| sl@0 |      1 | /*
 | 
| sl@0 |      2 | ** 2004 January 13
 | 
| sl@0 |      3 | **
 | 
| sl@0 |      4 | ** The author disclaims copyright to this source code.  In place of
 | 
| sl@0 |      5 | ** a legal notice, here is a blessing:
 | 
| sl@0 |      6 | **
 | 
| sl@0 |      7 | **    May you do good and not evil.
 | 
| sl@0 |      8 | **    May you find forgiveness for yourself and forgive others.
 | 
| sl@0 |      9 | **    May you share freely, never taking more than you give.
 | 
| sl@0 |     10 | **
 | 
| sl@0 |     11 | *************************************************************************
 | 
| sl@0 |     12 | ** This file implements a simple standalone program used to test whether
 | 
| sl@0 |     13 | ** or not the SQLite library is threadsafe.
 | 
| sl@0 |     14 | **
 | 
| sl@0 |     15 | ** This file is NOT part of the standard SQLite library.  It is used for
 | 
| sl@0 |     16 | ** testing only.
 | 
| sl@0 |     17 | */
 | 
| sl@0 |     18 | #include <stdio.h>
 | 
| sl@0 |     19 | #include <unistd.h>
 | 
| sl@0 |     20 | #include <pthread.h>
 | 
| sl@0 |     21 | #include <string.h>
 | 
| sl@0 |     22 | #include <stdlib.h>
 | 
| sl@0 |     23 | #include "sqlite.h"
 | 
| sl@0 |     24 | 
 | 
| sl@0 |     25 | /*
 | 
| sl@0 |     26 | ** Name of the database
 | 
| sl@0 |     27 | */
 | 
| sl@0 |     28 | #define DB_FILE "test.db"
 | 
| sl@0 |     29 | 
 | 
| sl@0 |     30 | /* 
 | 
| sl@0 |     31 | ** When this variable becomes non-zero, all threads stop
 | 
| sl@0 |     32 | ** what they are doing.
 | 
| sl@0 |     33 | */
 | 
| sl@0 |     34 | volatile int all_stop = 0;
 | 
| sl@0 |     35 | 
 | 
| sl@0 |     36 | /* 
 | 
| sl@0 |     37 | ** Callback from the integrity check.  If the result is anything other
 | 
| sl@0 |     38 | ** than "ok" it means the integrity check has failed.  Set the "all_stop"
 | 
| sl@0 |     39 | ** global variable to stop all other activity.  Print the error message
 | 
| sl@0 |     40 | ** or print OK if the string "ok" is seen.
 | 
| sl@0 |     41 | */
 | 
| sl@0 |     42 | int check_callback(void *pid, int argc, char **argv, char **notUsed2){
 | 
| sl@0 |     43 |   int id = (int)pid;
 | 
| sl@0 |     44 |   if( strcmp(argv[0],"ok") ){
 | 
| sl@0 |     45 |     all_stop = 1;
 | 
| sl@0 |     46 |     fprintf(stderr,"id: %s\n", id, argv[0]);
 | 
| sl@0 |     47 |   }else{
 | 
| sl@0 |     48 |     /* fprintf(stderr,"%d: OK\n", id); */
 | 
| sl@0 |     49 |   }
 | 
| sl@0 |     50 |   return 0;
 | 
| sl@0 |     51 | }
 | 
| sl@0 |     52 | 
 | 
| sl@0 |     53 | /*
 | 
| sl@0 |     54 | ** Do an integrity check on the database.  If the first integrity check
 | 
| sl@0 |     55 | ** fails, try it a second time.
 | 
| sl@0 |     56 | */
 | 
| sl@0 |     57 | int integrity_check(sqlite *db, int id){
 | 
| sl@0 |     58 |   int rc;
 | 
| sl@0 |     59 |   if( all_stop ) return 0;
 | 
| sl@0 |     60 |   /* fprintf(stderr,"%d: CHECK\n", id); */
 | 
| sl@0 |     61 |   rc = sqlite3_exec(db, "pragma integrity_check", check_callback, 0, 0);
 | 
| sl@0 |     62 |   if( rc!=SQLITE_OK && rc!=SQLITE_BUSY ){
 | 
| sl@0 |     63 |     fprintf(stderr,"%d, Integrity check returns %d\n", id, rc);
 | 
| sl@0 |     64 |   }
 | 
| sl@0 |     65 |   if( all_stop ){
 | 
| sl@0 |     66 |     sqlite3_exec(db, "pragma integrity_check", check_callback, 0, 0);
 | 
| sl@0 |     67 |   }
 | 
| sl@0 |     68 |   return 0;
 | 
| sl@0 |     69 | }
 | 
| sl@0 |     70 | 
 | 
