mirror of
https://github.com/MariaDB/server.git
synced 2025-01-22 23:04:20 +01:00
b0e3bdca20
note that when the object size is 1, the cache table acts on the number of objects as before. change the tree algorithms to update the object size in the cache table when the object is unpinned. git-svn-id: file:///svn/tokudb@334 c7de825b-a66e-492c-adef-691d508d4ae1
161 lines
5 KiB
C
161 lines
5 KiB
C
#include "memory.h"
|
|
#include "cachetable.h"
|
|
|
|
#include <assert.h>
|
|
#include <stdio.h>
|
|
#include <stdlib.h>
|
|
#include <unistd.h>
|
|
|
|
CACHETABLE ct;
|
|
|
|
enum { N_PRESENT_LIMIT = 4, TRIALS=200, N_FILES=2 };
|
|
int n_present=0;
|
|
struct present_items {
|
|
CACHEKEY key;
|
|
CACHEFILE cf;
|
|
} present_items[N_PRESENT_LIMIT];
|
|
|
|
static void print_ints(void) __attribute__((__unused__));
|
|
static void print_ints(void) {
|
|
int i;
|
|
for (i=0; i<n_present; i++) {
|
|
if (i==0) printf("{"); else printf(",");
|
|
printf("{%lld,%p}", present_items[i].key, present_items[i].cf);
|
|
}
|
|
printf("}\n");
|
|
}
|
|
|
|
static void item_becomes_present(CACHEFILE cf, CACHEKEY key) {
|
|
assert(n_present<N_PRESENT_LIMIT);
|
|
present_items[n_present].cf = cf;
|
|
present_items[n_present].key = key;
|
|
n_present++;
|
|
}
|
|
|
|
static void item_becomes_not_present(CACHEFILE cf, CACHEKEY key) {
|
|
int i;
|
|
//printf("Removing {%4lld %16p}: Initially: ", key, cf); print_ints();
|
|
assert(n_present<=N_PRESENT_LIMIT);
|
|
for (i=0; i<n_present; i++) {
|
|
if (present_items[i].cf==cf && present_items[i].key==key) {
|
|
present_items[i]=present_items[n_present-1];
|
|
n_present--;
|
|
//printf(" Finally: "); print_ints();
|
|
return;
|
|
}
|
|
}
|
|
printf("Whoops, %p,%lld was already not present\n", cf ,key);
|
|
abort();
|
|
}
|
|
|
|
static void file_is_not_present(CACHEFILE cf) {
|
|
int i;
|
|
for (i=0; i<n_present; i++) {
|
|
assert(present_items[i].cf!=cf);
|
|
}
|
|
}
|
|
|
|
|
|
static void flush_forchain (CACHEFILE f __attribute__((__unused__)), CACHEKEY key, void *value, long size __attribute__((__unused__)), int write_me __attribute__((__unused__)), int keep_me __attribute__((__unused__))) {
|
|
int *v = value;
|
|
//cachetable_print_state(ct);
|
|
//printf("Flush %lld %d\n", key, (int)value);
|
|
assert((long)v==(long)key);
|
|
item_becomes_not_present(f, key);
|
|
//print_ints();
|
|
}
|
|
|
|
static int fetch_forchain (CACHEFILE f __attribute__((__unused__)), CACHEKEY key, void**value, long *sizep __attribute__((__unused__)), void*extraargs) {
|
|
assert((long)extraargs==(long)key);
|
|
*value = (void*)(long)key;
|
|
return 0;
|
|
}
|
|
|
|
void verify_cachetable_against_present (void) {
|
|
int i;
|
|
for (i=0; i<n_present; i++) {
|
|
void *v;
|
|
int r;
|
|
assert(cachetable_maybe_get_and_pin(present_items[i].cf,
|
|
present_items[i].key,
|
|
&v)==0);
|
|
r = cachetable_unpin(present_items[i].cf, present_items[i].key, 0);
|
|
}
|
|
}
|
|
|
|
|
|
void test_chaining (void) {
|
|
/* Make sure that the hash chain and the LRU list don't get confused. */
|
|
CACHEFILE f[N_FILES];
|
|
enum { FILENAME_LEN=100 };
|
|
char fname[N_FILES][FILENAME_LEN];
|
|
int r;
|
|
long i, trial;
|
|
r = create_cachetable(&ct, N_PRESENT_LIMIT); assert(r==0);
|
|
for (i=0; i<N_FILES; i++) {
|
|
int r = snprintf(fname[i], FILENAME_LEN, "cachetabletest2.%ld.dat", i);
|
|
assert(r>0 && r<FILENAME_LEN);
|
|
unlink(fname[i]);
|
|
r = cachetable_openf(&f[i], ct, fname[i], O_RDWR|O_CREAT, 0777); assert(r==0);
|
|
}
|
|
for (i=0; i<N_PRESENT_LIMIT; i++) {
|
|
int fnum = i%N_FILES;
|
|
//printf("%s:%d Add %d\n", __FILE__, __LINE__, i);
|
|
r = cachetable_put(f[fnum], i, (void*)i, flush_forchain, fetch_forchain, (void*)i); assert(r==0);
|
|
item_becomes_present(f[fnum], i);
|
|
r = cachetable_unpin(f[fnum], i, 0); assert(r==0);
|
|
//print_ints();
|
|
}
|
|
for (trial=0; trial<TRIALS; trial++) {
|
|
if (n_present>0) {
|
|
// First touch some random ones
|
|
int whichone = random()%n_present;
|
|
void *value;
|
|
//printf("Touching %d (%lld, %p)\n", whichone, present_items[whichone].key, present_items[whichone].cf);
|
|
r = cachetable_get_and_pin(present_items[whichone].cf,
|
|
present_items[whichone].key,
|
|
&value,
|
|
flush_forchain,
|
|
fetch_forchain,
|
|
(void*)(long)present_items[whichone].key
|
|
);
|
|
assert(r==0);
|
|
r = cachetable_unpin(present_items[whichone].cf,
|
|
present_items[whichone].key,
|
|
0);
|
|
assert(r==0);
|
|
}
|
|
|
|
i += 1+ random()%100;
|
|
int fnum = i%N_FILES;
|
|
// i is always incrementing, so we need not worry about inserting a duplicate
|
|
//printf("%s:%d Add {%d,%p}\n", __FILE__, __LINE__, i, f[fnum]);
|
|
r = cachetable_put(f[fnum], i, (void*)i, flush_forchain, fetch_forchain, (void*)i); assert(r==0);
|
|
item_becomes_present(f[fnum], i);
|
|
//print_ints();
|
|
//cachetable_print_state(ct);
|
|
r = cachetable_unpin(f[fnum], i, 0); assert(r==0);
|
|
verify_cachetable_against_present();
|
|
|
|
if (random()%10==0) {
|
|
i = random()%N_FILES;
|
|
//printf("Close %d (%p), now n_present=%d\n", i, f[i], n_present);
|
|
//print_ints();
|
|
CACHEFILE oldcf=f[i];
|
|
r = cachefile_close(&f[i]); assert(r==0);
|
|
file_is_not_present(oldcf);
|
|
r = cachetable_openf(&f[i], ct, fname[i], O_RDWR, 0777); assert(r==0);
|
|
}
|
|
}
|
|
for (i=0; i<N_FILES; i++) {
|
|
r = cachefile_close(&f[i]); assert(r==0);
|
|
}
|
|
r = cachetable_close(&ct); assert(r==0);
|
|
}
|
|
|
|
int main (int argc __attribute__((__unused__)), char *argv[] __attribute__((__unused__))) {
|
|
test_chaining();
|
|
malloc_cleanup();
|
|
printf("ok\n");
|
|
return 0;
|
|
}
|