os/persistentdata/persistentstorage/sqlite3api/TEST/TclScript/threadtest2.c
author sl@SLION-WIN7.fritz.box
Fri, 15 Jun 2012 03:10:57 +0200
changeset 0 bde4ae8d615e
permissions -rw-r--r--
First public contribution.
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
}