os/persistentdata/persistentstorage/sqlite3api/TEST/TclScript/tester.tcl
author sl@SLION-WIN7.fritz.box
Fri, 15 Jun 2012 03:10:57 +0200
changeset 0 bde4ae8d615e
permissions -rw-r--r--
First public contribution.
     1 # 2001 September 15
     2 #
     3 # Portions Copyright (c) 2007-2010 Nokia Corporation and/or its subsidiaries. All rights reserved.
     4 #
     5 # The author disclaims copyright to this source code.  In place of
     6 # a legal notice, here is a blessing:
     7 #
     8 #    May you do good and not evil.
     9 #    May you find forgiveness for yourself and forgive others.
    10 #    May you share freely, never taking more than you give.
    11 #
    12 #***********************************************************************
    13 # This file implements some common TCL routines used for regression
    14 # testing the SQLite library
    15 #
    16 # $Id: tester.tcl,v 1.134 2008/08/05 17:53:24 drh Exp $
    17 
    18 #
    19 # What for user input before continuing.  This gives an opportunity
    20 # to connect profiling tools to the process.
    21 #
    22 for {set i 0} {$i<[llength $argv]} {incr i} {
    23   if {[regexp {^-+pause$} [lindex $argv $i] all value]} {
    24     puts -nonewline "Press RETURN to begin..."
    25     flush stdout
    26     gets stdin
    27     set argv [lreplace $argv $i $i]
    28   }
    29 }
    30 
    31 set tcl_precision 15
    32 set sqlite_pending_byte 0x0010000
    33 
    34 # 
    35 # Check the command-line arguments for a default soft-heap-limit.
    36 # Store this default value in the global variable ::soft_limit and
    37 # update the soft-heap-limit each time this script is run.  In that
    38 # way if an individual test file changes the soft-heap-limit, it
    39 # will be reset at the start of the next test file.
    40 #
    41 if {![info exists soft_limit]} {
    42   set soft_limit 0
    43   for {set i 0} {$i<[llength $argv]} {incr i} {
    44     if {[regexp {^--soft-heap-limit=(.+)$} [lindex $argv $i] all value]} {
    45       if {$value!="off"} {
    46         set soft_limit $value
    47       }
    48       set argv [lreplace $argv $i $i]
    49     }
    50   }
    51 }
    52 sqlite3_soft_heap_limit $soft_limit
    53 
    54 # 
    55 # Check the command-line arguments to set the memory debugger
    56 # backtrace depth.
    57 #
    58 # See the sqlite3_memdebug_backtrace() function in mem2.c or
    59 # test_malloc.c for additional information.
    60 #
    61 for {set i 0} {$i<[llength $argv]} {incr i} {
    62   if {[lindex $argv $i] eq "--malloctrace"} {
    63     set argv [lreplace $argv $i $i]
    64     sqlite3_memdebug_backtrace 10
    65     sqlite3_memdebug_log start
    66     set tester_do_malloctrace 1
    67   }
    68 }
    69 for {set i 0} {$i<[llength $argv]} {incr i} {
    70   if {[regexp {^--backtrace=(\d+)$} [lindex $argv $i] all value]} {
    71     sqlite3_memdebug_backtrace $value
    72     set argv [lreplace $argv $i $i]
    73   }
    74 }
    75 
    76 
    77 proc ostrace_call {zCall nClick zFile i32 i64} {
    78   set s "INSERT INTO ostrace VALUES('$zCall', $nClick, '$zFile', $i32, $i64);"
    79   puts $::ostrace_fd $s
    80 }
    81 
    82 for {set i 0} {$i<[llength $argv]} {incr i} {
    83   if {[lindex $argv $i] eq "--ossummary" || [lindex $argv $i] eq "--ostrace"} {
    84     sqlite3_instvfs create -default ostrace
    85     set tester_do_ostrace 1
    86     set ostrace_fd [open ostrace.sql w]
    87     puts $ostrace_fd "BEGIN;"
    88     if {[lindex $argv $i] eq "--ostrace"} {
    89       set    s "CREATE TABLE ostrace"
    90       append s "(method TEXT, clicks INT, file TEXT, i32 INT, i64 INT);"
    91       puts $ostrace_fd $s
    92       sqlite3_instvfs configure ostrace ostrace_call
    93       sqlite3_instvfs configure ostrace ostrace_call
    94     }
    95     set argv [lreplace $argv $i $i]
    96   }
    97   if {[lindex $argv $i] eq "--binarylog"} {
    98     set tester_do_binarylog 1
    99     set argv [lreplace $argv $i $i]
   100   }
   101 }
   102 
   103 # 
   104 # Check the command-line arguments to set the maximum number of
   105 # errors tolerated before halting.
   106 #
   107 if {![info exists maxErr]} {
   108   set maxErr 1000
   109 }
   110 for {set i 0} {$i<[llength $argv]} {incr i} {
   111   if {[regexp {^--maxerror=(\d+)$} [lindex $argv $i] all maxErr]} {
   112     set argv [lreplace $argv $i $i]
   113   }
   114 }
   115 #puts "Max error = $maxErr"
   116 
   117 
   118 # Use the pager codec if it is available
   119 #
   120 if {[sqlite3 -has-codec] && [info command sqlite_orig]==""} {
   121   rename sqlite3 sqlite_orig
   122   proc sqlite3 {args} {
   123     if {[llength $args]==2 && [string index [lindex $args 0] 0]!="-"} {
   124       lappend args -key {xyzzy}
   125     }
   126     uplevel 1 sqlite_orig $args
   127   }
   128 }
   129 
   130 
   131 # Create a test database
   132 #
   133 if {![info exists nTest]} {
   134   sqlite3_shutdown 
   135   install_malloc_faultsim 1 
   136   sqlite3_initialize
   137   if {[info exists tester_do_binarylog]} {
   138     sqlite3_instvfs binarylog -default binarylog ostrace.bin
   139     sqlite3_instvfs marker binarylog "$argv0 $argv"
   140   }
   141 }
   142 catch {db close}
   143 file delete -force test.db
   144 file delete -force test.db-journal
   145 sqlite3 db ./test.db
   146 set ::DB [sqlite3_connection_pointer db]
   147 if {[info exists ::SETUP_SQL]} {
   148   db eval $::SETUP_SQL
   149 }
   150 
   151 # Abort early if this script has been run before.
   152 #
   153 if {[info exists nTest]} return
   154 
   155 # Symbian OS globals
   156 set case_failure 0
   157 set nCases 0
   158 set nFailedCases 0
   159 
   160 # Set the test counters to zero
   161 #
   162 set nErr 0
   163 set nTest 0
   164 set skip_test 0
   165 set failList {}
   166 set omitList {}
   167 if {![info exists speedTest]} {
   168   set speedTest 0
   169 }
   170 
   171 # Record the fact that a sequence of tests were omitted.
   172 #
   173 proc omit_test {name reason} {
   174   global omitList
   175   lappend omitList [list $name $reason]
   176 }
   177 
   178 # Symbian OS: Added procedures to output test result in TEF format
   179 puts "<pre>"
   180 
   181 # Symbian OS: global procedure to handle test errors counter
   182 proc do_fail {name} {
   183   global case_failure nErr failList
   184   set case_failure 1
   185   incr nErr
   186   lappend ::failList $name
   187 }
   188 
   189 # Symbian OS: global procedure to output START_TESTCASE in TEF format
   190 proc start_case {name} {
   191   global case_failure nCases
   192   set case_failure 0
   193   incr nCases
   194   puts "<font color=00AF00> START_TESTCASE $name <\/font>"
   195 }
   196 
   197 # Symbian OS: global procedure to output END_TESTCASE in TEF format
   198 proc end_case {name} {
   199   global case_failure nFailedCases
   200   if {$case_failure} {
   201     incr nFailedCases
   202     puts "<font color=FF0000> END_TESTCASE $name ***TestCaseResult = FAIL <\/font>"
   203   } else {
   204     puts "<font color=00AF00> END_TESTCASE $name ***TestCaseResult = PASS <\/font>"
   205   }
   206 }
   207 
   208 # Symbian OS: global procedure to output test summary in TEF format
   209 proc tef_summary {} {
   210   global nCases nFailedCases
   211   set nPass [expr "$nCases - $nFailedCases"]
   212   puts "<font color=00AFFF>TEST STEP SUMMARY:<\/font>"
   213   puts "<font color=00AF00>PASS = $nCases<\/font>"
   214   puts "<font color=FF0000>FAIL = $nFailedCases<\/font>"
   215   puts "<font color=00AFFF>TEST CASE SUMMARY:<\/font>"
   216   puts "<font color=00AF00>PASS = $nCases<\/font>"
   217   puts "<font color=FF0000>FAIL = $nFailedCases<\/font>"
   218   puts "<\/pre>"
   219   flush stdout
   220 }
   221 
   222 # Invoke the do_test procedure to run a single test 
   223 #
   224 proc do_test {name cmd expected} {
   225   global argv nErr nTest skip_test maxErr
   226   sqlite3_memdebug_settitle $name
   227   if {[info exists ::tester_do_binarylog]} {
   228     sqlite3_instvfs marker binarylog "Start of $name"
   229   }
   230   if {$skip_test} {
   231     set skip_test 0
   232     return
   233   }
   234   if {[llength $argv]==0} { 
   235     set go 1
   236   } else {
   237     set go 0
   238     foreach pattern $argv {
   239       if {[string match $pattern $name]} {
   240         set go 1
   241         break
   242       }
   243     }
   244   }
   245   if {!$go} return
   246   incr nTest
   247   puts -nonewline $name...
   248   flush stdout
   249   if {[catch {uplevel #0 "$cmd;\n"} result]} {
   250     puts "\nError: $result"
   251     # Symbian OS: Set and increase error count with do_fail procedure (definition in Tester.tcl)
   252     do_fail $name 
   253     print_text $name "FAILED"  
   254     if {$nErr>$maxErr} {puts "*** Giving up..."; finalize_testing}
   255   } elseif {[string compare $result $expected]} {
   256     puts "\nExpected: \[$expected\]\n     Got: \[$result\]"
   257     # Symbian OS: Set and increase error count with do_fail procedure (definition in Tester.tcl)
   258     do_fail $name 
   259     print_text $name "FAILED"  
   260     if {$nErr>=$maxErr} {puts "*** Giving up..."; finalize_testing}
   261   } else {
   262     puts " Ok"
   263     print_text $name "OK"  
   264   }
   265   flush stdout
   266   if {[info exists ::tester_do_binarylog]} {
   267     sqlite3_instvfs marker binarylog "End of $name"
   268   }
   269 }
   270 
   271 # Run an SQL script.  
   272 # Return the number of microseconds per statement.
   273 #
   274 proc speed_trial {name numstmt units sql} {
   275   puts -nonewline [format {%-21.21s } $name...]
   276   flush stdout
   277   set speed [time {sqlite3_exec_nr db $sql}]
   278   set tm [lindex $speed 0]
   279   if {$tm == 0} {
   280     set rate [format %20s "many"]
   281   } else {
   282     set rate [format %20.5f [expr {1000000.0*$numstmt/$tm}]]
   283   }
   284   set u2 $units/s
   285   puts [format {%12d uS %s %s} $tm $rate $u2]
   286   global total_time
   287   set total_time [expr {$total_time+$tm}]
   288 }
   289 proc speed_trial_tcl {name numstmt units script} {
   290   puts -nonewline [format {%-21.21s } $name...]
   291   flush stdout
   292   set speed [time {eval $script}]
   293   set tm [lindex $speed 0]
   294   if {$tm == 0} {
   295     set rate [format %20s "many"]
   296   } else {
   297     set rate [format %20.5f [expr {1000000.0*$numstmt/$tm}]]
   298   }
   299   set u2 $units/s
   300   puts [format {%12d uS %s %s} $tm $rate $u2]
   301   global total_time
   302   set total_time [expr {$total_time+$tm}]
   303 }
   304 proc speed_trial_init {name} {
   305   global total_time
   306   set total_time 0
   307 }
   308 proc speed_trial_summary {name} {
   309   global total_time
   310   puts [format {%-21.21s %12d uS TOTAL} $name $total_time]
   311 }
   312 
   313 # Run this routine last
   314 #
   315 proc finish_test {} {
   316   finalize_testing
   317 }
   318 proc finalize_testing {} {
   319   global nTest nErr sqlite_open_file_count omitList
   320 
   321   catch {db close}
   322   catch {db2 close}
   323   catch {db3 close}
   324 
   325   vfs_unlink_test
   326   sqlite3 db {}
   327   # sqlite3_clear_tsd_memdebug
   328   db close
   329   sqlite3_reset_auto_extension
   330 
   331   set heaplimit [sqlite3_soft_heap_limit]
   332   if {$heaplimit!=$::soft_limit} {
   333     puts "soft-heap-limit changed by this script\
   334           from $::soft_limit to $heaplimit"
   335   } elseif {$heaplimit!="" && $heaplimit>0} {
   336     puts "soft-heap-limit set to $heaplimit"
   337   }
   338   
   339   sqlite3_soft_heap_limit 0
   340   incr nTest
   341   puts "$nErr errors out of $nTest tests"
   342   if {$nErr>0} {
   343     puts "Failures on these tests: $::failList"
   344   }
   345   
   346   if {[llength $omitList]>0} {
   347     puts "Omitted test cases:"
   348     set prec {}
   349     foreach {rec} [lsort $omitList] {
   350       if {$rec==$prec} continue
   351       set prec $rec
   352       puts [format {  %-12s %s} [lindex $rec 0] [lindex $rec 1]]
   353     }
   354   }
   355   
   356   if {$nErr>0 && ![working_64bit_int]} {
   357     puts "******************************************************************"
   358     puts "N.B.:  The version of TCL that you used to build this test harness"
   359     puts "is defective in that it does not support 64-bit integers.  Some or"
   360     puts "all of the test failures above might be a result from this defect"
   361     puts "in your TCL build."
   362     puts "******************************************************************"
   363   }
   364   if {[info exists ::tester_do_binarylog]} {
   365     sqlite3_instvfs destroy binarylog
   366   }
   367 
   368   if {$sqlite_open_file_count} {
   369     puts "$sqlite_open_file_count files were left open"
   370     incr nErr
   371   }
   372   
   373   if {[info exists ::tester_do_ostrace]} {
   374     puts "Writing ostrace.sql..."
   375     set fd $::ostrace_fd
   376 
   377     puts -nonewline $fd "CREATE TABLE ossummary"
   378     puts $fd "(method TEXT, clicks INTEGER, count INTEGER);"
   379     foreach row [sqlite3_instvfs report ostrace] {
   380       foreach {method count clicks} $row break
   381       puts $fd "INSERT INTO ossummary VALUES('$method', $clicks, $count);"
   382     }
   383     puts $fd "COMMIT;"
   384     close $fd
   385     sqlite3_instvfs destroy ostrace
   386   }
   387   
   388   if {[sqlite3_memory_used]>0} {
   389     puts "Unfreed memory: [sqlite3_memory_used] bytes"
   390     incr nErr
   391     ifcapable memdebug||mem5||(mem3&&debug) {
   392       puts "Writing unfreed memory log to \"./memleak.txt\""
   393       sqlite3_memdebug_dump ./memleak.txt
   394     }
   395   } else {
   396     puts "All memory allocations freed - no leaks"
   397     ifcapable memdebug||mem5 {
   398       sqlite3_memdebug_dump ./memusage.txt
   399     }
   400   }
   401   
   402   show_memstats
   403   puts "Maximum memory usage: [sqlite3_memory_highwater 1] bytes"
   404   puts "Current memory usage: [sqlite3_memory_highwater] bytes"
   405   if {[info commands sqlite3_memdebug_malloc_count] ne ""} {
   406     puts "Number of malloc()  : [sqlite3_memdebug_malloc_count] calls"
   407   }
   408 
   409   if {[info exists ::tester_do_malloctrace]} {
   410     puts "Writing mallocs.sql..."
   411     memdebug_log_sql
   412     sqlite3_memdebug_log stop
   413     sqlite3_memdebug_log clear
   414 
   415     if {[sqlite3_memory_used]>0} {
   416       puts "Writing leaks.sql..."
   417       sqlite3_memdebug_log sync
   418       memdebug_log_sql leaks.sql
   419     }
   420   }
   421 
   422   foreach f [glob -nocomplain test.db-*-journal] {
   423     file delete -force $f
   424   }
   425   
   426   foreach f [glob -nocomplain test.db-mj*] {
   427     file delete -force $f
   428   }
   429 
   430 #Symbian OS - delete_test_files() is called to cleanup after the tests execution
   431   delete_test_files
   432 
   433 # Symbian OS: output TEF format summary
   434   tef_summary
   435 
   436   exit [expr {$nErr>0}]
   437 }
   438 
   439 # Display memory statistics for analysis and debugging purposes.
   440 #
   441 proc show_memstats {} {
   442   set x [sqlite3_status SQLITE_STATUS_MEMORY_USED 0]
   443   set y [sqlite3_status SQLITE_STATUS_MALLOC_SIZE 0]
   444   set val [format {now %10d  max %10d  max-size %10d} \
   445               [lindex $x 1] [lindex $x 2] [lindex $y 2]]
   446   puts "Memory used:          $val"
   447   set x [sqlite3_status SQLITE_STATUS_PAGECACHE_USED 0]
   448   set y [sqlite3_status SQLITE_STATUS_PAGECACHE_SIZE 0]
   449   set val [format {now %10d  max %10d  max-size %10d} \
   450               [lindex $x 1] [lindex $x 2] [lindex $y 2]]
   451   puts "Page-cache used:      $val"
   452   set x [sqlite3_status SQLITE_STATUS_PAGECACHE_OVERFLOW 0]
   453   set val [format {now %10d  max %10d} [lindex $x 1] [lindex $x 2]]
   454   puts "Page-cache overflow:  $val"
   455   set x [sqlite3_status SQLITE_STATUS_SCRATCH_USED 0]
   456   set val [format {now %10d  max %10d} [lindex $x 1] [lindex $x 2]]
   457   puts "Scratch memory used:  $val"
   458   set x [sqlite3_status SQLITE_STATUS_SCRATCH_OVERFLOW 0]
   459   set y [sqlite3_status SQLITE_STATUS_SCRATCH_SIZE 0]
   460   set val [format {now %10d  max %10d  max-size %10d} \
   461                [lindex $x 1] [lindex $x 2] [lindex $y 2]]
   462   puts "Scratch overflow:     $val"
   463   ifcapable yytrackmaxstackdepth {
   464     set x [sqlite3_status SQLITE_STATUS_PARSER_STACK 0]
   465     set val [format {               max %10d} [lindex $x 2]]
   466     puts "Parser stack depth:    $val"
   467   }
   468 }
   469 
   470 # A procedure to execute SQL
   471 #
   472 proc execsql {sql {db db}} {
   473   # puts "SQL = $sql"
   474   uplevel [list $db eval $sql]
   475 }
   476 
   477 # Execute SQL and catch exceptions.
   478 #
   479 proc catchsql {sql {db db}} {
   480   # puts "SQL = $sql"
   481   set r [catch {$db eval $sql} msg]
   482   lappend r $msg
   483   return $r
   484 }
   485 
   486 # Do an VDBE code dump on the SQL given
   487 #
   488 proc explain {sql {db db}} {
   489   puts ""
   490   puts "addr  opcode        p1      p2      p3      p4               p5  #"
   491   puts "----  ------------  ------  ------  ------  ---------------  --  -"
   492   $db eval "explain $sql" {} {
   493     puts [format {%-4d  %-12.12s  %-6d  %-6d  %-6d  % -17s %s  %s} \
   494       $addr $opcode $p1 $p2 $p3 $p4 $p5 $comment
   495     ]
   496   }
   497 }
   498 
   499 # Show the VDBE program for an SQL statement but omit the Trace
   500 # opcode at the beginning.  This procedure can be used to prove
   501 # that different SQL statements generate exactly the same VDBE code.
   502 #
   503 proc explain_no_trace {sql} {
   504   set tr [db eval "EXPLAIN $sql"]
   505   return [lrange $tr 7 end]
   506 }
   507 
   508 # Another procedure to execute SQL.  This one includes the field
   509 # names in the returned list.
   510 #
   511 proc execsql2 {sql} {
   512   set result {}
   513   db eval $sql data {
   514     foreach f $data(*) {
   515       lappend result $f $data($f)
   516     }
   517   }
   518   return $result
   519 }
   520 
   521 # Use the non-callback API to execute multiple SQL statements
   522 #
   523 proc stepsql {dbptr sql} {
   524   set sql [string trim $sql]
   525   set r 0
   526   while {[string length $sql]>0} {
   527     if {[catch {sqlite3_prepare $dbptr $sql -1 sqltail} vm]} {
   528       return [list 1 $vm]
   529     }
   530     set sql [string trim $sqltail]
   531 #    while {[sqlite_step $vm N VAL COL]=="SQLITE_ROW"} {
   532 #      foreach v $VAL {lappend r $v}
   533 #    }
   534     while {[sqlite3_step $vm]=="SQLITE_ROW"} {
   535       for {set i 0} {$i<[sqlite3_data_count $vm]} {incr i} {
   536         lappend r [sqlite3_column_text $vm $i]
   537       }
   538     }
   539     if {[catch {sqlite3_finalize $vm} errmsg]} {
   540       return [list 1 $errmsg]
   541     }
   542   }
   543   return $r
   544 }
   545 
   546 # Delete a file or directory
   547 #
   548 proc forcedelete {filename} {
   549   if {[catch {file delete -force $filename}]} {
   550     exec rm -rf $filename
   551   }
   552 }
   553 
   554 # Do an integrity check of the entire database
   555 #
   556 proc integrity_check {name} {
   557   ifcapable integrityck {
   558     do_test $name {
   559       execsql {PRAGMA integrity_check}
   560     } {ok}
   561   }
   562 }
   563 
   564 proc fix_ifcapable_expr {expr} {
   565   set ret ""
   566   set state 0
   567   for {set i 0} {$i < [string length $expr]} {incr i} {
   568     set char [string range $expr $i $i]
   569     set newstate [expr {[string is alnum $char] || $char eq "_"}]
   570     if {$newstate && !$state} {
   571       append ret {$::sqlite_options(}
   572     }
   573     if {!$newstate && $state} {
   574       append ret )
   575     }
   576     append ret $char
   577     set state $newstate
   578   }
   579   if {$state} {append ret )}
   580   return $ret
   581 }
   582 
   583 # Evaluate a boolean expression of capabilities.  If true, execute the
   584 # code.  Omit the code if false.
   585 #
   586 proc ifcapable {expr code {else ""} {elsecode ""}} {
   587   #regsub -all {[a-z_0-9]+} $expr {$::sqlite_options(&)} e2
   588   set e2 [fix_ifcapable_expr $expr]
   589   if ($e2) {
   590     set c [catch {uplevel 1 $code} r]
   591   } else {
   592     set c [catch {uplevel 1 $elsecode} r]
   593   }
   594   return -code $c $r
   595 }
   596 
   597 # This proc execs a seperate process that crashes midway through executing
   598 # the SQL script $sql on database test.db.
   599 #
   600 # The crash occurs during a sync() of file $crashfile. When the crash
   601 # occurs a random subset of all unsynced writes made by the process are
   602 # written into the files on disk. Argument $crashdelay indicates the
   603 # number of file syncs to wait before crashing.
   604 #
   605 # The return value is a list of two elements. The first element is a
   606 # boolean, indicating whether or not the process actually crashed or
   607 # reported some other error. The second element in the returned list is the
   608 # error message. This is "child process exited abnormally" if the crash
   609 # occured.
   610 #
   611 #   crashsql -delay CRASHDELAY -file CRASHFILE ?-blocksize BLOCKSIZE? $sql
   612 #
   613 proc crashsql {args} {
   614   if {$::tcl_platform(platform)!="unix"} {
   615     error "crashsql should only be used on unix"
   616   }
   617 
   618   set blocksize ""
   619   set crashdelay 1
   620   set prngseed 0
   621   set tclbody {}
   622   set crashfile ""
   623   set dc ""
   624   set sql [lindex $args end]
   625   
   626   for {set ii 0} {$ii < [llength $args]-1} {incr ii 2} {
   627     set z [lindex $args $ii]
   628     set n [string length $z]
   629     set z2 [lindex $args [expr $ii+1]]
   630 
   631     if     {$n>1 && [string first $z -delay]==0}     {set crashdelay $z2} \
   632     elseif {$n>1 && [string first $z -seed]==0}      {set prngseed $z2} \
   633     elseif {$n>1 && [string first $z -file]==0}      {set crashfile $z2}  \
   634     elseif {$n>1 && [string first $z -tclbody]==0}   {set tclbody $z2}  \
   635     elseif {$n>1 && [string first $z -blocksize]==0} {set blocksize "-s $z2" } \
   636     elseif {$n>1 && [string first $z -characteristics]==0} {set dc "-c {$z2}" } \
   637     else   { error "Unrecognized option: $z" }
   638   }
   639 
   640   if {$crashfile eq ""} {
   641     error "Compulsory option -file missing"
   642   }
   643 
   644   set cfile [file join [pwd] $crashfile]
   645 
   646   set f [open crash.tcl w]
   647   puts $f "sqlite3_crash_enable 1"
   648   puts $f "sqlite3_crashparams $blocksize $dc $crashdelay $cfile"
   649   puts $f "set sqlite_pending_byte $::sqlite_pending_byte"
   650   puts $f "sqlite3 db test.db -vfs crash"
   651 
   652   # This block sets the cache size of the main database to 10
   653   # pages. This is done in case the build is configured to omit
   654   # "PRAGMA cache_size".
   655   puts $f {db eval {SELECT * FROM sqlite_master;}}
   656   puts $f {set bt [btree_from_db db]}
   657   puts $f {btree_set_cache_size $bt 10}
   658   if {$prngseed} {
   659     set seed [expr {$prngseed%10007+1}]
   660     # puts seed=$seed
   661     puts $f "db eval {SELECT randomblob($seed)}"
   662   }
   663 
   664   if {[string length $tclbody]>0} {
   665     puts $f $tclbody
   666   }
   667   if {[string length $sql]>0} {
   668     puts $f "db eval {"
   669     puts $f   "$sql"
   670     puts $f "}"
   671   }
   672   close $f
   673 
   674   set r [catch {
   675     exec [info nameofexec] crash.tcl >@stdout
   676   } msg]
   677   lappend r $msg
   678 }
   679 
   680 # Usage: do_ioerr_test <test number> <options...>
   681 #
   682 # This proc is used to implement test cases that check that IO errors
   683 # are correctly handled. The first argument, <test number>, is an integer 
   684 # used to name the tests executed by this proc. Options are as follows:
   685 #
   686 #     -tclprep          TCL script to run to prepare test.
   687 #     -sqlprep          SQL script to run to prepare test.
   688 #     -tclbody          TCL script to run with IO error simulation.
   689 #     -sqlbody          TCL script to run with IO error simulation.
   690 #     -exclude          List of 'N' values not to test.
   691 #     -erc              Use extended result codes
   692 #     -persist          Make simulated I/O errors persistent
   693 #     -start            Value of 'N' to begin with (default 1)
   694 #
   695 #     -cksum            Boolean. If true, test that the database does
   696 #                       not change during the execution of the test case.
   697 #
   698 proc do_ioerr_test {testname args} {
   699 
   700   set ::ioerropts(-start) 1
   701   set ::ioerropts(-cksum) 0
   702   set ::ioerropts(-erc) 0
   703   set ::ioerropts(-count) 100000000
   704   set ::ioerropts(-persist) 1
   705   set ::ioerropts(-ckrefcount) 0
   706   set ::ioerropts(-restoreprng) 1
   707   array set ::ioerropts $args
   708 
   709   # TEMPORARY: For 3.5.9, disable testing of extended result codes. There are
   710   # a couple of obscure IO errors that do not return them.
   711   set ::ioerropts(-erc) 0
   712 
   713   set ::go 1
   714   #reset_prng_state
   715   save_prng_state
   716   for {set n $::ioerropts(-start)} {$::go && $n<200} {incr n} {
   717     set ::TN $n
   718     incr ::ioerropts(-count) -1
   719     if {$::ioerropts(-count)<0} break
   720  
   721     # Skip this IO error if it was specified with the "-exclude" option.
   722     if {[info exists ::ioerropts(-exclude)]} {
   723       if {[lsearch $::ioerropts(-exclude) $n]!=-1} continue
   724     }
   725     if {$::ioerropts(-restoreprng)} {
   726       restore_prng_state
   727     }
   728 
   729     # Delete the files test.db and test2.db, then execute the TCL and 
   730     # SQL (in that order) to prepare for the test case.
   731     do_test $testname.$n.1 {
   732       set ::sqlite_io_error_pending 0
   733       catch {db close}
   734       catch {file delete -force test.db}
   735       catch {file delete -force test.db-journal}
   736       catch {file delete -force test2.db}
   737       catch {file delete -force test2.db-journal}
   738       set ::DB [sqlite3 db test.db; sqlite3_connection_pointer db]
   739       sqlite3_extended_result_codes $::DB $::ioerropts(-erc)
   740       if {[info exists ::ioerropts(-tclprep)]} {
   741         eval $::ioerropts(-tclprep)
   742       }
   743       if {[info exists ::ioerropts(-sqlprep)]} {
   744         execsql $::ioerropts(-sqlprep)
   745       }
   746       expr 0
   747     } {0}
   748 
   749     # Read the 'checksum' of the database.
   750     if {$::ioerropts(-cksum)} {
   751       set checksum [cksum]
   752     }
   753 
   754     # Set the Nth IO error to fail.
   755     do_test $testname.$n.2 [subst {
   756       set ::sqlite_io_error_persist $::ioerropts(-persist)
   757       set ::sqlite_io_error_pending $n
   758     }] $n
   759   
   760     # Create a single TCL script from the TCL and SQL specified
   761     # as the body of the test.
   762     set ::ioerrorbody {}
   763     if {[info exists ::ioerropts(-tclbody)]} {
   764       append ::ioerrorbody "$::ioerropts(-tclbody)\n"
   765     }
   766     if {[info exists ::ioerropts(-sqlbody)]} {
   767       append ::ioerrorbody "db eval {$::ioerropts(-sqlbody)}"
   768     }
   769 
   770     # Execute the TCL Script created in the above block. If
   771     # there are at least N IO operations performed by SQLite as
   772     # a result of the script, the Nth will fail.
   773     do_test $testname.$n.3 {
   774       set ::sqlite_io_error_hit 0
   775       set ::sqlite_io_error_hardhit 0
   776       set r [catch $::ioerrorbody msg]
   777       set ::errseen $r
   778       set rc [sqlite3_errcode $::DB]
   779       if {$::ioerropts(-erc)} {
   780         # If we are in extended result code mode, make sure all of the
   781         # IOERRs we get back really do have their extended code values.
   782         # If an extended result code is returned, the sqlite3_errcode
   783         # TCLcommand will return a string of the form:  SQLITE_IOERR+nnnn
   784         # where nnnn is a number
   785         if {[regexp {^SQLITE_IOERR} $rc] && ![regexp {IOERR\+\d} $rc]} {
   786           return $rc
   787         }
   788       } else {
   789         # If we are not in extended result code mode, make sure no
   790         # extended error codes are returned.
   791         if {[regexp {\+\d} $rc]} {
   792           return $rc
   793         }
   794       }
   795       # The test repeats as long as $::go is non-zero.  $::go starts out
   796       # as 1.  When a test runs to completion without hitting an I/O
   797       # error, that means there is no point in continuing with this test
   798       # case so set $::go to zero.
   799       #
   800       if {$::sqlite_io_error_pending>0} {
   801         set ::go 0
   802         set q 0
   803         set ::sqlite_io_error_pending 0
   804       } else {
   805         set q 1
   806       }
   807 
   808       set s [expr $::sqlite_io_error_hit==0]
   809       if {$::sqlite_io_error_hit>$::sqlite_io_error_hardhit && $r==0} {
   810         set r 1
   811       }
   812       set ::sqlite_io_error_hit 0
   813 
   814       # One of two things must have happened. either
   815       #   1.  We never hit the IO error and the SQL returned OK
   816       #   2.  An IO error was hit and the SQL failed
   817       #
   818       expr { ($s && !$r && !$q) || (!$s && $r && $q) }
   819     } {1}
   820 
   821     set ::sqlite_io_error_hit 0
   822     set ::sqlite_io_error_pending 0
   823 
   824     # Check that no page references were leaked. There should be 
   825     # a single reference if there is still an active transaction, 
   826     # or zero otherwise.
   827     #
   828     # UPDATE: If the IO error occurs after a 'BEGIN' but before any
   829     # locks are established on database files (i.e. if the error 
   830     # occurs while attempting to detect a hot-journal file), then
   831     # there may 0 page references and an active transaction according
   832     # to [sqlite3_get_autocommit].
   833     #
   834     if {$::go && $::sqlite_io_error_hardhit && $::ioerropts(-ckrefcount)} {
   835       do_test $testname.$n.4 {
   836         set bt [btree_from_db db]
   837         db_enter db
   838         array set stats [btree_pager_stats $bt]
   839         db_leave db
   840         set nRef $stats(ref)
   841         expr {$nRef == 0 || ([sqlite3_get_autocommit db]==0 && $nRef == 1)}
   842       } {1}
   843     }
   844 
   845     # If there is an open database handle and no open transaction, 
   846     # and the pager is not running in exclusive-locking mode,
   847     # check that the pager is in "unlocked" state. Theoretically,
   848     # if a call to xUnlock() failed due to an IO error the underlying
   849     # file may still be locked.
   850     #
   851     ifcapable pragma {
   852       if { [info commands db] ne ""
   853         && $::ioerropts(-ckrefcount)
   854         && [db one {pragma locking_mode}] eq "normal"
   855         && [sqlite3_get_autocommit db]
   856       } {
   857         do_test $testname.$n.5 {
   858           set bt [btree_from_db db]
   859           db_enter db
   860           array set stats [btree_pager_stats $bt]
   861           db_leave db
   862           set stats(state)
   863         } 0
   864       }
   865     }
   866 
   867     # If an IO error occured, then the checksum of the database should
   868     # be the same as before the script that caused the IO error was run.
   869     #
   870     if {$::go && $::sqlite_io_error_hardhit && $::ioerropts(-cksum)} {
   871       do_test $testname.$n.6 {
   872         catch {db close}
   873         catch {db2 close}
   874         set ::DB [sqlite3 db test.db; sqlite3_connection_pointer db]
   875         cksum
   876       } $checksum
   877     }
   878 
   879     set ::sqlite_io_error_hardhit 0
   880     set ::sqlite_io_error_pending 0
   881     if {[info exists ::ioerropts(-cleanup)]} {
   882       catch $::ioerropts(-cleanup)
   883     }
   884   }
   885   set ::sqlite_io_error_pending 0
   886   set ::sqlite_io_error_persist 0
   887   unset ::ioerropts
   888 }
   889 
   890 # Return a checksum based on the contents of the main database associated
   891 # with connection $db
   892 #
   893 proc cksum {{db db}} {
   894   set txt [$db eval {
   895       SELECT name, type, sql FROM sqlite_master order by name
   896   }]\n
   897   foreach tbl [$db eval {
   898       SELECT name FROM sqlite_master WHERE type='table' order by name
   899   }] {
   900     append txt [$db eval "SELECT * FROM $tbl"]\n
   901   }
   902   foreach prag {default_synchronous default_cache_size} {
   903     append txt $prag-[$db eval "PRAGMA $prag"]\n
   904   }
   905   set cksum [string length $txt]-[md5 $txt]
   906   # puts $cksum-[file size test.db]
   907   return $cksum
   908 }
   909 
   910 # Generate a checksum based on the contents of the main and temp tables
   911 # database $db. If the checksum of two databases is the same, and the
   912 # integrity-check passes for both, the two databases are identical.
   913 #
   914 proc allcksum {{db db}} {
   915   set ret [list]
   916   ifcapable tempdb {
   917     set sql {
   918       SELECT name FROM sqlite_master WHERE type = 'table' UNION
   919       SELECT name FROM sqlite_temp_master WHERE type = 'table' UNION
   920       SELECT 'sqlite_master' UNION
   921       SELECT 'sqlite_temp_master' ORDER BY 1
   922     }
   923   } else {
   924     set sql {
   925       SELECT name FROM sqlite_master WHERE type = 'table' UNION
   926       SELECT 'sqlite_master' ORDER BY 1
   927     }
   928   }
   929   set tbllist [$db eval $sql]
   930   set txt {}
   931   foreach tbl $tbllist {
   932     append txt [$db eval "SELECT * FROM $tbl"]
   933   }
   934   foreach prag {default_cache_size} {
   935     append txt $prag-[$db eval "PRAGMA $prag"]\n
   936   }
   937   # puts txt=$txt
   938   return [md5 $txt]
   939 }
   940 
   941 proc memdebug_log_sql {{filename mallocs.sql}} {
   942 
   943   set data [sqlite3_memdebug_log dump]
   944   set nFrame [expr [llength [lindex $data 0]]-2]
   945   if {$nFrame < 0} { return "" }
   946 
   947   set database temp
   948 
   949   set tbl "CREATE TABLE ${database}.malloc(nCall, nByte"
   950   for {set ii 1} {$ii <= $nFrame} {incr ii} {
   951     append tbl ", f${ii}"
   952   }
   953   append tbl ");\n"
   954 
   955   set sql ""
   956   foreach e $data {
   957     append sql "INSERT INTO ${database}.malloc VALUES([join $e ,]);\n"
   958     foreach f [lrange $e 2 end] {
   959       set frames($f) 1
   960     }
   961   }
   962 
   963   set tbl2 "CREATE TABLE ${database}.frame(frame INTEGER PRIMARY KEY, line);\n"
   964   set tbl3 "CREATE TABLE ${database}.file(name PRIMARY KEY, content);\n"
   965 
   966   foreach f [array names frames] {
   967     set addr [format %x $f]
   968     set cmd "addr2line -e [info nameofexec] $addr"
   969     set line [eval exec $cmd]
   970     append sql "INSERT INTO ${database}.frame VALUES($f, '$line');\n"
   971 
   972     set file [lindex [split $line :] 0]
   973     set files($file) 1
   974   }
   975 
   976   foreach f [array names files] {
   977     set contents ""
   978     catch {
   979       set fd [open $f]
   980       set contents [read $fd]
   981       close $fd
   982     }
   983     set contents [string map {' ''} $contents]
   984     append sql "INSERT INTO ${database}.file VALUES('$f', '$contents');\n"
   985   }
   986 
   987   set fd [open $filename w]
   988   puts $fd "BEGIN; ${tbl}${tbl2}${tbl3}${sql} ; COMMIT;"
   989   close $fd
   990 }
   991 
   992 # Copy file $from into $to. This is used because some versions of
   993 # TCL for windows (notably the 8.4.1 binary package shipped with the
   994 # current mingw release) have a broken "file copy" command.
   995 #
   996 proc copy_file {from to} {
   997   if {$::tcl_platform(platform)=="unix"} {
   998     file copy -force $from $to
   999   } else {
  1000     set f [open $from]
  1001     fconfigure $f -translation binary
  1002     set t [open $to w]
  1003     fconfigure $t -translation binary
  1004     puts -nonewline $t [read $f [file size $from]]
  1005     close $t
  1006     close $f
  1007   }
  1008 }
  1009 
  1010 # If the library is compiled with the SQLITE_DEFAULT_AUTOVACUUM macro set
  1011 # to non-zero, then set the global variable $AUTOVACUUM to 1.
  1012 set AUTOVACUUM $sqlite_options(default_autovacuum)