os/persistentdata/persistentstorage/sqlite3api/TEST/TclScript/malloc.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
# 2001 September 15
sl@0
     2
#
sl@0
     3
# Portions Copyright (c) 2007-2008 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
#
sl@0
    14
# This file attempts to check the behavior of the SQLite library in 
sl@0
    15
# an out-of-memory situation. When compiled with -DSQLITE_DEBUG=1, 
sl@0
    16
# the SQLite library accepts a special command (sqlite3_memdebug_fail N C)
sl@0
    17
# which causes the N-th malloc to fail.  This special feature is used
sl@0
    18
# to see what happens in the library if a malloc were to really fail
sl@0
    19
# due to an out-of-memory situation.
sl@0
    20
#
sl@0
    21
# $Id: malloc.test,v 1.67 2008/09/23 16:41:30 danielk1977 Exp $
sl@0
    22
sl@0
    23
set testdir [file dirname $argv0]
sl@0
    24
source $testdir/tester.tcl
sl@0
    25
sl@0
    26
sl@0
    27
# Only run these tests if memory debugging is turned on.
sl@0
    28
#
sl@0
    29
source $testdir/malloc_common.tcl
sl@0
    30
if {!$MEMDEBUG} {
sl@0
    31
   puts "Skipping malloc tests: not compiled with -DSQLITE_MEMDEBUG..."
sl@0
    32
   finish_test
sl@0
    33
   return
sl@0
    34
}
sl@0
    35
sl@0
    36
if {$tcl_platform(platform)!="symbian"} {
sl@0
    37
  # Do a couple of memory dumps just to exercise the memory dump logic
sl@0
    38
  # that that we can say that we have.
sl@0
    39
  #
sl@0
    40
  puts stderr "This is a test.  Ignore the error that follows:"
sl@0
    41
  sqlite3_memdebug_dump $testdir
sl@0
    42
  puts "Memory dump to file memdump.txt..."
sl@0
    43
  sqlite3_memdebug_dump memdump.txt
sl@0
    44
}  
sl@0
    45
sl@0
    46
ifcapable bloblit&&subquery {
sl@0
    47
  do_malloc_test 1 -tclprep {
sl@0
    48
    db close
sl@0
    49
  } -tclbody {
sl@0
    50
    if {[catch {sqlite3 db test.db}]} {
sl@0
    51
      error "out of memory"
sl@0
    52
    }
sl@0
    53
    sqlite3_extended_result_codes db 1
sl@0
    54
  } -sqlbody {
sl@0
    55
    DROP TABLE IF EXISTS t1;
sl@0
    56
    CREATE TABLE t1(
sl@0
    57
       a int, b float, c double, d text, e varchar(20),
sl@0
    58
       primary key(a,b,c)
sl@0
    59
    );
sl@0
    60
    CREATE INDEX i1 ON t1(a,b);
sl@0
    61
    INSERT INTO t1 VALUES(1,2.3,4.5,'hi',x'746865726500');
sl@0
    62
    INSERT INTO t1 VALUES(6,7.0,0.8,'hello','out yonder');
sl@0
    63
    SELECT * FROM t1;
sl@0
    64
    SELECT avg(b) FROM t1 GROUP BY a HAVING b>20.0;
sl@0
    65
    DELETE FROM t1 WHERE a IN (SELECT min(a) FROM t1);
sl@0
    66
    SELECT count(*), group_concat(e) FROM t1;
sl@0
    67
    SELECT b FROM t1 ORDER BY 1 COLLATE nocase;
sl@0
    68
  } 
sl@0
    69
}
sl@0
    70
sl@0
    71
# Ensure that no file descriptors were leaked.
sl@0
    72
do_test malloc-1.X {
sl@0
    73
  catch {db close}
sl@0
    74
  set sqlite_open_file_count
sl@0
    75
} {0}
sl@0
    76
sl@0
    77
