2020-09-30 16:53:55 +02:00

249 lines
9.3 KiB
C++

#ifndef _NEW_PAGE_HPP_
#define _NEW_PAGE_HPP_
// Ruler
// 1 2 3 4 5 6 7 8
//345678901234567890123456789012345678901234567890123456789012345678901234567890
/********************************************************************/
/* */
/* The standard layout. */
/* */
/* The standard layout for 'hpp' files for this code is as */
/* follows: */
/* */
/* 1. Include files. */
/* 2. Constants exported from the class. */
/* 3. Data structures exported from the class. */
/* 4. Forward references to other data structures. */
/* 5. Class specifications (including inline functions). */
/* 6. Additional large inline functions. */
/* */
/* Any portion that is not required is simply omitted. */
/* */
/********************************************************************/
#include "Global.hpp"
#include "Environment.hpp"
#include "Common.hpp"
#include "Find.hpp"
#include "Page.hpp"
#include "Rockall.hpp"
#include "Spinlock.hpp"
/********************************************************************/
/* */
/* Constants exported from the class. */
/* */
/* The constants supplied here relate to various failure */
/* conditions or situations where information is unknown. */
/* */
/********************************************************************/
CONST SBIT16 NoSizeKey = -1;
/********************************************************************/
/* */
/* Class forward references. */
/* */
/* We need to refer to the following classes before they are */
/* fully specified so here we list them as forward references. */
/* */
/********************************************************************/
class CACHE;
/********************************************************************/
/* */
/* Create and delete pages. */
/* */
/* We would normally expect a class to manage its own memory. */
/* However, this is quite difficult for the 'PAGE' class as it */
/* is also responsible for managing the memory for the memory */
/* allocator. So here we remove a potentially nasty chore */
/* and isolate it in a class of its own. */
/* */
/********************************************************************/
class NEW_PAGE : public ENVIRONMENT, public COMMON
{
//
// Private structures.
//
// All the pages descriptions created by this
// class and managed by the memory allocator
// are linked in three list. One of these lists
// is managed by this class and is called the
// 'NewPageList'. All pages are linked into
// of three sub-lists. The 'ExternalList' is
// a list of pages externally allocated pages.
// The 'FullList' is a list of sub-allocated
// space from the 'ExternalList' which is
// partially or completely filled with alocations.
// Finally, the 'FreeList' is a collection of
// empty page descriptions all of the same size.
//
//
typedef struct
{
SBIT32 Elements;
LIST ExternalList;
LIST FreeList;
LIST FullList;
SBIT32 Size;
}
NEW_PAGES;
//
// Private data.
//
// We manage a collection of data structures in
// this class. The fundamental data structure
// is a stack of externally allocated pages that
// typically contain page descriptions that are
// linked together into linked lists. The maximum
// size of this stack is given by 'MaxStack'.
// A few additional pages are consumed to allocate
// stacks for caches in other classes.
//
SBIT32 MaxCacheStack;
SBIT32 MaxNewPages;
SBIT32 MaxStack;
//
// We keep track of various values to save having
// to recompute them. The 'NaturalSize' is the
// natural allocation size of our host (i.e. the OS).
// The 'RootSize' is some multiple of the
// 'NaturalSize' that this class uses to consume
// memory. The 'ThreadSafe' flag indicates whether
// we need to use locks. The "TopOfStack' is the
// stack which contains pointers to the externally
// allocated space. The 'Version' is the global
// version number that is used to stamp each page
// whenever it is allocated or deallocated. The
// version number allows the code to ensure that
// a page description has not been changed while
// it was not holding the associated lock.
//
SBIT32 NaturalSize;
SBIT32 RootCoreSize;
SBIT32 RootStackSize;
BOOLEAN ThreadSafe;
SBIT32 TopOfStack;
SBIT32 Version;
//
// We keep pointers to all the interesting data
// structures we may need to update. The
// 'CacheStack' points to block of memory that
// is being sliced into small stacks for caches
// in other classes. The 'NewPages' points to
// an array of linked lists of page descriptions.
// Each collection of page descriptions is
// identical except for the size of the assocated
// bit vector.
//
CHAR *CacheStack;
NEW_PAGES *NewPages;
VOID **Stack;
//
// We sometimes need to interact with some of
// the other class. The 'Find" class is a hash
// table of all the currently allocated pages.
// The 'Rockall' class contains the external
// API which includes the external memory memory
// allocation functions. The 'TopCache' is the
// largest cache we support and contains details
// about top level allocations sizes.
//
FIND *Find;
ROCKALL *Rockall;
CACHE *TopCache;
SPINLOCK Spinlock;
public:
//
// Public functions.
//
// The public functions provide support for creating
// new page descriptions and caches for other
// classes. Although a lot of the fuinctionality
// of the heap is masked from this class various
// features such as deleting the entire heap
// (i.e. 'DeleteAll') are still visable.
//
NEW_PAGE
(
FIND *NewFind,
SBIT32 NewPageSizes[],
ROCKALL *NewRockall,
SBIT32 Size,
BOOLEAN NewThreadSafe
);
PAGE *CreatePage( CACHE *Cache,SBIT32 NewSize = NoSize );
VOID DeleteAll( BOOLEAN Recycle );
VOID DeletePage( PAGE *Page );
SBIT16 FindSizeKey( SBIT16 NumberOfElements );
VOID *NewCacheStack( SBIT32 Size );
VOID ResizeStack( VOID );
BOOLEAN Walk( SEARCH_PAGE *Details );
~NEW_PAGE( VOID );
//
// Public inline functions.
//
// The public inline functions are typically either
// small or highly performance sensitive. The
// functions here mainly relate to locking and
// updating various data structures.
//
INLINE VOID ClaimNewPageLock( VOID )
{
if ( (ThreadSafe) && (Find -> GetLockCount() == 0) )
{ Spinlock.ClaimLock(); }
}
INLINE VOID ReleaseNewPageLock( VOID )
{
if ( (ThreadSafe) && (Find -> GetLockCount() == 0) )
{ Spinlock.ReleaseLock(); }
}
INLINE VOID UpdateNewPage( CACHE *NewTopCache )
{ TopCache = NewTopCache; }
private:
//
// Private functions.
//
// We support the overloading of the external
// memory allocation routines. This is somewhat
// unusual and means that we need to verify
// that these functions do not supply us with
// total rubbish.
//
VOID *VerifyNewArea( SBIT32 AlignMask,SBIT32 Size );
//
// Disabled operations.
//
// All copy constructors and class assignment
// operations are disabled.
//
NEW_PAGE( CONST NEW_PAGE & Copy );
VOID operator=( CONST NEW_PAGE & Copy );
};
#endif