blob: 51c89da499bc9b53a6b75785677a58ecd3169d02 [file] [log] [blame]
drh0de8c112002-07-06 16:32:14 +00001/*
2** A utility for printing an SQLite database journal.
3*/
4#include <stdio.h>
5#include <ctype.h>
drh0de8c112002-07-06 16:32:14 +00006#include <stdlib.h>
drh4f788ec2010-10-01 13:28:43 +00007#include <string.h>
drh0de8c112002-07-06 16:32:14 +00008
drh4f788ec2010-10-01 13:28:43 +00009/*
10** state information
11*/
12static int pageSize = 1024;
13static int sectorSize = 512;
14static FILE *db = 0;
drh4f788ec2010-10-01 13:28:43 +000015static int fileSize = 0;
16static unsigned cksumNonce = 0;
drh0de8c112002-07-06 16:32:14 +000017
drh4f788ec2010-10-01 13:28:43 +000018/* Report a memory allocation error */
drh0de8c112002-07-06 16:32:14 +000019static void out_of_memory(void){
20 fprintf(stderr,"Out of memory...\n");
21 exit(1);
22}
23
drh4f788ec2010-10-01 13:28:43 +000024/*
25** Read N bytes of memory starting at iOfst into space obtained
26** from malloc().
27*/
drh4bb77ec2014-06-30 11:14:26 +000028static unsigned char *read_content(int N, int iOfst){
drh4f788ec2010-10-01 13:28:43 +000029 int got;
drh4bb77ec2014-06-30 11:14:26 +000030 unsigned char *pBuf = malloc(N);
drh4f788ec2010-10-01 13:28:43 +000031 if( pBuf==0 ) out_of_memory();
32 fseek(db, iOfst, SEEK_SET);
mistachkincdabd7b2015-10-14 20:34:57 +000033 got = (int)fread(pBuf, 1, N, db);
drh4f788ec2010-10-01 13:28:43 +000034 if( got<0 ){
35 fprintf(stderr, "I/O error reading %d bytes from %d\n", N, iOfst);
36 memset(pBuf, 0, N);
37 }else if( got<N ){
38 fprintf(stderr, "Short read: got only %d of %d bytes from %d\n",
39 got, N, iOfst);
40 memset(&pBuf[got], 0, N-got);
drh0de8c112002-07-06 16:32:14 +000041 }
drh4f788ec2010-10-01 13:28:43 +000042 return pBuf;
43}
44
45/* Print a line of decode output showing a 4-byte integer.
46*/
47static unsigned print_decode_line(
drh4bb77ec2014-06-30 11:14:26 +000048 const unsigned char *aData, /* Content being decoded */
49 int ofst, int nByte, /* Start and size of decode */
50 const char *zMsg /* Message to append */
drh4f788ec2010-10-01 13:28:43 +000051){
52 int i, j;
53 unsigned val = aData[ofst];
54 char zBuf[100];
drh4bb77ec2014-06-30 11:14:26 +000055 sprintf(zBuf, " %05x: %02x", ofst, aData[ofst]);
mistachkincdabd7b2015-10-14 20:34:57 +000056 i = (int)strlen(zBuf);
drh4f788ec2010-10-01 13:28:43 +000057 for(j=1; j<4; j++){
58 if( j>=nByte ){
59 sprintf(&zBuf[i], " ");
60 }else{
61 sprintf(&zBuf[i], " %02x", aData[ofst+j]);
62 val = val*256 + aData[ofst+j];
63 }
mistachkincdabd7b2015-10-14 20:34:57 +000064 i += (int)strlen(&zBuf[i]);
drh4f788ec2010-10-01 13:28:43 +000065 }
66 sprintf(&zBuf[i], " %10u", val);
67 printf("%s %s\n", zBuf, zMsg);
68 return val;
69}
70
71/*
72** Read and print a journal header. Store key information (page size, etc)
73** in global variables.
74*/
75static unsigned decode_journal_header(int iOfst){
drh4bb77ec2014-06-30 11:14:26 +000076 unsigned char *pHdr = read_content(64, iOfst);
drh4f788ec2010-10-01 13:28:43 +000077 unsigned nPage;
78 printf("Header at offset %d:\n", iOfst);
79 print_decode_line(pHdr, 0, 4, "Header part 1 (3654616569)");
80 print_decode_line(pHdr, 4, 4, "Header part 2 (547447767)");
81 nPage =
82 print_decode_line(pHdr, 8, 4, "page count");
83 cksumNonce =
84 print_decode_line(pHdr, 12, 4, "chksum nonce");
85 print_decode_line(pHdr, 16, 4, "initial database size in pages");
86 sectorSize =
87 print_decode_line(pHdr, 20, 4, "sector size");
88 pageSize =
89 print_decode_line(pHdr, 24, 4, "page size");
90 print_decode_line(pHdr, 28, 4, "zero");
91 print_decode_line(pHdr, 32, 4, "zero");
92 print_decode_line(pHdr, 36, 4, "zero");
93 print_decode_line(pHdr, 40, 4, "zero");
94 free(pHdr);
95 return nPage;
96}
97
98static void print_page(int iOfst){
99 unsigned char *aData;
100 char zTitle[50];
101 aData = read_content(pageSize+8, iOfst);
102 sprintf(zTitle, "page number for page at offset %d", iOfst);
drh4bb77ec2014-06-30 11:14:26 +0000103 print_decode_line(aData-iOfst, iOfst, 4, zTitle);
drh0de8c112002-07-06 16:32:14 +0000104 free(aData);
105}
106
107int main(int argc, char **argv){
drh4f788ec2010-10-01 13:28:43 +0000108 int nPage, cnt;
109 int iOfst;
drh0de8c112002-07-06 16:32:14 +0000110 if( argc!=2 ){
111 fprintf(stderr,"Usage: %s FILENAME\n", argv[0]);
112 exit(1);
113 }
drh4f788ec2010-10-01 13:28:43 +0000114 db = fopen(argv[1], "rb");
115 if( db==0 ){
drh0de8c112002-07-06 16:32:14 +0000116 fprintf(stderr,"%s: can't open %s\n", argv[0], argv[1]);
117 exit(1);
118 }
drh4f788ec2010-10-01 13:28:43 +0000119 fseek(db, 0, SEEK_END);
120 fileSize = ftell(db);
121 printf("journal file size: %d bytes\n", fileSize);
122 fseek(db, 0, SEEK_SET);
123 iOfst = 0;
124 while( iOfst<fileSize ){
125 cnt = nPage = (int)decode_journal_header(iOfst);
126 if( cnt==0 ){
127 cnt = (fileSize - sectorSize)/(pageSize+8);
drh240c5792004-02-08 00:40:52 +0000128 }
drh4f788ec2010-10-01 13:28:43 +0000129 iOfst += sectorSize;
130 while( cnt && iOfst<fileSize ){
131 print_page(iOfst);
132 iOfst += pageSize+8;
133 }
134 iOfst = (iOfst/sectorSize + 1)*sectorSize;
drh0de8c112002-07-06 16:32:14 +0000135 }
drh4f788ec2010-10-01 13:28:43 +0000136 fclose(db);
drh4bb77ec2014-06-30 11:14:26 +0000137 return 0;
drh0de8c112002-07-06 16:32:14 +0000138}