os/persistentdata/persistentstorage/sqlite3api/TEST/TclScript/io.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
# 2007 August 21
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
# The focus of this file is testing some specific characteristics of the 
sl@0
    13
# IO traffic generated by SQLite (making sure SQLite is not writing out
sl@0
    14
# more database pages than it has to, stuff like that).
sl@0
    15
#
sl@0
    16
# $Id: io.test,v 1.19 2008/09/18 11:18:41 danielk1977 Exp $
sl@0
    17
sl@0
    18
set testdir [file dirname $argv0]
sl@0
    19
source $testdir/tester.tcl
sl@0
    20
sl@0
    21
db close
sl@0
    22
sqlite3_simulate_device
sl@0
    23
sqlite3 db test.db -vfs devsym
sl@0
    24
sl@0
    25
# Test summary:
sl@0
    26
#
sl@0
    27
# io-1.* -  Test that quick-balance does not journal pages unnecessarily.
sl@0
    28
#
sl@0
    29
# io-2.* -  Test the "atomic-write optimization".
sl@0
    30
#
sl@0
    31
# io-3.* -  Test the IO traffic enhancements triggered when the 
sl@0
    32
#           IOCAP_SEQUENTIAL device capability flag is set (no 
sl@0
    33
#           fsync() calls on the journal file).
sl@0
    34
#
sl@0
    35
# io-4.* -  Test the IO traffic enhancements triggered when the 
sl@0
    36
#           IOCAP_SAFE_APPEND device capability flag is set (fewer 
sl@0
    37
#           fsync() calls on the journal file, no need to set nRec
sl@0
    38
#           field in the single journal header).
sl@0
    39
#
sl@0
    40
# io-5.* -  Test that the default page size is selected and used 
sl@0
    41
#           correctly.
sl@0
    42
#           
sl@0
    43
sl@0
    44
set ::nWrite 0
sl@0
    45
proc nWrite {db} {
sl@0
    46
  set bt [btree_from_db $db]
sl@0
    47
  db_enter $db
sl@0
    48
  array set stats [btree_pager_stats $bt]
sl@0
    49
  db_leave $db
sl@0
    50
  set res [expr $stats(write) - $::nWrite]
sl@0
    51
  set ::nWrite $stats(write)
sl@0
    52
  set res
sl@0
    53
}
sl@0
    54
sl@0
    55
set ::nSync 0
sl@0
    56
proc nSync {} {
sl@0
    57
  set res [expr {$::sqlite_sync_count - $::nSync}]
sl@0
    58
  set ::nSync $::sqlite_sync_count
sl@0
    59
  set res
sl@0
    60
}
sl@0
    61
sl@0
    62
do_test io-1.1 {
sl@0
    63
  execsql {
sl@0
    64
    PRAGMA auto_vacuum = OFF;
sl@0
    65
    PRAGMA page_size = 1024;
sl@0
    66
    CREATE TABLE abc(a,b);
sl@0
    67
  }
sl@0
    68
  nWrite db
sl@0
    69
} {2}
sl@0
    70
sl@0
    71
# Insert into the table 4 records of aproximately 240 bytes each.
sl@0
    72
# This should completely fill the root-page of the table. Each
sl@0
    73
# INSERT causes 2 db pages to be written - the root-page of "abc"
sl@0
    74
# and page 1 (db change-counter page).
sl@0
    75
do_test io-1.2 {
sl@0
    76
  set ret [list]
sl@0
    77
  execsql { INSERT INTO abc VALUES(1,randstr(230,230)); }
sl@0
    78
  lappend ret [nWrite db]
sl@0
    79
  execsql { INSERT INTO abc VALUES(2,randstr(230,230)); }
sl@0
    80
  lappend ret [nWrite db]
sl@0
    81
  execsql { INSERT INTO abc VALUES(3,randstr(230,230)); }
sl@0
    82
  lappend ret [nWrite db]
sl@0
    83
  execsql { INSERT INTO abc VALUES(4,randstr(230,230)); }
sl@0
    84
  lappend ret [nWrite db]
sl@0
    85
} {2 2 2 2}
sl@0
    86
sl@0
    87
# Insert another 240 byte record. This causes two leaf pages
sl@0
    88