ifcapable subquery {
sl@0
    78
  do_malloc_test 2 -sqlbody {
sl@0
    79
    CREATE TABLE t1(a int, b int default 'abc', c int default 1);
sl@0
    80
    CREATE INDEX i1 ON t1(a,b);
sl@0
    81
    INSERT INTO t1 VALUES(1,1,'99 abcdefghijklmnopqrstuvwxyz');
sl@0
    82
    INSERT INTO t1 VALUES(2,4,'98 abcdefghijklmnopqrstuvwxyz');
sl@0
    83
    INSERT INTO t1 VALUES(3,9,'97 abcdefghijklmnopqrstuvwxyz');
sl@0
    84
    INSERT INTO t1 VALUES(4,16,'96 abcdefghijklmnopqrstuvwxyz');
sl@0
    85
    INSERT INTO t1 VALUES(5,25,'95 abcdefghijklmnopqrstuvwxyz');
sl@0
    86
    INSERT INTO t1 VALUES(6,36,'94 abcdefghijklmnopqrstuvwxyz');
sl@0
    87
    SELECT 'stuff', count(*) as 'other stuff', max(a+10) FROM t1;
sl@0
    88
    UPDATE t1 SET b=b||b||b||b;
sl@0
    89
    UPDATE t1 SET b=a WHERE a in (10,12,22);
sl@0
    90
    INSERT INTO t1(c,b,a) VALUES(20,10,5);
sl@0
    91
    INSERT INTO t1 SELECT * FROM t1
sl@0
    92
        WHERE a IN (SELECT a FROM t1 WHERE a<10);
sl@0
    93
    DELETE FROM t1 WHERE a>=10;
sl@0
    94
    DROP INDEX i1;
sl@0
    95
    DELETE FROM t1;
sl@0
    96
  } 
sl@0
    97
}
sl@0
    98
sl@0
    99
# Ensure that no file descriptors were leaked.
sl@0
   100
do_test malloc-2.X {
sl@0
   101
  catch {db close}
sl@0
   102
  set sqlite_open_file_count
sl@0
   103
} {0}
sl@0
   104
sl@0
   105
do_malloc_test 3 -sqlbody {
sl@0
   106
  BEGIN TRANSACTION;
sl@0
   107
  CREATE TABLE t1(a int, b int, c int);
sl@0
   108
  CREATE INDEX i1 ON t1(a,b);
sl@0
   109
  INSERT INTO t1 VALUES(1,1,99);
sl@0
   110
  INSERT INTO t1 VALUES(2,4,98);
sl@0
   111
  INSERT INTO t1 VALUES(3,9,97);
sl@0
   112
  INSERT INTO t1 VALUES(4,16,96);
sl@0
   113
  INSERT INTO t1 VALUES(5,25,95);
sl@0
   114
  INSERT INTO t1 VALUES(6,36,94);
sl@0
   115
  INSERT INTO t1(c,b,a) VALUES(20,10,5);
sl@0
   116
  DELETE FROM t1 WHERE a>=10;
sl@0
   117
  DROP INDEX i1;
sl@0
   118
  DELETE FROM t1;
sl@0
   119
  ROLLBACK;
sl@0
   120
} 
sl@0
   121
sl@0
   122
sl@0
   123
# Ensure that no file descriptors were leaked.
sl@0
   124
do_test malloc-3.X {
sl@0
   125
  catch {db close}
sl@0
   126
  set sqlite_open_file_count
sl@0
   127
} {0}
sl@0
   128
sl@0
   129
ifcapable subquery {
sl@0
   130
  do_malloc_test 4 -sqlbody {
sl@0
   131
    BEGIN TRANSACTION;
sl@0
   132
    CREATE TABLE t1(a int, b int, c int);
sl@0
   133
    CREATE INDEX i1 ON t1(a,b);
sl@0
   134
    INSERT INTO t1 VALUES(1,1,99);
sl@0
   135
    INSERT INTO t1 VALUES(2,4,98);
sl@0
   136
    INSERT INTO t1 VALUES(3,9,97);
sl@0
   137
    INSERT INTO t1 VALUES(4,16,96);
sl@0
   138
    INSERT INTO t1 VALUES(5,25,95);
sl@0
   139
    INSERT INTO t1 VALUES(6,36,94);
sl@0
   140
    UPDATE t1 SET b=a WHERE a in (10,12,22);
sl@0
   141
    INSERT INTO t1 SELECT * FROM t1
sl@0
   142
       WHERE a IN (SELECT a FROM t1 WHERE a<10);
sl@0
   143
    DROP INDEX i1;
sl@0
   144
    DELETE FROM t1;
sl@0
   145
    COMMIT;
sl@0
   146
  } 
sl@0
   147
}
sl@0
   148
sl@0
   149
# Ensure that no file descriptors were leaked.
sl@0
   150
do_test malloc-4.X {
sl@0
   151
  catch {db close}
sl@0
   152
  set sqlite_open_file_count
sl@0
   153
} {0}
sl@0
   154
sl@0
   155
ifcapable trigger {
sl@0
   156
  do_malloc_test 5 -sqlbody {
sl@0
   157
    BEGIN TRANSACTION;
sl@0
   158
    CREATE TABLE t1(a,b);
sl@0
   159
    CREATE TABLE t2(x,y);
sl@0
   160
    CREATE TRIGGER r1 AFTER INSERT ON t1 WHEN new.a = 2 BEGIN
sl@0
   161
      INSERT INTO t2(x,y) VALUES(new.rowid,1);
sl@0
   162
      INSERT INTO t2(x,y) SELECT * FROM t2;
sl@0
   163
      INSERT INTO t2 SELECT * FROM t2;
sl@0
   164
      UPDATE t2 SET y=y+1 WHERE x=new.rowid;
sl@0
   165
      SELECT 123;
sl@0
   166
      DELETE FROM t2 WHERE x=new.rowid;
sl@0
   167
    END;
sl@0
   168
    INSERT INTO t1(a,b) VALUES(2,3);
sl@0
   169
    COMMIT;
sl@0
   170
  } 
sl@0
   171
}
sl@0
   172
