First public contribution.
3 # The author disclaims copyright to this source code. In place of
4 # a legal notice, here is a blessing:
6 # May you do good and not evil.
7 # May you find forgiveness for yourself and forgive others.
8 # May you share freely, never taking more than you give.
10 #***********************************************************************
12 # The focus of this file is testing some specific characteristics of the
13 # IO traffic generated by SQLite (making sure SQLite is not writing out
14 # more database pages than it has to, stuff like that).
16 # $Id: io.test,v 1.19 2008/09/18 11:18:41 danielk1977 Exp $
18 set testdir [file dirname $argv0]
19 source $testdir/tester.tcl
22 sqlite3_simulate_device
23 sqlite3 db test.db -vfs devsym
27 # io-1.* - Test that quick-balance does not journal pages unnecessarily.
29 # io-2.* - Test the "atomic-write optimization".
31 # io-3.* - Test the IO traffic enhancements triggered when the
32 # IOCAP_SEQUENTIAL device capability flag is set (no
33 # fsync() calls on the journal file).
35 # io-4.* - Test the IO traffic enhancements triggered when the
36 # IOCAP_SAFE_APPEND device capability flag is set (fewer
37 # fsync() calls on the journal file, no need to set nRec
38 # field in the single journal header).
40 # io-5.* - Test that the default page size is selected and used
46 set bt [btree_from_db $db]
48 array set stats [btree_pager_stats $bt]
50 set res [expr $stats(write) - $::nWrite]
51 set ::nWrite $stats(write)
57 set res [expr {$::sqlite_sync_count - $::nSync}]
58 set ::nSync $::sqlite_sync_count
64 PRAGMA auto_vacuum = OFF;
65 PRAGMA page_size = 1024;
66 CREATE TABLE abc(a,b);
71 # Insert into the table 4 records of aproximately 240 bytes each.
72 # This should completely fill the root-page of the table. Each
73 # INSERT causes 2 db pages to be written - the root-page of "abc"
74 # and page 1 (db change-counter page).
77 execsql { INSERT INTO abc VALUES(1,randstr(230,230)); }
78 lappend ret [nWrite db]
79 execsql { INSERT INTO abc VALUES(2,randstr(230,230)); }
80 lappend ret [nWrite db]
81 execsql { INSERT INTO abc VALUES(3,randstr(230,230)); }
82 lappend ret [nWrite db]
83 execsql { INSERT INTO abc VALUES(4,randstr(230,230)); }
84 lappend ret [nWrite db]
87 # Insert another 240 byte record. This causes two leaf pages
88 # to be added to the root page of abc. 4 pages in total
89 # are written to the db file - the two leaf pages, the root
90 # of abc and the change-counter page.
92 execsql { INSERT INTO abc VALUES(5,randstr(230,230)); }
96 # Insert another 3 240 byte records. After this, the tree consists of
97 # the root-node, which is close to empty, and two leaf pages, both of
101 execsql { INSERT INTO abc VALUES(6,randstr(230,230)); }
102 lappend ret [nWrite db]
103 execsql { INSERT INTO abc VALUES(7,randstr(230,230)); }
104 lappend ret [nWrite db]
105 execsql { INSERT INTO abc VALUES(8,randstr(230,230)); }
106 lappend ret [nWrite db]
109 # This insert should use the quick-balance trick to add a third leaf
110 # to the b-tree used to store table abc. It should only be necessary to
111 # write to 3 pages to do this: the change-counter, the root-page and
114 execsql { INSERT INTO abc VALUES(9,randstr(230,230)); }
118 ifcapable atomicwrite {
120 #----------------------------------------------------------------------
121 # Test cases io-2.* test the atomic-write optimization.
124 execsql { DELETE FROM abc; VACUUM; }
127 # Clear the write and sync counts.
130 # The following INSERT updates 2 pages and requires 4 calls to fsync():
132 # 1) The directory in which the journal file is created,
133 # 2) The journal file (to sync the page data),
134 # 3) The journal file (to sync the journal file header),
135 # 4) The database file.
138 execsql { INSERT INTO abc VALUES(1, 2) }
139 list [nWrite db] [nSync]
142 # Set the device-characteristic mask to include the SQLITE_IOCAP_ATOMIC,
143 # then do another INSERT similar to the one in io-2.2. This should
144 # only write 1 page and require a single fsync().
146 # The single fsync() is the database file. Only one page is reported as
147 # written because page 1 - the change-counter page - is written using
148 # an out-of-band method that bypasses the write counter.
150 sqlite3_simulate_device -char atomic
152 execsql { INSERT INTO abc VALUES(3, 4) }
153 list [nWrite db] [nSync]
156 # Test that the journal file is not created and the change-counter is
157 # updated when the atomic-write optimization is used.
162 INSERT INTO abc VALUES(5, 6);
164 sqlite3 db2 test.db -vfs devsym
165 execsql { SELECT * FROM abc } db2
168 file exists test.db-journal
172 execsql { SELECT * FROM abc } db2
176 # Test that the journal file is created and sync()d if the transaction
177 # modifies more than one database page, even if the IOCAP_ATOMIC flag
181 execsql { CREATE TABLE def(d, e) }
185 INSERT INTO abc VALUES(7, 8);
187 file exists test.db-journal
190 execsql { INSERT INTO def VALUES('a', 'b'); }
191 file exists test.db-journal
195 list [nWrite db] [nSync]
198 # Test that the journal file is created and sync()d if the transaction
199 # modifies a single database page and also appends a page to the file.
200 # Internally, this case is handled differently to the one above. The
201 # journal file is not actually created until the 'COMMIT' statement
207 INSERT INTO abc VALUES(9, randstr(1000,1000));
209 file exists test.db-journal
212 # Create a file at "test.db-journal". This will prevent SQLite from
213 # opening the journal for exclusive access. As a result, the COMMIT
214 # should fail with SQLITE_CANTOPEN and the transaction rolled back.
216 set fd [open test.db-journal w]
217 puts $fd "This is not a journal file"
220 } {1 {unable to open database file}}
222 file delete -force test.db-journal
224 } {1 {cannot commit - no transaction is active}}
226 execsql { SELECT * FROM abc }
230 # Test that if the database modification is part of multi-file commit,
231 # the journal file is always created. In this case, the journal file
232 # is created during execution of the COMMIT statement, so we have to
233 # use the same technique to check that it is created as in the above
235 file delete -force test2.db test2.db-journal
239 ATTACH 'test2.db' AS aux;
240 PRAGMA aux.page_size = 1024;
241 CREATE TABLE aux.abc2(a, b);
243 INSERT INTO abc VALUES(9, 10);
245 file exists test.db-journal
248 execsql { INSERT INTO abc2 SELECT * FROM abc }
249 file exists test2.db-journal
252 execsql { SELECT * FROM abc UNION ALL SELECT * FROM abc2 }
253 } {1 2 3 4 5 6 7 8 9 10 1 2 3 4 5 6 7 8 9 10}
255 set fd [open test2.db-journal w]
256 puts $fd "This is not a journal file"
259 } {1 {unable to open database file}}
261 file delete -force test2.db-journal
263 } {1 {cannot commit - no transaction is active}}
265 execsql { SELECT * FROM abc UNION ALL SELECT * FROM abc2 }
269 # Try an explicit ROLLBACK before the journal file is created.
276 file exists test.db-journal
279 execsql { SELECT * FROM abc }
288 # Test that the atomic write optimisation is not enabled if the sector
289 # size is larger than the page-size.
292 sqlite3_simulate_device -char atomic -sectorsize 2048
295 INSERT INTO abc VALUES(9, 10);
297 file exists test.db-journal
300 execsql { ROLLBACK; }
302 file delete -force test.db test.db-journal
303 sqlite3 db test.db -vfs devsym
305 PRAGMA auto_vacuum = OFF;
306 PRAGMA page_size = 2048;
307 CREATE TABLE abc(a, b);
311 INSERT INTO abc VALUES(9, 10);
313 file exists test.db-journal
319 # Test a couple of the more specific IOCAP_ATOMIC flags
320 # (i.e IOCAP_ATOMIC2K etc.).
323 sqlite3_simulate_device -char atomic1k
326 INSERT INTO abc VALUES(11, 12);
328 file exists test.db-journal
332 sqlite3_simulate_device -char atomic2k
335 INSERT INTO abc VALUES(11, 12);
337 file exists test.db-journal
345 PRAGMA locking_mode = exclusive;
348 } {exclusive exclusive}
351 INSERT INTO abc VALUES(11, 12);
353 file exists test.db-journal
358 PRAGMA locking_mode = normal;
359 INSERT INTO abc VALUES(13, 14);
361 file exists test.db-journal
364 } ;# /* ifcapable atomicwrite */
366 #----------------------------------------------------------------------
367 # Test cases io-3.* test the IOCAP_SEQUENTIAL optimization.
369 sqlite3_simulate_device -char sequential -sectorsize 0
370 ifcapable pager_pragmas {
373 file delete -force test.db test.db-journal
374 sqlite3 db test.db -vfs devsym
376 PRAGMA auto_vacuum=OFF;
378 # File size might be 1 due to the hack to work around ticket #3260.
379 # Search for #3260 in os_unix.c for additional information.
380 expr {[file size test.db]>1}
383 execsql { CREATE TABLE abc(a, b) }
386 PRAGMA temp_store = memory;
387 PRAGMA cache_size = 10;
389 INSERT INTO abc VALUES('hello', 'world');
390 INSERT INTO abc SELECT * FROM abc;
391 INSERT INTO abc SELECT * FROM abc;
392 INSERT INTO abc SELECT * FROM abc;
393 INSERT INTO abc SELECT * FROM abc;
394 INSERT INTO abc SELECT * FROM abc;
395 INSERT INTO abc SELECT * FROM abc;
396 INSERT INTO abc SELECT * FROM abc;
397 INSERT INTO abc SELECT * FROM abc;
398 INSERT INTO abc SELECT * FROM abc;
399 INSERT INTO abc SELECT * FROM abc;
400 INSERT INTO abc SELECT * FROM abc;
402 # File has grown - showing there was a cache-spill - but there
403 # have been no calls to fsync(). The file is probably about 30KB.
404 # But some VFS implementations (symbian) buffer writes so the actual
405 # size may be a little less than that. So this test case just tests
406 # that the file is now greater than 20000 bytes in size.
407 list [expr [file size test.db]>20000] [nSync]
410 # The COMMIT requires a single fsync() - to the database file.
412 list [file size test.db] [nSync]
416 #----------------------------------------------------------------------
417 # Test cases io-4.* test the IOCAP_SAFE_APPEND optimization.
419 sqlite3_simulate_device -char safe_append
421 # With the SAFE_APPEND flag set, simple transactions require 3, rather
422 # than 4, calls to fsync(). The fsync() calls are on:
424 # 1) The directory in which the journal file is created, (unix only)
425 # 2) The journal file (to sync the page data),
426 # 3) The database file.
428 # Normally, when the SAFE_APPEND flag is not set, there is another fsync()
429 # on the journal file between steps (2) and (3) above.
431 if {$::tcl_platform(platform)=="unix"} {
432 set expected_sync_count 3
434 set expected_sync_count 2
437 execsql { DELETE FROM abc }
439 execsql { INSERT INTO abc VALUES('a', 'b') }
441 } $expected_sync_count
443 # With SAFE_APPEND set, the nRec field of the journal file header should
444 # be set to 0xFFFFFFFF before the first journal sync. The nRec field
445 # occupies bytes 8-11 of the journal file.
449 execsql { INSERT INTO abc VALUES('c', 'd') }
450 file exists test.db-journal
452 if {$::tcl_platform(platform)=="unix"} {
454 hexio_read test.db-journal 8 4
460 } $expected_sync_count
461 sqlite3_simulate_device -char safe_append
463 # With SAFE_APPEND set, there should only ever be one journal-header
464 # written to the database, even though the sync-mode is "full".
468 INSERT INTO abc SELECT * FROM abc;
469 INSERT INTO abc SELECT * FROM abc;
470 INSERT INTO abc SELECT * FROM abc;
471 INSERT INTO abc SELECT * FROM abc;
472 INSERT INTO abc SELECT * FROM abc;
473 INSERT INTO abc SELECT * FROM abc;
474 INSERT INTO abc SELECT * FROM abc;
475 INSERT INTO abc SELECT * FROM abc;
476 INSERT INTO abc SELECT * FROM abc;
477 INSERT INTO abc SELECT * FROM abc;
478 INSERT INTO abc SELECT * FROM abc;
480 expr {[file size test.db]/1024}
482 ifcapable pager_pragmas {
485 PRAGMA synchronous = full;
486 PRAGMA cache_size = 10;
494 UPDATE abc SET a = 'x';
496 file exists test.db-journal
498 if {$tcl_platform(platform) != "symbian"} {
499 # This test is not run on symbian because the file-buffer makes it
500 # difficult to predict the exact size of the file as reported by
503 # The UPDATE statement in the statement above modifies 41 pages
504 # (all pages in the database except page 1 and the root page of
505 # abc). Because the cache_size is set to 10, this must have required
506 # at least 4 cache-spills. If there were no journal headers written
507 # to the journal file after the cache-spill, then the size of the
508 # journal file is give by:
510 # <jrnl file size> = <jrnl header size> + nPage * (<page-size> + 8)
512 # If the journal file contains additional headers, this formula
513 # will not predict the size of the journal file.
515 file size test.db-journal
516 } [expr 512 + (1024+8)*41]
519 #----------------------------------------------------------------------
520 # Test cases io-5.* test that the default page size is selected and
524 foreach {char sectorsize pgsize} {
534 {atomic2K atomic} 512 8192
538 if {$pgsize>$::SQLITE_MAX_PAGE_SIZE} continue
540 file delete -force test.db test.db-journal
541 sqlite3_simulate_device -char $char -sectorsize $sectorsize
542 sqlite3 db test.db -vfs devsym
544 PRAGMA auto_vacuum=OFF;
546 ifcapable !atomicwrite {
547 if {[regexp {^atomic} $char]} continue
551 CREATE TABLE abc(a, b, c);
553 expr {[file size test.db]/2}
557 sqlite3_simulate_device -char {} -sectorsize 0