# to be added to the root page of abc. 4 pages in total
sl@0
    89
# are written to the db file - the two leaf pages, the root
sl@0
    90
# of abc and the change-counter page.
sl@0
    91
do_test io-1.3 {
sl@0
    92
  execsql { INSERT INTO abc VALUES(5,randstr(230,230)); }
sl@0
    93
  nWrite db
sl@0
    94
} {4}
sl@0
    95
sl@0
    96
# Insert another 3 240 byte records. After this, the tree consists of 
sl@0
    97
# the root-node, which is close to empty, and two leaf pages, both of 
sl@0
    98
# which are full. 
sl@0
    99
do_test io-1.4 {
sl@0
   100
  set ret [list]
sl@0
   101
  execsql { INSERT INTO abc VALUES(6,randstr(230,230)); }
sl@0
   102
  lappend ret [nWrite db]
sl@0
   103
  execsql { INSERT INTO abc VALUES(7,randstr(230,230)); }
sl@0
   104
  lappend ret [nWrite db]
sl@0
   105
  execsql { INSERT INTO abc VALUES(8,randstr(230,230)); }
sl@0
   106
  lappend ret [nWrite db]
sl@0
   107
} {2 2 2}
sl@0
   108
sl@0
   109
# This insert should use the quick-balance trick to add a third leaf
sl@0
   110
# to the b-tree used to store table abc. It should only be necessary to
sl@0
   111
# write to 3 pages to do this: the change-counter, the root-page and
sl@0
   112
# the new leaf page.
sl@0
   113
do_test io-1.5 {
sl@0
   114
  execsql { INSERT INTO abc VALUES(9,randstr(230,230)); }
sl@0
   115
  nWrite db
sl@0
   116
} {3}
sl@0
   117
sl@0
   118
