[Zodb-checkins] CVS: ZODB3/ZEO - ClientCache.py:1.37 stats.py:1.17
Guido van Rossum
guido@python.org
Wed, 18 Sep 2002 17:22:59 -0400
Update of /cvs-repository/ZODB3/ZEO
In directory cvs.zope.org:/tmp/cvs-serv822
Modified Files:
ClientCache.py stats.py
Log Message:
Cache policy improvement:
When load() finds a hit in the non-current file, copy it to the
current file (except when this would cause a cache flip).
It is hoped that this improves the cache performance.
(Now merged into the trunk! Jeremy & Jim want this in the ZEO and ZODB3 beta2 releases.)
=== ZODB3/ZEO/ClientCache.py 1.36 => 1.37 ===
--- ZODB3/ZEO/ClientCache.py:1.36 Fri Sep 13 16:59:28 2002
+++ ZODB3/ZEO/ClientCache.py Wed Sep 18 17:22:58 2002
@@ -303,6 +303,8 @@
if dlen:
data = read(dlen)
self._trace(0x2A, oid, version, h[19:], dlen)
+ if (p < 0) != self._current:
+ self._copytocurrent(ap, tlen, dlen, vlen, h, data)
return data, h[19:]
else:
self._trace(0x26, oid, version)
@@ -317,6 +319,9 @@
seek(ap+27)
data = read(dlen)
self._trace(0x2C, oid, version, h[19:], dlen)
+ if (p < 0) != self._current:
+ self._copytocurrent(ap, tlen, dlen, vlen, h,
+ data, vheader)
return data, h[19:]
else:
self._trace(0x28, oid, version)
@@ -326,10 +331,71 @@
vdata = read(vdlen)
vserial = read(8)
self._trace(0x2E, oid, version, vserial, vdlen)
+ if (p < 0) != self._current:
+ self._copytocurrent(ap, tlen, dlen, vlen, h,
+ None, vheader, vdata, vserial)
return vdata, vserial
finally:
self._release()
+ def _copytocurrent(self, pos, tlen, dlen, vlen, header,
+ data=None, vheader=None, vdata=None, vserial=None):
+ """Copy a cache hit from the non-current file to the current file.
+
+ Arguments are the file position in the non-current file,
+ record length, data length, version string length, header, and
+ optionally parts of the record that have already been read.
+ """
+ if self._pos + tlen > self._limit:
+ return # Don't let this cause a cache flip
+ assert len(header) == 27
+ if header[8] == 'n':
+ # Rewrite the header to drop the version data.
+ # This shortens the record.
+ tlen = 31 + dlen
+ vlen = 0
+ # (oid:8, status:1, tlen:4, vlen:2, dlen:4, serial:8)
+ header = header[:9] + pack(">IHI", tlen, vlen, dlen) + header[-8:]
+ else:
+ assert header[8] == 'v'
+ f = self._f[not self._current]
+ if data is None:
+ f.seek(pos+27)
+ data = f.read(dlen)
+ if len(data) != dlen:
+ return
+ l = [header, data]
+ if vlen:
+ assert vheader is not None
+ l.append(vheader)
+ assert (vdata is None) == (vserial is None)
+ if vdata is None:
+ vdlen = unpack(">I", vheader[-4:])[0]
+ f.seek(pos+27+dlen+vlen+4)
+ vdata = f.read(vdlen)
+ if len(vdata) != vdlen:
+ return
+ vserial = f.read(8)
+ if len(vserial) != 8:
+ return
+ l.append(vdata)
+ l.append(vserial)
+ else:
+ assert None is vheader is vdata is vserial
+ l.append(header[9:13]) # copy of tlen
+ g = self._f[self._current]
+ g.seek(self._pos)
+ g.writelines(l)
+ assert g.tell() == self._pos + tlen
+ oid = header[:8]
+ if self._current:
+ self._index[oid] = - self._pos
+ else:
+ self._index[oid] = self._pos
+ self._pos += tlen
+ self._trace(0x6A, header[:8], vlen and vheader[:-4] or '',
+ vlen and vserial or header[-8:], dlen)
+
def update(self, oid, serial, version, data):
self._acquire()
try:
@@ -476,7 +542,7 @@
l.append(stlen)
f = self._f[self._current]
f.seek(self._pos)
- f.write("".join(l))
+ f.writelines(l) # write all list elements
if self._current:
self._index[oid] = - self._pos
=== ZODB3/ZEO/stats.py 1.16 => 1.17 ===
--- ZODB3/ZEO/stats.py:1.16 Mon Sep 9 23:13:04 2002
+++ ZODB3/ZEO/stats.py Wed Sep 18 17:22:58 2002
@@ -364,6 +364,8 @@
0x5A: "store (non-version data present)",
0x5C: "store (only version data present)",
+ 0x6A: "_copytocurrent",
+
0x70: "checkSize (cache flip)",
}