123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139 |
- /*
- ** 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;
- }
|