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
|