Remove the timing and counter metrics from the OPFS VFS, as they're long-since unused so are just wasted CPU cycles.

FossilOrigin-Name: 67c035c34fb916e66bfe115a132660771e8fa2921e6d46756975c5df3ebcd73c
This commit is contained in:
stephan 2024-07-11 14:07:46 +00:00
parent 9b183a234c
commit 20ff50ad10
4 changed files with 10 additions and 141 deletions

View File

@ -87,35 +87,6 @@ const installAsyncProxy = function(){
const log = (...args)=>logImpl(2, ...args); const log = (...args)=>logImpl(2, ...args);
const warn = (...args)=>logImpl(1, ...args); const warn = (...args)=>logImpl(1, ...args);
const error = (...args)=>logImpl(0, ...args); const error = (...args)=>logImpl(0, ...args);
const metrics = Object.create(null);
metrics.reset = ()=>{
let k;
const r = (m)=>(m.count = m.time = m.wait = 0);
for(k in state.opIds){
r(metrics[k] = Object.create(null));
}
let s = metrics.s11n = Object.create(null);
s = s.serialize = Object.create(null);
s.count = s.time = 0;
s = metrics.s11n.deserialize = Object.create(null);
s.count = s.time = 0;
};
metrics.dump = ()=>{
let k, n = 0, t = 0, w = 0;
for(k in state.opIds){
const m = metrics[k];
n += m.count;
t += m.time;
w += m.wait;
m.avgTime = (m.count && m.time) ? (m.time / m.count) : 0;
}
console.log(globalThis?.location?.href,
"metrics for",globalThis?.location?.href,":\n",
metrics,
"\nTotal of",n,"op(s) for",t,"ms",
"approx",w,"ms spent waiting on OPFS APIs.");
console.log("Serialization metrics:",metrics.s11n);
};
/** /**
__openFiles is a map of sqlite3_file pointers (integers) to __openFiles is a map of sqlite3_file pointers (integers) to
@ -372,37 +343,6 @@ const installAsyncProxy = function(){
if(fh.readOnly) toss(opName+"(): File is read-only: "+fh.filenameAbs); if(fh.readOnly) toss(opName+"(): File is read-only: "+fh.filenameAbs);
}; };
/**
We track 2 different timers: the "metrics" timer records how much
time we spend performing work. The "wait" timer records how much
time we spend waiting on the underlying OPFS timer. See the calls
to mTimeStart(), mTimeEnd(), wTimeStart(), and wTimeEnd()
throughout this file to see how they're used.
*/
const __mTimer = Object.create(null);
__mTimer.op = undefined;
__mTimer.start = undefined;
const mTimeStart = (op)=>{
__mTimer.start = performance.now();
__mTimer.op = op;
//metrics[op] || toss("Maintenance required: missing metrics for",op);
++metrics[op].count;
};
const mTimeEnd = ()=>(
metrics[__mTimer.op].time += performance.now() - __mTimer.start
);
const __wTimer = Object.create(null);
__wTimer.op = undefined;
__wTimer.start = undefined;
const wTimeStart = (op)=>{
__wTimer.start = performance.now();
__wTimer.op = op;
//metrics[op] || toss("Maintenance required: missing metrics for",op);
};
const wTimeEnd = ()=>(
metrics[__wTimer.op].wait += performance.now() - __wTimer.start
);
/** /**
Gets set to true by the 'opfs-async-shutdown' command to quit the Gets set to true by the 'opfs-async-shutdown' command to quit the
wait loop. This is only intended for debugging purposes: we cannot wait loop. This is only intended for debugging purposes: we cannot
@ -413,37 +353,24 @@ const installAsyncProxy = function(){
/** /**
Asynchronous wrappers for sqlite3_vfs and sqlite3_io_methods Asynchronous wrappers for sqlite3_vfs and sqlite3_io_methods
methods, as well as helpers like mkdir(). Maintenance reminder: methods, as well as helpers like mkdir().
members are in alphabetical order to simplify finding them.
*/ */
const vfsAsyncImpls = { const vfsAsyncImpls = {
'opfs-async-metrics': async ()=>{
mTimeStart('opfs-async-metrics');
metrics.dump();
storeAndNotify('opfs-async-metrics', 0);
mTimeEnd();
},
'opfs-async-shutdown': async ()=>{ 'opfs-async-shutdown': async ()=>{
flagAsyncShutdown = true; flagAsyncShutdown = true;
storeAndNotify('opfs-async-shutdown', 0); storeAndNotify('opfs-async-shutdown', 0);
}, },
mkdir: async (dirname)=>{ mkdir: async (dirname)=>{
mTimeStart('mkdir');
let rc = 0; let rc = 0;
wTimeStart('mkdir');
try { try {
await getDirForFilename(dirname+"/filepart", true); await getDirForFilename(dirname+"/filepart", true);
}catch(e){ }catch(e){
state.s11n.storeException(2,e); state.s11n.storeException(2,e);
rc = state.sq3Codes.SQLITE_IOERR; rc = state.sq3Codes.SQLITE_IOERR;
}finally{
wTimeEnd();
} }
storeAndNotify('mkdir', rc); storeAndNotify('mkdir', rc);
mTimeEnd();
}, },
xAccess: async (filename)=>{ xAccess: async (filename)=>{
mTimeStart('xAccess');
/* OPFS cannot support the full range of xAccess() queries /* OPFS cannot support the full range of xAccess() queries
sqlite3 calls for. We can essentially just tell if the file sqlite3 calls for. We can essentially just tell if the file
is accessible, but if it is then it's automatically writable is accessible, but if it is then it's automatically writable
@ -456,26 +383,20 @@ const installAsyncProxy = function(){
accessible, non-0 means not accessible. accessible, non-0 means not accessible.
*/ */
let rc = 0; let rc = 0;
wTimeStart('xAccess');
try{ try{
const [dh, fn] = await getDirForFilename(filename); const [dh, fn] = await getDirForFilename(filename);
await dh.getFileHandle(fn); await dh.getFileHandle(fn);
}catch(e){ }catch(e){
state.s11n.storeException(2,e); state.s11n.storeException(2,e);
rc = state.sq3Codes.SQLITE_IOERR; rc = state.sq3Codes.SQLITE_IOERR;
}finally{
wTimeEnd();
} }
storeAndNotify('xAccess', rc); storeAndNotify('xAccess', rc);
mTimeEnd();
}, },
xClose: async function(fid/*sqlite3_file pointer*/){ xClose: async function(fid/*sqlite3_file pointer*/){
const opName = 'xClose'; const opName = 'xClose';
mTimeStart(opName);
__implicitLocks.delete(fid); __implicitLocks.delete(fid);
const fh = __openFiles[fid]; const fh = __openFiles[fid];
let rc = 0; let rc = 0;
wTimeStart(opName);
if(fh){ if(fh){
delete __openFiles[fid]; delete __openFiles[fid];
await closeSyncHandle(fh); await closeSyncHandle(fh);
@ -487,15 +408,11 @@ const installAsyncProxy = function(){
state.s11n.serialize(); state.s11n.serialize();
rc = state.sq3Codes.SQLITE_NOTFOUND; rc = state.sq3Codes.SQLITE_NOTFOUND;
} }
wTimeEnd();
storeAndNotify(opName, rc); storeAndNotify(opName, rc);
mTimeEnd();
}, },
xDelete: async function(...args){ xDelete: async function(...args){
mTimeStart('xDelete');
const rc = await vfsAsyncImpls.xDeleteNoWait(...args); const rc = await vfsAsyncImpls.xDeleteNoWait(...args);
storeAndNotify('xDelete', rc); storeAndNotify('xDelete', rc);
mTimeEnd();
}, },
xDeleteNoWait: async function(filename, syncDir = 0, recursive = false){ xDeleteNoWait: async function(filename, syncDir = 0, recursive = false){
/* The syncDir flag is, for purposes of the VFS API's semantics, /* The syncDir flag is, for purposes of the VFS API's semantics,
@ -511,7 +428,6 @@ const installAsyncProxy = function(){
is false. is false.
*/ */
let rc = 0; let rc = 0;
wTimeStart('xDelete');
try { try {
while(filename){ while(filename){
const [hDir, filenamePart] = await getDirForFilename(filename, false); const [hDir, filenamePart] = await getDirForFilename(filename, false);
@ -527,14 +443,11 @@ const installAsyncProxy = function(){
state.s11n.storeException(2,e); state.s11n.storeException(2,e);
rc = state.sq3Codes.SQLITE_IOERR_DELETE; rc = state.sq3Codes.SQLITE_IOERR_DELETE;
} }
wTimeEnd();
return rc; return rc;
}, },
xFileSize: async function(fid/*sqlite3_file pointer*/){ xFileSize: async function(fid/*sqlite3_file pointer*/){
mTimeStart('xFileSize');
const fh = __openFiles[fid]; const fh = __openFiles[fid];
let rc = 0; let rc = 0;
wTimeStart('xFileSize');
try{ try{
const sz = await (await getSyncHandle(fh,'xFileSize')).getSize(); const sz = await (await getSyncHandle(fh,'xFileSize')).getSize();
state.s11n.serialize(Number(sz)); state.s11n.serialize(Number(sz));
@ -543,19 +456,15 @@ const installAsyncProxy = function(){
rc = GetSyncHandleError.convertRc(e,state.sq3Codes.SQLITE_IOERR); rc = GetSyncHandleError.convertRc(e,state.sq3Codes.SQLITE_IOERR);
} }
await releaseImplicitLock(fh); await releaseImplicitLock(fh);
wTimeEnd();
storeAndNotify('xFileSize', rc); storeAndNotify('xFileSize', rc);
mTimeEnd();
}, },
xLock: async function(fid/*sqlite3_file pointer*/, xLock: async function(fid/*sqlite3_file pointer*/,
lockType/*SQLITE_LOCK_...*/){ lockType/*SQLITE_LOCK_...*/){
mTimeStart('xLock');
const fh = __openFiles[fid]; const fh = __openFiles[fid];
let rc = 0; let rc = 0;
const oldLockType = fh.xLock; const oldLockType = fh.xLock;
fh.xLock = lockType; fh.xLock = lockType;
if( !fh.syncHandle ){ if( !fh.syncHandle ){
wTimeStart('xLock');
try { try {
await getSyncHandle(fh,'xLock'); await getSyncHandle(fh,'xLock');
__implicitLocks.delete(fid); __implicitLocks.delete(fid);
@ -564,18 +473,14 @@ const installAsyncProxy = function(){
rc = GetSyncHandleError.convertRc(e,state.sq3Codes.SQLITE_IOERR_LOCK); rc = GetSyncHandleError.convertRc(e,state.sq3Codes.SQLITE_IOERR_LOCK);
fh.xLock = oldLockType; fh.xLock = oldLockType;
} }
wTimeEnd();
} }
storeAndNotify('xLock',rc); storeAndNotify('xLock',rc);
mTimeEnd();
}, },
xOpen: async function(fid/*sqlite3_file pointer*/, filename, xOpen: async function(fid/*sqlite3_file pointer*/, filename,
flags/*SQLITE_OPEN_...*/, flags/*SQLITE_OPEN_...*/,
opfsFlags/*OPFS_...*/){ opfsFlags/*OPFS_...*/){
const opName = 'xOpen'; const opName = 'xOpen';
mTimeStart(opName);
const create = (state.sq3Codes.SQLITE_OPEN_CREATE & flags); const create = (state.sq3Codes.SQLITE_OPEN_CREATE & flags);
wTimeStart('xOpen');
try{ try{
let hDir, filenamePart; let hDir, filenamePart;
try { try {
@ -583,8 +488,6 @@ const installAsyncProxy = function(){
}catch(e){ }catch(e){
state.s11n.storeException(1,e); state.s11n.storeException(1,e);
storeAndNotify(opName, state.sq3Codes.SQLITE_NOTFOUND); storeAndNotify(opName, state.sq3Codes.SQLITE_NOTFOUND);
mTimeEnd();
wTimeEnd();
return; return;
} }
if( state.opfsFlags.OPFS_UNLINK_BEFORE_OPEN & opfsFlags ){ if( state.opfsFlags.OPFS_UNLINK_BEFORE_OPEN & opfsFlags ){
@ -596,7 +499,6 @@ const installAsyncProxy = function(){
} }
} }
const hFile = await hDir.getFileHandle(filenamePart, {create}); const hFile = await hDir.getFileHandle(filenamePart, {create});
wTimeEnd();
const fh = Object.assign(Object.create(null),{ const fh = Object.assign(Object.create(null),{
fid: fid, fid: fid,
filenameAbs: filename, filenameAbs: filename,
@ -614,60 +516,47 @@ const installAsyncProxy = function(){
__openFiles[fid] = fh; __openFiles[fid] = fh;
storeAndNotify(opName, 0); storeAndNotify(opName, 0);
}catch(e){ }catch(e){
wTimeEnd();
error(opName,e); error(opName,e);
state.s11n.storeException(1,e); state.s11n.storeException(1,e);
storeAndNotify(opName, state.sq3Codes.SQLITE_IOERR); storeAndNotify(opName, state.sq3Codes.SQLITE_IOERR);
} }
mTimeEnd();
}, },
xRead: async function(fid/*sqlite3_file pointer*/,n,offset64){ xRead: async function(fid/*sqlite3_file pointer*/,n,offset64){
mTimeStart('xRead');
let rc = 0, nRead; let rc = 0, nRead;
const fh = __openFiles[fid]; const fh = __openFiles[fid];
try{ try{
wTimeStart('xRead');
nRead = (await getSyncHandle(fh,'xRead')).read( nRead = (await getSyncHandle(fh,'xRead')).read(
fh.sabView.subarray(0, n), fh.sabView.subarray(0, n),
{at: Number(offset64)} {at: Number(offset64)}
); );
wTimeEnd();
if(nRead < n){/* Zero-fill remaining bytes */ if(nRead < n){/* Zero-fill remaining bytes */
fh.sabView.fill(0, nRead, n); fh.sabView.fill(0, nRead, n);
rc = state.sq3Codes.SQLITE_IOERR_SHORT_READ; rc = state.sq3Codes.SQLITE_IOERR_SHORT_READ;
} }
}catch(e){ }catch(e){
if(undefined===nRead) wTimeEnd();
error("xRead() failed",e,fh); error("xRead() failed",e,fh);
state.s11n.storeException(1,e); state.s11n.storeException(1,e);
rc = GetSyncHandleError.convertRc(e,state.sq3Codes.SQLITE_IOERR_READ); rc = GetSyncHandleError.convertRc(e,state.sq3Codes.SQLITE_IOERR_READ);
} }
await releaseImplicitLock(fh); await releaseImplicitLock(fh);
storeAndNotify('xRead',rc); storeAndNotify('xRead',rc);
mTimeEnd();
}, },
xSync: async function(fid/*sqlite3_file pointer*/,flags/*ignored*/){ xSync: async function(fid/*sqlite3_file pointer*/,flags/*ignored*/){
mTimeStart('xSync');
const fh = __openFiles[fid]; const fh = __openFiles[fid];
let rc = 0; let rc = 0;
if(!fh.readOnly && fh.syncHandle){ if(!fh.readOnly && fh.syncHandle){
try { try {
wTimeStart('xSync');
await fh.syncHandle.flush(); await fh.syncHandle.flush();
}catch(e){ }catch(e){
state.s11n.storeException(2,e); state.s11n.storeException(2,e);
rc = state.sq3Codes.SQLITE_IOERR_FSYNC; rc = state.sq3Codes.SQLITE_IOERR_FSYNC;
} }
wTimeEnd();
} }
storeAndNotify('xSync',rc); storeAndNotify('xSync',rc);
mTimeEnd();
}, },
xTruncate: async function(fid/*sqlite3_file pointer*/,size){ xTruncate: async function(fid/*sqlite3_file pointer*/,size){
mTimeStart('xTruncate');
let rc = 0; let rc = 0;
const fh = __openFiles[fid]; const fh = __openFiles[fid];
wTimeStart('xTruncate');
try{ try{
affirmNotRO('xTruncate', fh); affirmNotRO('xTruncate', fh);
await (await getSyncHandle(fh,'xTruncate')).truncate(size); await (await getSyncHandle(fh,'xTruncate')).truncate(size);
@ -677,35 +566,27 @@ const installAsyncProxy = function(){
rc = GetSyncHandleError.convertRc(e,state.sq3Codes.SQLITE_IOERR_TRUNCATE); rc = GetSyncHandleError.convertRc(e,state.sq3Codes.SQLITE_IOERR_TRUNCATE);
} }
await releaseImplicitLock(fh); await releaseImplicitLock(fh);
wTimeEnd();
storeAndNotify('xTruncate',rc); storeAndNotify('xTruncate',rc);
mTimeEnd();
}, },
xUnlock: async function(fid/*sqlite3_file pointer*/, xUnlock: async function(fid/*sqlite3_file pointer*/,
lockType/*SQLITE_LOCK_...*/){ lockType/*SQLITE_LOCK_...*/){
mTimeStart('xUnlock');
let rc = 0; let rc = 0;
const fh = __openFiles[fid]; const fh = __openFiles[fid];
if( fh.syncHandle if( fh.syncHandle
&& state.sq3Codes.SQLITE_LOCK_NONE===lockType && state.sq3Codes.SQLITE_LOCK_NONE===lockType
/* Note that we do not differentiate between lock types in /* Note that we do not differentiate between lock types in
this VFS. We're either locked or unlocked. */ ){ this VFS. We're either locked or unlocked. */ ){
wTimeStart('xUnlock');
try { await closeSyncHandle(fh) } try { await closeSyncHandle(fh) }
catch(e){ catch(e){
state.s11n.storeException(1,e); state.s11n.storeException(1,e);
rc = state.sq3Codes.SQLITE_IOERR_UNLOCK; rc = state.sq3Codes.SQLITE_IOERR_UNLOCK;
} }
wTimeEnd();
} }
storeAndNotify('xUnlock',rc); storeAndNotify('xUnlock',rc);
mTimeEnd();
}, },
xWrite: async function(fid/*sqlite3_file pointer*/,n,offset64){ xWrite: async function(fid/*sqlite3_file pointer*/,n,offset64){
mTimeStart('xWrite');
let rc; let rc;
const fh = __openFiles[fid]; const fh = __openFiles[fid];
wTimeStart('xWrite');
try{ try{
affirmNotRO('xWrite', fh); affirmNotRO('xWrite', fh);
rc = ( rc = (
@ -719,9 +600,7 @@ const installAsyncProxy = function(){
rc = GetSyncHandleError.convertRc(e,state.sq3Codes.SQLITE_IOERR_WRITE); rc = GetSyncHandleError.convertRc(e,state.sq3Codes.SQLITE_IOERR_WRITE);
} }
await releaseImplicitLock(fh); await releaseImplicitLock(fh);
wTimeEnd();
storeAndNotify('xWrite',rc); storeAndNotify('xWrite',rc);
mTimeEnd();
} }
}/*vfsAsyncImpls*/; }/*vfsAsyncImpls*/;
@ -755,8 +634,6 @@ const installAsyncProxy = function(){
} }
}; };
state.s11n.deserialize = function(clear=false){ state.s11n.deserialize = function(clear=false){
++metrics.s11n.deserialize.count;
const t = performance.now();
const argc = viewU8[0]; const argc = viewU8[0];
const rc = argc ? [] : null; const rc = argc ? [] : null;
if(argc){ if(argc){
@ -781,12 +658,9 @@ const installAsyncProxy = function(){
} }
if(clear) viewU8[0] = 0; if(clear) viewU8[0] = 0;
//log("deserialize:",argc, rc); //log("deserialize:",argc, rc);
metrics.s11n.deserialize.time += performance.now() - t;
return rc; return rc;
}; };
state.s11n.serialize = function(...args){ state.s11n.serialize = function(...args){
const t = performance.now();
++metrics.s11n.serialize.count;
if(args.length){ if(args.length){
//log("serialize():",args); //log("serialize():",args);
const typeIds = []; const typeIds = [];
@ -817,7 +691,6 @@ const installAsyncProxy = function(){
}else{ }else{
viewU8[0] = 0; viewU8[0] = 0;
} }
metrics.s11n.serialize.time += performance.now() - t;
}; };
state.s11n.storeException = state.asyncS11nExceptions state.s11n.storeException = state.asyncS11nExceptions
@ -899,7 +772,6 @@ const installAsyncProxy = function(){
} }
}); });
initS11n(); initS11n();
metrics.reset();
log("init state",state); log("init state",state);
wPost('opfs-async-inited'); wPost('opfs-async-inited');
waitLoop(); waitLoop();
@ -912,9 +784,6 @@ const installAsyncProxy = function(){
waitLoop(); waitLoop();
} }
break; break;
case 'opfs-async-metrics':
metrics.dump();
break;
} }
}; };
wPost('opfs-async-loaded'); wPost('opfs-async-loaded');

View File

@ -443,7 +443,7 @@ const installOpfsVfs = function callee(options){
OPFS_UNLINK_BEFORE_OPEN: 0x02, OPFS_UNLINK_BEFORE_OPEN: 0x02,
/** /**
If true, any async routine which implicitly acquires a sync If true, any async routine which implicitly acquires a sync
access handle (i.e. an OPFS lock) will release that locks at access handle (i.e. an OPFS lock) will release that lock at
the end of the call which acquires it. If false, such the end of the call which acquires it. If false, such
"autolocks" are not released until the VFS is idle for some "autolocks" are not released until the VFS is idle for some
brief amount of time. brief amount of time.

View File

@ -1,5 +1,5 @@
C Add\ssome\sdocs\sexplaining\s[64f4f3cd878b1a72]. C Remove\sthe\stiming\sand\scounter\smetrics\sfrom\sthe\sOPFS\sVFS,\sas\sthey're\slong-since\sunused\sso\sare\sjust\swasted\sCPU\scycles.
D 2024-07-11T12:57:58.720 D 2024-07-11T14:07:46.554
F .fossil-settings/empty-dirs dbb81e8fc0401ac46a1491ab34a7f2c7c0452f2f06b54ebb845d024ca8283ef1 F .fossil-settings/empty-dirs dbb81e8fc0401ac46a1491ab34a7f2c7c0452f2f06b54ebb845d024ca8283ef1
F .fossil-settings/ignore-glob 35175cdfcf539b2318cb04a9901442804be81cd677d8b889fcc9149c21f239ea F .fossil-settings/ignore-glob 35175cdfcf539b2318cb04a9901442804be81cd677d8b889fcc9149c21f239ea
F LICENSE.md df5091916dbb40e6e9686186587125e1b2ff51f022cc334e886c19a0e9982724 F LICENSE.md df5091916dbb40e6e9686186587125e1b2ff51f022cc334e886c19a0e9982724
@ -616,10 +616,10 @@ F ext/wasm/api/sqlite3-api-oo1.c-pp.js c373cc04625a96bd3f01ce8ebeac93a5d38dbda62
F ext/wasm/api/sqlite3-api-prologue.js b347a0c5350247f90174a0ad9b9e72a99a5f837f31f78f60fcdb829b2ca30b63 F ext/wasm/api/sqlite3-api-prologue.js b347a0c5350247f90174a0ad9b9e72a99a5f837f31f78f60fcdb829b2ca30b63
F ext/wasm/api/sqlite3-api-worker1.c-pp.js 5cc22a3c0d52828cb32aad8691488719f47d27567e63e8bc8b832d74371c352d F ext/wasm/api/sqlite3-api-worker1.c-pp.js 5cc22a3c0d52828cb32aad8691488719f47d27567e63e8bc8b832d74371c352d
F ext/wasm/api/sqlite3-license-version-header.js 0c807a421f0187e778dc1078f10d2994b915123c1223fe752b60afdcd1263f89 F ext/wasm/api/sqlite3-license-version-header.js 0c807a421f0187e778dc1078f10d2994b915123c1223fe752b60afdcd1263f89
F ext/wasm/api/sqlite3-opfs-async-proxy.js 881af4643f037b6590c491cef5fac8bcdd4118993197a1fa222ccb8b01e3504a F ext/wasm/api/sqlite3-opfs-async-proxy.js e8f1df56e97a29004a95a2eddd26778f52c33b3e797d32d4b1b668a38e6493dd
F ext/wasm/api/sqlite3-vfs-helper.c-pp.js 3f828cc66758acb40e9c5b4dcfd87fd478a14c8fb7f0630264e6c7fa0e57515d F ext/wasm/api/sqlite3-vfs-helper.c-pp.js 3f828cc66758acb40e9c5b4dcfd87fd478a14c8fb7f0630264e6c7fa0e57515d
F ext/wasm/api/sqlite3-vfs-opfs-sahpool.c-pp.js 8433ee332d5f5e39fb19427fccb7bad7f44aa99b5504daad3343fc128c311e78 F ext/wasm/api/sqlite3-vfs-opfs-sahpool.c-pp.js 8433ee332d5f5e39fb19427fccb7bad7f44aa99b5504daad3343fc128c311e78
F ext/wasm/api/sqlite3-vfs-opfs.c-pp.js 1d0a2d9d9c8b1eb0126e1abd00ef26a9be9d348786d7e415556ecc345832eabe F ext/wasm/api/sqlite3-vfs-opfs.c-pp.js 258a0d7c2a952ee360c13d7e4551b11d3f4fbe6dec1df6162866eca4d54e8443
F ext/wasm/api/sqlite3-vtab-helper.c-pp.js a2fcbc3fecdd0eea229283584ebc122f29d98194083675dbe5cb2cf3a17fe309 F ext/wasm/api/sqlite3-vtab-helper.c-pp.js a2fcbc3fecdd0eea229283584ebc122f29d98194083675dbe5cb2cf3a17fe309
F ext/wasm/api/sqlite3-wasm.c 9267174b9b0591b4f71193542ab57adf95bb9415f7d3453acf4a8ca8052f5e6c F ext/wasm/api/sqlite3-wasm.c 9267174b9b0591b4f71193542ab57adf95bb9415f7d3453acf4a8ca8052f5e6c
F ext/wasm/api/sqlite3-worker1-promiser.c-pp.js 46f303ba8ddd1b2f0a391798837beddfa72e8c897038c8047eda49ce7d5ed46b F ext/wasm/api/sqlite3-worker1-promiser.c-pp.js 46f303ba8ddd1b2f0a391798837beddfa72e8c897038c8047eda49ce7d5ed46b
@ -2195,8 +2195,8 @@ F vsixtest/vsixtest.tcl 6a9a6ab600c25a91a7acc6293828957a386a8a93
F vsixtest/vsixtest.vcxproj.data 2ed517e100c66dc455b492e1a33350c1b20fbcdc F vsixtest/vsixtest.vcxproj.data 2ed517e100c66dc455b492e1a33350c1b20fbcdc
F vsixtest/vsixtest.vcxproj.filters 37e51ffedcdb064aad6ff33b6148725226cd608e F vsixtest/vsixtest.vcxproj.filters 37e51ffedcdb064aad6ff33b6148725226cd608e
F vsixtest/vsixtest_TemporaryKey.pfx e5b1b036facdb453873e7084e1cae9102ccc67a0 F vsixtest/vsixtest_TemporaryKey.pfx e5b1b036facdb453873e7084e1cae9102ccc67a0
P 64f4f3cd878b1a723caa0cd54b21f10fdb27d4e2ecb19fc233cbc8dd9d06f259 P 4fdf9a322fccf7d68b03d07a42a526f18f951e2e19228b7d41e01121b667cc86
R 14a1d62c6182eceaa22bb6feac58e8ac R 5866522ad6fbcc120fd61fe54902049d
U stephan U stephan
Z d0a56f821b38fd7e590974af97a06665 Z c3a08dde48aa22cdaaa7d36ef45af62c
# Remove this line to create a well-formed Fossil manifest. # Remove this line to create a well-formed Fossil manifest.

View File

@ -1 +1 @@
4fdf9a322fccf7d68b03d07a42a526f18f951e2e19228b7d41e01121b667cc86 67c035c34fb916e66bfe115a132660771e8fa2921e6d46756975c5df3ebcd73c