os/persistentdata/persistentstorage/sqlite3api/TEST/TclScript/malloc_common.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
# 2007 May 05
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 common code used by many different malloc tests
sl@0
    13
# within the test suite.
sl@0
    14
#
sl@0
    15
# $Id: malloc_common.tcl,v 1.22 2008/09/23 16:41:30 danielk1977 Exp $
sl@0
    16
sl@0
    17
# If we did not compile with malloc testing enabled, then do nothing.
sl@0
    18
#
sl@0
    19
ifcapable builtin_test {
sl@0
    20
  set MEMDEBUG 1
sl@0
    21
} else {
sl@0
    22
  set MEMDEBUG 0
sl@0
    23
  return 0
sl@0
    24
}
sl@0
    25
sl@0
    26
# Usage: do_malloc_test <test number> <options...>
sl@0
    27
#
sl@0
    28
# The first argument, <test number>, is an integer used to name the
sl@0
    29
# tests executed by this proc. Options are as follows:
sl@0
    30
#
sl@0
    31
#     -tclprep          TCL script to run to prepare test.
sl@0
    32
#     -sqlprep          SQL script to run to prepare test.
sl@0
    33
#     -tclbody          TCL script to run with malloc failure simulation.
sl@0
    34
#     -sqlbody          TCL script to run with malloc failure simulation.
sl@0
    35
#     -cleanup          TCL script to run after the test.
sl@0
    36
#
sl@0
    37
# This command runs a series of tests to verify SQLite's ability
sl@0
    38
# to handle an out-of-memory condition gracefully. It is assumed
sl@0
    39
# that if this condition occurs a malloc() call will return a
sl@0
    40
# NULL pointer. Linux, for example, doesn't do that by default. See
sl@0
    41
# the "BUGS" section of malloc(3).
sl@0
    42
#
sl@0
    43
# Each iteration of a loop, the TCL commands in any argument passed
sl@0
    44
# to the -tclbody switch, followed by the SQL commands in any argument
sl@0
    45
# passed to the -sqlbody switch are executed. Each iteration the
sl@0
    46
# Nth call to sqliteMalloc() is made to fail, where N is increased
sl@0
    47
# each time the loop runs starting from 1. When all commands execute
sl@0
    48
# successfully, the loop ends.
sl@0
    49
#
sl@0
    50
