139 lines
		
	
	
		
			3.6 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			139 lines
		
	
	
		
			3.6 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| /*
 | |
| ** 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;
 | |
| }
 | 