ifcapable atomicwrite {
sl@0
   119
sl@0
   120
#----------------------------------------------------------------------
sl@0
   121
# Test cases io-2.* test the atomic-write optimization.
sl@0
   122
#
sl@0
   123
do_test io-2.1 {
sl@0
   124
  execsql { DELETE FROM abc; VACUUM; }
sl@0
   125
} {}
sl@0
   126
sl@0
   127
# Clear the write and sync counts.
sl@0
   128
nWrite db ; nSync
sl@0
   129
sl@0
   130
# The following INSERT updates 2 pages and requires 4 calls to fsync():
sl@0
   131
#
sl@0
   132
#   1) The directory in which the journal file is created,
sl@0
   133
#   2) The journal file (to sync the page data),
sl@0
   134
#   3) The journal file (to sync the journal file header),
sl@0
   135
#   4) The database file.
sl@0
   136
#
sl@0
   137
do_test io-2.2 {
sl@0
   138
  execsql { INSERT INTO abc VALUES(1, 2) }
sl@0
   139
  list [nWrite db] [nSync]
sl@0
   140
} {2 4}
sl@0
   141
sl@0
   142
# Set the device-characteristic mask to include the SQLITE_IOCAP_ATOMIC,
sl@0
   143
# then do another INSERT similar to the one in io-2.2. This should
sl@0
   144
# only write 1 page and require a single fsync().
sl@0
   145
# 
sl@0
   146
# The single fsync() is the database file. Only one page is reported as
sl@0
   147
# written because page 1 - the change-counter page - is written using
sl@0
   148
# an out-of-band method that bypasses the write counter.
sl@0
   149
#
sl@0
   150
sqlite3_simulate_device -char atomic
sl@0
   151
do_test io-2.3 {
sl@0
   152
  execsql { INSERT INTO abc VALUES(3, 4) }
sl@0
   153
  list [nWrite db] [nSync]
sl@0
   154
} {1 1}
sl@0
   155
sl@0
   156
# Test that the journal file is not created and the change-counter is
sl@0
   157
# updated when the atomic-write optimization is used.
sl@0
   158
#
sl@0
   159
do_test io-2.4.1 {
sl@0
   160
  execsql {
sl@0
   161
    BEGIN;
sl@0
   162
    INSERT INTO abc VALUES(5, 6);
sl@0
   163
  }
sl@0
   164
  sqlite3 db2 test.db -vfs devsym
sl@0
   165
  execsql { SELECT * FROM abc } db2
sl@0
   166
} {1 2 3 4}
sl@0
   167
do_test io-2.4.2 {
sl@0
   168
  file exists test.db-journal
sl@0
   169
} {0}
sl@0
   170
do_test io-2.4.3 {
sl@0
   171
  execsql { COMMIT }
sl@0
   172
  execsql { SELECT * FROM abc } db2
sl@0
   173
} {1 2 3 4 5 6}
sl@0
   174
db2 close
sl@0
   175
sl@0
   176
# Test that the journal file is created and sync()d if the transaction
sl@0
   177
# modifies more than one database page, even if the IOCAP_ATOMIC flag
sl@0
   178
# is set.
sl@0
   179
#
sl@0
   180
do_test io-2.5.1 {
sl@0
   181
  execsql { CREATE TABLE def(d, e) }
sl@0
   182
  nWrite db ; nSync
sl@0
   183
  execsql {
sl@0
   184
    BEGIN;
sl@0
   185
    INSERT INTO abc VALUES(7, 8);
sl@0
   186
  }
sl@0
   187
  file exists test.db-journal
sl@0
   188
} {0}
sl@0
   189
do_test io-2.5.2 {
sl@0
   190
  execsql { INSERT INTO def VALUES('a', 'b'); }
sl@0
   191
  file exists test.db-journal
sl@0
   192
} {1}
sl@0
   193
do_test io-2.5.3 {
sl@0
   194
  execsql { COMMIT }
sl@0
   195
  list [nWrite db] [nSync]
sl@0
   196
} {3 4}
sl@0
   197
sl@0
   198
# Test that the journal file is created and sync()d if the transaction
sl@0
   199
# modifies a single database page and also appends a page to the file.
sl@0
   200
# Internally, this case is handled differently to the one above. The
sl@0
   201
# journal file is not actually created until the 'COMMIT' statement
sl@0
   202
# is executed.
sl@0
   203
#
sl@0
   204
do_test io-2.6.1 {
sl@0
   205
  execsql {
sl@0
   206
    BEGIN;
sl@0
   207
    INSERT INTO abc VALUES(9, randstr(1000,1000));
sl@0
   208
  }
sl@0
   209
  file exists test.db-journal
sl@0
   210
} {0}
sl@0
   211
do_test io-2.6.2 {
sl@0
   212
  # Create a file at "test.db-journal". This will prevent SQLite from
sl@0
   213
  # opening the journal for exclusive access. As a result, the COMMIT
sl@0
   214
  # should fail with SQLITE_CANTOPEN and the transaction rolled back.
sl@0
   215
  #
sl@0
   216
  set fd [open test.db-journal w]
sl@0
   217
  puts $fd "This is not a journal file"
sl@0
   218
  close $fd
sl@0
   219
  catchsql { COMMIT }
sl@0
   220
} {1 {unable to open database file}}
sl@0
   221
do_test io-2.6.3 {
sl@0
   222
  file delete -force test.db-journal
sl@0
   223
  catchsql { COMMIT }
sl@0
   224
} {1 {cannot commit - no transaction is active}}
sl@0
   225
do_test io-2.6.4 {
sl@0
   226
  execsql { SELECT * FROM abc }
sl@0
   227
} {1 2 3 4 5 6 7 8}
sl@0
   228
sl@0
   229
sl@0
   230
# Test that if the database modification is part of multi-file commit,
sl@0
   231
# the journal file is always created. In this case, the journal file
sl@0
   232
# is created during execution of the COMMIT statement, so we have to
sl@0
   233
# use the same technique to check that it is created as in the above 
sl@0
   234
# block.
sl@0
   235
file delete -force test2.db test2.db-journal
sl@0
   236
ifcapable attach {
sl@0
   237
  do_test io-2.7.1 {
sl@0
   238
    execsql {
sl@0
   239
      ATTACH 'test2.db' AS aux;
sl@0
   240
      PRAGMA aux.page_size = 1024;
sl@0
   241
      CREATE TABLE aux.abc2(a, b);
sl@0
   242
      BEGIN;
sl@0
   243
      INSERT INTO abc VALUES(9, 10);
sl@0
   244
    }
sl@0
   245
    file exists test.db-journal
sl@0
   246
  } {0}
sl@0
   247
  do_test io-2.7.2 {
sl@0
   248
    execsql { INSERT INTO abc2 SELECT * FROM abc }
sl@0
   249
    file exists test2.db-journal
sl@0
   250
  } {0}
sl@0
   251
  do_test io-2.7.3 {
sl@0
   252
    execsql { SELECT * FROM abc UNION ALL SELECT * FROM abc2 }
sl@0
   253
  } {1 2 3 4 5 6 7 8 9 10 1 2 3 4 5 6 7 8 9 10}
sl@0
   254
  do_test io-2.7.4 {
sl@0
   255
    set fd [open test2.db-journal w]
sl@0
   256
    puts $fd "This is not a journal file"
sl@0
   257
    close $fd
sl@0
   258
    catchsql { COMMIT }
sl@0
   259
  } {1 {unable to open database file}}
sl@0
   260
  do_test io-2.7.5 {
sl@0
   261
    file delete -force test2.db-journal
sl@0
   262
    catchsql { COMMIT }
sl@0
   263
  } {1 {cannot commit - no transaction is active}}
sl@0
   264
  do_test io-2.7.6 {
sl@0
   265
    execsql { SELECT * FROM abc UNION ALL SELECT * FROM abc2 }
sl@0
   266
  } {1 2 3 4 5 6 7 8}
sl@0
   267
}
sl@0
   268
sl@0
   269
# Try an explicit ROLLBACK before the journal file is created.
sl@0
   270
#
sl@0
   271
do_test io-2.8.1 {
sl@0
   272
  execsql {
sl@0
   273
    BEGIN;
sl@0
   274
    DELETE FROM abc;
sl@0
   275
  }
sl@0
   276
  file exists test.db-journal
sl@0
   277
} {0}
sl@0
   278
do_test io-2.8.2 {
sl@0
   279
  execsql { SELECT * FROM abc }
sl@0
   280
} {}
sl@0
   281
do_test io-2.8.3 {
sl@0
   282
  execsql {
sl@0
   283
    ROLLBACK;
sl@0
   284
    SELECT * FROM abc;
sl@0
   285
  }
sl@0
   286
} {1 2 3 4 5 6 7 8}
sl@0
   287
sl@0
   288
# Test that the atomic write optimisation is not enabled if the sector
sl@0
   289
# size is larger than the page-size.
sl@0
   290
#
sl@0
   291
do_test io-2.9.1 {
sl@0
   292
  sqlite3_simulate_device -char atomic -sectorsize 2048
sl@0
   293
  execsql {
sl@0
   294
    BEGIN;
sl@0
   295
    INSERT INTO abc VALUES(9, 10);
sl@0
   296
  }
sl@0
   297
  file exists test.db-journal
sl@0
   298
} {1}
sl@0
   299
do_test io-2.9.2 {
sl@0
   300
  execsql { ROLLBACK; }
sl@0
   301
  db close
sl@0
   302
  file delete -force test.db test.db-journal
sl@0
   303
  sqlite3 db test.db -vfs devsym
sl@0
   304
  execsql {
sl@0
   305
    PRAGMA auto_vacuum = OFF;
sl@0
   306
    PRAGMA page_size = 2048;
sl@0
   307
    CREATE TABLE abc(a, b);
sl@0
   308
  }
sl@0
   309
  execsql {
sl@0
   310
    BEGIN;
sl@0
   311
    INSERT INTO abc VALUES(9, 10);
sl@0
   312
  }
sl@0
   313
  file exists test.db-journal
sl@0
   314
} {0}
sl@0
   315
do_test io-2.9.3 {
sl@0
   316
  execsql { COMMIT }
sl@0
   317
} {}
sl@0
   318
sl@0
   319
# Test a couple of the more specific IOCAP_ATOMIC flags 
sl@0
   320
# (i.e IOCAP_ATOMIC2K etc.).
sl@0
   321
#
sl@0
   322
do_test io-2.10.1 {
sl@0
   323
  sqlite3_simulate_device -char atomic1k
sl@0
   324
  execsql {
sl@0
   325
    BEGIN;
sl@0
   326
    INSERT INTO abc VALUES(11, 12);
sl@0
   327
  }
sl@0
   328
  file exists test.db-journal
sl@0
   329
} {1}
sl@0
   330
do_test io-2.10.2 {
sl@0
   331
  execsql { ROLLBACK }
sl@0
   332
  sqlite3_simulate_device -char atomic2k
sl@0
   333
  execsql {
sl@0
   334
    BEGIN;
sl@0
   335
    INSERT INTO abc VALUES(11, 12);
sl@0
   336
  }
sl@0
   337
  file exists test.db-journal
sl@0
   338
} {0}
sl@0
   339
do_test io-2.10.3 {
sl@0
   340
  execsql { ROLLBACK }
sl@0
   341
} {}
sl@0
   342
sl@0
   343
do_test io-2.11.0 {
sl@0
   344
  execsql { 
sl@0
   345
    PRAGMA locking_mode = exclusive;
sl@0
   346
    PRAGMA locking_mode;
sl@0
   347
  }
sl@0
   348
} {exclusive exclusive}
sl@0
   349
do_test io-2.11.1 {
sl@0
   350
  execsql { 
sl@0
   351
    INSERT INTO abc VALUES(11, 12);
sl@0
   352
  }
sl@0
   353
  file exists test.db-journal
sl@0
   354
} {0}
sl@0
   355
sl@0
   356
do_test io-2.11.2 {
sl@0
   357
  execsql { 
sl@0
   358
    PRAGMA locking_mode = normal;
sl@0
   359
    INSERT INTO abc VALUES(13, 14);
sl@0
   360
  }
sl@0
   361
  file exists test.db-journal
sl@0
   362
} {0}
sl@0
   363
sl@0
   364
} ;# /* ifcapable atomicwrite */
sl@0
   365