sl@0
   173
# Ensure that no file descriptors were leaked.
sl@0
   174
do_test malloc-5.X {
sl@0
   175
  catch {db close}
sl@0
   176
  set sqlite_open_file_count
sl@0
   177
} {0}
sl@0
   178
sl@0
   179
ifcapable vacuum {
sl@0
   180
  do_malloc_test 6 -sqlprep {
sl@0
   181
    BEGIN TRANSACTION;
sl@0
   182
    CREATE TABLE t1(a);
sl@0
   183
    INSERT INTO t1 VALUES(1);
sl@0
   184
    INSERT INTO t1 SELECT a*2 FROM t1;
sl@0
   185
    INSERT INTO t1 SELECT a*2 FROM t1;
sl@0
   186
    INSERT INTO t1 SELECT a*2 FROM t1;
sl@0
   187
    INSERT INTO t1 SELECT a*2 FROM t1;
sl@0
   188
    INSERT INTO t1 SELECT a*2 FROM t1;
sl@0
   189
    INSERT INTO t1 SELECT a*2 FROM t1;
sl@0
   190
    INSERT INTO t1 SELECT a*2 FROM t1;
sl@0
   191
    INSERT INTO t1 SELECT a*2 FROM t1;
sl@0
   192
    INSERT INTO t1 SELECT a*2 FROM t1;
sl@0
   193
    INSERT INTO t1 SELECT a*2 FROM t1;
sl@0
   194
    DELETE FROM t1 where rowid%5 = 0;
sl@0
   195
    COMMIT;
sl@0
   196
  } -sqlbody {
sl@0
   197
    VACUUM;
sl@0
   198
  } 
sl@0
   199
}
sl@0
   200
sl@0
   201
do_malloc_test 7 -sqlprep {
sl@0
   202
  CREATE TABLE t1(a, b);
sl@0
   203
  INSERT INTO t1 VALUES(1, 2);
sl@0
   204
  INSERT INTO t1 VALUES(3, 4);
sl@0
   205
  INSERT INTO t1 VALUES(5, 6);
sl@0
   206
  INSERT INTO t1 VALUES(7, randstr(1200,1200));
sl@0
   207
} -sqlbody {
sl@0
   208
  SELECT min(a) FROM t1 WHERE a<6 GROUP BY b;
sl@0
   209
  SELECT a FROM t1 WHERE a<6 ORDER BY a;
sl@0
   210
  SELECT b FROM t1 WHERE a>6;
sl@0
   211
}
sl@0
   212
sl@0
   213
# This block is designed to test that some malloc failures that may
sl@0
   214
# occur in vdbeapi.c. Specifically, if a malloc failure that occurs
sl@0
   215
# when converting UTF-16 text to integers and real numbers is handled
sl@0
   216
# correctly. 
sl@0
   217
#
sl@0
   218
# This is done by retrieving a string from the database engine and
sl@0
   219
# manipulating it using the sqlite3_column_*** APIs. This doesn't 
sl@0
   220
# actually return an error to the user when a malloc() fails.. That 
sl@0
   221
# could be viewed as a bug.
sl@0
   222
#
sl@0
   223
# These tests only run if UTF-16 support is compiled in.
sl@0
   224
#
sl@0
   225
ifcapable utf16 {
sl@0
   226
  set ::STMT {}
sl@0
   227
  do_malloc_test 8 -tclprep {
sl@0
   228
    set sql "SELECT '[string repeat abc 20]', '[string repeat def 20]', ?"
sl@0
   229
    set ::STMT [sqlite3_prepare db $sql -1 X]
sl@0
   230
    sqlite3_step $::STMT
sl@0
   231
    if { $::tcl_platform(byteOrder)=="littleEndian" } {
sl@0
   232
      set ::bomstr "\xFF\xFE"
sl@0
   233
    } else {
sl@0
   234
      set ::bomstr "\xFE\xFF"
sl@0
   235
    }
sl@0
   236
    append ::bomstr [encoding convertto unicode "123456789_123456789_12345678"]
sl@0
   237
  } -tclbody {
sl@0
   238
    sqlite3_column_text16 $::STMT 0
sl@0
   239
    sqlite3_column_int $::STMT 0
sl@0
   240
    sqlite3_column_text16 $::STMT 1
sl@0
   241
    sqlite3_column_double $::STMT 1
sl@0
   242
    set rc [sqlite3_reset $::STMT]
sl@0
   243
    if {$rc eq "SQLITE_NOMEM"} {error "out of memory"}
sl@0
   244
    sqlite3_bind_text16 $::STMT 1 $::bomstr 60
sl@0
   245
    #catch {sqlite3_finalize $::STMT}
sl@0
   246
    #if {[lindex [sqlite_malloc_stat] 2]<=0} {
sl@0
   247
    #  error "out of memory"
sl@0
   248
    #}
sl@0
   249
  } -cleanup {
sl@0
   250
    if {$::STMT!=""} {
sl@0
   251
      sqlite3_finalize $::STMT
sl@0
   252
      set ::STMT {}
sl@0
   253
    }
sl@0
   254
  }
sl@0
   255
}
sl@0
   256
