summaryrefslogtreecommitdiffstats
path: root/test/thread003.test
diff options
context:
space:
mode:
Diffstat (limited to '')
-rw-r--r--test/thread003.test189
1 files changed, 189 insertions, 0 deletions
diff --git a/test/thread003.test b/test/thread003.test
new file mode 100644
index 0000000..8c7440c
--- /dev/null
+++ b/test/thread003.test
@@ -0,0 +1,189 @@
+# 2007 September 10
+#
+# The author disclaims copyright to this source code. In place of
+# a legal notice, here is a blessing:
+#
+# May you do good and not evil.
+# May you find forgiveness for yourself and forgive others.
+# May you share freely, never taking more than you give.
+#
+#***********************************************************************
+#
+# This file contains tests that attempt to break the pcache module
+# by bombarding it with simultaneous requests from multiple threads.
+#
+# $Id: thread003.test,v 1.8 2009/03/26 14:48:07 danielk1977 Exp $
+
+set testdir [file dirname $argv0]
+
+source $testdir/tester.tcl
+if {[run_thread_tests]==0} { finish_test ; return }
+
+# Set up a couple of different databases full of pseudo-randomly
+# generated data.
+#
+do_test thread003.1.1 {
+ execsql {
+ BEGIN;
+ CREATE TABLE t1(a, b, c);
+ }
+ for {set ii 0} {$ii < 5000} {incr ii} {
+ execsql {INSERT INTO t1 VALUES($ii, randomblob(200), randomblob(200))}
+ }
+ execsql {
+ CREATE INDEX i1 ON t1(a, b);
+ COMMIT;
+ }
+} {}
+do_test thread003.1.2 {
+ expr {([file size test.db] / 1024) > 2000}
+} {1}
+do_test thread003.1.3 {
+ db close
+ forcedelete test2.db
+ sqlite3 db test2.db
+} {}
+do_test thread003.1.4 {
+ execsql {
+ BEGIN;
+ CREATE TABLE t1(a, b, c);
+ }
+ for {set ii 0} {$ii < 5000} {incr ii} {
+ execsql {INSERT INTO t1 VALUES($ii, randomblob(200), randomblob(200))}
+ }
+ execsql {
+ CREATE INDEX i1 ON t1(a, b);
+ COMMIT;
+ }
+} {}
+do_test thread003.1.5 {
+ expr {([file size test.db] / 1024) > 2000}
+} {1}
+do_test thread003.1.6 {
+ db close
+} {}
+
+
+# This test opens a connection on each of the large (>2MB) database files
+# created by the previous block. The connections do not share a cache.
+# Both "cache_size" parameters are set to 15, so there is a maximum of
+# 30 pages available globally.
+#
+# Then, in separate threads, the databases are randomly queried over and
+# over again. This will force the connections to recycle clean pages from
+# each other. If there is a thread-safety problem, a segfault or assertion
+# failure may eventually occur.
+#
+set nSecond 30
+puts "Starting thread003.2 (should run for ~$nSecond seconds)"
+do_test thread003.2 {
+ foreach zFile {test.db test2.db} {
+ set SCRIPT [format {
+ set iEnd [expr {[clock_seconds] + %d}]
+ set ::DB [sqlthread open %s xyzzy]
+
+ # Set the cache size to 15 pages per cache. 30 available globally.
+ execsql { PRAGMA cache_size = 15 }
+
+ while {[clock_seconds] < $iEnd} {
+ set iQuery [expr {int(rand()*5000)}]
+ execsql " SELECT * FROM t1 WHERE a = $iQuery "
+ }
+
+ sqlite3_close $::DB
+ expr 1
+ } $nSecond $zFile]
+
+ unset -nocomplain finished($zFile)
+ thread_spawn finished($zFile) $thread_procs $SCRIPT
+ }
+ foreach zFile {test.db test2.db} {
+ if {![info exists finished($zFile)]} {
+ vwait finished($zFile)
+ }
+ }
+ expr 0
+} {0}
+
+# This test is the same as the test above, except that each thread also
+# writes to the database. This causes pages to be moved back and forth
+# between the caches internal dirty and clean lists, which is another
+# opportunity for a thread-related bug to present itself.
+#
+set nSecond 30
+puts "Starting thread003.3 (should run for ~$nSecond seconds)"
+do_test thread003.3 {
+ foreach zFile {test.db test2.db} {
+ set SCRIPT [format {
+ set iStart [clock_seconds]
+ set iEnd [expr {[clock_seconds] + %d}]
+ set ::DB [sqlthread open %s xyzzy]
+
+ # Set the cache size to 15 pages per cache. 30 available globally.
+ execsql { PRAGMA cache_size = 15 }
+
+ while {[clock_seconds] < $iEnd} {
+ set iQuery [expr {int(rand()*5000)}]
+ execsql "SELECT * FROM t1 WHERE a = $iQuery"
+ execsql "UPDATE t1 SET b = randomblob(200)
+ WHERE a < $iQuery AND a > $iQuery + 20
+ "
+ }
+
+ sqlite3_close $::DB
+ expr 1
+ } $nSecond $zFile]
+
+ unset -nocomplain finished($zFile)
+ thread_spawn finished($zFile) $thread_procs $SCRIPT
+ }
+ foreach zFile {test.db test2.db} {
+ if {![info exists finished($zFile)]} {
+ vwait finished($zFile)
+ }
+ }
+ expr 0
+} {0}
+
+# In this test case, one thread is continually querying the database.
+# The other thread does not have a database connection, but calls
+# sqlite3_release_memory() over and over again.
+#
+set nSecond 30
+puts "Starting thread003.4 (should run for ~$nSecond seconds)"
+unset -nocomplain finished(1)
+unset -nocomplain finished(2)
+do_test thread003.4 {
+ thread_spawn finished(1) $thread_procs [format {
+ set iEnd [expr {[clock_seconds] + %d}]
+ set ::DB [sqlthread open test.db xyzzy]
+
+ # Set the cache size to 15 pages per cache. 30 available globally.
+ execsql { PRAGMA cache_size = 15 }
+
+ while {[clock_seconds] < $iEnd} {
+ set iQuery [expr {int(rand()*5000)}]
+ execsql "SELECT * FROM t1 WHERE a = $iQuery"
+ }
+
+ sqlite3_close $::DB
+ expr 1
+ } $nSecond]
+ thread_spawn finished(2) [format {
+ set iEnd [expr {[clock_seconds] + %d}]
+
+ while {[clock_seconds] < $iEnd} {
+ sqlite3_release_memory 1000
+ }
+ } $nSecond]
+
+ foreach ii {1 2} {
+ if {![info exists finished($ii)]} {
+ vwait finished($ii)
+ }
+ }
+ expr 0
+} {0}
+
+set sqlite_open_file_count 0
+finish_test