2014-06-24 02:59:15 +02:00
|
|
|
/*
|
|
|
|
** This utility program decodes and displays the content of the
|
|
|
|
** sqlite_stat4 table in the database file named on the command
|
|
|
|
** line.
|
|
|
|
*/
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <string.h>
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <ctype.h>
|
|
|
|
#include "sqlite3.h"
|
|
|
|
|
2015-10-29 14:48:15 +01:00
|
|
|
#define ISPRINT(X) isprint((unsigned char)(X))
|
|
|
|
|
2014-06-24 02:59:15 +02:00
|
|
|
typedef sqlite3_int64 i64; /* 64-bit signed integer type */
|
|
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
** Convert the var-int format into i64. Return the number of bytes
|
|
|
|
** in the var-int. Write the var-int value into *pVal.
|
|
|
|
*/
|
|
|
|
static int decodeVarint(const unsigned char *z, i64 *pVal){
|
|
|
|
i64 v = 0;
|
|
|
|
int i;
|
|
|
|
for(i=0; i<8; i++){
|
|
|
|
v = (v<<7) + (z[i]&0x7f);
|
|
|
|
if( (z[i]&0x80)==0 ){ *pVal = v; return i+1; }
|
|
|
|
}
|
|
|
|
v = (v<<8) + (z[i]&0xff);
|
|
|
|
*pVal = v;
|
|
|
|
return 9;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
int main(int argc, char **argv){
|
|
|
|
sqlite3 *db;
|
|
|
|
sqlite3_stmt *pStmt;
|
|
|
|
char *zIdx = 0;
|
|
|
|
int rc, j, x, y, mxHdr;
|
|
|
|
const unsigned char *aSample;
|
|
|
|
int nSample;
|
|
|
|
i64 iVal;
|
|
|
|
const char *zSep;
|
2014-11-04 22:38:45 +01:00
|
|
|
int iRow = 0;
|
2014-06-24 02:59:15 +02:00
|
|
|
|
|
|
|
if( argc!=2 ){
|
|
|
|
fprintf(stderr, "Usage: %s DATABASE-FILE\n", argv[0]);
|
|
|
|
exit(1);
|
|
|
|
}
|
|
|
|
rc = sqlite3_open(argv[1], &db);
|
|
|
|
if( rc!=SQLITE_OK || db==0 ){
|
|
|
|
fprintf(stderr, "Cannot open database file [%s]\n", argv[1]);
|
|
|
|
exit(1);
|
|
|
|
}
|
|
|
|
rc = sqlite3_prepare_v2(db,
|
|
|
|
"SELECT tbl||'.'||idx, nEq, nLT, nDLt, sample "
|
|
|
|
"FROM sqlite_stat4 ORDER BY 1", -1,
|
|
|
|
&pStmt, 0);
|
|
|
|
if( rc!=SQLITE_OK || pStmt==0 ){
|
|
|
|
fprintf(stderr, "%s\n", sqlite3_errmsg(db));
|
|
|
|
sqlite3_close(db);
|
|
|
|
exit(1);
|
|
|
|
}
|
|
|
|
while( SQLITE_ROW==sqlite3_step(pStmt) ){
|
|
|
|
if( zIdx==0 || strcmp(zIdx, (const char*)sqlite3_column_text(pStmt,0))!=0 ){
|
2014-11-04 22:38:45 +01:00
|
|
|
if( zIdx ) printf("\n**************************************"
|
|
|
|
"**************\n\n");
|
2014-06-24 02:59:15 +02:00
|
|
|
sqlite3_free(zIdx);
|
|
|
|
zIdx = sqlite3_mprintf("%s", sqlite3_column_text(pStmt,0));
|
2014-11-04 22:38:45 +01:00
|
|
|
iRow = 0;
|
2014-06-24 02:59:15 +02:00
|
|
|
}
|
2014-11-04 22:38:45 +01:00
|
|
|
printf("%s sample %d ------------------------------------\n", zIdx, ++iRow);
|
2014-06-24 02:59:15 +02:00
|
|
|
printf(" nEq = %s\n", sqlite3_column_text(pStmt,1));
|
|
|
|
printf(" nLt = %s\n", sqlite3_column_text(pStmt,2));
|
|
|
|
printf(" nDLt = %s\n", sqlite3_column_text(pStmt,3));
|
|
|
|
printf(" sample = x'");
|
|
|
|
aSample = sqlite3_column_blob(pStmt,4);
|
|
|
|
nSample = sqlite3_column_bytes(pStmt,4);
|
|
|
|
for(j=0; j<nSample; j++) printf("%02x", aSample[j]);
|
|
|
|
printf("'\n ");
|
|
|
|
zSep = " ";
|
|
|
|
x = decodeVarint(aSample, &iVal);
|
|
|
|
if( iVal<x || iVal>nSample ){
|
|
|
|
printf(" <error>\n");
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
y = mxHdr = (int)iVal;
|
|
|
|
while( x<mxHdr ){
|
|
|
|
int sz;
|
|
|
|
i64 v;
|
|
|
|
x += decodeVarint(aSample+x, &iVal);
|
|
|
|
if( x>mxHdr ) break;
|
|
|
|
if( iVal<0 ) break;
|
|
|
|
switch( iVal ){
|
|
|
|
case 0: sz = 0; break;
|
|
|
|
case 1: sz = 1; break;
|
|
|
|
case 2: sz = 2; break;
|
|
|
|
case 3: sz = 3; break;
|
|
|
|
case 4: sz = 4; break;
|
|
|
|
case 5: sz = 6; break;
|
|
|
|
case 6: sz = 8; break;
|
|
|
|
case 7: sz = 8; break;
|
|
|
|
case 8: sz = 0; break;
|
|
|
|
case 9: sz = 0; break;
|
|
|
|
case 10:
|
|
|
|
case 11: sz = 0; break;
|
|
|
|
default: sz = (int)(iVal-12)/2; break;
|
|
|
|
}
|
|
|
|
if( y+sz>nSample ) break;
|
|
|
|
if( iVal==0 ){
|
|
|
|
printf("%sNULL", zSep);
|
2014-06-24 22:19:21 +02:00
|
|
|
}else if( iVal==8 || iVal==9 ){
|
|
|
|
printf("%s%d", zSep, ((int)iVal)-8);
|
2014-06-24 02:59:15 +02:00
|
|
|
}else if( iVal<=7 ){
|
|
|
|
v = (signed char)aSample[y];
|
|
|
|
for(j=1; j<sz; j++){
|
|
|
|
v = (v<<8) + aSample[y+j];
|
|
|
|
}
|
|
|
|
if( iVal==7 ){
|
|
|
|
double r;
|
2017-07-15 19:57:31 +02:00
|
|
|
char *z;
|
2014-06-24 02:59:15 +02:00
|
|
|
memcpy(&r, &v, sizeof(r));
|
2017-07-15 19:57:31 +02:00
|
|
|
z = sqlite3_mprintf("%s%!.15g", zSep, r);
|
|
|
|
printf("%s", z);
|
|
|
|
sqlite3_free(z);
|
2014-06-24 02:59:15 +02:00
|
|
|
}else{
|
|
|
|
printf("%s%lld", zSep, v);
|
|
|
|
}
|
|
|
|
}else if( (iVal&1)==0 ){
|
|
|
|
printf("%sx'", zSep);
|
|
|
|
for(j=0; j<sz; j++){
|
|
|
|
printf("%02x", aSample[y+j]);
|
|
|
|
}
|
|
|
|
printf("'");
|
|
|
|
}else{
|
2016-12-09 00:52:38 +01:00
|
|
|
printf("%s'", zSep);
|
2014-06-24 02:59:15 +02:00
|
|
|
for(j=0; j<sz; j++){
|
|
|
|
char c = (char)aSample[y+j];
|
2015-10-29 14:48:15 +01:00
|
|
|
if( ISPRINT(c) ){
|
2016-12-09 00:52:38 +01:00
|
|
|
if( c=='\'' || c=='\\' ) putchar('\\');
|
2014-06-24 02:59:15 +02:00
|
|
|
putchar(c);
|
|
|
|
}else if( c=='\n' ){
|
|
|
|
printf("\\n");
|
|
|
|
}else if( c=='\t' ){
|
|
|
|
printf("\\t");
|
|
|
|
}else if( c=='\r' ){
|
|
|
|
printf("\\r");
|
|
|
|
}else{
|
|
|
|
printf("\\%03o", c);
|
|
|
|
}
|
|
|
|
}
|
2016-12-09 00:52:38 +01:00
|
|
|
printf("'");
|
2014-06-24 02:59:15 +02:00
|
|
|
}
|
|
|
|
zSep = ",";
|
|
|
|
y += sz;
|
|
|
|
}
|
|
|
|
printf("\n");
|
|
|
|
}
|
|
|
|
sqlite3_free(zIdx);
|
|
|
|
sqlite3_finalize(pStmt);
|
|
|
|
sqlite3_close(db);
|
|
|
|
return 0;
|
|
|
|
}
|