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