sl@0
   366
#----------------------------------------------------------------------
sl@0
   367
# Test cases io-3.* test the IOCAP_SEQUENTIAL optimization.
sl@0
   368
#
sl@0
   369
sqlite3_simulate_device -char sequential -sectorsize 0
sl@0
   370
ifcapable pager_pragmas {
sl@0
   371
  do_test io-3.1 {
sl@0
   372
    db close
sl@0
   373
    file delete -force test.db test.db-journal
sl@0
   374
    sqlite3 db test.db -vfs devsym
sl@0
   375
    db eval {
sl@0
   376
      PRAGMA auto_vacuum=OFF;
sl@0
   377
    }
sl@0
   378
    # File size might be 1 due to the hack to work around ticket #3260.
sl@0
   379
    # Search for #3260 in os_unix.c for additional information.
sl@0
   380
    expr {[file size test.db]>1}
sl@0
   381
  } {0}
sl@0
   382
  do_test io-3.2 {
sl@0
   383
    execsql { CREATE TABLE abc(a, b) }
sl@0
   384
    nSync
sl@0
   385
    execsql {
sl@0
   386
      PRAGMA temp_store = memory;
sl@0
   387
      PRAGMA cache_size = 10;
sl@0
   388
      BEGIN;
sl@0
   389
      INSERT INTO abc VALUES('hello', 'world');
sl@0
   390
      INSERT INTO abc SELECT * FROM abc;
sl@0
   391
      INSERT INTO abc SELECT * FROM abc;
sl@0
   392
      INSERT INTO abc SELECT * FROM abc;
sl@0
   393
      INSERT INTO abc SELECT * FROM abc;
sl@0
   394
      INSERT INTO abc SELECT * FROM abc;
sl@0
   395
      INSERT INTO abc SELECT * FROM abc;
sl@0
   396
      INSERT INTO abc SELECT * FROM abc;
sl@0
   397
      INSERT INTO abc SELECT * FROM abc;
sl@0
   398
      INSERT INTO abc SELECT * FROM abc;
sl@0
   399
      INSERT INTO abc SELECT * FROM abc;
sl@0
   400
      INSERT INTO abc SELECT * FROM abc;
sl@0
   401
    }
sl@0
   402
    # File has grown - showing there was a cache-spill - but there 
sl@0
   403
    # have been no calls to fsync(). The file is probably about 30KB.
sl@0
   404
    # But some VFS implementations (symbian) buffer writes so the actual
sl@0
   405
    # size may be a little less than that. So this test case just tests
sl@0
   406
    # that the file is now greater than 20000 bytes in size.
sl@0
   407
    list [expr [file size test.db]>20000] [nSync]
sl@0
   408
  } {1 0}
sl@0
   409
  do_test io-3.3 {
sl@0
   410
    # The COMMIT requires a single fsync() - to the database file.
sl@0
   411
    execsql { COMMIT }
sl@0
   412
    list [file size test.db] [nSync]
sl@0
   413
  } {39936 1}
sl@0
   414
}
sl@0
   415