sl@0
   257
# This block tests that malloc() failures that occur whilst commiting
sl@0
   258
# a multi-file transaction are handled correctly.
sl@0
   259
#
sl@0
   260
do_malloc_test 9 -sqlprep {
sl@0
   261
  ATTACH 'test2.db' as test2;
sl@0
   262
  CREATE TABLE abc1(a, b, c);
sl@0
   263
  CREATE TABLE test2.abc2(a, b, c);
sl@0
   264
} -sqlbody {
sl@0
   265
  BEGIN;
sl@0
   266
  INSERT INTO abc1 VALUES(1, 2, 3);
sl@0
   267
  INSERT INTO abc2 VALUES(1, 2, 3);
sl@0
   268
  COMMIT;
sl@0
   269
}
sl@0
   270
sl@0
   271
# This block tests malloc() failures that occur while opening a 
sl@0
   272
# connection to a database.
sl@0
   273
do_malloc_test 10 -tclprep {
sl@0
   274
  catch {db2 close}
sl@0
   275
  db close
sl@0
   276
  file delete -force test.db test.db-journal
sl@0
   277
  sqlite3 db test.db
sl@0
   278
  sqlite3_extended_result_codes db 1
sl@0
   279
  db eval {CREATE TABLE abc(a, b, c)}
sl@0
   280
} -tclbody {
sl@0
   281
  db close
sl@0
   282
  sqlite3 db2 test.db
sl@0
   283
  sqlite3_extended_result_codes db2 1
sl@0
   284
  db2 eval {SELECT * FROM sqlite_master}
sl@0
   285
  db2 close
sl@0
   286
}
sl@0
   287
sl@0
   288
# This block tests malloc() failures that occur within calls to
sl@0
   289
# sqlite3_create_function().
sl@0
   290
do_malloc_test 11 -tclbody {
sl@0
   291
  set rc [sqlite3_create_function db]
sl@0
   292
  if {[string match $rc SQLITE_OK]} {
sl@0
   293
    set rc [sqlite3_create_aggregate db]
sl@0
   294
  }
sl@0
   295
  if {[string match $rc SQLITE_NOMEM]} {
sl@0
   296
    error "out of memory"
sl@0
   297
  }
sl@0
   298
}
sl@0
   299
sl@0
   300
do_malloc_test 12 -tclbody {
sl@0
   301
  set sql16 [encoding convertto unicode "SELECT * FROM sqlite_master"]
sl@0
   302
  append sql16 "\00\00"
sl@0
   303
  set ::STMT [sqlite3_prepare16 db $sql16 -1 DUMMY]
sl@0
   304
  sqlite3_finalize $::STMT
sl@0
   305
}
sl@0
   306
sl@0
   307
# Test malloc errors when replaying two hot journals from a 2-file 
sl@0
   308
# transaction.
sl@0
   309
ifcapable crashtest&&attach {
sl@0
   310
  do_malloc_test 13 -tclprep {
sl@0
   311
    set rc [crashsql -delay 1 -file test2.db {
sl@0
   312
      ATTACH 'test2.db' as aux;
sl@0
   313
      PRAGMA cache_size = 10;
sl@0
   314
      BEGIN;
sl@0
   315
      CREATE TABLE aux.t2(a, b, c);
sl@0
   316
      CREATE TABLE t1(a, b, c);
sl@0
   317
      COMMIT;
sl@0
   318
    }]
sl@0
   319
    if {$rc!="1 {child process exited abnormally}"} {
sl@0
   320
      error "Wrong error message: $rc"
sl@0
   321
    }
sl@0
   322
  } -tclbody {
sl@0
   323
    db eval {ATTACH 'test2.db' as aux;}
sl@0
   324
    set rc [catch {db eval {
sl@0
   325
      SELECT * FROM t1; 
sl@0
   326
      SELECT * FROM t2;
sl@0
   327
    }} err]
sl@0
   328
    if {$rc && $err!="no such table: t1"} {
sl@0
   329
      error $err
sl@0
   330
    }
sl@0
   331
  }
sl@0
   332
}
sl@0
   333
