Fixes default log output to console for macOS
[sqlcipher.git] / tool / getlock.c
blob7eff04d7f97c00efa08ac2b9aac253f2acd7a98a
1 /*
2 ** This utility program looks at an SQLite database and determines whether
3 ** or not it is locked, the kind of lock, and who is holding this lock.
4 **
5 ** This only works on unix when the posix advisory locking method is used
6 ** (which is the default on unix) and when the PENDING_BYTE is in its
7 ** usual place.
8 */
9 #include <sys/types.h>
10 #include <sys/stat.h>
11 #include <unistd.h>
12 #include <fcntl.h>
13 #include <string.h>
14 #include <stdio.h>
15 #include <stdlib.h>
16 #include <errno.h>
18 static void usage(const char *argv0){
19 fprintf(stderr, "Usage: %s database\n", argv0);
20 exit(1);
23 /* Check for a conflicting lock. If one is found, print an this
24 ** on standard output using the format string given and return 1.
25 ** If there are no conflicting locks, return 0.
27 static int isLocked(
28 int h, /* File descriptor to check */
29 int type, /* F_RDLCK or F_WRLCK */
30 unsigned int iOfst, /* First byte of the lock */
31 unsigned int iCnt, /* Number of bytes in the lock range */
32 const char *zType /* Type of lock */
34 struct flock lk;
36 memset(&lk, 0, sizeof(lk));
37 lk.l_type = type;
38 lk.l_whence = SEEK_SET;
39 lk.l_start = iOfst;
40 lk.l_len = iCnt;
41 if( fcntl(h, F_GETLK, &lk)==(-1) ){
42 fprintf(stderr, "fcntl(%d) failed: errno=%d\n", h, errno);
43 exit(1);
45 if( lk.l_type==F_UNLCK ) return 0;
46 printf("%s lock held by %d\n", zType, (int)lk.l_pid);
47 return 1;
51 ** Location of locking bytes in the database file
53 #define PENDING_BYTE (0x40000000)
54 #define RESERVED_BYTE (PENDING_BYTE+1)
55 #define SHARED_FIRST (PENDING_BYTE+2)
56 #define SHARED_SIZE 510
59 ** Lock locations for shared-memory locks used by WAL mode.
61 #define SHM_BASE 120
62 #define SHM_WRITE SHM_BASE
63 #define SHM_CHECKPOINT (SHM_BASE+1)
64 #define SHM_RECOVER (SHM_BASE+2)
65 #define SHM_READ_FIRST (SHM_BASE+3)
66 #define SHM_READ_SIZE 5
69 int main(int argc, char **argv){
70 int hDb; /* File descriptor for the open database file */
71 int hShm; /* File descriptor for WAL shared-memory file */
72 char *zShm; /* Name of the shared-memory file for WAL mode */
73 ssize_t got; /* Bytes read from header */
74 int isWal; /* True if in WAL mode */
75 int nName; /* Length of filename */
76 unsigned char aHdr[100]; /* Database header */
77 int nLock = 0; /* Number of locks held */
78 int i; /* Loop counter */
80 if( argc!=2 ) usage(argv[0]);
81 hDb = open(argv[1], O_RDONLY, 0);
82 if( hDb<0 ){
83 fprintf(stderr, "cannot open %s\n", argv[1]);
84 return 1;
87 /* Make sure we are dealing with an database file */
88 got = read(hDb, aHdr, 100);
89 if( got!=100 || memcmp(aHdr, "SQLite format 3",16)!=0 ){
90 fprintf(stderr, "not an SQLite database: %s\n", argv[1]);
91 exit(1);
94 /* First check for an exclusive lock */
95 if( isLocked(hDb, F_RDLCK, SHARED_FIRST, SHARED_SIZE, "EXCLUSIVE") ){
96 return 0;
98 isWal = aHdr[18]==2;
99 if( isWal==0 ){
100 /* Rollback mode */
101 if( isLocked(hDb, F_RDLCK, PENDING_BYTE, 1, "PENDING") ) return 0;
102 if( isLocked(hDb, F_RDLCK, RESERVED_BYTE, 1, "RESERVED") ) return 0;
103 if( isLocked(hDb, F_WRLCK, SHARED_FIRST, SHARED_SIZE, "SHARED") ){
104 return 0;
106 }else{
107 /* WAL mode */
108 nName = (int)strlen(argv[1]);
109 zShm = malloc( nName + 100 );
110 if( zShm==0 ){
111 fprintf(stderr, "out of memory\n");
112 exit(1);
114 memcpy(zShm, argv[1], nName);
115 memcpy(&zShm[nName], "-shm", 5);
116 hShm = open(zShm, O_RDONLY, 0);
117 if( hShm<0 ){
118 fprintf(stderr, "cannot open %s\n", zShm);
119 return 1;
121 if( isLocked(hShm, F_RDLCK, SHM_RECOVER, 1, "WAL-RECOVERY") ){
122 return 0;
124 nLock += isLocked(hShm, F_RDLCK, SHM_CHECKPOINT, 1, "WAL-CHECKPOINT");
125 nLock += isLocked(hShm, F_RDLCK, SHM_WRITE, 1, "WAL-WRITE");
126 for(i=0; i<SHM_READ_SIZE; i++){
127 nLock += isLocked(hShm, F_WRLCK, SHM_READ_FIRST+i, 1, "WAL-READ");
130 if( nLock==0 ){
131 printf("file is not locked\n");
133 return 0;