proc do_malloc_test {tn args} {
sl@0
    51
  array unset ::mallocopts 
sl@0
    52
  array set ::mallocopts $args
sl@0
    53
sl@0
    54
  if {[string is integer $tn]} {
sl@0
    55
    set tn malloc-$tn
sl@0
    56
  }
sl@0
    57
  if {[info exists ::mallocopts(-start)]} {
sl@0
    58
    set start $::mallocopts(-start)
sl@0
    59
  } else {
sl@0
    60
    set start 0
sl@0
    61
  }
sl@0
    62
  if {[info exists ::mallocopts(-end)]} {
sl@0
    63
    set end $::mallocopts(-end)
sl@0
    64
  } else {
sl@0
    65
    set end 50000
sl@0
    66
  }
sl@0
    67
  save_prng_state
sl@0
    68
sl@0
    69
  foreach ::iRepeat {0 10000000} {
sl@0
    70
    set ::go 1
sl@0
    71
    for {set ::n $start} {$::go && $::n <= $end} {incr ::n} {
sl@0
    72
sl@0
    73
      # If $::iRepeat is 0, then the malloc() failure is transient - it
sl@0
    74
      # fails and then subsequent calls succeed. If $::iRepeat is 1, 
sl@0
    75
      # then the failure is persistent - once malloc() fails it keeps
sl@0
    76
      # failing.
sl@0
    77
      #
sl@0
    78
      set zRepeat "transient"
sl@0
    79
      if {$::iRepeat} {set zRepeat "persistent"}
sl@0
    80
      restore_prng_state
sl@0
    81
      foreach file [glob -nocomplain test.db-mj*] {file delete -force $file}
sl@0
    82
sl@0
    83
      do_test ${tn}.${zRepeat}.${::n} {
sl@0
    84
  
sl@0
    85
        # Remove all traces of database files test.db and test2.db 
sl@0
    86
        # from the file-system. Then open (empty database) "test.db" 
sl@0
    87
        # with the handle [db].
sl@0
    88
        # 
sl@0
    89
        catch {db close} 
sl@0
    90
        catch {file delete -force test.db}
sl@0
    91
        catch {file delete -force test.db-journal}
sl@0
    92
        catch {file delete -force test2.db}
sl@0
    93
        catch {file delete -force test2.db-journal}
sl@0
    94
        if {[info exists ::mallocopts(-testdb)]} {
sl@0
    95
          file copy $::mallocopts(-testdb) test.db
sl@0
    96
        }
sl@0
    97
        catch { sqlite3 db test.db }
sl@0
    98
        if {[info commands db] ne ""} {
sl@0
    99
          sqlite3_extended_result_codes db 1
sl@0
   100
        }
sl@0
   101
        sqlite3_db_config_lookaside db 0 0 0
sl@0
   102
  
sl@0
   103
        # Execute any -tclprep and -sqlprep scripts.
sl@0
   104
        #
sl@0
   105
        if {[info exists ::mallocopts(-tclprep)]} {
sl@0
   106
          eval $::mallocopts(-tclprep)
sl@0
   107
        }
sl@0
   108
        if {[info exists ::mallocopts(-sqlprep)]} {
sl@0
   109
          execsql $::mallocopts(-sqlprep)
sl@0
   110
        }
sl@0
   111
  
sl@0
   112
        # Now set the ${::n}th malloc() to fail and execute the -tclbody 
sl@0
   113
        # and -sqlbody scripts.
sl@0
   114
        #
sl@0
   115
        sqlite3_memdebug_fail $::n -repeat $::iRepeat
sl@0
   116
        set ::mallocbody {}
sl@0
   117
        if {[info exists ::mallocopts(-tclbody)]} {
sl@0
   118
          append ::mallocbody "$::mallocopts(-tclbody)\n"
sl@0
   119
        }
sl@0
   120
        if {[info exists ::mallocopts(-sqlbody)]} {
sl@0
   121
          append ::mallocbody "db eval {$::mallocopts(-sqlbody)}"
sl@0
   122
        }
sl@0
   123
sl@0
   124
        # The following block sets local variables as follows:
sl@0
   125
        #
sl@0
   126
        #     isFail  - True if an error (any error) was reported by sqlite.
sl@0
   127
        #     nFail   - The total number of simulated malloc() failures.
sl@0
   128
        #     nBenign - The number of benign simulated malloc() failures.
sl@0
   129
        #
sl@0
   130
        set isFail [catch $::mallocbody msg]
sl@0
   131
        set nFail [sqlite3_memdebug_fail -1 -benigncnt nBenign]
sl@0
   132
        # puts -nonewline " (isFail=$isFail nFail=$nFail nBenign=$nBenign) "
sl@0
   133
sl@0
   134
        # If one or more mallocs failed, run this loop body again.
sl@0
   135
        #
sl@0
   136
        set go [expr {$nFail>0}]
sl@0
   137
sl@0
   138
        if {($nFail-$nBenign)==0} {
sl@0
   139
          if {$isFail} {
sl@0
   140
            set v2 $msg
sl@0
   141
          } else {
sl@0
   142
            set isFail 1
sl@0
   143
            set v2 1
sl@0
   144
          }
sl@0
   145
        } elseif {!$isFail} {
sl@0
   146
          set v2 $msg
sl@0
   147
        } elseif {
sl@0
   148
          [info command db]=="" || 
sl@0
   149
          [db errorcode]==7 ||
sl@0
   150
          $msg=="out of memory"
sl@0
   151
        } {
sl@0
   152
          set v2 1
sl@0
   153
        } else {
sl@0
   154
          set v2 $msg
sl@0
   155
          puts [db errorcode]
sl@0
   156
        }
sl@0
   157
        lappend isFail $v2
sl@0
   158
      } {1 1}
sl@0
   159
  
sl@0
   160
      if {[info exists ::mallocopts(-cleanup)]} {
sl@0
   161
        catch [list uplevel #0 $::mallocopts(-cleanup)] msg
sl@0
   162
      }
sl@0
   163
    }
sl@0
   164
  }
sl@0
   165
  unset ::mallocopts
sl@0
   166
  sqlite3_memdebug_fail -1
sl@0
   167
}