1 # 2010 May 24 2 # 3 # The author disclaims copyright to this source code. In place of 4 # a legal notice, here is a blessing: 5 # 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. 9 # 10 #*********************************************************************** 11 # 12 13 set testdir [file dirname $argv0] 14 source $testdir/tester.tcl 15 source $testdir/lock_common.tcl 16 source $testdir/wal_common.tcl 17 18 ifcapable !wal {finish_test ; return } 19 20 # Read and return the contents of file $filename. Treat the content as 21 # binary data. 22 # 23 proc readfile {filename} { 24 set fd [open $filename] 25 fconfigure $fd -encoding binary 26 fconfigure $fd -translation binary 27 set data [read $fd] 28 close $fd 29 return $data 30 } 31 32 # 33 # File $filename must be a WAL file on disk. Check that the checksum of frame 34 # $iFrame in the file is correct when interpreting data as $endian-endian 35 # integers ($endian must be either "big" or "little"). If the checksum looks 36 # correct, return 1. Otherwise 0. 37 # 38 proc log_checksum_verify {filename iFrame endian} { 39 set data [readfile $filename] 40 41 foreach {offset c1 c2} [log_checksum_calc $data $iFrame $endian] {} 42 43 binary scan [string range $data $offset [expr $offset+7]] II expect1 expect2 44 set expect1 [expr $expect1&0xFFFFFFFF] 45 set expect2 [expr $expect2&0xFFFFFFFF] 46 47 expr {$c1==$expect1 && $c2==$expect2} 48 } 49 50 # File $filename must be a WAL file on disk. Compute the checksum for frame 51 # $iFrame in the file by interpreting data as $endian-endian integers 52 # ($endian must be either "big" or "little"). Then write the computed 53 # checksum into the file. 54 # 55 proc log_checksum_write {filename iFrame endian} { 56 set data [readfile $filename] 57 58 foreach {offset c1 c2} [log_checksum_calc $data $iFrame $endian] {} 59 60 set bin [binary format II $c1 $c2] 61 set fd [open $filename r+] 62 fconfigure $fd -encoding binary 63 fconfigure $fd -translation binary 64 seek $fd $offset 65 puts -nonewline $fd $bin 66 close $fd 67 } 68 69 # Calculate and return the checksum for a particular frame in a WAL. 70 # 71 # Arguments are: 72 # 73 # $data Blob containing the entire contents of a WAL. 74 # 75 # $iFrame Frame number within the $data WAL. Frames are numbered 76 # starting at 1. 77 # 78 # $endian One of "big" or "little". 79 # 80 # Returns a list of three elements, as follows: 81 # 82 # * The byte offset of the checksum belonging to frame $iFrame in the WAL. 83 # * The first integer in the calculated version of the checksum. 84 # * The second integer in the calculated version of the checksum. 85 # 86 proc log_checksum_calc {data iFrame endian} { 87 88 binary scan [string range $data 8 11] I pgsz 89 if {$iFrame > 1} { 90 set n [wal_file_size [expr $iFrame-2] $pgsz] 91 binary scan [string range $data [expr $n+16] [expr $n+23]] II c1 c2 92 } else { 93 set c1 0 94 set c2 0 95 wal_cksum $endian c1 c2 [string range $data 0 23] 96 } 97 98 set n [wal_file_size [expr $iFrame-1] $pgsz] 99 wal_cksum $endian c1 c2 [string range $data $n [expr $n+7]] 100 wal_cksum $endian c1 c2 [string range $data [expr $n+24] [expr $n+24+$pgsz-1]] 101 102 list [expr $n+16] $c1 $c2 103 } 104 105 # 106 # File $filename must be a WAL file on disk. Set the 'magic' field of the 107 # WAL header to indicate that checksums are $endian-endian ($endian must be 108 # either "big" or "little"). 109 # 110 # Also update the wal header checksum (since the wal header contents may 111 # have changed). 112 # 113 proc log_checksum_writemagic {filename endian} { 114 set val [expr {0x377f0682 | ($endian == "big" ? 1 : 0)}] 115 set bin [binary format I $val] 116 set fd [open $filename r+] 117 fconfigure $fd -encoding binary 118 fconfigure $fd -translation binary 119 puts -nonewline $fd $bin 120 121 seek $fd 0 122 set blob [read $fd 24] 123 set c1 0 124 set c2 0 125 wal_cksum $endian c1 c2 $blob 126 seek $fd 24 127 puts -nonewline $fd [binary format II $c1 $c2] 128 129 close $fd 130 } 131 132 #------------------------------------------------------------------------- 133 # Test cases walcksum-1.* attempt to verify the following: 134 # 135 # * That both native and non-native order checksum log files can 136 # be recovered. 137 # 138 # * That when appending to native or non-native checksum log files 139 # SQLite continues to use the right kind of checksums. 140 # 141 # * Test point 2 when the appending process is not one that recovered 142 # the log file. 143 # 144 # * Test that both native and non-native checksum log files can be 145 # checkpointed. And that after doing so the next write to the log 146 # file occurs using native byte-order checksums. 147 # 148 set native "big" 149 if {$::tcl_platform(byteOrder) == "littleEndian"} { set native "little" } 150 foreach endian {big little} { 151 152 # Create a database. Leave some data in the log file. 153 # 154 do_test walcksum-1.$endian.1 { 155 catch { db close } 156 file delete -force test.db test.db-wal test.db-journal 157 sqlite3 db test.db 158 execsql { 159 PRAGMA page_size = 1024; 160 PRAGMA auto_vacuum = 0; 161 PRAGMA synchronous = NORMAL; 162 163 CREATE TABLE t1(a PRIMARY KEY, b); 164 INSERT INTO t1 VALUES(1, 'one'); 165 INSERT INTO t1 VALUES(2, 'two'); 166 INSERT INTO t1 VALUES(3, 'three'); 167 INSERT INTO t1 VALUES(5, 'five'); 168 169 PRAGMA journal_mode = WAL; 170 INSERT INTO t1 VALUES(8, 'eight'); 171 INSERT INTO t1 VALUES(13, 'thirteen'); 172 INSERT INTO t1 VALUES(21, 'twentyone'); 173 } 174 175 file copy -force test.db test2.db 176 file copy -force test.db-wal test2.db-wal 177 db close 178 179 list [file size test2.db] [file size test2.db-wal] 180 } [list [expr 1024*3] [wal_file_size 6 1024]] 181 182 # Verify that the checksums are valid for all frames and that they 183 # are calculated by interpreting data in native byte-order. 184 # 185 for {set f 1} {$f <= 6} {incr f} { 186 do_test walcksum-1.$endian.2.$f { 187 log_checksum_verify test2.db-wal $f $native 188 } 1 189 } 190 191 # Replace all checksums in the current WAL file with $endian versions. 192 # Then check that it is still possible to recover and read the database. 193 # 194 log_checksum_writemagic test2.db-wal $endian 195 for {set f 1} {$f <= 6} {incr f} { 196 do_test walcksum-1.$endian.3.$f { 197 log_checksum_write test2.db-wal $f $endian 198 log_checksum_verify test2.db-wal $f $endian 199 } {1} 200 } 201 do_test walcksum-1.$endian.4.1 { 202 file copy -force test2.db test.db 203 file copy -force test2.db-wal test.db-wal 204 sqlite3 db test.db 205 execsql { SELECT a FROM t1 } 206 } {1 2 3 5 8 13 21} 207 208 # Following recovery, any frames written to the log should use the same 209 # endianness as the existing frames. Check that this is the case. 210 # 211 do_test walcksum-1.$endian.5.0 { 212 execsql { 213 PRAGMA synchronous = NORMAL; 214 INSERT INTO t1 VALUES(34, 'thirtyfour'); 215 } 216 list [file size test.db] [file size test.db-wal] 217 } [list [expr 1024*3] [wal_file_size 8 1024]] 218 for {set f 1} {$f <= 8} {incr f} { 219 do_test walcksum-1.$endian.5.$f { 220 log_checksum_verify test.db-wal $f $endian 221 } {1} 222 } 223 224 # Now connect a second connection to the database. Check that this one 225 # (not the one that did recovery) also appends frames to the log using 226 # the same endianness for checksums as the existing frames. 227 # 228 do_test walcksum-1.$endian.6 { 229 sqlite3 db2 test.db 230 execsql { 231 PRAGMA integrity_check; 232 SELECT a FROM t1; 233 } db2 234 } {ok 1 2 3 5 8 13 21 34} 235 do_test walcksum-1.$endian.7.0 { 236 execsql { 237 PRAGMA synchronous = NORMAL; 238 INSERT INTO t1 VALUES(55, 'fiftyfive'); 239 } db2 240 list [file size test.db] [file size test.db-wal] 241 } [list [expr 1024*3] [wal_file_size 10 1024]] 242 for {set f 1} {$f <= 10} {incr f} { 243 do_test walcksum-1.$endian.7.$f { 244 log_checksum_verify test.db-wal $f $endian 245 } {1} 246 } 247 248 # Now that both the recoverer and non-recoverer have added frames to the 249 # log file, check that it can still be recovered. 250 # 251 file copy -force test.db test2.db 252 file copy -force test.db-wal test2.db-wal 253 do_test walcksum-1.$endian.7.11 { 254 sqlite3 db3 test2.db 255 execsql { 256 PRAGMA integrity_check; 257 SELECT a FROM t1; 258 } db3 259 } {ok 1 2 3 5 8 13 21 34 55} 260 db3 close 261 262 # Run a checkpoint on the database file. Then, check that any frames written 263 # to the start of the log use native byte-order checksums. 264 # 265 do_test walcksum-1.$endian.8.1 { 266 execsql { 267 PRAGMA wal_checkpoint; 268 INSERT INTO t1 VALUES(89, 'eightynine'); 269 } 270 log_checksum_verify test.db-wal 1 $native 271 } {1} 272 do_test walcksum-1.$endian.8.2 { 273 log_checksum_verify test.db-wal 2 $native 274 } {1} 275 do_test walcksum-1.$endian.8.3 { 276 log_checksum_verify test.db-wal 3 $native 277 } {0} 278 279 do_test walcksum-1.$endian.9 { 280 execsql { 281 PRAGMA integrity_check; 282 SELECT a FROM t1; 283 } db2 284 } {ok 1 2 3 5 8 13 21 34 55 89} 285 286 catch { db close } 287 catch { db2 close } 288 } 289 290 #------------------------------------------------------------------------- 291 # Test case walcksum-2.* tests that if a statement transaction is rolled 292 # back after frames are written to the WAL, and then (after writing some 293 # more) the outer transaction is committed, the WAL file is still correctly 294 # formatted (and can be recovered by a second process if required). 295 # 296 do_test walcksum-2.1 { 297 file delete -force test.db test.db-wal test.db-journal 298 sqlite3 db test.db 299 execsql { 300 PRAGMA synchronous = NORMAL; 301 PRAGMA page_size = 1024; 302 PRAGMA journal_mode = WAL; 303 PRAGMA cache_size = 10; 304 CREATE TABLE t1(x PRIMARY KEY); 305 PRAGMA wal_checkpoint; 306 INSERT INTO t1 VALUES(randomblob(800)); 307 BEGIN; 308 INSERT INTO t1 SELECT randomblob(800) FROM t1; /* 2 */ 309 INSERT INTO t1 SELECT randomblob(800) FROM t1; /* 4 */ 310 INSERT INTO t1 SELECT randomblob(800) FROM t1; /* 8 */ 311 INSERT INTO t1 SELECT randomblob(800) FROM t1; /* 16 */ 312 SAVEPOINT one; 313 INSERT INTO t1 SELECT randomblob(800) FROM t1; /* 32 */ 314 INSERT INTO t1 SELECT randomblob(800) FROM t1; /* 64 */ 315 INSERT INTO t1 SELECT randomblob(800) FROM t1; /* 128 */ 316 INSERT INTO t1 SELECT randomblob(800) FROM t1; /* 256 */ 317 ROLLBACK TO one; 318 INSERT INTO t1 SELECT randomblob(800) FROM t1; /* 32 */ 319 INSERT INTO t1 SELECT randomblob(800) FROM t1; /* 64 */ 320 INSERT INTO t1 SELECT randomblob(800) FROM t1; /* 128 */ 321 INSERT INTO t1 SELECT randomblob(800) FROM t1; /* 256 */ 322 COMMIT; 323 } 324 325 file copy -force test.db test2.db 326 file copy -force test.db-wal test2.db-wal 327 328 sqlite3 db2 test2.db 329 execsql { 330 PRAGMA integrity_check; 331 SELECT count(*) FROM t1; 332 } db2 333 } {ok 256} 334 catch { db close } 335 catch { db2 close } 336 337 #------------------------------------------------------------------------- 338 # Test case walcksum-3.* tests that the checksum calculation detects single 339 # byte changes to frame or frame-header data and considers the frame 340 # invalid as a result. 341 # 342 do_test walcksum-3.1 { 343 file delete -force test.db test.db-wal test.db-journal 344 sqlite3 db test.db 345 346 execsql { 347 PRAGMA synchronous = NORMAL; 348 PRAGMA page_size = 1024; 349 CREATE TABLE t1(a, b); 350 INSERT INTO t1 VALUES(1, randomblob(300)); 351 INSERT INTO t1 VALUES(2, randomblob(300)); 352 PRAGMA journal_mode = WAL; 353 INSERT INTO t1 VALUES(3, randomblob(300)); 354 } 355 356 file size test.db-wal 357 } [wal_file_size 1 1024] 358 do_test walcksum-3.2 { 359 file copy -force test.db-wal test2.db-wal 360 file copy -force test.db test2.db 361 sqlite3 db2 test2.db 362 execsql { SELECT a FROM t1 } db2 363 } {1 2 3} 364 db2 close 365 file copy -force test.db test2.db 366 367 368 foreach incr {1 2 3 20 40 60 80 100 120 140 160 180 200 220 240 253 254 255} { 369 do_test walcksum-3.3.$incr { 370 set FAIL 0 371 for {set iOff 0} {$iOff < [wal_file_size 1 1024]} {incr iOff} { 372 373 file copy -force test.db-wal test2.db-wal 374 set fd [open test2.db-wal r+] 375 fconfigure $fd -encoding binary 376 fconfigure $fd -translation binary 377 378 seek $fd $iOff 379 binary scan [read $fd 1] c x 380 seek $fd $iOff 381 puts -nonewline $fd [binary format c [expr {($x+$incr)&0xFF}]] 382 close $fd 383 384 sqlite3 db2 test2.db 385 if { [execsql { SELECT a FROM t1 } db2] != "1 2" } {set FAIL 1} 386 db2 close 387 } 388 set FAIL 389 } {0} 390 } 391 392 finish_test 393 394