1997-12-19 09:32:58 +00:00
/* ========================================================================== **
* ubi_Cache . c
*
* Copyright ( C ) 1997 by Christopher R . Hertel
*
* Email : crh @ ubiqx . mn . org
* - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - * *
*
* This module implements a generic cache .
*
* - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - * *
*
* This library is free software ; you can redistribute it and / or
* modify it under the terms of the GNU Library General Public
* License as published by the Free Software Foundation ; either
* version 2 of the License , or ( at your option ) any later version .
*
* This library is distributed in the hope that it will be useful ,
* but WITHOUT ANY WARRANTY ; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE . See the GNU
* Library General Public License for more details .
*
* You should have received a copy of the GNU Library General Public
* License along with this library ; if not , write to the Free
* Software Foundation , Inc . , 675 Mass Ave , Cambridge , MA 0213 9 , USA .
*
* - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - * *
*
* This module uses a splay tree to implement a simple cache . The cache
* module adds a thin layer of functionality to the splay tree . In
* particular :
*
* - The tree ( cache ) may be limited in size by the number of
* entries permitted or the amount of memory used . When either
* limit is exceeded cache entries are removed until the cache
* conforms .
* - Some statistical information is kept so that an approximate
* " hit ratio " can be calculated .
* - There are several functions available that provide access to
* and management of cache size limits , hit ratio , and tree
* trimming .
*
* The splay tree is used because recently accessed items tend toward the
* top of the tree and less recently accessed items tend toward the bottom .
* This makes it easy to purge less recently used items should the cache
* exceed its limits .
*
* To use this module , you will need to supply a comparison function of
* type ubi_trCompFunc and a node - freeing function of type
* ubi_trKillNodeTrn . See ubi_BinTree . h for more information on
* these . ( This is all basic ubiqx tree management stuff . )
*
* Notes :
*
* - Cache performance will start to suffer dramatically if the
* cache becomes large enough to force the OS to start swapping
* memory to disk . This is because the nodes of the underlying tree
* will be scattered across memory in an order that is completely
* unrelated to their traversal order . As more and more of the
* cache is placed into swap space , more and more swaps will be
* required for a simple traversal ( . . . and then there ' s the splay
* operation ) .
*
* In one simple test under Linux , the load and dump of a cache of
* 400 , 000 entries took only 1 min , 40 sec of real time . The same
* test with 450 , 000 records took 2 * hours * and eight minutes .
*
* - In an effort to save memory , I considered using an unsigned
* short to save the per - entry entry size . I would have tucked this
* value into some unused space in the tree node structure . On
* 32 - bit word aligned systems this would have saved an additional
* four bytes per entry . I may revisit this issue , but for now I ' ve
* decided against it .
*
* Using an unsigned short would limit the size of an entry to 64 K
* bytes . That ' s probably more than enough for most applications .
* The key word in that last sentence , however , is " probably " . I
* really dislike imposing such limits on things .
*
* - Each entry keeps track of the amount of memory it used and the
* cache header keeps the total . This information is provided via
* the EntrySize parameter in ubi_cachePut ( ) , so it is up to you to
* make sure that the numbers are accurate . ( The numbers don ' t even
* have to represent bytes used . )
*
* As you consider this , note that the strdup ( ) function - - as an
* example - - will call malloc ( ) . The latter generally allocates a
* multiple of the system word size , which may be more than the
* number of bytes needed to store the string .
*
* - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - * *
*
1998-05-21 17:14:20 +00:00
* Log : ubi_Cache . c , v
* Revision 0.1 1998 / 05 / 20 04 : 36 : 02 crh
* The C file now includes ubi_null . h . See ubi_null . h for more info .
*
1997-12-19 09:32:58 +00:00
* Revision 0.0 1997 / 12 / 18 06 : 24 : 33 crh
* Initial Revision .
*
* = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = * *
*/
1998-05-21 17:14:20 +00:00
# include "ubi_null.h" /* ubiqx NULL source. */
# include "ubi_Cache.h" /* Header for *this* module. */
1997-12-19 09:32:58 +00:00
/* -------------------------------------------------------------------------- **
* Static data . . .
*/
1998-05-21 17:14:20 +00:00
/* commented out until I make use of it...
static char ModuleID [ ] =
" ubi_Cache \n \
\ tRevision : 0.1 \ n \
\ tDate : 1998 / 05 / 20 04 : 36 : 02 \ n \
\ tAuthor : crh \ n " ;
*/
1997-12-19 09:32:58 +00:00
/* -------------------------------------------------------------------------- **
* Internal functions . . .
*/
static void free_entry ( ubi_cacheRootPtr CachePtr , ubi_cacheEntryPtr EntryPtr )
/* ------------------------------------------------------------------------ **
* Free a ubi_cacheEntry , and adjust the mem_used counter accordingly .
*
* Input : CachePtr - A pointer to the cache from which the entry has
* been removed .
* EntryPtr - A pointer to the already removed entry .
*
* Output : none .
*
* Notes : The entry must be removed from the cache * before * this function
* is called ! ! ! !
* - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - * *
*/
{
CachePtr - > mem_used - = EntryPtr - > entry_size ;
( * CachePtr - > free_func ) ( ( void * ) EntryPtr ) ;
} /* free_entry */
static void cachetrim ( ubi_cacheRootPtr crptr )
/* ------------------------------------------------------------------------ **
* Remove entries from the cache until the number of entries and the amount
* of memory used are * both * below or at the maximum .
*
* Input : crptr - pointer to the cache to be trimmed .
*
* Output : None .
*
* - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - * *
*/
{
while ( ( crptr - > max_entries & & ( crptr - > max_entries < crptr - > root . count ) )
| | ( crptr - > max_memory & & ( crptr - > max_memory < crptr - > mem_used ) ) )
{
if ( ! ubi_cacheReduce ( crptr , 1 ) )
return ;
}
} /* cachetrim */
/* -------------------------------------------------------------------------- **
* Exported functions . . .
*/
ubi_cacheRootPtr ubi_cacheInit ( ubi_cacheRootPtr CachePtr ,
ubi_trCompFunc CompFunc ,
ubi_trKillNodeRtn FreeFunc ,
unsigned long MaxEntries ,
unsigned long MaxMemory )
/* ------------------------------------------------------------------------ **
* Initialize a cache header structure .
*
* Input : CachePtr - A pointer to a ubi_cacheRoot structure that is
* to be initialized .
* CompFunc - A pointer to the function that will be called
* to compare two cache values . See the module
* comments , above , for more information .
* FreeFunc - A pointer to a function that will be called
* to free a cache entry . If you allocated
* the cache entry using malloc ( ) , then this
* will likely be free ( ) . If you are allocating
* cache entries from a free list , then this will
* likely be a function that returns memory to the
* free list , etc .
* MaxEntries - The maximum number of entries that will be
* allowed to exist in the cache . If this limit
* is exceeded , then existing entries will be
* removed from the cache . A value of zero
* indicates that there is no limit on the number
* of cache entries . See ubi_cachePut ( ) .
* MaxMemory - The maximum amount of memory , in bytes , to be
* allocated to the cache ( excluding the cache
* header ) . If this is exceeded , existing entries
* in the cache will be removed until enough memory
* has been freed to meet the condition . See
* ubi_cachePut ( ) .
*
* Output : A pointer to the initialized cache ( i . e . , the same as CachePtr ) .
*
* Notes : Both MaxEntries and MaxMemory may be changed after the cache
* has been created . See
* ubi_cacheSetMaxEntries ( )
* ubi_cacheSetMaxMemory ( )
* ubi_cacheGetMaxEntries ( )
* ubi_cacheGetMaxMemory ( ) ( the latter two are macros ) .
*
* - Memory is allocated in multiples of the word size . The
* return value of the strlen ( ) function does not reflect
* this ; it will allways be less than or equal to the amount
* of memory actually allocated . Keep this in mind when
* choosing a value for MaxMemory .
*
* - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - * *
*/
{
if ( CachePtr )
{
( void ) ubi_trInitTree ( CachePtr , CompFunc , ubi_trOVERWRITE ) ;
CachePtr - > free_func = FreeFunc ;
CachePtr - > max_entries = MaxEntries ;
CachePtr - > max_memory = MaxMemory ;
CachePtr - > mem_used = 0 ;
CachePtr - > cache_hits = 0 ;
CachePtr - > cache_trys = 0 ;
}
return ( CachePtr ) ;
} /* ubi_cacheInit */
ubi_cacheRootPtr ubi_cacheClear ( ubi_cacheRootPtr CachePtr )
/* ------------------------------------------------------------------------ **
* Remove and free all entries in an existing cache .
*
* Input : CachePtr - A pointer to the cache that is to be cleared .
*
* Output : A pointer to the cache header ( i . e . , the same as CachePtr ) .
* This function re - initializes the cache header .
*
* - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - * *
*/
{
if ( CachePtr )
{
( void ) ubi_trKillTree ( CachePtr , CachePtr - > free_func ) ;
CachePtr - > mem_used = 0 ;
CachePtr - > cache_hits = 0 ;
CachePtr - > cache_trys = 0 ;
}
return ( CachePtr ) ;
} /* ubi_cacheClear */
void ubi_cachePut ( ubi_cacheRootPtr CachePtr ,
unsigned long EntrySize ,
ubi_cacheEntryPtr EntryPtr ,
ubi_trItemPtr Key )
/* ------------------------------------------------------------------------ **
* Add an entry to the cache .
*
* Input : CachePtr - A pointer to the cache into which the entry
* will be added .
* EntrySize - The size , in bytes , of the memory block indicated
* by EntryPtr . This will be copied into the
* EntryPtr - > entry_size field .
* EntryPtr - A pointer to a memory block that begins with a
* ubi_cacheEntry structure . The entry structure
* should be followed immediately by the data to be
* cached ( even if that is a pointer to yet more data ) .
* Key - Pointer used to identify the lookup key within the
* Entry .
*
* Output : None .
*
* Notes : After adding the new node , the cache is " trimmed " . This
* removes extra nodes if the tree has exceeded it ' s memory or
* entry count limits . It is unlikely that the newly added node
* will be purged from the cache ( assuming a reasonably large
* cache ) , since new nodes in a splay tree ( which is what this
* module was designed to use ) are moved to the top of the tree
* and the cache purge process removes nodes from the bottom of
* the tree .
* - The underlying splay tree is opened in OVERWRITE mode . If
* the input key matches an existing key , the existing entry will
* be politely removed from the tree and freed .
* - Memory is allocated in multiples of the word size . The
* return value of the strlen ( ) function does not reflect
* this ; it will allways be less than or equal to the amount
* of memory actually allocated .
*
* - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - * *
*/
{
ubi_trNodePtr OldNode ;
EntryPtr - > entry_size = EntrySize ;
CachePtr - > mem_used + = EntrySize ;
( void ) ubi_trInsert ( CachePtr , EntryPtr , Key , & OldNode ) ;
if ( OldNode )
free_entry ( CachePtr , ( ubi_cacheEntryPtr ) OldNode ) ;
cachetrim ( CachePtr ) ;
} /* ubi_cachePut */
ubi_cacheEntryPtr ubi_cacheGet ( ubi_cacheRootPtr CachePtr ,
ubi_trItemPtr FindMe )
/* ------------------------------------------------------------------------ **
* Attempt to retrieve an entry from the cache .
*
* Input : CachePtr - A ponter to the cache that is to be searched .
* FindMe - A ubi_trItemPtr that indicates the key for which
* to search .
*
* Output : A pointer to the cache entry that was found , or NULL if no
* matching entry was found .
*
* Notes : This function also updates the hit ratio counters .
* The counters are unsigned short . If the number of cache tries
* reaches 32768 , then both the number of tries and the number of
* hits are divided by two . This prevents the counters from
* overflowing . See the comments in ubi_cacheHitRatio ( ) for
* additional notes .
*
* - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - * *
*/
{
ubi_trNodePtr FoundPtr ;
FoundPtr = ubi_trFind ( CachePtr , FindMe ) ;
if ( FoundPtr )
CachePtr - > cache_hits + + ;
CachePtr - > cache_trys + + ;
if ( CachePtr - > cache_trys & 0x8000 )
{
CachePtr - > cache_hits = CachePtr - > cache_hits / 2 ;
CachePtr - > cache_trys = CachePtr - > cache_trys / 2 ;
}
return ( ( ubi_cacheEntryPtr ) FoundPtr ) ;
} /* ubi_cacheGet */
ubi_trBool ubi_cacheDelete ( ubi_cacheRootPtr CachePtr , ubi_trItemPtr DeleteMe )
/* ------------------------------------------------------------------------ **
* Find and delete the specified cache entry .
*
* Input : CachePtr - A pointer to the cache .
* DeleteMe - The key of the entry to be deleted .
*
* Output : TRUE if the entry was found & freed , else FALSE .
*
* - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - * *
*/
{
ubi_trNodePtr FoundPtr ;
FoundPtr = ubi_trFind ( CachePtr , DeleteMe ) ;
if ( FoundPtr )
{
( void ) ubi_trRemove ( CachePtr , FoundPtr ) ;
free_entry ( CachePtr , ( ubi_cacheEntryPtr ) FoundPtr ) ;
return ( ubi_trTRUE ) ;
}
return ( ubi_trFALSE ) ;
} /* ubi_cacheDelete */
ubi_trBool ubi_cacheReduce ( ubi_cacheRootPtr CachePtr , unsigned long count )
/* ------------------------------------------------------------------------ **
* Remove < count > entries from the bottom of the cache .
*
* Input : CachePtr - A pointer to the cache which is to be reduced in
* size .
* count - The number of entries to remove .
*
* Output : The function will return TRUE if < count > entries were removed ,
* else FALSE . A return value of FALSE should indicate that
* there were less than < count > entries in the cache , and that the
* cache is now empty .
*
* Notes : This function forces a reduction in the number of cache entries
* without requiring that the MaxMemory or MaxEntries values be
* changed .
*
* - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - * *
*/
{
ubi_trNodePtr NodePtr ;
while ( count )
{
NodePtr = ubi_trLeafNode ( CachePtr - > root . root ) ;
if ( NULL = = NodePtr )
return ( ubi_trFALSE ) ;
else
{
( void ) ubi_trRemove ( CachePtr , NodePtr ) ;
free_entry ( CachePtr , ( ubi_cacheEntryPtr ) NodePtr ) ;
}
count - - ;
}
return ( ubi_trTRUE ) ;
} /* ubi_cacheReduce */
unsigned long ubi_cacheSetMaxEntries ( ubi_cacheRootPtr CachePtr ,
unsigned long NewSize )
/* ------------------------------------------------------------------------ **
* Change the maximum number of entries allowed to exist in the cache .
*
* Input : CachePtr - A pointer to the cache to be modified .
* NewSize - The new maximum number of cache entries .
*
* Output : The maximum number of entries previously allowed to exist in
* the cache .
*
* Notes : If the new size is less than the old size , this function will
* trim the cache ( remove excess entries ) .
* - A value of zero indicates an unlimited number of entries .
*
* - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - * *
*/
{
unsigned long oldsize = CachePtr - > max_entries ; /* Save the old value. */
CachePtr - > max_entries = NewSize ; /* Apply the new value. */
if ( ( NewSize < oldsize ) | | ( NewSize & & ! oldsize ) ) /* If size is smaller, */
cachetrim ( CachePtr ) ; /* remove excess. */
return ( oldsize ) ;
} /* ubi_cacheSetMaxEntries */
unsigned long ubi_cacheSetMaxMemory ( ubi_cacheRootPtr CachePtr ,
unsigned long NewSize )
/* ------------------------------------------------------------------------ **
* Change the maximum amount of memory to be used for storing cache
* entries .
*
* Input : CachePtr - A pointer to the cache to be modified .
* NewSize - The new cache memory size .
*
* Output : The previous maximum memory size .
*
* Notes : If the new size is less than the old size , this function will
* trim the cache ( remove excess entries ) .
* - A value of zero indicates that the cache has no memory limit .
*
* - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - * *
*/
{
unsigned long oldsize = CachePtr - > max_memory ; /* Save the old value. */
CachePtr - > max_memory = NewSize ; /* Apply the new value. */
if ( ( NewSize < oldsize ) | | ( NewSize & & ! oldsize ) ) /* If size is smaller, */
cachetrim ( CachePtr ) ; /* remove excess. */
return ( oldsize ) ;
} /* ubi_cacheSetMaxMemory */
int ubi_cacheHitRatio ( ubi_cacheRootPtr CachePtr )
/* ------------------------------------------------------------------------ **
* Returns a value that is 10 , 000 times the slightly weighted average hit
* ratio for the cache .
*
* Input : CachePtr - Pointer to the cache to be queried .
*
* Output : An integer that is 10 , 000 times the number of successful
* cache hits divided by the number of cache lookups , or :
* ( 10000 * hits ) / trys
* You can easily convert this to a float , or do something
* like this ( where i is the return value of this function ) :
*
* printf ( " Hit rate : %d.%02d%% \n " , ( i / 100 ) , ( i % 100 ) ) ;
*
* Notes : I say " slightly-weighted " , because the numerator and
* denominator are both accumulated in locations of type
* ' unsigned short ' . If the number of cache trys becomes
* large enough , both are divided by two . ( See function
* ubi_cacheGet ( ) . )
* Dividing both numerator and denominator by two does not
* change the ratio ( much . . . it is an integer divide ) , but it
* does mean that subsequent increments to either counter will
* have twice as much significance as previous ones .
*
* - The value returned by this function will be in the range
* [ 0. .10000 ] because ( 0 < = cache_hits < = cache_trys ) will
* always be true .
*
* - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - * *
*/
{
int tmp = 0 ;
if ( CachePtr - > cache_trys )
tmp = ( int ) ( ( 10000 * ( long ) ( CachePtr - > cache_hits ) )
/ ( long ) ( CachePtr - > cache_trys ) ) ;
return ( tmp ) ;
} /* ubi_cacheHitRatio */
/* -------------------------------------------------------------------------- */