os/persistentdata/persistentstorage/sqlite3api/TEST/TclScript/thread003.test
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
# 2007 September 10
sl@0
     2
#
sl@0
     3
# The author disclaims copyright to this source code.  In place of
sl@0
     4
# a legal notice, here is a blessing:
sl@0
     5
#
sl@0
     6
#    May you do good and not evil.
sl@0
     7
#    May you find forgiveness for yourself and forgive others.
sl@0
     8
#    May you share freely, never taking more than you give.
sl@0
     9
#
sl@0
    10
#***********************************************************************
sl@0
    11
#
sl@0
    12
#   This file contains tests that attempt to break the pcache module
sl@0
    13
#   by bombarding it with simultaneous requests from multiple threads.
sl@0
    14
#     
sl@0
    15
# $Id: thread003.test,v 1.4 2008/08/30 09:10:17 danielk1977 Exp $
sl@0
    16
sl@0
    17
set testdir [file dirname $argv0]
sl@0
    18
sl@0
    19
source $testdir/tester.tcl
sl@0
    20
source $testdir/thread_common.tcl
sl@0
    21
if {[info commands sqlthread] eq ""} {
sl@0
    22
  finish_test
sl@0
    23
  return
sl@0
    24
}
sl@0
    25
sl@0
    26
# Set up a couple of different databases full of pseudo-randomly 
sl@0
    27
# generated data.
sl@0
    28
#
sl@0
    29
do_test thread003.1.1 {
sl@0
    30
  execsql {
sl@0
    31
    BEGIN;
sl@0
    32
    CREATE TABLE t1(a, b, c);
sl@0
    33
  }
sl@0
    34
  for {set ii 0} {$ii < 5000} {incr ii} {
sl@0
    35
    execsql {INSERT INTO t1 VALUES($ii, randomblob(200), randomblob(200))}
sl@0
    36
  }
sl@0
    37
  execsql { 
sl@0
    38
    CREATE INDEX i1 ON t1(a, b); 
sl@0
    39
    COMMIT;
sl@0
    40
  }
sl@0
    41
} {}
sl@0
    42
do_test thread003.1.2 {
sl@0
    43
  expr {([file size test.db] / 1024) > 2000}
sl@0
    44
} {1}
sl@0
    45
do_test thread003.1.3 {
sl@0
    46
  db close
sl@0
    47
  file delete -force test2.db
sl@0
    48
  sqlite3 db test2.db
sl@0
    49
} {}
sl@0
    50
do_test thread003.1.4 {
sl@0
    51
  execsql {
sl@0
    52
    BEGIN;
sl@0
    53
    CREATE TABLE t1(a, b, c);
sl@0
    54
  }
sl@0
    55
  for {set ii 0} {$ii < 5000} {incr ii} {
sl@0
    56
    execsql {INSERT INTO t1 VALUES($ii, randomblob(200), randomblob(200))}
sl@0
    57
  }
sl@0
    58
  execsql { 
sl@0
    59
    CREATE INDEX i1 ON t1(a, b); 
sl@0
    60
    COMMIT;
sl@0
    61
  }
sl@0
    62
} {}
sl@0
    63
do_test thread003.1.5 {
sl@0
    64
  expr {([file size test.db] / 1024) > 2000}
sl@0
    65
} {1}
sl@0
    66
do_test thread003.1.6 {
sl@0
    67
  db close
sl@0
    68
} {}
sl@0
    69
sl@0
    70
sl@0
    71
# This test opens a connection on each of the large (>2MB) database files
sl@0
    72
# created by the previous block. The connections do not share a cache.
sl@0
    73
# Both "cache_size" parameters are set to 15, so there is a maximum of
sl@0
    74
# 30 pages available globally.
sl@0
    75
#
sl@0
    76
# Then, in separate threads, the databases are randomly queried over and
sl@0
    77
# over again. This will force the connections to recycle clean pages from
sl@0
    78
# each other. If there is a thread-safety problem, a segfault or assertion
sl@0
    79
# failure may eventually occur.
sl@0
    80
#
sl@0
    81
set nSecond 30
sl@0
    82
puts "Starting thread003.2 (should run for ~$nSecond seconds)"
sl@0
    83
do_test thread003.2 {
sl@0
    84
  foreach zFile {test.db test2.db} {
sl@0
    85
    set SCRIPT [format {
sl@0
    86
      set iEnd [expr {[clock_seconds] + %d}]
sl@0
    87
      set ::DB [sqlthread open %s]
sl@0
    88
  
sl@0
    89
      # Set the cache size to 15 pages per cache. 30 available globally.
sl@0
    90
      execsql { PRAGMA cache_size = 15 }
sl@0
    91
  
sl@0
    92
      while {[clock_seconds] < $iEnd} {
sl@0
    93
        set iQuery [expr {int(rand()*5000)}]
sl@0
    94
        execsql " SELECT * FROM t1 WHERE a = $iQuery "
sl@0
    95
      }
sl@0
    96
  
sl@0
    97
      sqlite3_close $::DB
sl@0
    98
      expr 1
sl@0
    99
    } $nSecond $zFile]
sl@0
   100
  
sl@0
   101
    unset -nocomplain finished($zFile)
sl@0
   102
    thread_spawn finished($zFile) $thread_procs $SCRIPT
sl@0
   103
  }
sl@0
   104
  foreach zFile {test.db test2.db} {
sl@0
   105
    if {![info exists finished($zFile)]} {
sl@0
   106
      vwait finished($zFile)
sl@0
   107
    }
sl@0
   108
  }
sl@0
   109
  expr 0
sl@0
   110
} {0}
sl@0
   111
