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