sl@0
   334
if {$tcl_platform(platform)!="windows" && $tcl_platform(platform)!="symbian"} {
sl@0
   335
  do_malloc_test 14 -tclprep {
sl@0
   336
    catch {db close}
sl@0
   337
    sqlite3 db2 test2.db
sl@0
   338
    sqlite3_extended_result_codes db2 1
sl@0
   339
    db2 eval {
sl@0
   340
      PRAGMA synchronous = 0;
sl@0
   341
      CREATE TABLE t1(a, b);
sl@0
   342
      INSERT INTO t1 VALUES(1, 2);
sl@0
   343
      BEGIN;
sl@0
   344
      INSERT INTO t1 VALUES(3, 4);
sl@0
   345
    }
sl@0
   346
    copy_file test2.db test.db
sl@0
   347
    copy_file test2.db-journal test.db-journal
sl@0
   348
    db2 close
sl@0
   349
  } -tclbody {
sl@0
   350
    sqlite3 db test.db
sl@0
   351
    sqlite3_extended_result_codes db 1
sl@0
   352
sl@0
   353
    # If an out-of-memory occurs within a call to a VFS layer function during
sl@0
   354
    # hot-journal rollback, sqlite will report SQLITE_CORRUPT. See commit
sl@0
   355
    # [5668] for details.
sl@0
   356
    set rc [catch {db eval { SELECT * FROM t1 }} msg]
sl@0
   357
    if {$msg eq "database disk image is malformed"} { set msg "out of memory" }
sl@0
   358
    if {$rc} { error $msg }
sl@0
   359
    set msg
sl@0
   360
  }
sl@0
   361
}
sl@0
   362
sl@0
   363
proc string_compare {a b} {
sl@0
   364
  return [string compare $a $b]
sl@0
   365
}
sl@0
   366
sl@0
   367
# Test for malloc() failures in sqlite3_create_collation() and 
sl@0
   368
# sqlite3_create_collation16().
sl@0
   369
#
sl@0
   370
ifcapable utf16 {
sl@0
   371
  do_malloc_test 15 -start 4 -tclbody {
sl@0
   372
    db collate string_compare string_compare
sl@0
   373
    if {[catch {add_test_collate db 1 1 1} msg]} {
sl@0
   374
      if {$msg=="SQLITE_NOMEM"} {set msg "out of memory"}
sl@0
   375
      error $msg
sl@0
   376
    }
sl@0
   377
  
sl@0
   378
    db complete {SELECT "hello """||'world"' [microsoft], * FROM anicetable;}
sl@0
   379
    db complete {-- Useful comment}
sl@0
   380
  
sl@0
   381
    execsql {
sl@0
   382
      CREATE TABLE t1(a, b COLLATE string_compare);
sl@0
   383
      INSERT INTO t1 VALUES(10, 'string');
sl@0
   384
      INSERT INTO t1 VALUES(10, 'string2');
sl@0
   385
    }
sl@0
   386
  }
sl@0
   387
}
sl@0
   388
sl@0
   389
# Also test sqlite3_complete(). There are (currently) no malloc()
sl@0
   390
# calls in this function, but test anyway against future changes.
sl@0
   391
#
sl@0
   392
do_malloc_test 16 -tclbody {
sl@0
   393
  db complete {SELECT "hello """||'world"' [microsoft], * FROM anicetable;}
sl@0
   394
  db complete {-- Useful comment}
sl@0
   395
  db eval {
sl@0
   396
    SELECT * FROM sqlite_master;
sl@0
   397
  }
sl@0
   398
}
sl@0
   399
sl@0
   400
# Test handling of malloc() failures in sqlite3_open16().
sl@0
   401
#
sl@0
   402