sl@0
   416
#----------------------------------------------------------------------
sl@0
   417
# Test cases io-4.* test the IOCAP_SAFE_APPEND optimization.
sl@0
   418
#
sl@0
   419
sqlite3_simulate_device -char safe_append
sl@0
   420
sl@0
   421
# With the SAFE_APPEND flag set, simple transactions require 3, rather
sl@0
   422
# than 4, calls to fsync(). The fsync() calls are on:
sl@0
   423
#
sl@0
   424
#   1) The directory in which the journal file is created, (unix only)
sl@0
   425
#   2) The journal file (to sync the page data),
sl@0
   426
#   3) The database file.
sl@0
   427
#
sl@0
   428
# Normally, when the SAFE_APPEND flag is not set, there is another fsync()
sl@0
   429
# on the journal file between steps (2) and (3) above.
sl@0
   430
#
sl@0
   431
if {$::tcl_platform(platform)=="unix"} {
sl@0
   432
  set expected_sync_count 3
sl@0
   433
} else {
sl@0
   434
  set expected_sync_count 2
sl@0
   435
}
sl@0
   436
do_test io-4.1 {
sl@0
   437
  execsql { DELETE FROM abc }
sl@0
   438
  nSync
sl@0
   439
  execsql { INSERT INTO abc VALUES('a', 'b') }
sl@0
   440
  nSync
sl@0
   441
} $expected_sync_count
sl@0
   442
