[Zope-Checkins] CVS: ZODB3/ZEO - simul.py:1.12.8.2.18.3
Jeremy Hylton
cvs-admin at zope.org
Tue Dec 2 22:45:02 EST 2003
Update of /cvs-repository/ZODB3/ZEO
In directory cvs.zope.org:/tmp/cvs-serv1324
Modified Files:
Tag: Zope-2_6-branch
simul.py
Log Message:
Add circular cache simulation.
Extend reporting code to support multiple simulator-specific
statistics.
=== ZODB3/ZEO/simul.py 1.12.8.2.18.2 => 1.12.8.2.18.3 ===
--- ZODB3/ZEO/simul.py:1.12.8.2.18.2 Wed Nov 26 16:44:39 2003
+++ ZODB3/ZEO/simul.py Tue Dec 2 22:45:00 2003
@@ -43,8 +43,9 @@
MB = 1000*1000
cachelimit = 20*MB
simclass = ZEOCacheSimulation
+ omicron = None
try:
- opts, args = getopt.getopt(sys.argv[1:], "bflyz2s:")
+ opts, args = getopt.getopt(sys.argv[1:], "bflyz2cs:o:")
except getopt.error, msg:
usage(msg)
return 2
@@ -63,11 +64,19 @@
cachelimit = int(float(a)*MB)
if o == '-2':
simclass = TwoQSimluation
+ if o == '-c':
+ simclass = CircularCacheSimulation
+ if o == '-o':
+ omicron = float(a)
if len(args) != 1:
usage("exactly one file argument required")
return 2
filename = args[0]
+ if omicron is not None and simclass != CircularCacheSimulation:
+ usage("-o flag only useful with -c (CircularCacheSimulation)")
+ return 2
+
# Open file
if filename.endswith(".gz"):
# Open gzipped file
@@ -93,7 +102,10 @@
return 1
# Create simulation object
- sim = simclass(cachelimit)
+ if omicron is not None:
+ sim = simclass(cachelimit, omicron)
+ else:
+ sim = simclass(cachelimit)
# Print output header
sim.printheader()
@@ -154,6 +166,9 @@
self.total_hits = 0 # Subclass must increment
self.total_invals = 0
self.total_writes = 0
+ if not hasattr(self, "extras"):
+ self.extras = (self.extraname,)
+ self.format = self.format + " %6s" * len(self.extras)
# Reset per-run statistics and set up simulation data
self.restart()
@@ -204,7 +219,7 @@
def inval(self, oid):
pass
- format = "%12s %9s %8s %8s %6s %6s %6s %6s"
+ format = "%12s %9s %8s %8s %6s %6s %6s"
# Subclass should override extraname to name known instance variables;
# if extraname is 'foo', both self.foo and self.total_foo must exist:
@@ -213,34 +228,41 @@
def printheader(self):
print "%s, cache size %s bytes" % (self.__class__.__name__,
addcommas(self.cachelimit))
- print self.format % (
- "START TIME", "DURATION", "LOADS", "HITS",
- "INVALS", "WRITES", self.extraname.upper(), "HITRATE")
+ self.extraheader()
+ extranames = tuple([s.upper() for s in self.extras])
+ args = ("START TIME", "DURATION", "LOADS", "HITS",
+ "INVALS", "WRITES", "HITRATE") + extranames
+ print self.format % args
+
+ def extraheader(self):
+ pass
nreports = 0
def report(self):
if self.loads:
self.nreports += 1
- print self.format % (
- time.ctime(self.ts0)[4:-8],
- duration(self.ts1 - self.ts0),
- self.loads, self.hits, self.invals, self.writes,
- getattr(self, self.extraname),
- hitrate(self.loads, self.hits))
+ args = (time.ctime(self.ts0)[4:-8],
+ duration(self.ts1 - self.ts0),
+ self.loads, self.hits, self.invals, self.writes,
+ hitrate(self.loads, self.hits))
+ args += tuple([getattr(self, name) for name in self.extras])
+ print self.format % args
def finish(self):
self.report()
if self.nreports > 1:
- print (self.format + " OVERALL") % (
+ args = (
time.ctime(self.epoch)[4:-8],
duration(self.ts1 - self.epoch),
self.total_loads,
self.total_hits,
self.total_invals,
self.total_writes,
- getattr(self, "total_" + self.extraname),
hitrate(self.total_loads, self.total_hits))
+ args += tuple([getattr(self, "total_" + name)
+ for name in self.extras])
+ print (self.format + " OVERALL") % args
class ZEOCacheSimulation(Simulation):
@@ -442,11 +464,12 @@
# object based, so, for example, it's hard to compute the number
# of oids to store in a1out based on the size of a1in.
- extraname = "evicts"
+ extras = "evicts", "hothits"
def __init__(self, cachelimit, outlen=100000):
Simulation.__init__(self, cachelimit)
self.total_evicts = 0
+ self.total_hothits = 0
self.a1out_limit = outlen
# An LRU queue of hot objects
@@ -535,6 +558,8 @@
if node.kind is am:
self.hits += 1
self.total_hits += 1
+ self.hothits += 1
+ self.total_hothits += 1
node.linkbefore(self.am)
elif node.kind is a1in:
self.hits += 1
@@ -574,6 +599,7 @@
Simulation.restart(self)
self.evicts = 0
+ self.hothits = 0
self.cache = {}
self.am_limit = 3 * self.cachelimit / 4
@@ -594,6 +620,146 @@
if next.kind != self.kind:
self.kind = next.kind
Node.linkbefore(self, next)
+
+class CircularCacheSimulation(Simulation):
+
+ # The cache is managed as a single file with a pointer that
+ # goes around the file, circularly, forever. New objects
+ # are written at the current pointer, evicting whatever was
+ # there previously.
+
+ # For each cache hit, there is some distance between the current
+ # pointer offset and the offset of the cached data record. The
+ # cache can be configured to copy objects to the current offset
+ # depending on how far away they are now. The omicron parameter
+ # specifies a percentage
+
+ extras = "evicts", "copies"
+
+ def __init__(self, cachelimit, omicron=0):
+ Simulation.__init__(self, cachelimit)
+ self.omicron = omicron
+ self.total_evicts = 0
+ self.total_copies = 0
+ # Current offset in file
+ self.offset = 0
+ # Map offset in file to tuple of size, oid
+ self.filemap = {0: (self.cachelimit, None)}
+ # Map oid to offset
+ self.cache = {}
+
+ def extraheader(self):
+ print "omicron = %s" % self.omicron
+
+ def restart(self):
+ Simulation.restart(self)
+ self.evicts = 0
+ self.copies = 0
+
+ def load(self, oid, size):
+ pos = self.cache.get(oid)
+ if pos is None:
+ self.add(oid, size)
+ else:
+ self.hits += 1
+ self.total_hits += 1
+ self.copy(oid, size, pos)
+
+ def check(self):
+ d = dict(self.filemap)
+ done = {}
+ while d:
+ pos, (size, oid) = d.popitem()
+ next = pos + size
+ assert next in d or next in done or next == self.cachelimit
+ done[pos] = pos
+ print "checked", self.offset
+
+ def dump(self):
+ L = list(self.filemap)
+ L.sort()
+ print "cache size", len(L)
+ next = None
+ for pos in L:
+ size, oid = self.filemap[pos]
+ if oid:
+ print pos, size, repr(oid), self.cache[oid]
+ else:
+ print pos, size, None
+ if next is not None:
+ assert next == pos
+ next = pos + size
+ print
+
+ def add(self, oid, size):
+ avail = self.makeroom(size)
+ assert oid not in self.cache
+ self.filemap[self.offset] = size, oid
+ self.cache[oid] = self.offset
+ self.offset += size
+ # All the space made available must be accounted for in filemap.
+ excess = avail - size
+ if excess:
+ self.filemap[self.offset] = excess, None
+
+ def makeroom(self, need):
+ if self.offset + need > self.cachelimit:
+ self.offset = 0
+ pos = self.offset
+ # Evict enough objects to make the necessary space available.
+ while need > 0:
+ try:
+ size, oid = self.filemap.pop(pos)
+ except:
+ raise
+
+ if oid is not None:
+ self.evicts += 1
+ self.total_evicts += 1
+ del self.cache[oid]
+ pos += size
+ need -= size
+ return pos - self.offset
+
+ def copy(self, oid, size, pos):
+ # Copy only if the distance is greater than omicron.
+ dist = self.offset - pos
+ if dist < 0:
+ dist += self.cachelimit
+ if dist < self.omicron * self.cachelimit:
+ self.copies += 1
+ self.total_copies += 1
+ self.filemap[pos] = size, None
+ del self.cache[oid]
+ self.add(oid, size)
+
+ def inval(self, oid):
+ pos = self.cache.get(oid)
+ if pos is None:
+ return
+ self.invals += 1
+ self.total_invals += 1
+ size, _oid = self.filemap[pos]
+ assert oid == _oid
+ self.filemap[pos] = size, None
+ del self.cache[oid]
+
+ def write(self, oid, size):
+ pos = self.cache.get(oid)
+ if pos is None:
+ return
+ oldsize, _oid = self.filemap[pos]
+ assert oid == _oid
+ if size == oldsize:
+ return
+ if size < oldsize:
+ excess = oldsize - size
+ self.filemap[pos] = size, oid
+ self.filemap[pos + size] = excess, None
+ else:
+ self.filemap[pos] = oldsize, None
+ del self.cache[oid]
+ self.add(oid, size)
class BuddyCacheSimulation(LRUCacheSimulation):
More information about the Zope-Checkins
mailing list