os/persistentdata/persistentstorage/sqlite3api/TEST/TclScript/malloc4.test
author sl@SLION-WIN7.fritz.box
Fri, 15 Jun 2012 03:10:57 +0200
changeset 0 bde4ae8d615e
permissions -rw-r--r--
First public contribution.
sl@0
     1
# 2005 November 30
sl@0
     2
#
sl@0
     3
# The author disclaims copyright to this source code.  In place of
sl@0
     4
# a legal notice, here is a blessing:
sl@0
     5
#
sl@0
     6
#    May you do good and not evil.
sl@0
     7
#    May you find forgiveness for yourself and forgive others.
sl@0
     8
#    May you share freely, never taking more than you give.
sl@0
     9
#
sl@0
    10
#***********************************************************************
sl@0
    11
#
sl@0
    12
# This file contains tests to ensure that the library handles malloc() failures
sl@0
    13
# correctly. The emphasis in this file is on sqlite3_column_XXX() APIs.
sl@0
    14
#
sl@0
    15
# $Id: malloc4.test,v 1.10 2008/02/18 22:24:58 drh Exp $
sl@0
    16
sl@0
    17
#---------------------------------------------------------------------------
sl@0
    18
# NOTES ON EXPECTED BEHAVIOUR
sl@0
    19
#
sl@0
    20
# [193] When a memory allocation failure occurs during sqlite3_column_name(),
sl@0
    21
#       sqlite3_column_name16(), sqlite3_column_decltype(), or
sl@0
    22
#       sqlite3_column_decltype16() the function shall return NULL.
sl@0
    23
#
sl@0
    24
#---------------------------------------------------------------------------
sl@0
    25
sl@0
    26
set testdir [file dirname $argv0]
sl@0
    27
source $testdir/tester.tcl
sl@0
    28
source $testdir/malloc_common.tcl
sl@0
    29
sl@0
    30
# Only run these tests if memory debugging is turned on.
sl@0
    31
if {!$MEMDEBUG} {
sl@0
    32
   puts "Skipping malloc4 tests: not compiled with -DSQLITE_MEMDEBUG..."
sl@0
    33
   finish_test
sl@0
    34
   return
sl@0
    35
}
sl@0
    36
sl@0
    37
ifcapable !utf16 {
sl@0
    38
  finish_test
sl@0
    39
  return
sl@0
    40
}
sl@0
    41
sl@0
    42
