| /* |
| ** A utility for printing an SQLite database journal. |
| */ |
| #include <stdio.h> |
| #include <ctype.h> |
| #include <stdlib.h> |
| #include <string.h> |
| |
| /* |
| ** state information |
| */ |
| static int pageSize = 1024; |
| static int sectorSize = 512; |
| static FILE *db = 0; |
| static int fileSize = 0; |
| static unsigned cksumNonce = 0; |
| |
| /* Report a memory allocation error */ |
| static void out_of_memory(void){ |
| fprintf(stderr,"Out of memory...\n"); |
| exit(1); |
| } |
| |
| /* |
| ** Read N bytes of memory starting at iOfst into space obtained |
| ** from malloc(). |
| */ |
| static unsigned char *read_content(int N, int iOfst){ |
| int got; |
| unsigned char *pBuf = malloc(N); |
| if( pBuf==0 ) out_of_memory(); |
| fseek(db, iOfst, SEEK_SET); |
| got = (int)fread(pBuf, 1, N, db); |
| if( got<0 ){ |
| fprintf(stderr, "I/O error reading %d bytes from %d\n", N, iOfst); |
| memset(pBuf, 0, N); |
| }else if( got<N ){ |
| fprintf(stderr, "Short read: got only %d of %d bytes from %d\n", |
| got, N, iOfst); |
| memset(&pBuf[got], 0, N-got); |
| } |
| return pBuf; |
| } |
| |
| /* Print a line of decode output showing a 4-byte integer. |
| */ |
| static unsigned print_decode_line( |
| const unsigned char *aData, /* Content being decoded */ |
| int ofst, int nByte, /* Start and size of decode */ |
| const char *zMsg /* Message to append */ |
| ){ |
| int i, j; |
| unsigned val = aData[ofst]; |
| char zBuf[100]; |
| sprintf(zBuf, " %05x: %02x", ofst, aData[ofst]); |
| i = (int)strlen(zBuf); |
| for(j=1; j<4; j++){ |
| if( j>=nByte ){ |
| sprintf(&zBuf[i], " "); |
| }else{ |
| sprintf(&zBuf[i], " %02x", aData[ofst+j]); |
| val = val*256 + aData[ofst+j]; |
| } |
| i += (int)strlen(&zBuf[i]); |
| } |
| sprintf(&zBuf[i], " %10u", val); |
| printf("%s %s\n", zBuf, zMsg); |
| return val; |
| } |
| |
| /* |
| ** Read and print a journal header. Store key information (page size, etc) |
| ** in global variables. |
| */ |
| static unsigned decode_journal_header(int iOfst){ |
| unsigned char *pHdr = read_content(64, iOfst); |
| unsigned nPage; |
| printf("Header at offset %d:\n", iOfst); |
| print_decode_line(pHdr, 0, 4, "Header part 1 (3654616569)"); |
| print_decode_line(pHdr, 4, 4, "Header part 2 (547447767)"); |
| nPage = |
| print_decode_line(pHdr, 8, 4, "page count"); |
| cksumNonce = |
| print_decode_line(pHdr, 12, 4, "chksum nonce"); |
| print_decode_line(pHdr, 16, 4, "initial database size in pages"); |
| sectorSize = |
| print_decode_line(pHdr, 20, 4, "sector size"); |
| pageSize = |
| print_decode_line(pHdr, 24, 4, "page size"); |
| print_decode_line(pHdr, 28, 4, "zero"); |
| print_decode_line(pHdr, 32, 4, "zero"); |
| print_decode_line(pHdr, 36, 4, "zero"); |
| print_decode_line(pHdr, 40, 4, "zero"); |
| free(pHdr); |
| return nPage; |
| } |
| |
| static void print_page(int iOfst){ |
| unsigned char *aData; |
| char zTitle[50]; |
| aData = read_content(pageSize+8, iOfst); |
| sprintf(zTitle, "page number for page at offset %d", iOfst); |
| print_decode_line(aData-iOfst, iOfst, 4, zTitle); |
| free(aData); |
| } |
| |
| int main(int argc, char **argv){ |
| int nPage, cnt; |
| int iOfst; |
| if( argc!=2 ){ |
| fprintf(stderr,"Usage: %s FILENAME\n", argv[0]); |
| exit(1); |
| } |
| db = fopen(argv[1], "rb"); |
| if( db==0 ){ |
| fprintf(stderr,"%s: can't open %s\n", argv[0], argv[1]); |
| exit(1); |
| } |
| fseek(db, 0, SEEK_END); |
| fileSize = ftell(db); |
| printf("journal file size: %d bytes\n", fileSize); |
| fseek(db, 0, SEEK_SET); |
| iOfst = 0; |
| while( iOfst<fileSize ){ |
| cnt = nPage = (int)decode_journal_header(iOfst); |
| if( cnt==0 ){ |
| cnt = (fileSize - sectorSize)/(pageSize+8); |
| } |
| iOfst += sectorSize; |
| while( cnt && iOfst<fileSize ){ |
| print_page(iOfst); |
| iOfst += pageSize+8; |
| } |
| iOfst = (iOfst/sectorSize + 1)*sectorSize; |
| } |
| fclose(db); |
| return 0; |
| } |