sl@0
   443
# With SAFE_APPEND set, the nRec field of the journal file header should
sl@0
   444
# be set to 0xFFFFFFFF before the first journal sync. The nRec field
sl@0
   445
# occupies bytes 8-11 of the journal file.
sl@0
   446
#
sl@0
   447
do_test io-4.2.1 {
sl@0
   448
  execsql { BEGIN }
sl@0
   449
  execsql { INSERT INTO abc VALUES('c', 'd') }
sl@0
   450
  file exists test.db-journal
sl@0
   451
} {1}
sl@0
   452
if {$::tcl_platform(platform)=="unix"} {
sl@0
   453
  do_test io-4.2.2 {
sl@0
   454
    hexio_read test.db-journal 8 4
sl@0
   455
  } {FFFFFFFF}
sl@0
   456
}
sl@0
   457
do_test io-4.2.3 {
sl@0
   458
  execsql { COMMIT }
sl@0
   459
  nSync
sl@0
   460
} $expected_sync_count
sl@0
   461
sqlite3_simulate_device -char safe_append
sl@0
   462
sl@0
   463
# With SAFE_APPEND set, there should only ever be one journal-header
sl@0
   464
# written to the database, even though the sync-mode is "full".
sl@0
   465
#
sl@0
   466
do_test io-4.3.1 {
sl@0
   467
  execsql {
sl@0
   468
    INSERT INTO abc SELECT * FROM abc;
sl@0
   469
    INSERT INTO abc SELECT * FROM abc;
sl@0
   470
    INSERT INTO abc SELECT * FROM abc;
sl@0
   471
    INSERT INTO abc SELECT * FROM abc;
sl@0
   472
    INSERT INTO abc SELECT * FROM abc;
sl@0
   473
    INSERT INTO abc SELECT * FROM abc;
sl@0
   474
    INSERT INTO abc SELECT * FROM abc;
sl@0
   475
    INSERT INTO abc SELECT * FROM abc;
sl@0
   476
    INSERT INTO abc SELECT * FROM abc;
sl@0
   477
    INSERT INTO abc SELECT * FROM abc;
sl@0
   478
    INSERT INTO abc SELECT * FROM abc;
sl@0
   479
  }
sl@0
   480
  expr {[file size test.db]/1024}
sl@0
   481
} {43}
sl@0
   482
