123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643 |
- # 2007 August 21
- #
- # The author disclaims copyright to this source code. In place of
- # a legal notice, here is a blessing:
- #
- # May you do good and not evil.
- # May you find forgiveness for yourself and forgive others.
- # May you share freely, never taking more than you give.
- #
- #***********************************************************************
- #
- # The focus of this file is testing some specific characteristics of the
- # IO traffic generated by SQLite (making sure SQLite is not writing out
- # more database pages than it has to, stuff like that).
- #
- set testdir [file dirname $argv0]
- source $testdir/tester.tcl
- set ::testprefix io
- db close
- sqlite3_simulate_device
- sqlite3 db test.db -vfs devsym
- # Test summary:
- #
- # io-1.* - Test that quick-balance does not journal pages unnecessarily.
- #
- # io-2.* - Test the "atomic-write optimization".
- #
- # io-3.* - Test the IO traffic enhancements triggered when the
- # IOCAP_SEQUENTIAL device capability flag is set (no
- # fsync() calls on the journal file).
- #
- # io-4.* - Test the IO traffic enhancements triggered when the
- # IOCAP_SAFE_APPEND device capability flag is set (fewer
- # fsync() calls on the journal file, no need to set nRec
- # field in the single journal header).
- #
- # io-5.* - Test that the default page size is selected and used
- # correctly.
- #
- # io-6.* - Test that the pager-cache is not being flushed unnecessarily
- # after a transaction that uses the special atomic-write path
- # is committed.
- #
- set ::nWrite 0
- proc nWrite {db} {
- set bt [btree_from_db $db]
- db_enter $db
- array set stats [btree_pager_stats $bt]
- db_leave $db
- set res [expr $stats(write) - $::nWrite]
- set ::nWrite $stats(write)
- set res
- }
- set ::nSync 0
- proc nSync {} {
- set res [expr {$::sqlite_sync_count - $::nSync}]
- set ::nSync $::sqlite_sync_count
- set res
- }
- do_test io-1.1 {
- execsql {
- PRAGMA auto_vacuum = OFF;
- PRAGMA page_size = 1024;
- CREATE TABLE abc(a,b);
- }
- nWrite db
- } {2}
- # Insert into the table 4 records of aproximately 240 bytes each.
- # This should completely fill the root-page of the table. Each
- # INSERT causes 2 db pages to be written - the root-page of "abc"
- # and page 1 (db change-counter page).
- do_test io-1.2 {
- set ret [list]
- execsql { INSERT INTO abc VALUES(1,randstr(230,230)); }
- lappend ret [nWrite db]
- execsql { INSERT INTO abc VALUES(2,randstr(230,230)); }
- lappend ret [nWrite db]
- execsql { INSERT INTO abc VALUES(3,randstr(230,230)); }
- lappend ret [nWrite db]
- execsql { INSERT INTO abc VALUES(4,randstr(230,230)); }
- lappend ret [nWrite db]
- } {2 2 2 2}
- # Insert another 240 byte record. This causes two leaf pages
- # to be added to the root page of abc. 4 pages in total
- # are written to the db file - the two leaf pages, the root
- # of abc and the change-counter page.
- do_test io-1.3 {
- execsql { INSERT INTO abc VALUES(5,randstr(230,230)); }
- nWrite db
- } {4}
- # Insert another 3 240 byte records. After this, the tree consists of
- # the root-node, which is close to empty, and two leaf pages, both of
- # which are full.
- do_test io-1.4 {
- set ret [list]
- execsql { INSERT INTO abc VALUES(6,randstr(230,230)); }
- lappend ret [nWrite db]
- execsql { INSERT INTO abc VALUES(7,randstr(230,230)); }
- lappend ret [nWrite db]
- execsql { INSERT INTO abc VALUES(8,randstr(230,230)); }
- lappend ret [nWrite db]
- } {2 2 2}
- # This insert should use the quick-balance trick to add a third leaf
- # to the b-tree used to store table abc. It should only be necessary to
- # write to 3 pages to do this: the change-counter, the root-page and
- # the new leaf page.
- do_test io-1.5 {
- execsql { INSERT INTO abc VALUES(9,randstr(230,230)); }
- nWrite db
- } {3}
- ifcapable atomicwrite {
- #----------------------------------------------------------------------
- # Test cases io-2.* test the atomic-write optimization.
- #
- do_test io-2.1 {
- execsql { DELETE FROM abc; VACUUM; }
- } {}
- # Clear the write and sync counts.
- nWrite db ; nSync
- # The following INSERT updates 2 pages and requires 4 calls to fsync():
- #
- # 1) The directory in which the journal file is created,
- # 2) The journal file (to sync the page data),
- # 3) The journal file (to sync the journal file header),
- # 4) The database file.
- #
- do_test io-2.2 {
- execsql { INSERT INTO abc VALUES(1, 2) }
- list [nWrite db] [nSync]
- } {2 4}
- # Set the device-characteristic mask to include the SQLITE_IOCAP_ATOMIC,
- # then do another INSERT similar to the one in io-2.2. This should
- # only write 1 page and require a single fsync().
- #
- # The single fsync() is the database file. Only one page is reported as
- # written because page 1 - the change-counter page - is written using
- # an out-of-band method that bypasses the write counter.
- #
- # UPDATE: As of [05f98d4eec] (adding SQLITE_DBSTATUS_CACHE_WRITE), the
- # second write is also counted. So this now reports two writes and a
- # single fsync.
- #
- sqlite3_simulate_device -char atomic
- do_test io-2.3 {
- execsql { INSERT INTO abc VALUES(3, 4) }
- list [nWrite db] [nSync]
- } {2 1}
- # Test that the journal file is not created and the change-counter is
- # updated when the atomic-write optimization is used.
- #
- do_test io-2.4.1 {
- execsql {
- BEGIN;
- INSERT INTO abc VALUES(5, 6);
- }
- sqlite3 db2 test.db -vfs devsym
- execsql { SELECT * FROM abc } db2
- } {1 2 3 4}
- do_test io-2.4.2 {
- file exists test.db-journal
- } {0}
- do_test io-2.4.3 {
- execsql { COMMIT }
- execsql { SELECT * FROM abc } db2
- } {1 2 3 4 5 6}
- db2 close
- # Test that the journal file is created and sync()d if the transaction
- # modifies more than one database page, even if the IOCAP_ATOMIC flag
- # is set.
- #
- do_test io-2.5.1 {
- execsql { CREATE TABLE def(d, e) }
- nWrite db ; nSync
- execsql {
- BEGIN;
- INSERT INTO abc VALUES(7, 8);
- }
- file exists test.db-journal
- } {0}
- do_test io-2.5.2 {
- execsql { INSERT INTO def VALUES('a', 'b'); }
- file exists test.db-journal
- } {1}
- do_test io-2.5.3 {
- execsql { COMMIT }
- list [nWrite db] [nSync]
- } {3 4}
- # Test that the journal file is created and sync()d if the transaction
- # modifies a single database page and also appends a page to the file.
- # Internally, this case is handled differently to the one above. The
- # journal file is not actually created until the 'COMMIT' statement
- # is executed.
- #
- # Changed 2010-03-27: The size of the database is now stored in
- # bytes 28..31 and so when a page is added to the database, page 1
- # is immediately modified and the journal file immediately comes into
- # existence. To fix this test, the BEGIN is changed into a a
- # BEGIN IMMEDIATE and the INSERT is omitted.
- #
- do_test io-2.6.1 {
- execsql {
- BEGIN IMMEDIATE;
- -- INSERT INTO abc VALUES(9, randstr(1000,1000));
- }
- file exists test.db-journal
- } {0}
- do_test io-2.6.2 {
- # Create a file at "test.db-journal". This will prevent SQLite from
- # opening the journal for exclusive access. As a result, the COMMIT
- # should fail with SQLITE_CANTOPEN and the transaction rolled back.
- #
- file mkdir test.db-journal
- catchsql {
- INSERT INTO abc VALUES(9, randstr(1000,1000));
- COMMIT
- }
- } {1 {unable to open database file}}
- do_test io-2.6.3 {
- forcedelete test.db-journal
- catchsql { COMMIT }
- } {0 {}}
- do_test io-2.6.4 {
- execsql { SELECT * FROM abc }
- } {1 2 3 4 5 6 7 8}
- # Test that if the database modification is part of multi-file commit,
- # the journal file is always created. In this case, the journal file
- # is created during execution of the COMMIT statement, so we have to
- # use the same technique to check that it is created as in the above
- # block.
- forcedelete test2.db test2.db-journal
- ifcapable attach {
- do_test io-2.7.1 {
- execsql {
- ATTACH 'test2.db' AS aux;
- PRAGMA aux.page_size = 1024;
- CREATE TABLE aux.abc2(a, b);
- BEGIN;
- INSERT INTO abc VALUES(9, 10);
- }
- file exists test.db-journal
- } {0}
- do_test io-2.7.2 {
- execsql { INSERT INTO abc2 SELECT * FROM abc }
- file exists test2.db-journal
- } {0}
- do_test io-2.7.3 {
- execsql { SELECT * FROM abc UNION ALL SELECT * FROM abc2 }
- } {1 2 3 4 5 6 7 8 9 10 1 2 3 4 5 6 7 8 9 10}
- do_test io-2.7.4 {
- file mkdir test2.db-journal
- catchsql { COMMIT }
- } {1 {unable to open database file}}
- do_test io-2.7.5 {
- forcedelete test2.db-journal
- catchsql { COMMIT }
- } {1 {cannot commit - no transaction is active}}
- do_test io-2.7.6 {
- execsql { SELECT * FROM abc UNION ALL SELECT * FROM abc2 }
- } {1 2 3 4 5 6 7 8}
- }
- # Try an explicit ROLLBACK before the journal file is created.
- #
- do_test io-2.8.1 {
- execsql {
- BEGIN;
- DELETE FROM abc;
- }
- file exists test.db-journal
- } {0}
- do_test io-2.8.2 {
- execsql { SELECT * FROM abc }
- } {}
- do_test io-2.8.3 {
- execsql {
- ROLLBACK;
- SELECT * FROM abc;
- }
- } {1 2 3 4 5 6 7 8}
- # Test that the atomic write optimisation is not enabled if the sector
- # size is larger than the page-size.
- #
- do_test io-2.9.1 {
- db close
- sqlite3 db test.db
- sqlite3_simulate_device -char atomic -sectorsize 2048
- execsql {
- BEGIN;
- INSERT INTO abc VALUES(9, 10);
- }
- file exists test.db-journal
- } {1}
- do_test io-2.9.2 {
- execsql { ROLLBACK; }
- db close
- forcedelete test.db test.db-journal
- sqlite3 db test.db -vfs devsym
- execsql {
- PRAGMA auto_vacuum = OFF;
- PRAGMA page_size = 2048;
- CREATE TABLE abc(a, b);
- }
- execsql {
- BEGIN;
- INSERT INTO abc VALUES(9, 10);
- }
- file exists test.db-journal
- } {0}
- do_test io-2.9.3 {
- execsql { COMMIT }
- } {}
- # Test a couple of the more specific IOCAP_ATOMIC flags
- # (i.e IOCAP_ATOMIC2K etc.).
- #
- do_test io-2.10.1 {
- sqlite3_simulate_device -char atomic1k
- execsql {
- BEGIN;
- INSERT INTO abc VALUES(11, 12);
- }
- file exists test.db-journal
- } {1}
- do_test io-2.10.2 {
- execsql { ROLLBACK }
- sqlite3_simulate_device -char atomic2k
- execsql {
- BEGIN;
- INSERT INTO abc VALUES(11, 12);
- }
- file exists test.db-journal
- } {0}
- do_test io-2.10.3 {
- execsql { ROLLBACK }
- } {}
- do_test io-2.11.0 {
- execsql {
- PRAGMA locking_mode = exclusive;
- PRAGMA locking_mode;
- }
- } {exclusive exclusive}
- do_test io-2.11.1 {
- execsql {
- INSERT INTO abc VALUES(11, 12);
- }
- file exists test.db-journal
- } {0}
- do_test io-2.11.2 {
- execsql {
- PRAGMA locking_mode = normal;
- INSERT INTO abc VALUES(13, 14);
- }
- file exists test.db-journal
- } {0}
- } ;# /* ifcapable atomicwrite */
- #----------------------------------------------------------------------
- # Test cases io-3.* test the IOCAP_SEQUENTIAL optimization.
- #
- sqlite3_simulate_device -char sequential -sectorsize 0
- ifcapable pager_pragmas {
- do_test io-3.1 {
- db close
- forcedelete test.db test.db-journal
- sqlite3 db test.db -vfs devsym
- db eval {
- PRAGMA auto_vacuum=OFF;
- }
- # File size might be 1 due to the hack to work around ticket #3260.
- # Search for #3260 in os_unix.c for additional information.
- expr {[file size test.db]>1}
- } {0}
- do_test io-3.2 {
- execsql { CREATE TABLE abc(a, b) }
- nSync
- execsql {
- PRAGMA temp_store = memory;
- PRAGMA cache_size = 10;
- BEGIN;
- INSERT INTO abc VALUES('hello', 'world');
- INSERT INTO abc SELECT * FROM abc;
- INSERT INTO abc SELECT * FROM abc;
- INSERT INTO abc SELECT * FROM abc;
- INSERT INTO abc SELECT * FROM abc;
- INSERT INTO abc SELECT * FROM abc;
- INSERT INTO abc SELECT * FROM abc;
- INSERT INTO abc SELECT * FROM abc;
- INSERT INTO abc SELECT * FROM abc;
- INSERT INTO abc SELECT * FROM abc;
- INSERT INTO abc SELECT * FROM abc;
- INSERT INTO abc SELECT * FROM abc;
- }
- # File has grown - showing there was a cache-spill - but there
- # have been no calls to fsync(). The file is probably about 30KB.
- # But some VFS implementations (symbian) buffer writes so the actual
- # size may be a little less than that. So this test case just tests
- # that the file is now greater than 20000 bytes in size.
- list [expr [file size test.db]>20000] [nSync]
- } {1 0}
- do_test io-3.3 {
- # The COMMIT requires a single fsync() - to the database file.
- execsql { COMMIT }
- list [file size test.db] [nSync]
- } {39936 1}
- }
- #----------------------------------------------------------------------
- # Test cases io-4.* test the IOCAP_SAFE_APPEND optimization.
- #
- sqlite3_simulate_device -char safe_append
- # With the SAFE_APPEND flag set, simple transactions require 3, rather
- # than 4, calls to fsync(). The fsync() calls are on:
- #
- # 1) The directory in which the journal file is created, (unix only)
- # 2) The journal file (to sync the page data),
- # 3) The database file.
- #
- # Normally, when the SAFE_APPEND flag is not set, there is another fsync()
- # on the journal file between steps (2) and (3) above.
- #
- set expected_sync_count 2
- if {$::tcl_platform(platform)=="unix"} {
- ifcapable dirsync {
- incr expected_sync_count
- }
- }
- do_test io-4.1 {
- execsql { DELETE FROM abc }
- nSync
- execsql { INSERT INTO abc VALUES('a', 'b') }
- nSync
- } $expected_sync_count
- # With SAFE_APPEND set, the nRec field of the journal file header should
- # be set to 0xFFFFFFFF before the first journal sync. The nRec field
- # occupies bytes 8-11 of the journal file.
- #
- do_test io-4.2.1 {
- execsql { BEGIN }
- execsql { INSERT INTO abc VALUES('c', 'd') }
- file exists test.db-journal
- } {1}
- if {$::tcl_platform(platform)=="unix"} {
- do_test io-4.2.2 {
- hexio_read test.db-journal 8 4
- } {FFFFFFFF}
- }
- do_test io-4.2.3 {
- execsql { COMMIT }
- nSync
- } $expected_sync_count
- sqlite3_simulate_device -char safe_append
- # With SAFE_APPEND set, there should only ever be one journal-header
- # written to the database, even though the sync-mode is "full".
- #
- do_test io-4.3.1 {
- execsql {
- INSERT INTO abc SELECT * FROM abc;
- INSERT INTO abc SELECT * FROM abc;
- INSERT INTO abc SELECT * FROM abc;
- INSERT INTO abc SELECT * FROM abc;
- INSERT INTO abc SELECT * FROM abc;
- INSERT INTO abc SELECT * FROM abc;
- INSERT INTO abc SELECT * FROM abc;
- INSERT INTO abc SELECT * FROM abc;
- INSERT INTO abc SELECT * FROM abc;
- INSERT INTO abc SELECT * FROM abc;
- INSERT INTO abc SELECT * FROM abc;
- }
- expr {[file size test.db]/1024}
- } {43}
- ifcapable pager_pragmas {
- do_test io-4.3.2 {
- execsql {
- PRAGMA synchronous = full;
- PRAGMA cache_size = 10;
- PRAGMA synchronous;
- }
- } {2}
- }
- do_test io-4.3.3 {
- execsql {
- BEGIN;
- UPDATE abc SET a = 'x';
- }
- file exists test.db-journal
- } {1}
- if {$tcl_platform(platform) != "symbian"} {
- # This test is not run on symbian because the file-buffer makes it
- # difficult to predict the exact size of the file as reported by
- # [file size].
- do_test io-4.3.4 {
- # The UPDATE statement in the statement above modifies 41 pages
- # (all pages in the database except page 1 and the root page of
- # abc). Because the cache_size is set to 10, this must have required
- # at least 4 cache-spills. If there were no journal headers written
- # to the journal file after the cache-spill, then the size of the
- # journal file is give by:
- #
- # <jrnl file size> = <jrnl header size> + nPage * (<page-size> + 8)
- #
- # If the journal file contains additional headers, this formula
- # will not predict the size of the journal file.
- #
- file size test.db-journal
- } [expr 512 + (1024+8)*41]
- }
- #----------------------------------------------------------------------
- # Test cases io-5.* test that the default page size is selected and
- # used correctly.
- #
- set tn 0
- foreach {char sectorsize pgsize} {
- {} 512 1024
- {} 1024 1024
- {} 2048 2048
- {} 8192 8192
- {} 16384 8192
- {atomic} 512 8192
- {atomic512} 512 1024
- {atomic2K} 512 2048
- {atomic2K} 4096 4096
- {atomic2K atomic} 512 8192
- {atomic64K} 512 1024
- } {
- incr tn
- if {$pgsize>$::SQLITE_MAX_PAGE_SIZE} continue
- db close
- forcedelete test.db test.db-journal
- sqlite3_simulate_device -char $char -sectorsize $sectorsize
- sqlite3 db test.db -vfs devsym
- db eval {
- PRAGMA auto_vacuum=OFF;
- }
- ifcapable !atomicwrite {
- if {[regexp {^atomic} $char]} continue
- }
- do_test io-5.$tn {
- execsql {
- CREATE TABLE abc(a, b, c);
- }
- expr {[file size test.db]/2}
- } $pgsize
- }
- #----------------------------------------------------------------------
- #
- do_test io-6.1 {
- db close
- sqlite3_simulate_device -char atomic
- forcedelete test.db
- sqlite3 db test.db -vfs devsym
- execsql {
- PRAGMA mmap_size = 0;
- PRAGMA page_size = 1024;
- PRAGMA cache_size = 2000;
- CREATE TABLE t1(x);
- CREATE TABLE t2(x);
- CREATE TABLE t3(x);
- CREATE INDEX i3 ON t3(x);
- INSERT INTO t3 VALUES(randomblob(100));
- INSERT INTO t3 SELECT randomblob(100) FROM t3;
- INSERT INTO t3 SELECT randomblob(100) FROM t3;
- INSERT INTO t3 SELECT randomblob(100) FROM t3;
- INSERT INTO t3 SELECT randomblob(100) FROM t3;
- INSERT INTO t3 SELECT randomblob(100) FROM t3;
- INSERT INTO t3 SELECT randomblob(100) FROM t3;
- INSERT INTO t3 SELECT randomblob(100) FROM t3;
- INSERT INTO t3 SELECT randomblob(100) FROM t3;
- INSERT INTO t3 SELECT randomblob(100) FROM t3;
- INSERT INTO t3 SELECT randomblob(100) FROM t3;
- INSERT INTO t3 SELECT randomblob(100) FROM t3;
- }
- db_save_and_close
- } {}
- foreach {tn sql} {
- 1 { BEGIN;
- INSERT INTO t1 VALUES('123');
- INSERT INTO t2 VALUES('456');
- COMMIT;
- }
- 2 { BEGIN;
- INSERT INTO t1 VALUES('123');
- COMMIT;
- }
- } {
- # These tests don't work with memsubsys1, as it causes the effective page
- # cache size to become too small to hold the entire db in memory.
- if {[permutation] == "memsubsys1"} continue
- db_restore
- sqlite3 db test.db -vfs devsym
- execsql {
- PRAGMA cache_size = 2000;
- PRAGMA mmap_size = 0;
- SELECT x FROM t3 ORDER BY rowid;
- SELECT x FROM t3 ORDER BY x;
- }
- do_execsql_test 6.2.$tn.1 { PRAGMA integrity_check } {ok}
- do_execsql_test 6.2.$tn.2 $sql
- # Corrupt the database file on disk. This should not matter for the
- # purposes of the following "PRAGMA integrity_check", as the entire
- # database should be cached in the pager-cache. If corruption is
- # reported, it indicates that executing $sql caused the pager cache
- # to be flushed. Which is a bug.
- hexio_write test.db [expr 1024 * 5] [string repeat 00 2048]
- do_execsql_test 6.2.$tn.3 { PRAGMA integrity_check } {ok}
- db close
- }
- sqlite3_simulate_device -char {} -sectorsize 0
- finish_test
|