2007-12-18 09:41:03 +01:00
/*
Unix SMB / CIFS implementation .
In - memory cache
Copyright ( C ) Volker Lendecke 2007
This program is free software ; you can redistribute it and / or modify
it under the terms of the GNU General Public License as published by
the Free Software Foundation ; either version 3 of the License , or
( at your option ) any later version .
This program 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 General Public License for more details .
You should have received a copy of the GNU General Public License
along with this program . If not , see < http : //www.gnu.org/licenses/>.
*/
2014-07-17 12:49:48 +02:00
# include "replace.h"
# include <talloc.h>
# include "../lib/util/debug.h"
2015-03-05 11:26:46 +01:00
# include "../lib/util/samba_util.h"
2014-07-17 12:49:48 +02:00
# include "../lib/util/dlinklist.h"
2008-10-12 16:27:00 +02:00
# include "../lib/util/rbtree.h"
2014-07-17 12:49:48 +02:00
# include "memcache.h"
2007-12-18 09:41:03 +01:00
2007-12-20 10:55:45 +01:00
static struct memcache * global_cache ;
2019-04-01 15:38:59 -07:00
struct memcache_talloc_value {
void * ptr ;
2019-04-01 16:23:35 -07:00
size_t len ;
2019-04-01 15:38:59 -07:00
} ;
2007-12-18 09:41:03 +01:00
struct memcache_element {
struct rb_node rb_node ;
struct memcache_element * prev , * next ;
size_t keylength , valuelength ;
2014-07-17 12:41:20 +02:00
uint8_t n ; /* This is really an enum, but save memory */
2007-12-18 09:41:03 +01:00
char data [ 1 ] ; /* placeholder for offsetof */
} ;
struct memcache {
2010-02-06 12:40:03 +11:00
struct memcache_element * mru ;
2007-12-18 09:41:03 +01:00
struct rb_root tree ;
size_t size ;
size_t max_size ;
} ;
2007-12-20 14:41:58 +01:00
static void memcache_element_parse ( struct memcache_element * e ,
DATA_BLOB * key , DATA_BLOB * value ) ;
2008-11-13 23:50:19 +01:00
static bool memcache_is_talloc ( enum memcache_number n )
{
bool result ;
switch ( n ) {
case GETPWNAM_CACHE :
case PDB_GETPWSID_CACHE :
case SINGLETON_CACHE_TALLOC :
2015-04-02 11:19:22 -07:00
case SHARE_MODE_LOCK_CACHE :
2017-06-29 14:32:47 -07:00
case GETWD_CACHE :
2016-10-18 13:37:19 -06:00
case VIRUSFILTER_SCAN_RESULTS_CACHE_TALLOC :
2008-11-13 23:50:19 +01:00
result = true ;
break ;
default :
result = false ;
break ;
}
return result ;
}
2007-12-18 09:41:03 +01:00
static int memcache_destructor ( struct memcache * cache ) {
struct memcache_element * e , * next ;
for ( e = cache - > mru ; e ! = NULL ; e = next ) {
next = e - > next ;
2014-07-17 12:48:51 +02:00
TALLOC_FREE ( e ) ;
2007-12-18 09:41:03 +01:00
}
return 0 ;
}
struct memcache * memcache_init ( TALLOC_CTX * mem_ctx , size_t max_size )
{
struct memcache * result ;
2011-06-07 11:44:43 +10:00
result = talloc_zero ( mem_ctx , struct memcache ) ;
2007-12-18 09:41:03 +01:00
if ( result = = NULL ) {
return NULL ;
}
result - > max_size = max_size ;
talloc_set_destructor ( result , memcache_destructor ) ;
return result ;
}
2007-12-20 10:55:45 +01:00
void memcache_set_global ( struct memcache * cache )
{
TALLOC_FREE ( global_cache ) ;
global_cache = cache ;
}
2007-12-18 09:41:03 +01:00
static struct memcache_element * memcache_node2elem ( struct rb_node * node )
{
return ( struct memcache_element * )
( ( char * ) node - offsetof ( struct memcache_element , rb_node ) ) ;
}
static void memcache_element_parse ( struct memcache_element * e ,
DATA_BLOB * key , DATA_BLOB * value )
{
2014-07-17 12:41:20 +02:00
key - > data = ( ( uint8_t * ) e ) + offsetof ( struct memcache_element , data ) ;
2007-12-18 09:41:03 +01:00
key - > length = e - > keylength ;
value - > data = key - > data + e - > keylength ;
value - > length = e - > valuelength ;
}
static size_t memcache_element_size ( size_t key_length , size_t value_length )
{
return sizeof ( struct memcache_element ) - 1 + key_length + value_length ;
}
static int memcache_compare ( struct memcache_element * e , enum memcache_number n ,
DATA_BLOB key )
{
DATA_BLOB this_key , this_value ;
2008-05-20 18:35:23 +02:00
if ( ( int ) e - > n < ( int ) n ) return 1 ;
if ( ( int ) e - > n > ( int ) n ) return - 1 ;
2007-12-18 09:41:03 +01:00
2008-05-20 18:35:23 +02:00
if ( e - > keylength < key . length ) return 1 ;
if ( e - > keylength > key . length ) return - 1 ;
2007-12-18 09:41:03 +01:00
memcache_element_parse ( e , & this_key , & this_value ) ;
return memcmp ( this_key . data , key . data , key . length ) ;
}
static struct memcache_element * memcache_find (
struct memcache * cache , enum memcache_number n , DATA_BLOB key )
{
struct rb_node * node ;
node = cache - > tree . rb_node ;
while ( node ! = NULL ) {
struct memcache_element * elem = memcache_node2elem ( node ) ;
int cmp ;
cmp = memcache_compare ( elem , n , key ) ;
if ( cmp = = 0 ) {
return elem ;
}
node = ( cmp < 0 ) ? node - > rb_left : node - > rb_right ;
}
return NULL ;
}
bool memcache_lookup ( struct memcache * cache , enum memcache_number n ,
DATA_BLOB key , DATA_BLOB * value )
{
struct memcache_element * e ;
2007-12-20 10:55:45 +01:00
if ( cache = = NULL ) {
cache = global_cache ;
}
if ( cache = = NULL ) {
return false ;
}
2007-12-18 09:41:03 +01:00
e = memcache_find ( cache , n , key ) ;
if ( e = = NULL ) {
return false ;
}
if ( cache - > size ! = 0 ) {
DLIST_PROMOTE ( cache - > mru , e ) ;
}
memcache_element_parse ( e , & key , value ) ;
return true ;
}
2007-12-20 14:41:58 +01:00
void * memcache_lookup_talloc ( struct memcache * cache , enum memcache_number n ,
DATA_BLOB key )
{
DATA_BLOB value ;
2019-04-01 15:38:59 -07:00
struct memcache_talloc_value mtv ;
2007-12-20 14:41:58 +01:00
if ( ! memcache_lookup ( cache , n , key , & value ) ) {
return NULL ;
}
2019-04-01 15:38:59 -07:00
if ( value . length ! = sizeof ( mtv ) ) {
2007-12-20 14:41:58 +01:00
return NULL ;
}
2019-04-01 15:38:59 -07:00
memcpy ( & mtv , value . data , sizeof ( mtv ) ) ;
2007-12-20 14:41:58 +01:00
2019-04-01 15:38:59 -07:00
return mtv . ptr ;
2007-12-20 14:41:58 +01:00
}
2007-12-18 09:41:03 +01:00
static void memcache_delete_element ( struct memcache * cache ,
struct memcache_element * e )
{
rb_erase ( & e - > rb_node , & cache - > tree ) ;
DLIST_REMOVE ( cache - > mru , e ) ;
2008-11-13 23:50:19 +01:00
if ( memcache_is_talloc ( e - > n ) ) {
DATA_BLOB cache_key , cache_value ;
2019-04-01 15:38:59 -07:00
struct memcache_talloc_value mtv ;
2008-11-13 23:50:19 +01:00
memcache_element_parse ( e , & cache_key , & cache_value ) ;
2019-04-01 15:38:59 -07:00
SMB_ASSERT ( cache_value . length = = sizeof ( mtv ) ) ;
memcpy ( & mtv , cache_value . data , sizeof ( mtv ) ) ;
2019-04-01 16:23:35 -07:00
cache - > size - = mtv . len ;
2019-04-01 15:38:59 -07:00
TALLOC_FREE ( mtv . ptr ) ;
2008-11-13 23:50:19 +01:00
}
2007-12-18 09:41:03 +01:00
cache - > size - = memcache_element_size ( e - > keylength , e - > valuelength ) ;
2014-07-17 12:48:51 +02:00
TALLOC_FREE ( e ) ;
2007-12-18 09:41:03 +01:00
}
2021-02-02 18:10:38 +01:00
static void memcache_trim ( struct memcache * cache , struct memcache_element * e )
2007-12-18 09:41:03 +01:00
{
2021-02-02 18:10:38 +01:00
struct memcache_element * tail = NULL ;
2007-12-18 09:41:03 +01:00
if ( cache - > max_size = = 0 ) {
return ;
}
2021-02-02 18:10:38 +01:00
for ( tail = DLIST_TAIL ( cache - > mru ) ;
( cache - > size > cache - > max_size ) & & ( tail ! = NULL ) ;
tail = DLIST_TAIL ( cache - > mru ) )
{
if ( tail = = e ) {
tail = DLIST_PREV ( tail ) ;
if ( tail = = NULL ) {
break ;
}
}
memcache_delete_element ( cache , tail ) ;
2007-12-18 09:41:03 +01:00
}
}
void memcache_delete ( struct memcache * cache , enum memcache_number n ,
DATA_BLOB key )
{
struct memcache_element * e ;
2007-12-20 10:55:45 +01:00
if ( cache = = NULL ) {
cache = global_cache ;
}
if ( cache = = NULL ) {
return ;
}
2007-12-18 09:41:03 +01:00
e = memcache_find ( cache , n , key ) ;
if ( e = = NULL ) {
return ;
}
memcache_delete_element ( cache , e ) ;
}
void memcache_add ( struct memcache * cache , enum memcache_number n ,
DATA_BLOB key , DATA_BLOB value )
{
struct memcache_element * e ;
struct rb_node * * p ;
struct rb_node * parent ;
DATA_BLOB cache_key , cache_value ;
size_t element_size ;
2007-12-20 10:55:45 +01:00
if ( cache = = NULL ) {
cache = global_cache ;
}
if ( cache = = NULL ) {
return ;
}
2007-12-18 09:41:03 +01:00
if ( key . length = = 0 ) {
return ;
}
e = memcache_find ( cache , n , key ) ;
if ( e ! = NULL ) {
memcache_element_parse ( e , & cache_key , & cache_value ) ;
if ( value . length < = cache_value . length ) {
2008-11-13 23:50:19 +01:00
if ( memcache_is_talloc ( e - > n ) ) {
2019-04-01 15:38:59 -07:00
struct memcache_talloc_value mtv ;
SMB_ASSERT ( cache_value . length = = sizeof ( mtv ) ) ;
memcpy ( & mtv , cache_value . data , sizeof ( mtv ) ) ;
2019-04-01 16:23:35 -07:00
cache - > size - = mtv . len ;
2019-04-01 15:38:59 -07:00
TALLOC_FREE ( mtv . ptr ) ;
2008-11-13 23:50:19 +01:00
}
2007-12-18 09:41:03 +01:00
/*
* We can reuse the existing record
*/
memcpy ( cache_value . data , value . data , value . length ) ;
e - > valuelength = value . length ;
2019-04-01 16:23:35 -07:00
if ( memcache_is_talloc ( e - > n ) ) {
struct memcache_talloc_value mtv ;
SMB_ASSERT ( cache_value . length = = sizeof ( mtv ) ) ;
memcpy ( & mtv , cache_value . data , sizeof ( mtv ) ) ;
cache - > size + = mtv . len ;
}
2007-12-18 09:41:03 +01:00
return ;
}
memcache_delete_element ( cache , e ) ;
}
element_size = memcache_element_size ( key . length , value . length ) ;
2014-07-17 12:48:51 +02:00
e = talloc_size ( cache , element_size ) ;
2007-12-18 09:41:03 +01:00
if ( e = = NULL ) {
2014-07-17 12:48:51 +02:00
DEBUG ( 0 , ( " talloc failed \n " ) ) ;
2007-12-18 09:41:03 +01:00
return ;
}
2014-07-17 12:48:51 +02:00
talloc_set_type ( e , struct memcache_element ) ;
2007-12-18 09:41:03 +01:00
e - > n = n ;
e - > keylength = key . length ;
e - > valuelength = value . length ;
memcache_element_parse ( e , & cache_key , & cache_value ) ;
memcpy ( cache_key . data , key . data , key . length ) ;
memcpy ( cache_value . data , value . data , value . length ) ;
parent = NULL ;
p = & cache - > tree . rb_node ;
while ( * p ) {
struct memcache_element * elem = memcache_node2elem ( * p ) ;
int cmp ;
parent = ( * p ) ;
cmp = memcache_compare ( elem , n , key ) ;
p = ( cmp < 0 ) ? & ( * p ) - > rb_left : & ( * p ) - > rb_right ;
}
rb_link_node ( & e - > rb_node , parent , p ) ;
rb_insert_color ( & e - > rb_node , & cache - > tree ) ;
DLIST_ADD ( cache - > mru , e ) ;
cache - > size + = element_size ;
2019-04-01 16:23:35 -07:00
if ( memcache_is_talloc ( e - > n ) ) {
struct memcache_talloc_value mtv ;
SMB_ASSERT ( cache_value . length = = sizeof ( mtv ) ) ;
memcpy ( & mtv , cache_value . data , sizeof ( mtv ) ) ;
cache - > size + = mtv . len ;
}
2021-02-02 18:10:38 +01:00
memcache_trim ( cache , e ) ;
2007-12-18 09:41:03 +01:00
}
2007-12-20 14:41:58 +01:00
void memcache_add_talloc ( struct memcache * cache , enum memcache_number n ,
2008-11-15 00:50:33 +01:00
DATA_BLOB key , void * pptr )
2007-12-20 14:41:58 +01:00
{
2019-04-01 15:38:59 -07:00
struct memcache_talloc_value mtv ;
2008-11-14 13:42:54 +01:00
void * * ptr = ( void * * ) pptr ;
if ( cache = = NULL ) {
cache = global_cache ;
}
if ( cache = = NULL ) {
return ;
}
2019-04-01 16:23:35 -07:00
mtv . len = talloc_total_size ( * ptr ) ;
2019-04-01 15:38:59 -07:00
mtv . ptr = talloc_move ( cache , ptr ) ;
memcache_add ( cache , n , key , data_blob_const ( & mtv , sizeof ( mtv ) ) ) ;
2007-12-20 14:41:58 +01:00
}
2007-12-18 09:41:03 +01:00
void memcache_flush ( struct memcache * cache , enum memcache_number n )
{
struct rb_node * node ;
2007-12-20 10:55:45 +01:00
if ( cache = = NULL ) {
cache = global_cache ;
}
if ( cache = = NULL ) {
return ;
}
2007-12-18 09:41:03 +01:00
/*
* Find the smallest element of number n
*/
node = cache - > tree . rb_node ;
if ( node = = NULL ) {
return ;
}
2008-05-20 18:35:23 +02:00
/*
* First , find * any * element of number n
*/
2007-12-18 09:41:03 +01:00
while ( true ) {
struct memcache_element * elem = memcache_node2elem ( node ) ;
struct rb_node * next ;
2008-05-20 18:35:23 +02:00
if ( ( int ) elem - > n = = ( int ) n ) {
break ;
}
2007-12-18 09:41:03 +01:00
if ( ( int ) elem - > n < ( int ) n ) {
next = node - > rb_right ;
}
else {
next = node - > rb_left ;
}
if ( next = = NULL ) {
break ;
}
node = next ;
}
2008-05-20 18:35:23 +02:00
/*
* Then , find the leftmost element with number n
*/
while ( true ) {
struct rb_node * prev = rb_prev ( node ) ;
struct memcache_element * elem ;
if ( prev = = NULL ) {
break ;
}
elem = memcache_node2elem ( prev ) ;
if ( ( int ) elem - > n ! = ( int ) n ) {
break ;
}
node = prev ;
}
2007-12-18 09:41:03 +01:00
while ( node ! = NULL ) {
struct memcache_element * e = memcache_node2elem ( node ) ;
struct rb_node * next = rb_next ( node ) ;
2008-05-20 18:35:23 +02:00
if ( e - > n ! = n ) {
break ;
}
2007-12-18 09:41:03 +01:00
memcache_delete_element ( cache , e ) ;
node = next ;
}
}