/ Hex Artifact Content
Login

Artifact 4b413718ab3ef5e4f9f1ab0502b89707c5a9ccdc:


0000: 2f 2a 0a 2a 2a 20 32 30 31 30 2d 30 37 2d 32 32  /*.** 2010-07-22
0010: 0a 2a 2a 0a 2a 2a 20 54 68 65 20 61 75 74 68 6f  .**.** The autho
0020: 72 20 64 69 73 63 6c 61 69 6d 73 20 63 6f 70 79  r disclaims copy
0030: 72 69 67 68 74 20 74 6f 20 74 68 69 73 20 73 6f  right to this so
0040: 75 72 63 65 20 63 6f 64 65 2e 20 20 49 6e 20 70  urce code.  In p
0050: 6c 61 63 65 20 6f 66 0a 2a 2a 20 61 20 6c 65 67  lace of.** a leg
0060: 61 6c 20 6e 6f 74 69 63 65 2c 20 68 65 72 65 20  al notice, here 
0070: 69 73 20 61 20 62 6c 65 73 73 69 6e 67 3a 0a 2a  is a blessing:.*
0080: 2a 0a 2a 2a 20 20 20 20 4d 61 79 20 79 6f 75 20  *.**    May you 
0090: 64 6f 20 67 6f 6f 64 20 61 6e 64 20 6e 6f 74 20  do good and not 
00a0: 65 76 69 6c 2e 0a 2a 2a 20 20 20 20 4d 61 79 20  evil..**    May 
00b0: 79 6f 75 20 66 69 6e 64 20 66 6f 72 67 69 76 65  you find forgive
00c0: 6e 65 73 73 20 66 6f 72 20 79 6f 75 72 73 65 6c  ness for yoursel
00d0: 66 20 61 6e 64 20 66 6f 72 67 69 76 65 20 6f 74  f and forgive ot
00e0: 68 65 72 73 2e 0a 2a 2a 20 20 20 20 4d 61 79 20  hers..**    May 
00f0: 79 6f 75 20 73 68 61 72 65 20 66 72 65 65 6c 79  you share freely
0100: 2c 20 6e 65 76 65 72 20 74 61 6b 69 6e 67 20 6d  , never taking m
0110: 6f 72 65 20 74 68 61 6e 20 79 6f 75 20 67 69 76  ore than you giv
0120: 65 2e 0a 2a 2a 0a 2a 2a 2a 2a 2a 2a 2a 2a 2a 2a  e..**.**********
0130: 2a 2a 2a 2a 2a 2a 2a 2a 2a 2a 2a 2a 2a 2a 2a 2a  ****************
0140: 2a 2a 2a 2a 2a 2a 2a 2a 2a 2a 2a 2a 2a 2a 2a 2a  ****************
0150: 2a 2a 2a 2a 2a 2a 2a 2a 2a 2a 2a 2a 2a 2a 2a 2a  ****************
0160: 2a 2a 2a 2a 2a 2a 2a 2a 2a 2a 2a 2a 2a 2a 2a 0a  ***************.
0170: 2a 2a 0a 2a 2a 20 54 68 65 20 63 6f 64 65 20 69  **.** The code i
0180: 6e 20 74 68 69 73 20 66 69 6c 65 20 72 75 6e 73  n this file runs
0190: 20 61 20 66 65 77 20 6d 75 6c 74 69 2d 74 68 72   a few multi-thr
01a0: 65 61 64 65 64 20 74 65 73 74 20 63 61 73 65 73  eaded test cases
01b0: 20 75 73 69 6e 67 20 74 68 65 0a 2a 2a 20 53 51   using the.** SQ
01c0: 4c 69 74 65 20 6c 69 62 72 61 72 79 2e 20 49 74  Lite library. It
01d0: 20 63 61 6e 20 62 65 20 63 6f 6d 70 69 6c 65 64   can be compiled
01e0: 20 74 6f 20 61 6e 20 65 78 65 63 75 74 61 62 6c   to an executabl
01f0: 65 20 6f 6e 20 75 6e 69 78 20 75 73 69 6e 67 20  e on unix using 
0200: 74 68 65 0a 2a 2a 20 66 6f 6c 6c 6f 77 69 6e 67  the.** following
0210: 20 63 6f 6d 6d 61 6e 64 3a 0a 2a 2a 0a 2a 2a 20   command:.**.** 
0220: 20 20 67 63 63 20 2d 4f 32 20 74 68 72 65 61 64    gcc -O2 thread
0230: 74 65 73 74 33 2e 63 20 73 71 6c 69 74 65 33 2e  test3.c sqlite3.
0240: 63 20 2d 6c 64 6c 20 2d 6c 70 74 68 72 65 61 64  c -ldl -lpthread
0250: 20 2d 6c 6d 0a 2a 2a 0a 2a 2a 20 45 76 65 6e 20   -lm.**.** Even 
0260: 74 68 6f 75 67 68 20 74 68 72 65 61 64 74 65 73  though threadtes
0270: 74 33 2e 63 20 69 73 20 74 68 65 20 6f 6e 6c 79  t3.c is the only
0280: 20 43 20 73 6f 75 72 63 65 20 63 6f 64 65 20 66   C source code f
0290: 69 6c 65 20 6d 65 6e 74 69 6f 6e 65 64 20 6f 6e  ile mentioned on
02a0: 0a 2a 2a 20 74 68 65 20 63 6f 6d 70 69 6c 65 72  .** the compiler
02b0: 20 63 6f 6d 6d 61 6e 64 2d 6c 69 6e 65 2c 20 23   command-line, #
02c0: 69 6e 63 6c 75 64 65 20 6d 61 63 72 6f 73 20 61  include macros a
02d0: 72 65 20 75 73 65 64 20 74 6f 20 70 75 6c 6c 20  re used to pull 
02e0: 69 6e 20 61 64 64 69 74 69 6f 6e 61 6c 0a 2a 2a  in additional.**
02f0: 20 43 20 63 6f 64 65 20 66 69 6c 65 73 20 6e 61   C code files na
0300: 6d 65 64 20 22 74 74 33 5f 2a 2e 63 22 2e 0a 2a  med "tt3_*.c"..*
0310: 2a 0a 2a 2a 20 41 66 74 65 72 20 63 6f 6d 70 69  *.** After compi
0320: 6c 69 6e 67 2c 20 72 75 6e 20 74 68 69 73 20 70  ling, run this p
0330: 72 6f 67 72 61 6d 20 77 69 74 68 20 61 6e 20 6f  rogram with an o
0340: 70 74 69 6f 6e 61 6c 20 61 72 67 75 6d 65 6e 74  ptional argument
0350: 20 74 65 6c 6c 69 6e 67 0a 2a 2a 20 77 68 69 63   telling.** whic
0360: 68 20 74 65 73 74 20 74 6f 20 72 75 6e 2e 20 20  h test to run.  
0370: 41 6c 6c 20 74 65 73 74 73 20 61 72 65 20 72 75  All tests are ru
0380: 6e 20 69 66 20 6e 6f 20 61 72 67 75 6d 65 6e 74  n if no argument
0390: 20 69 73 20 67 69 76 65 6e 2e 20 20 54 68 65 0a   is given.  The.
03a0: 2a 2a 20 61 72 67 75 6d 65 6e 74 20 63 61 6e 20  ** argument can 
03b0: 62 65 20 61 20 67 6c 6f 62 20 70 61 74 74 65 72  be a glob patter
03c0: 6e 20 74 6f 20 6d 61 74 63 68 20 6d 75 6c 74 69  n to match multi
03d0: 70 6c 65 20 74 65 73 74 73 2e 20 20 45 78 61 6d  ple tests.  Exam
03e0: 70 6c 65 73 3a 0a 2a 2a 0a 2a 2a 20 20 20 20 20  ples:.**.**     
03f0: 20 20 20 2e 2f 61 2e 6f 75 74 20 20 20 20 20 20     ./a.out      
0400: 20 20 20 20 20 20 20 20 20 20 20 2d 2d 20 52 75             -- Ru
0410: 6e 20 61 6c 6c 20 74 65 73 74 73 0a 2a 2a 20 20  n all tests.**  
0420: 20 20 20 20 20 20 2e 2f 61 2e 6f 75 74 20 77 61        ./a.out wa
0430: 6c 74 68 72 65 61 64 33 20 20 20 20 20 20 2d 2d  lthread3      --
0440: 20 52 75 6e 20 74 68 65 20 22 77 61 6c 74 68 72   Run the "walthr
0450: 65 61 64 33 22 20 74 65 73 74 0a 2a 2a 20 20 20  ead3" test.**   
0460: 20 20 20 20 20 2e 2f 61 2e 6f 75 74 20 27 77 61       ./a.out 'wa
0470: 6c 2a 27 20 20 20 20 20 20 20 20 20 20 2d 2d 20  l*'          -- 
0480: 52 75 6e 20 61 6c 6c 20 6f 66 20 74 68 65 20 77  Run all of the w
0490: 61 6c 2a 20 74 65 73 74 73 0a 2a 2a 20 20 20 20  al* tests.**    
04a0: 20 20 20 20 2e 2f 61 2e 6f 75 74 20 2d 2d 68 65      ./a.out --he
04b0: 6c 70 20 20 20 20 20 20 20 20 20 20 2d 2d 20 4c  lp          -- L
04c0: 69 73 74 20 61 6c 6c 20 61 76 61 69 6c 61 62 6c  ist all availabl
04d0: 65 20 74 65 73 74 73 0a 2a 2a 0a 2a 2a 20 54 68  e tests.**.** Th
04e0: 65 20 65 78 69 74 20 73 74 61 74 75 73 20 69 73  e exit status is
04f0: 20 6e 6f 6e 2d 7a 65 72 6f 20 69 66 20 61 6e 79   non-zero if any
0500: 20 74 65 73 74 20 66 61 69 6c 73 2e 0a 2a 2f 0a   test fails..*/.
0510: 0a 0a 0a 0a 0a 23 69 6e 63 6c 75 64 65 20 3c 73  .....#include <s
0520: 71 6c 69 74 65 33 2e 68 3e 0a 0a 23 69 6e 63 6c  qlite3.h>..#incl
0530: 75 64 65 20 22 74 65 73 74 5f 6d 75 6c 74 69 70  ude "test_multip
0540: 6c 65 78 2e 68 22 0a 23 69 6e 63 6c 75 64 65 20  lex.h".#include 
0550: 22 74 74 33 5f 63 6f 72 65 2e 63 22 0a 0a 2f 2a  "tt3_core.c"../*
0560: 20 52 65 71 75 69 72 65 64 20 74 6f 20 6c 69 6e   Required to lin
0570: 6b 20 74 65 73 74 5f 6d 75 6c 74 69 70 6c 65 78  k test_multiplex
0580: 2e 63 20 2a 2f 0a 23 69 66 6e 64 65 66 20 53 51  .c */.#ifndef SQ
0590: 4c 49 54 45 5f 4f 4d 49 54 5f 57 53 44 0a 69 6e  LITE_OMIT_WSD.in
05a0: 74 20 73 71 6c 69 74 65 33 50 65 6e 64 69 6e 67  t sqlite3Pending
05b0: 42 79 74 65 20 3d 20 30 78 34 30 30 30 30 30 30  Byte = 0x4000000
05c0: 30 3b 0a 23 65 6e 64 69 66 0a 0a 0a 2f 2a 2a 2a  0;.#endif.../***
05d0: 2a 2a 2a 2a 2a 2a 2a 2a 2a 2a 2a 2a 2a 2a 2a 2a  ****************
05e0: 2a 2a 2a 2a 2a 2a 2a 2a 2a 2a 2a 2a 2a 2a 2a 2a  ****************
05f0: 2a 2a 2a 2a 2a 2a 2a 2a 2a 2a 2a 2a 2a 2a 2a 2a  ****************
0600: 2a 2a 2a 2a 2a 2a 2a 2a 2a 2a 2a 2a 2a 2a 2a 2a  ****************
0610: 2a 2a 2a 2a 2a 2a 0a 2a 2a 2a 2a 2a 2a 2a 2a 2a  ******.*********
0620: 2a 2a 2a 2a 2a 2a 2a 2a 2a 2a 2a 2a 2a 2a 2a 2a  ****************
0630: 2a 2a 2a 2a 2a 2a 2a 2a 2a 2a 2a 2a 2a 2a 2a 2a  ****************
0640: 2a 2a 2a 2a 2a 2a 2a 2a 2a 2a 2a 2a 2a 2a 2a 2a  ****************
0650: 2a 2a 2a 2a 2a 2a 2a 2a 2a 2a 2a 2a 2a 2a 2a 2a  ****************
0660: 2a 0a 2a 2a 2a 2a 2a 2a 2a 2a 2a 2a 2a 2a 2a 2a  *.**************
0670: 2a 2a 2a 2a 2a 2a 2a 2a 2a 2a 2a 2a 2a 2a 2a 2a  ****************
0680: 2a 2a 2a 2a 2a 2a 2a 2a 2a 2a 2a 2a 2a 2a 2a 2a  ****************
0690: 2a 2a 2a 2a 2a 2a 2a 2a 2a 2a 2a 2a 2a 2a 2a 2a  ****************
06a0: 2a 2a 2a 2a 2a 2a 2a 2a 2a 2a 2a 2a 0a 2a 2a 20  ************.** 
06b0: 45 6e 64 20 69 6e 66 72 61 73 74 72 75 63 74 75  End infrastructu
06c0: 72 65 2e 20 42 65 67 69 6e 20 74 65 73 74 73 2e  re. Begin tests.
06d0: 0a 2a 2f 0a 0a 23 64 65 66 69 6e 65 20 57 41 4c  .*/..#define WAL
06e0: 54 48 52 45 41 44 31 5f 4e 54 48 52 45 41 44 20  THREAD1_NTHREAD 
06f0: 20 31 30 0a 23 64 65 66 69 6e 65 20 57 41 4c 54   10.#define WALT
0700: 48 52 45 41 44 33 5f 4e 54 48 52 45 41 44 20 20  HREAD3_NTHREAD  
0710: 36 0a 0a 73 74 61 74 69 63 20 63 68 61 72 20 2a  6..static char *
0720: 77 61 6c 74 68 72 65 61 64 31 5f 74 68 72 65 61  walthread1_threa
0730: 64 28 69 6e 74 20 69 54 69 64 2c 20 76 6f 69 64  d(int iTid, void
0740: 20 2a 70 41 72 67 29 7b 0a 20 20 45 72 72 6f 72   *pArg){.  Error
0750: 20 65 72 72 20 3d 20 7b 30 7d 3b 20 20 20 20 20   err = {0};     
0760: 20 20 20 20 20 20 20 20 20 20 20 2f 2a 20 45 72             /* Er
0770: 72 6f 72 20 63 6f 64 65 20 61 6e 64 20 6d 65 73  ror code and mes
0780: 73 61 67 65 20 2a 2f 0a 20 20 53 71 6c 69 74 65  sage */.  Sqlite
0790: 20 64 62 20 3d 20 7b 30 7d 3b 20 20 20 20 20 20   db = {0};      
07a0: 20 20 20 20 20 20 20 20 20 20 2f 2a 20 53 51 4c            /* SQL
07b0: 69 74 65 20 64 61 74 61 62 61 73 65 20 63 6f 6e  ite database con
07c0: 6e 65 63 74 69 6f 6e 20 2a 2f 0a 20 20 69 6e 74  nection */.  int
07d0: 20 6e 49 74 65 72 20 3d 20 30 3b 20 20 20 20 20   nIter = 0;     
07e0: 20 20 20 20 20 20 20 20 20 20 20 20 20 2f 2a 20               /* 
07f0: 49 74 65 72 61 74 69 6f 6e 73 20 73 6f 20 66 61  Iterations so fa
0800: 72 20 2a 2f 0a 0a 20 20 6f 70 65 6e 64 62 28 26  r */..  opendb(&
0810: 65 72 72 2c 20 26 64 62 2c 20 22 74 65 73 74 2e  err, &db, "test.
0820: 64 62 22 2c 20 30 29 3b 0a 20 20 77 68 69 6c 65  db", 0);.  while
0830: 28 20 21 74 69 6d 65 74 6f 73 74 6f 70 28 26 65  ( !timetostop(&e
0840: 72 72 29 20 29 7b 0a 20 20 20 20 63 6f 6e 73 74  rr) ){.    const
0850: 20 63 68 61 72 20 2a 61 7a 53 71 6c 5b 5d 20 3d   char *azSql[] =
0860: 20 7b 0a 20 20 20 20 20 20 22 53 45 4c 45 43 54   {.      "SELECT
0870: 20 6d 64 35 73 75 6d 28 78 29 20 46 52 4f 4d 20   md5sum(x) FROM 
0880: 74 31 20 57 48 45 52 45 20 72 6f 77 69 64 20 21  t1 WHERE rowid !
0890: 3d 20 28 53 45 4c 45 43 54 20 6d 61 78 28 72 6f  = (SELECT max(ro
08a0: 77 69 64 29 20 46 52 4f 4d 20 74 31 29 22 2c 0a  wid) FROM t1)",.
08b0: 20 20 20 20 20 20 22 53 45 4c 45 43 54 20 78 20        "SELECT x 
08c0: 46 52 4f 4d 20 74 31 20 57 48 45 52 45 20 72 6f  FROM t1 WHERE ro
08d0: 77 69 64 20 3d 20 28 53 45 4c 45 43 54 20 6d 61  wid = (SELECT ma
08e0: 78 28 72 6f 77 69 64 29 20 46 52 4f 4d 20 74 31  x(rowid) FROM t1
08f0: 29 22 2c 0a 20 20 20 20 7d 3b 0a 20 20 20 20 63  )",.    };.    c
0900: 68 61 72 20 2a 7a 31 2c 20 2a 7a 32 2c 20 2a 7a  har *z1, *z2, *z
0910: 33 3b 0a 0a 20 20 20 20 65 78 65 63 73 71 6c 28  3;..    execsql(
0920: 26 65 72 72 2c 20 26 64 62 2c 20 22 42 45 47 49  &err, &db, "BEGI
0930: 4e 22 29 3b 0a 20 20 20 20 69 6e 74 65 67 72 69  N");.    integri
0940: 74 79 5f 63 68 65 63 6b 28 26 65 72 72 2c 20 26  ty_check(&err, &
0950: 64 62 29 3b 0a 20 20 20 20 7a 31 20 3d 20 65 78  db);.    z1 = ex
0960: 65 63 73 71 6c 5f 74 65 78 74 28 26 65 72 72 2c  ecsql_text(&err,
0970: 20 26 64 62 2c 20 31 2c 20 61 7a 53 71 6c 5b 30   &db, 1, azSql[0
0980: 5d 29 3b 0a 20 20 20 20 7a 32 20 3d 20 65 78 65  ]);.    z2 = exe
0990: 63 73 71 6c 5f 74 65 78 74 28 26 65 72 72 2c 20  csql_text(&err, 
09a0: 26 64 62 2c 20 32 2c 20 61 7a 53 71 6c 5b 31 5d  &db, 2, azSql[1]
09b0: 29 3b 0a 20 20 20 20 7a 33 20 3d 20 65 78 65 63  );.    z3 = exec
09c0: 73 71 6c 5f 74 65 78 74 28 26 65 72 72 2c 20 26  sql_text(&err, &
09d0: 64 62 2c 20 33 2c 20 61 7a 53 71 6c 5b 30 5d 29  db, 3, azSql[0])
09e0: 3b 0a 20 20 20 20 65 78 65 63 73 71 6c 28 26 65  ;.    execsql(&e
09f0: 72 72 2c 20 26 64 62 2c 20 22 43 4f 4d 4d 49 54  rr, &db, "COMMIT
0a00: 22 29 3b 0a 0a 20 20 20 20 69 66 28 20 73 74 72  ");..    if( str
0a10: 63 6d 70 28 7a 31 2c 20 7a 32 29 20 7c 7c 20 73  cmp(z1, z2) || s
0a20: 74 72 63 6d 70 28 7a 31 2c 20 7a 33 29 20 29 7b  trcmp(z1, z3) ){
0a30: 0a 20 20 20 20 20 20 74 65 73 74 5f 65 72 72 6f  .      test_erro
0a40: 72 28 26 65 72 72 2c 20 22 46 61 69 6c 65 64 20  r(&err, "Failed 
0a50: 72 65 61 64 3a 20 25 73 20 25 73 20 25 73 22 2c  read: %s %s %s",
0a60: 20 7a 31 2c 20 7a 32 2c 20 7a 33 29 3b 0a 20 20   z1, z2, z3);.  
0a70: 20 20 7d 0a 0a 20 20 20 20 73 71 6c 5f 73 63 72    }..    sql_scr
0a80: 69 70 74 28 26 65 72 72 2c 20 26 64 62 2c 0a 20  ipt(&err, &db,. 
0a90: 20 20 20 20 20 20 20 22 42 45 47 49 4e 3b 22 0a         "BEGIN;".
0aa0: 20 20 20 20 20 20 20 20 20 20 22 49 4e 53 45 52            "INSER
0ab0: 54 20 49 4e 54 4f 20 74 31 20 56 41 4c 55 45 53  T INTO t1 VALUES
0ac0: 28 72 61 6e 64 6f 6d 62 6c 6f 62 28 31 30 30 29  (randomblob(100)
0ad0: 29 3b 22 0a 20 20 20 20 20 20 20 20 20 20 22 49  );".          "I
0ae0: 4e 53 45 52 54 20 49 4e 54 4f 20 74 31 20 56 41  NSERT INTO t1 VA
0af0: 4c 55 45 53 28 72 61 6e 64 6f 6d 62 6c 6f 62 28  LUES(randomblob(
0b00: 31 30 30 29 29 3b 22 0a 20 20 20 20 20 20 20 20  100));".        
0b10: 20 20 22 49 4e 53 45 52 54 20 49 4e 54 4f 20 74    "INSERT INTO t
0b20: 31 20 53 45 4c 45 43 54 20 6d 64 35 73 75 6d 28  1 SELECT md5sum(
0b30: 78 29 20 46 52 4f 4d 20 74 31 3b 22 0a 20 20 20  x) FROM t1;".   
0b40: 20 20 20 20 20 22 43 4f 4d 4d 49 54 3b 22 0a 20       "COMMIT;". 
0b50: 20 20 20 29 3b 0a 20 20 20 20 6e 49 74 65 72 2b     );.    nIter+
0b60: 2b 3b 0a 20 20 7d 0a 20 20 63 6c 6f 73 65 64 62  +;.  }.  closedb
0b70: 28 26 65 72 72 2c 20 26 64 62 29 3b 0a 0a 20 20  (&err, &db);..  
0b80: 70 72 69 6e 74 5f 61 6e 64 5f 66 72 65 65 5f 65  print_and_free_e
0b90: 72 72 28 26 65 72 72 29 3b 0a 20 20 72 65 74 75  rr(&err);.  retu
0ba0: 72 6e 20 73 71 6c 69 74 65 33 5f 6d 70 72 69 6e  rn sqlite3_mprin
0bb0: 74 66 28 22 25 64 20 69 74 65 72 61 74 69 6f 6e  tf("%d iteration
0bc0: 73 22 2c 20 6e 49 74 65 72 29 3b 0a 7d 0a 0a 73  s", nIter);.}..s
0bd0: 74 61 74 69 63 20 63 68 61 72 20 2a 77 61 6c 74  tatic char *walt
0be0: 68 72 65 61 64 31 5f 63 6b 70 74 5f 74 68 72 65  hread1_ckpt_thre
0bf0: 61 64 28 69 6e 74 20 69 54 69 64 2c 20 76 6f 69  ad(int iTid, voi
0c00: 64 20 2a 70 41 72 67 29 7b 0a 20 20 45 72 72 6f  d *pArg){.  Erro
0c10: 72 20 65 72 72 20 3d 20 7b 30 7d 3b 20 20 20 20  r err = {0};    
0c20: 20 20 20 20 20 20 20 20 20 20 20 20 2f 2a 20 45              /* E
0c30: 72 72 6f 72 20 63 6f 64 65 20 61 6e 64 20 6d 65  rror code and me
0c40: 73 73 61 67 65 20 2a 2f 0a 20 20 53 71 6c 69 74  ssage */.  Sqlit
0c50: 65 20 64 62 20 3d 20 7b 30 7d 3b 20 20 20 20 20  e db = {0};     
0c60: 20 20 20 20 20 20 20 20 20 20 20 2f 2a 20 53 51             /* SQ
0c70: 4c 69 74 65 20 64 61 74 61 62 61 73 65 20 63 6f  Lite database co
0c80: 6e 6e 65 63 74 69 6f 6e 20 2a 2f 0a 20 20 69 6e  nnection */.  in
0c90: 74 20 6e 43 6b 70 74 20 3d 20 30 3b 20 20 20 20  t nCkpt = 0;    
0ca0: 20 20 20 20 20 20 20 20 20 20 20 20 20 20 2f 2a                /*
0cb0: 20 43 68 65 63 6b 70 6f 69 6e 74 73 20 73 6f 20   Checkpoints so 
0cc0: 66 61 72 20 2a 2f 0a 0a 20 20 6f 70 65 6e 64 62  far */..  opendb
0cd0: 28 26 65 72 72 2c 20 26 64 62 2c 20 22 74 65 73  (&err, &db, "tes
0ce0: 74 2e 64 62 22 2c 20 30 29 3b 0a 20 20 77 68 69  t.db", 0);.  whi
0cf0: 6c 65 28 20 21 74 69 6d 65 74 6f 73 74 6f 70 28  le( !timetostop(
0d00: 26 65 72 72 29 20 29 7b 0a 20 20 20 20 75 73 6c  &err) ){.    usl
0d10: 65 65 70 28 35 30 30 2a 31 30 30 30 29 3b 0a 20  eep(500*1000);. 
0d20: 20 20 20 65 78 65 63 73 71 6c 28 26 65 72 72 2c     execsql(&err,
0d30: 20 26 64 62 2c 20 22 50 52 41 47 4d 41 20 77 61   &db, "PRAGMA wa
0d40: 6c 5f 63 68 65 63 6b 70 6f 69 6e 74 22 29 3b 0a  l_checkpoint");.
0d50: 20 20 20 20 69 66 28 20 65 72 72 2e 72 63 3d 3d      if( err.rc==
0d60: 53 51 4c 49 54 45 5f 4f 4b 20 29 20 6e 43 6b 70  SQLITE_OK ) nCkp
0d70: 74 2b 2b 3b 0a 20 20 20 20 63 6c 65 61 72 5f 65  t++;.    clear_e
0d80: 72 72 6f 72 28 26 65 72 72 2c 20 53 51 4c 49 54  rror(&err, SQLIT
0d90: 45 5f 42 55 53 59 29 3b 0a 20 20 7d 0a 20 20 63  E_BUSY);.  }.  c
0da0: 6c 6f 73 65 64 62 28 26 65 72 72 2c 20 26 64 62  losedb(&err, &db
0db0: 29 3b 0a 0a 20 20 70 72 69 6e 74 5f 61 6e 64 5f  );..  print_and_
0dc0: 66 72 65 65 5f 65 72 72 28 26 65 72 72 29 3b 0a  free_err(&err);.
0dd0: 20 20 72 65 74 75 72 6e 20 73 71 6c 69 74 65 33    return sqlite3
0de0: 5f 6d 70 72 69 6e 74 66 28 22 25 64 20 63 68 65  _mprintf("%d che
0df0: 63 6b 70 6f 69 6e 74 73 22 2c 20 6e 43 6b 70 74  ckpoints", nCkpt
0e00: 29 3b 0a 7d 0a 0a 73 74 61 74 69 63 20 76 6f 69  );.}..static voi
0e10: 64 20 77 61 6c 74 68 72 65 61 64 31 28 69 6e 74  d walthread1(int
0e20: 20 6e 4d 73 29 7b 0a 20 20 45 72 72 6f 72 20 65   nMs){.  Error e
0e30: 72 72 20 3d 20 7b 30 7d 3b 20 20 20 20 20 20 20  rr = {0};       
0e40: 20 20 20 20 20 20 20 20 20 2f 2a 20 45 72 72 6f           /* Erro
0e50: 72 20 63 6f 64 65 20 61 6e 64 20 6d 65 73 73 61  r code and messa
0e60: 67 65 20 2a 2f 0a 20 20 53 71 6c 69 74 65 20 64  ge */.  Sqlite d
0e70: 62 20 3d 20 7b 30 7d 3b 20 20 20 20 20 20 20 20  b = {0};        
0e80: 20 20 20 20 20 20 20 20 2f 2a 20 53 51 4c 69 74          /* SQLit
0e90: 65 20 64 61 74 61 62 61 73 65 20 63 6f 6e 6e 65  e database conne
0ea0: 63 74 69 6f 6e 20 2a 2f 0a 20 20 54 68 72 65 61  ction */.  Threa
0eb0: 64 73 65 74 20 74 68 72 65 61 64 73 20 3d 20 7b  dset threads = {
0ec0: 30 7d 3b 20 20 20 20 20 20 20 20 2f 2a 20 54 65  0};        /* Te
0ed0: 73 74 20 74 68 72 65 61 64 73 20 2a 2f 0a 20 20  st threads */.  
0ee0: 69 6e 74 20 69 3b 20 20 20 20 20 20 20 20 20 20  int i;          
0ef0: 20 20 20 20 20 20 20 20 20 20 20 20 20 20 20 20                  
0f00: 2f 2a 20 49 74 65 72 61 74 6f 72 20 76 61 72 69  /* Iterator vari
0f10: 61 62 6c 65 20 2a 2f 0a 0a 20 20 6f 70 65 6e 64  able */..  opend
0f20: 62 28 26 65 72 72 2c 20 26 64 62 2c 20 22 74 65  b(&err, &db, "te
0f30: 73 74 2e 64 62 22 2c 20 31 29 3b 0a 20 20 73 71  st.db", 1);.  sq
0f40: 6c 5f 73 63 72 69 70 74 28 26 65 72 72 2c 20 26  l_script(&err, &
0f50: 64 62 2c 0a 20 20 20 20 20 20 22 50 52 41 47 4d  db,.      "PRAGM
0f60: 41 20 6a 6f 75 72 6e 61 6c 5f 6d 6f 64 65 20 3d  A journal_mode =
0f70: 20 57 41 4c 3b 22 0a 20 20 20 20 20 20 22 43 52   WAL;".      "CR
0f80: 45 41 54 45 20 54 41 42 4c 45 20 74 31 28 78 20  EATE TABLE t1(x 
0f90: 50 52 49 4d 41 52 59 20 4b 45 59 29 3b 22 0a 20  PRIMARY KEY);". 
0fa0: 20 20 20 20 20 22 49 4e 53 45 52 54 20 49 4e 54       "INSERT INT
0fb0: 4f 20 74 31 20 56 41 4c 55 45 53 28 72 61 6e 64  O t1 VALUES(rand
0fc0: 6f 6d 62 6c 6f 62 28 31 30 30 29 29 3b 22 0a 20  omblob(100));". 
0fd0: 20 20 20 20 20 22 49 4e 53 45 52 54 20 49 4e 54       "INSERT INT
0fe0: 4f 20 74 31 20 56 41 4c 55 45 53 28 72 61 6e 64  O t1 VALUES(rand
0ff0: 6f 6d 62 6c 6f 62 28 31 30 30 29 29 3b 22 0a 20  omblob(100));". 
1000: 20 20 20 20 20 22 49 4e 53 45 52 54 20 49 4e 54       "INSERT INT
1010: 4f 20 74 31 20 53 45 4c 45 43 54 20 6d 64 35 73  O t1 SELECT md5s
1020: 75 6d 28 78 29 20 46 52 4f 4d 20 74 31 3b 22 0a  um(x) FROM t1;".
1030: 20 20 29 3b 0a 20 20 63 6c 6f 73 65 64 62 28 26    );.  closedb(&
1040: 65 72 72 2c 20 26 64 62 29 3b 0a 0a 20 20 73 65  err, &db);..  se
1050: 74 73 74 6f 70 74 69 6d 65 28 26 65 72 72 2c 20  tstoptime(&err, 
1060: 6e 4d 73 29 3b 0a 20 20 66 6f 72 28 69 3d 30 3b  nMs);.  for(i=0;
1070: 20 69 3c 57 41 4c 54 48 52 45 41 44 31 5f 4e 54   i<WALTHREAD1_NT
1080: 48 52 45 41 44 3b 20 69 2b 2b 29 7b 0a 20 20 20  HREAD; i++){.   
1090: 20 6c 61 75 6e 63 68 5f 74 68 72 65 61 64 28 26   launch_thread(&
10a0: 65 72 72 2c 20 26 74 68 72 65 61 64 73 2c 20 77  err, &threads, w
10b0: 61 6c 74 68 72 65 61 64 31 5f 74 68 72 65 61 64  althread1_thread
10c0: 2c 20 30 29 3b 0a 20 20 7d 0a 20 20 6c 61 75 6e  , 0);.  }.  laun
10d0: 63 68 5f 74 68 72 65 61 64 28 26 65 72 72 2c 20  ch_thread(&err, 
10e0: 26 74 68 72 65 61 64 73 2c 20 77 61 6c 74 68 72  &threads, walthr
10f0: 65 61 64 31 5f 63 6b 70 74 5f 74 68 72 65 61 64  ead1_ckpt_thread
1100: 2c 20 30 29 3b 0a 20 20 6a 6f 69 6e 5f 61 6c 6c  , 0);.  join_all
1110: 5f 74 68 72 65 61 64 73 28 26 65 72 72 2c 20 26  _threads(&err, &
1120: 74 68 72 65 61 64 73 29 3b 0a 0a 20 20 70 72 69  threads);..  pri
1130: 6e 74 5f 61 6e 64 5f 66 72 65 65 5f 65 72 72 28  nt_and_free_err(
1140: 26 65 72 72 29 3b 0a 7d 0a 0a 73 74 61 74 69 63  &err);.}..static
1150: 20 63 68 61 72 20 2a 77 61 6c 74 68 72 65 61 64   char *walthread
1160: 32 5f 74 68 72 65 61 64 28 69 6e 74 20 69 54 69  2_thread(int iTi
1170: 64 2c 20 76 6f 69 64 20 2a 70 41 72 67 29 7b 0a  d, void *pArg){.
1180: 20 20 45 72 72 6f 72 20 65 72 72 20 3d 20 7b 30    Error err = {0
1190: 7d 3b 20 20 20 20 20 20 20 20 20 20 20 20 20 20  };              
11a0: 20 20 2f 2a 20 45 72 72 6f 72 20 63 6f 64 65 20    /* Error code 
11b0: 61 6e 64 20 6d 65 73 73 61 67 65 20 2a 2f 0a 20  and message */. 
11c0: 20 53 71 6c 69 74 65 20 64 62 20 3d 20 7b 30 7d   Sqlite db = {0}
11d0: 3b 20 20 20 20 20 20 20 20 20 20 20 20 20 20 20  ;               
11e0: 20 2f 2a 20 53 51 4c 69 74 65 20 64 61 74 61 62   /* SQLite datab
11f0: 61 73 65 20 63 6f 6e 6e 65 63 74 69 6f 6e 20 2a  ase connection *
1200: 2f 0a 20 20 69 6e 74 20 61 6e 54 72 61 6e 73 5b  /.  int anTrans[
1210: 32 5d 20 3d 20 7b 30 2c 20 30 7d 3b 20 20 20 20  2] = {0, 0};    
1220: 20 20 20 20 2f 2a 20 4e 75 6d 62 65 72 20 6f 66      /* Number of
1230: 20 57 41 4c 20 61 6e 64 20 52 6f 6c 6c 62 61 63   WAL and Rollbac
1240: 6b 20 74 72 61 6e 73 61 63 74 69 6f 6e 73 20 2a  k transactions *
1250: 2f 0a 20 20 69 6e 74 20 69 41 72 67 20 3d 20 50  /.  int iArg = P
1260: 54 52 32 49 4e 54 28 70 41 72 67 29 3b 0a 0a 20  TR2INT(pArg);.. 
1270: 20 63 6f 6e 73 74 20 63 68 61 72 20 2a 7a 4a 6f   const char *zJo
1280: 75 72 6e 61 6c 20 3d 20 22 50 52 41 47 4d 41 20  urnal = "PRAGMA 
1290: 6a 6f 75 72 6e 61 6c 5f 6d 6f 64 65 20 3d 20 57  journal_mode = W
12a0: 41 4c 22 3b 0a 20 20 69 66 28 20 69 41 72 67 20  AL";.  if( iArg 
12b0: 29 7b 20 7a 4a 6f 75 72 6e 61 6c 20 3d 20 22 50  ){ zJournal = "P
12c0: 52 41 47 4d 41 20 6a 6f 75 72 6e 61 6c 5f 6d 6f  RAGMA journal_mo
12d0: 64 65 20 3d 20 44 45 4c 45 54 45 22 3b 20 7d 0a  de = DELETE"; }.
12e0: 0a 20 20 77 68 69 6c 65 28 20 21 74 69 6d 65 74  .  while( !timet
12f0: 6f 73 74 6f 70 28 26 65 72 72 29 20 29 7b 0a 20  ostop(&err) ){. 
1300: 20 20 20 69 6e 74 20 6a 6f 75 72 6e 61 6c 5f 65     int journal_e
1310: 78 69 73 74 73 20 3d 20 30 3b 0a 20 20 20 20 69  xists = 0;.    i
1320: 6e 74 20 77 61 6c 5f 65 78 69 73 74 73 20 3d 20  nt wal_exists = 
1330: 30 3b 0a 0a 20 20 20 20 6f 70 65 6e 64 62 28 26  0;..    opendb(&
1340: 65 72 72 2c 20 26 64 62 2c 20 22 74 65 73 74 2e  err, &db, "test.
1350: 64 62 22 2c 20 30 29 3b 0a 0a 20 20 20 20 73 71  db", 0);..    sq
1360: 6c 5f 73 63 72 69 70 74 28 26 65 72 72 2c 20 26  l_script(&err, &
1370: 64 62 2c 20 7a 4a 6f 75 72 6e 61 6c 29 3b 0a 20  db, zJournal);. 
1380: 20 20 20 63 6c 65 61 72 5f 65 72 72 6f 72 28 26     clear_error(&
1390: 65 72 72 2c 20 53 51 4c 49 54 45 5f 42 55 53 59  err, SQLITE_BUSY
13a0: 29 3b 0a 20 20 20 20 73 71 6c 5f 73 63 72 69 70  );.    sql_scrip
13b0: 74 28 26 65 72 72 2c 20 26 64 62 2c 20 22 42 45  t(&err, &db, "BE
13c0: 47 49 4e 22 29 3b 0a 20 20 20 20 73 71 6c 5f 73  GIN");.    sql_s
13d0: 63 72 69 70 74 28 26 65 72 72 2c 20 26 64 62 2c  cript(&err, &db,
13e0: 20 22 49 4e 53 45 52 54 20 49 4e 54 4f 20 74 31   "INSERT INTO t1
13f0: 20 56 41 4c 55 45 53 28 4e 55 4c 4c 2c 20 72 61   VALUES(NULL, ra
1400: 6e 64 6f 6d 62 6c 6f 62 28 31 30 30 29 29 22 29  ndomblob(100))")
1410: 3b 0a 0a 20 20 20 20 6a 6f 75 72 6e 61 6c 5f 65  ;..    journal_e
1420: 78 69 73 74 73 20 3d 20 28 66 69 6c 65 73 69 7a  xists = (filesiz
1430: 65 28 26 65 72 72 2c 20 22 74 65 73 74 2e 64 62  e(&err, "test.db
1440: 2d 6a 6f 75 72 6e 61 6c 22 29 20 3e 3d 20 30 29  -journal") >= 0)
1450: 3b 0a 20 20 20 20 77 61 6c 5f 65 78 69 73 74 73  ;.    wal_exists
1460: 20 3d 20 28 66 69 6c 65 73 69 7a 65 28 26 65 72   = (filesize(&er
1470: 72 2c 20 22 74 65 73 74 2e 64 62 2d 77 61 6c 22  r, "test.db-wal"
1480: 29 20 3e 3d 20 30 29 3b 0a 20 20 20 20 69 66 28  ) >= 0);.    if(
1490: 20 28 6a 6f 75 72 6e 61 6c 5f 65 78 69 73 74 73   (journal_exists
14a0: 2b 77 61 6c 5f 65 78 69 73 74 73 29 21 3d 31 20  +wal_exists)!=1 
14b0: 29 7b 0a 20 20 20 20 20 20 74 65 73 74 5f 65 72  ){.      test_er
14c0: 72 6f 72 28 26 65 72 72 2c 20 22 46 69 6c 65 20  ror(&err, "File 
14d0: 73 79 73 74 65 6d 20 6c 6f 6f 6b 73 20 69 6e 63  system looks inc
14e0: 6f 72 72 65 63 74 20 28 25 64 2c 20 25 64 29 22  orrect (%d, %d)"
14f0: 2c 20 0a 20 20 20 20 20 20 20 20 20 20 6a 6f 75  , .          jou
1500: 72 6e 61 6c 5f 65 78 69 73 74 73 2c 20 77 61 6c  rnal_exists, wal
1510: 5f 65 78 69 73 74 73 0a 20 20 20 20 20 20 29 3b  _exists.      );
1520: 0a 20 20 20 20 7d 0a 20 20 20 20 61 6e 54 72 61  .    }.    anTra
1530: 6e 73 5b 6a 6f 75 72 6e 61 6c 5f 65 78 69 73 74  ns[journal_exist
1540: 73 5d 2b 2b 3b 0a 0a 20 20 20 20 73 71 6c 5f 73  s]++;..    sql_s
1550: 63 72 69 70 74 28 26 65 72 72 2c 20 26 64 62 2c  cript(&err, &db,
1560: 20 22 43 4f 4d 4d 49 54 22 29 3b 0a 20 20 20 20   "COMMIT");.    
1570: 69 6e 74 65 67 72 69 74 79 5f 63 68 65 63 6b 28  integrity_check(
1580: 26 65 72 72 2c 20 26 64 62 29 3b 0a 20 20 20 20  &err, &db);.    
1590: 63 6c 6f 73 65 64 62 28 26 65 72 72 2c 20 26 64  closedb(&err, &d
15a0: 62 29 3b 0a 20 20 7d 0a 0a 20 20 70 72 69 6e 74  b);.  }..  print
15b0: 5f 61 6e 64 5f 66 72 65 65 5f 65 72 72 28 26 65  _and_free_err(&e
15c0: 72 72 29 3b 0a 20 20 72 65 74 75 72 6e 20 73 71  rr);.  return sq
15d0: 6c 69 74 65 33 5f 6d 70 72 69 6e 74 66 28 22 57  lite3_mprintf("W
15e0: 20 25 64 20 52 20 25 64 22 2c 20 61 6e 54 72 61   %d R %d", anTra
15f0: 6e 73 5b 30 5d 2c 20 61 6e 54 72 61 6e 73 5b 31  ns[0], anTrans[1
1600: 5d 29 3b 0a 7d 0a 0a 73 74 61 74 69 63 20 76 6f  ]);.}..static vo
1610: 69 64 20 77 61 6c 74 68 72 65 61 64 32 28 69 6e  id walthread2(in
1620: 74 20 6e 4d 73 29 7b 0a 20 20 45 72 72 6f 72 20  t nMs){.  Error 
1630: 65 72 72 20 3d 20 7b 30 7d 3b 0a 20 20 53 71 6c  err = {0};.  Sql
1640: 69 74 65 20 64 62 20 3d 20 7b 30 7d 3b 0a 20 20  ite db = {0};.  
1650: 54 68 72 65 61 64 73 65 74 20 74 68 72 65 61 64  Threadset thread
1660: 73 20 3d 20 7b 30 7d 3b 0a 0a 20 20 6f 70 65 6e  s = {0};..  open
1670: 64 62 28 26 65 72 72 2c 20 26 64 62 2c 20 22 74  db(&err, &db, "t
1680: 65 73 74 2e 64 62 22 2c 20 31 29 3b 0a 20 20 73  est.db", 1);.  s
1690: 71 6c 5f 73 63 72 69 70 74 28 26 65 72 72 2c 20  ql_script(&err, 
16a0: 26 64 62 2c 20 22 43 52 45 41 54 45 20 54 41 42  &db, "CREATE TAB
16b0: 4c 45 20 74 31 28 78 20 49 4e 54 45 47 45 52 20  LE t1(x INTEGER 
16c0: 50 52 49 4d 41 52 59 20 4b 45 59 2c 20 79 20 55  PRIMARY KEY, y U
16d0: 4e 49 51 55 45 29 22 29 3b 0a 20 20 63 6c 6f 73  NIQUE)");.  clos
16e0: 65 64 62 28 26 65 72 72 2c 20 26 64 62 29 3b 0a  edb(&err, &db);.
16f0: 0a 20 20 73 65 74 73 74 6f 70 74 69 6d 65 28 26  .  setstoptime(&
1700: 65 72 72 2c 20 6e 4d 73 29 3b 0a 20 20 6c 61 75  err, nMs);.  lau
1710: 6e 63 68 5f 74 68 72 65 61 64 28 26 65 72 72 2c  nch_thread(&err,
1720: 20 26 74 68 72 65 61 64 73 2c 20 77 61 6c 74 68   &threads, walth
1730: 72 65 61 64 32 5f 74 68 72 65 61 64 2c 20 30 29  read2_thread, 0)
1740: 3b 0a 20 20 6c 61 75 6e 63 68 5f 74 68 72 65 61  ;.  launch_threa
1750: 64 28 26 65 72 72 2c 20 26 74 68 72 65 61 64 73  d(&err, &threads
1760: 2c 20 77 61 6c 74 68 72 65 61 64 32 5f 74 68 72  , walthread2_thr
1770: 65 61 64 2c 20 30 29 3b 0a 20 20 6c 61 75 6e 63  ead, 0);.  launc
1780: 68 5f 74 68 72 65 61 64 28 26 65 72 72 2c 20 26  h_thread(&err, &
1790: 74 68 72 65 61 64 73 2c 20 77 61 6c 74 68 72 65  threads, walthre
17a0: 61 64 32 5f 74 68 72 65 61 64 2c 20 28 76 6f 69  ad2_thread, (voi
17b0: 64 2a 29 31 29 3b 0a 20 20 6c 61 75 6e 63 68 5f  d*)1);.  launch_
17c0: 74 68 72 65 61 64 28 26 65 72 72 2c 20 26 74 68  thread(&err, &th
17d0: 72 65 61 64 73 2c 20 77 61 6c 74 68 72 65 61 64  reads, walthread
17e0: 32 5f 74 68 72 65 61 64 2c 20 28 76 6f 69 64 2a  2_thread, (void*
17f0: 29 31 29 3b 0a 20 20 6a 6f 69 6e 5f 61 6c 6c 5f  )1);.  join_all_
1800: 74 68 72 65 61 64 73 28 26 65 72 72 2c 20 26 74  threads(&err, &t
1810: 68 72 65 61 64 73 29 3b 0a 0a 20 20 70 72 69 6e  hreads);..  prin
1820: 74 5f 61 6e 64 5f 66 72 65 65 5f 65 72 72 28 26  t_and_free_err(&
1830: 65 72 72 29 3b 0a 7d 0a 0a 73 74 61 74 69 63 20  err);.}..static 
1840: 63 68 61 72 20 2a 77 61 6c 74 68 72 65 61 64 33  char *walthread3
1850: 5f 74 68 72 65 61 64 28 69 6e 74 20 69 54 69 64  _thread(int iTid
1860: 2c 20 76 6f 69 64 20 2a 70 41 72 67 29 7b 0a 20  , void *pArg){. 
1870: 20 45 72 72 6f 72 20 65 72 72 20 3d 20 7b 30 7d   Error err = {0}
1880: 3b 20 20 20 20 20 20 20 20 20 20 20 20 20 20 20  ;               
1890: 20 2f 2a 20 45 72 72 6f 72 20 63 6f 64 65 20 61   /* Error code a
18a0: 6e 64 20 6d 65 73 73 61 67 65 20 2a 2f 0a 20 20  nd message */.  
18b0: 53 71 6c 69 74 65 20 64 62 20 3d 20 7b 30 7d 3b  Sqlite db = {0};
18c0: 20 20 20 20 20 20 20 20 20 20 20 20 20 20 20 20                  
18d0: 2f 2a 20 53 51 4c 69 74 65 20 64 61 74 61 62 61  /* SQLite databa
18e0: 73 65 20 63 6f 6e 6e 65 63 74 69 6f 6e 20 2a 2f  se connection */
18f0: 0a 20 20 69 36 34 20 69 4e 65 78 74 57 72 69 74  .  i64 iNextWrit
1900: 65 3b 20 20 20 20 20 20 20 20 20 20 20 20 20 20  e;              
1910: 20 20 20 2f 2a 20 4e 65 78 74 20 76 61 6c 75 65     /* Next value
1920: 20 74 68 69 73 20 74 68 72 65 61 64 20 77 69 6c   this thread wil
1930: 6c 20 77 72 69 74 65 20 2a 2f 0a 20 20 69 6e 74  l write */.  int
1940: 20 69 41 72 67 20 3d 20 50 54 52 32 49 4e 54 28   iArg = PTR2INT(
1950: 70 41 72 67 29 3b 0a 0a 20 20 6f 70 65 6e 64 62  pArg);..  opendb
1960: 28 26 65 72 72 2c 20 26 64 62 2c 20 22 74 65 73  (&err, &db, "tes
1970: 74 2e 64 62 22 2c 20 30 29 3b 0a 20 20 73 71 6c  t.db", 0);.  sql
1980: 5f 73 63 72 69 70 74 28 26 65 72 72 2c 20 26 64  _script(&err, &d
1990: 62 2c 20 22 50 52 41 47 4d 41 20 77 61 6c 5f 61  b, "PRAGMA wal_a
19a0: 75 74 6f 63 68 65 63 6b 70 6f 69 6e 74 20 3d 20  utocheckpoint = 
19b0: 31 30 22 29 3b 0a 0a 20 20 69 4e 65 78 74 57 72  10");..  iNextWr
19c0: 69 74 65 20 3d 20 69 41 72 67 2b 31 3b 0a 20 20  ite = iArg+1;.  
19d0: 77 68 69 6c 65 28 20 31 20 29 7b 0a 20 20 20 20  while( 1 ){.    
19e0: 69 36 34 20 73 75 6d 31 3b 0a 20 20 20 20 69 36  i64 sum1;.    i6
19f0: 34 20 73 75 6d 32 3b 0a 20 20 20 20 69 6e 74 20  4 sum2;.    int 
1a00: 73 74 6f 70 20 3d 20 30 3b 20 20 20 20 20 20 20  stop = 0;       
1a10: 20 20 20 20 20 20 20 20 20 20 2f 2a 20 54 72 75            /* Tru
1a20: 65 20 74 6f 20 73 74 6f 70 20 65 78 65 63 75 74  e to stop execut
1a30: 69 6e 67 20 28 74 65 73 74 20 74 69 6d 65 64 20  ing (test timed 
1a40: 6f 75 74 29 20 2a 2f 0a 0a 20 20 20 20 77 68 69  out) */..    whi
1a50: 6c 65 28 20 30 3d 3d 28 73 74 6f 70 20 3d 20 74  le( 0==(stop = t
1a60: 69 6d 65 74 6f 73 74 6f 70 28 26 65 72 72 29 29  imetostop(&err))
1a70: 20 29 7b 0a 20 20 20 20 20 20 69 36 34 20 69 4d   ){.      i64 iM
1a80: 61 78 20 3d 20 65 78 65 63 73 71 6c 5f 69 36 34  ax = execsql_i64
1a90: 28 26 65 72 72 2c 20 26 64 62 2c 20 22 53 45 4c  (&err, &db, "SEL
1aa0: 45 43 54 20 6d 61 78 28 63 6e 74 29 20 46 52 4f  ECT max(cnt) FRO
1ab0: 4d 20 74 31 22 29 3b 0a 20 20 20 20 20 20 69 66  M t1");.      if
1ac0: 28 20 69 4d 61 78 2b 31 3d 3d 69 4e 65 78 74 57  ( iMax+1==iNextW
1ad0: 72 69 74 65 20 29 20 62 72 65 61 6b 3b 0a 20 20  rite ) break;.  
1ae0: 20 20 7d 0a 20 20 20 20 69 66 28 20 73 74 6f 70    }.    if( stop
1af0: 20 29 20 62 72 65 61 6b 3b 0a 0a 20 20 20 20 73   ) break;..    s
1b00: 75 6d 31 20 3d 20 65 78 65 63 73 71 6c 5f 69 36  um1 = execsql_i6
1b10: 34 28 26 65 72 72 2c 20 26 64 62 2c 20 22 53 45  4(&err, &db, "SE
1b20: 4c 45 43 54 20 73 75 6d 28 63 6e 74 29 20 46 52  LECT sum(cnt) FR
1b30: 4f 4d 20 74 31 22 29 3b 0a 20 20 20 20 73 75 6d  OM t1");.    sum
1b40: 32 20 3d 20 65 78 65 63 73 71 6c 5f 69 36 34 28  2 = execsql_i64(
1b50: 26 65 72 72 2c 20 26 64 62 2c 20 22 53 45 4c 45  &err, &db, "SELE
1b60: 43 54 20 73 75 6d 28 73 75 6d 31 29 20 46 52 4f  CT sum(sum1) FRO
1b70: 4d 20 74 31 22 29 3b 0a 20 20 20 20 65 78 65 63  M t1");.    exec
1b80: 73 71 6c 5f 69 36 34 28 26 65 72 72 2c 20 26 64  sql_i64(&err, &d
1b90: 62 2c 20 0a 20 20 20 20 20 20 20 20 22 49 4e 53  b, .        "INS
1ba0: 45 52 54 20 49 4e 54 4f 20 74 31 20 56 41 4c 55  ERT INTO t1 VALU
1bb0: 45 53 28 3a 69 4e 65 78 74 57 72 69 74 65 2c 20  ES(:iNextWrite, 
1bc0: 3a 69 53 75 6d 31 2c 20 3a 69 53 75 6d 32 29 22  :iSum1, :iSum2)"
1bd0: 2c 0a 20 20 20 20 20 20 20 20 26 69 4e 65 78 74  ,.        &iNext
1be0: 57 72 69 74 65 2c 20 26 73 75 6d 31 2c 20 26 73  Write, &sum1, &s
1bf0: 75 6d 32 0a 20 20 20 20 29 3b 0a 20 20 20 20 69  um2.    );.    i
1c00: 6e 74 65 67 72 69 74 79 5f 63 68 65 63 6b 28 26  ntegrity_check(&
1c10: 65 72 72 2c 20 26 64 62 29 3b 0a 0a 20 20 20 20  err, &db);..    
1c20: 69 4e 65 78 74 57 72 69 74 65 20 2b 3d 20 57 41  iNextWrite += WA
1c30: 4c 54 48 52 45 41 44 33 5f 4e 54 48 52 45 41 44  LTHREAD3_NTHREAD
1c40: 3b 0a 20 20 7d 0a 0a 20 20 63 6c 6f 73 65 64 62  ;.  }..  closedb
1c50: 28 26 65 72 72 2c 20 26 64 62 29 3b 0a 20 20 70  (&err, &db);.  p
1c60: 72 69 6e 74 5f 61 6e 64 5f 66 72 65 65 5f 65 72  rint_and_free_er
1c70: 72 28 26 65 72 72 29 3b 0a 20 20 72 65 74 75 72  r(&err);.  retur
1c80: 6e 20 30 3b 0a 7d 0a 0a 73 74 61 74 69 63 20 76  n 0;.}..static v
1c90: 6f 69 64 20 77 61 6c 74 68 72 65 61 64 33 28 69  oid walthread3(i
1ca0: 6e 74 20 6e 4d 73 29 7b 0a 20 20 45 72 72 6f 72  nt nMs){.  Error
1cb0: 20 65 72 72 20 3d 20 7b 30 7d 3b 0a 20 20 53 71   err = {0};.  Sq
1cc0: 6c 69 74 65 20 64 62 20 3d 20 7b 30 7d 3b 0a 20  lite db = {0};. 
1cd0: 20 54 68 72 65 61 64 73 65 74 20 74 68 72 65 61   Threadset threa
1ce0: 64 73 20 3d 20 7b 30 7d 3b 0a 20 20 69 6e 74 20  ds = {0};.  int 
1cf0: 69 3b 0a 0a 20 20 6f 70 65 6e 64 62 28 26 65 72  i;..  opendb(&er
1d00: 72 2c 20 26 64 62 2c 20 22 74 65 73 74 2e 64 62  r, &db, "test.db
1d10: 22 2c 20 31 29 3b 0a 20 20 73 71 6c 5f 73 63 72  ", 1);.  sql_scr
1d20: 69 70 74 28 26 65 72 72 2c 20 26 64 62 2c 20 0a  ipt(&err, &db, .
1d30: 20 20 20 20 20 20 22 50 52 41 47 4d 41 20 6a 6f        "PRAGMA jo
1d40: 75 72 6e 61 6c 5f 6d 6f 64 65 20 3d 20 57 41 4c  urnal_mode = WAL
1d50: 3b 22 0a 20 20 20 20 20 20 22 43 52 45 41 54 45  ;".      "CREATE
1d60: 20 54 41 42 4c 45 20 74 31 28 63 6e 74 20 50 52   TABLE t1(cnt PR
1d70: 49 4d 41 52 59 20 4b 45 59 2c 20 73 75 6d 31 2c  IMARY KEY, sum1,
1d80: 20 73 75 6d 32 29 3b 22 0a 20 20 20 20 20 20 22   sum2);".      "
1d90: 43 52 45 41 54 45 20 49 4e 44 45 58 20 69 31 20  CREATE INDEX i1 
1da0: 4f 4e 20 74 31 28 73 75 6d 31 29 3b 22 0a 20 20  ON t1(sum1);".  
1db0: 20 20 20 20 22 43 52 45 41 54 45 20 49 4e 44 45      "CREATE INDE
1dc0: 58 20 69 32 20 4f 4e 20 74 31 28 73 75 6d 32 29  X i2 ON t1(sum2)
1dd0: 3b 22 0a 20 20 20 20 20 20 22 49 4e 53 45 52 54  ;".      "INSERT
1de0: 20 49 4e 54 4f 20 74 31 20 56 41 4c 55 45 53 28   INTO t1 VALUES(
1df0: 30 2c 20 30 2c 20 30 29 3b 22 0a 20 20 29 3b 0a  0, 0, 0);".  );.
1e00: 20 20 63 6c 6f 73 65 64 62 28 26 65 72 72 2c 20    closedb(&err, 
1e10: 26 64 62 29 3b 0a 0a 20 20 73 65 74 73 74 6f 70  &db);..  setstop
1e20: 74 69 6d 65 28 26 65 72 72 2c 20 6e 4d 73 29 3b  time(&err, nMs);
1e30: 0a 20 20 66 6f 72 28 69 3d 30 3b 20 69 3c 57 41  .  for(i=0; i<WA
1e40: 4c 54 48 52 45 41 44 33 5f 4e 54 48 52 45 41 44  LTHREAD3_NTHREAD
1e50: 3b 20 69 2b 2b 29 7b 0a 20 20 20 20 6c 61 75 6e  ; i++){.    laun
1e60: 63 68 5f 74 68 72 65 61 64 28 26 65 72 72 2c 20  ch_thread(&err, 
1e70: 26 74 68 72 65 61 64 73 2c 20 77 61 6c 74 68 72  &threads, walthr
1e80: 65 61 64 33 5f 74 68 72 65 61 64 2c 20 49 4e 54  ead3_thread, INT
1e90: 32 50 54 52 28 69 29 29 3b 0a 20 20 7d 0a 20 20  2PTR(i));.  }.  
1ea0: 6a 6f 69 6e 5f 61 6c 6c 5f 74 68 72 65 61 64 73  join_all_threads
1eb0: 28 26 65 72 72 2c 20 26 74 68 72 65 61 64 73 29  (&err, &threads)
1ec0: 3b 0a 0a 20 20 70 72 69 6e 74 5f 61 6e 64 5f 66  ;..  print_and_f
1ed0: 72 65 65 5f 65 72 72 28 26 65 72 72 29 3b 0a 7d  ree_err(&err);.}
1ee0: 0a 0a 73 74 61 74 69 63 20 63 68 61 72 20 2a 77  ..static char *w
1ef0: 61 6c 74 68 72 65 61 64 34 5f 72 65 61 64 65 72  althread4_reader
1f00: 5f 74 68 72 65 61 64 28 69 6e 74 20 69 54 69 64  _thread(int iTid
1f10: 2c 20 76 6f 69 64 20 2a 70 41 72 67 29 7b 0a 20  , void *pArg){. 
1f20: 20 45 72 72 6f 72 20 65 72 72 20 3d 20 7b 30 7d   Error err = {0}
1f30: 3b 20 20 20 20 20 20 20 20 20 20 20 20 20 20 20  ;               
1f40: 20 2f 2a 20 45 72 72 6f 72 20 63 6f 64 65 20 61   /* Error code a
1f50: 6e 64 20 6d 65 73 73 61 67 65 20 2a 2f 0a 20 20  nd message */.  
1f60: 53 71 6c 69 74 65 20 64 62 20 3d 20 7b 30 7d 3b  Sqlite db = {0};
1f70: 20 20 20 20 20 20 20 20 20 20 20 20 20 20 20 20                  
1f80: 2f 2a 20 53 51 4c 69 74 65 20 64 61 74 61 62 61  /* SQLite databa
1f90: 73 65 20 63 6f 6e 6e 65 63 74 69 6f 6e 20 2a 2f  se connection */
1fa0: 0a 0a 20 20 6f 70 65 6e 64 62 28 26 65 72 72 2c  ..  opendb(&err,
1fb0: 20 26 64 62 2c 20 22 74 65 73 74 2e 64 62 22 2c   &db, "test.db",
1fc0: 20 30 29 3b 0a 20 20 77 68 69 6c 65 28 20 21 74   0);.  while( !t
1fd0: 69 6d 65 74 6f 73 74 6f 70 28 26 65 72 72 29 20  imetostop(&err) 
1fe0: 29 7b 0a 20 20 20 20 69 6e 74 65 67 72 69 74 79  ){.    integrity
1ff0: 5f 63 68 65 63 6b 28 26 65 72 72 2c 20 26 64 62  _check(&err, &db
2000: 29 3b 0a 20 20 7d 0a 20 20 63 6c 6f 73 65 64 62  );.  }.  closedb
2010: 28 26 65 72 72 2c 20 26 64 62 29 3b 0a 0a 20 20  (&err, &db);..  
2020: 70 72 69 6e 74 5f 61 6e 64 5f 66 72 65 65 5f 65  print_and_free_e
2030: 72 72 28 26 65 72 72 29 3b 0a 20 20 72 65 74 75  rr(&err);.  retu
2040: 72 6e 20 30 3b 0a 7d 0a 0a 73 74 61 74 69 63 20  rn 0;.}..static 
2050: 63 68 61 72 20 2a 77 61 6c 74 68 72 65 61 64 34  char *walthread4
2060: 5f 77 72 69 74 65 72 5f 74 68 72 65 61 64 28 69  _writer_thread(i
2070: 6e 74 20 69 54 69 64 2c 20 76 6f 69 64 20 2a 70  nt iTid, void *p
2080: 41 72 67 29 7b 0a 20 20 45 72 72 6f 72 20 65 72  Arg){.  Error er
2090: 72 20 3d 20 7b 30 7d 3b 20 20 20 20 20 20 20 20  r = {0};        
20a0: 20 20 20 20 20 20 20 20 2f 2a 20 45 72 72 6f 72          /* Error
20b0: 20 63 6f 64 65 20 61 6e 64 20 6d 65 73 73 61 67   code and messag
20c0: 65 20 2a 2f 0a 20 20 53 71 6c 69 74 65 20 64 62  e */.  Sqlite db
20d0: 20 3d 20 7b 30 7d 3b 20 20 20 20 20 20 20 20 20   = {0};         
20e0: 20 20 20 20 20 20 20 2f 2a 20 53 51 4c 69 74 65         /* SQLite
20f0: 20 64 61 74 61 62 61 73 65 20 63 6f 6e 6e 65 63   database connec
2100: 74 69 6f 6e 20 2a 2f 0a 20 20 69 36 34 20 69 52  tion */.  i64 iR
2110: 6f 77 20 3d 20 31 3b 0a 0a 20 20 6f 70 65 6e 64  ow = 1;..  opend
2120: 62 28 26 65 72 72 2c 20 26 64 62 2c 20 22 74 65  b(&err, &db, "te
2130: 73 74 2e 64 62 22 2c 20 30 29 3b 0a 20 20 73 71  st.db", 0);.  sq
2140: 6c 5f 73 63 72 69 70 74 28 26 65 72 72 2c 20 26  l_script(&err, &
2150: 64 62 2c 20 22 50 52 41 47 4d 41 20 77 61 6c 5f  db, "PRAGMA wal_
2160: 61 75 74 6f 63 68 65 63 6b 70 6f 69 6e 74 20 3d  autocheckpoint =
2170: 20 31 35 3b 22 29 3b 0a 20 20 77 68 69 6c 65 28   15;");.  while(
2180: 20 21 74 69 6d 65 74 6f 73 74 6f 70 28 26 65 72   !timetostop(&er
2190: 72 29 20 29 7b 0a 20 20 20 20 65 78 65 63 73 71  r) ){.    execsq
21a0: 6c 5f 69 36 34 28 0a 20 20 20 20 20 20 20 20 26  l_i64(.        &
21b0: 65 72 72 2c 20 26 64 62 2c 20 22 52 45 50 4c 41  err, &db, "REPLA
21c0: 43 45 20 49 4e 54 4f 20 74 31 20 56 41 4c 55 45  CE INTO t1 VALUE
21d0: 53 28 3a 69 52 6f 77 2c 20 72 61 6e 64 6f 6d 62  S(:iRow, randomb
21e0: 6c 6f 62 28 33 30 30 29 29 22 2c 20 26 69 52 6f  lob(300))", &iRo
21f0: 77 0a 20 20 20 20 29 3b 0a 20 20 20 20 69 52 6f  w.    );.    iRo
2200: 77 2b 2b 3b 0a 20 20 20 20 69 66 28 20 69 52 6f  w++;.    if( iRo
2210: 77 3d 3d 31 30 20 29 20 69 52 6f 77 20 3d 20 30  w==10 ) iRow = 0
2220: 3b 0a 20 20 7d 0a 20 20 63 6c 6f 73 65 64 62 28  ;.  }.  closedb(
2230: 26 65 72 72 2c 20 26 64 62 29 3b 0a 0a 20 20 70  &err, &db);..  p
2240: 72 69 6e 74 5f 61 6e 64 5f 66 72 65 65 5f 65 72  rint_and_free_er
2250: 72 28 26 65 72 72 29 3b 0a 20 20 72 65 74 75 72  r(&err);.  retur
2260: 6e 20 30 3b 0a 7d 0a 0a 73 74 61 74 69 63 20 76  n 0;.}..static v
2270: 6f 69 64 20 77 61 6c 74 68 72 65 61 64 34 28 69  oid walthread4(i
2280: 6e 74 20 6e 4d 73 29 7b 0a 20 20 45 72 72 6f 72  nt nMs){.  Error
2290: 20 65 72 72 20 3d 20 7b 30 7d 3b 0a 20 20 53 71   err = {0};.  Sq
22a0: 6c 69 74 65 20 64 62 20 3d 20 7b 30 7d 3b 0a 20  lite db = {0};. 
22b0: 20 54 68 72 65 61 64 73 65 74 20 74 68 72 65 61   Threadset threa
22c0: 64 73 20 3d 20 7b 30 7d 3b 0a 0a 20 20 6f 70 65  ds = {0};..  ope
22d0: 6e 64 62 28 26 65 72 72 2c 20 26 64 62 2c 20 22  ndb(&err, &db, "
22e0: 74 65 73 74 2e 64 62 22 2c 20 31 29 3b 0a 20 20  test.db", 1);.  
22f0: 73 71 6c 5f 73 63 72 69 70 74 28 26 65 72 72 2c  sql_script(&err,
2300: 20 26 64 62 2c 20 0a 20 20 20 20 20 20 22 50 52   &db, .      "PR
2310: 41 47 4d 41 20 6a 6f 75 72 6e 61 6c 5f 6d 6f 64  AGMA journal_mod
2320: 65 20 3d 20 57 41 4c 3b 22 0a 20 20 20 20 20 20  e = WAL;".      
2330: 22 43 52 45 41 54 45 20 54 41 42 4c 45 20 74 31  "CREATE TABLE t1
2340: 28 61 20 49 4e 54 45 47 45 52 20 50 52 49 4d 41  (a INTEGER PRIMA
2350: 52 59 20 4b 45 59 2c 20 62 20 55 4e 49 51 55 45  RY KEY, b UNIQUE
2360: 29 3b 22 0a 20 20 29 3b 0a 20 20 63 6c 6f 73 65  );".  );.  close
2370: 64 62 28 26 65 72 72 2c 20 26 64 62 29 3b 0a 0a  db(&err, &db);..
2380: 20 20 73 65 74 73 74 6f 70 74 69 6d 65 28 26 65    setstoptime(&e
2390: 72 72 2c 20 6e 4d 73 29 3b 0a 20 20 6c 61 75 6e  rr, nMs);.  laun
23a0: 63 68 5f 74 68 72 65 61 64 28 26 65 72 72 2c 20  ch_thread(&err, 
23b0: 26 74 68 72 65 61 64 73 2c 20 77 61 6c 74 68 72  &threads, walthr
23c0: 65 61 64 34 5f 72 65 61 64 65 72 5f 74 68 72 65  ead4_reader_thre
23d0: 61 64 2c 20 30 29 3b 0a 20 20 6c 61 75 6e 63 68  ad, 0);.  launch
23e0: 5f 74 68 72 65 61 64 28 26 65 72 72 2c 20 26 74  _thread(&err, &t
23f0: 68 72 65 61 64 73 2c 20 77 61 6c 74 68 72 65 61  hreads, walthrea
2400: 64 34 5f 77 72 69 74 65 72 5f 74 68 72 65 61 64  d4_writer_thread
2410: 2c 20 30 29 3b 0a 20 20 6a 6f 69 6e 5f 61 6c 6c  , 0);.  join_all
2420: 5f 74 68 72 65 61 64 73 28 26 65 72 72 2c 20 26  _threads(&err, &
2430: 74 68 72 65 61 64 73 29 3b 0a 0a 20 20 70 72 69  threads);..  pri
2440: 6e 74 5f 61 6e 64 5f 66 72 65 65 5f 65 72 72 28  nt_and_free_err(
2450: 26 65 72 72 29 3b 0a 7d 0a 0a 73 74 61 74 69 63  &err);.}..static
2460: 20 63 68 61 72 20 2a 77 61 6c 74 68 72 65 61 64   char *walthread
2470: 35 5f 74 68 72 65 61 64 28 69 6e 74 20 69 54 69  5_thread(int iTi
2480: 64 2c 20 76 6f 69 64 20 2a 70 41 72 67 29 7b 0a  d, void *pArg){.
2490: 20 20 45 72 72 6f 72 20 65 72 72 20 3d 20 7b 30    Error err = {0
24a0: 7d 3b 20 20 20 20 20 20 20 20 20 20 20 20 20 20  };              
24b0: 20 20 2f 2a 20 45 72 72 6f 72 20 63 6f 64 65 20    /* Error code 
24c0: 61 6e 64 20 6d 65 73 73 61 67 65 20 2a 2f 0a 20  and message */. 
24d0: 20 53 71 6c 69 74 65 20 64 62 20 3d 20 7b 30 7d   Sqlite db = {0}
24e0: 3b 20 20 20 20 20 20 20 20 20 20 20 20 20 20 20  ;               
24f0: 20 2f 2a 20 53 51 4c 69 74 65 20 64 61 74 61 62   /* SQLite datab
2500: 61 73 65 20 63 6f 6e 6e 65 63 74 69 6f 6e 20 2a  ase connection *
2510: 2f 0a 20 20 69 36 34 20 6e 52 6f 77 3b 0a 0a 20  /.  i64 nRow;.. 
2520: 20 6f 70 65 6e 64 62 28 26 65 72 72 2c 20 26 64   opendb(&err, &d
2530: 62 2c 20 22 74 65 73 74 2e 64 62 22 2c 20 30 29  b, "test.db", 0)
2540: 3b 0a 20 20 6e 52 6f 77 20 3d 20 65 78 65 63 73  ;.  nRow = execs
2550: 71 6c 5f 69 36 34 28 26 65 72 72 2c 20 26 64 62  ql_i64(&err, &db
2560: 2c 20 22 53 45 4c 45 43 54 20 63 6f 75 6e 74 28  , "SELECT count(
2570: 2a 29 20 46 52 4f 4d 20 74 31 22 29 3b 0a 20 20  *) FROM t1");.  
2580: 63 6c 6f 73 65 64 62 28 26 65 72 72 2c 20 26 64  closedb(&err, &d
2590: 62 29 3b 0a 0a 20 20 69 66 28 20 6e 52 6f 77 21  b);..  if( nRow!
25a0: 3d 36 35 35 33 36 20 29 20 74 65 73 74 5f 65 72  =65536 ) test_er
25b0: 72 6f 72 28 26 65 72 72 2c 20 22 42 61 64 20 72  ror(&err, "Bad r
25c0: 6f 77 20 63 6f 75 6e 74 3a 20 25 64 22 2c 20 28  ow count: %d", (
25d0: 69 6e 74 29 6e 52 6f 77 29 3b 0a 20 20 70 72 69  int)nRow);.  pri
25e0: 6e 74 5f 61 6e 64 5f 66 72 65 65 5f 65 72 72 28  nt_and_free_err(
25f0: 26 65 72 72 29 3b 0a 20 20 72 65 74 75 72 6e 20  &err);.  return 
2600: 30 3b 0a 7d 0a 73 74 61 74 69 63 20 76 6f 69 64  0;.}.static void
2610: 20 77 61 6c 74 68 72 65 61 64 35 28 69 6e 74 20   walthread5(int 
2620: 6e 4d 73 29 7b 0a 20 20 45 72 72 6f 72 20 65 72  nMs){.  Error er
2630: 72 20 3d 20 7b 30 7d 3b 0a 20 20 53 71 6c 69 74  r = {0};.  Sqlit
2640: 65 20 64 62 20 3d 20 7b 30 7d 3b 0a 20 20 54 68  e db = {0};.  Th
2650: 72 65 61 64 73 65 74 20 74 68 72 65 61 64 73 20  readset threads 
2660: 3d 20 7b 30 7d 3b 0a 0a 20 20 6f 70 65 6e 64 62  = {0};..  opendb
2670: 28 26 65 72 72 2c 20 26 64 62 2c 20 22 74 65 73  (&err, &db, "tes
2680: 74 2e 64 62 22 2c 20 31 29 3b 0a 20 20 73 71 6c  t.db", 1);.  sql
2690: 5f 73 63 72 69 70 74 28 26 65 72 72 2c 20 26 64  _script(&err, &d
26a0: 62 2c 20 0a 20 20 20 20 20 20 22 50 52 41 47 4d  b, .      "PRAGM
26b0: 41 20 77 61 6c 5f 61 75 74 6f 63 68 65 63 6b 70  A wal_autocheckp
26c0: 6f 69 6e 74 20 3d 20 30 3b 22 0a 20 20 20 20 20  oint = 0;".     
26d0: 20 22 50 52 41 47 4d 41 20 70 61 67 65 5f 73 69   "PRAGMA page_si
26e0: 7a 65 20 3d 20 31 30 32 34 3b 22 0a 20 20 20 20  ze = 1024;".    
26f0: 20 20 22 50 52 41 47 4d 41 20 6a 6f 75 72 6e 61    "PRAGMA journa
2700: 6c 5f 6d 6f 64 65 20 3d 20 57 41 4c 3b 22 0a 20  l_mode = WAL;". 
2710: 20 20 20 20 20 22 43 52 45 41 54 45 20 54 41 42       "CREATE TAB
2720: 4c 45 20 74 31 28 78 29 3b 22 0a 20 20 20 20 20  LE t1(x);".     
2730: 20 22 42 45 47 49 4e 3b 22 0a 20 20 20 20 20 20   "BEGIN;".      
2740: 22 49 4e 53 45 52 54 20 49 4e 54 4f 20 74 31 20  "INSERT INTO t1 
2750: 56 41 4c 55 45 53 28 72 61 6e 64 6f 6d 62 6c 6f  VALUES(randomblo
2760: 62 28 39 30 30 29 29 3b 22 0a 20 20 20 20 20 20  b(900));".      
2770: 22 49 4e 53 45 52 54 20 49 4e 54 4f 20 74 31 20  "INSERT INTO t1 
2780: 53 45 4c 45 43 54 20 72 61 6e 64 6f 6d 62 6c 6f  SELECT randomblo
2790: 62 28 39 30 30 29 20 46 52 4f 4d 20 74 31 3b 20  b(900) FROM t1; 
27a0: 20 20 20 20 20 2f 2a 20 20 20 20 20 32 20 2a 2f       /*     2 */
27b0: 22 0a 20 20 20 20 20 20 22 49 4e 53 45 52 54 20  ".      "INSERT 
27c0: 49 4e 54 4f 20 74 31 20 53 45 4c 45 43 54 20 72  INTO t1 SELECT r
27d0: 61 6e 64 6f 6d 62 6c 6f 62 28 39 30 30 29 20 46  andomblob(900) F
27e0: 52 4f 4d 20 74 31 3b 20 20 20 20 20 20 2f 2a 20  ROM t1;      /* 
27f0: 20 20 20 20 34 20 2a 2f 22 0a 20 20 20 20 20 20      4 */".      
2800: 22 49 4e 53 45 52 54 20 49 4e 54 4f 20 74 31 20  "INSERT INTO t1 
2810: 53 45 4c 45 43 54 20 72 61 6e 64 6f 6d 62 6c 6f  SELECT randomblo
2820: 62 28 39 30 30 29 20 46 52 4f 4d 20 74 31 3b 20  b(900) FROM t1; 
2830: 20 20 20 20 20 2f 2a 20 20 20 20 20 38 20 2a 2f       /*     8 */
2840: 22 0a 20 20 20 20 20 20 22 49 4e 53 45 52 54 20  ".      "INSERT 
2850: 49 4e 54 4f 20 74 31 20 53 45 4c 45 43 54 20 72  INTO t1 SELECT r
2860: 61 6e 64 6f 6d 62 6c 6f 62 28 39 30 30 29 20 46  andomblob(900) F
2870: 52 4f 4d 20 74 31 3b 20 20 20 20 20 20 2f 2a 20  ROM t1;      /* 
2880: 20 20 20 31 36 20 2a 2f 22 0a 20 20 20 20 20 20     16 */".      
2890: 22 49 4e 53 45 52 54 20 49 4e 54 4f 20 74 31 20  "INSERT INTO t1 
28a0: 53 45 4c 45 43 54 20 72 61 6e 64 6f 6d 62 6c 6f  SELECT randomblo
28b0: 62 28 39 30 30 29 20 46 52 4f 4d 20 74 31 3b 20  b(900) FROM t1; 
28c0: 20 20 20 20 20 2f 2a 20 20 20 20 33 32 20 2a 2f       /*    32 */
28d0: 22 0a 20 20 20 20 20 20 22 49 4e 53 45 52 54 20  ".      "INSERT 
28e0: 49 4e 54 4f 20 74 31 20 53 45 4c 45 43 54 20 72  INTO t1 SELECT r
28f0: 61 6e 64 6f 6d 62 6c 6f 62 28 39 30 30 29 20 46  andomblob(900) F
2900: 52 4f 4d 20 74 31 3b 20 20 20 20 20 20 2f 2a 20  ROM t1;      /* 
2910: 20 20 20 36 34 20 2a 2f 22 0a 20 20 20 20 20 20     64 */".      
2920: 22 49 4e 53 45 52 54 20 49 4e 54 4f 20 74 31 20  "INSERT INTO t1 
2930: 53 45 4c 45 43 54 20 72 61 6e 64 6f 6d 62 6c 6f  SELECT randomblo
2940: 62 28 39 30 30 29 20 46 52 4f 4d 20 74 31 3b 20  b(900) FROM t1; 
2950: 20 20 20 20 20 2f 2a 20 20 20 31 32 38 20 2a 2f       /*   128 */
2960: 22 0a 20 20 20 20 20 20 22 49 4e 53 45 52 54 20  ".      "INSERT 
2970: 49 4e 54 4f 20 74 31 20 53 45 4c 45 43 54 20 72  INTO t1 SELECT r
2980: 61 6e 64 6f 6d 62 6c 6f 62 28 39 30 30 29 20 46  andomblob(900) F
2990: 52 4f 4d 20 74 31 3b 20 20 20 20 20 20 2f 2a 20  ROM t1;      /* 
29a0: 20 20 32 35 36 20 2a 2f 22 0a 20 20 20 20 20 20    256 */".      
29b0: 22 49 4e 53 45 52 54 20 49 4e 54 4f 20 74 31 20  "INSERT INTO t1 
29c0: 53 45 4c 45 43 54 20 72 61 6e 64 6f 6d 62 6c 6f  SELECT randomblo
29d0: 62 28 39 30 30 29 20 46 52 4f 4d 20 74 31 3b 20  b(900) FROM t1; 
29e0: 20 20 20 20 20 2f 2a 20 20 20 35 31 32 20 2a 2f       /*   512 */
29f0: 22 0a 20 20 20 20 20 20 22 49 4e 53 45 52 54 20  ".      "INSERT 
2a00: 49 4e 54 4f 20 74 31 20 53 45 4c 45 43 54 20 72  INTO t1 SELECT r
2a10: 61 6e 64 6f 6d 62 6c 6f 62 28 39 30 30 29 20 46  andomblob(900) F
2a20: 52 4f 4d 20 74 31 3b 20 20 20 20 20 20 2f 2a 20  ROM t1;      /* 
2a30: 20 31 30 32 34 20 2a 2f 22 0a 20 20 20 20 20 20   1024 */".      
2a40: 22 49 4e 53 45 52 54 20 49 4e 54 4f 20 74 31 20  "INSERT INTO t1 
2a50: 53 45 4c 45 43 54 20 72 61 6e 64 6f 6d 62 6c 6f  SELECT randomblo
2a60: 62 28 39 30 30 29 20 46 52 4f 4d 20 74 31 3b 20  b(900) FROM t1; 
2a70: 20 20 20 20 20 2f 2a 20 20 32 30 34 38 20 2a 2f       /*  2048 */
2a80: 22 0a 20 20 20 20 20 20 22 49 4e 53 45 52 54 20  ".      "INSERT 
2a90: 49 4e 54 4f 20 74 31 20 53 45 4c 45 43 54 20 72  INTO t1 SELECT r
2aa0: 61 6e 64 6f 6d 62 6c 6f 62 28 39 30 30 29 20 46  andomblob(900) F
2ab0: 52 4f 4d 20 74 31 3b 20 20 20 20 20 20 2f 2a 20  ROM t1;      /* 
2ac0: 20 34 30 39 36 20 2a 2f 22 0a 20 20 20 20 20 20   4096 */".      
2ad0: 22 49 4e 53 45 52 54 20 49 4e 54 4f 20 74 31 20  "INSERT INTO t1 
2ae0: 53 45 4c 45 43 54 20 72 61 6e 64 6f 6d 62 6c 6f  SELECT randomblo
2af0: 62 28 39 30 30 29 20 46 52 4f 4d 20 74 31 3b 20  b(900) FROM t1; 
2b00: 20 20 20 20 20 2f 2a 20 20 38 31 39 32 20 2a 2f       /*  8192 */
2b10: 22 0a 20 20 20 20 20 20 22 49 4e 53 45 52 54 20  ".      "INSERT 
2b20: 49 4e 54 4f 20 74 31 20 53 45 4c 45 43 54 20 72  INTO t1 SELECT r
2b30: 61 6e 64 6f 6d 62 6c 6f 62 28 39 30 30 29 20 46  andomblob(900) F
2b40: 52 4f 4d 20 74 31 3b 20 20 20 20 20 20 2f 2a 20  ROM t1;      /* 
2b50: 31 36 33 38 34 20 2a 2f 22 0a 20 20 20 20 20 20  16384 */".      
2b60: 22 49 4e 53 45 52 54 20 49 4e 54 4f 20 74 31 20  "INSERT INTO t1 
2b70: 53 45 4c 45 43 54 20 72 61 6e 64 6f 6d 62 6c 6f  SELECT randomblo
2b80: 62 28 39 30 30 29 20 46 52 4f 4d 20 74 31 3b 20  b(900) FROM t1; 
2b90: 20 20 20 20 20 2f 2a 20 33 32 37 36 38 20 2a 2f       /* 32768 */
2ba0: 22 0a 20 20 20 20 20 20 22 49 4e 53 45 52 54 20  ".      "INSERT 
2bb0: 49 4e 54 4f 20 74 31 20 53 45 4c 45 43 54 20 72  INTO t1 SELECT r
2bc0: 61 6e 64 6f 6d 62 6c 6f 62 28 39 30 30 29 20 46  andomblob(900) F
2bd0: 52 4f 4d 20 74 31 3b 20 20 20 20 20 20 2f 2a 20  ROM t1;      /* 
2be0: 36 35 35 33 36 20 2a 2f 22 0a 20 20 20 20 20 20  65536 */".      
2bf0: 22 43 4f 4d 4d 49 54 3b 22 0a 20 20 29 3b 0a 20  "COMMIT;".  );. 
2c00: 20 66 69 6c 65 63 6f 70 79 28 26 65 72 72 2c 20   filecopy(&err, 
2c10: 22 74 65 73 74 2e 64 62 22 2c 20 22 74 65 73 74  "test.db", "test
2c20: 5f 73 76 2e 64 62 22 29 3b 0a 20 20 66 69 6c 65  _sv.db");.  file
2c30: 63 6f 70 79 28 26 65 72 72 2c 20 22 74 65 73 74  copy(&err, "test
2c40: 2e 64 62 2d 77 61 6c 22 2c 20 22 74 65 73 74 5f  .db-wal", "test_
2c50: 73 76 2e 64 62 2d 77 61 6c 22 29 3b 0a 20 20 63  sv.db-wal");.  c
2c60: 6c 6f 73 65 64 62 28 26 65 72 72 2c 20 26 64 62  losedb(&err, &db
2c70: 29 3b 0a 0a 20 20 66 69 6c 65 63 6f 70 79 28 26  );..  filecopy(&
2c80: 65 72 72 2c 20 22 74 65 73 74 5f 73 76 2e 64 62  err, "test_sv.db
2c90: 22 2c 20 22 74 65 73 74 2e 64 62 22 29 3b 0a 20  ", "test.db");. 
2ca0: 20 66 69 6c 65 63 6f 70 79 28 26 65 72 72 2c 20   filecopy(&err, 
2cb0: 22 74 65 73 74 5f 73 76 2e 64 62 2d 77 61 6c 22  "test_sv.db-wal"
2cc0: 2c 20 22 74 65 73 74 2e 64 62 2d 77 61 6c 22 29  , "test.db-wal")
2cd0: 3b 0a 0a 20 20 69 66 28 20 65 72 72 2e 72 63 3d  ;..  if( err.rc=
2ce0: 3d 53 51 4c 49 54 45 5f 4f 4b 20 29 7b 0a 20 20  =SQLITE_OK ){.  
2cf0: 20 20 70 72 69 6e 74 66 28 22 20 20 57 41 4c 20    printf("  WAL 
2d00: 66 69 6c 65 20 69 73 20 25 64 20 62 79 74 65 73  file is %d bytes
2d10: 2c 22 2c 20 28 69 6e 74 29 66 69 6c 65 73 69 7a  ,", (int)filesiz
2d20: 65 28 26 65 72 72 2c 22 74 65 73 74 2e 64 62 2d  e(&err,"test.db-
2d30: 77 61 6c 22 29 29 3b 0a 20 20 20 20 70 72 69 6e  wal"));.    prin
2d40: 74 66 28 22 20 44 42 20 66 69 6c 65 20 69 73 20  tf(" DB file is 
2d50: 25 64 2e 5c 6e 22 2c 20 28 69 6e 74 29 66 69 6c  %d.\n", (int)fil
2d60: 65 73 69 7a 65 28 26 65 72 72 2c 22 74 65 73 74  esize(&err,"test
2d70: 2e 64 62 22 29 29 3b 0a 20 20 7d 0a 0a 20 20 73  .db"));.  }..  s
2d80: 65 74 73 74 6f 70 74 69 6d 65 28 26 65 72 72 2c  etstoptime(&err,
2d90: 20 6e 4d 73 29 3b 0a 20 20 6c 61 75 6e 63 68 5f   nMs);.  launch_
2da0: 74 68 72 65 61 64 28 26 65 72 72 2c 20 26 74 68  thread(&err, &th
2db0: 72 65 61 64 73 2c 20 77 61 6c 74 68 72 65 61 64  reads, walthread
2dc0: 35 5f 74 68 72 65 61 64 2c 20 30 29 3b 0a 20 20  5_thread, 0);.  
2dd0: 6c 61 75 6e 63 68 5f 74 68 72 65 61 64 28 26 65  launch_thread(&e
2de0: 72 72 2c 20 26 74 68 72 65 61 64 73 2c 20 77 61  rr, &threads, wa
2df0: 6c 74 68 72 65 61 64 35 5f 74 68 72 65 61 64 2c  lthread5_thread,
2e00: 20 30 29 3b 0a 20 20 6c 61 75 6e 63 68 5f 74 68   0);.  launch_th
2e10: 72 65 61 64 28 26 65 72 72 2c 20 26 74 68 72 65  read(&err, &thre
2e20: 61 64 73 2c 20 77 61 6c 74 68 72 65 61 64 35 5f  ads, walthread5_
2e30: 74 68 72 65 61 64 2c 20 30 29 3b 0a 20 20 6c 61  thread, 0);.  la
2e40: 75 6e 63 68 5f 74 68 72 65 61 64 28 26 65 72 72  unch_thread(&err
2e50: 2c 20 26 74 68 72 65 61 64 73 2c 20 77 61 6c 74  , &threads, walt
2e60: 68 72 65 61 64 35 5f 74 68 72 65 61 64 2c 20 30  hread5_thread, 0
2e70: 29 3b 0a 20 20 6c 61 75 6e 63 68 5f 74 68 72 65  );.  launch_thre
2e80: 61 64 28 26 65 72 72 2c 20 26 74 68 72 65 61 64  ad(&err, &thread
2e90: 73 2c 20 77 61 6c 74 68 72 65 61 64 35 5f 74 68  s, walthread5_th
2ea0: 72 65 61 64 2c 20 30 29 3b 0a 20 20 6a 6f 69 6e  read, 0);.  join
2eb0: 5f 61 6c 6c 5f 74 68 72 65 61 64 73 28 26 65 72  _all_threads(&er
2ec0: 72 2c 20 26 74 68 72 65 61 64 73 29 3b 0a 0a 20  r, &threads);.. 
2ed0: 20 69 66 28 20 65 72 72 2e 72 63 3d 3d 53 51 4c   if( err.rc==SQL
2ee0: 49 54 45 5f 4f 4b 20 29 7b 0a 20 20 20 20 70 72  ITE_OK ){.    pr
2ef0: 69 6e 74 66 28 22 20 20 57 41 4c 20 66 69 6c 65  intf("  WAL file
2f00: 20 69 73 20 25 64 20 62 79 74 65 73 2c 22 2c 20   is %d bytes,", 
2f10: 28 69 6e 74 29 66 69 6c 65 73 69 7a 65 28 26 65  (int)filesize(&e
2f20: 72 72 2c 22 74 65 73 74 2e 64 62 2d 77 61 6c 22  rr,"test.db-wal"
2f30: 29 29 3b 0a 20 20 20 20 70 72 69 6e 74 66 28 22  ));.    printf("
2f40: 20 44 42 20 66 69 6c 65 20 69 73 20 25 64 2e 5c   DB file is %d.\
2f50: 6e 22 2c 20 28 69 6e 74 29 66 69 6c 65 73 69 7a  n", (int)filesiz
2f60: 65 28 26 65 72 72 2c 22 74 65 73 74 2e 64 62 22  e(&err,"test.db"
2f70: 29 29 3b 0a 20 20 7d 0a 0a 20 20 70 72 69 6e 74  ));.  }..  print
2f80: 5f 61 6e 64 5f 66 72 65 65 5f 65 72 72 28 26 65  _and_free_err(&e
2f90: 72 72 29 3b 0a 7d 0a 0a 2f 2a 2d 2d 2d 2d 2d 2d  rr);.}../*------
2fa0: 2d 2d 2d 2d 2d 2d 2d 2d 2d 2d 2d 2d 2d 2d 2d 2d  ----------------
2fb0: 2d 2d 2d 2d 2d 2d 2d 2d 2d 2d 2d 2d 2d 2d 2d 2d  ----------------
2fc0: 2d 2d 2d 2d 2d 2d 2d 2d 2d 2d 2d 2d 2d 2d 2d 2d  ----------------
2fd0: 2d 2d 2d 2d 2d 2d 2d 2d 2d 2d 2d 2d 2d 2d 2d 2d  ----------------
2fe0: 2d 2d 0a 2a 2a 20 54 65 73 74 20 63 61 73 65 20  --.** Test case 
2ff0: 22 63 67 74 5f 70 61 67 65 72 5f 31 22 0a 2a 2f  "cgt_pager_1".*/
3000: 0a 23 64 65 66 69 6e 65 20 43 41 4c 4c 47 52 49  .#define CALLGRI
3010: 4e 44 54 45 53 54 31 5f 4e 52 4f 57 20 31 30 30  NDTEST1_NROW 100
3020: 30 30 0a 73 74 61 74 69 63 20 76 6f 69 64 20 63  00.static void c
3030: 67 74 5f 70 61 67 65 72 5f 31 5f 70 6f 70 75 6c  gt_pager_1_popul
3040: 61 74 65 28 45 72 72 6f 72 20 2a 70 45 72 72 2c  ate(Error *pErr,
3050: 20 53 71 6c 69 74 65 20 2a 70 44 62 29 7b 0a 20   Sqlite *pDb){. 
3060: 20 63 6f 6e 73 74 20 63 68 61 72 20 2a 7a 49 6e   const char *zIn
3070: 73 65 72 74 20 3d 20 22 49 4e 53 45 52 54 20 49  sert = "INSERT I
3080: 4e 54 4f 20 74 31 20 56 41 4c 55 45 53 28 3a 69  NTO t1 VALUES(:i
3090: 52 6f 77 2c 20 7a 65 72 6f 62 6c 6f 62 28 3a 69  Row, zeroblob(:i
30a0: 42 6c 6f 62 29 29 22 3b 0a 20 20 69 36 34 20 69  Blob))";.  i64 i
30b0: 52 6f 77 3b 0a 20 20 73 71 6c 5f 73 63 72 69 70  Row;.  sql_scrip
30c0: 74 28 70 45 72 72 2c 20 70 44 62 2c 20 22 42 45  t(pErr, pDb, "BE
30d0: 47 49 4e 22 29 3b 0a 20 20 66 6f 72 28 69 52 6f  GIN");.  for(iRo
30e0: 77 3d 31 3b 20 69 52 6f 77 3c 3d 43 41 4c 4c 47  w=1; iRow<=CALLG
30f0: 52 49 4e 44 54 45 53 54 31 5f 4e 52 4f 57 3b 20  RINDTEST1_NROW; 
3100: 69 52 6f 77 2b 2b 29 7b 0a 20 20 20 20 69 36 34  iRow++){.    i64
3110: 20 69 42 6c 6f 62 20 3d 20 36 30 30 20 2b 20 28   iBlob = 600 + (
3120: 69 52 6f 77 25 33 30 30 29 3b 0a 20 20 20 20 65  iRow%300);.    e
3130: 78 65 63 73 71 6c 28 70 45 72 72 2c 20 70 44 62  xecsql(pErr, pDb
3140: 2c 20 7a 49 6e 73 65 72 74 2c 20 26 69 52 6f 77  , zInsert, &iRow
3150: 2c 20 26 69 42 6c 6f 62 29 3b 0a 20 20 7d 0a 20  , &iBlob);.  }. 
3160: 20 73 71 6c 5f 73 63 72 69 70 74 28 70 45 72 72   sql_script(pErr
3170: 2c 20 70 44 62 2c 20 22 43 4f 4d 4d 49 54 22 29  , pDb, "COMMIT")
3180: 3b 0a 7d 0a 73 74 61 74 69 63 20 76 6f 69 64 20  ;.}.static void 
3190: 63 67 74 5f 70 61 67 65 72 5f 31 5f 75 70 64 61  cgt_pager_1_upda
31a0: 74 65 28 45 72 72 6f 72 20 2a 70 45 72 72 2c 20  te(Error *pErr, 
31b0: 53 71 6c 69 74 65 20 2a 70 44 62 29 7b 0a 20 20  Sqlite *pDb){.  
31c0: 63 6f 6e 73 74 20 63 68 61 72 20 2a 7a 55 70 64  const char *zUpd
31d0: 61 74 65 20 3d 20 22 55 50 44 41 54 45 20 74 31  ate = "UPDATE t1
31e0: 20 53 45 54 20 62 20 3d 20 7a 65 72 6f 62 6c 6f   SET b = zeroblo
31f0: 62 28 3a 69 42 6c 6f 62 29 20 57 48 45 52 45 20  b(:iBlob) WHERE 
3200: 61 20 3d 20 3a 69 52 6f 77 22 3b 0a 20 20 69 36  a = :iRow";.  i6
3210: 34 20 69 52 6f 77 3b 0a 20 20 73 71 6c 5f 73 63  4 iRow;.  sql_sc
3220: 72 69 70 74 28 70 45 72 72 2c 20 70 44 62 2c 20  ript(pErr, pDb, 
3230: 22 42 45 47 49 4e 22 29 3b 0a 20 20 66 6f 72 28  "BEGIN");.  for(
3240: 69 52 6f 77 3d 31 3b 20 69 52 6f 77 3c 3d 43 41  iRow=1; iRow<=CA
3250: 4c 4c 47 52 49 4e 44 54 45 53 54 31 5f 4e 52 4f  LLGRINDTEST1_NRO
3260: 57 3b 20 69 52 6f 77 2b 2b 29 7b 0a 20 20 20 20  W; iRow++){.    
3270: 69 36 34 20 69 42 6c 6f 62 20 3d 20 36 30 30 20  i64 iBlob = 600 
3280: 2b 20 28 28 69 52 6f 77 2b 31 30 30 29 25 33 30  + ((iRow+100)%30
3290: 30 29 3b 0a 20 20 20 20 65 78 65 63 73 71 6c 28  0);.    execsql(
32a0: 70 45 72 72 2c 20 70 44 62 2c 20 7a 55 70 64 61  pErr, pDb, zUpda
32b0: 74 65 2c 20 26 69 42 6c 6f 62 2c 20 26 69 52 6f  te, &iBlob, &iRo
32c0: 77 29 3b 0a 20 20 7d 0a 20 20 73 71 6c 5f 73 63  w);.  }.  sql_sc
32d0: 72 69 70 74 28 70 45 72 72 2c 20 70 44 62 2c 20  ript(pErr, pDb, 
32e0: 22 43 4f 4d 4d 49 54 22 29 3b 0a 7d 0a 73 74 61  "COMMIT");.}.sta
32f0: 74 69 63 20 76 6f 69 64 20 63 67 74 5f 70 61 67  tic void cgt_pag
3300: 65 72 5f 31 5f 72 65 61 64 28 45 72 72 6f 72 20  er_1_read(Error 
3310: 2a 70 45 72 72 2c 20 53 71 6c 69 74 65 20 2a 70  *pErr, Sqlite *p
3320: 44 62 29 7b 0a 20 20 69 36 34 20 69 52 6f 77 3b  Db){.  i64 iRow;
3330: 0a 20 20 73 71 6c 5f 73 63 72 69 70 74 28 70 45  .  sql_script(pE
3340: 72 72 2c 20 70 44 62 2c 20 22 42 45 47 49 4e 22  rr, pDb, "BEGIN"
3350: 29 3b 0a 20 20 66 6f 72 28 69 52 6f 77 3d 31 3b  );.  for(iRow=1;
3360: 20 69 52 6f 77 3c 3d 43 41 4c 4c 47 52 49 4e 44   iRow<=CALLGRIND
3370: 54 45 53 54 31 5f 4e 52 4f 57 3b 20 69 52 6f 77  TEST1_NROW; iRow
3380: 2b 2b 29 7b 0a 20 20 20 20 65 78 65 63 73 71 6c  ++){.    execsql
3390: 28 70 45 72 72 2c 20 70 44 62 2c 20 22 53 45 4c  (pErr, pDb, "SEL
33a0: 45 43 54 20 2a 20 46 52 4f 4d 20 74 31 20 57 48  ECT * FROM t1 WH
33b0: 45 52 45 20 61 20 3d 20 3a 69 52 6f 77 22 2c 20  ERE a = :iRow", 
33c0: 26 69 52 6f 77 29 3b 0a 20 20 7d 0a 20 20 73 71  &iRow);.  }.  sq
33d0: 6c 5f 73 63 72 69 70 74 28 70 45 72 72 2c 20 70  l_script(pErr, p
33e0: 44 62 2c 20 22 43 4f 4d 4d 49 54 22 29 3b 0a 7d  Db, "COMMIT");.}
33f0: 0a 73 74 61 74 69 63 20 76 6f 69 64 20 63 67 74  .static void cgt
3400: 5f 70 61 67 65 72 5f 31 28 69 6e 74 20 6e 4d 73  _pager_1(int nMs
3410: 29 7b 0a 20 20 76 6f 69 64 20 28 2a 78 53 75 62  ){.  void (*xSub
3420: 29 28 45 72 72 6f 72 20 2a 2c 20 53 71 6c 69 74  )(Error *, Sqlit
3430: 65 20 2a 29 3b 0a 20 20 45 72 72 6f 72 20 65 72  e *);.  Error er
3440: 72 20 3d 20 7b 30 7d 3b 0a 20 20 53 71 6c 69 74  r = {0};.  Sqlit
3450: 65 20 64 62 20 3d 20 7b 30 7d 3b 0a 0a 20 20 6f  e db = {0};..  o
3460: 70 65 6e 64 62 28 26 65 72 72 2c 20 26 64 62 2c  pendb(&err, &db,
3470: 20 22 74 65 73 74 2e 64 62 22 2c 20 31 29 3b 0a   "test.db", 1);.
3480: 20 20 73 71 6c 5f 73 63 72 69 70 74 28 26 65 72    sql_script(&er
3490: 72 2c 20 26 64 62 2c 0a 20 20 20 20 20 20 22 50  r, &db,.      "P
34a0: 52 41 47 4d 41 20 63 61 63 68 65 5f 73 69 7a 65  RAGMA cache_size
34b0: 20 3d 20 32 30 30 30 3b 22 0a 20 20 20 20 20 20   = 2000;".      
34c0: 22 50 52 41 47 4d 41 20 70 61 67 65 5f 73 69 7a  "PRAGMA page_siz
34d0: 65 20 3d 20 31 30 32 34 3b 22 0a 20 20 20 20 20  e = 1024;".     
34e0: 20 22 43 52 45 41 54 45 20 54 41 42 4c 45 20 74   "CREATE TABLE t
34f0: 31 28 61 20 49 4e 54 45 47 45 52 20 50 52 49 4d  1(a INTEGER PRIM
3500: 41 52 59 20 4b 45 59 2c 20 62 20 42 4c 4f 42 29  ARY KEY, b BLOB)
3510: 3b 22 0a 20 20 29 3b 0a 0a 20 20 78 53 75 62 20  ;".  );..  xSub 
3520: 3d 20 63 67 74 5f 70 61 67 65 72 5f 31 5f 70 6f  = cgt_pager_1_po
3530: 70 75 6c 61 74 65 3b 20 78 53 75 62 28 26 65 72  pulate; xSub(&er
3540: 72 2c 20 26 64 62 29 3b 0a 20 20 78 53 75 62 20  r, &db);.  xSub 
3550: 3d 20 63 67 74 5f 70 61 67 65 72 5f 31 5f 75 70  = cgt_pager_1_up
3560: 64 61 74 65 3b 20 20 20 78 53 75 62 28 26 65 72  date;   xSub(&er
3570: 72 2c 20 26 64 62 29 3b 0a 20 20 78 53 75 62 20  r, &db);.  xSub 
3580: 3d 20 63 67 74 5f 70 61 67 65 72 5f 31 5f 72 65  = cgt_pager_1_re
3590: 61 64 3b 20 20 20 20 20 78 53 75 62 28 26 65 72  ad;     xSub(&er
35a0: 72 2c 20 26 64 62 29 3b 0a 0a 20 20 63 6c 6f 73  r, &db);..  clos
35b0: 65 64 62 28 26 65 72 72 2c 20 26 64 62 29 3b 0a  edb(&err, &db);.
35c0: 20 20 70 72 69 6e 74 5f 61 6e 64 5f 66 72 65 65    print_and_free
35d0: 5f 65 72 72 28 26 65 72 72 29 3b 0a 7d 0a 0a 2f  _err(&err);.}../
35e0: 2a 2d 2d 2d 2d 2d 2d 2d 2d 2d 2d 2d 2d 2d 2d 2d  *---------------
35f0: 2d 2d 2d 2d 2d 2d 2d 2d 2d 2d 2d 2d 2d 2d 2d 2d  ----------------
3600: 2d 2d 2d 2d 2d 2d 2d 2d 2d 2d 2d 2d 2d 2d 2d 2d  ----------------
3610: 2d 2d 2d 2d 2d 2d 2d 2d 2d 2d 2d 2d 2d 2d 2d 2d  ----------------
3620: 2d 2d 2d 2d 2d 2d 2d 2d 2d 0a 2a 2a 20 54 65 73  ---------.** Tes
3630: 74 20 63 61 73 65 20 22 64 79 6e 61 6d 69 63 5f  t case "dynamic_
3640: 74 72 69 67 67 65 72 73 22 0a 2a 2a 0a 2a 2a 20  triggers".**.** 
3650: 20 20 54 77 6f 20 74 68 72 65 61 64 73 20 65 78    Two threads ex
3660: 65 63 75 74 69 6e 67 20 73 74 61 74 65 6d 65 6e  ecuting statemen
3670: 74 73 20 74 68 61 74 20 63 61 75 73 65 20 64 65  ts that cause de
3680: 65 70 6c 79 20 6e 65 73 74 65 64 20 74 72 69 67  eply nested trig
3690: 67 65 72 73 0a 2a 2a 20 20 20 74 6f 20 66 69 72  gers.**   to fir
36a0: 65 2e 20 41 6e 64 20 6f 6e 65 20 74 68 72 65 61  e. And one threa
36b0: 64 20 62 75 73 69 6c 79 20 63 72 65 61 74 69 6e  d busily creatin
36c0: 67 20 61 6e 64 20 64 65 6c 65 74 69 6e 67 20 74  g and deleting t
36d0: 72 69 67 67 65 72 73 2e 20 54 68 69 73 0a 2a 2a  riggers. This.**
36e0: 20 20 20 69 73 20 61 6e 20 61 74 74 65 6d 70 74     is an attempt
36f0: 20 74 6f 20 66 69 6e 64 20 61 20 62 75 67 20 72   to find a bug r
3700: 65 70 6f 72 74 65 64 20 74 6f 20 75 73 2e 0a 2a  eported to us..*
3710: 2f 0a 0a 73 74 61 74 69 63 20 63 68 61 72 20 2a  /..static char *
3720: 64 79 6e 61 6d 69 63 5f 74 72 69 67 67 65 72 73  dynamic_triggers
3730: 5f 31 28 69 6e 74 20 69 54 69 64 2c 20 76 6f 69  _1(int iTid, voi
3740: 64 20 2a 70 41 72 67 29 7b 0a 20 20 45 72 72 6f  d *pArg){.  Erro
3750: 72 20 65 72 72 20 3d 20 7b 30 7d 3b 20 20 20 20  r err = {0};    
3760: 20 20 20 20 20 20 20 20 20 20 20 20 2f 2a 20 45              /* E
3770: 72 72 6f 72 20 63 6f 64 65 20 61 6e 64 20 6d 65  rror code and me
3780: 73 73 61 67 65 20 2a 2f 0a 20 20 53 71 6c 69 74  ssage */.  Sqlit
3790: 65 20 64 62 20 3d 20 7b 30 7d 3b 20 20 20 20 20  e db = {0};     
37a0: 20 20 20 20 20 20 20 20 20 20 20 2f 2a 20 53 51             /* SQ
37b0: 4c 69 74 65 20 64 61 74 61 62 61 73 65 20 63 6f  Lite database co
37c0: 6e 6e 65 63 74 69 6f 6e 20 2a 2f 0a 20 20 69 6e  nnection */.  in
37d0: 74 20 6e 44 72 6f 70 20 3d 20 30 3b 0a 20 20 69  t nDrop = 0;.  i
37e0: 6e 74 20 6e 43 72 65 61 74 65 20 3d 20 30 3b 0a  nt nCreate = 0;.
37f0: 0a 20 20 6f 70 65 6e 64 62 28 26 65 72 72 2c 20  .  opendb(&err, 
3800: 26 64 62 2c 20 22 74 65 73 74 2e 64 62 22 2c 20  &db, "test.db", 
3810: 30 29 3b 0a 20 20 77 68 69 6c 65 28 20 21 74 69  0);.  while( !ti
3820: 6d 65 74 6f 73 74 6f 70 28 26 65 72 72 29 20 29  metostop(&err) )
3830: 7b 0a 20 20 20 20 69 6e 74 20 69 3b 0a 0a 20 20  {.    int i;..  
3840: 20 20 66 6f 72 28 69 3d 31 3b 20 69 3c 39 3b 20    for(i=1; i<9; 
3850: 69 2b 2b 29 7b 0a 20 20 20 20 20 20 63 68 61 72  i++){.      char
3860: 20 2a 7a 53 71 6c 20 3d 20 73 71 6c 69 74 65 33   *zSql = sqlite3
3870: 5f 6d 70 72 69 6e 74 66 28 0a 20 20 20 20 20 20  _mprintf(.      
3880: 20 20 22 43 52 45 41 54 45 20 54 52 49 47 47 45    "CREATE TRIGGE
3890: 52 20 69 74 72 25 64 20 42 45 46 4f 52 45 20 49  R itr%d BEFORE I
38a0: 4e 53 45 52 54 20 4f 4e 20 74 25 64 20 42 45 47  NSERT ON t%d BEG
38b0: 49 4e 20 22 0a 20 20 20 20 20 20 20 20 20 20 22  IN ".          "
38c0: 49 4e 53 45 52 54 20 49 4e 54 4f 20 74 25 64 20  INSERT INTO t%d 
38d0: 56 41 4c 55 45 53 28 6e 65 77 2e 78 2c 20 6e 65  VALUES(new.x, ne
38e0: 77 2e 79 29 3b 22 0a 20 20 20 20 20 20 20 20 22  w.y);".        "
38f0: 45 4e 44 3b 22 2c 20 69 2c 20 69 2c 20 69 2b 31  END;", i, i, i+1
3900: 0a 20 20 20 20 20 20 29 3b 0a 20 20 20 20 20 20  .      );.      
3910: 65 78 65 63 73 71 6c 28 26 65 72 72 2c 20 26 64  execsql(&err, &d
3920: 62 2c 20 7a 53 71 6c 29 3b 0a 20 20 20 20 20 20  b, zSql);.      
3930: 73 71 6c 69 74 65 33 5f 66 72 65 65 28 7a 53 71  sqlite3_free(zSq
3940: 6c 29 3b 0a 20 20 20 20 20 20 6e 43 72 65 61 74  l);.      nCreat
3950: 65 2b 2b 3b 0a 20 20 20 20 7d 0a 0a 20 20 20 20  e++;.    }..    
3960: 66 6f 72 28 69 3d 31 3b 20 69 3c 39 3b 20 69 2b  for(i=1; i<9; i+
3970: 2b 29 7b 0a 20 20 20 20 20 20 63 68 61 72 20 2a  +){.      char *
3980: 7a 53 71 6c 20 3d 20 73 71 6c 69 74 65 33 5f 6d  zSql = sqlite3_m
3990: 70 72 69 6e 74 66 28 0a 20 20 20 20 20 20 20 20  printf(.        
39a0: 22 43 52 45 41 54 45 20 54 52 49 47 47 45 52 20  "CREATE TRIGGER 
39b0: 64 74 72 25 64 20 42 45 46 4f 52 45 20 44 45 4c  dtr%d BEFORE DEL
39c0: 45 54 45 20 4f 4e 20 74 25 64 20 42 45 47 49 4e  ETE ON t%d BEGIN
39d0: 20 22 0a 20 20 20 20 20 20 20 20 20 20 22 44 45   ".          "DE
39e0: 4c 45 54 45 20 46 52 4f 4d 20 74 25 64 20 57 48  LETE FROM t%d WH
39f0: 45 52 45 20 78 20 3d 20 6f 6c 64 2e 78 3b 20 22  ERE x = old.x; "
3a00: 0a 20 20 20 20 20 20 20 20 22 45 4e 44 3b 22 2c  .        "END;",
3a10: 20 69 2c 20 69 2c 20 69 2b 31 0a 20 20 20 20 20   i, i, i+1.     
3a20: 20 29 3b 0a 20 20 20 20 20 20 65 78 65 63 73 71   );.      execsq
3a30: 6c 28 26 65 72 72 2c 20 26 64 62 2c 20 7a 53 71  l(&err, &db, zSq
3a40: 6c 29 3b 0a 20 20 20 20 20 20 73 71 6c 69 74 65  l);.      sqlite
3a50: 33 5f 66 72 65 65 28 7a 53 71 6c 29 3b 0a 20 20  3_free(zSql);.  
3a60: 20 20 20 20 6e 43 72 65 61 74 65 2b 2b 3b 0a 20      nCreate++;. 
3a70: 20 20 20 7d 0a 0a 20 20 20 20 66 6f 72 28 69 3d     }..    for(i=
3a80: 31 3b 20 69 3c 39 3b 20 69 2b 2b 29 7b 0a 20 20  1; i<9; i++){.  
3a90: 20 20 20 20 63 68 61 72 20 2a 7a 53 71 6c 20 3d      char *zSql =
3aa0: 20 73 71 6c 69 74 65 33 5f 6d 70 72 69 6e 74 66   sqlite3_mprintf
3ab0: 28 22 44 52 4f 50 20 54 52 49 47 47 45 52 20 69  ("DROP TRIGGER i
3ac0: 74 72 25 64 22 2c 20 69 29 3b 0a 20 20 20 20 20  tr%d", i);.     
3ad0: 20 65 78 65 63 73 71 6c 28 26 65 72 72 2c 20 26   execsql(&err, &
3ae0: 64 62 2c 20 7a 53 71 6c 29 3b 0a 20 20 20 20 20  db, zSql);.     
3af0: 20 73 71 6c 69 74 65 33 5f 66 72 65 65 28 7a 53   sqlite3_free(zS
3b00: 71 6c 29 3b 0a 20 20 20 20 20 20 6e 44 72 6f 70  ql);.      nDrop
3b10: 2b 2b 3b 0a 20 20 20 20 7d 0a 0a 20 20 20 20 66  ++;.    }..    f
3b20: 6f 72 28 69 3d 31 3b 20 69 3c 39 3b 20 69 2b 2b  or(i=1; i<9; i++
3b30: 29 7b 0a 20 20 20 20 20 20 63 68 61 72 20 2a 7a  ){.      char *z
3b40: 53 71 6c 20 3d 20 73 71 6c 69 74 65 33 5f 6d 70  Sql = sqlite3_mp
3b50: 72 69 6e 74 66 28 22 44 52 4f 50 20 54 52 49 47  rintf("DROP TRIG
3b60: 47 45 52 20 64 74 72 25 64 22 2c 20 69 29 3b 0a  GER dtr%d", i);.
3b70: 20 20 20 20 20 20 65 78 65 63 73 71 6c 28 26 65        execsql(&e
3b80: 72 72 2c 20 26 64 62 2c 20 7a 53 71 6c 29 3b 0a  rr, &db, zSql);.
3b90: 20 20 20 20 20 20 73 71 6c 69 74 65 33 5f 66 72        sqlite3_fr
3ba0: 65 65 28 7a 53 71 6c 29 3b 0a 20 20 20 20 20 20  ee(zSql);.      
3bb0: 6e 44 72 6f 70 2b 2b 3b 0a 20 20 20 20 7d 0a 20  nDrop++;.    }. 
3bc0: 20 7d 0a 20 20 63 6c 6f 73 65 64 62 28 26 65 72   }.  closedb(&er
3bd0: 72 2c 20 26 64 62 29 3b 0a 0a 20 20 70 72 69 6e  r, &db);..  prin
3be0: 74 5f 61 6e 64 5f 66 72 65 65 5f 65 72 72 28 26  t_and_free_err(&
3bf0: 65 72 72 29 3b 0a 20 20 72 65 74 75 72 6e 20 73  err);.  return s
3c00: 71 6c 69 74 65 33 5f 6d 70 72 69 6e 74 66 28 22  qlite3_mprintf("
3c10: 25 64 20 63 72 65 61 74 65 64 2c 20 25 64 20 64  %d created, %d d
3c20: 72 6f 70 70 65 64 22 2c 20 6e 43 72 65 61 74 65  ropped", nCreate
3c30: 2c 20 6e 44 72 6f 70 29 3b 0a 7d 0a 0a 73 74 61  , nDrop);.}..sta
3c40: 74 69 63 20 63 68 61 72 20 2a 64 79 6e 61 6d 69  tic char *dynami
3c50: 63 5f 74 72 69 67 67 65 72 73 5f 32 28 69 6e 74  c_triggers_2(int
3c60: 20 69 54 69 64 2c 20 76 6f 69 64 20 2a 70 41 72   iTid, void *pAr
3c70: 67 29 7b 0a 20 20 45 72 72 6f 72 20 65 72 72 20  g){.  Error err 
3c80: 3d 20 7b 30 7d 3b 20 20 20 20 20 20 20 20 20 20  = {0};          
3c90: 20 20 20 20 20 20 2f 2a 20 45 72 72 6f 72 20 63        /* Error c
3ca0: 6f 64 65 20 61 6e 64 20 6d 65 73 73 61 67 65 20  ode and message 
3cb0: 2a 2f 0a 20 20 53 71 6c 69 74 65 20 64 62 20 3d  */.  Sqlite db =
3cc0: 20 7b 30 7d 3b 20 20 20 20 20 20 20 20 20 20 20   {0};           
3cd0: 20 20 20 20 20 2f 2a 20 53 51 4c 69 74 65 20 64       /* SQLite d
3ce0: 61 74 61 62 61 73 65 20 63 6f 6e 6e 65 63 74 69  atabase connecti
3cf0: 6f 6e 20 2a 2f 0a 20 20 69 36 34 20 69 56 61 6c  on */.  i64 iVal
3d00: 20 3d 20 30 3b 0a 20 20 69 6e 74 20 6e 49 6e 73   = 0;.  int nIns
3d10: 65 72 74 20 3d 20 30 3b 0a 20 20 69 6e 74 20 6e  ert = 0;.  int n
3d20: 44 65 6c 65 74 65 20 3d 20 30 3b 0a 0a 20 20 6f  Delete = 0;..  o
3d30: 70 65 6e 64 62 28 26 65 72 72 2c 20 26 64 62 2c  pendb(&err, &db,
3d40: 20 22 74 65 73 74 2e 64 62 22 2c 20 30 29 3b 0a   "test.db", 0);.
3d50: 20 20 77 68 69 6c 65 28 20 21 74 69 6d 65 74 6f    while( !timeto
3d60: 73 74 6f 70 28 26 65 72 72 29 20 29 7b 0a 20 20  stop(&err) ){.  
3d70: 20 20 64 6f 20 7b 0a 20 20 20 20 20 20 69 56 61    do {.      iVa
3d80: 6c 20 3d 20 28 69 56 61 6c 2b 31 29 25 31 30 30  l = (iVal+1)%100
3d90: 3b 0a 20 20 20 20 20 20 65 78 65 63 73 71 6c 28  ;.      execsql(
3da0: 26 65 72 72 2c 20 26 64 62 2c 20 22 49 4e 53 45  &err, &db, "INSE
3db0: 52 54 20 49 4e 54 4f 20 74 31 20 56 41 4c 55 45  RT INTO t1 VALUE
3dc0: 53 28 3a 69 58 2c 20 3a 69 59 2b 31 29 22 2c 20  S(:iX, :iY+1)", 
3dd0: 26 69 56 61 6c 2c 20 26 69 56 61 6c 29 3b 0a 20  &iVal, &iVal);. 
3de0: 20 20 20 20 20 6e 49 6e 73 65 72 74 2b 2b 3b 0a       nInsert++;.
3df0: 20 20 20 20 7d 20 77 68 69 6c 65 28 20 69 56 61      } while( iVa
3e00: 6c 20 29 3b 0a 0a 20 20 20 20 64 6f 20 7b 0a 20  l );..    do {. 
3e10: 20 20 20 20 20 69 56 61 6c 20 3d 20 28 69 56 61       iVal = (iVa
3e20: 6c 2b 31 29 25 31 30 30 3b 0a 20 20 20 20 20 20  l+1)%100;.      
3e30: 65 78 65 63 73 71 6c 28 26 65 72 72 2c 20 26 64  execsql(&err, &d
3e40: 62 2c 20 22 44 45 4c 45 54 45 20 46 52 4f 4d 20  b, "DELETE FROM 
3e50: 74 31 20 57 48 45 52 45 20 78 20 3d 20 3a 69 58  t1 WHERE x = :iX
3e60: 22 2c 20 26 69 56 61 6c 29 3b 0a 20 20 20 20 20  ", &iVal);.     
3e70: 20 6e 44 65 6c 65 74 65 2b 2b 3b 0a 20 20 20 20   nDelete++;.    
3e80: 7d 20 77 68 69 6c 65 28 20 69 56 61 6c 20 29 3b  } while( iVal );
3e90: 0a 20 20 7d 0a 20 20 63 6c 6f 73 65 64 62 28 26  .  }.  closedb(&
3ea0: 65 72 72 2c 20 26 64 62 29 3b 0a 0a 20 20 70 72  err, &db);..  pr
3eb0: 69 6e 74 5f 61 6e 64 5f 66 72 65 65 5f 65 72 72  int_and_free_err
3ec0: 28 26 65 72 72 29 3b 0a 20 20 72 65 74 75 72 6e  (&err);.  return
3ed0: 20 73 71 6c 69 74 65 33 5f 6d 70 72 69 6e 74 66   sqlite3_mprintf
3ee0: 28 22 25 64 20 69 6e 73 65 72 74 73 2c 20 25 64  ("%d inserts, %d
3ef0: 20 64 65 6c 65 74 65 73 22 2c 20 6e 49 6e 73 65   deletes", nInse
3f00: 72 74 2c 20 6e 44 65 6c 65 74 65 29 3b 0a 7d 0a  rt, nDelete);.}.
3f10: 0a 73 74 61 74 69 63 20 76 6f 69 64 20 64 79 6e  .static void dyn
3f20: 61 6d 69 63 5f 74 72 69 67 67 65 72 73 28 69 6e  amic_triggers(in
3f30: 74 20 6e 4d 73 29 7b 0a 20 20 45 72 72 6f 72 20  t nMs){.  Error 
3f40: 65 72 72 20 3d 20 7b 30 7d 3b 0a 20 20 53 71 6c  err = {0};.  Sql
3f50: 69 74 65 20 64 62 20 3d 20 7b 30 7d 3b 0a 20 20  ite db = {0};.  
3f60: 54 68 72 65 61 64 73 65 74 20 74 68 72 65 61 64  Threadset thread
3f70: 73 20 3d 20 7b 30 7d 3b 0a 0a 20 20 6f 70 65 6e  s = {0};..  open
3f80: 64 62 28 26 65 72 72 2c 20 26 64 62 2c 20 22 74  db(&err, &db, "t
3f90: 65 73 74 2e 64 62 22 2c 20 31 29 3b 0a 20 20 73  est.db", 1);.  s
3fa0: 71 6c 5f 73 63 72 69 70 74 28 26 65 72 72 2c 20  ql_script(&err, 
3fb0: 26 64 62 2c 20 0a 20 20 20 20 20 20 22 50 52 41  &db, .      "PRA
3fc0: 47 4d 41 20 70 61 67 65 5f 73 69 7a 65 20 3d 20  GMA page_size = 
3fd0: 31 30 32 34 3b 22 0a 20 20 20 20 20 20 22 50 52  1024;".      "PR
3fe0: 41 47 4d 41 20 6a 6f 75 72 6e 61 6c 5f 6d 6f 64  AGMA journal_mod
3ff0: 65 20 3d 20 57 41 4c 3b 22 0a 20 20 20 20 20 20  e = WAL;".      
4000: 22 43 52 45 41 54 45 20 54 41 42 4c 45 20 74 31  "CREATE TABLE t1
4010: 28 78 2c 20 79 29 3b 22 0a 20 20 20 20 20 20 22  (x, y);".      "
4020: 43 52 45 41 54 45 20 54 41 42 4c 45 20 74 32 28  CREATE TABLE t2(
4030: 78 2c 20 79 29 3b 22 0a 20 20 20 20 20 20 22 43  x, y);".      "C
4040: 52 45 41 54 45 20 54 41 42 4c 45 20 74 33 28 78  REATE TABLE t3(x
4050: 2c 20 79 29 3b 22 0a 20 20 20 20 20 20 22 43 52  , y);".      "CR
4060: 45 41 54 45 20 54 41 42 4c 45 20 74 34 28 78 2c  EATE TABLE t4(x,
4070: 20 79 29 3b 22 0a 20 20 20 20 20 20 22 43 52 45   y);".      "CRE
4080: 41 54 45 20 54 41 42 4c 45 20 74 35 28 78 2c 20  ATE TABLE t5(x, 
4090: 79 29 3b 22 0a 20 20 20 20 20 20 22 43 52 45 41  y);".      "CREA
40a0: 54 45 20 54 41 42 4c 45 20 74 36 28 78 2c 20 79  TE TABLE t6(x, y
40b0: 29 3b 22 0a 20 20 20 20 20 20 22 43 52 45 41 54  );".      "CREAT
40c0: 45 20 54 41 42 4c 45 20 74 37 28 78 2c 20 79 29  E TABLE t7(x, y)
40d0: 3b 22 0a 20 20 20 20 20 20 22 43 52 45 41 54 45  ;".      "CREATE
40e0: 20 54 41 42 4c 45 20 74 38 28 78 2c 20 79 29 3b   TABLE t8(x, y);
40f0: 22 0a 20 20 20 20 20 20 22 43 52 45 41 54 45 20  ".      "CREATE 
4100: 54 41 42 4c 45 20 74 39 28 78 2c 20 79 29 3b 22  TABLE t9(x, y);"
4110: 0a 20 20 29 3b 0a 20 20 63 6c 6f 73 65 64 62 28  .  );.  closedb(
4120: 26 65 72 72 2c 20 26 64 62 29 3b 0a 0a 20 20 73  &err, &db);..  s
4130: 65 74 73 74 6f 70 74 69 6d 65 28 26 65 72 72 2c  etstoptime(&err,
4140: 20 6e 4d 73 29 3b 0a 0a 20 20 73 71 6c 69 74 65   nMs);..  sqlite
4150: 33 5f 65 6e 61 62 6c 65 5f 73 68 61 72 65 64 5f  3_enable_shared_
4160: 63 61 63 68 65 28 31 29 3b 0a 20 20 6c 61 75 6e  cache(1);.  laun
4170: 63 68 5f 74 68 72 65 61 64 28 26 65 72 72 2c 20  ch_thread(&err, 
4180: 26 74 68 72 65 61 64 73 2c 20 64 79 6e 61 6d 69  &threads, dynami
4190: 63 5f 74 72 69 67 67 65 72 73 5f 32 2c 20 30 29  c_triggers_2, 0)
41a0: 3b 0a 20 20 6c 61 75 6e 63 68 5f 74 68 72 65 61  ;.  launch_threa
41b0: 64 28 26 65 72 72 2c 20 26 74 68 72 65 61 64 73  d(&err, &threads
41c0: 2c 20 64 79 6e 61 6d 69 63 5f 74 72 69 67 67 65  , dynamic_trigge
41d0: 72 73 5f 32 2c 20 30 29 3b 0a 0a 20 20 73 6c 65  rs_2, 0);..  sle
41e0: 65 70 28 32 29 3b 0a 20 20 73 71 6c 69 74 65 33  ep(2);.  sqlite3
41f0: 5f 65 6e 61 62 6c 65 5f 73 68 61 72 65 64 5f 63  _enable_shared_c
4200: 61 63 68 65 28 30 29 3b 0a 0a 20 20 6c 61 75 6e  ache(0);..  laun
4210: 63 68 5f 74 68 72 65 61 64 28 26 65 72 72 2c 20  ch_thread(&err, 
4220: 26 74 68 72 65 61 64 73 2c 20 64 79 6e 61 6d 69  &threads, dynami
4230: 63 5f 74 72 69 67 67 65 72 73 5f 32 2c 20 30 29  c_triggers_2, 0)
4240: 3b 0a 20 20 6c 61 75 6e 63 68 5f 74 68 72 65 61  ;.  launch_threa
4250: 64 28 26 65 72 72 2c 20 26 74 68 72 65 61 64 73  d(&err, &threads
4260: 2c 20 64 79 6e 61 6d 69 63 5f 74 72 69 67 67 65  , dynamic_trigge
4270: 72 73 5f 31 2c 20 30 29 3b 0a 0a 20 20 6a 6f 69  rs_1, 0);..  joi
4280: 6e 5f 61 6c 6c 5f 74 68 72 65 61 64 73 28 26 65  n_all_threads(&e
4290: 72 72 2c 20 26 74 68 72 65 61 64 73 29 3b 0a 0a  rr, &threads);..
42a0: 20 20 70 72 69 6e 74 5f 61 6e 64 5f 66 72 65 65    print_and_free
42b0: 5f 65 72 72 28 26 65 72 72 29 3b 0a 7d 0a 0a 0a  _err(&err);.}...
42c0: 0a 23 69 6e 63 6c 75 64 65 20 22 74 74 33 5f 63  .#include "tt3_c
42d0: 68 65 63 6b 70 6f 69 6e 74 2e 63 22 0a 23 69 6e  heckpoint.c".#in
42e0: 63 6c 75 64 65 20 22 74 74 33 5f 69 6e 64 65 78  clude "tt3_index
42f0: 2e 63 22 0a 23 69 6e 63 6c 75 64 65 20 22 74 74  .c".#include "tt
4300: 33 5f 6c 6f 6f 6b 61 73 69 64 65 31 2e 63 22 0a  3_lookaside1.c".
4310: 23 69 6e 63 6c 75 64 65 20 22 74 74 33 5f 76 61  #include "tt3_va
4320: 63 75 75 6d 2e 63 22 0a 23 69 6e 63 6c 75 64 65  cuum.c".#include
4330: 20 22 74 74 33 5f 73 74 72 65 73 73 2e 63 22 0a   "tt3_stress.c".
4340: 0a 69 6e 74 20 6d 61 69 6e 28 69 6e 74 20 61 72  .int main(int ar
4350: 67 63 2c 20 63 68 61 72 20 2a 2a 61 72 67 76 29  gc, char **argv)
4360: 7b 0a 20 20 73 74 72 75 63 74 20 54 68 72 65 61  {.  struct Threa
4370: 64 54 65 73 74 20 7b 0a 20 20 20 20 76 6f 69 64  dTest {.    void
4380: 20 28 2a 78 54 65 73 74 29 28 69 6e 74 29 3b 20   (*xTest)(int); 
4390: 20 20 2f 2a 20 52 6f 75 74 69 6e 65 20 66 6f 72    /* Routine for
43a0: 20 72 75 6e 6e 69 6e 67 20 74 68 69 73 20 74 65   running this te
43b0: 73 74 20 2a 2f 0a 20 20 20 20 63 6f 6e 73 74 20  st */.    const 
43c0: 63 68 61 72 20 2a 7a 54 65 73 74 3b 20 20 20 20  char *zTest;    
43d0: 2f 2a 20 4e 61 6d 65 20 6f 66 20 74 68 69 73 20  /* Name of this 
43e0: 74 65 73 74 20 2a 2f 0a 20 20 20 20 69 6e 74 20  test */.    int 
43f0: 6e 4d 73 3b 20 20 20 20 20 20 20 20 20 20 20 20  nMs;            
4400: 20 20 2f 2a 20 48 6f 77 20 6c 6f 6e 67 20 74 6f    /* How long to
4410: 20 72 75 6e 20 74 68 69 73 20 74 65 73 74 2c 20   run this test, 
4420: 69 6e 20 6d 69 6c 6c 69 73 65 63 6f 6e 64 73 20  in milliseconds 
4430: 2a 2f 0a 20 20 7d 20 61 54 65 73 74 5b 5d 20 3d  */.  } aTest[] =
4440: 20 7b 0a 20 20 20 20 7b 20 77 61 6c 74 68 72 65   {.    { walthre
4450: 61 64 31 2c 20 22 77 61 6c 74 68 72 65 61 64 31  ad1, "walthread1
4460: 22 2c 20 32 30 30 30 30 20 7d 2c 0a 20 20 20 20  ", 20000 },.    
4470: 7b 20 77 61 6c 74 68 72 65 61 64 32 2c 20 22 77  { walthread2, "w
4480: 61 6c 74 68 72 65 61 64 32 22 2c 20 32 30 30 30  althread2", 2000
4490: 30 20 7d 2c 0a 20 20 20 20 7b 20 77 61 6c 74 68  0 },.    { walth
44a0: 72 65 61 64 33 2c 20 22 77 61 6c 74 68 72 65 61  read3, "walthrea
44b0: 64 33 22 2c 20 32 30 30 30 30 20 7d 2c 0a 20 20  d3", 20000 },.  
44c0: 20 20 7b 20 77 61 6c 74 68 72 65 61 64 34 2c 20    { walthread4, 
44d0: 22 77 61 6c 74 68 72 65 61 64 34 22 2c 20 32 30  "walthread4", 20
44e0: 30 30 30 20 7d 2c 0a 20 20 20 20 7b 20 77 61 6c  000 },.    { wal
44f0: 74 68 72 65 61 64 35 2c 20 22 77 61 6c 74 68 72  thread5, "walthr
4500: 65 61 64 35 22 2c 20 20 31 30 30 30 20 7d 2c 0a  ead5",  1000 },.
4510: 20 20 20 20 0a 20 20 20 20 7b 20 63 67 74 5f 70      .    { cgt_p
4520: 61 67 65 72 5f 31 2c 20 20 20 20 20 20 22 63 67  ager_1,      "cg
4530: 74 5f 70 61 67 65 72 5f 31 22 2c 20 30 20 7d 2c  t_pager_1", 0 },
4540: 0a 20 20 20 20 7b 20 64 79 6e 61 6d 69 63 5f 74  .    { dynamic_t
4550: 72 69 67 67 65 72 73 2c 20 22 64 79 6e 61 6d 69  riggers, "dynami
4560: 63 5f 74 72 69 67 67 65 72 73 22 2c 20 32 30 30  c_triggers", 200
4570: 30 30 20 7d 2c 0a 0a 20 20 20 20 7b 20 63 68 65  00 },..    { che
4580: 63 6b 70 6f 69 6e 74 5f 73 74 61 72 76 61 74 69  ckpoint_starvati
4590: 6f 6e 5f 31 2c 20 22 63 68 65 63 6b 70 6f 69 6e  on_1, "checkpoin
45a0: 74 5f 73 74 61 72 76 61 74 69 6f 6e 5f 31 22 2c  t_starvation_1",
45b0: 20 31 30 30 30 30 20 7d 2c 0a 20 20 20 20 7b 20   10000 },.    { 
45c0: 63 68 65 63 6b 70 6f 69 6e 74 5f 73 74 61 72 76  checkpoint_starv
45d0: 61 74 69 6f 6e 5f 32 2c 20 22 63 68 65 63 6b 70  ation_2, "checkp
45e0: 6f 69 6e 74 5f 73 74 61 72 76 61 74 69 6f 6e 5f  oint_starvation_
45f0: 32 22 2c 20 31 30 30 30 30 20 7d 2c 0a 0a 20 20  2", 10000 },..  
4600: 20 20 7b 20 63 72 65 61 74 65 5f 64 72 6f 70 5f    { create_drop_
4610: 69 6e 64 65 78 5f 31 2c 20 22 63 72 65 61 74 65  index_1, "create
4620: 5f 64 72 6f 70 5f 69 6e 64 65 78 5f 31 22 2c 20  _drop_index_1", 
4630: 31 30 30 30 30 20 7d 2c 0a 20 20 20 20 7b 20 6c  10000 },.    { l
4640: 6f 6f 6b 61 73 69 64 65 31 2c 20 20 20 20 20 20  ookaside1,      
4650: 20 20 20 20 22 6c 6f 6f 6b 61 73 69 64 65 31 22      "lookaside1"
4660: 2c 20 31 30 30 30 30 20 7d 2c 0a 20 20 20 20 7b  , 10000 },.    {
4670: 20 76 61 63 75 75 6d 31 2c 20 20 20 20 20 20 20   vacuum1,       
4680: 20 20 20 20 20 20 22 76 61 63 75 75 6d 31 22 2c        "vacuum1",
4690: 20 31 30 30 30 30 20 7d 2c 0a 20 20 20 20 7b 20   10000 },.    { 
46a0: 73 74 72 65 73 73 31 2c 20 20 20 20 20 20 20 20  stress1,        
46b0: 20 20 20 20 20 22 73 74 72 65 73 73 31 22 2c 20       "stress1", 
46c0: 31 30 30 30 30 20 7d 2c 0a 20 20 20 20 7b 20 73  10000 },.    { s
46d0: 74 72 65 73 73 32 2c 20 20 20 20 20 20 20 20 20  tress2,         
46e0: 20 20 20 20 22 73 74 72 65 73 73 32 22 2c 20 36      "stress2", 6
46f0: 30 30 30 30 20 7d 2c 0a 20 20 7d 3b 0a 20 20 73  0000 },.  };.  s
4700: 74 61 74 69 63 20 63 68 61 72 20 2a 73 75 62 73  tatic char *subs
4710: 74 41 72 67 76 5b 5d 20 3d 20 7b 20 30 2c 20 22  tArgv[] = { 0, "
4720: 2a 22 2c 20 30 20 7d 3b 0a 20 20 69 6e 74 20 69  *", 0 };.  int i
4730: 2c 20 69 41 72 67 3b 0a 20 20 69 6e 74 20 6e 54  , iArg;.  int nT
4740: 65 73 74 66 6f 75 6e 64 20 3d 20 30 3b 0a 0a 20  estfound = 0;.. 
4750: 20 73 71 6c 69 74 65 33 5f 63 6f 6e 66 69 67 28   sqlite3_config(
4760: 53 51 4c 49 54 45 5f 43 4f 4e 46 49 47 5f 4d 55  SQLITE_CONFIG_MU
4770: 4c 54 49 54 48 52 45 41 44 29 3b 0a 20 20 69 66  LTITHREAD);.  if
4780: 28 20 61 72 67 63 3c 32 20 29 7b 0a 20 20 20 20  ( argc<2 ){.    
4790: 61 72 67 63 20 3d 20 32 3b 0a 20 20 20 20 61 72  argc = 2;.    ar
47a0: 67 76 20 3d 20 73 75 62 73 74 41 72 67 76 3b 0a  gv = substArgv;.
47b0: 20 20 7d 0a 0a 20 20 2f 2a 20 4c 6f 6f 70 20 74    }..  /* Loop t
47c0: 68 72 6f 75 67 68 20 74 68 65 20 63 6f 6d 6d 61  hrough the comma
47d0: 6e 64 2d 6c 69 6e 65 20 61 72 67 75 6d 65 6e 74  nd-line argument
47e0: 73 20 74 6f 20 65 6e 73 75 72 65 20 74 68 61 74  s to ensure that
47f0: 20 65 61 63 68 20 61 72 67 75 6d 65 6e 74 0a 20   each argument. 
4800: 20 2a 2a 20 73 65 6c 65 63 74 73 20 61 74 20 6c   ** selects at l
4810: 65 61 73 74 20 6f 6e 65 20 74 65 73 74 2e 20 49  east one test. I
4820: 66 20 6e 6f 74 2c 20 61 73 73 75 6d 65 20 74 68  f not, assume th
4830: 65 72 65 20 69 73 20 61 20 74 79 70 6f 20 6f 6e  ere is a typo on
4840: 20 74 68 65 20 0a 20 20 2a 2a 20 63 6f 6d 6d 61   the .  ** comma
4850: 6e 64 2d 6c 69 6e 65 20 61 6e 64 20 62 61 69 6c  nd-line and bail
4860: 20 6f 75 74 20 77 69 74 68 20 74 68 65 20 75 73   out with the us
4870: 61 67 65 20 6d 65 73 73 61 67 65 2e 20 20 2a 2f  age message.  */
4880: 0a 20 20 66 6f 72 28 69 41 72 67 3d 31 3b 20 69  .  for(iArg=1; i
4890: 41 72 67 3c 61 72 67 63 3b 20 69 41 72 67 2b 2b  Arg<argc; iArg++
48a0: 29 7b 0a 20 20 20 20 63 6f 6e 73 74 20 63 68 61  ){.    const cha
48b0: 72 20 2a 7a 41 72 67 20 3d 20 61 72 67 76 5b 69  r *zArg = argv[i
48c0: 41 72 67 5d 3b 0a 20 20 20 20 69 66 28 20 7a 41  Arg];.    if( zA
48d0: 72 67 5b 30 5d 3d 3d 27 2d 27 20 29 7b 0a 20 20  rg[0]=='-' ){.  
48e0: 20 20 20 20 69 66 28 20 73 71 6c 69 74 65 33 5f      if( sqlite3_
48f0: 73 74 72 69 63 6d 70 28 7a 41 72 67 2c 20 22 2d  stricmp(zArg, "-
4900: 6d 75 6c 74 69 70 6c 65 78 6f 72 22 29 3d 3d 30  multiplexor")==0
4910: 20 29 7b 0a 20 20 20 20 20 20 20 20 2f 2a 20 49   ){.        /* I
4920: 6e 73 74 61 6c 6c 20 74 68 65 20 6d 75 6c 74 69  nstall the multi
4930: 70 6c 65 78 6f 72 20 56 46 53 20 61 73 20 74 68  plexor VFS as th
4940: 65 20 64 65 66 61 75 6c 74 20 2a 2f 0a 20 20 20  e default */.   
4950: 20 20 20 20 20 69 6e 74 20 72 63 20 3d 20 73 71       int rc = sq
4960: 6c 69 74 65 33 5f 6d 75 6c 74 69 70 6c 65 78 5f  lite3_multiplex_
4970: 69 6e 69 74 69 61 6c 69 7a 65 28 30 2c 20 31 29  initialize(0, 1)
4980: 3b 0a 20 20 20 20 20 20 20 20 69 66 28 20 72 63  ;.        if( rc
4990: 21 3d 53 51 4c 49 54 45 5f 4f 4b 20 29 7b 0a 20  !=SQLITE_OK ){. 
49a0: 20 20 20 20 20 20 20 20 20 66 70 72 69 6e 74 66           fprintf
49b0: 28 73 74 64 65 72 72 2c 20 22 46 61 69 6c 65 64  (stderr, "Failed
49c0: 20 74 6f 20 69 6e 73 74 61 6c 6c 20 6d 75 6c 74   to install mult
49d0: 69 70 6c 65 78 6f 72 20 56 46 53 20 28 25 64 29  iplexor VFS (%d)
49e0: 5c 6e 22 2c 20 72 63 29 3b 0a 20 20 20 20 20 20  \n", rc);.      
49f0: 20 20 20 20 72 65 74 75 72 6e 20 32 35 33 3b 0a      return 253;.
4a00: 20 20 20 20 20 20 20 20 7d 0a 20 20 20 20 20 20          }.      
4a10: 7d 0a 20 20 20 20 20 20 65 6c 73 65 20 7b 0a 20  }.      else {. 
4a20: 20 20 20 20 20 20 20 67 6f 74 6f 20 75 73 61 67         goto usag
4a30: 65 3b 0a 20 20 20 20 20 20 7d 0a 0a 20 20 20 20  e;.      }..    
4a40: 20 20 63 6f 6e 74 69 6e 75 65 3b 0a 20 20 20 20    continue;.    
4a50: 7d 0a 0a 20 20 20 20 66 6f 72 28 69 3d 30 3b 20  }..    for(i=0; 
4a60: 69 3c 73 69 7a 65 6f 66 28 61 54 65 73 74 29 2f  i<sizeof(aTest)/
4a70: 73 69 7a 65 6f 66 28 61 54 65 73 74 5b 30 5d 29  sizeof(aTest[0])
4a80: 3b 20 69 2b 2b 29 7b 0a 20 20 20 20 20 20 69 66  ; i++){.      if
4a90: 28 20 73 71 6c 69 74 65 33 5f 73 74 72 67 6c 6f  ( sqlite3_strglo
4aa0: 62 28 7a 41 72 67 2c 20 61 54 65 73 74 5b 69 5d  b(zArg, aTest[i]
4ab0: 2e 7a 54 65 73 74 29 3d 3d 30 20 29 20 62 72 65  .zTest)==0 ) bre
4ac0: 61 6b 3b 0a 20 20 20 20 7d 0a 20 20 20 20 69 66  ak;.    }.    if
4ad0: 28 20 69 3e 3d 73 69 7a 65 6f 66 28 61 54 65 73  ( i>=sizeof(aTes
4ae0: 74 29 2f 73 69 7a 65 6f 66 28 61 54 65 73 74 5b  t)/sizeof(aTest[
4af0: 30 5d 29 20 29 20 67 6f 74 6f 20 75 73 61 67 65  0]) ) goto usage
4b00: 3b 20 20 20 0a 20 20 7d 0a 0a 20 20 66 6f 72 28  ;   .  }..  for(
4b10: 69 41 72 67 3d 31 3b 20 69 41 72 67 3c 61 72 67  iArg=1; iArg<arg
4b20: 63 3b 20 69 41 72 67 2b 2b 29 7b 0a 20 20 20 20  c; iArg++){.    
4b30: 69 66 28 20 61 72 67 76 5b 69 41 72 67 5d 5b 30  if( argv[iArg][0
4b40: 5d 3d 3d 27 2d 27 20 29 20 63 6f 6e 74 69 6e 75  ]=='-' ) continu
4b50: 65 3b 0a 20 20 20 20 66 6f 72 28 69 3d 30 3b 20  e;.    for(i=0; 
4b60: 69 3c 73 69 7a 65 6f 66 28 61 54 65 73 74 29 2f  i<sizeof(aTest)/
4b70: 73 69 7a 65 6f 66 28 61 54 65 73 74 5b 30 5d 29  sizeof(aTest[0])
4b80: 3b 20 69 2b 2b 29 7b 0a 20 20 20 20 20 20 63 68  ; i++){.      ch
4b90: 61 72 20 63 6f 6e 73 74 20 2a 7a 20 3d 20 61 54  ar const *z = aT
4ba0: 65 73 74 5b 69 5d 2e 7a 54 65 73 74 3b 0a 20 20  est[i].zTest;.  
4bb0: 20 20 20 20 69 66 28 20 73 71 6c 69 74 65 33 5f      if( sqlite3_
4bc0: 73 74 72 67 6c 6f 62 28 61 72 67 76 5b 69 41 72  strglob(argv[iAr
4bd0: 67 5d 2c 7a 29 3d 3d 30 20 29 7b 0a 20 20 20 20  g],z)==0 ){.    
4be0: 20 20 20 20 70 72 69 6e 74 66 28 22 52 75 6e 6e      printf("Runn
4bf0: 69 6e 67 20 25 73 20 66 6f 72 20 25 64 20 73 65  ing %s for %d se
4c00: 63 6f 6e 64 73 2e 2e 2e 5c 6e 22 2c 20 7a 2c 20  conds...\n", z, 
4c10: 61 54 65 73 74 5b 69 5d 2e 6e 4d 73 2f 31 30 30  aTest[i].nMs/100
4c20: 30 29 3b 0a 20 20 20 20 20 20 20 20 66 66 6c 75  0);.        fflu
4c30: 73 68 28 73 74 64 6f 75 74 29 3b 0a 20 20 20 20  sh(stdout);.    
4c40: 20 20 20 20 61 54 65 73 74 5b 69 5d 2e 78 54 65      aTest[i].xTe
4c50: 73 74 28 61 54 65 73 74 5b 69 5d 2e 6e 4d 73 29  st(aTest[i].nMs)
4c60: 3b 0a 20 20 20 20 20 20 20 20 6e 54 65 73 74 66  ;.        nTestf
4c70: 6f 75 6e 64 2b 2b 3b 0a 20 20 20 20 20 20 7d 0a  ound++;.      }.
4c80: 20 20 20 20 7d 0a 20 20 7d 0a 20 20 69 66 28 20      }.  }.  if( 
4c90: 6e 54 65 73 74 66 6f 75 6e 64 3d 3d 30 20 29 20  nTestfound==0 ) 
4ca0: 67 6f 74 6f 20 75 73 61 67 65 3b 0a 0a 20 20 70  goto usage;..  p
4cb0: 72 69 6e 74 66 28 22 25 64 20 65 72 72 6f 72 73  rintf("%d errors
4cc0: 20 6f 75 74 20 6f 66 20 25 64 20 74 65 73 74 73   out of %d tests
4cd0: 5c 6e 22 2c 20 6e 47 6c 6f 62 61 6c 45 72 72 2c  \n", nGlobalErr,
4ce0: 20 6e 54 65 73 74 66 6f 75 6e 64 29 3b 0a 20 20   nTestfound);.  
4cf0: 72 65 74 75 72 6e 20 28 6e 47 6c 6f 62 61 6c 45  return (nGlobalE
4d00: 72 72 3e 30 20 3f 20 32 35 35 20 3a 20 30 29 3b  rr>0 ? 255 : 0);
4d10: 0a 0a 20 75 73 61 67 65 3a 0a 20 20 70 72 69 6e  .. usage:.  prin
4d20: 74 66 28 22 55 73 61 67 65 3a 20 25 73 20 5b 2d  tf("Usage: %s [-
4d30: 6d 75 6c 74 69 70 6c 65 78 6f 72 5d 20 5b 74 65  multiplexor] [te
4d40: 73 74 6e 61 6d 65 7c 74 65 73 74 70 72 65 66 69  stname|testprefi
4d50: 78 2a 5d 2e 2e 2e 5c 6e 22 2c 20 61 72 67 76 5b  x*]...\n", argv[
4d60: 30 5d 29 3b 0a 20 20 70 72 69 6e 74 66 28 22 41  0]);.  printf("A
4d70: 76 61 69 6c 61 62 6c 65 20 74 65 73 74 73 20 61  vailable tests a
4d80: 72 65 3a 5c 6e 22 29 3b 0a 20 20 66 6f 72 28 69  re:\n");.  for(i
4d90: 3d 30 3b 20 69 3c 73 69 7a 65 6f 66 28 61 54 65  =0; i<sizeof(aTe
4da0: 73 74 29 2f 73 69 7a 65 6f 66 28 61 54 65 73 74  st)/sizeof(aTest
4db0: 5b 30 5d 29 3b 20 69 2b 2b 29 7b 0a 20 20 20 20  [0]); i++){.    
4dc0: 70 72 69 6e 74 66 28 22 20 20 20 25 73 5c 6e 22  printf("   %s\n"
4dd0: 2c 20 61 54 65 73 74 5b 69 5d 2e 7a 54 65 73 74  , aTest[i].zTest
4de0: 29 3b 0a 20 20 7d 0a 0a 20 20 72 65 74 75 72 6e  );.  }..  return
4df0: 20 32 35 34 3b 0a 7d 0a                           254;.}.