|  | typedef struct Config		Config; | 
|  | typedef struct AMap		AMap; | 
|  | typedef struct AMapN		AMapN; | 
|  | typedef struct Arena		Arena; | 
|  | typedef struct AState	AState; | 
|  | typedef struct ArenaCIG	ArenaCIG; | 
|  | typedef struct ArenaHead	ArenaHead; | 
|  | typedef struct ArenaPart	ArenaPart; | 
|  | typedef struct ArenaTail	ArenaTail; | 
|  | typedef struct ATailStats	ATailStats; | 
|  | typedef struct CIBlock		CIBlock; | 
|  | typedef struct Clump		Clump; | 
|  | typedef struct ClumpInfo	ClumpInfo; | 
|  | typedef struct Graph Graph; | 
|  | typedef struct IAddr		IAddr; | 
|  | typedef struct IBucket		IBucket; | 
|  | typedef struct IEStream		IEStream; | 
|  | typedef struct IEntry		IEntry; | 
|  | typedef struct IFile		IFile; | 
|  | typedef struct ISect		ISect; | 
|  | typedef struct Index		Index; | 
|  | typedef struct Lump		Lump; | 
|  | typedef struct DBlock		DBlock; | 
|  | typedef struct Part		Part; | 
|  | typedef struct Statbin Statbin; | 
|  | typedef struct Statdesc	Statdesc; | 
|  | typedef struct Stats		Stats; | 
|  | typedef struct ZBlock		ZBlock; | 
|  | typedef struct Round	Round; | 
|  | typedef struct Bloom	Bloom; | 
|  |  | 
|  | #pragma incomplete IEStream | 
|  |  | 
|  | #define	TWID32	((u32int)~(u32int)0) | 
|  | #define	TWID64	((u64int)~(u64int)0) | 
|  | #define	TWID8	((u8int)~(u8int)0) | 
|  |  | 
|  | enum | 
|  | { | 
|  | /* | 
|  | * formerly fundamental constant, | 
|  | * now a server-imposed limitation. | 
|  | */ | 
|  | VtMaxLumpSize	= 56*1024, | 
|  |  | 
|  | ABlockLog		= 9,		/* log2(512), the quantum for reading arenas */ | 
|  | ANameSize		= 64, | 
|  | MaxDiskBlock		= 64*1024,	/* max. allowed size for a disk block */ | 
|  | MaxIoSize		= 64*1024,	/* max. allowed size for a disk io operation */ | 
|  | PartBlank		= 256*1024,	/* untouched section at beginning of partition */ | 
|  | HeadSize		= 512,		/* size of a header after PartBlank */ | 
|  | MinArenaSize		= 1*1024*1024,	/* smallest reasonable arena size */ | 
|  | IndexBase		= 1024*1024,	/* initial address to use in an index */ | 
|  | MaxIo			= 64*1024,	/* max size of a single read or write operation */ | 
|  | ICacheBits		= 16,		/* default bits for indexing icache */ | 
|  | MaxAMap			= 31*1024,	/* max. allowed arenas in an address mapping; must be < 32*1024 */ | 
|  | Unspecified		= TWID32, | 
|  |  | 
|  | /* | 
|  | * return codes from syncarena | 
|  | */ | 
|  | SyncDataErr	= 1 << 0,		/* problem reading the clump data */ | 
|  | SyncCIErr	= 1 << 1,		/* found erroneous clump directory entries */ | 
|  | SyncCIZero	= 1 << 2,		/* found unwritten clump directory entries */ | 
|  | SyncFixErr	= 1 << 3,		/* error writing fixed data */ | 
|  | SyncHeader	= 1 << 4,		/* altered header fields */ | 
|  |  | 
|  | /* | 
|  | * error severity | 
|  | */ | 
|  | EOk			= 0,		/* error expected in normal operation */ | 
|  | EStrange,				/* strange error that should be logged */ | 
|  | ECorrupt,				/* corrupted data found in arenas */ | 
|  | EICorrupt,				/* corrupted data found in index */ | 
|  | EAdmin,					/* should be brought to administrators' attention */ | 
|  | ECrash,					/* really bad internal error */ | 
|  | EBug,					/* a limitation which should be fixed */ | 
|  | EInconsist,				/* inconsistencies between index and arena */ | 
|  | EMax, | 
|  |  | 
|  | /* | 
|  | * internal disk formats for the venti archival storage system | 
|  | */ | 
|  | /* | 
|  | * magic numbers on disk | 
|  | */ | 
|  | _ClumpMagic		= 0xd15cb10cU,	/* clump header, deprecated */ | 
|  | ClumpFreeMagic		= 0,		/* free clump; terminates active clump log */ | 
|  |  | 
|  | ArenaPartMagic		= 0xa9e4a5e7U,	/* arena partition header */ | 
|  | ArenaMagic		= 0xf2a14eadU,	/* arena trailer */ | 
|  | ArenaHeadMagic		= 0xd15c4eadU,	/* arena header */ | 
|  |  | 
|  | BloomMagic		= 0xb1004eadU,	/* bloom filter header */ | 
|  | BloomMaxHash	= 32, | 
|  |  | 
|  | ISectMagic		= 0xd15c5ec7U,	/* index header */ | 
|  |  | 
|  | ArenaPartVersion	= 3, | 
|  | ArenaVersion4		= 4, | 
|  | ArenaVersion5		= 5, | 
|  | BloomVersion		= 1, | 
|  | IndexVersion		= 1, | 
|  | ISectVersion1		= 1, | 
|  | ISectVersion2		= 2, | 
|  |  | 
|  | /* | 
|  | * encodings of clumps on disk | 
|  | */ | 
|  | ClumpEErr		= 0,		/* can't happen */ | 
|  | ClumpENone,				/* plain */ | 
|  | ClumpECompress,				/* compressed */ | 
|  | ClumpEMax, | 
|  |  | 
|  | /* | 
|  | * sizes in bytes on disk | 
|  | */ | 
|  | U8Size			= 1, | 
|  | U16Size			= 2, | 
|  | U32Size			= 4, | 
|  | U64Size			= 8, | 
|  |  | 
|  | ArenaPartSize		= 4 * U32Size, | 
|  | ArenaSize4		= 2 * U64Size + 6 * U32Size + ANameSize + U8Size, | 
|  | ArenaSize5			= ArenaSize4 + U32Size, | 
|  | ArenaSize5a		= ArenaSize5 + 2 * U8Size + 2 * U32Size + 2 * U64Size, | 
|  | ArenaHeadSize4		= U64Size + 3 * U32Size + ANameSize, | 
|  | ArenaHeadSize5		= ArenaHeadSize4 + U32Size, | 
|  | BloomHeadSize	= 4 * U32Size, | 
|  | ISectSize1		= 7 * U32Size + 2 * ANameSize, | 
|  | ISectSize2		= ISectSize1 + U32Size, | 
|  | ClumpInfoSize		= U8Size + 2 * U16Size + VtScoreSize, | 
|  | ClumpSize		= ClumpInfoSize + U8Size + 3 * U32Size, | 
|  | MaxBloomSize		= 1<<(32-3),	/* 2^32 bits */ | 
|  | MaxBloomHash	= 32,		/* bits per score */ | 
|  | /* | 
|  | * BUG - The various block copies that manipulate entry buckets | 
|  | * would be faster if we bumped IBucketSize up to 8 and IEntrySize up to 40, | 
|  | * so that everything is word-aligned.  Buildindex is actually cpu-bound | 
|  | * by the (byte at a time) copying in qsort. | 
|  | */ | 
|  | IBucketSize		= U32Size + U16Size, | 
|  | IEntrySize		= U64Size + U32Size + 2*U16Size + 2*U8Size + VtScoreSize, | 
|  | IEntryTypeOff		= VtScoreSize + U32Size + U16Size + U64Size + U16Size, | 
|  | IEntryAddrOff		= VtScoreSize + U32Size + U16Size, | 
|  |  | 
|  | MaxClumpBlocks		=  (VtMaxLumpSize + ClumpSize + (1 << ABlockLog) - 1) >> ABlockLog, | 
|  |  | 
|  | IcacheFrac		= 1000000,	/* denominator */ | 
|  |  | 
|  | SleepForever		= 1000000000,	/* magic value for sleep time */ | 
|  | /* | 
|  | * dirty flags - order controls disk write order | 
|  | */ | 
|  | DirtyArena		= 1, | 
|  | DirtyArenaCib, | 
|  | DirtyArenaTrailer, | 
|  | DirtyMax, | 
|  |  | 
|  | ArenaCIGSize = 10*1024,	// about 0.5 MB worth of IEntry. | 
|  |  | 
|  | VentiZZZZZZZZ | 
|  | }; | 
|  |  | 
|  | extern char TraceDisk[]; | 
|  | extern char TraceLump[]; | 
|  | extern char TraceBlock[]; | 
|  | extern char TraceProc[]; | 
|  | extern char TraceWork[]; | 
|  | extern char TraceQuiet[]; | 
|  | extern char TraceRpc[]; | 
|  |  | 
|  | /* | 
|  | * results of parsing and initializing a config file | 
|  | */ | 
|  | struct Config | 
|  | { | 
|  | char		*index;			/* name of the index to initialize */ | 
|  | int		naparts;		/* arena partitions initialized */ | 
|  | ArenaPart	**aparts; | 
|  | int		nsects;			/* index sections initialized */ | 
|  | ISect		**sects; | 
|  | Bloom	*bloom;		/* bloom filter */ | 
|  | u32int	bcmem; | 
|  | u32int	mem; | 
|  | u32int	icmem; | 
|  | int		queuewrites; | 
|  | char*	haddr; | 
|  | char*	vaddr; | 
|  | char*	webroot; | 
|  | }; | 
|  |  | 
|  | /* | 
|  | * a Part is the low level interface to files or disks. | 
|  | * there are two main types of partitions | 
|  | *	arena paritions, which some number of arenas, each in a sub-partition. | 
|  | *	index partition, which only have one subpartition. | 
|  | */ | 
|  | struct Part | 
|  | { | 
|  | int		fd;			/* rock for accessing the disk */ | 
|  | int		mode; | 
|  | u64int		offset; | 
|  | u64int		size;			/* size of the partiton */ | 
|  | u32int		blocksize;		/* block size for reads and writes */ | 
|  | u32int		fsblocksize;	/* minimum file system block size */ | 
|  | char		*name; | 
|  | char		*filename; | 
|  | Channel		*writechan;		/* chan[dcache.nblock](DBlock*) */ | 
|  | }; | 
|  |  | 
|  | /* | 
|  | * a cached block from the partition | 
|  | * yuck -- most of this is internal structure for the cache | 
|  | * all other routines should only use data | 
|  | */ | 
|  | struct DBlock | 
|  | { | 
|  | u8int	*data; | 
|  |  | 
|  | Part	*part;			/* partition in which cached */ | 
|  | u64int	addr;			/* base address on the partition */ | 
|  | u32int	size;			/* amount of data available, not amount allocated; should go away */ | 
|  | u32int	mode; | 
|  | u32int	dirty; | 
|  | u32int	dirtying; | 
|  | DBlock	*next;			/* doubly linked hash chains */ | 
|  | DBlock	*prev; | 
|  | u32int	heap;			/* index in heap table */ | 
|  | u32int	used;			/* last reference times */ | 
|  | u32int	used2; | 
|  | u32int	ref;			/* reference count */ | 
|  | RWLock	lock;			/* for access to data only */ | 
|  | Channel	*writedonechan; | 
|  | void*	chanbuf[1];		/* buffer for the chan! */ | 
|  | }; | 
|  |  | 
|  | /* | 
|  | * a cached block from the partition | 
|  | * yuck -- most of this is internal structure for the cache | 
|  | * all other routines should only use data | 
|  | * double yuck -- this is mostly the same as a DBlock | 
|  | */ | 
|  | struct Lump | 
|  | { | 
|  | Packet	*data; | 
|  |  | 
|  | Part	*part;			/* partition in which cached */ | 
|  | u8int	score[VtScoreSize];	/* score of packet */ | 
|  | u8int	type;			/* type of packet */ | 
|  | u32int	size;			/* amount of data allocated to hold packet */ | 
|  | Lump	*next;			/* doubly linked hash chains */ | 
|  | Lump	*prev; | 
|  | u32int	heap;			/* index in heap table */ | 
|  | u32int	used;			/* last reference times */ | 
|  | u32int	used2; | 
|  | u32int	ref;			/* reference count */ | 
|  | QLock	lock;			/* for access to data only */ | 
|  | }; | 
|  |  | 
|  | /* | 
|  | * mapping between names and address ranges | 
|  | */ | 
|  | struct AMap | 
|  | { | 
|  | u64int		start; | 
|  | u64int		stop; | 
|  | char		name[ANameSize]; | 
|  | }; | 
|  |  | 
|  | /* | 
|  | * an AMap along with a length | 
|  | */ | 
|  | struct AMapN | 
|  | { | 
|  | int		n; | 
|  | AMap		*map; | 
|  | }; | 
|  |  | 
|  | /* | 
|  | * an ArenaPart is a partition made up of Arenas | 
|  | * it exists because most os's don't support many partitions, | 
|  | * and we want to have many different Arenas | 
|  | */ | 
|  | struct ArenaPart | 
|  | { | 
|  | Part		*part; | 
|  | u64int		size;			/* size of underlying partition, rounded down to blocks */ | 
|  | Arena		**arenas; | 
|  | u32int		tabbase;		/* base address of arena table on disk */ | 
|  | u32int		tabsize;		/* max. bytes in arena table */ | 
|  |  | 
|  | /* | 
|  | * fields stored on disk | 
|  | */ | 
|  | u32int		version; | 
|  | u32int		blocksize;		/* "optimal" block size for reads and writes */ | 
|  | u32int		arenabase;		/* base address of first arena */ | 
|  |  | 
|  | /* | 
|  | * stored in the arena mapping table on disk | 
|  | */ | 
|  | AMap		*map; | 
|  | int		narenas; | 
|  | }; | 
|  |  | 
|  | /* | 
|  | * info about one block in the clump info cache | 
|  | */ | 
|  | struct CIBlock | 
|  | { | 
|  | u32int		block;			/* blocks in the directory */ | 
|  | int		offset;			/* offsets of one clump in the data */ | 
|  | DBlock		*data; | 
|  | }; | 
|  |  | 
|  | /* | 
|  | * Statistics kept in the tail. | 
|  | */ | 
|  | struct ATailStats | 
|  | { | 
|  | u32int		clumps;		/* number of clumps */ | 
|  | u32int		cclumps;		/* number of compressed clumps */ | 
|  | u64int		used; | 
|  | u64int		uncsize; | 
|  | u8int		sealed; | 
|  | }; | 
|  |  | 
|  | /* | 
|  | * Arena state - represents a point in the data log | 
|  | */ | 
|  | struct AState | 
|  | { | 
|  | Arena		*arena; | 
|  | u64int		aa;			/* index address */ | 
|  | ATailStats		stats; | 
|  | }; | 
|  |  | 
|  | /* | 
|  | * an Arena is a log of Clumps, preceeded by an ArenaHeader, | 
|  | * and followed by a Arena, each in one disk block. | 
|  | * struct on disk is not always up to date, but should be self-consistent. | 
|  | * to sync after reboot, follow clumps starting at used until ClumpFreeMagic if found. | 
|  | * <struct name="Arena" type="Arena *"> | 
|  | *	<field name="name" val="s->name" type="AName"/> | 
|  | *	<field name="version" val="s->version" type="U32int"/> | 
|  | *	<field name="partition" val="s->part->name" type="AName"/> | 
|  | *	<field name="blocksize" val="s->blocksize" type="U32int"/> | 
|  | *	<field name="start" val="s->base" type="U64int"/> | 
|  | *	<field name="stop" val="s->base+2*s->blocksize" type="U64int"/> | 
|  | *	<field name="created" val="s->ctime" type="U32int"/> | 
|  | *	<field name="modified" val="s->wtime" type="U32int"/> | 
|  | *	<field name="sealed" val="s->sealed" type="Sealed"/> | 
|  | *	<field name="score" val="s->score" type="Score"/> | 
|  | *	<field name="clumps" val="s->clumps" type="U32int"/> | 
|  | *	<field name="compressedclumps" val="s->cclumps" type="U32int"/> | 
|  | *	<field name="data" val="s->uncsize" type="U64int"/> | 
|  | *	<field name="compresseddata" val="s->used - s->clumps * ClumpSize" type="U64int"/> | 
|  | *	<field name="storage" val="s->used + s->clumps * ClumpInfoSize" type="U64int"/> | 
|  | * </struct> | 
|  | */ | 
|  | struct Arena | 
|  | { | 
|  | QLock		lock;			/* lock for arena fields, writing to disk */ | 
|  | Part		*part;			/* partition in which arena lives */ | 
|  | int		blocksize;		/* size of block to read or write */ | 
|  | u64int		base;			/* base address on disk */ | 
|  | u64int		size;			/* total space in the arena */ | 
|  | u8int		score[VtScoreSize];	/* score of the entire sealed & summed arena */ | 
|  |  | 
|  | int		clumpmax;		/* ClumpInfos per block */ | 
|  | AState		mem; | 
|  | int		inqueue; | 
|  |  | 
|  | /* | 
|  | * fields stored on disk | 
|  | */ | 
|  | u32int		version; | 
|  | char		name[ANameSize];	/* text label */ | 
|  | ATailStats		memstats; | 
|  | ATailStats		diskstats; | 
|  | u32int		ctime;			/* first time a block was written */ | 
|  | u32int		wtime;			/* last time a block was written */ | 
|  | u32int		clumpmagic; | 
|  |  | 
|  | ArenaCIG	*cig; | 
|  | int	ncig; | 
|  | }; | 
|  |  | 
|  | struct ArenaCIG | 
|  | { | 
|  | u64int	offset;  // from arena base | 
|  | }; | 
|  |  | 
|  | /* | 
|  | * redundant storage of some fields at the beginning of each arena | 
|  | */ | 
|  | struct ArenaHead | 
|  | { | 
|  | u32int		version; | 
|  | char		name[ANameSize]; | 
|  | u32int		blocksize; | 
|  | u64int		size; | 
|  | u32int		clumpmagic; | 
|  | }; | 
|  |  | 
|  | /* | 
|  | * most interesting meta information for a clump. | 
|  | * stored in each clump's header and in the Arena's directory, | 
|  | * stored in reverse order just prior to the arena trailer | 
|  | */ | 
|  | struct ClumpInfo | 
|  | { | 
|  | u8int		type; | 
|  | u16int		size;			/* size of disk data, not including header */ | 
|  | u16int		uncsize;		/* size of uncompressed data */ | 
|  | u8int		score[VtScoreSize];	/* score of the uncompressed data only */ | 
|  | }; | 
|  |  | 
|  | /* | 
|  | * header for an immutable clump of data | 
|  | */ | 
|  | struct Clump | 
|  | { | 
|  | ClumpInfo	info; | 
|  | u8int		encoding; | 
|  | u32int		creator;		/* initial client which wrote the block */ | 
|  | u32int		time;			/* creation at gmt seconds since 1/1/1970 */ | 
|  | }; | 
|  |  | 
|  | /* | 
|  | * index of all clumps according to their score | 
|  | * this is just a wrapper to tie together the index sections | 
|  | * <struct name="Index" type="Index *"> | 
|  | *	<field name="name" val="s->name" type="AName"/> | 
|  | *	<field name="version" val="s->version" type="U32int"/> | 
|  | *	<field name="blocksize" val="s->blocksize" type="U32int"/> | 
|  | *	<field name="tabsize" val="s->tabsize" type="U32int"/> | 
|  | *	<field name="buckets" val="s->buckets" type="U32int"/> | 
|  | *	<field name="buckdiv" val="s->div" type="U32int"/> | 
|  | *	<field name="bitblocks" val="s->div" type="U32int"/> | 
|  | *	<field name="maxdepth" val="s->div" type="U32int"/> | 
|  | *	<field name="bitkeylog" val="s->div" type="U32int"/> | 
|  | *	<field name="bitkeymask" val="s->div" type="U32int"/> | 
|  | *	<array name="sect" val="&s->smap[i]" elems="s->nsects" type="Amap"/> | 
|  | *	<array name="amap" val="&s->amap[i]" elems="s->narenas" type="Amap"/> | 
|  | *	<array name="arena" val="s->arenas[i]" elems="s->narenas" type="Arena"/> | 
|  | * </struct> | 
|  | * <struct name="Amap" type="AMap *"> | 
|  | *	<field name="name" val="s->name" type="AName"/> | 
|  | *	<field name="start" val="s->start" type="U64int"/> | 
|  | *	<field name="stop" val="s->stop" type="U64int"/> | 
|  | * </struct> | 
|  | */ | 
|  | struct Index | 
|  | { | 
|  | u32int		div;			/* divisor for mapping score to bucket */ | 
|  | u32int		buckets;		/* last bucket used in disk hash table */ | 
|  | u32int		blocksize; | 
|  | u32int		tabsize;		/* max. bytes in index config */ | 
|  |  | 
|  | int		mapalloc;		/* first arena to check when adding a lump */ | 
|  | Arena		**arenas;		/* arenas in the mapping */ | 
|  | ISect		**sects;		/* sections which hold the buckets */ | 
|  | Bloom		*bloom;	/* bloom filter */ | 
|  |  | 
|  | /* | 
|  | * fields stored in config file | 
|  | */ | 
|  | u32int		version; | 
|  | char		name[ANameSize];	/* text label */ | 
|  | int		nsects; | 
|  | AMap		*smap;			/* mapping of buckets to index sections */ | 
|  | int		narenas; | 
|  | AMap		*amap;			/* mapping from index addesses to arenas */ | 
|  |  | 
|  | QLock	writing; | 
|  | }; | 
|  |  | 
|  | /* | 
|  | * one part of the bucket storage for an index. | 
|  | * the index blocks are sequentially allocated | 
|  | * across all of the sections. | 
|  | */ | 
|  | struct ISect | 
|  | { | 
|  | Part		*part; | 
|  | int		blocklog;		/* log2(blocksize) */ | 
|  | int		buckmax;		/* max. entries in a index bucket */ | 
|  | u32int		tabbase;		/* base address of index config table on disk */ | 
|  | u32int		tabsize;		/* max. bytes in index config */ | 
|  | Channel	*writechan; | 
|  | Channel	*writedonechan; | 
|  | void		*ig;		/* used by buildindex only */ | 
|  | int		ng; | 
|  |  | 
|  | /* | 
|  | * fields stored on disk | 
|  | */ | 
|  | u32int		version; | 
|  | u32int		bucketmagic; | 
|  | char		name[ANameSize];	/* text label */ | 
|  | char		index[ANameSize];	/* index owning the section */ | 
|  | u32int		blocksize;		/* size of hash buckets in index */ | 
|  | u32int		blockbase;		/* address of start of on disk index table */ | 
|  | u32int		blocks;			/* total blocks on disk; some may be unused */ | 
|  | u32int		start;			/* first bucket in this section */ | 
|  | u32int		stop;			/* limit of buckets in this section */ | 
|  | }; | 
|  |  | 
|  | /* | 
|  | * externally interesting part of an IEntry | 
|  | */ | 
|  | struct IAddr | 
|  | { | 
|  | u64int		addr; | 
|  | u16int		size;			/* uncompressed size */ | 
|  | u8int		type;			/* type of block */ | 
|  | u8int		blocks;			/* arena io quanta for Clump + data */ | 
|  | }; | 
|  |  | 
|  | /* | 
|  | * entries in the index | 
|  | * kept in IBuckets in the disk index table, | 
|  | * cached in the memory ICache. | 
|  | */ | 
|  | struct IEntry | 
|  | { | 
|  | /* on disk data - 32 bytes*/ | 
|  | u8int	score[VtScoreSize]; | 
|  | IAddr	ia; | 
|  |  | 
|  | IEntry	*nexthash; | 
|  | IEntry	*nextdirty; | 
|  | IEntry	*next; | 
|  | IEntry	*prev; | 
|  | u8int	state; | 
|  | }; | 
|  | enum { | 
|  | IEClean = 0, | 
|  | IEDirty = 1, | 
|  | IESummary = 2, | 
|  | }; | 
|  |  | 
|  | /* | 
|  | * buckets in the on disk index table | 
|  | */ | 
|  | struct IBucket | 
|  | { | 
|  | u16int		n;			/* number of active indices */ | 
|  | u32int		buck;		/* used by buildindex/checkindex only */ | 
|  | u8int		*data; | 
|  | }; | 
|  |  | 
|  | /* | 
|  | * temporary buffers used by individual threads | 
|  | */ | 
|  | struct ZBlock | 
|  | { | 
|  | u32int		len; | 
|  | u32int		_size; | 
|  | u8int		*data; | 
|  | u8int		*free; | 
|  | }; | 
|  |  | 
|  | /* | 
|  | * simple input buffer for a '\0' terminated text file | 
|  | */ | 
|  | struct IFile | 
|  | { | 
|  | char		*name;				/* name of the file */ | 
|  | ZBlock		*b;				/* entire contents of file */ | 
|  | u32int		pos;				/* current position in the file */ | 
|  | }; | 
|  |  | 
|  | struct Statdesc | 
|  | { | 
|  | char *name; | 
|  | ulong max; | 
|  | }; | 
|  |  | 
|  | /* keep in sync with stats.c:/statdesc and httpd.c:/graphname*/ | 
|  | enum | 
|  | { | 
|  | StatRpcTotal, | 
|  | StatRpcRead, | 
|  | StatRpcReadOk, | 
|  | StatRpcReadFail, | 
|  | StatRpcReadBytes, | 
|  | StatRpcReadTime, | 
|  | StatRpcReadCached, | 
|  | StatRpcReadCachedTime, | 
|  | StatRpcReadUncached, | 
|  | StatRpcReadUncachedTime, | 
|  | StatRpcWrite, | 
|  | StatRpcWriteNew, | 
|  | StatRpcWriteOld, | 
|  | StatRpcWriteFail, | 
|  | StatRpcWriteBytes, | 
|  | StatRpcWriteTime, | 
|  | StatRpcWriteNewTime, | 
|  | StatRpcWriteOldTime, | 
|  |  | 
|  | StatLcacheHit, | 
|  | StatLcacheMiss, | 
|  | StatLcacheRead, | 
|  | StatLcacheWrite, | 
|  | StatLcacheSize, | 
|  | StatLcacheStall, | 
|  | StatLcacheReadTime, | 
|  |  | 
|  | StatDcacheHit, | 
|  | StatDcacheMiss, | 
|  | StatDcacheLookup, | 
|  | StatDcacheRead, | 
|  | StatDcacheWrite, | 
|  | StatDcacheDirty, | 
|  | StatDcacheSize, | 
|  | StatDcacheFlush, | 
|  | StatDcacheStall, | 
|  | StatDcacheLookupTime, | 
|  |  | 
|  | StatDblockStall, | 
|  | StatLumpStall, | 
|  |  | 
|  | StatIcacheHit, | 
|  | StatIcacheMiss, | 
|  | StatIcacheRead, | 
|  | StatIcacheWrite, | 
|  | StatIcacheFill, | 
|  | StatIcachePrefetch, | 
|  | StatIcacheDirty, | 
|  | StatIcacheSize, | 
|  | StatIcacheFlush, | 
|  | StatIcacheStall, | 
|  | StatIcacheReadTime, | 
|  | StatIcacheLookup, | 
|  | StatScacheHit, | 
|  | StatScachePrefetch, | 
|  |  | 
|  | StatBloomHit, | 
|  | StatBloomMiss, | 
|  | StatBloomFalseMiss, | 
|  | StatBloomLookup, | 
|  | StatBloomOnes, | 
|  | StatBloomBits, | 
|  |  | 
|  | StatApartRead, | 
|  | StatApartReadBytes, | 
|  | StatApartWrite, | 
|  | StatApartWriteBytes, | 
|  |  | 
|  | StatIsectRead, | 
|  | StatIsectReadBytes, | 
|  | StatIsectWrite, | 
|  | StatIsectWriteBytes, | 
|  |  | 
|  | StatSumRead, | 
|  | StatSumReadBytes, | 
|  |  | 
|  | StatCigLoad, | 
|  | StatCigLoadTime, | 
|  |  | 
|  | NStat | 
|  | }; | 
|  |  | 
|  | extern Statdesc statdesc[NStat]; | 
|  |  | 
|  | /* | 
|  | * statistics about the operation of the server | 
|  | * mainly for performance monitoring and profiling. | 
|  | */ | 
|  | struct Stats | 
|  | { | 
|  | ulong		now; | 
|  | ulong		n[NStat]; | 
|  | }; | 
|  |  | 
|  | struct Statbin | 
|  | { | 
|  | uint nsamp; | 
|  | uint min; | 
|  | uint max; | 
|  | uint avg; | 
|  | }; | 
|  |  | 
|  | struct Graph | 
|  | { | 
|  | long (*fn)(Stats*, Stats*, void*); | 
|  | void *arg; | 
|  | long t0; | 
|  | long t1; | 
|  | long min; | 
|  | long max; | 
|  | long wid; | 
|  | long ht; | 
|  | int fill; | 
|  | }; | 
|  |  | 
|  | /* | 
|  | * for kicking background processes that run one round after another after another | 
|  | */ | 
|  | struct Round | 
|  | { | 
|  | QLock	lock; | 
|  | Rendez	start; | 
|  | Rendez	finish; | 
|  | Rendez	delaywait; | 
|  | int		delaytime; | 
|  | int		delaykick; | 
|  | char*	name; | 
|  | int		last; | 
|  | int		current; | 
|  | int		next; | 
|  | int		doanother; | 
|  | }; | 
|  |  | 
|  | /* | 
|  | * Bloom filter of stored block hashes | 
|  | */ | 
|  | struct Bloom | 
|  | { | 
|  | RWLock lk;		/* protects nhash, nbits, tab, mb */ | 
|  | QLock mod;		/* one marker at a time, protects nb */ | 
|  | int nhash; | 
|  | ulong size;		/* bytes in tab */ | 
|  | ulong bitmask;		/* to produce bit index */ | 
|  | u8int *data; | 
|  | Part *part; | 
|  | Channel *writechan; | 
|  | Channel *writedonechan; | 
|  | }; | 
|  |  | 
|  | extern	Index		*mainindex; | 
|  | extern	u32int		maxblocksize;		/* max. block size used by any partition */ | 
|  | extern	int		paranoid;		/* should verify hashes on disk read */ | 
|  | extern	int		queuewrites;		/* put all lump writes on a queue and finish later */ | 
|  | extern	int		readonly;		/* only allowed to read the disk data */ | 
|  | extern	Stats		stats; | 
|  | extern	u8int		zeroscore[VtScoreSize]; | 
|  | extern	int		compressblocks; | 
|  | extern	int		writestodevnull;	/* dangerous - for performance debugging */ | 
|  | extern	int		bootstrap;		/* writes but does not index - cannot read */ | 
|  | extern	int		collectstats; | 
|  | extern	QLock	memdrawlock; | 
|  | extern	int		icachesleeptime; | 
|  | extern	int		minicachesleeptime; | 
|  | extern	int		arenasumsleeptime; | 
|  | extern	int		manualscheduling; | 
|  | extern	int		l0quantum; | 
|  | extern	int		l1quantum; | 
|  | extern	int		ignorebloom; | 
|  | extern	int		icacheprefetch; | 
|  | extern	int		syncwrites; | 
|  | extern	int		debugarena; /* print in arena error msgs; -1==unknown */ | 
|  |  | 
|  | extern	Stats	*stathist; | 
|  | extern	int	nstathist; | 
|  | extern	ulong	stattime; | 
|  |  | 
|  | #ifndef PLAN9PORT | 
|  | #pragma varargck type "V" uchar* | 
|  | #define ODIRECT 0 | 
|  | #endif | 
|  |  |