ifcapable utf16 {
sl@0
   403
  do_malloc_test 17 -tclbody {
sl@0
   404
    set DB2 0
sl@0
   405
    set STMT 0
sl@0
   406
  
sl@0
   407
    # open database using sqlite3_open16()
sl@0
   408
    set filename [encoding convertto unicode test.db]
sl@0
   409
    append filename "\x00\x00"
sl@0
   410
    set DB2 [sqlite3_open16 $filename -unused]
sl@0
   411
    if {0==$DB2} {
sl@0
   412
      error "out of memory"
sl@0
   413
    }
sl@0
   414
    sqlite3_extended_result_codes $DB2 1
sl@0
   415
  
sl@0
   416
    # Prepare statement
sl@0
   417
    set rc [catch {sqlite3_prepare $DB2 {SELECT * FROM sqlite_master} -1 X} msg]
sl@0
   418
    if {[sqlite3_errcode $DB2] eq "SQLITE_IOERR+12"} {
sl@0
   419
      error "out of memory"
sl@0
   420
    }
sl@0
   421
    if {[regexp ".*automatic extension loading.*" [sqlite3_errmsg $DB2]]} {
sl@0
   422
      error "out of memory"
sl@0
   423
    }
sl@0
   424
    if {$rc} {
sl@0
   425
      error [string range $msg 4 end]
sl@0
   426
    }
sl@0
   427
    set STMT $msg
sl@0
   428
  
sl@0
   429
    # Finalize statement
sl@0
   430
    set rc [sqlite3_finalize $STMT]
sl@0
   431
    if {$rc!="SQLITE_OK"} {
sl@0
   432
      error [sqlite3_errmsg $DB2]
sl@0
   433
    }
sl@0
   434
    set STMT 0
sl@0
   435
  
sl@0
   436
    # Close database
sl@0
   437
    set rc [sqlite3_close $DB2]
sl@0
   438
    if {$rc!="SQLITE_OK"} {
sl@0
   439
      error [sqlite3_errmsg $DB2]
sl@0
   440
    }
sl@0
   441
    set DB2 0
sl@0
   442
  } -cleanup {
sl@0
   443
    if {$STMT!="0"} {
sl@0
   444
      sqlite3_finalize $STMT
sl@0
   445
    }
sl@0
   446
    if {$DB2!="0"} {
sl@0
   447
      set rc [sqlite3_close $DB2]
sl@0
   448
    }
sl@0
   449
  }
sl@0
   450
}
sl@0
   451
sl@0
   452
# Test handling of malloc() failures in sqlite3_errmsg16().
sl@0
   453
#
sl@0
   454
ifcapable utf16 {
sl@0
   455
  do_malloc_test 18 -tclprep {
sl@0
   456
    catch {
sl@0
   457
      db eval "SELECT [string repeat longcolumnname 10] FROM sqlite_master"
sl@0
   458
    }
sl@0
   459
  } -tclbody {
sl@0
   460
    set utf16 [sqlite3_errmsg16 [sqlite3_connection_pointer db]]
sl@0
   461
    binary scan $utf16 c* bytes
sl@0
   462
    if {[llength $bytes]==0} {
sl@0
   463
      error "out of memory"
sl@0
   464
    }
sl@0
   465
  }
sl@0
   466
}
sl@0
   467
sl@0
   468
# This test is aimed at coverage testing. Specificly, it is supposed to
sl@0
   469
# cause a malloc() only used when converting between the two utf-16 
sl@0
   470
# encodings to fail (i.e. little-endian->big-endian). It only actually 
sl@0
   471
# hits this malloc() on little-endian hosts.
sl@0
   472
#
sl@0
   473
set static_string "\x00h\x00e\x00l\x00l\x00o"
sl@0
   474
for {set l 0} {$l<10} {incr l} {
sl@0
   475
  append static_string $static_string
sl@0
   476
}
sl@0
   477
append static_string "\x00\x00"
sl@0
   478
do_malloc_test 19 -tclprep {
sl@0
   479
  execsql {
sl@0
   480
    PRAGMA encoding = "UTF16be";
sl@0
   481
    CREATE TABLE abc(a, b, c);
sl@0
   482
  }
sl@0
   483
} -tclbody {
sl@0
   484
  unset -nocomplain ::STMT
sl@0
   485
  set r [catch {
sl@0
   486
    set ::STMT [sqlite3_prepare db {SELECT ?} -1 DUMMY]
sl@0
   487
    sqlite3_bind_text16 -static $::STMT 1 $static_string 112
sl@0
   488
  } msg]
sl@0
   489
  if {$r} {error [string range $msg 4 end]}
sl@0
   490
  set msg
sl@0
   491
} -cleanup {
sl@0
   492
  if {[info exists ::STMT]} {
sl@0
   493
    sqlite3_finalize $::STMT
sl@0
   494
  }
sl@0
   495
}
sl@0
   496
unset static_string
sl@0
   497
sl@0
   498
# Make sure SQLITE_NOMEM is reported out on an ATTACH failure even
sl@0
   499
# when the malloc failure occurs within the nested parse.
sl@0
   500
#
sl@0
   501
ifcapable attach {
sl@0
   502
  do_malloc_test 20 -tclprep {
sl@0
   503
    db close
sl@0
   504
    file delete -force test2.db test2.db-journal
sl@0
   505
    sqlite3 db test2.db
sl@0
   506
    sqlite3_extended_result_codes db 1
sl@0
   507
    db eval {CREATE TABLE t1(x);}
sl@0
   508
    db close
sl@0
   509
  } -tclbody {
sl@0
   510
    if {[catch {sqlite3 db test.db}]} {
sl@0
   511
      error "out of memory"
sl@0
   512
    }
sl@0
   513
    sqlite3_extended_result_codes db 1
sl@0
   514
  } -sqlbody {
sl@0
   515
    ATTACH DATABASE 'test2.db' AS t2;
sl@0
   516
    SELECT * FROM t1;
sl@0
   517
    DETACH DATABASE t2;
sl@0
   518
  } 
sl@0
   519
}
sl@0
   520
