|  | #include "stdinc.h" | 
|  | #include "dat.h" | 
|  | #include "fns.h" | 
|  |  | 
|  | /* #define CHECK(x)	x */ | 
|  | #define CHECK(x) | 
|  |  | 
|  | typedef struct LumpCache	LumpCache; | 
|  |  | 
|  | enum | 
|  | { | 
|  | HashLog		= 9, | 
|  | HashSize	= 1<<HashLog, | 
|  | HashMask	= HashSize - 1, | 
|  | }; | 
|  |  | 
|  | struct LumpCache | 
|  | { | 
|  | QLock		lock; | 
|  | Rendez		full; | 
|  | Lump		*free;			/* list of available lumps */ | 
|  | u32int		allowed;		/* total allowable space for packets */ | 
|  | u32int		avail;			/* remaining space for packets */ | 
|  | u32int		now;			/* ticks for usage timestamps */ | 
|  | Lump		**heads;		/* hash table for finding address */ | 
|  | int		nheap;			/* number of available victims */ | 
|  | Lump		**heap;			/* heap for locating victims */ | 
|  | int		nblocks;		/* number of blocks allocated */ | 
|  | Lump		*blocks;		/* array of block descriptors */ | 
|  | }; | 
|  |  | 
|  | static LumpCache	lumpcache; | 
|  |  | 
|  | static void	delheap(Lump *db); | 
|  | static int	downheap(int i, Lump *b); | 
|  | static void	fixheap(int i, Lump *b); | 
|  | static int	upheap(int i, Lump *b); | 
|  | static Lump	*bumplump(void); | 
|  |  | 
|  | void | 
|  | initlumpcache(u32int size, u32int nblocks) | 
|  | { | 
|  | Lump *last, *b; | 
|  | int i; | 
|  |  | 
|  | lumpcache.full.l = &lumpcache.lock; | 
|  | lumpcache.nblocks = nblocks; | 
|  | lumpcache.allowed = size; | 
|  | lumpcache.avail = size; | 
|  | lumpcache.heads = MKNZ(Lump*, HashSize); | 
|  | lumpcache.heap = MKNZ(Lump*, nblocks); | 
|  | lumpcache.blocks = MKNZ(Lump, nblocks); | 
|  | setstat(StatLcacheSize, lumpcache.nblocks); | 
|  |  | 
|  | last = nil; | 
|  | for(i = 0; i < nblocks; i++){ | 
|  | b = &lumpcache.blocks[i]; | 
|  | b->type = TWID8; | 
|  | b->heap = TWID32; | 
|  | b->next = last; | 
|  | last = b; | 
|  | } | 
|  | lumpcache.free = last; | 
|  | lumpcache.nheap = 0; | 
|  | } | 
|  |  | 
|  | Lump* | 
|  | lookuplump(u8int *score, int type) | 
|  | { | 
|  | uint ms; | 
|  | Lump *b; | 
|  | u32int h; | 
|  |  | 
|  | ms = 0; | 
|  | trace(TraceLump, "lookuplump enter"); | 
|  |  | 
|  | h = hashbits(score, HashLog); | 
|  |  | 
|  | /* | 
|  | * look for the block in the cache | 
|  | */ | 
|  | qlock(&lumpcache.lock); | 
|  | CHECK(checklumpcache()); | 
|  | again: | 
|  | for(b = lumpcache.heads[h]; b != nil; b = b->next){ | 
|  | if(scorecmp(score, b->score)==0 && type == b->type){ | 
|  | addstat(StatLcacheHit, 1); | 
|  | trace(TraceLump, "lookuplump hit"); | 
|  | goto found; | 
|  | } | 
|  | } | 
|  |  | 
|  | trace(TraceLump, "lookuplump miss"); | 
|  |  | 
|  | /* | 
|  | * missed: locate the block with the oldest second to last use. | 
|  | * remove it from the heap, and fix up the heap. | 
|  | */ | 
|  | while(lumpcache.free == nil){ | 
|  | trace(TraceLump, "lookuplump bump"); | 
|  | CHECK(checklumpcache()); | 
|  | if(bumplump() == nil){ | 
|  | CHECK(checklumpcache()); | 
|  | logerr(EAdmin, "all lump cache blocks in use"); | 
|  | addstat(StatLcacheStall, 1); | 
|  | CHECK(checklumpcache()); | 
|  | rsleep(&lumpcache.full); | 
|  | CHECK(checklumpcache()); | 
|  | addstat(StatLcacheStall, -1); | 
|  | goto again; | 
|  | } | 
|  | CHECK(checklumpcache()); | 
|  | } | 
|  |  | 
|  | /* start timer on cache miss to avoid system call on cache hit */ | 
|  | ms = msec(); | 
|  |  | 
|  | addstat(StatLcacheMiss, 1); | 
|  | b = lumpcache.free; | 
|  | lumpcache.free = b->next; | 
|  |  | 
|  | /* | 
|  | * the new block has no last use, so assume it happens sometime in the middle | 
|  | ZZZ this is not reasonable | 
|  | */ | 
|  | b->used = (b->used2 + lumpcache.now) / 2; | 
|  |  | 
|  | /* | 
|  | * rechain the block on the correct hash chain | 
|  | */ | 
|  | b->next = lumpcache.heads[h]; | 
|  | lumpcache.heads[h] = b; | 
|  | if(b->next != nil) | 
|  | b->next->prev = b; | 
|  | b->prev = nil; | 
|  |  | 
|  | scorecp(b->score, score); | 
|  | b->type = type; | 
|  | b->size = 0; | 
|  | b->data = nil; | 
|  |  | 
|  | found: | 
|  | b->ref++; | 
|  | b->used2 = b->used; | 
|  | b->used = lumpcache.now++; | 
|  | if(b->heap != TWID32) | 
|  | fixheap(b->heap, b); | 
|  | CHECK(checklumpcache()); | 
|  | qunlock(&lumpcache.lock); | 
|  |  | 
|  |  | 
|  | addstat(StatLumpStall, 1); | 
|  | qlock(&b->lock); | 
|  | addstat(StatLumpStall, -1); | 
|  |  | 
|  | trace(TraceLump, "lookuplump exit"); | 
|  | addstat2(StatLcacheRead, 1, StatLcacheReadTime, ms ? msec()-ms : 0); | 
|  | return b; | 
|  | } | 
|  |  | 
|  | void | 
|  | insertlump(Lump *b, Packet *p) | 
|  | { | 
|  | u32int size; | 
|  |  | 
|  | /* | 
|  | * look for the block in the cache | 
|  | */ | 
|  | trace(TraceLump, "insertlump enter"); | 
|  | qlock(&lumpcache.lock); | 
|  | CHECK(checklumpcache()); | 
|  | again: | 
|  |  | 
|  | addstat(StatLcacheWrite, 1); | 
|  |  | 
|  | /* | 
|  | * missed: locate the block with the oldest second to last use. | 
|  | * remove it from the heap, and fix up the heap. | 
|  | */ | 
|  | size = packetasize(p); | 
|  | while(lumpcache.avail < size){ | 
|  | trace(TraceLump, "insertlump bump"); | 
|  | CHECK(checklumpcache()); | 
|  | if(bumplump() == nil){ | 
|  | logerr(EAdmin, "all lump cache blocks in use"); | 
|  | addstat(StatLcacheStall, 1); | 
|  | CHECK(checklumpcache()); | 
|  | rsleep(&lumpcache.full); | 
|  | CHECK(checklumpcache()); | 
|  | addstat(StatLcacheStall, -1); | 
|  | goto again; | 
|  | } | 
|  | CHECK(checklumpcache()); | 
|  | } | 
|  | b->data = p; | 
|  | b->size = size; | 
|  | lumpcache.avail -= size; | 
|  | CHECK(checklumpcache()); | 
|  | qunlock(&lumpcache.lock); | 
|  | trace(TraceLump, "insertlump exit"); | 
|  | } | 
|  |  | 
|  | void | 
|  | putlump(Lump *b) | 
|  | { | 
|  | if(b == nil) | 
|  | return; | 
|  |  | 
|  | trace(TraceLump, "putlump"); | 
|  | qunlock(&b->lock); | 
|  | qlock(&lumpcache.lock); | 
|  | CHECK(checklumpcache()); | 
|  | if(--b->ref == 0){ | 
|  | if(b->heap == TWID32) | 
|  | upheap(lumpcache.nheap++, b); | 
|  | trace(TraceLump, "putlump wakeup"); | 
|  | rwakeupall(&lumpcache.full); | 
|  | } | 
|  | CHECK(checklumpcache()); | 
|  | qunlock(&lumpcache.lock); | 
|  | } | 
|  |  | 
|  | /* | 
|  | * remove some lump from use and update the free list and counters | 
|  | */ | 
|  | static Lump* | 
|  | bumplump(void) | 
|  | { | 
|  | Lump *b; | 
|  | u32int h; | 
|  |  | 
|  | /* | 
|  | * remove blocks until we find one that is unused | 
|  | * referenced blocks are left in the heap even though | 
|  | * they can't be scavenged; this is simple a speed optimization | 
|  | */ | 
|  | CHECK(checklumpcache()); | 
|  | for(;;){ | 
|  | if(lumpcache.nheap == 0){ | 
|  | trace(TraceLump, "bumplump emptyheap"); | 
|  | return nil; | 
|  | } | 
|  | b = lumpcache.heap[0]; | 
|  | delheap(b); | 
|  | if(!b->ref){ | 
|  | trace(TraceLump, "bumplump wakeup"); | 
|  | rwakeupall(&lumpcache.full); | 
|  | break; | 
|  | } | 
|  | } | 
|  |  | 
|  | /* | 
|  | * unchain the block | 
|  | */ | 
|  | trace(TraceLump, "bumplump unchain"); | 
|  | if(b->prev == nil){ | 
|  | h = hashbits(b->score, HashLog); | 
|  | if(lumpcache.heads[h] != b) | 
|  | sysfatal("bad hash chains in lump cache"); | 
|  | lumpcache.heads[h] = b->next; | 
|  | }else | 
|  | b->prev->next = b->next; | 
|  | if(b->next != nil) | 
|  | b->next->prev = b->prev; | 
|  |  | 
|  | if(b->data != nil){ | 
|  | packetfree(b->data); | 
|  | b->data = nil; | 
|  | lumpcache.avail += b->size; | 
|  | b->size = 0; | 
|  | } | 
|  | b->type = TWID8; | 
|  |  | 
|  | b->next = lumpcache.free; | 
|  | lumpcache.free = b; | 
|  |  | 
|  | CHECK(checklumpcache()); | 
|  | trace(TraceLump, "bumplump exit"); | 
|  | return b; | 
|  | } | 
|  |  | 
|  | void | 
|  | emptylumpcache(void) | 
|  | { | 
|  | qlock(&lumpcache.lock); | 
|  | while(bumplump()) | 
|  | ; | 
|  | qunlock(&lumpcache.lock); | 
|  | } | 
|  |  | 
|  | /* | 
|  | * delete an arbitrary block from the heap | 
|  | */ | 
|  | static void | 
|  | delheap(Lump *db) | 
|  | { | 
|  | fixheap(db->heap, lumpcache.heap[--lumpcache.nheap]); | 
|  | db->heap = TWID32; | 
|  | } | 
|  |  | 
|  | /* | 
|  | * push an element up or down to it's correct new location | 
|  | */ | 
|  | static void | 
|  | fixheap(int i, Lump *b) | 
|  | { | 
|  | if(upheap(i, b) == i) | 
|  | downheap(i, b); | 
|  | } | 
|  |  | 
|  | static int | 
|  | upheap(int i, Lump *b) | 
|  | { | 
|  | Lump *bb; | 
|  | u32int now; | 
|  | int p; | 
|  |  | 
|  | now = lumpcache.now; | 
|  | for(; i != 0; i = p){ | 
|  | p = (i - 1) >> 1; | 
|  | bb = lumpcache.heap[p]; | 
|  | if(b->used2 - now >= bb->used2 - now) | 
|  | break; | 
|  | lumpcache.heap[i] = bb; | 
|  | bb->heap = i; | 
|  | } | 
|  |  | 
|  | lumpcache.heap[i] = b; | 
|  | b->heap = i; | 
|  | return i; | 
|  | } | 
|  |  | 
|  | static int | 
|  | downheap(int i, Lump *b) | 
|  | { | 
|  | Lump *bb; | 
|  | u32int now; | 
|  | int k; | 
|  |  | 
|  | now = lumpcache.now; | 
|  | for(; ; i = k){ | 
|  | k = (i << 1) + 1; | 
|  | if(k >= lumpcache.nheap) | 
|  | break; | 
|  | if(k + 1 < lumpcache.nheap && lumpcache.heap[k]->used2 - now > lumpcache.heap[k + 1]->used2 - now) | 
|  | k++; | 
|  | bb = lumpcache.heap[k]; | 
|  | if(b->used2 - now <= bb->used2 - now) | 
|  | break; | 
|  | lumpcache.heap[i] = bb; | 
|  | bb->heap = i; | 
|  | } | 
|  |  | 
|  | lumpcache.heap[i] = b; | 
|  | b->heap = i; | 
|  | return i; | 
|  | } | 
|  |  | 
|  | static void | 
|  | findblock(Lump *bb) | 
|  | { | 
|  | Lump *b, *last; | 
|  | int h; | 
|  |  | 
|  | last = nil; | 
|  | h = hashbits(bb->score, HashLog); | 
|  | for(b = lumpcache.heads[h]; b != nil; b = b->next){ | 
|  | if(last != b->prev) | 
|  | sysfatal("bad prev link"); | 
|  | if(b == bb) | 
|  | return; | 
|  | last = b; | 
|  | } | 
|  | sysfatal("block score=%V type=%#x missing from hash table", bb->score, bb->type); | 
|  | } | 
|  |  | 
|  | void | 
|  | checklumpcache(void) | 
|  | { | 
|  | Lump *b; | 
|  | u32int size, now, nfree; | 
|  | int i, k, refed; | 
|  |  | 
|  | now = lumpcache.now; | 
|  | for(i = 0; i < lumpcache.nheap; i++){ | 
|  | if(lumpcache.heap[i]->heap != i) | 
|  | sysfatal("lc: mis-heaped at %d: %d", i, lumpcache.heap[i]->heap); | 
|  | if(i > 0 && lumpcache.heap[(i - 1) >> 1]->used2 - now > lumpcache.heap[i]->used2 - now) | 
|  | sysfatal("lc: bad heap ordering"); | 
|  | k = (i << 1) + 1; | 
|  | if(k < lumpcache.nheap && lumpcache.heap[i]->used2 - now > lumpcache.heap[k]->used2 - now) | 
|  | sysfatal("lc: bad heap ordering"); | 
|  | k++; | 
|  | if(k < lumpcache.nheap && lumpcache.heap[i]->used2 - now > lumpcache.heap[k]->used2 - now) | 
|  | sysfatal("lc: bad heap ordering"); | 
|  | } | 
|  |  | 
|  | refed = 0; | 
|  | size = 0; | 
|  | for(i = 0; i < lumpcache.nblocks; i++){ | 
|  | b = &lumpcache.blocks[i]; | 
|  | if(b->data == nil && b->size != 0) | 
|  | sysfatal("bad size: %d data=%p", b->size, b->data); | 
|  | if(b->ref && b->heap == TWID32) | 
|  | refed++; | 
|  | if(b->type != TWID8){ | 
|  | findblock(b); | 
|  | size += b->size; | 
|  | } | 
|  | if(b->heap != TWID32 | 
|  | && lumpcache.heap[b->heap] != b) | 
|  | sysfatal("lc: spurious heap value"); | 
|  | } | 
|  | if(lumpcache.avail != lumpcache.allowed - size){ | 
|  | fprint(2, "mismatched available=%d and allowed=%d - used=%d space", lumpcache.avail, lumpcache.allowed, size); | 
|  | *(volatile int*)0=0; | 
|  | } | 
|  |  | 
|  | nfree = 0; | 
|  | for(b = lumpcache.free; b != nil; b = b->next){ | 
|  | if(b->type != TWID8 || b->heap != TWID32) | 
|  | sysfatal("lc: bad free list"); | 
|  | nfree++; | 
|  | } | 
|  |  | 
|  | if(lumpcache.nheap + nfree + refed != lumpcache.nblocks) | 
|  | sysfatal("lc: missing blocks: %d %d %d %d", lumpcache.nheap, refed, nfree, lumpcache.nblocks); | 
|  | } | 
|  |  |