ifcapable pager_pragmas {
sl@0
   483
  do_test io-4.3.2 {
sl@0
   484
    execsql {
sl@0
   485
      PRAGMA synchronous = full;
sl@0
   486
      PRAGMA cache_size = 10;
sl@0
   487
      PRAGMA synchronous;
sl@0
   488
    }
sl@0
   489
  } {2}
sl@0
   490
}
sl@0
   491
do_test io-4.3.3 {
sl@0
   492
  execsql {
sl@0
   493
    BEGIN;
sl@0
   494
    UPDATE abc SET a = 'x';
sl@0
   495
  }
sl@0
   496
  file exists test.db-journal
sl@0
   497
} {1}
sl@0
   498
if {$tcl_platform(platform) != "symbian"} {
sl@0
   499
  # This test is not run on symbian because the file-buffer makes it
sl@0
   500
  # difficult to predict the exact size of the file as reported by 
sl@0
   501
  # [file size].
sl@0
   502
  do_test io-4.3.4 {
sl@0
   503
    # The UPDATE statement in the statement above modifies 41 pages 
sl@0
   504
    # (all pages in the database except page 1 and the root page of 
sl@0
   505
    # abc). Because the cache_size is set to 10, this must have required
sl@0
   506
    # at least 4 cache-spills. If there were no journal headers written
sl@0
   507
    # to the journal file after the cache-spill, then the size of the
sl@0
   508
    # journal file is give by:
sl@0
   509
    #
sl@0
   510
    #    <jrnl file size> = <jrnl header size> + nPage * (<page-size> + 8)
sl@0
   511
    #
sl@0
   512
    # If the journal file contains additional headers, this formula
sl@0
   513
    # will not predict the size of the journal file.
sl@0
   514
    #
sl@0
   515
    file size test.db-journal
sl@0
   516
  } [expr 512 + (1024+8)*41]
sl@0
   517
}
sl@0
   518
sl@0
   519
#----------------------------------------------------------------------
sl@0
   520
# Test cases io-5.* test that the default page size is selected and
sl@0
   521
# used correctly.
sl@0
   522
#
sl@0
   523
set tn 0
sl@0
   524
foreach {char                 sectorsize pgsize} {
sl@0
   525
         {}                     512      1024
sl@0
   526
         {}                    1024      1024
sl@0
   527
         {}                    2048      2048
sl@0
   528
         {}                    8192      8192
sl@0
   529
         {}                   16384      8192
sl@0
   530
         {atomic}               512      8192
sl@0
   531
         {atomic512}            512      1024
sl@0
   532
         {atomic2K}             512      2048
sl@0
   533
         {atomic2K}            4096      4096
sl@0
   534
         {atomic2K atomic}      512      8192
sl@0
   535
         {atomic64K}            512      1024
sl@0
   536
} {
sl@0
   537
  incr tn
sl@0
   538
  if {$pgsize>$::SQLITE_MAX_PAGE_SIZE} continue
sl@0
   539
  db close
sl@0
   540
  file delete -force test.db test.db-journal
sl@0
   541
  sqlite3_simulate_device -char $char -sectorsize $sectorsize
sl@0
   542
  sqlite3 db test.db -vfs devsym
sl@0
   543
  db eval {
sl@0
   544
    PRAGMA auto_vacuum=OFF;
sl@0
   545
  }
sl@0
   546
  ifcapable !atomicwrite {
sl@0
   547
    if {[regexp {^atomic} $char]} continue
sl@0
   548
  }
sl@0
   549
  do_test io-5.$tn {
sl@0
   550
    execsql {
sl@0
   551
      CREATE TABLE abc(a, b, c);
sl@0
   552
    }
sl@0
   553
    expr {[file size test.db]/2}
sl@0
   554
  } $pgsize
sl@0
   555
}
sl@0
   556
sl@0
   557
sqlite3_simulate_device -char {} -sectorsize 0
sl@0
   558
finish_test