sl@0
   521
# Test malloc failure whilst installing a foreign key.
sl@0
   522
#
sl@0
   523
ifcapable foreignkey {
sl@0
   524
  do_malloc_test 21 -sqlbody {
sl@0
   525
    CREATE TABLE abc(a, b, c, FOREIGN KEY(a) REFERENCES abc(b))
sl@0
   526
  } 
sl@0
   527
}
sl@0
   528
sl@0
   529
# Test malloc failure in an sqlite3_prepare_v2() call.
sl@0
   530
#
sl@0
   531
do_malloc_test 22 -tclbody {
sl@0
   532
  set ::STMT ""
sl@0
   533
  set r [catch {
sl@0
   534
    set ::STMT [
sl@0
   535
      sqlite3_prepare_v2 db "SELECT * FROM sqlite_master" -1 DUMMY
sl@0
   536
    ]
sl@0
   537
  } msg]
sl@0
   538
  if {$r} {error [string range $msg 4 end]}
sl@0
   539
} -cleanup {
sl@0
   540
  if {$::STMT ne ""} {
sl@0
   541
    sqlite3_finalize $::STMT
sl@0
   542
    set ::STMT ""
sl@0
   543
  }
sl@0
   544
}
sl@0
   545
sl@0
   546
ifcapable {pager_pragmas} {
sl@0
   547
  # This tests a special case - that an error that occurs while the pager
sl@0
   548
  # is trying to recover from error-state in exclusive-access mode works.
sl@0
   549
  #
sl@0
   550
  do_malloc_test 23 -tclprep {
sl@0
   551
    db eval {
sl@0
   552
      PRAGMA cache_size = 10;
sl@0
   553
      PRAGMA locking_mode = exclusive;
sl@0
   554
      BEGIN;
sl@0
   555
      CREATE TABLE abc(a, b, c);
sl@0
   556
      CREATE INDEX abc_i ON abc(a, b, c);
sl@0
   557
      INSERT INTO abc 
sl@0
   558
        VALUES(randstr(100,100), randstr(100,100), randstr(100,100));
sl@0
   559
      INSERT INTO abc 
sl@0
   560
        SELECT randstr(100,100), randstr(100,100), randstr(100,100) FROM abc;
sl@0
   561
      INSERT INTO abc 
sl@0
   562
        SELECT randstr(100,100), randstr(100,100), randstr(100,100) FROM abc;
sl@0
   563
      INSERT INTO abc 
sl@0
   564
        SELECT randstr(100,100), randstr(100,100), randstr(100,100) FROM abc;
sl@0
   565
      INSERT INTO abc 
sl@0
   566
        SELECT randstr(100,100), randstr(100,100), randstr(100,100) FROM abc;
sl@0
   567
      INSERT INTO abc 
sl@0
   568
        SELECT randstr(100,100), randstr(100,100), randstr(100,100) FROM abc;
sl@0
   569
      COMMIT;
sl@0
   570
    }
sl@0
   571
  
sl@0
   572
    # This puts the pager into error state.
sl@0
   573
    #
sl@0
   574
    db eval BEGIN
sl@0
   575
    db eval {UPDATE abc SET a = 0 WHERE oid%2}
sl@0
   576
    set ::sqlite_io_error_pending 10
sl@0
   577
    catch {db eval {ROLLBACK}} msg
sl@0
   578
sl@0
   579
  } -tclbody {
sl@0
   580
    # If an out-of-memory occurs within a call to a VFS layer function during
sl@0
   581
    # hot-journal rollback, sqlite will report SQLITE_CORRUPT. See commit
sl@0
   582
    # [5668] for details.
sl@0
   583
    set rc [catch {db eval { SELECT * FROM abc LIMIT 10 }} msg]
sl@0
   584
    if {$msg eq "database disk image is malformed"} { set msg "out of memory" }
sl@0
   585
    if {$rc} { error $msg }
sl@0
   586
    set msg
sl@0
   587
  } -cleanup {
sl@0
   588
    set e [db eval {PRAGMA integrity_check}]
sl@0
   589
    if {$e ne "ok"} {error $e}
sl@0
   590
  }
sl@0
   591
}
sl@0
   592
sl@0
   593