sl@0
   112
# This test is the same as the test above, except that each thread also
sl@0
   113
# writes to the database. This causes pages to be moved back and forth 
sl@0
   114
# between the caches internal dirty and clean lists, which is another
sl@0
   115
# opportunity for a thread-related bug to present itself.
sl@0
   116
#
sl@0
   117
set nSecond 30
sl@0
   118
puts "Starting thread003.3 (should run for ~$nSecond seconds)"
sl@0
   119
do_test thread003.3 {
sl@0
   120
  foreach zFile {test.db test2.db} {
sl@0
   121
    set SCRIPT [format {
sl@0
   122
      set iStart [clock_seconds]
sl@0
   123
      set iEnd [expr {[clock_seconds] + %d}]
sl@0
   124
      set ::DB [sqlthread open %s]
sl@0
   125
  
sl@0
   126
      # Set the cache size to 15 pages per cache. 30 available globally.
sl@0
   127
      execsql { PRAGMA cache_size = 15 }
sl@0
   128
  
sl@0
   129
      while {[clock_seconds] < $iEnd} {
sl@0
   130
        set iQuery [expr {int(rand()*5000)}]
sl@0
   131
        execsql "SELECT * FROM t1 WHERE a = $iQuery"
sl@0
   132
        execsql "UPDATE t1 SET b = randomblob(200) 
sl@0
   133
                 WHERE a < $iQuery AND a > $iQuery + 20
sl@0
   134
        "
sl@0
   135
      }
sl@0
   136
  
sl@0
   137
      sqlite3_close $::DB
sl@0
   138
      expr 1
sl@0
   139
    } $nSecond $zFile]
sl@0
   140
  
sl@0
   141
    unset -nocomplain finished($zFile)
sl@0
   142
    thread_spawn finished($zFile) $thread_procs $SCRIPT
sl@0
   143
  }
sl@0
   144
  foreach zFile {test.db test2.db} {
sl@0
   145
    if {![info exists finished($zFile)]} {
sl@0
   146
      vwait finished($zFile)
sl@0
   147
    }
sl@0
   148
  }
sl@0
   149
  expr 0
sl@0
   150
} {0}
sl@0
   151
sl@0
   152
# In this test case, one thread is continually querying the database.
sl@0
   153
# The other thread does not have a database connection, but calls
sl@0
   154
# sqlite3_release_memory() over and over again.
sl@0
   155
#
sl@0
   156
set nSecond 30
sl@0
   157
puts "Starting thread003.3 (should run for ~$nSecond seconds)"
sl@0
   158
unset -nocomplain finished(1)
sl@0
   159
unset -nocomplain finished(2)
sl@0
   160
do_test thread003.4 {
sl@0
   161
  thread_spawn finished(1) $thread_procs [format {
sl@0
   162
    set iEnd [expr {[clock_seconds] + %d}]
sl@0
   163
    set ::DB [sqlthread open test.db]
sl@0
   164
sl@0
   165
    # Set the cache size to 15 pages per cache. 30 available globally.
sl@0
   166
    execsql { PRAGMA cache_size = 15 }
sl@0
   167
sl@0
   168
    while {[clock_seconds] < $iEnd} {
sl@0
   169
      set iQuery [expr {int(rand()*5000)}]
sl@0
   170
      execsql "SELECT * FROM t1 WHERE a = $iQuery"
sl@0
   171
    }
sl@0
   172
sl@0
   173
    sqlite3_close $::DB
sl@0
   174
    expr 1
sl@0
   175
  } $nSecond] 
sl@0
   176
  thread_spawn finished(2) [format {
sl@0
   177
    set iEnd [expr {[clock_seconds] + %d}]
sl@0
   178
sl@0
   179
    while {[clock_seconds] < $iEnd} {
sl@0
   180
      sqlite3_release_memory 1000
sl@0
   181
    }
sl@0
   182
  } $nSecond]
sl@0
   183
  
sl@0
   184
  foreach ii {1 2} {
sl@0
   185
    if {![info exists finished($ii)]} {
sl@0
   186
      vwait finished($ii)
sl@0
   187
    }
sl@0
   188
  }
sl@0
   189
  expr 0
sl@0
   190
} {0}
sl@0
   191
sl@0
   192
finish_test
sl@0
   193
sl@0
   194