| sl@0 |     71 | /*
 | 
| sl@0 |     72 | ** This is the worker thread
 | 
| sl@0 |     73 | */
 | 
| sl@0 |     74 | void *worker(void *workerArg){
 | 
| sl@0 |     75 |   sqlite *db;
 | 
| sl@0 |     76 |   int id = (int)workerArg;
 | 
| sl@0 |     77 |   int rc;
 | 
| sl@0 |     78 |   int cnt = 0;
 | 
| sl@0 |     79 |   fprintf(stderr, "Starting worker %d\n", id);
 | 
| sl@0 |     80 |   while( !all_stop && cnt++<10000 ){
 | 
| sl@0 |     81 |     if( cnt%100==0 ) printf("%d: %d\n", id, cnt);
 | 
| sl@0 |     82 |     while( (sqlite3_open(DB_FILE, &db))!=SQLITE_OK ) sched_yield();
 | 
| sl@0 |     83 |     sqlite3_exec(db, "PRAGMA synchronous=OFF", 0, 0, 0);
 | 
| sl@0 |     84 |     /* integrity_check(db, id); */
 | 
| sl@0 |     85 |     if( all_stop ){ sqlite3_close(db); break; }
 | 
| sl@0 |     86 |     /* fprintf(stderr, "%d: BEGIN\n", id); */
 | 
| sl@0 |     87 |     rc = sqlite3_exec(db, "INSERT INTO t1 VALUES('bogus data')", 0, 0, 0);
 | 
| sl@0 |     88 |     /* fprintf(stderr, "%d: END rc=%d\n", id, rc); */
 | 
| sl@0 |     89 |     sqlite3_close(db);
 | 
| sl@0 |     90 |   }
 | 
| sl@0 |     91 |   fprintf(stderr, "Worker %d finished\n", id);
 | 
| sl@0 |     92 |   return 0;
 | 
| sl@0 |     93 | }
 | 
| sl@0 |     94 | 
 | 
| sl@0 |     95 | /*
 | 
| sl@0 |     96 | ** Initialize the database and start the threads
 | 
| sl@0 |     97 | */
 | 
| sl@0 |     98 | int main(int argc, char **argv){
 | 
| sl@0 |     99 |   sqlite *db;
 | 
| sl@0 |    100 |   int i, rc;
 | 
| sl@0 |    101 |   pthread_t aThread[5];
 | 
| sl@0 |    102 | 
 | 
| sl@0 |    103 |   if( strcmp(DB_FILE,":memory:") ){
 | 
| sl@0 |    104 |     char *zJournal = sqlite3_mprintf("%s-journal", DB_FILE);
 | 
| sl@0 |    105 |     unlink(DB_FILE);
 | 
| sl@0 |    106 |     unlink(zJournal);
 | 
| sl@0 |    107 |     sqlite3_free(zJournal);
 | 
| sl@0 |    108 |   }  
 | 
| sl@0 |    109 |   sqlite3_open(DB_FILE, &db);
 | 
| sl@0 |    110 |   if( db==0 ){
 | 
| sl@0 |    111 |     fprintf(stderr,"unable to initialize database\n");
 | 
| sl@0 |    112 |     exit(1);
 | 
| sl@0 |    113 |   }
 | 
| sl@0 |    114 |   rc = sqlite3_exec(db, "CREATE TABLE t1(x);", 0,0,0);
 | 
| sl@0 |    115 |   if( rc ){
 | 
| sl@0 |    116 |     fprintf(stderr,"cannot create table t1: %d\n", rc);
 | 
| sl@0 |    117 |     exit(1);
 | 
| sl@0 |    118 |   }
 | 
| sl@0 |    119 |   sqlite3_close(db);
 | 
| sl@0 |    120 |   for(i=0; i<sizeof(aThread)/sizeof(aThread[0]); i++){
 | 
| sl@0 |    121 |     pthread_create(&aThread[i], 0, worker, (void*)i);
 | 
| sl@0 |    122 |   }
 | 
| sl@0 |    123 |   for(i=0; i<sizeof(aThread)/sizeof(aThread[i]); i++){
 | 
| sl@0 |    124 |     pthread_join(aThread[i], 0);
 | 
| sl@0 |    125 |   }
 | 
| sl@0 |    126 |   if( !all_stop ){
 | 
| sl@0 |    127 |     printf("Everything seems ok.\n");
 | 
| sl@0 |    128 |     return 0;
 | 
| sl@0 |    129 |   }else{
 | 
| sl@0 |    130 |     printf("We hit an error.\n");
 | 
| sl@0 |    131 |     return 1;
 | 
| sl@0 |    132 |   }
 | 
| sl@0 |    133 | }
 |