proc do_stmt_test {id sql} {
sl@0
    43
  set ::sql $sql
sl@0
    44
  set go 1
sl@0
    45
  for {set n 0} {$go} {incr n} {
sl@0
    46
    set testid "malloc4-$id.$n"
sl@0
    47
sl@0
    48
    # Prepare the statement
sl@0
    49
    do_test ${testid}.1 {
sl@0
    50
      set ::STMT [sqlite3_prepare $::DB $sql -1 TAIL]
sl@0
    51
      expr [string length $::STMT] > 0
sl@0
    52
    } {1}
sl@0
    53
sl@0
    54
    # Set the Nth malloc() to fail.
sl@0
    55
    sqlite3_memdebug_fail $n -repeat 0
sl@0
    56
sl@0
    57
    # Test malloc failure in the _name(), _name16(), decltype() and
sl@0
    58
    # decltype16() APIs. Calls that occur after the malloc() failure should
sl@0
    59
    # return NULL. No error is raised though.
sl@0
    60
    #
sl@0
    61
    # ${testid}.2.1 - Call _name()
sl@0
    62
    # ${testid}.2.2 - Call _name16()
sl@0
    63
    # ${testid}.2.3 - Call _name()
sl@0
    64
    # ${testid}.2.4 - Check that the return values of the above three calls are
sl@0
    65
    #                 consistent with each other and with the simulated
sl@0
    66
    #                 malloc() failures.
sl@0
    67
    #
sl@0
    68
    # Because the code that implements the _decltype() and _decltype16() APIs
sl@0
    69
    # is the same as the _name() and _name16() implementations, we don't worry
sl@0
    70
    # about explicitly testing them.
sl@0
    71
    #
sl@0
    72
    do_test ${testid}.2.1 {
sl@0
    73
      set mf1 [expr [sqlite3_memdebug_pending] < 0]
sl@0
    74
      set ::name8  [sqlite3_column_name $::STMT 0]
sl@0
    75
      set mf2 [expr [sqlite3_memdebug_pending] < 0]
sl@0
    76
      expr {$mf1 == $mf2 || $::name8 == ""}
sl@0
    77
    } {1}
sl@0
    78
    do_test ${testid}.2.2 {
sl@0
    79
      set mf1 [expr [sqlite3_memdebug_pending] < 0]
sl@0
    80
      set ::name16 [sqlite3_column_name16 $::STMT 0]
sl@0
    81
      set ::name16 [encoding convertfrom unicode $::name16]
sl@0
    82
      set ::name16 [string range $::name16 0 end-1]
sl@0
    83
      set mf2 [expr [sqlite3_memdebug_pending] < 0]
sl@0
    84
      expr {$mf1 == $mf2 || $::name16 == ""}
sl@0
    85
    } {1}
sl@0
    86
    do_test ${testid}.2.3 {
sl@0
    87
      set mf1 [expr [sqlite3_memdebug_pending] < 0]
sl@0
    88
      set ::name8_2 [sqlite3_column_name $::STMT 0]
sl@0
    89
      set mf2 [expr [sqlite3_memdebug_pending] < 0]
sl@0
    90
      expr {$mf1 == $mf2 || $::name8_2 == ""}
sl@0
    91
    } {1}
sl@0
    92
    set ::mallocFailed [expr [sqlite3_memdebug_pending] < 0]
sl@0
    93
    do_test ${testid}.2.4 {
sl@0
    94
      expr {
sl@0
    95
        $::name8 == $::name8_2 && $::name16 == $::name8 && !$::mallocFailed ||
sl@0
    96
        $::name8 == $::name8_2 && $::name16 == "" &&        $::mallocFailed ||
sl@0
    97
        $::name8 == $::name16 && $::name8_2 == "" &&        $::mallocFailed ||
sl@0
    98
        $::name8_2 == $::name16 && $::name8 == "" &&        $::mallocFailed
sl@0
    99
      }
sl@0
   100
    } {1}
sl@0
   101
sl@0
   102
    # Step the statement so that we can call _text() and _text16().  Before
sl@0
   103
    # running sqlite3_step(), make sure that malloc() is not about to fail.
sl@0
   104
    # Memory allocation failures that occur within sqlite3_step() are tested
sl@0
   105
    # elsewhere.
sl@0
   106
    set mf [sqlite3_memdebug_pending]
sl@0
   107
    sqlite3_memdebug_fail -1
sl@0
   108
    do_test ${testid}.3 {
sl@0
   109
      sqlite3_step $::STMT
sl@0
   110
    } {SQLITE_ROW}
sl@0
   111
    sqlite3_memdebug_fail $mf
sl@0
   112
sl@0
   113
    # Test for malloc() failures within _text() and _text16().
sl@0
   114
    #
sl@0
   115
    do_test ${testid}.4.1 {
sl@0
   116
      set ::text8 [sqlite3_column_text $::STMT 0]
sl@0
   117
      set mf [expr [sqlite3_memdebug_pending] < 0 && !$::mallocFailed]
sl@0
   118
      expr {$mf==0 || $::text8 == ""}
sl@0
   119
    } {1}
sl@0
   120
    do_test ${testid}.4.2 {
sl@0
   121
      set ::text16 [sqlite3_column_text16 $::STMT 0]
sl@0
   122
      set ::text16 [encoding convertfrom unicode $::text16]
sl@0
   123
      set ::text16 [string range $::text16 0 end-1]
sl@0
   124
      set mf [expr [sqlite3_memdebug_pending] < 0 && !$::mallocFailed]
sl@0
   125
      expr {$mf==0 || $::text16 == ""}
sl@0
   126
    } {1}
sl@0
   127
    do_test ${testid}.4.3 {
sl@0
   128
      set ::text8_2 [sqlite3_column_text $::STMT 0]
sl@0
   129
      set mf [expr [sqlite3_memdebug_pending] < 0 && !$::mallocFailed]
sl@0
   130
      expr {$mf==0 || $::text8_2 == "" || ($::text16 == "" && $::text8 != "")}
sl@0
   131
    } {1}
sl@0
   132
sl@0
   133
    # Test for malloc() failures within _int(), _int64() and _real(). The only
sl@0
   134
    # way this can occur is if the string has to be translated from UTF-16 to
sl@0
   135
    # UTF-8 before being converted to a numeric value.
sl@0
   136
    do_test ${testid}.4.4.1 {
sl@0
   137
      set mf [sqlite3_memdebug_pending]
sl@0
   138
      sqlite3_memdebug_fail -1
sl@0
   139
      sqlite3_column_text16 $::STMT 0
sl@0
   140
      sqlite3_memdebug_fail $mf
sl@0
   141
      sqlite3_column_int $::STMT 0
sl@0
   142
    } {0}
sl@0
   143
    do_test ${testid}.4.5 {
sl@0
   144
      set mf [sqlite3_memdebug_pending]
sl@0
   145
      sqlite3_memdebug_fail -1
sl@0
   146
      sqlite3_column_text16 $::STMT 0
sl@0
   147
      sqlite3_memdebug_fail $mf
sl@0
   148
      sqlite3_column_int64 $::STMT 0
sl@0
   149
    } {0}
sl@0
   150
sl@0
   151
    do_test ${testid}.4.6 {
sl@0
   152
      set mf [sqlite3_memdebug_pending]
sl@0
   153
      sqlite3_memdebug_fail -1
sl@0
   154
      sqlite3_column_text16 $::STMT 0
sl@0
   155
      sqlite3_memdebug_fail $mf
sl@0
   156
      sqlite3_column_double $::STMT 0
sl@0
   157
    } {0.0}
sl@0
   158
sl@0
   159
    set mallocFailedAfterStep [expr \
sl@0
   160
      [sqlite3_memdebug_pending] < 0 && !$::mallocFailed
sl@0
   161
    ]
sl@0
   162
sl@0
   163
    sqlite3_memdebug_fail -1
sl@0
   164
    # Test that if a malloc() failed the next call to sqlite3_step() returns
sl@0
   165
    # SQLITE_ERROR. If malloc() did not fail, it should return SQLITE_DONE.
sl@0
   166
    #
sl@0
   167
    do_test ${testid}.5 {
sl@0
   168
      sqlite3_step $::STMT
sl@0
   169
    } [expr {$mallocFailedAfterStep ? "SQLITE_ERROR" : "SQLITE_DONE"}]
sl@0
   170
sl@0
   171
    do_test ${testid}.6 {
sl@0
   172
      sqlite3_finalize $::STMT
sl@0
   173
    } [expr {$mallocFailedAfterStep ? "SQLITE_NOMEM" : "SQLITE_OK"}]
sl@0
   174
sl@0
   175
    if {$::mallocFailed == 0 && $mallocFailedAfterStep == 0} {
sl@0
   176
      sqlite3_memdebug_fail -1
sl@0
   177
      set go 0
sl@0
   178
    }
sl@0
   179
  }
sl@0
   180
}
sl@0
   181
sl@0
   182
execsql {
sl@0
   183
  CREATE TABLE tbl(
sl@0
   184
    the_first_reasonably_long_column_name that_also_has_quite_a_lengthy_type
sl@0
   185
  );
sl@0
   186
  INSERT INTO tbl VALUES(
sl@0
   187
    'An extra long string. Far too long to be stored in NBFS bytes.'
sl@0
   188
  );
sl@0
   189
}
sl@0
   190
sl@0
   191
do_stmt_test 1 "SELECT * FROM tbl"
sl@0
   192
sl@0
   193
sqlite3_memdebug_fail -1
sl@0
   194
finish_test