ifcapable compound {
sl@0
   594
  do_malloc_test 24 -sqlprep {
sl@0
   595
    CREATE TABLE t1(a, b, c)
sl@0
   596
  } -sqlbody {
sl@0
   597
    SELECT 1 FROM t1 UNION SELECT 2 FROM t1 ORDER BY 1
sl@0
   598
  }
sl@0
   599
}
sl@0
   600
sl@0
   601
ifcapable view&&trigger {
sl@0
   602
  do_malloc_test 25 -sqlprep {
sl@0
   603
    CREATE TABLE t1(a, b, c);
sl@0
   604
    CREATE VIEW v1 AS SELECT * FROM t1;
sl@0
   605
    CREATE TRIGGER v1t1 INSTEAD OF DELETE ON v1 BEGIN SELECT 1; END;
sl@0
   606
    CREATE TRIGGER v1t2 INSTEAD OF INSERT ON v1 BEGIN SELECT 1; END;
sl@0
   607
    CREATE TRIGGER v1t3 INSTEAD OF UPDATE ON v1 BEGIN SELECT 1; END;
sl@0
   608
  } -sqlbody {
sl@0
   609
    DELETE FROM v1 WHERE a = 1;
sl@0
   610
    INSERT INTO v1 VALUES(1, 2, 3);
sl@0
   611
    UPDATE v1 SET a = 1 WHERE b = 2;
sl@0
   612
  }
sl@0
   613
}
sl@0
   614
sl@0
   615
do_malloc_test 25 -sqlprep {
sl@0
   616
  CREATE TABLE abc(a, b, c);
sl@0
   617
  CREATE INDEX i1 ON abc(a, b);
sl@0
   618
  INSERT INTO abc VALUES(1, 2, 3);
sl@0
   619
  INSERT INTO abc VALUES(4, 5, 6);
sl@0
   620
} -tclbody {
sl@0
   621
  # For each UPDATE executed, the cursor used for the SELECT statement
sl@0
   622
  # must be "saved". Because the cursor is open on an index, this requires
sl@0
   623
  # a malloc() to allocate space to save the index key. This test case is
sl@0
   624
  # aimed at testing the response of the library to a failure in that
sl@0
   625
  # particular malloc() call.
sl@0
   626
  db eval {SELECT a FROM abc ORDER BY a} {
sl@0
   627
    db eval {UPDATE abc SET b = b - 1 WHERE a = $a}
sl@0
   628
  }
sl@0
   629
}
sl@0
   630
sl@0
   631
# This test is designed to test a specific juncture in the sqlite code.
sl@0
   632
# The database set up by -sqlprep script contains a single table B-Tree
sl@0
   633
# of height 2. In the -tclbody script, the existing database connection
sl@0
   634
# is closed and a new one opened and used to insert a new row into the
sl@0
   635
# table B-Tree. By using a new connection, the outcome of a malloc() 
sl@0
   636
# failure while seeking to the right-hand side of the B-Tree to insert 
sl@0
   637
# a new record can be tested.
sl@0
   638
#
sl@0
   639
do_malloc_test 26 -sqlprep {
sl@0
   640
  BEGIN;
sl@0
   641
  CREATE TABLE t1(a, b);
sl@0
   642
  INSERT INTO t1 VALUES(1, randomblob(210));
sl@0
   643
  INSERT INTO t1 VALUES(1, randomblob(210));
sl@0
   644
  INSERT INTO t1 VALUES(1, randomblob(210));
sl@0
   645
  INSERT INTO t1 VALUES(1, randomblob(210));
sl@0
   646
  INSERT INTO t1 VALUES(1, randomblob(210));
sl@0
   647
  COMMIT;
sl@0
   648
} -tclbody {
sl@0
   649
  db close
sl@0
   650
  sqlite3 db test.db
sl@0
   651
  db eval { INSERT INTO t1 VALUES(1, randomblob(210)) }
sl@0
   652
}
sl@0
   653
sl@0
   654
# Test that no memory is leaked following a malloc() failure in
sl@0
   655
# sqlite3_initialize().
sl@0
   656
#
sl@0
   657
do_malloc_test 27 -tclprep {
sl@0
   658
  db close
sl@0
   659
  sqlite3_shutdown
sl@0
   660
} -tclbody {
sl@0
   661
  set rc [sqlite3_initialize]
sl@0
   662
  if {$rc == "SQLITE_NOMEM"} {
sl@0
   663
    error "out of memory"
sl@0
   664
  }
sl@0
   665
}
sl@0
   666
sl@0
   667
# Ensure that no file descriptors were leaked.
sl@0
   668
do_test malloc-99.X {
sl@0
   669
  catch {db close}
sl@0
   670
  set sqlite_open_file_count
sl@0
   671
} {0}
sl@0
   672
sl@0
   673
puts open-file-count=$sqlite_open_file_count
sl@0
   674
finish_test