/ Hex Artifact Content
Login

Artifact c2c0f5dc907f1346f5d4b65eb5799f11eb9e4071:


0000: 2f 2a 0a 2a 2a 20 32 30 30 33 20 44 65 63 65 6d  /*.** 2003 Decem
0010: 62 65 72 20 31 38 0a 2a 2a 0a 2a 2a 20 54 68 65  ber 18.**.** The
0020: 20 61 75 74 68 6f 72 20 64 69 73 63 6c 61 69 6d   author disclaim
0030: 73 20 63 6f 70 79 72 69 67 68 74 20 74 6f 20 74  s copyright to t
0040: 68 69 73 20 73 6f 75 72 63 65 20 63 6f 64 65 2e  his source code.
0050: 20 20 49 6e 20 70 6c 61 63 65 20 6f 66 0a 2a 2a    In place of.**
0060: 20 61 20 6c 65 67 61 6c 20 6e 6f 74 69 63 65 2c   a legal notice,
0070: 20 68 65 72 65 20 69 73 20 61 20 62 6c 65 73 73   here is a bless
0080: 69 6e 67 3a 0a 2a 2a 0a 2a 2a 20 20 20 20 4d 61  ing:.**.**    Ma
0090: 79 20 79 6f 75 20 64 6f 20 67 6f 6f 64 20 61 6e  y you do good an
00a0: 64 20 6e 6f 74 20 65 76 69 6c 2e 0a 2a 2a 20 20  d not evil..**  
00b0: 20 20 4d 61 79 20 79 6f 75 20 66 69 6e 64 20 66    May you find f
00c0: 6f 72 67 69 76 65 6e 65 73 73 20 66 6f 72 20 79  orgiveness for y
00d0: 6f 75 72 73 65 6c 66 20 61 6e 64 20 66 6f 72 67  ourself and forg
00e0: 69 76 65 20 6f 74 68 65 72 73 2e 0a 2a 2a 20 20  ive others..**  
00f0: 20 20 4d 61 79 20 79 6f 75 20 73 68 61 72 65 20    May you share 
0100: 66 72 65 65 6c 79 2c 20 6e 65 76 65 72 20 74 61  freely, never ta
0110: 6b 69 6e 67 20 6d 6f 72 65 20 74 68 61 6e 20 79  king more than y
0120: 6f 75 20 67 69 76 65 2e 0a 2a 2a 0a 2a 2a 2a 2a  ou give..**.****
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 2a  ****************
0170: 2a 2a 2a 2a 2a 0a 2a 2a 20 43 6f 64 65 20 66 6f  *****.** Code fo
0180: 72 20 74 65 73 74 69 6e 67 20 74 68 65 20 74 68  r testing the th
0190: 65 20 53 51 4c 69 74 65 20 6c 69 62 72 61 72 79  e SQLite library
01a0: 20 69 6e 20 61 20 6d 75 6c 74 69 74 68 72 65 61   in a multithrea
01b0: 64 65 64 20 65 6e 76 69 72 6f 6e 6d 65 6e 74 2e  ded environment.
01c0: 0a 2a 2a 0a 2a 2a 20 24 49 64 3a 20 74 65 73 74  .**.** $Id: test
01d0: 34 2e 63 2c 76 20 31 2e 32 31 20 32 30 30 37 2f  4.c,v 1.21 2007/
01e0: 30 38 2f 32 32 20 31 31 3a 34 31 3a 31 38 20 64  08/22 11:41:18 d
01f0: 72 68 20 45 78 70 20 24 0a 2a 2f 0a 23 69 6e 63  rh Exp $.*/.#inc
0200: 6c 75 64 65 20 22 73 71 6c 69 74 65 49 6e 74 2e  lude "sqliteInt.
0210: 68 22 0a 23 69 6e 63 6c 75 64 65 20 22 74 63 6c  h".#include "tcl
0220: 2e 68 22 0a 23 69 66 20 64 65 66 69 6e 65 64 28  .h".#if defined(
0230: 4f 53 5f 55 4e 49 58 29 20 26 26 20 4f 53 5f 55  OS_UNIX) && OS_U
0240: 4e 49 58 3d 3d 31 20 26 26 20 53 51 4c 49 54 45  NIX==1 && SQLITE
0250: 5f 54 48 52 45 41 44 53 41 46 45 0a 23 69 6e 63  _THREADSAFE.#inc
0260: 6c 75 64 65 20 3c 73 74 64 6c 69 62 2e 68 3e 0a  lude <stdlib.h>.
0270: 23 69 6e 63 6c 75 64 65 20 3c 73 74 72 69 6e 67  #include <string
0280: 2e 68 3e 0a 23 69 6e 63 6c 75 64 65 20 3c 70 74  .h>.#include <pt
0290: 68 72 65 61 64 2e 68 3e 0a 23 69 6e 63 6c 75 64  hread.h>.#includ
02a0: 65 20 3c 73 63 68 65 64 2e 68 3e 0a 23 69 6e 63  e <sched.h>.#inc
02b0: 6c 75 64 65 20 3c 63 74 79 70 65 2e 68 3e 0a 0a  lude <ctype.h>..
02c0: 2f 2a 0a 2a 2a 20 45 61 63 68 20 74 68 72 65 61  /*.** Each threa
02d0: 64 20 69 73 20 63 6f 6e 74 72 6f 6c 6c 65 64 20  d is controlled 
02e0: 62 79 20 61 6e 20 69 6e 73 74 61 6e 63 65 20 6f  by an instance o
02f0: 66 20 74 68 65 20 66 6f 6c 6c 6f 77 69 6e 67 0a  f the following.
0300: 2a 2a 20 73 74 72 75 63 74 75 72 65 2e 0a 2a 2f  ** structure..*/
0310: 0a 74 79 70 65 64 65 66 20 73 74 72 75 63 74 20  .typedef struct 
0320: 54 68 72 65 61 64 20 54 68 72 65 61 64 3b 0a 73  Thread Thread;.s
0330: 74 72 75 63 74 20 54 68 72 65 61 64 20 7b 0a 20  truct Thread {. 
0340: 20 2f 2a 20 54 68 65 20 66 69 72 73 74 20 67 72   /* The first gr
0350: 6f 75 70 20 6f 66 20 66 69 65 6c 64 73 20 61 72  oup of fields ar
0360: 65 20 77 72 69 74 61 62 6c 65 20 62 79 20 74 68  e writable by th
0370: 65 20 6d 61 73 74 65 72 20 61 6e 64 20 72 65 61  e master and rea
0380: 64 2d 6f 6e 6c 79 0a 20 20 2a 2a 20 74 6f 20 74  d-only.  ** to t
0390: 68 65 20 74 68 72 65 61 64 2e 20 2a 2f 0a 20 20  he thread. */.  
03a0: 63 68 61 72 20 2a 7a 46 69 6c 65 6e 61 6d 65 3b  char *zFilename;
03b0: 20 20 20 20 20 20 20 2f 2a 20 4e 61 6d 65 20 6f         /* Name o
03c0: 66 20 64 61 74 61 62 61 73 65 20 66 69 6c 65 20  f database file 
03d0: 2a 2f 0a 20 20 76 6f 69 64 20 28 2a 78 4f 70 29  */.  void (*xOp)
03e0: 28 54 68 72 65 61 64 2a 29 3b 20 20 2f 2a 20 6e  (Thread*);  /* n
03f0: 65 78 74 20 6f 70 65 72 61 74 69 6f 6e 20 74 6f  ext operation to
0400: 20 64 6f 20 2a 2f 0a 20 20 63 68 61 72 20 2a 7a   do */.  char *z
0410: 41 72 67 3b 20 20 20 20 20 20 20 20 20 20 20 20  Arg;            
0420: 2f 2a 20 61 72 67 75 6d 65 6e 74 20 75 73 61 62  /* argument usab
0430: 6c 65 20 62 79 20 78 4f 70 20 2a 2f 0a 20 20 69  le by xOp */.  i
0440: 6e 74 20 6f 70 6e 75 6d 3b 20 20 20 20 20 20 20  nt opnum;       
0450: 20 20 20 20 20 20 2f 2a 20 4f 70 65 72 61 74 69        /* Operati
0460: 6f 6e 20 6e 75 6d 62 65 72 20 2a 2f 0a 20 20 69  on number */.  i
0470: 6e 74 20 62 75 73 79 3b 20 20 20 20 20 20 20 20  nt busy;        
0480: 20 20 20 20 20 20 2f 2a 20 54 72 75 65 20 69 66        /* True if
0490: 20 74 68 69 73 20 74 68 72 65 61 64 20 69 73 20   this thread is 
04a0: 69 6e 20 75 73 65 20 2a 2f 0a 0a 20 20 2f 2a 20  in use */..  /* 
04b0: 54 68 65 20 6e 65 78 74 20 67 72 6f 75 70 20 6f  The next group o
04c0: 66 20 66 69 65 6c 64 73 20 61 72 65 20 77 72 69  f fields are wri
04d0: 74 61 62 6c 65 20 62 79 20 74 68 65 20 74 68 72  table by the thr
04e0: 65 61 64 20 62 75 74 20 72 65 61 64 2d 6f 6e 6c  ead but read-onl
04f0: 79 20 74 6f 20 74 68 65 0a 20 20 2a 2a 20 6d 61  y to the.  ** ma
0500: 73 74 65 72 2e 20 2a 2f 0a 20 20 69 6e 74 20 63  ster. */.  int c
0510: 6f 6d 70 6c 65 74 65 64 3b 20 20 20 20 20 20 20  ompleted;       
0520: 20 2f 2a 20 4e 75 6d 62 65 72 20 6f 66 20 6f 70   /* Number of op
0530: 65 72 61 74 69 6f 6e 73 20 63 6f 6d 70 6c 65 74  erations complet
0540: 65 64 20 2a 2f 0a 20 20 73 71 6c 69 74 65 33 20  ed */.  sqlite3 
0550: 2a 64 62 3b 20 20 20 20 20 20 20 20 20 20 20 2f  *db;           /
0560: 2a 20 4f 70 65 6e 20 64 61 74 61 62 61 73 65 20  * Open database 
0570: 2a 2f 0a 20 20 73 71 6c 69 74 65 33 5f 73 74 6d  */.  sqlite3_stm
0580: 74 20 2a 70 53 74 6d 74 3b 20 20 20 20 20 2f 2a  t *pStmt;     /*
0590: 20 50 65 6e 64 69 6e 67 20 6f 70 65 72 61 74 69   Pending operati
05a0: 6f 6e 20 2a 2f 0a 20 20 63 68 61 72 20 2a 7a 45  on */.  char *zE
05b0: 72 72 3b 20 20 20 20 20 20 20 20 20 20 20 2f 2a  rr;           /*
05c0: 20 6f 70 65 72 61 74 69 6f 6e 20 65 72 72 6f 72   operation error
05d0: 20 2a 2f 0a 20 20 63 68 61 72 20 2a 7a 53 74 61   */.  char *zSta
05e0: 74 69 63 45 72 72 3b 20 20 20 20 20 2f 2a 20 53  ticErr;     /* S
05f0: 74 61 74 69 63 20 65 72 72 6f 72 20 6d 65 73 73  tatic error mess
0600: 61 67 65 20 2a 2f 0a 20 20 69 6e 74 20 72 63 3b  age */.  int rc;
0610: 20 20 20 20 20 20 20 20 20 20 20 20 20 20 20 2f                 /
0620: 2a 20 6f 70 65 72 61 74 69 6f 6e 20 72 65 74 75  * operation retu
0630: 72 6e 20 63 6f 64 65 20 2a 2f 0a 20 20 69 6e 74  rn code */.  int
0640: 20 61 72 67 63 3b 20 20 20 20 20 20 20 20 20 20   argc;          
0650: 20 20 20 2f 2a 20 6e 75 6d 62 65 72 20 6f 66 20     /* number of 
0660: 63 6f 6c 75 6d 6e 73 20 69 6e 20 72 65 73 75 6c  columns in resul
0670: 74 20 2a 2f 0a 20 20 63 6f 6e 73 74 20 63 68 61  t */.  const cha
0680: 72 20 2a 61 72 67 76 5b 31 30 30 5d 3b 20 20 20  r *argv[100];   
0690: 20 2f 2a 20 72 65 73 75 6c 74 20 63 6f 6c 75 6d   /* result colum
06a0: 6e 73 20 2a 2f 0a 20 20 63 6f 6e 73 74 20 63 68  ns */.  const ch
06b0: 61 72 20 2a 63 6f 6c 76 5b 31 30 30 5d 3b 20 20  ar *colv[100];  
06c0: 20 20 2f 2a 20 72 65 73 75 6c 74 20 63 6f 6c 75    /* result colu
06d0: 6d 6e 20 6e 61 6d 65 73 20 2a 2f 0a 7d 3b 0a 0a  mn names */.};..
06e0: 2f 2a 0a 2a 2a 20 54 68 65 72 65 20 63 61 6e 20  /*.** There can 
06f0: 62 65 20 61 73 20 6d 61 6e 79 20 61 73 20 32 36  be as many as 26
0700: 20 74 68 72 65 61 64 73 20 72 75 6e 6e 69 6e 67   threads running
0710: 20 61 74 20 6f 6e 63 65 2e 20 20 45 61 63 68 20   at once.  Each 
0720: 69 73 20 6e 61 6d 65 64 0a 2a 2a 20 62 79 20 61  is named.** by a
0730: 20 63 61 70 69 74 61 6c 20 6c 65 74 74 65 72 3a   capital letter:
0740: 20 41 2c 20 42 2c 20 43 2c 20 2e 2e 2e 2c 20 59   A, B, C, ..., Y
0750: 2c 20 5a 2e 0a 2a 2f 0a 23 64 65 66 69 6e 65 20  , Z..*/.#define 
0760: 4e 5f 54 48 52 45 41 44 20 32 36 0a 73 74 61 74  N_THREAD 26.stat
0770: 69 63 20 54 68 72 65 61 64 20 74 68 72 65 61 64  ic Thread thread
0780: 73 65 74 5b 4e 5f 54 48 52 45 41 44 5d 3b 0a 0a  set[N_THREAD];..
0790: 0a 2f 2a 0a 2a 2a 20 54 68 65 20 6d 61 69 6e 20  ./*.** The main 
07a0: 6c 6f 6f 70 20 66 6f 72 20 61 20 74 68 72 65 61  loop for a threa
07b0: 64 2e 20 20 54 68 72 65 61 64 73 20 75 73 65 20  d.  Threads use 
07c0: 62 75 73 79 20 77 61 69 74 69 6e 67 2e 20 0a 2a  busy waiting. .*
07d0: 2f 0a 73 74 61 74 69 63 20 76 6f 69 64 20 2a 74  /.static void *t
07e0: 68 72 65 61 64 5f 6d 61 69 6e 28 76 6f 69 64 20  hread_main(void 
07f0: 2a 70 41 72 67 29 7b 0a 20 20 54 68 72 65 61 64  *pArg){.  Thread
0800: 20 2a 70 20 3d 20 28 54 68 72 65 61 64 2a 29 70   *p = (Thread*)p
0810: 41 72 67 3b 0a 20 20 69 66 28 20 70 2d 3e 64 62  Arg;.  if( p->db
0820: 20 29 7b 0a 20 20 20 20 73 71 6c 69 74 65 33 5f   ){.    sqlite3_
0830: 63 6c 6f 73 65 28 70 2d 3e 64 62 29 3b 0a 20 20  close(p->db);.  
0840: 7d 0a 20 20 73 71 6c 69 74 65 33 5f 6f 70 65 6e  }.  sqlite3_open
0850: 28 70 2d 3e 7a 46 69 6c 65 6e 61 6d 65 2c 20 26  (p->zFilename, &
0860: 70 2d 3e 64 62 29 3b 0a 20 20 69 66 28 20 53 51  p->db);.  if( SQ
0870: 4c 49 54 45 5f 4f 4b 21 3d 73 71 6c 69 74 65 33  LITE_OK!=sqlite3
0880: 5f 65 72 72 63 6f 64 65 28 70 2d 3e 64 62 29 20  _errcode(p->db) 
0890: 29 7b 0a 20 20 20 20 70 2d 3e 7a 45 72 72 20 3d  ){.    p->zErr =
08a0: 20 73 74 72 64 75 70 28 73 71 6c 69 74 65 33 5f   strdup(sqlite3_
08b0: 65 72 72 6d 73 67 28 70 2d 3e 64 62 29 29 3b 0a  errmsg(p->db));.
08c0: 20 20 20 20 73 71 6c 69 74 65 33 5f 63 6c 6f 73      sqlite3_clos
08d0: 65 28 70 2d 3e 64 62 29 3b 0a 20 20 20 20 70 2d  e(p->db);.    p-
08e0: 3e 64 62 20 3d 20 30 3b 0a 20 20 7d 0a 20 20 70  >db = 0;.  }.  p
08f0: 2d 3e 70 53 74 6d 74 20 3d 20 30 3b 0a 20 20 70  ->pStmt = 0;.  p
0900: 2d 3e 63 6f 6d 70 6c 65 74 65 64 20 3d 20 31 3b  ->completed = 1;
0910: 0a 20 20 77 68 69 6c 65 28 20 70 2d 3e 6f 70 6e  .  while( p->opn
0920: 75 6d 3c 3d 70 2d 3e 63 6f 6d 70 6c 65 74 65 64  um<=p->completed
0930: 20 29 20 73 63 68 65 64 5f 79 69 65 6c 64 28 29   ) sched_yield()
0940: 3b 0a 20 20 77 68 69 6c 65 28 20 70 2d 3e 78 4f  ;.  while( p->xO
0950: 70 20 29 7b 0a 20 20 20 20 69 66 28 20 70 2d 3e  p ){.    if( p->
0960: 7a 45 72 72 20 26 26 20 70 2d 3e 7a 45 72 72 21  zErr && p->zErr!
0970: 3d 70 2d 3e 7a 53 74 61 74 69 63 45 72 72 20 29  =p->zStaticErr )
0980: 7b 0a 20 20 20 20 20 20 73 71 6c 69 74 65 33 5f  {.      sqlite3_
0990: 66 72 65 65 28 70 2d 3e 7a 45 72 72 29 3b 0a 20  free(p->zErr);. 
09a0: 20 20 20 20 20 70 2d 3e 7a 45 72 72 20 3d 20 30       p->zErr = 0
09b0: 3b 0a 20 20 20 20 7d 0a 20 20 20 20 28 2a 70 2d  ;.    }.    (*p-
09c0: 3e 78 4f 70 29 28 70 29 3b 0a 20 20 20 20 70 2d  >xOp)(p);.    p-
09d0: 3e 63 6f 6d 70 6c 65 74 65 64 2b 2b 3b 0a 20 20  >completed++;.  
09e0: 20 20 77 68 69 6c 65 28 20 70 2d 3e 6f 70 6e 75    while( p->opnu
09f0: 6d 3c 3d 70 2d 3e 63 6f 6d 70 6c 65 74 65 64 20  m<=p->completed 
0a00: 29 20 73 63 68 65 64 5f 79 69 65 6c 64 28 29 3b  ) sched_yield();
0a10: 0a 20 20 7d 0a 20 20 69 66 28 20 70 2d 3e 70 53  .  }.  if( p->pS
0a20: 74 6d 74 20 29 7b 0a 20 20 20 20 73 71 6c 69 74  tmt ){.    sqlit
0a30: 65 33 5f 66 69 6e 61 6c 69 7a 65 28 70 2d 3e 70  e3_finalize(p->p
0a40: 53 74 6d 74 29 3b 0a 20 20 20 20 70 2d 3e 70 53  Stmt);.    p->pS
0a50: 74 6d 74 20 3d 20 30 3b 0a 20 20 7d 0a 20 20 69  tmt = 0;.  }.  i
0a60: 66 28 20 70 2d 3e 64 62 20 29 7b 0a 20 20 20 20  f( p->db ){.    
0a70: 73 71 6c 69 74 65 33 5f 63 6c 6f 73 65 28 70 2d  sqlite3_close(p-
0a80: 3e 64 62 29 3b 0a 20 20 20 20 70 2d 3e 64 62 20  >db);.    p->db 
0a90: 3d 20 30 3b 0a 20 20 7d 0a 20 20 69 66 28 20 70  = 0;.  }.  if( p
0aa0: 2d 3e 7a 45 72 72 20 26 26 20 70 2d 3e 7a 45 72  ->zErr && p->zEr
0ab0: 72 21 3d 70 2d 3e 7a 53 74 61 74 69 63 45 72 72  r!=p->zStaticErr
0ac0: 20 29 7b 0a 20 20 20 20 73 71 6c 69 74 65 33 5f   ){.    sqlite3_
0ad0: 66 72 65 65 28 70 2d 3e 7a 45 72 72 29 3b 0a 20  free(p->zErr);. 
0ae0: 20 20 20 70 2d 3e 7a 45 72 72 20 3d 20 30 3b 0a     p->zErr = 0;.
0af0: 20 20 7d 0a 20 20 70 2d 3e 63 6f 6d 70 6c 65 74    }.  p->complet
0b00: 65 64 2b 2b 3b 0a 20 20 73 71 6c 69 74 65 33 5f  ed++;.  sqlite3_
0b10: 74 68 72 65 61 64 5f 63 6c 65 61 6e 75 70 28 29  thread_cleanup()
0b20: 3b 0a 20 20 72 65 74 75 72 6e 20 30 3b 0a 7d 0a  ;.  return 0;.}.
0b30: 0a 2f 2a 0a 2a 2a 20 47 65 74 20 61 20 74 68 72  ./*.** Get a thr
0b40: 65 61 64 20 49 44 20 77 68 69 63 68 20 69 73 20  ead ID which is 
0b50: 61 6e 20 75 70 70 65 72 20 63 61 73 65 20 6c 65  an upper case le
0b60: 74 74 65 72 2e 20 20 52 65 74 75 72 6e 20 74 68  tter.  Return th
0b70: 65 20 69 6e 64 65 78 2e 0a 2a 2a 20 49 66 20 74  e index..** If t
0b80: 68 65 20 61 72 67 75 6d 65 6e 74 20 69 73 20 6e  he argument is n
0b90: 6f 74 20 61 20 76 61 6c 69 64 20 74 68 72 65 61  ot a valid threa
0ba0: 64 20 49 44 20 70 75 74 20 61 6e 20 65 72 72 6f  d ID put an erro
0bb0: 72 20 6d 65 73 73 61 67 65 20 69 6e 0a 2a 2a 20  r message in.** 
0bc0: 74 68 65 20 69 6e 74 65 72 70 72 65 74 65 72 20  the interpreter 
0bd0: 61 6e 64 20 72 65 74 75 72 6e 20 2d 31 2e 0a 2a  and return -1..*
0be0: 2f 0a 73 74 61 74 69 63 20 69 6e 74 20 70 61 72  /.static int par
0bf0: 73 65 5f 74 68 72 65 61 64 5f 69 64 28 54 63 6c  se_thread_id(Tcl
0c00: 5f 49 6e 74 65 72 70 20 2a 69 6e 74 65 72 70 2c  _Interp *interp,
0c10: 20 63 6f 6e 73 74 20 63 68 61 72 20 2a 7a 41 72   const char *zAr
0c20: 67 29 7b 0a 20 20 69 66 28 20 7a 41 72 67 3d 3d  g){.  if( zArg==
0c30: 30 20 7c 7c 20 7a 41 72 67 5b 30 5d 3d 3d 30 20  0 || zArg[0]==0 
0c40: 7c 7c 20 7a 41 72 67 5b 31 5d 21 3d 30 20 7c 7c  || zArg[1]!=0 ||
0c50: 20 21 69 73 75 70 70 65 72 28 28 75 6e 73 69 67   !isupper((unsig
0c60: 6e 65 64 20 63 68 61 72 29 7a 41 72 67 5b 30 5d  ned char)zArg[0]
0c70: 29 20 29 7b 0a 20 20 20 20 54 63 6c 5f 41 70 70  ) ){.    Tcl_App
0c80: 65 6e 64 52 65 73 75 6c 74 28 69 6e 74 65 72 70  endResult(interp
0c90: 2c 20 22 74 68 72 65 61 64 20 49 44 20 6d 75 73  , "thread ID mus
0ca0: 74 20 62 65 20 61 6e 20 75 70 70 65 72 20 63 61  t be an upper ca
0cb0: 73 65 20 6c 65 74 74 65 72 22 2c 20 30 29 3b 0a  se letter", 0);.
0cc0: 20 20 20 20 72 65 74 75 72 6e 20 2d 31 3b 0a 20      return -1;. 
0cd0: 20 7d 0a 20 20 72 65 74 75 72 6e 20 7a 41 72 67   }.  return zArg
0ce0: 5b 30 5d 20 2d 20 27 41 27 3b 0a 7d 0a 0a 2f 2a  [0] - 'A';.}../*
0cf0: 0a 2a 2a 20 55 73 61 67 65 3a 20 20 20 20 74 68  .** Usage:    th
0d00: 72 65 61 64 5f 63 72 65 61 74 65 20 4e 41 4d 45  read_create NAME
0d10: 20 20 46 49 4c 45 4e 41 4d 45 0a 2a 2a 0a 2a 2a    FILENAME.**.**
0d20: 20 4e 41 4d 45 20 73 68 6f 75 6c 64 20 62 65 20   NAME should be 
0d30: 61 6e 20 75 70 70 65 72 20 63 61 73 65 20 6c 65  an upper case le
0d40: 74 74 65 72 2e 20 20 53 74 61 72 74 20 74 68 65  tter.  Start the
0d50: 20 74 68 72 65 61 64 20 72 75 6e 6e 69 6e 67 20   thread running 
0d60: 77 69 74 68 0a 2a 2a 20 61 6e 20 6f 70 65 6e 20  with.** an open 
0d70: 63 6f 6e 6e 65 63 74 69 6f 6e 20 74 6f 20 74 68  connection to th
0d80: 65 20 67 69 76 65 6e 20 64 61 74 61 62 61 73 65  e given database
0d90: 2e 0a 2a 2f 0a 73 74 61 74 69 63 20 69 6e 74 20  ..*/.static int 
0da0: 74 63 6c 5f 74 68 72 65 61 64 5f 63 72 65 61 74  tcl_thread_creat
0db0: 65 28 0a 20 20 76 6f 69 64 20 2a 4e 6f 74 55 73  e(.  void *NotUs
0dc0: 65 64 2c 0a 20 20 54 63 6c 5f 49 6e 74 65 72 70  ed,.  Tcl_Interp
0dd0: 20 2a 69 6e 74 65 72 70 2c 20 20 20 20 2f 2a 20   *interp,    /* 
0de0: 54 68 65 20 54 43 4c 20 69 6e 74 65 72 70 72 65  The TCL interpre
0df0: 74 65 72 20 74 68 61 74 20 69 6e 76 6f 6b 65 64  ter that invoked
0e00: 20 74 68 69 73 20 63 6f 6d 6d 61 6e 64 20 2a 2f   this command */
0e10: 0a 20 20 69 6e 74 20 61 72 67 63 2c 20 20 20 20  .  int argc,    
0e20: 20 20 20 20 20 20 20 20 20 20 2f 2a 20 4e 75 6d            /* Num
0e30: 62 65 72 20 6f 66 20 61 72 67 75 6d 65 6e 74 73  ber of arguments
0e40: 20 2a 2f 0a 20 20 63 6f 6e 73 74 20 63 68 61 72   */.  const char
0e50: 20 2a 2a 61 72 67 76 20 20 20 20 20 20 2f 2a 20   **argv      /* 
0e60: 54 65 78 74 20 6f 66 20 65 61 63 68 20 61 72 67  Text of each arg
0e70: 75 6d 65 6e 74 20 2a 2f 0a 29 7b 0a 20 20 69 6e  ument */.){.  in
0e80: 74 20 69 3b 0a 20 20 70 74 68 72 65 61 64 5f 74  t i;.  pthread_t
0e90: 20 78 3b 0a 20 20 69 6e 74 20 72 63 3b 0a 0a 20   x;.  int rc;.. 
0ea0: 20 69 66 28 20 61 72 67 63 21 3d 33 20 29 7b 0a   if( argc!=3 ){.
0eb0: 20 20 20 20 54 63 6c 5f 41 70 70 65 6e 64 52 65      Tcl_AppendRe
0ec0: 73 75 6c 74 28 69 6e 74 65 72 70 2c 20 22 77 72  sult(interp, "wr
0ed0: 6f 6e 67 20 23 20 61 72 67 73 3a 20 73 68 6f 75  ong # args: shou
0ee0: 6c 64 20 62 65 20 5c 22 22 2c 20 61 72 67 76 5b  ld be \"", argv[
0ef0: 30 5d 2c 0a 20 20 20 20 20 20 20 22 20 49 44 20  0],.       " ID 
0f00: 46 49 4c 45 4e 41 4d 45 22 2c 20 30 29 3b 0a 20  FILENAME", 0);. 
0f10: 20 20 20 72 65 74 75 72 6e 20 54 43 4c 5f 45 52     return TCL_ER
0f20: 52 4f 52 3b 0a 20 20 7d 0a 20 20 69 20 3d 20 70  ROR;.  }.  i = p
0f30: 61 72 73 65 5f 74 68 72 65 61 64 5f 69 64 28 69  arse_thread_id(i
0f40: 6e 74 65 72 70 2c 20 61 72 67 76 5b 31 5d 29 3b  nterp, argv[1]);
0f50: 0a 20 20 69 66 28 20 69 3c 30 20 29 20 72 65 74  .  if( i<0 ) ret
0f60: 75 72 6e 20 54 43 4c 5f 45 52 52 4f 52 3b 0a 20  urn TCL_ERROR;. 
0f70: 20 69 66 28 20 74 68 72 65 61 64 73 65 74 5b 69   if( threadset[i
0f80: 5d 2e 62 75 73 79 20 29 7b 0a 20 20 20 20 54 63  ].busy ){.    Tc
0f90: 6c 5f 41 70 70 65 6e 64 52 65 73 75 6c 74 28 69  l_AppendResult(i
0fa0: 6e 74 65 72 70 2c 20 22 74 68 72 65 61 64 20 22  nterp, "thread "
0fb0: 2c 20 61 72 67 76 5b 31 5d 2c 20 22 20 69 73 20  , argv[1], " is 
0fc0: 61 6c 72 65 61 64 79 20 72 75 6e 6e 69 6e 67 22  already running"
0fd0: 2c 20 30 29 3b 0a 20 20 20 20 72 65 74 75 72 6e  , 0);.    return
0fe0: 20 54 43 4c 5f 45 52 52 4f 52 3b 0a 20 20 7d 0a   TCL_ERROR;.  }.
0ff0: 20 20 74 68 72 65 61 64 73 65 74 5b 69 5d 2e 62    threadset[i].b
1000: 75 73 79 20 3d 20 31 3b 0a 20 20 73 71 6c 69 74  usy = 1;.  sqlit
1010: 65 33 5f 66 72 65 65 28 74 68 72 65 61 64 73 65  e3_free(threadse
1020: 74 5b 69 5d 2e 7a 46 69 6c 65 6e 61 6d 65 29 3b  t[i].zFilename);
1030: 0a 20 20 74 68 72 65 61 64 73 65 74 5b 69 5d 2e  .  threadset[i].
1040: 7a 46 69 6c 65 6e 61 6d 65 20 3d 20 73 71 6c 69  zFilename = sqli
1050: 74 65 33 53 74 72 44 75 70 28 61 72 67 76 5b 32  te3StrDup(argv[2
1060: 5d 29 3b 0a 20 20 74 68 72 65 61 64 73 65 74 5b  ]);.  threadset[
1070: 69 5d 2e 6f 70 6e 75 6d 20 3d 20 31 3b 0a 20 20  i].opnum = 1;.  
1080: 74 68 72 65 61 64 73 65 74 5b 69 5d 2e 63 6f 6d  threadset[i].com
1090: 70 6c 65 74 65 64 20 3d 20 30 3b 0a 20 20 72 63  pleted = 0;.  rc
10a0: 20 3d 20 70 74 68 72 65 61 64 5f 63 72 65 61 74   = pthread_creat
10b0: 65 28 26 78 2c 20 30 2c 20 74 68 72 65 61 64 5f  e(&x, 0, thread_
10c0: 6d 61 69 6e 2c 20 26 74 68 72 65 61 64 73 65 74  main, &threadset
10d0: 5b 69 5d 29 3b 0a 20 20 69 66 28 20 72 63 20 29  [i]);.  if( rc )
10e0: 7b 0a 20 20 20 20 54 63 6c 5f 41 70 70 65 6e 64  {.    Tcl_Append
10f0: 52 65 73 75 6c 74 28 69 6e 74 65 72 70 2c 20 22  Result(interp, "
1100: 66 61 69 6c 65 64 20 74 6f 20 63 72 65 61 74 65  failed to create
1110: 20 74 68 65 20 74 68 72 65 61 64 22 2c 20 30 29   the thread", 0)
1120: 3b 0a 20 20 20 20 73 71 6c 69 74 65 33 5f 66 72  ;.    sqlite3_fr
1130: 65 65 28 74 68 72 65 61 64 73 65 74 5b 69 5d 2e  ee(threadset[i].
1140: 7a 46 69 6c 65 6e 61 6d 65 29 3b 0a 20 20 20 20  zFilename);.    
1150: 74 68 72 65 61 64 73 65 74 5b 69 5d 2e 62 75 73  threadset[i].bus
1160: 79 20 3d 20 30 3b 0a 20 20 20 20 72 65 74 75 72  y = 0;.    retur
1170: 6e 20 54 43 4c 5f 45 52 52 4f 52 3b 0a 20 20 7d  n TCL_ERROR;.  }
1180: 0a 20 20 70 74 68 72 65 61 64 5f 64 65 74 61 63  .  pthread_detac
1190: 68 28 78 29 3b 0a 20 20 72 65 74 75 72 6e 20 54  h(x);.  return T
11a0: 43 4c 5f 4f 4b 3b 0a 7d 0a 0a 2f 2a 0a 2a 2a 20  CL_OK;.}../*.** 
11b0: 57 61 69 74 20 66 6f 72 20 61 20 74 68 72 65 61  Wait for a threa
11c0: 64 20 74 6f 20 72 65 61 63 68 20 69 74 73 20 69  d to reach its i
11d0: 64 6c 65 20 73 74 61 74 65 2e 0a 2a 2f 0a 73 74  dle state..*/.st
11e0: 61 74 69 63 20 76 6f 69 64 20 74 68 72 65 61 64  atic void thread
11f0: 5f 77 61 69 74 28 54 68 72 65 61 64 20 2a 70 29  _wait(Thread *p)
1200: 7b 0a 20 20 77 68 69 6c 65 28 20 70 2d 3e 6f 70  {.  while( p->op
1210: 6e 75 6d 3e 70 2d 3e 63 6f 6d 70 6c 65 74 65 64  num>p->completed
1220: 20 29 20 73 63 68 65 64 5f 79 69 65 6c 64 28 29   ) sched_yield()
1230: 3b 0a 7d 0a 0a 2f 2a 0a 2a 2a 20 55 73 61 67 65  ;.}../*.** Usage
1240: 3a 20 20 74 68 72 65 61 64 5f 77 61 69 74 20 49  :  thread_wait I
1250: 44 0a 2a 2a 0a 2a 2a 20 57 61 69 74 20 6f 6e 20  D.**.** Wait on 
1260: 74 68 72 65 61 64 20 49 44 20 74 6f 20 72 65 61  thread ID to rea
1270: 63 68 20 69 74 73 20 69 64 6c 65 20 73 74 61 74  ch its idle stat
1280: 65 2e 0a 2a 2f 0a 73 74 61 74 69 63 20 69 6e 74  e..*/.static int
1290: 20 74 63 6c 5f 74 68 72 65 61 64 5f 77 61 69 74   tcl_thread_wait
12a0: 28 0a 20 20 76 6f 69 64 20 2a 4e 6f 74 55 73 65  (.  void *NotUse
12b0: 64 2c 0a 20 20 54 63 6c 5f 49 6e 74 65 72 70 20  d,.  Tcl_Interp 
12c0: 2a 69 6e 74 65 72 70 2c 20 20 20 20 2f 2a 20 54  *interp,    /* T
12d0: 68 65 20 54 43 4c 20 69 6e 74 65 72 70 72 65 74  he TCL interpret
12e0: 65 72 20 74 68 61 74 20 69 6e 76 6f 6b 65 64 20  er that invoked 
12f0: 74 68 69 73 20 63 6f 6d 6d 61 6e 64 20 2a 2f 0a  this command */.
1300: 20 20 69 6e 74 20 61 72 67 63 2c 20 20 20 20 20    int argc,     
1310: 20 20 20 20 20 20 20 20 20 2f 2a 20 4e 75 6d 62           /* Numb
1320: 65 72 20 6f 66 20 61 72 67 75 6d 65 6e 74 73 20  er of arguments 
1330: 2a 2f 0a 20 20 63 6f 6e 73 74 20 63 68 61 72 20  */.  const char 
1340: 2a 2a 61 72 67 76 20 20 20 20 20 20 2f 2a 20 54  **argv      /* T
1350: 65 78 74 20 6f 66 20 65 61 63 68 20 61 72 67 75  ext of each argu
1360: 6d 65 6e 74 20 2a 2f 0a 29 7b 0a 20 20 69 6e 74  ment */.){.  int
1370: 20 69 3b 0a 0a 20 20 69 66 28 20 61 72 67 63 21   i;..  if( argc!
1380: 3d 32 20 29 7b 0a 20 20 20 20 54 63 6c 5f 41 70  =2 ){.    Tcl_Ap
1390: 70 65 6e 64 52 65 73 75 6c 74 28 69 6e 74 65 72  pendResult(inter
13a0: 70 2c 20 22 77 72 6f 6e 67 20 23 20 61 72 67 73  p, "wrong # args
13b0: 3a 20 73 68 6f 75 6c 64 20 62 65 20 5c 22 22 2c  : should be \"",
13c0: 20 61 72 67 76 5b 30 5d 2c 0a 20 20 20 20 20 20   argv[0],.      
13d0: 20 22 20 49 44 22 2c 20 30 29 3b 0a 20 20 20 20   " ID", 0);.    
13e0: 72 65 74 75 72 6e 20 54 43 4c 5f 45 52 52 4f 52  return TCL_ERROR
13f0: 3b 0a 20 20 7d 0a 20 20 69 20 3d 20 70 61 72 73  ;.  }.  i = pars
1400: 65 5f 74 68 72 65 61 64 5f 69 64 28 69 6e 74 65  e_thread_id(inte
1410: 72 70 2c 20 61 72 67 76 5b 31 5d 29 3b 0a 20 20  rp, argv[1]);.  
1420: 69 66 28 20 69 3c 30 20 29 20 72 65 74 75 72 6e  if( i<0 ) return
1430: 20 54 43 4c 5f 45 52 52 4f 52 3b 0a 20 20 69 66   TCL_ERROR;.  if
1440: 28 20 21 74 68 72 65 61 64 73 65 74 5b 69 5d 2e  ( !threadset[i].
1450: 62 75 73 79 20 29 7b 0a 20 20 20 20 54 63 6c 5f  busy ){.    Tcl_
1460: 41 70 70 65 6e 64 52 65 73 75 6c 74 28 69 6e 74  AppendResult(int
1470: 65 72 70 2c 20 22 6e 6f 20 73 75 63 68 20 74 68  erp, "no such th
1480: 72 65 61 64 22 2c 20 30 29 3b 0a 20 20 20 20 72  read", 0);.    r
1490: 65 74 75 72 6e 20 54 43 4c 5f 45 52 52 4f 52 3b  eturn TCL_ERROR;
14a0: 0a 20 20 7d 0a 20 20 74 68 72 65 61 64 5f 77 61  .  }.  thread_wa
14b0: 69 74 28 26 74 68 72 65 61 64 73 65 74 5b 69 5d  it(&threadset[i]
14c0: 29 3b 0a 20 20 72 65 74 75 72 6e 20 54 43 4c 5f  );.  return TCL_
14d0: 4f 4b 3b 0a 7d 0a 0a 2f 2a 0a 2a 2a 20 53 74 6f  OK;.}../*.** Sto
14e0: 70 20 61 20 74 68 72 65 61 64 2e 0a 2a 2f 0a 73  p a thread..*/.s
14f0: 74 61 74 69 63 20 76 6f 69 64 20 73 74 6f 70 5f  tatic void stop_
1500: 74 68 72 65 61 64 28 54 68 72 65 61 64 20 2a 70  thread(Thread *p
1510: 29 7b 0a 20 20 74 68 72 65 61 64 5f 77 61 69 74  ){.  thread_wait
1520: 28 70 29 3b 0a 20 20 70 2d 3e 78 4f 70 20 3d 20  (p);.  p->xOp = 
1530: 30 3b 0a 20 20 70 2d 3e 6f 70 6e 75 6d 2b 2b 3b  0;.  p->opnum++;
1540: 0a 20 20 74 68 72 65 61 64 5f 77 61 69 74 28 70  .  thread_wait(p
1550: 29 3b 0a 20 20 73 71 6c 69 74 65 33 5f 66 72 65  );.  sqlite3_fre
1560: 65 28 70 2d 3e 7a 41 72 67 29 3b 0a 20 20 70 2d  e(p->zArg);.  p-
1570: 3e 7a 41 72 67 20 3d 20 30 3b 0a 20 20 73 71 6c  >zArg = 0;.  sql
1580: 69 74 65 33 5f 66 72 65 65 28 70 2d 3e 7a 46 69  ite3_free(p->zFi
1590: 6c 65 6e 61 6d 65 29 3b 0a 20 20 70 2d 3e 7a 46  lename);.  p->zF
15a0: 69 6c 65 6e 61 6d 65 20 3d 20 30 3b 0a 20 20 70  ilename = 0;.  p
15b0: 2d 3e 62 75 73 79 20 3d 20 30 3b 0a 7d 0a 0a 2f  ->busy = 0;.}../
15c0: 2a 0a 2a 2a 20 55 73 61 67 65 3a 20 20 74 68 72  *.** Usage:  thr
15d0: 65 61 64 5f 68 61 6c 74 20 49 44 0a 2a 2a 0a 2a  ead_halt ID.**.*
15e0: 2a 20 43 61 75 73 65 20 61 20 74 68 72 65 61 64  * Cause a thread
15f0: 20 74 6f 20 73 68 75 74 20 69 74 73 65 6c 66 20   to shut itself 
1600: 64 6f 77 6e 2e 20 20 57 61 69 74 20 66 6f 72 20  down.  Wait for 
1610: 74 68 65 20 73 68 75 74 64 6f 77 6e 20 74 6f 20  the shutdown to 
1620: 62 65 0a 2a 2a 20 63 6f 6d 70 6c 65 74 65 64 2e  be.** completed.
1630: 20 20 49 66 20 49 44 20 69 73 20 22 2a 22 20 74    If ID is "*" t
1640: 68 65 6e 20 73 74 6f 70 20 61 6c 6c 20 74 68 72  hen stop all thr
1650: 65 61 64 73 2e 0a 2a 2f 0a 73 74 61 74 69 63 20  eads..*/.static 
1660: 69 6e 74 20 74 63 6c 5f 74 68 72 65 61 64 5f 68  int tcl_thread_h
1670: 61 6c 74 28 0a 20 20 76 6f 69 64 20 2a 4e 6f 74  alt(.  void *Not
1680: 55 73 65 64 2c 0a 20 20 54 63 6c 5f 49 6e 74 65  Used,.  Tcl_Inte
1690: 72 70 20 2a 69 6e 74 65 72 70 2c 20 20 20 20 2f  rp *interp,    /
16a0: 2a 20 54 68 65 20 54 43 4c 20 69 6e 74 65 72 70  * The TCL interp
16b0: 72 65 74 65 72 20 74 68 61 74 20 69 6e 76 6f 6b  reter that invok
16c0: 65 64 20 74 68 69 73 20 63 6f 6d 6d 61 6e 64 20  ed this command 
16d0: 2a 2f 0a 20 20 69 6e 74 20 61 72 67 63 2c 20 20  */.  int argc,  
16e0: 20 20 20 20 20 20 20 20 20 20 20 20 2f 2a 20 4e              /* N
16f0: 75 6d 62 65 72 20 6f 66 20 61 72 67 75 6d 65 6e  umber of argumen
1700: 74 73 20 2a 2f 0a 20 20 63 6f 6e 73 74 20 63 68  ts */.  const ch
1710: 61 72 20 2a 2a 61 72 67 76 20 20 20 20 20 20 2f  ar **argv      /
1720: 2a 20 54 65 78 74 20 6f 66 20 65 61 63 68 20 61  * Text of each a
1730: 72 67 75 6d 65 6e 74 20 2a 2f 0a 29 7b 0a 20 20  rgument */.){.  
1740: 69 6e 74 20 69 3b 0a 0a 20 20 69 66 28 20 61 72  int i;..  if( ar
1750: 67 63 21 3d 32 20 29 7b 0a 20 20 20 20 54 63 6c  gc!=2 ){.    Tcl
1760: 5f 41 70 70 65 6e 64 52 65 73 75 6c 74 28 69 6e  _AppendResult(in
1770: 74 65 72 70 2c 20 22 77 72 6f 6e 67 20 23 20 61  terp, "wrong # a
1780: 72 67 73 3a 20 73 68 6f 75 6c 64 20 62 65 20 5c  rgs: should be \
1790: 22 22 2c 20 61 72 67 76 5b 30 5d 2c 0a 20 20 20  "", argv[0],.   
17a0: 20 20 20 20 22 20 49 44 22 2c 20 30 29 3b 0a 20      " ID", 0);. 
17b0: 20 20 20 72 65 74 75 72 6e 20 54 43 4c 5f 45 52     return TCL_ER
17c0: 52 4f 52 3b 0a 20 20 7d 0a 20 20 69 66 28 20 61  ROR;.  }.  if( a
17d0: 72 67 76 5b 31 5d 5b 30 5d 3d 3d 27 2a 27 20 26  rgv[1][0]=='*' &
17e0: 26 20 61 72 67 76 5b 31 5d 5b 31 5d 3d 3d 30 20  & argv[1][1]==0 
17f0: 29 7b 0a 20 20 20 20 66 6f 72 28 69 3d 30 3b 20  ){.    for(i=0; 
1800: 69 3c 4e 5f 54 48 52 45 41 44 3b 20 69 2b 2b 29  i<N_THREAD; i++)
1810: 7b 0a 20 20 20 20 20 20 69 66 28 20 74 68 72 65  {.      if( thre
1820: 61 64 73 65 74 5b 69 5d 2e 62 75 73 79 20 29 20  adset[i].busy ) 
1830: 73 74 6f 70 5f 74 68 72 65 61 64 28 26 74 68 72  stop_thread(&thr
1840: 65 61 64 73 65 74 5b 69 5d 29 3b 0a 20 20 20 20  eadset[i]);.    
1850: 7d 0a 20 20 7d 65 6c 73 65 7b 0a 20 20 20 20 69  }.  }else{.    i
1860: 20 3d 20 70 61 72 73 65 5f 74 68 72 65 61 64 5f   = parse_thread_
1870: 69 64 28 69 6e 74 65 72 70 2c 20 61 72 67 76 5b  id(interp, argv[
1880: 31 5d 29 3b 0a 20 20 20 20 69 66 28 20 69 3c 30  1]);.    if( i<0
1890: 20 29 20 72 65 74 75 72 6e 20 54 43 4c 5f 45 52   ) return TCL_ER
18a0: 52 4f 52 3b 0a 20 20 20 20 69 66 28 20 21 74 68  ROR;.    if( !th
18b0: 72 65 61 64 73 65 74 5b 69 5d 2e 62 75 73 79 20  readset[i].busy 
18c0: 29 7b 0a 20 20 20 20 20 20 54 63 6c 5f 41 70 70  ){.      Tcl_App
18d0: 65 6e 64 52 65 73 75 6c 74 28 69 6e 74 65 72 70  endResult(interp
18e0: 2c 20 22 6e 6f 20 73 75 63 68 20 74 68 72 65 61  , "no such threa
18f0: 64 22 2c 20 30 29 3b 0a 20 20 20 20 20 20 72 65  d", 0);.      re
1900: 74 75 72 6e 20 54 43 4c 5f 45 52 52 4f 52 3b 0a  turn TCL_ERROR;.
1910: 20 20 20 20 7d 0a 20 20 20 20 73 74 6f 70 5f 74      }.    stop_t
1920: 68 72 65 61 64 28 26 74 68 72 65 61 64 73 65 74  hread(&threadset
1930: 5b 69 5d 29 3b 0a 20 20 7d 0a 20 20 72 65 74 75  [i]);.  }.  retu
1940: 72 6e 20 54 43 4c 5f 4f 4b 3b 0a 7d 0a 0a 2f 2a  rn TCL_OK;.}../*
1950: 0a 2a 2a 20 55 73 61 67 65 3a 20 74 68 72 65 61  .** Usage: threa
1960: 64 5f 61 72 67 63 20 20 49 44 0a 2a 2a 0a 2a 2a  d_argc  ID.**.**
1970: 20 57 61 69 74 20 6f 6e 20 74 68 65 20 6d 6f 73   Wait on the mos
1980: 74 20 72 65 63 65 6e 74 20 74 68 72 65 61 64 5f  t recent thread_
1990: 73 74 65 70 20 74 6f 20 63 6f 6d 70 6c 65 74 65  step to complete
19a0: 2c 20 74 68 65 6e 20 72 65 74 75 72 6e 20 74 68  , then return th
19b0: 65 0a 2a 2a 20 6e 75 6d 62 65 72 20 6f 66 20 63  e.** number of c
19c0: 6f 6c 75 6d 6e 73 20 69 6e 20 74 68 65 20 72 65  olumns in the re
19d0: 73 75 6c 74 20 73 65 74 2e 0a 2a 2f 0a 73 74 61  sult set..*/.sta
19e0: 74 69 63 20 69 6e 74 20 74 63 6c 5f 74 68 72 65  tic int tcl_thre
19f0: 61 64 5f 61 72 67 63 28 0a 20 20 76 6f 69 64 20  ad_argc(.  void 
1a00: 2a 4e 6f 74 55 73 65 64 2c 0a 20 20 54 63 6c 5f  *NotUsed,.  Tcl_
1a10: 49 6e 74 65 72 70 20 2a 69 6e 74 65 72 70 2c 20  Interp *interp, 
1a20: 20 20 20 2f 2a 20 54 68 65 20 54 43 4c 20 69 6e     /* The TCL in
1a30: 74 65 72 70 72 65 74 65 72 20 74 68 61 74 20 69  terpreter that i
1a40: 6e 76 6f 6b 65 64 20 74 68 69 73 20 63 6f 6d 6d  nvoked this comm
1a50: 61 6e 64 20 2a 2f 0a 20 20 69 6e 74 20 61 72 67  and */.  int arg
1a60: 63 2c 20 20 20 20 20 20 20 20 20 20 20 20 20 20  c,              
1a70: 2f 2a 20 4e 75 6d 62 65 72 20 6f 66 20 61 72 67  /* Number of arg
1a80: 75 6d 65 6e 74 73 20 2a 2f 0a 20 20 63 6f 6e 73  uments */.  cons
1a90: 74 20 63 68 61 72 20 2a 2a 61 72 67 76 20 20 20  t char **argv   
1aa0: 20 20 20 2f 2a 20 54 65 78 74 20 6f 66 20 65 61     /* Text of ea
1ab0: 63 68 20 61 72 67 75 6d 65 6e 74 20 2a 2f 0a 29  ch argument */.)
1ac0: 7b 0a 20 20 69 6e 74 20 69 3b 0a 20 20 63 68 61  {.  int i;.  cha
1ad0: 72 20 7a 42 75 66 5b 31 30 30 5d 3b 0a 0a 20 20  r zBuf[100];..  
1ae0: 69 66 28 20 61 72 67 63 21 3d 32 20 29 7b 0a 20  if( argc!=2 ){. 
1af0: 20 20 20 54 63 6c 5f 41 70 70 65 6e 64 52 65 73     Tcl_AppendRes
1b00: 75 6c 74 28 69 6e 74 65 72 70 2c 20 22 77 72 6f  ult(interp, "wro
1b10: 6e 67 20 23 20 61 72 67 73 3a 20 73 68 6f 75 6c  ng # args: shoul
1b20: 64 20 62 65 20 5c 22 22 2c 20 61 72 67 76 5b 30  d be \"", argv[0
1b30: 5d 2c 0a 20 20 20 20 20 20 20 22 20 49 44 22 2c  ],.       " ID",
1b40: 20 30 29 3b 0a 20 20 20 20 72 65 74 75 72 6e 20   0);.    return 
1b50: 54 43 4c 5f 45 52 52 4f 52 3b 0a 20 20 7d 0a 20  TCL_ERROR;.  }. 
1b60: 20 69 20 3d 20 70 61 72 73 65 5f 74 68 72 65 61   i = parse_threa
1b70: 64 5f 69 64 28 69 6e 74 65 72 70 2c 20 61 72 67  d_id(interp, arg
1b80: 76 5b 31 5d 29 3b 0a 20 20 69 66 28 20 69 3c 30  v[1]);.  if( i<0
1b90: 20 29 20 72 65 74 75 72 6e 20 54 43 4c 5f 45 52   ) return TCL_ER
1ba0: 52 4f 52 3b 0a 20 20 69 66 28 20 21 74 68 72 65  ROR;.  if( !thre
1bb0: 61 64 73 65 74 5b 69 5d 2e 62 75 73 79 20 29 7b  adset[i].busy ){
1bc0: 0a 20 20 20 20 54 63 6c 5f 41 70 70 65 6e 64 52  .    Tcl_AppendR
1bd0: 65 73 75 6c 74 28 69 6e 74 65 72 70 2c 20 22 6e  esult(interp, "n
1be0: 6f 20 73 75 63 68 20 74 68 72 65 61 64 22 2c 20  o such thread", 
1bf0: 30 29 3b 0a 20 20 20 20 72 65 74 75 72 6e 20 54  0);.    return T
1c00: 43 4c 5f 45 52 52 4f 52 3b 0a 20 20 7d 0a 20 20  CL_ERROR;.  }.  
1c10: 74 68 72 65 61 64 5f 77 61 69 74 28 26 74 68 72  thread_wait(&thr
1c20: 65 61 64 73 65 74 5b 69 5d 29 3b 0a 20 20 73 70  eadset[i]);.  sp
1c30: 72 69 6e 74 66 28 7a 42 75 66 2c 20 22 25 64 22  rintf(zBuf, "%d"
1c40: 2c 20 74 68 72 65 61 64 73 65 74 5b 69 5d 2e 61  , threadset[i].a
1c50: 72 67 63 29 3b 0a 20 20 54 63 6c 5f 41 70 70 65  rgc);.  Tcl_Appe
1c60: 6e 64 52 65 73 75 6c 74 28 69 6e 74 65 72 70 2c  ndResult(interp,
1c70: 20 7a 42 75 66 2c 20 30 29 3b 0a 20 20 72 65 74   zBuf, 0);.  ret
1c80: 75 72 6e 20 54 43 4c 5f 4f 4b 3b 0a 7d 0a 0a 2f  urn TCL_OK;.}../
1c90: 2a 0a 2a 2a 20 55 73 61 67 65 3a 20 74 68 72 65  *.** Usage: thre
1ca0: 61 64 5f 61 72 67 76 20 20 49 44 20 20 20 4e 0a  ad_argv  ID   N.
1cb0: 2a 2a 0a 2a 2a 20 57 61 69 74 20 6f 6e 20 74 68  **.** Wait on th
1cc0: 65 20 6d 6f 73 74 20 72 65 63 65 6e 74 20 74 68  e most recent th
1cd0: 72 65 61 64 5f 73 74 65 70 20 74 6f 20 63 6f 6d  read_step to com
1ce0: 70 6c 65 74 65 2c 20 74 68 65 6e 20 72 65 74 75  plete, then retu
1cf0: 72 6e 20 74 68 65 0a 2a 2a 20 76 61 6c 75 65 20  rn the.** value 
1d00: 6f 66 20 74 68 65 20 4e 2d 74 68 20 63 6f 6c 75  of the N-th colu
1d10: 6d 6e 73 20 69 6e 20 74 68 65 20 72 65 73 75 6c  mns in the resul
1d20: 74 20 73 65 74 2e 0a 2a 2f 0a 73 74 61 74 69 63  t set..*/.static
1d30: 20 69 6e 74 20 74 63 6c 5f 74 68 72 65 61 64 5f   int tcl_thread_
1d40: 61 72 67 76 28 0a 20 20 76 6f 69 64 20 2a 4e 6f  argv(.  void *No
1d50: 74 55 73 65 64 2c 0a 20 20 54 63 6c 5f 49 6e 74  tUsed,.  Tcl_Int
1d60: 65 72 70 20 2a 69 6e 74 65 72 70 2c 20 20 20 20  erp *interp,    
1d70: 2f 2a 20 54 68 65 20 54 43 4c 20 69 6e 74 65 72  /* The TCL inter
1d80: 70 72 65 74 65 72 20 74 68 61 74 20 69 6e 76 6f  preter that invo
1d90: 6b 65 64 20 74 68 69 73 20 63 6f 6d 6d 61 6e 64  ked this command
1da0: 20 2a 2f 0a 20 20 69 6e 74 20 61 72 67 63 2c 20   */.  int argc, 
1db0: 20 20 20 20 20 20 20 20 20 20 20 20 20 2f 2a 20               /* 
1dc0: 4e 75 6d 62 65 72 20 6f 66 20 61 72 67 75 6d 65  Number of argume
1dd0: 6e 74 73 20 2a 2f 0a 20 20 63 6f 6e 73 74 20 63  nts */.  const c
1de0: 68 61 72 20 2a 2a 61 72 67 76 20 20 20 20 20 20  har **argv      
1df0: 2f 2a 20 54 65 78 74 20 6f 66 20 65 61 63 68 20  /* Text of each 
1e00: 61 72 67 75 6d 65 6e 74 20 2a 2f 0a 29 7b 0a 20  argument */.){. 
1e10: 20 69 6e 74 20 69 3b 0a 20 20 69 6e 74 20 6e 3b   int i;.  int n;
1e20: 0a 0a 20 20 69 66 28 20 61 72 67 63 21 3d 33 20  ..  if( argc!=3 
1e30: 29 7b 0a 20 20 20 20 54 63 6c 5f 41 70 70 65 6e  ){.    Tcl_Appen
1e40: 64 52 65 73 75 6c 74 28 69 6e 74 65 72 70 2c 20  dResult(interp, 
1e50: 22 77 72 6f 6e 67 20 23 20 61 72 67 73 3a 20 73  "wrong # args: s
1e60: 68 6f 75 6c 64 20 62 65 20 5c 22 22 2c 20 61 72  hould be \"", ar
1e70: 67 76 5b 30 5d 2c 0a 20 20 20 20 20 20 20 22 20  gv[0],.       " 
1e80: 49 44 20 4e 22 2c 20 30 29 3b 0a 20 20 20 20 72  ID N", 0);.    r
1e90: 65 74 75 72 6e 20 54 43 4c 5f 45 52 52 4f 52 3b  eturn TCL_ERROR;
1ea0: 0a 20 20 7d 0a 20 20 69 20 3d 20 70 61 72 73 65  .  }.  i = parse
1eb0: 5f 74 68 72 65 61 64 5f 69 64 28 69 6e 74 65 72  _thread_id(inter
1ec0: 70 2c 20 61 72 67 76 5b 31 5d 29 3b 0a 20 20 69  p, argv[1]);.  i
1ed0: 66 28 20 69 3c 30 20 29 20 72 65 74 75 72 6e 20  f( i<0 ) return 
1ee0: 54 43 4c 5f 45 52 52 4f 52 3b 0a 20 20 69 66 28  TCL_ERROR;.  if(
1ef0: 20 21 74 68 72 65 61 64 73 65 74 5b 69 5d 2e 62   !threadset[i].b
1f00: 75 73 79 20 29 7b 0a 20 20 20 20 54 63 6c 5f 41  usy ){.    Tcl_A
1f10: 70 70 65 6e 64 52 65 73 75 6c 74 28 69 6e 74 65  ppendResult(inte
1f20: 72 70 2c 20 22 6e 6f 20 73 75 63 68 20 74 68 72  rp, "no such thr
1f30: 65 61 64 22 2c 20 30 29 3b 0a 20 20 20 20 72 65  ead", 0);.    re
1f40: 74 75 72 6e 20 54 43 4c 5f 45 52 52 4f 52 3b 0a  turn TCL_ERROR;.
1f50: 20 20 7d 0a 20 20 69 66 28 20 54 63 6c 5f 47 65    }.  if( Tcl_Ge
1f60: 74 49 6e 74 28 69 6e 74 65 72 70 2c 20 61 72 67  tInt(interp, arg
1f70: 76 5b 32 5d 2c 20 26 6e 29 20 29 20 72 65 74 75  v[2], &n) ) retu
1f80: 72 6e 20 54 43 4c 5f 45 52 52 4f 52 3b 0a 20 20  rn TCL_ERROR;.  
1f90: 74 68 72 65 61 64 5f 77 61 69 74 28 26 74 68 72  thread_wait(&thr
1fa0: 65 61 64 73 65 74 5b 69 5d 29 3b 0a 20 20 69 66  eadset[i]);.  if
1fb0: 28 20 6e 3c 30 20 7c 7c 20 6e 3e 3d 74 68 72 65  ( n<0 || n>=thre
1fc0: 61 64 73 65 74 5b 69 5d 2e 61 72 67 63 20 29 7b  adset[i].argc ){
1fd0: 0a 20 20 20 20 54 63 6c 5f 41 70 70 65 6e 64 52  .    Tcl_AppendR
1fe0: 65 73 75 6c 74 28 69 6e 74 65 72 70 2c 20 22 63  esult(interp, "c
1ff0: 6f 6c 75 6d 6e 20 6e 75 6d 62 65 72 20 6f 75 74  olumn number out
2000: 20 6f 66 20 72 61 6e 67 65 22 2c 20 30 29 3b 0a   of range", 0);.
2010: 20 20 20 20 72 65 74 75 72 6e 20 54 43 4c 5f 45      return TCL_E
2020: 52 52 4f 52 3b 0a 20 20 7d 0a 20 20 54 63 6c 5f  RROR;.  }.  Tcl_
2030: 41 70 70 65 6e 64 52 65 73 75 6c 74 28 69 6e 74  AppendResult(int
2040: 65 72 70 2c 20 74 68 72 65 61 64 73 65 74 5b 69  erp, threadset[i
2050: 5d 2e 61 72 67 76 5b 6e 5d 2c 20 30 29 3b 0a 20  ].argv[n], 0);. 
2060: 20 72 65 74 75 72 6e 20 54 43 4c 5f 4f 4b 3b 0a   return TCL_OK;.
2070: 7d 0a 0a 2f 2a 0a 2a 2a 20 55 73 61 67 65 3a 20  }../*.** Usage: 
2080: 74 68 72 65 61 64 5f 63 6f 6c 6e 61 6d 65 20 20  thread_colname  
2090: 49 44 20 20 20 4e 0a 2a 2a 0a 2a 2a 20 57 61 69  ID   N.**.** Wai
20a0: 74 20 6f 6e 20 74 68 65 20 6d 6f 73 74 20 72 65  t on the most re
20b0: 63 65 6e 74 20 74 68 72 65 61 64 5f 73 74 65 70  cent thread_step
20c0: 20 74 6f 20 63 6f 6d 70 6c 65 74 65 2c 20 74 68   to complete, th
20d0: 65 6e 20 72 65 74 75 72 6e 20 74 68 65 0a 2a 2a  en return the.**
20e0: 20 6e 61 6d 65 20 6f 66 20 74 68 65 20 4e 2d 74   name of the N-t
20f0: 68 20 63 6f 6c 75 6d 6e 73 20 69 6e 20 74 68 65  h columns in the
2100: 20 72 65 73 75 6c 74 20 73 65 74 2e 0a 2a 2f 0a   result set..*/.
2110: 73 74 61 74 69 63 20 69 6e 74 20 74 63 6c 5f 74  static int tcl_t
2120: 68 72 65 61 64 5f 63 6f 6c 6e 61 6d 65 28 0a 20  hread_colname(. 
2130: 20 76 6f 69 64 20 2a 4e 6f 74 55 73 65 64 2c 0a   void *NotUsed,.
2140: 20 20 54 63 6c 5f 49 6e 74 65 72 70 20 2a 69 6e    Tcl_Interp *in
2150: 74 65 72 70 2c 20 20 20 20 2f 2a 20 54 68 65 20  terp,    /* The 
2160: 54 43 4c 20 69 6e 74 65 72 70 72 65 74 65 72 20  TCL interpreter 
2170: 74 68 61 74 20 69 6e 76 6f 6b 65 64 20 74 68 69  that invoked thi
2180: 73 20 63 6f 6d 6d 61 6e 64 20 2a 2f 0a 20 20 69  s command */.  i
2190: 6e 74 20 61 72 67 63 2c 20 20 20 20 20 20 20 20  nt argc,        
21a0: 20 20 20 20 20 20 2f 2a 20 4e 75 6d 62 65 72 20        /* Number 
21b0: 6f 66 20 61 72 67 75 6d 65 6e 74 73 20 2a 2f 0a  of arguments */.
21c0: 20 20 63 6f 6e 73 74 20 63 68 61 72 20 2a 2a 61    const char **a
21d0: 72 67 76 20 20 20 20 20 20 2f 2a 20 54 65 78 74  rgv      /* Text
21e0: 20 6f 66 20 65 61 63 68 20 61 72 67 75 6d 65 6e   of each argumen
21f0: 74 20 2a 2f 0a 29 7b 0a 20 20 69 6e 74 20 69 3b  t */.){.  int i;
2200: 0a 20 20 69 6e 74 20 6e 3b 0a 0a 20 20 69 66 28  .  int n;..  if(
2210: 20 61 72 67 63 21 3d 33 20 29 7b 0a 20 20 20 20   argc!=3 ){.    
2220: 54 63 6c 5f 41 70 70 65 6e 64 52 65 73 75 6c 74  Tcl_AppendResult
2230: 28 69 6e 74 65 72 70 2c 20 22 77 72 6f 6e 67 20  (interp, "wrong 
2240: 23 20 61 72 67 73 3a 20 73 68 6f 75 6c 64 20 62  # args: should b
2250: 65 20 5c 22 22 2c 20 61 72 67 76 5b 30 5d 2c 0a  e \"", argv[0],.
2260: 20 20 20 20 20 20 20 22 20 49 44 20 4e 22 2c 20         " ID N", 
2270: 30 29 3b 0a 20 20 20 20 72 65 74 75 72 6e 20 54  0);.    return T
2280: 43 4c 5f 45 52 52 4f 52 3b 0a 20 20 7d 0a 20 20  CL_ERROR;.  }.  
2290: 69 20 3d 20 70 61 72 73 65 5f 74 68 72 65 61 64  i = parse_thread
22a0: 5f 69 64 28 69 6e 74 65 72 70 2c 20 61 72 67 76  _id(interp, argv
22b0: 5b 31 5d 29 3b 0a 20 20 69 66 28 20 69 3c 30 20  [1]);.  if( i<0 
22c0: 29 20 72 65 74 75 72 6e 20 54 43 4c 5f 45 52 52  ) return TCL_ERR
22d0: 4f 52 3b 0a 20 20 69 66 28 20 21 74 68 72 65 61  OR;.  if( !threa
22e0: 64 73 65 74 5b 69 5d 2e 62 75 73 79 20 29 7b 0a  dset[i].busy ){.
22f0: 20 20 20 20 54 63 6c 5f 41 70 70 65 6e 64 52 65      Tcl_AppendRe
2300: 73 75 6c 74 28 69 6e 74 65 72 70 2c 20 22 6e 6f  sult(interp, "no
2310: 20 73 75 63 68 20 74 68 72 65 61 64 22 2c 20 30   such thread", 0
2320: 29 3b 0a 20 20 20 20 72 65 74 75 72 6e 20 54 43  );.    return TC
2330: 4c 5f 45 52 52 4f 52 3b 0a 20 20 7d 0a 20 20 69  L_ERROR;.  }.  i
2340: 66 28 20 54 63 6c 5f 47 65 74 49 6e 74 28 69 6e  f( Tcl_GetInt(in
2350: 74 65 72 70 2c 20 61 72 67 76 5b 32 5d 2c 20 26  terp, argv[2], &
2360: 6e 29 20 29 20 72 65 74 75 72 6e 20 54 43 4c 5f  n) ) return TCL_
2370: 45 52 52 4f 52 3b 0a 20 20 74 68 72 65 61 64 5f  ERROR;.  thread_
2380: 77 61 69 74 28 26 74 68 72 65 61 64 73 65 74 5b  wait(&threadset[
2390: 69 5d 29 3b 0a 20 20 69 66 28 20 6e 3c 30 20 7c  i]);.  if( n<0 |
23a0: 7c 20 6e 3e 3d 74 68 72 65 61 64 73 65 74 5b 69  | n>=threadset[i
23b0: 5d 2e 61 72 67 63 20 29 7b 0a 20 20 20 20 54 63  ].argc ){.    Tc
23c0: 6c 5f 41 70 70 65 6e 64 52 65 73 75 6c 74 28 69  l_AppendResult(i
23d0: 6e 74 65 72 70 2c 20 22 63 6f 6c 75 6d 6e 20 6e  nterp, "column n
23e0: 75 6d 62 65 72 20 6f 75 74 20 6f 66 20 72 61 6e  umber out of ran
23f0: 67 65 22 2c 20 30 29 3b 0a 20 20 20 20 72 65 74  ge", 0);.    ret
2400: 75 72 6e 20 54 43 4c 5f 45 52 52 4f 52 3b 0a 20  urn TCL_ERROR;. 
2410: 20 7d 0a 20 20 54 63 6c 5f 41 70 70 65 6e 64 52   }.  Tcl_AppendR
2420: 65 73 75 6c 74 28 69 6e 74 65 72 70 2c 20 74 68  esult(interp, th
2430: 72 65 61 64 73 65 74 5b 69 5d 2e 63 6f 6c 76 5b  readset[i].colv[
2440: 6e 5d 2c 20 30 29 3b 0a 20 20 72 65 74 75 72 6e  n], 0);.  return
2450: 20 54 43 4c 5f 4f 4b 3b 0a 7d 0a 0a 2f 2a 0a 2a   TCL_OK;.}../*.*
2460: 2a 20 55 73 61 67 65 3a 20 74 68 72 65 61 64 5f  * Usage: thread_
2470: 72 65 73 75 6c 74 20 20 49 44 0a 2a 2a 0a 2a 2a  result  ID.**.**
2480: 20 57 61 69 74 20 6f 6e 20 74 68 65 20 6d 6f 73   Wait on the mos
2490: 74 20 72 65 63 65 6e 74 20 6f 70 65 72 61 74 69  t recent operati
24a0: 6f 6e 20 74 6f 20 63 6f 6d 70 6c 65 74 65 2c 20  on to complete, 
24b0: 74 68 65 6e 20 72 65 74 75 72 6e 20 74 68 65 0a  then return the.
24c0: 2a 2a 20 72 65 73 75 6c 74 20 63 6f 64 65 20 66  ** result code f
24d0: 72 6f 6d 20 74 68 61 74 20 6f 70 65 72 61 74 69  rom that operati
24e0: 6f 6e 2e 0a 2a 2f 0a 73 74 61 74 69 63 20 69 6e  on..*/.static in
24f0: 74 20 74 63 6c 5f 74 68 72 65 61 64 5f 72 65 73  t tcl_thread_res
2500: 75 6c 74 28 0a 20 20 76 6f 69 64 20 2a 4e 6f 74  ult(.  void *Not
2510: 55 73 65 64 2c 0a 20 20 54 63 6c 5f 49 6e 74 65  Used,.  Tcl_Inte
2520: 72 70 20 2a 69 6e 74 65 72 70 2c 20 20 20 20 2f  rp *interp,    /
2530: 2a 20 54 68 65 20 54 43 4c 20 69 6e 74 65 72 70  * The TCL interp
2540: 72 65 74 65 72 20 74 68 61 74 20 69 6e 76 6f 6b  reter that invok
2550: 65 64 20 74 68 69 73 20 63 6f 6d 6d 61 6e 64 20  ed this command 
2560: 2a 2f 0a 20 20 69 6e 74 20 61 72 67 63 2c 20 20  */.  int argc,  
2570: 20 20 20 20 20 20 20 20 20 20 20 20 2f 2a 20 4e              /* N
2580: 75 6d 62 65 72 20 6f 66 20 61 72 67 75 6d 65 6e  umber of argumen
2590: 74 73 20 2a 2f 0a 20 20 63 6f 6e 73 74 20 63 68  ts */.  const ch
25a0: 61 72 20 2a 2a 61 72 67 76 20 20 20 20 20 20 2f  ar **argv      /
25b0: 2a 20 54 65 78 74 20 6f 66 20 65 61 63 68 20 61  * Text of each a
25c0: 72 67 75 6d 65 6e 74 20 2a 2f 0a 29 7b 0a 20 20  rgument */.){.  
25d0: 69 6e 74 20 69 3b 0a 20 20 63 6f 6e 73 74 20 63  int i;.  const c
25e0: 68 61 72 20 2a 7a 4e 61 6d 65 3b 0a 0a 20 20 69  har *zName;..  i
25f0: 66 28 20 61 72 67 63 21 3d 32 20 29 7b 0a 20 20  f( argc!=2 ){.  
2600: 20 20 54 63 6c 5f 41 70 70 65 6e 64 52 65 73 75    Tcl_AppendResu
2610: 6c 74 28 69 6e 74 65 72 70 2c 20 22 77 72 6f 6e  lt(interp, "wron
2620: 67 20 23 20 61 72 67 73 3a 20 73 68 6f 75 6c 64  g # args: should
2630: 20 62 65 20 5c 22 22 2c 20 61 72 67 76 5b 30 5d   be \"", argv[0]
2640: 2c 0a 20 20 20 20 20 20 20 22 20 49 44 22 2c 20  ,.       " ID", 
2650: 30 29 3b 0a 20 20 20 20 72 65 74 75 72 6e 20 54  0);.    return T
2660: 43 4c 5f 45 52 52 4f 52 3b 0a 20 20 7d 0a 20 20  CL_ERROR;.  }.  
2670: 69 20 3d 20 70 61 72 73 65 5f 74 68 72 65 61 64  i = parse_thread
2680: 5f 69 64 28 69 6e 74 65 72 70 2c 20 61 72 67 76  _id(interp, argv
2690: 5b 31 5d 29 3b 0a 20 20 69 66 28 20 69 3c 30 20  [1]);.  if( i<0 
26a0: 29 20 72 65 74 75 72 6e 20 54 43 4c 5f 45 52 52  ) return TCL_ERR
26b0: 4f 52 3b 0a 20 20 69 66 28 20 21 74 68 72 65 61  OR;.  if( !threa
26c0: 64 73 65 74 5b 69 5d 2e 62 75 73 79 20 29 7b 0a  dset[i].busy ){.
26d0: 20 20 20 20 54 63 6c 5f 41 70 70 65 6e 64 52 65      Tcl_AppendRe
26e0: 73 75 6c 74 28 69 6e 74 65 72 70 2c 20 22 6e 6f  sult(interp, "no
26f0: 20 73 75 63 68 20 74 68 72 65 61 64 22 2c 20 30   such thread", 0
2700: 29 3b 0a 20 20 20 20 72 65 74 75 72 6e 20 54 43  );.    return TC
2710: 4c 5f 45 52 52 4f 52 3b 0a 20 20 7d 0a 20 20 74  L_ERROR;.  }.  t
2720: 68 72 65 61 64 5f 77 61 69 74 28 26 74 68 72 65  hread_wait(&thre
2730: 61 64 73 65 74 5b 69 5d 29 3b 0a 20 20 73 77 69  adset[i]);.  swi
2740: 74 63 68 28 20 74 68 72 65 61 64 73 65 74 5b 69  tch( threadset[i
2750: 5d 2e 72 63 20 29 7b 0a 20 20 20 20 63 61 73 65  ].rc ){.    case
2760: 20 53 51 4c 49 54 45 5f 4f 4b 3a 20 20 20 20 20   SQLITE_OK:     
2770: 20 20 20 20 7a 4e 61 6d 65 20 3d 20 22 53 51 4c      zName = "SQL
2780: 49 54 45 5f 4f 4b 22 3b 20 20 20 20 20 20 20 20  ITE_OK";        
2790: 20 20 62 72 65 61 6b 3b 0a 20 20 20 20 63 61 73    break;.    cas
27a0: 65 20 53 51 4c 49 54 45 5f 45 52 52 4f 52 3a 20  e SQLITE_ERROR: 
27b0: 20 20 20 20 20 7a 4e 61 6d 65 20 3d 20 22 53 51       zName = "SQ
27c0: 4c 49 54 45 5f 45 52 52 4f 52 22 3b 20 20 20 20  LITE_ERROR";    
27d0: 20 20 20 62 72 65 61 6b 3b 0a 20 20 20 20 63 61     break;.    ca
27e0: 73 65 20 53 51 4c 49 54 45 5f 50 45 52 4d 3a 20  se SQLITE_PERM: 
27f0: 20 20 20 20 20 20 7a 4e 61 6d 65 20 3d 20 22 53        zName = "S
2800: 51 4c 49 54 45 5f 50 45 52 4d 22 3b 20 20 20 20  QLITE_PERM";    
2810: 20 20 20 20 62 72 65 61 6b 3b 0a 20 20 20 20 63      break;.    c
2820: 61 73 65 20 53 51 4c 49 54 45 5f 41 42 4f 52 54  ase SQLITE_ABORT
2830: 3a 20 20 20 20 20 20 7a 4e 61 6d 65 20 3d 20 22  :      zName = "
2840: 53 51 4c 49 54 45 5f 41 42 4f 52 54 22 3b 20 20  SQLITE_ABORT";  
2850: 20 20 20 20 20 62 72 65 61 6b 3b 0a 20 20 20 20       break;.    
2860: 63 61 73 65 20 53 51 4c 49 54 45 5f 42 55 53 59  case SQLITE_BUSY
2870: 3a 20 20 20 20 20 20 20 7a 4e 61 6d 65 20 3d 20  :       zName = 
2880: 22 53 51 4c 49 54 45 5f 42 55 53 59 22 3b 20 20  "SQLITE_BUSY";  
2890: 20 20 20 20 20 20 62 72 65 61 6b 3b 0a 20 20 20        break;.   
28a0: 20 63 61 73 65 20 53 51 4c 49 54 45 5f 4c 4f 43   case SQLITE_LOC
28b0: 4b 45 44 3a 20 20 20 20 20 7a 4e 61 6d 65 20 3d  KED:     zName =
28c0: 20 22 53 51 4c 49 54 45 5f 4c 4f 43 4b 45 44 22   "SQLITE_LOCKED"
28d0: 3b 20 20 20 20 20 20 62 72 65 61 6b 3b 0a 20 20  ;      break;.  
28e0: 20 20 63 61 73 65 20 53 51 4c 49 54 45 5f 4e 4f    case SQLITE_NO
28f0: 4d 45 4d 3a 20 20 20 20 20 20 7a 4e 61 6d 65 20  MEM:      zName 
2900: 3d 20 22 53 51 4c 49 54 45 5f 4e 4f 4d 45 4d 22  = "SQLITE_NOMEM"
2910: 3b 20 20 20 20 20 20 20 62 72 65 61 6b 3b 0a 20  ;       break;. 
2920: 20 20 20 63 61 73 65 20 53 51 4c 49 54 45 5f 52     case SQLITE_R
2930: 45 41 44 4f 4e 4c 59 3a 20 20 20 7a 4e 61 6d 65  EADONLY:   zName
2940: 20 3d 20 22 53 51 4c 49 54 45 5f 52 45 41 44 4f   = "SQLITE_READO
2950: 4e 4c 59 22 3b 20 20 20 20 62 72 65 61 6b 3b 0a  NLY";    break;.
2960: 20 20 20 20 63 61 73 65 20 53 51 4c 49 54 45 5f      case SQLITE_
2970: 49 4e 54 45 52 52 55 50 54 3a 20 20 7a 4e 61 6d  INTERRUPT:  zNam
2980: 65 20 3d 20 22 53 51 4c 49 54 45 5f 49 4e 54 45  e = "SQLITE_INTE
2990: 52 52 55 50 54 22 3b 20 20 20 62 72 65 61 6b 3b  RRUPT";   break;
29a0: 0a 20 20 20 20 63 61 73 65 20 53 51 4c 49 54 45  .    case SQLITE
29b0: 5f 49 4f 45 52 52 3a 20 20 20 20 20 20 7a 4e 61  _IOERR:      zNa
29c0: 6d 65 20 3d 20 22 53 51 4c 49 54 45 5f 49 4f 45  me = "SQLITE_IOE
29d0: 52 52 22 3b 20 20 20 20 20 20 20 62 72 65 61 6b  RR";       break
29e0: 3b 0a 20 20 20 20 63 61 73 65 20 53 51 4c 49 54  ;.    case SQLIT
29f0: 45 5f 43 4f 52 52 55 50 54 3a 20 20 20 20 7a 4e  E_CORRUPT:    zN
2a00: 61 6d 65 20 3d 20 22 53 51 4c 49 54 45 5f 43 4f  ame = "SQLITE_CO
2a10: 52 52 55 50 54 22 3b 20 20 20 20 20 62 72 65 61  RRUPT";     brea
2a20: 6b 3b 0a 20 20 20 20 63 61 73 65 20 53 51 4c 49  k;.    case SQLI
2a30: 54 45 5f 46 55 4c 4c 3a 20 20 20 20 20 20 20 7a  TE_FULL:       z
2a40: 4e 61 6d 65 20 3d 20 22 53 51 4c 49 54 45 5f 46  Name = "SQLITE_F
2a50: 55 4c 4c 22 3b 20 20 20 20 20 20 20 20 62 72 65  ULL";        bre
2a60: 61 6b 3b 0a 20 20 20 20 63 61 73 65 20 53 51 4c  ak;.    case SQL
2a70: 49 54 45 5f 43 41 4e 54 4f 50 45 4e 3a 20 20 20  ITE_CANTOPEN:   
2a80: 7a 4e 61 6d 65 20 3d 20 22 53 51 4c 49 54 45 5f  zName = "SQLITE_
2a90: 43 41 4e 54 4f 50 45 4e 22 3b 20 20 20 20 62 72  CANTOPEN";    br
2aa0: 65 61 6b 3b 0a 20 20 20 20 63 61 73 65 20 53 51  eak;.    case SQ
2ab0: 4c 49 54 45 5f 50 52 4f 54 4f 43 4f 4c 3a 20 20  LITE_PROTOCOL:  
2ac0: 20 7a 4e 61 6d 65 20 3d 20 22 53 51 4c 49 54 45   zName = "SQLITE
2ad0: 5f 50 52 4f 54 4f 43 4f 4c 22 3b 20 20 20 20 62  _PROTOCOL";    b
2ae0: 72 65 61 6b 3b 0a 20 20 20 20 63 61 73 65 20 53  reak;.    case S
2af0: 51 4c 49 54 45 5f 45 4d 50 54 59 3a 20 20 20 20  QLITE_EMPTY:    
2b00: 20 20 7a 4e 61 6d 65 20 3d 20 22 53 51 4c 49 54    zName = "SQLIT
2b10: 45 5f 45 4d 50 54 59 22 3b 20 20 20 20 20 20 20  E_EMPTY";       
2b20: 62 72 65 61 6b 3b 0a 20 20 20 20 63 61 73 65 20  break;.    case 
2b30: 53 51 4c 49 54 45 5f 53 43 48 45 4d 41 3a 20 20  SQLITE_SCHEMA:  
2b40: 20 20 20 7a 4e 61 6d 65 20 3d 20 22 53 51 4c 49     zName = "SQLI
2b50: 54 45 5f 53 43 48 45 4d 41 22 3b 20 20 20 20 20  TE_SCHEMA";     
2b60: 20 62 72 65 61 6b 3b 0a 20 20 20 20 63 61 73 65   break;.    case
2b70: 20 53 51 4c 49 54 45 5f 43 4f 4e 53 54 52 41 49   SQLITE_CONSTRAI
2b80: 4e 54 3a 20 7a 4e 61 6d 65 20 3d 20 22 53 51 4c  NT: zName = "SQL
2b90: 49 54 45 5f 43 4f 4e 53 54 52 41 49 4e 54 22 3b  ITE_CONSTRAINT";
2ba0: 20 20 62 72 65 61 6b 3b 0a 20 20 20 20 63 61 73    break;.    cas
2bb0: 65 20 53 51 4c 49 54 45 5f 4d 49 53 4d 41 54 43  e SQLITE_MISMATC
2bc0: 48 3a 20 20 20 7a 4e 61 6d 65 20 3d 20 22 53 51  H:   zName = "SQ
2bd0: 4c 49 54 45 5f 4d 49 53 4d 41 54 43 48 22 3b 20  LITE_MISMATCH"; 
2be0: 20 20 20 62 72 65 61 6b 3b 0a 20 20 20 20 63 61     break;.    ca
2bf0: 73 65 20 53 51 4c 49 54 45 5f 4d 49 53 55 53 45  se SQLITE_MISUSE
2c00: 3a 20 20 20 20 20 7a 4e 61 6d 65 20 3d 20 22 53  :     zName = "S
2c10: 51 4c 49 54 45 5f 4d 49 53 55 53 45 22 3b 20 20  QLITE_MISUSE";  
2c20: 20 20 20 20 62 72 65 61 6b 3b 0a 20 20 20 20 63      break;.    c
2c30: 61 73 65 20 53 51 4c 49 54 45 5f 4e 4f 4c 46 53  ase SQLITE_NOLFS
2c40: 3a 20 20 20 20 20 20 7a 4e 61 6d 65 20 3d 20 22  :      zName = "
2c50: 53 51 4c 49 54 45 5f 4e 4f 4c 46 53 22 3b 20 20  SQLITE_NOLFS";  
2c60: 20 20 20 20 20 62 72 65 61 6b 3b 0a 20 20 20 20       break;.    
2c70: 63 61 73 65 20 53 51 4c 49 54 45 5f 41 55 54 48  case SQLITE_AUTH
2c80: 3a 20 20 20 20 20 20 20 7a 4e 61 6d 65 20 3d 20  :       zName = 
2c90: 22 53 51 4c 49 54 45 5f 41 55 54 48 22 3b 20 20  "SQLITE_AUTH";  
2ca0: 20 20 20 20 20 20 62 72 65 61 6b 3b 0a 20 20 20        break;.   
2cb0: 20 63 61 73 65 20 53 51 4c 49 54 45 5f 46 4f 52   case SQLITE_FOR
2cc0: 4d 41 54 3a 20 20 20 20 20 7a 4e 61 6d 65 20 3d  MAT:     zName =
2cd0: 20 22 53 51 4c 49 54 45 5f 46 4f 52 4d 41 54 22   "SQLITE_FORMAT"
2ce0: 3b 20 20 20 20 20 20 62 72 65 61 6b 3b 0a 20 20  ;      break;.  
2cf0: 20 20 63 61 73 65 20 53 51 4c 49 54 45 5f 52 41    case SQLITE_RA
2d00: 4e 47 45 3a 20 20 20 20 20 20 7a 4e 61 6d 65 20  NGE:      zName 
2d10: 3d 20 22 53 51 4c 49 54 45 5f 52 41 4e 47 45 22  = "SQLITE_RANGE"
2d20: 3b 20 20 20 20 20 20 20 62 72 65 61 6b 3b 0a 20  ;       break;. 
2d30: 20 20 20 63 61 73 65 20 53 51 4c 49 54 45 5f 52     case SQLITE_R
2d40: 4f 57 3a 20 20 20 20 20 20 20 20 7a 4e 61 6d 65  OW:        zName
2d50: 20 3d 20 22 53 51 4c 49 54 45 5f 52 4f 57 22 3b   = "SQLITE_ROW";
2d60: 20 20 20 20 20 20 20 20 20 62 72 65 61 6b 3b 0a           break;.
2d70: 20 20 20 20 63 61 73 65 20 53 51 4c 49 54 45 5f      case SQLITE_
2d80: 44 4f 4e 45 3a 20 20 20 20 20 20 20 7a 4e 61 6d  DONE:       zNam
2d90: 65 20 3d 20 22 53 51 4c 49 54 45 5f 44 4f 4e 45  e = "SQLITE_DONE
2da0: 22 3b 20 20 20 20 20 20 20 20 62 72 65 61 6b 3b  ";        break;
2db0: 0a 20 20 20 20 64 65 66 61 75 6c 74 3a 20 20 20  .    default:   
2dc0: 20 20 20 20 20 20 20 20 20 20 20 20 20 7a 4e 61               zNa
2dd0: 6d 65 20 3d 20 22 53 51 4c 49 54 45 5f 55 6e 6b  me = "SQLITE_Unk
2de0: 6e 6f 77 6e 22 3b 20 20 20 20 20 62 72 65 61 6b  nown";     break
2df0: 3b 0a 20 20 7d 0a 20 20 54 63 6c 5f 41 70 70 65  ;.  }.  Tcl_Appe
2e00: 6e 64 52 65 73 75 6c 74 28 69 6e 74 65 72 70 2c  ndResult(interp,
2e10: 20 7a 4e 61 6d 65 2c 20 30 29 3b 0a 20 20 72 65   zName, 0);.  re
2e20: 74 75 72 6e 20 54 43 4c 5f 4f 4b 3b 0a 7d 0a 0a  turn TCL_OK;.}..
2e30: 2f 2a 0a 2a 2a 20 55 73 61 67 65 3a 20 74 68 72  /*.** Usage: thr
2e40: 65 61 64 5f 65 72 72 6f 72 20 20 49 44 0a 2a 2a  ead_error  ID.**
2e50: 0a 2a 2a 20 57 61 69 74 20 6f 6e 20 74 68 65 20  .** Wait on the 
2e60: 6d 6f 73 74 20 72 65 63 65 6e 74 20 6f 70 65 72  most recent oper
2e70: 61 74 69 6f 6e 20 74 6f 20 63 6f 6d 70 6c 65 74  ation to complet
2e80: 65 2c 20 74 68 65 6e 20 72 65 74 75 72 6e 20 74  e, then return t
2e90: 68 65 0a 2a 2a 20 65 72 72 6f 72 20 73 74 72 69  he.** error stri
2ea0: 6e 67 2e 0a 2a 2f 0a 73 74 61 74 69 63 20 69 6e  ng..*/.static in
2eb0: 74 20 74 63 6c 5f 74 68 72 65 61 64 5f 65 72 72  t tcl_thread_err
2ec0: 6f 72 28 0a 20 20 76 6f 69 64 20 2a 4e 6f 74 55  or(.  void *NotU
2ed0: 73 65 64 2c 0a 20 20 54 63 6c 5f 49 6e 74 65 72  sed,.  Tcl_Inter
2ee0: 70 20 2a 69 6e 74 65 72 70 2c 20 20 20 20 2f 2a  p *interp,    /*
2ef0: 20 54 68 65 20 54 43 4c 20 69 6e 74 65 72 70 72   The TCL interpr
2f00: 65 74 65 72 20 74 68 61 74 20 69 6e 76 6f 6b 65  eter that invoke
2f10: 64 20 74 68 69 73 20 63 6f 6d 6d 61 6e 64 20 2a  d this command *
2f20: 2f 0a 20 20 69 6e 74 20 61 72 67 63 2c 20 20 20  /.  int argc,   
2f30: 20 20 20 20 20 20 20 20 20 20 20 2f 2a 20 4e 75             /* Nu
2f40: 6d 62 65 72 20 6f 66 20 61 72 67 75 6d 65 6e 74  mber of argument
2f50: 73 20 2a 2f 0a 20 20 63 6f 6e 73 74 20 63 68 61  s */.  const cha
2f60: 72 20 2a 2a 61 72 67 76 20 20 20 20 20 20 2f 2a  r **argv      /*
2f70: 20 54 65 78 74 20 6f 66 20 65 61 63 68 20 61 72   Text of each ar
2f80: 67 75 6d 65 6e 74 20 2a 2f 0a 29 7b 0a 20 20 69  gument */.){.  i
2f90: 6e 74 20 69 3b 0a 0a 20 20 69 66 28 20 61 72 67  nt i;..  if( arg
2fa0: 63 21 3d 32 20 29 7b 0a 20 20 20 20 54 63 6c 5f  c!=2 ){.    Tcl_
2fb0: 41 70 70 65 6e 64 52 65 73 75 6c 74 28 69 6e 74  AppendResult(int
2fc0: 65 72 70 2c 20 22 77 72 6f 6e 67 20 23 20 61 72  erp, "wrong # ar
2fd0: 67 73 3a 20 73 68 6f 75 6c 64 20 62 65 20 5c 22  gs: should be \"
2fe0: 22 2c 20 61 72 67 76 5b 30 5d 2c 0a 20 20 20 20  ", argv[0],.    
2ff0: 20 20 20 22 20 49 44 22 2c 20 30 29 3b 0a 20 20     " ID", 0);.  
3000: 20 20 72 65 74 75 72 6e 20 54 43 4c 5f 45 52 52    return TCL_ERR
3010: 4f 52 3b 0a 20 20 7d 0a 20 20 69 20 3d 20 70 61  OR;.  }.  i = pa
3020: 72 73 65 5f 74 68 72 65 61 64 5f 69 64 28 69 6e  rse_thread_id(in
3030: 74 65 72 70 2c 20 61 72 67 76 5b 31 5d 29 3b 0a  terp, argv[1]);.
3040: 20 20 69 66 28 20 69 3c 30 20 29 20 72 65 74 75    if( i<0 ) retu
3050: 72 6e 20 54 43 4c 5f 45 52 52 4f 52 3b 0a 20 20  rn TCL_ERROR;.  
3060: 69 66 28 20 21 74 68 72 65 61 64 73 65 74 5b 69  if( !threadset[i
3070: 5d 2e 62 75 73 79 20 29 7b 0a 20 20 20 20 54 63  ].busy ){.    Tc
3080: 6c 5f 41 70 70 65 6e 64 52 65 73 75 6c 74 28 69  l_AppendResult(i
3090: 6e 74 65 72 70 2c 20 22 6e 6f 20 73 75 63 68 20  nterp, "no such 
30a0: 74 68 72 65 61 64 22 2c 20 30 29 3b 0a 20 20 20  thread", 0);.   
30b0: 20 72 65 74 75 72 6e 20 54 43 4c 5f 45 52 52 4f   return TCL_ERRO
30c0: 52 3b 0a 20 20 7d 0a 20 20 74 68 72 65 61 64 5f  R;.  }.  thread_
30d0: 77 61 69 74 28 26 74 68 72 65 61 64 73 65 74 5b  wait(&threadset[
30e0: 69 5d 29 3b 0a 20 20 54 63 6c 5f 41 70 70 65 6e  i]);.  Tcl_Appen
30f0: 64 52 65 73 75 6c 74 28 69 6e 74 65 72 70 2c 20  dResult(interp, 
3100: 74 68 72 65 61 64 73 65 74 5b 69 5d 2e 7a 45 72  threadset[i].zEr
3110: 72 2c 20 30 29 3b 0a 20 20 72 65 74 75 72 6e 20  r, 0);.  return 
3120: 54 43 4c 5f 4f 4b 3b 0a 7d 0a 0a 2f 2a 0a 2a 2a  TCL_OK;.}../*.**
3130: 20 54 68 69 73 20 70 72 6f 63 65 64 75 72 65 20   This procedure 
3140: 72 75 6e 73 20 69 6e 20 74 68 65 20 74 68 72 65  runs in the thre
3150: 61 64 20 74 6f 20 63 6f 6d 70 69 6c 65 20 61 6e  ad to compile an
3160: 20 53 51 4c 20 73 74 61 74 65 6d 65 6e 74 2e 0a   SQL statement..
3170: 2a 2f 0a 73 74 61 74 69 63 20 76 6f 69 64 20 64  */.static void d
3180: 6f 5f 63 6f 6d 70 69 6c 65 28 54 68 72 65 61 64  o_compile(Thread
3190: 20 2a 70 29 7b 0a 20 20 69 66 28 20 70 2d 3e 64   *p){.  if( p->d
31a0: 62 3d 3d 30 20 29 7b 0a 20 20 20 20 70 2d 3e 7a  b==0 ){.    p->z
31b0: 45 72 72 20 3d 20 70 2d 3e 7a 53 74 61 74 69 63  Err = p->zStatic
31c0: 45 72 72 20 3d 20 22 6e 6f 20 64 61 74 61 62 61  Err = "no databa
31d0: 73 65 20 69 73 20 6f 70 65 6e 22 3b 0a 20 20 20  se is open";.   
31e0: 20 70 2d 3e 72 63 20 3d 20 53 51 4c 49 54 45 5f   p->rc = SQLITE_
31f0: 45 52 52 4f 52 3b 0a 20 20 20 20 72 65 74 75 72  ERROR;.    retur
3200: 6e 3b 0a 20 20 7d 0a 20 20 69 66 28 20 70 2d 3e  n;.  }.  if( p->
3210: 70 53 74 6d 74 20 29 7b 0a 20 20 20 20 73 71 6c  pStmt ){.    sql
3220: 69 74 65 33 5f 66 69 6e 61 6c 69 7a 65 28 70 2d  ite3_finalize(p-
3230: 3e 70 53 74 6d 74 29 3b 0a 20 20 20 20 70 2d 3e  >pStmt);.    p->
3240: 70 53 74 6d 74 20 3d 20 30 3b 0a 20 20 7d 0a 20  pStmt = 0;.  }. 
3250: 20 70 2d 3e 72 63 20 3d 20 73 71 6c 69 74 65 33   p->rc = sqlite3
3260: 5f 70 72 65 70 61 72 65 28 70 2d 3e 64 62 2c 20  _prepare(p->db, 
3270: 70 2d 3e 7a 41 72 67 2c 20 2d 31 2c 20 26 70 2d  p->zArg, -1, &p-
3280: 3e 70 53 74 6d 74 2c 20 30 29 3b 0a 7d 0a 0a 2f  >pStmt, 0);.}../
3290: 2a 0a 2a 2a 20 55 73 61 67 65 3a 20 74 68 72 65  *.** Usage: thre
32a0: 61 64 5f 63 6f 6d 70 69 6c 65 20 49 44 20 53 51  ad_compile ID SQ
32b0: 4c 0a 2a 2a 0a 2a 2a 20 43 6f 6d 70 69 6c 65 20  L.**.** Compile 
32c0: 61 20 6e 65 77 20 76 69 72 74 75 61 6c 20 6d 61  a new virtual ma
32d0: 63 68 69 6e 65 2e 0a 2a 2f 0a 73 74 61 74 69 63  chine..*/.static
32e0: 20 69 6e 74 20 74 63 6c 5f 74 68 72 65 61 64 5f   int tcl_thread_
32f0: 63 6f 6d 70 69 6c 65 28 0a 20 20 76 6f 69 64 20  compile(.  void 
3300: 2a 4e 6f 74 55 73 65 64 2c 0a 20 20 54 63 6c 5f  *NotUsed,.  Tcl_
3310: 49 6e 74 65 72 70 20 2a 69 6e 74 65 72 70 2c 20  Interp *interp, 
3320: 20 20 20 2f 2a 20 54 68 65 20 54 43 4c 20 69 6e     /* The TCL in
3330: 74 65 72 70 72 65 74 65 72 20 74 68 61 74 20 69  terpreter that i
3340: 6e 76 6f 6b 65 64 20 74 68 69 73 20 63 6f 6d 6d  nvoked this comm
3350: 61 6e 64 20 2a 2f 0a 20 20 69 6e 74 20 61 72 67  and */.  int arg
3360: 63 2c 20 20 20 20 20 20 20 20 20 20 20 20 20 20  c,              
3370: 2f 2a 20 4e 75 6d 62 65 72 20 6f 66 20 61 72 67  /* Number of arg
3380: 75 6d 65 6e 74 73 20 2a 2f 0a 20 20 63 6f 6e 73  uments */.  cons
3390: 74 20 63 68 61 72 20 2a 2a 61 72 67 76 20 20 20  t char **argv   
33a0: 20 20 20 2f 2a 20 54 65 78 74 20 6f 66 20 65 61     /* Text of ea
33b0: 63 68 20 61 72 67 75 6d 65 6e 74 20 2a 2f 0a 29  ch argument */.)
33c0: 7b 0a 20 20 69 6e 74 20 69 3b 0a 20 20 69 66 28  {.  int i;.  if(
33d0: 20 61 72 67 63 21 3d 33 20 29 7b 0a 20 20 20 20   argc!=3 ){.    
33e0: 54 63 6c 5f 41 70 70 65 6e 64 52 65 73 75 6c 74  Tcl_AppendResult
33f0: 28 69 6e 74 65 72 70 2c 20 22 77 72 6f 6e 67 20  (interp, "wrong 
3400: 23 20 61 72 67 73 3a 20 73 68 6f 75 6c 64 20 62  # args: should b
3410: 65 20 5c 22 22 2c 20 61 72 67 76 5b 30 5d 2c 0a  e \"", argv[0],.
3420: 20 20 20 20 20 20 20 22 20 49 44 20 53 51 4c 22         " ID SQL"
3430: 2c 20 30 29 3b 0a 20 20 20 20 72 65 74 75 72 6e  , 0);.    return
3440: 20 54 43 4c 5f 45 52 52 4f 52 3b 0a 20 20 7d 0a   TCL_ERROR;.  }.
3450: 20 20 69 20 3d 20 70 61 72 73 65 5f 74 68 72 65    i = parse_thre
3460: 61 64 5f 69 64 28 69 6e 74 65 72 70 2c 20 61 72  ad_id(interp, ar
3470: 67 76 5b 31 5d 29 3b 0a 20 20 69 66 28 20 69 3c  gv[1]);.  if( i<
3480: 30 20 29 20 72 65 74 75 72 6e 20 54 43 4c 5f 45  0 ) return TCL_E
3490: 52 52 4f 52 3b 0a 20 20 69 66 28 20 21 74 68 72  RROR;.  if( !thr
34a0: 65 61 64 73 65 74 5b 69 5d 2e 62 75 73 79 20 29  eadset[i].busy )
34b0: 7b 0a 20 20 20 20 54 63 6c 5f 41 70 70 65 6e 64  {.    Tcl_Append
34c0: 52 65 73 75 6c 74 28 69 6e 74 65 72 70 2c 20 22  Result(interp, "
34d0: 6e 6f 20 73 75 63 68 20 74 68 72 65 61 64 22 2c  no such thread",
34e0: 20 30 29 3b 0a 20 20 20 20 72 65 74 75 72 6e 20   0);.    return 
34f0: 54 43 4c 5f 45 52 52 4f 52 3b 0a 20 20 7d 0a 20  TCL_ERROR;.  }. 
3500: 20 74 68 72 65 61 64 5f 77 61 69 74 28 26 74 68   thread_wait(&th
3510: 72 65 61 64 73 65 74 5b 69 5d 29 3b 0a 20 20 74  readset[i]);.  t
3520: 68 72 65 61 64 73 65 74 5b 69 5d 2e 78 4f 70 20  hreadset[i].xOp 
3530: 3d 20 64 6f 5f 63 6f 6d 70 69 6c 65 3b 0a 20 20  = do_compile;.  
3540: 73 71 6c 69 74 65 33 5f 66 72 65 65 28 74 68 72  sqlite3_free(thr
3550: 65 61 64 73 65 74 5b 69 5d 2e 7a 41 72 67 29 3b  eadset[i].zArg);
3560: 0a 20 20 74 68 72 65 61 64 73 65 74 5b 69 5d 2e  .  threadset[i].
3570: 7a 41 72 67 20 3d 20 73 71 6c 69 74 65 33 53 74  zArg = sqlite3St
3580: 72 44 75 70 28 61 72 67 76 5b 32 5d 29 3b 0a 20  rDup(argv[2]);. 
3590: 20 74 68 72 65 61 64 73 65 74 5b 69 5d 2e 6f 70   threadset[i].op
35a0: 6e 75 6d 2b 2b 3b 0a 20 20 72 65 74 75 72 6e 20  num++;.  return 
35b0: 54 43 4c 5f 4f 4b 3b 0a 7d 0a 0a 2f 2a 0a 2a 2a  TCL_OK;.}../*.**
35c0: 20 54 68 69 73 20 70 72 6f 63 65 64 75 72 65 20   This procedure 
35d0: 72 75 6e 73 20 69 6e 20 74 68 65 20 74 68 72 65  runs in the thre
35e0: 61 64 20 74 6f 20 73 74 65 70 20 74 68 65 20 76  ad to step the v
35f0: 69 72 74 75 61 6c 20 6d 61 63 68 69 6e 65 2e 0a  irtual machine..
3600: 2a 2f 0a 73 74 61 74 69 63 20 76 6f 69 64 20 64  */.static void d
3610: 6f 5f 73 74 65 70 28 54 68 72 65 61 64 20 2a 70  o_step(Thread *p
3620: 29 7b 0a 20 20 69 6e 74 20 69 3b 0a 20 20 69 66  ){.  int i;.  if
3630: 28 20 70 2d 3e 70 53 74 6d 74 3d 3d 30 20 29 7b  ( p->pStmt==0 ){
3640: 0a 20 20 20 20 70 2d 3e 7a 45 72 72 20 3d 20 70  .    p->zErr = p
3650: 2d 3e 7a 53 74 61 74 69 63 45 72 72 20 3d 20 22  ->zStaticErr = "
3660: 6e 6f 20 76 69 72 74 75 61 6c 20 6d 61 63 68 69  no virtual machi
3670: 6e 65 20 61 76 61 69 6c 61 62 6c 65 22 3b 0a 20  ne available";. 
3680: 20 20 20 70 2d 3e 72 63 20 3d 20 53 51 4c 49 54     p->rc = SQLIT
3690: 45 5f 45 52 52 4f 52 3b 0a 20 20 20 20 72 65 74  E_ERROR;.    ret
36a0: 75 72 6e 3b 0a 20 20 7d 0a 20 20 70 2d 3e 72 63  urn;.  }.  p->rc
36b0: 20 3d 20 73 71 6c 69 74 65 33 5f 73 74 65 70 28   = sqlite3_step(
36c0: 70 2d 3e 70 53 74 6d 74 29 3b 0a 20 20 69 66 28  p->pStmt);.  if(
36d0: 20 70 2d 3e 72 63 3d 3d 53 51 4c 49 54 45 5f 52   p->rc==SQLITE_R
36e0: 4f 57 20 29 7b 0a 20 20 20 20 70 2d 3e 61 72 67  OW ){.    p->arg
36f0: 63 20 3d 20 73 71 6c 69 74 65 33 5f 63 6f 6c 75  c = sqlite3_colu
3700: 6d 6e 5f 63 6f 75 6e 74 28 70 2d 3e 70 53 74 6d  mn_count(p->pStm
3710: 74 29 3b 0a 20 20 20 20 66 6f 72 28 69 3d 30 3b  t);.    for(i=0;
3720: 20 69 3c 73 71 6c 69 74 65 33 5f 64 61 74 61 5f   i<sqlite3_data_
3730: 63 6f 75 6e 74 28 70 2d 3e 70 53 74 6d 74 29 3b  count(p->pStmt);
3740: 20 69 2b 2b 29 7b 0a 20 20 20 20 20 20 70 2d 3e   i++){.      p->
3750: 61 72 67 76 5b 69 5d 20 3d 20 28 63 68 61 72 2a  argv[i] = (char*
3760: 29 73 71 6c 69 74 65 33 5f 63 6f 6c 75 6d 6e 5f  )sqlite3_column_
3770: 74 65 78 74 28 70 2d 3e 70 53 74 6d 74 2c 20 69  text(p->pStmt, i
3780: 29 3b 0a 20 20 20 20 7d 0a 20 20 20 20 66 6f 72  );.    }.    for
3790: 28 69 3d 30 3b 20 69 3c 70 2d 3e 61 72 67 63 3b  (i=0; i<p->argc;
37a0: 20 69 2b 2b 29 7b 0a 20 20 20 20 20 20 70 2d 3e   i++){.      p->
37b0: 63 6f 6c 76 5b 69 5d 20 3d 20 73 71 6c 69 74 65  colv[i] = sqlite
37c0: 33 5f 63 6f 6c 75 6d 6e 5f 6e 61 6d 65 28 70 2d  3_column_name(p-
37d0: 3e 70 53 74 6d 74 2c 20 69 29 3b 0a 20 20 20 20  >pStmt, i);.    
37e0: 7d 0a 20 20 7d 0a 7d 0a 0a 2f 2a 0a 2a 2a 20 55  }.  }.}../*.** U
37f0: 73 61 67 65 3a 20 74 68 72 65 61 64 5f 73 74 65  sage: thread_ste
3800: 70 20 49 44 0a 2a 2a 0a 2a 2a 20 41 64 76 61 6e  p ID.**.** Advan
3810: 63 65 20 74 68 65 20 76 69 72 74 75 61 6c 20 6d  ce the virtual m
3820: 61 63 68 69 6e 65 20 62 79 20 6f 6e 65 20 73 74  achine by one st
3830: 65 70 0a 2a 2f 0a 73 74 61 74 69 63 20 69 6e 74  ep.*/.static int
3840: 20 74 63 6c 5f 74 68 72 65 61 64 5f 73 74 65 70   tcl_thread_step
3850: 28 0a 20 20 76 6f 69 64 20 2a 4e 6f 74 55 73 65  (.  void *NotUse
3860: 64 2c 0a 20 20 54 63 6c 5f 49 6e 74 65 72 70 20  d,.  Tcl_Interp 
3870: 2a 69 6e 74 65 72 70 2c 20 20 20 20 2f 2a 20 54  *interp,    /* T
3880: 68 65 20 54 43 4c 20 69 6e 74 65 72 70 72 65 74  he TCL interpret
3890: 65 72 20 74 68 61 74 20 69 6e 76 6f 6b 65 64 20  er that invoked 
38a0: 74 68 69 73 20 63 6f 6d 6d 61 6e 64 20 2a 2f 0a  this command */.
38b0: 20 20 69 6e 74 20 61 72 67 63 2c 20 20 20 20 20    int argc,     
38c0: 20 20 20 20 20 20 20 20 20 2f 2a 20 4e 75 6d 62           /* Numb
38d0: 65 72 20 6f 66 20 61 72 67 75 6d 65 6e 74 73 20  er of arguments 
38e0: 2a 2f 0a 20 20 63 6f 6e 73 74 20 63 68 61 72 20  */.  const char 
38f0: 2a 2a 61 72 67 76 20 20 20 20 20 20 2f 2a 20 54  **argv      /* T
3900: 65 78 74 20 6f 66 20 65 61 63 68 20 61 72 67 75  ext of each argu
3910: 6d 65 6e 74 20 2a 2f 0a 29 7b 0a 20 20 69 6e 74  ment */.){.  int
3920: 20 69 3b 0a 20 20 69 66 28 20 61 72 67 63 21 3d   i;.  if( argc!=
3930: 32 20 29 7b 0a 20 20 20 20 54 63 6c 5f 41 70 70  2 ){.    Tcl_App
3940: 65 6e 64 52 65 73 75 6c 74 28 69 6e 74 65 72 70  endResult(interp
3950: 2c 20 22 77 72 6f 6e 67 20 23 20 61 72 67 73 3a  , "wrong # args:
3960: 20 73 68 6f 75 6c 64 20 62 65 20 5c 22 22 2c 20   should be \"", 
3970: 61 72 67 76 5b 30 5d 2c 0a 20 20 20 20 20 20 20  argv[0],.       
3980: 22 20 49 44 4c 22 2c 20 30 29 3b 0a 20 20 20 20  " IDL", 0);.    
3990: 72 65 74 75 72 6e 20 54 43 4c 5f 45 52 52 4f 52  return TCL_ERROR
39a0: 3b 0a 20 20 7d 0a 20 20 69 20 3d 20 70 61 72 73  ;.  }.  i = pars
39b0: 65 5f 74 68 72 65 61 64 5f 69 64 28 69 6e 74 65  e_thread_id(inte
39c0: 72 70 2c 20 61 72 67 76 5b 31 5d 29 3b 0a 20 20  rp, argv[1]);.  
39d0: 69 66 28 20 69 3c 30 20 29 20 72 65 74 75 72 6e  if( i<0 ) return
39e0: 20 54 43 4c 5f 45 52 52 4f 52 3b 0a 20 20 69 66   TCL_ERROR;.  if
39f0: 28 20 21 74 68 72 65 61 64 73 65 74 5b 69 5d 2e  ( !threadset[i].
3a00: 62 75 73 79 20 29 7b 0a 20 20 20 20 54 63 6c 5f  busy ){.    Tcl_
3a10: 41 70 70 65 6e 64 52 65 73 75 6c 74 28 69 6e 74  AppendResult(int
3a20: 65 72 70 2c 20 22 6e 6f 20 73 75 63 68 20 74 68  erp, "no such th
3a30: 72 65 61 64 22 2c 20 30 29 3b 0a 20 20 20 20 72  read", 0);.    r
3a40: 65 74 75 72 6e 20 54 43 4c 5f 45 52 52 4f 52 3b  eturn TCL_ERROR;
3a50: 0a 20 20 7d 0a 20 20 74 68 72 65 61 64 5f 77 61  .  }.  thread_wa
3a60: 69 74 28 26 74 68 72 65 61 64 73 65 74 5b 69 5d  it(&threadset[i]
3a70: 29 3b 0a 20 20 74 68 72 65 61 64 73 65 74 5b 69  );.  threadset[i
3a80: 5d 2e 78 4f 70 20 3d 20 64 6f 5f 73 74 65 70 3b  ].xOp = do_step;
3a90: 0a 20 20 74 68 72 65 61 64 73 65 74 5b 69 5d 2e  .  threadset[i].
3aa0: 6f 70 6e 75 6d 2b 2b 3b 0a 20 20 72 65 74 75 72  opnum++;.  retur
3ab0: 6e 20 54 43 4c 5f 4f 4b 3b 0a 7d 0a 0a 2f 2a 0a  n TCL_OK;.}../*.
3ac0: 2a 2a 20 54 68 69 73 20 70 72 6f 63 65 64 75 72  ** This procedur
3ad0: 65 20 72 75 6e 73 20 69 6e 20 74 68 65 20 74 68  e runs in the th
3ae0: 72 65 61 64 20 74 6f 20 66 69 6e 61 6c 69 7a 65  read to finalize
3af0: 20 61 20 76 69 72 74 75 61 6c 20 6d 61 63 68 69   a virtual machi
3b00: 6e 65 2e 0a 2a 2f 0a 73 74 61 74 69 63 20 76 6f  ne..*/.static vo
3b10: 69 64 20 64 6f 5f 66 69 6e 61 6c 69 7a 65 28 54  id do_finalize(T
3b20: 68 72 65 61 64 20 2a 70 29 7b 0a 20 20 69 66 28  hread *p){.  if(
3b30: 20 70 2d 3e 70 53 74 6d 74 3d 3d 30 20 29 7b 0a   p->pStmt==0 ){.
3b40: 20 20 20 20 70 2d 3e 7a 45 72 72 20 3d 20 70 2d      p->zErr = p-
3b50: 3e 7a 53 74 61 74 69 63 45 72 72 20 3d 20 22 6e  >zStaticErr = "n
3b60: 6f 20 76 69 72 74 75 61 6c 20 6d 61 63 68 69 6e  o virtual machin
3b70: 65 20 61 76 61 69 6c 61 62 6c 65 22 3b 0a 20 20  e available";.  
3b80: 20 20 70 2d 3e 72 63 20 3d 20 53 51 4c 49 54 45    p->rc = SQLITE
3b90: 5f 45 52 52 4f 52 3b 0a 20 20 20 20 72 65 74 75  _ERROR;.    retu
3ba0: 72 6e 3b 0a 20 20 7d 0a 20 20 70 2d 3e 72 63 20  rn;.  }.  p->rc 
3bb0: 3d 20 73 71 6c 69 74 65 33 5f 66 69 6e 61 6c 69  = sqlite3_finali
3bc0: 7a 65 28 70 2d 3e 70 53 74 6d 74 29 3b 0a 20 20  ze(p->pStmt);.  
3bd0: 70 2d 3e 70 53 74 6d 74 20 3d 20 30 3b 0a 7d 0a  p->pStmt = 0;.}.
3be0: 0a 2f 2a 0a 2a 2a 20 55 73 61 67 65 3a 20 74 68  ./*.** Usage: th
3bf0: 72 65 61 64 5f 66 69 6e 61 6c 69 7a 65 20 49 44  read_finalize ID
3c00: 0a 2a 2a 0a 2a 2a 20 46 69 6e 61 6c 69 7a 65 20  .**.** Finalize 
3c10: 74 68 65 20 76 69 72 74 75 61 6c 20 6d 61 63 68  the virtual mach
3c20: 69 6e 65 2e 0a 2a 2f 0a 73 74 61 74 69 63 20 69  ine..*/.static i
3c30: 6e 74 20 74 63 6c 5f 74 68 72 65 61 64 5f 66 69  nt tcl_thread_fi
3c40: 6e 61 6c 69 7a 65 28 0a 20 20 76 6f 69 64 20 2a  nalize(.  void *
3c50: 4e 6f 74 55 73 65 64 2c 0a 20 20 54 63 6c 5f 49  NotUsed,.  Tcl_I
3c60: 6e 74 65 72 70 20 2a 69 6e 74 65 72 70 2c 20 20  nterp *interp,  
3c70: 20 20 2f 2a 20 54 68 65 20 54 43 4c 20 69 6e 74    /* The TCL int
3c80: 65 72 70 72 65 74 65 72 20 74 68 61 74 20 69 6e  erpreter that in
3c90: 76 6f 6b 65 64 20 74 68 69 73 20 63 6f 6d 6d 61  voked this comma
3ca0: 6e 64 20 2a 2f 0a 20 20 69 6e 74 20 61 72 67 63  nd */.  int argc
3cb0: 2c 20 20 20 20 20 20 20 20 20 20 20 20 20 20 2f  ,              /
3cc0: 2a 20 4e 75 6d 62 65 72 20 6f 66 20 61 72 67 75  * Number of argu
3cd0: 6d 65 6e 74 73 20 2a 2f 0a 20 20 63 6f 6e 73 74  ments */.  const
3ce0: 20 63 68 61 72 20 2a 2a 61 72 67 76 20 20 20 20   char **argv    
3cf0: 20 20 2f 2a 20 54 65 78 74 20 6f 66 20 65 61 63    /* Text of eac
3d00: 68 20 61 72 67 75 6d 65 6e 74 20 2a 2f 0a 29 7b  h argument */.){
3d10: 0a 20 20 69 6e 74 20 69 3b 0a 20 20 69 66 28 20  .  int i;.  if( 
3d20: 61 72 67 63 21 3d 32 20 29 7b 0a 20 20 20 20 54  argc!=2 ){.    T
3d30: 63 6c 5f 41 70 70 65 6e 64 52 65 73 75 6c 74 28  cl_AppendResult(
3d40: 69 6e 74 65 72 70 2c 20 22 77 72 6f 6e 67 20 23  interp, "wrong #
3d50: 20 61 72 67 73 3a 20 73 68 6f 75 6c 64 20 62 65   args: should be
3d60: 20 5c 22 22 2c 20 61 72 67 76 5b 30 5d 2c 0a 20   \"", argv[0],. 
3d70: 20 20 20 20 20 20 22 20 49 44 4c 22 2c 20 30 29        " IDL", 0)
3d80: 3b 0a 20 20 20 20 72 65 74 75 72 6e 20 54 43 4c  ;.    return TCL
3d90: 5f 45 52 52 4f 52 3b 0a 20 20 7d 0a 20 20 69 20  _ERROR;.  }.  i 
3da0: 3d 20 70 61 72 73 65 5f 74 68 72 65 61 64 5f 69  = parse_thread_i
3db0: 64 28 69 6e 74 65 72 70 2c 20 61 72 67 76 5b 31  d(interp, argv[1
3dc0: 5d 29 3b 0a 20 20 69 66 28 20 69 3c 30 20 29 20  ]);.  if( i<0 ) 
3dd0: 72 65 74 75 72 6e 20 54 43 4c 5f 45 52 52 4f 52  return TCL_ERROR
3de0: 3b 0a 20 20 69 66 28 20 21 74 68 72 65 61 64 73  ;.  if( !threads
3df0: 65 74 5b 69 5d 2e 62 75 73 79 20 29 7b 0a 20 20  et[i].busy ){.  
3e00: 20 20 54 63 6c 5f 41 70 70 65 6e 64 52 65 73 75    Tcl_AppendResu
3e10: 6c 74 28 69 6e 74 65 72 70 2c 20 22 6e 6f 20 73  lt(interp, "no s
3e20: 75 63 68 20 74 68 72 65 61 64 22 2c 20 30 29 3b  uch thread", 0);
3e30: 0a 20 20 20 20 72 65 74 75 72 6e 20 54 43 4c 5f  .    return TCL_
3e40: 45 52 52 4f 52 3b 0a 20 20 7d 0a 20 20 74 68 72  ERROR;.  }.  thr
3e50: 65 61 64 5f 77 61 69 74 28 26 74 68 72 65 61 64  ead_wait(&thread
3e60: 73 65 74 5b 69 5d 29 3b 0a 20 20 74 68 72 65 61  set[i]);.  threa
3e70: 64 73 65 74 5b 69 5d 2e 78 4f 70 20 3d 20 64 6f  dset[i].xOp = do
3e80: 5f 66 69 6e 61 6c 69 7a 65 3b 0a 20 20 73 71 6c  _finalize;.  sql
3e90: 69 74 65 33 5f 66 72 65 65 28 74 68 72 65 61 64  ite3_free(thread
3ea0: 73 65 74 5b 69 5d 2e 7a 41 72 67 29 3b 0a 20 20  set[i].zArg);.  
3eb0: 74 68 72 65 61 64 73 65 74 5b 69 5d 2e 7a 41 72  threadset[i].zAr
3ec0: 67 20 3d 20 30 3b 0a 20 20 74 68 72 65 61 64 73  g = 0;.  threads
3ed0: 65 74 5b 69 5d 2e 6f 70 6e 75 6d 2b 2b 3b 0a 20  et[i].opnum++;. 
3ee0: 20 72 65 74 75 72 6e 20 54 43 4c 5f 4f 4b 3b 0a   return TCL_OK;.
3ef0: 7d 0a 0a 2f 2a 0a 2a 2a 20 55 73 61 67 65 3a 20  }../*.** Usage: 
3f00: 74 68 72 65 61 64 5f 73 77 61 70 20 49 44 20 49  thread_swap ID I
3f10: 44 0a 2a 2a 0a 2a 2a 20 49 6e 74 65 72 63 68 61  D.**.** Intercha
3f20: 6e 67 65 20 74 68 65 20 73 71 6c 69 74 65 2a 20  nge the sqlite* 
3f30: 70 6f 69 6e 74 65 72 20 62 65 74 77 65 65 6e 20  pointer between 
3f40: 74 77 6f 20 74 68 72 65 61 64 73 2e 0a 2a 2f 0a  two threads..*/.
3f50: 73 74 61 74 69 63 20 69 6e 74 20 74 63 6c 5f 74  static int tcl_t
3f60: 68 72 65 61 64 5f 73 77 61 70 28 0a 20 20 76 6f  hread_swap(.  vo
3f70: 69 64 20 2a 4e 6f 74 55 73 65 64 2c 0a 20 20 54  id *NotUsed,.  T
3f80: 63 6c 5f 49 6e 74 65 72 70 20 2a 69 6e 74 65 72  cl_Interp *inter
3f90: 70 2c 20 20 20 20 2f 2a 20 54 68 65 20 54 43 4c  p,    /* The TCL
3fa0: 20 69 6e 74 65 72 70 72 65 74 65 72 20 74 68 61   interpreter tha
3fb0: 74 20 69 6e 76 6f 6b 65 64 20 74 68 69 73 20 63  t invoked this c
3fc0: 6f 6d 6d 61 6e 64 20 2a 2f 0a 20 20 69 6e 74 20  ommand */.  int 
3fd0: 61 72 67 63 2c 20 20 20 20 20 20 20 20 20 20 20  argc,           
3fe0: 20 20 20 2f 2a 20 4e 75 6d 62 65 72 20 6f 66 20     /* Number of 
3ff0: 61 72 67 75 6d 65 6e 74 73 20 2a 2f 0a 20 20 63  arguments */.  c
4000: 6f 6e 73 74 20 63 68 61 72 20 2a 2a 61 72 67 76  onst char **argv
4010: 20 20 20 20 20 20 2f 2a 20 54 65 78 74 20 6f 66        /* Text of
4020: 20 65 61 63 68 20 61 72 67 75 6d 65 6e 74 20 2a   each argument *
4030: 2f 0a 29 7b 0a 20 20 69 6e 74 20 69 2c 20 6a 3b  /.){.  int i, j;
4040: 0a 20 20 73 71 6c 69 74 65 33 20 2a 74 65 6d 70  .  sqlite3 *temp
4050: 3b 0a 20 20 69 66 28 20 61 72 67 63 21 3d 33 20  ;.  if( argc!=3 
4060: 29 7b 0a 20 20 20 20 54 63 6c 5f 41 70 70 65 6e  ){.    Tcl_Appen
4070: 64 52 65 73 75 6c 74 28 69 6e 74 65 72 70 2c 20  dResult(interp, 
4080: 22 77 72 6f 6e 67 20 23 20 61 72 67 73 3a 20 73  "wrong # args: s
4090: 68 6f 75 6c 64 20 62 65 20 5c 22 22 2c 20 61 72  hould be \"", ar
40a0: 67 76 5b 30 5d 2c 0a 20 20 20 20 20 20 20 22 20  gv[0],.       " 
40b0: 49 44 31 20 49 44 32 22 2c 20 30 29 3b 0a 20 20  ID1 ID2", 0);.  
40c0: 20 20 72 65 74 75 72 6e 20 54 43 4c 5f 45 52 52    return TCL_ERR
40d0: 4f 52 3b 0a 20 20 7d 0a 20 20 69 20 3d 20 70 61  OR;.  }.  i = pa
40e0: 72 73 65 5f 74 68 72 65 61 64 5f 69 64 28 69 6e  rse_thread_id(in
40f0: 74 65 72 70 2c 20 61 72 67 76 5b 31 5d 29 3b 0a  terp, argv[1]);.
4100: 20 20 69 66 28 20 69 3c 30 20 29 20 72 65 74 75    if( i<0 ) retu
4110: 72 6e 20 54 43 4c 5f 45 52 52 4f 52 3b 0a 20 20  rn TCL_ERROR;.  
4120: 69 66 28 20 21 74 68 72 65 61 64 73 65 74 5b 69  if( !threadset[i
4130: 5d 2e 62 75 73 79 20 29 7b 0a 20 20 20 20 54 63  ].busy ){.    Tc
4140: 6c 5f 41 70 70 65 6e 64 52 65 73 75 6c 74 28 69  l_AppendResult(i
4150: 6e 74 65 72 70 2c 20 22 6e 6f 20 73 75 63 68 20  nterp, "no such 
4160: 74 68 72 65 61 64 22 2c 20 30 29 3b 0a 20 20 20  thread", 0);.   
4170: 20 72 65 74 75 72 6e 20 54 43 4c 5f 45 52 52 4f   return TCL_ERRO
4180: 52 3b 0a 20 20 7d 0a 20 20 74 68 72 65 61 64 5f  R;.  }.  thread_
4190: 77 61 69 74 28 26 74 68 72 65 61 64 73 65 74 5b  wait(&threadset[
41a0: 69 5d 29 3b 0a 20 20 6a 20 3d 20 70 61 72 73 65  i]);.  j = parse
41b0: 5f 74 68 72 65 61 64 5f 69 64 28 69 6e 74 65 72  _thread_id(inter
41c0: 70 2c 20 61 72 67 76 5b 32 5d 29 3b 0a 20 20 69  p, argv[2]);.  i
41d0: 66 28 20 6a 3c 30 20 29 20 72 65 74 75 72 6e 20  f( j<0 ) return 
41e0: 54 43 4c 5f 45 52 52 4f 52 3b 0a 20 20 69 66 28  TCL_ERROR;.  if(
41f0: 20 21 74 68 72 65 61 64 73 65 74 5b 6a 5d 2e 62   !threadset[j].b
4200: 75 73 79 20 29 7b 0a 20 20 20 20 54 63 6c 5f 41  usy ){.    Tcl_A
4210: 70 70 65 6e 64 52 65 73 75 6c 74 28 69 6e 74 65  ppendResult(inte
4220: 72 70 2c 20 22 6e 6f 20 73 75 63 68 20 74 68 72  rp, "no such thr
4230: 65 61 64 22 2c 20 30 29 3b 0a 20 20 20 20 72 65  ead", 0);.    re
4240: 74 75 72 6e 20 54 43 4c 5f 45 52 52 4f 52 3b 0a  turn TCL_ERROR;.
4250: 20 20 7d 0a 20 20 74 68 72 65 61 64 5f 77 61 69    }.  thread_wai
4260: 74 28 26 74 68 72 65 61 64 73 65 74 5b 6a 5d 29  t(&threadset[j])
4270: 3b 0a 20 20 74 65 6d 70 20 3d 20 74 68 72 65 61  ;.  temp = threa
4280: 64 73 65 74 5b 69 5d 2e 64 62 3b 0a 20 20 74 68  dset[i].db;.  th
4290: 72 65 61 64 73 65 74 5b 69 5d 2e 64 62 20 3d 20  readset[i].db = 
42a0: 74 68 72 65 61 64 73 65 74 5b 6a 5d 2e 64 62 3b  threadset[j].db;
42b0: 0a 20 20 74 68 72 65 61 64 73 65 74 5b 6a 5d 2e  .  threadset[j].
42c0: 64 62 20 3d 20 74 65 6d 70 3b 0a 20 20 72 65 74  db = temp;.  ret
42d0: 75 72 6e 20 54 43 4c 5f 4f 4b 3b 0a 7d 0a 0a 2f  urn TCL_OK;.}../
42e0: 2a 0a 2a 2a 20 55 73 61 67 65 3a 20 74 68 72 65  *.** Usage: thre
42f0: 61 64 5f 64 62 5f 67 65 74 20 49 44 0a 2a 2a 0a  ad_db_get ID.**.
4300: 2a 2a 20 52 65 74 75 72 6e 20 74 68 65 20 64 61  ** Return the da
4310: 74 61 62 61 73 65 20 63 6f 6e 6e 65 63 74 69 6f  tabase connectio
4320: 6e 20 70 6f 69 6e 74 65 72 20 66 6f 72 20 74 68  n pointer for th
4330: 65 20 67 69 76 65 6e 20 74 68 72 65 61 64 2e 20  e given thread. 
4340: 20 54 68 65 6e 0a 2a 2a 20 72 65 6d 6f 76 65 20   Then.** remove 
4350: 74 68 65 20 70 6f 69 6e 74 65 72 20 66 72 6f 6d  the pointer from
4360: 20 74 68 65 20 74 68 72 65 61 64 20 69 74 73 65   the thread itse
4370: 6c 66 2e 20 20 41 66 74 65 72 77 61 72 64 73 2c  lf.  Afterwards,
4380: 20 74 68 65 20 74 68 72 65 61 64 0a 2a 2a 20 63   the thread.** c
4390: 61 6e 20 62 65 20 73 74 6f 70 70 65 64 20 61 6e  an be stopped an
43a0: 64 20 74 68 65 20 63 6f 6e 6e 65 63 74 69 6f 6e  d the connection
43b0: 20 63 61 6e 20 62 65 20 75 73 65 64 20 62 79 20   can be used by 
43c0: 74 68 65 20 6d 61 69 6e 20 74 68 72 65 61 64 2e  the main thread.
43d0: 0a 2a 2f 0a 73 74 61 74 69 63 20 69 6e 74 20 74  .*/.static int t
43e0: 63 6c 5f 74 68 72 65 61 64 5f 64 62 5f 67 65 74  cl_thread_db_get
43f0: 28 0a 20 20 76 6f 69 64 20 2a 4e 6f 74 55 73 65  (.  void *NotUse
4400: 64 2c 0a 20 20 54 63 6c 5f 49 6e 74 65 72 70 20  d,.  Tcl_Interp 
4410: 2a 69 6e 74 65 72 70 2c 20 20 20 20 2f 2a 20 54  *interp,    /* T
4420: 68 65 20 54 43 4c 20 69 6e 74 65 72 70 72 65 74  he TCL interpret
4430: 65 72 20 74 68 61 74 20 69 6e 76 6f 6b 65 64 20  er that invoked 
4440: 74 68 69 73 20 63 6f 6d 6d 61 6e 64 20 2a 2f 0a  this command */.
4450: 20 20 69 6e 74 20 61 72 67 63 2c 20 20 20 20 20    int argc,     
4460: 20 20 20 20 20 20 20 20 20 2f 2a 20 4e 75 6d 62           /* Numb
4470: 65 72 20 6f 66 20 61 72 67 75 6d 65 6e 74 73 20  er of arguments 
4480: 2a 2f 0a 20 20 63 6f 6e 73 74 20 63 68 61 72 20  */.  const char 
4490: 2a 2a 61 72 67 76 20 20 20 20 20 20 2f 2a 20 54  **argv      /* T
44a0: 65 78 74 20 6f 66 20 65 61 63 68 20 61 72 67 75  ext of each argu
44b0: 6d 65 6e 74 20 2a 2f 0a 29 7b 0a 20 20 69 6e 74  ment */.){.  int
44c0: 20 69 3b 0a 20 20 63 68 61 72 20 7a 42 75 66 5b   i;.  char zBuf[
44d0: 31 30 30 5d 3b 0a 20 20 65 78 74 65 72 6e 20 69  100];.  extern i
44e0: 6e 74 20 73 71 6c 69 74 65 33 54 65 73 74 4d 61  nt sqlite3TestMa
44f0: 6b 65 50 6f 69 6e 74 65 72 53 74 72 28 54 63 6c  kePointerStr(Tcl
4500: 5f 49 6e 74 65 72 70 2a 2c 20 63 68 61 72 2a 2c  _Interp*, char*,
4510: 20 76 6f 69 64 2a 29 3b 0a 20 20 69 66 28 20 61   void*);.  if( a
4520: 72 67 63 21 3d 32 20 29 7b 0a 20 20 20 20 54 63  rgc!=2 ){.    Tc
4530: 6c 5f 41 70 70 65 6e 64 52 65 73 75 6c 74 28 69  l_AppendResult(i
4540: 6e 74 65 72 70 2c 20 22 77 72 6f 6e 67 20 23 20  nterp, "wrong # 
4550: 61 72 67 73 3a 20 73 68 6f 75 6c 64 20 62 65 20  args: should be 
4560: 5c 22 22 2c 20 61 72 67 76 5b 30 5d 2c 0a 20 20  \"", argv[0],.  
4570: 20 20 20 20 20 22 20 49 44 22 2c 20 30 29 3b 0a       " ID", 0);.
4580: 20 20 20 20 72 65 74 75 72 6e 20 54 43 4c 5f 45      return TCL_E
4590: 52 52 4f 52 3b 0a 20 20 7d 0a 20 20 69 20 3d 20  RROR;.  }.  i = 
45a0: 70 61 72 73 65 5f 74 68 72 65 61 64 5f 69 64 28  parse_thread_id(
45b0: 69 6e 74 65 72 70 2c 20 61 72 67 76 5b 31 5d 29  interp, argv[1])
45c0: 3b 0a 20 20 69 66 28 20 69 3c 30 20 29 20 72 65  ;.  if( i<0 ) re
45d0: 74 75 72 6e 20 54 43 4c 5f 45 52 52 4f 52 3b 0a  turn TCL_ERROR;.
45e0: 20 20 69 66 28 20 21 74 68 72 65 61 64 73 65 74    if( !threadset
45f0: 5b 69 5d 2e 62 75 73 79 20 29 7b 0a 20 20 20 20  [i].busy ){.    
4600: 54 63 6c 5f 41 70 70 65 6e 64 52 65 73 75 6c 74  Tcl_AppendResult
4610: 28 69 6e 74 65 72 70 2c 20 22 6e 6f 20 73 75 63  (interp, "no suc
4620: 68 20 74 68 72 65 61 64 22 2c 20 30 29 3b 0a 20  h thread", 0);. 
4630: 20 20 20 72 65 74 75 72 6e 20 54 43 4c 5f 45 52     return TCL_ER
4640: 52 4f 52 3b 0a 20 20 7d 0a 20 20 74 68 72 65 61  ROR;.  }.  threa
4650: 64 5f 77 61 69 74 28 26 74 68 72 65 61 64 73 65  d_wait(&threadse
4660: 74 5b 69 5d 29 3b 0a 20 20 73 71 6c 69 74 65 33  t[i]);.  sqlite3
4670: 54 65 73 74 4d 61 6b 65 50 6f 69 6e 74 65 72 53  TestMakePointerS
4680: 74 72 28 69 6e 74 65 72 70 2c 20 7a 42 75 66 2c  tr(interp, zBuf,
4690: 20 74 68 72 65 61 64 73 65 74 5b 69 5d 2e 64 62   threadset[i].db
46a0: 29 3b 0a 20 20 74 68 72 65 61 64 73 65 74 5b 69  );.  threadset[i
46b0: 5d 2e 64 62 20 3d 20 30 3b 0a 20 20 54 63 6c 5f  ].db = 0;.  Tcl_
46c0: 41 70 70 65 6e 64 52 65 73 75 6c 74 28 69 6e 74  AppendResult(int
46d0: 65 72 70 2c 20 7a 42 75 66 2c 20 28 63 68 61 72  erp, zBuf, (char
46e0: 2a 29 30 29 3b 0a 20 20 72 65 74 75 72 6e 20 54  *)0);.  return T
46f0: 43 4c 5f 4f 4b 3b 0a 7d 0a 0a 2f 2a 0a 2a 2a 20  CL_OK;.}../*.** 
4700: 55 73 61 67 65 3a 20 74 68 72 65 61 64 5f 73 74  Usage: thread_st
4710: 6d 74 5f 67 65 74 20 49 44 0a 2a 2a 0a 2a 2a 20  mt_get ID.**.** 
4720: 52 65 74 75 72 6e 20 74 68 65 20 64 61 74 61 62  Return the datab
4730: 61 73 65 20 73 74 6d 74 20 70 6f 69 6e 74 65 72  ase stmt pointer
4740: 20 66 6f 72 20 74 68 65 20 67 69 76 65 6e 20 74   for the given t
4750: 68 72 65 61 64 2e 20 20 54 68 65 6e 0a 2a 2a 20  hread.  Then.** 
4760: 72 65 6d 6f 76 65 20 74 68 65 20 70 6f 69 6e 74  remove the point
4770: 65 72 20 66 72 6f 6d 20 74 68 65 20 74 68 72 65  er from the thre
4780: 61 64 20 69 74 73 65 6c 66 2e 20 0a 2a 2f 0a 73  ad itself. .*/.s
4790: 74 61 74 69 63 20 69 6e 74 20 74 63 6c 5f 74 68  tatic int tcl_th
47a0: 72 65 61 64 5f 73 74 6d 74 5f 67 65 74 28 0a 20  read_stmt_get(. 
47b0: 20 76 6f 69 64 20 2a 4e 6f 74 55 73 65 64 2c 0a   void *NotUsed,.
47c0: 20 20 54 63 6c 5f 49 6e 74 65 72 70 20 2a 69 6e    Tcl_Interp *in
47d0: 74 65 72 70 2c 20 20 20 20 2f 2a 20 54 68 65 20  terp,    /* The 
47e0: 54 43 4c 20 69 6e 74 65 72 70 72 65 74 65 72 20  TCL interpreter 
47f0: 74 68 61 74 20 69 6e 76 6f 6b 65 64 20 74 68 69  that invoked thi
4800: 73 20 63 6f 6d 6d 61 6e 64 20 2a 2f 0a 20 20 69  s command */.  i
4810: 6e 74 20 61 72 67 63 2c 20 20 20 20 20 20 20 20  nt argc,        
4820: 20 20 20 20 20 20 2f 2a 20 4e 75 6d 62 65 72 20        /* Number 
4830: 6f 66 20 61 72 67 75 6d 65 6e 74 73 20 2a 2f 0a  of arguments */.
4840: 20 20 63 6f 6e 73 74 20 63 68 61 72 20 2a 2a 61    const char **a
4850: 72 67 76 20 20 20 20 20 20 2f 2a 20 54 65 78 74  rgv      /* Text
4860: 20 6f 66 20 65 61 63 68 20 61 72 67 75 6d 65 6e   of each argumen
4870: 74 20 2a 2f 0a 29 7b 0a 20 20 69 6e 74 20 69 3b  t */.){.  int i;
4880: 0a 20 20 63 68 61 72 20 7a 42 75 66 5b 31 30 30  .  char zBuf[100
4890: 5d 3b 0a 20 20 65 78 74 65 72 6e 20 69 6e 74 20  ];.  extern int 
48a0: 73 71 6c 69 74 65 33 54 65 73 74 4d 61 6b 65 50  sqlite3TestMakeP
48b0: 6f 69 6e 74 65 72 53 74 72 28 54 63 6c 5f 49 6e  ointerStr(Tcl_In
48c0: 74 65 72 70 2a 2c 20 63 68 61 72 2a 2c 20 76 6f  terp*, char*, vo
48d0: 69 64 2a 29 3b 0a 20 20 69 66 28 20 61 72 67 63  id*);.  if( argc
48e0: 21 3d 32 20 29 7b 0a 20 20 20 20 54 63 6c 5f 41  !=2 ){.    Tcl_A
48f0: 70 70 65 6e 64 52 65 73 75 6c 74 28 69 6e 74 65  ppendResult(inte
4900: 72 70 2c 20 22 77 72 6f 6e 67 20 23 20 61 72 67  rp, "wrong # arg
4910: 73 3a 20 73 68 6f 75 6c 64 20 62 65 20 5c 22 22  s: should be \""
4920: 2c 20 61 72 67 76 5b 30 5d 2c 0a 20 20 20 20 20  , argv[0],.     
4930: 20 20 22 20 49 44 22 2c 20 30 29 3b 0a 20 20 20    " ID", 0);.   
4940: 20 72 65 74 75 72 6e 20 54 43 4c 5f 45 52 52 4f   return TCL_ERRO
4950: 52 3b 0a 20 20 7d 0a 20 20 69 20 3d 20 70 61 72  R;.  }.  i = par
4960: 73 65 5f 74 68 72 65 61 64 5f 69 64 28 69 6e 74  se_thread_id(int
4970: 65 72 70 2c 20 61 72 67 76 5b 31 5d 29 3b 0a 20  erp, argv[1]);. 
4980: 20 69 66 28 20 69 3c 30 20 29 20 72 65 74 75 72   if( i<0 ) retur
4990: 6e 20 54 43 4c 5f 45 52 52 4f 52 3b 0a 20 20 69  n TCL_ERROR;.  i
49a0: 66 28 20 21 74 68 72 65 61 64 73 65 74 5b 69 5d  f( !threadset[i]
49b0: 2e 62 75 73 79 20 29 7b 0a 20 20 20 20 54 63 6c  .busy ){.    Tcl
49c0: 5f 41 70 70 65 6e 64 52 65 73 75 6c 74 28 69 6e  _AppendResult(in
49d0: 74 65 72 70 2c 20 22 6e 6f 20 73 75 63 68 20 74  terp, "no such t
49e0: 68 72 65 61 64 22 2c 20 30 29 3b 0a 20 20 20 20  hread", 0);.    
49f0: 72 65 74 75 72 6e 20 54 43 4c 5f 45 52 52 4f 52  return TCL_ERROR
4a00: 3b 0a 20 20 7d 0a 20 20 74 68 72 65 61 64 5f 77  ;.  }.  thread_w
4a10: 61 69 74 28 26 74 68 72 65 61 64 73 65 74 5b 69  ait(&threadset[i
4a20: 5d 29 3b 0a 20 20 73 71 6c 69 74 65 33 54 65 73  ]);.  sqlite3Tes
4a30: 74 4d 61 6b 65 50 6f 69 6e 74 65 72 53 74 72 28  tMakePointerStr(
4a40: 69 6e 74 65 72 70 2c 20 7a 42 75 66 2c 20 74 68  interp, zBuf, th
4a50: 72 65 61 64 73 65 74 5b 69 5d 2e 70 53 74 6d 74  readset[i].pStmt
4a60: 29 3b 0a 20 20 74 68 72 65 61 64 73 65 74 5b 69  );.  threadset[i
4a70: 5d 2e 70 53 74 6d 74 20 3d 20 30 3b 0a 20 20 54  ].pStmt = 0;.  T
4a80: 63 6c 5f 41 70 70 65 6e 64 52 65 73 75 6c 74 28  cl_AppendResult(
4a90: 69 6e 74 65 72 70 2c 20 7a 42 75 66 2c 20 28 63  interp, zBuf, (c
4aa0: 68 61 72 2a 29 30 29 3b 0a 20 20 72 65 74 75 72  har*)0);.  retur
4ab0: 6e 20 54 43 4c 5f 4f 4b 3b 0a 7d 0a 0a 2f 2a 0a  n TCL_OK;.}../*.
4ac0: 2a 2a 20 52 65 67 69 73 74 65 72 20 63 6f 6d 6d  ** Register comm
4ad0: 61 6e 64 73 20 77 69 74 68 20 74 68 65 20 54 43  ands with the TC
4ae0: 4c 20 69 6e 74 65 72 70 72 65 74 65 72 2e 0a 2a  L interpreter..*
4af0: 2f 0a 69 6e 74 20 53 71 6c 69 74 65 74 65 73 74  /.int Sqlitetest
4b00: 34 5f 49 6e 69 74 28 54 63 6c 5f 49 6e 74 65 72  4_Init(Tcl_Inter
4b10: 70 20 2a 69 6e 74 65 72 70 29 7b 0a 20 20 73 74  p *interp){.  st
4b20: 61 74 69 63 20 73 74 72 75 63 74 20 7b 0a 20 20  atic struct {.  
4b30: 20 20 20 63 68 61 72 20 2a 7a 4e 61 6d 65 3b 0a     char *zName;.
4b40: 20 20 20 20 20 54 63 6c 5f 43 6d 64 50 72 6f 63       Tcl_CmdProc
4b50: 20 2a 78 50 72 6f 63 3b 0a 20 20 7d 20 61 43 6d   *xProc;.  } aCm
4b60: 64 5b 5d 20 3d 20 7b 0a 20 20 20 20 20 7b 20 22  d[] = {.     { "
4b70: 74 68 72 65 61 64 5f 63 72 65 61 74 65 22 2c 20  thread_create", 
4b80: 20 20 20 20 28 54 63 6c 5f 43 6d 64 50 72 6f 63      (Tcl_CmdProc
4b90: 2a 29 74 63 6c 5f 74 68 72 65 61 64 5f 63 72 65  *)tcl_thread_cre
4ba0: 61 74 65 20 20 20 20 20 7d 2c 0a 20 20 20 20 20  ate     },.     
4bb0: 7b 20 22 74 68 72 65 61 64 5f 77 61 69 74 22 2c  { "thread_wait",
4bc0: 20 20 20 20 20 20 20 28 54 63 6c 5f 43 6d 64 50         (Tcl_CmdP
4bd0: 72 6f 63 2a 29 74 63 6c 5f 74 68 72 65 61 64 5f  roc*)tcl_thread_
4be0: 77 61 69 74 20 20 20 20 20 20 20 7d 2c 0a 20 20  wait       },.  
4bf0: 20 20 20 7b 20 22 74 68 72 65 61 64 5f 68 61 6c     { "thread_hal
4c00: 74 22 2c 20 20 20 20 20 20 20 28 54 63 6c 5f 43  t",       (Tcl_C
4c10: 6d 64 50 72 6f 63 2a 29 74 63 6c 5f 74 68 72 65  mdProc*)tcl_thre
4c20: 61 64 5f 68 61 6c 74 20 20 20 20 20 20 20 7d 2c  ad_halt       },
4c30: 0a 20 20 20 20 20 7b 20 22 74 68 72 65 61 64 5f  .     { "thread_
4c40: 61 72 67 63 22 2c 20 20 20 20 20 20 20 28 54 63  argc",       (Tc
4c50: 6c 5f 43 6d 64 50 72 6f 63 2a 29 74 63 6c 5f 74  l_CmdProc*)tcl_t
4c60: 68 72 65 61 64 5f 61 72 67 63 20 20 20 20 20 20  hread_argc      
4c70: 20 7d 2c 0a 20 20 20 20 20 7b 20 22 74 68 72 65   },.     { "thre
4c80: 61 64 5f 61 72 67 76 22 2c 20 20 20 20 20 20 20  ad_argv",       
4c90: 28 54 63 6c 5f 43 6d 64 50 72 6f 63 2a 29 74 63  (Tcl_CmdProc*)tc
4ca0: 6c 5f 74 68 72 65 61 64 5f 61 72 67 76 20 20 20  l_thread_argv   
4cb0: 20 20 20 20 7d 2c 0a 20 20 20 20 20 7b 20 22 74      },.     { "t
4cc0: 68 72 65 61 64 5f 63 6f 6c 6e 61 6d 65 22 2c 20  hread_colname", 
4cd0: 20 20 20 28 54 63 6c 5f 43 6d 64 50 72 6f 63 2a     (Tcl_CmdProc*
4ce0: 29 74 63 6c 5f 74 68 72 65 61 64 5f 63 6f 6c 6e  )tcl_thread_coln
4cf0: 61 6d 65 20 20 20 20 7d 2c 0a 20 20 20 20 20 7b  ame    },.     {
4d00: 20 22 74 68 72 65 61 64 5f 72 65 73 75 6c 74 22   "thread_result"
4d10: 2c 20 20 20 20 20 28 54 63 6c 5f 43 6d 64 50 72  ,     (Tcl_CmdPr
4d20: 6f 63 2a 29 74 63 6c 5f 74 68 72 65 61 64 5f 72  oc*)tcl_thread_r
4d30: 65 73 75 6c 74 20 20 20 20 20 7d 2c 0a 20 20 20  esult     },.   
4d40: 20 20 7b 20 22 74 68 72 65 61 64 5f 65 72 72 6f    { "thread_erro
4d50: 72 22 2c 20 20 20 20 20 20 28 54 63 6c 5f 43 6d  r",      (Tcl_Cm
4d60: 64 50 72 6f 63 2a 29 74 63 6c 5f 74 68 72 65 61  dProc*)tcl_threa
4d70: 64 5f 65 72 72 6f 72 20 20 20 20 20 20 7d 2c 0a  d_error      },.
4d80: 20 20 20 20 20 7b 20 22 74 68 72 65 61 64 5f 63       { "thread_c
4d90: 6f 6d 70 69 6c 65 22 2c 20 20 20 20 28 54 63 6c  ompile",    (Tcl
4da0: 5f 43 6d 64 50 72 6f 63 2a 29 74 63 6c 5f 74 68  _CmdProc*)tcl_th
4db0: 72 65 61 64 5f 63 6f 6d 70 69 6c 65 20 20 20 20  read_compile    
4dc0: 7d 2c 0a 20 20 20 20 20 7b 20 22 74 68 72 65 61  },.     { "threa
4dd0: 64 5f 73 74 65 70 22 2c 20 20 20 20 20 20 20 28  d_step",       (
4de0: 54 63 6c 5f 43 6d 64 50 72 6f 63 2a 29 74 63 6c  Tcl_CmdProc*)tcl
4df0: 5f 74 68 72 65 61 64 5f 73 74 65 70 20 20 20 20  _thread_step    
4e00: 20 20 20 7d 2c 0a 20 20 20 20 20 7b 20 22 74 68     },.     { "th
4e10: 72 65 61 64 5f 66 69 6e 61 6c 69 7a 65 22 2c 20  read_finalize", 
4e20: 20 20 28 54 63 6c 5f 43 6d 64 50 72 6f 63 2a 29    (Tcl_CmdProc*)
4e30: 74 63 6c 5f 74 68 72 65 61 64 5f 66 69 6e 61 6c  tcl_thread_final
4e40: 69 7a 65 20 20 20 7d 2c 0a 20 20 20 20 20 7b 20  ize   },.     { 
4e50: 22 74 68 72 65 61 64 5f 73 77 61 70 22 2c 20 20  "thread_swap",  
4e60: 20 20 20 20 20 28 54 63 6c 5f 43 6d 64 50 72 6f       (Tcl_CmdPro
4e70: 63 2a 29 74 63 6c 5f 74 68 72 65 61 64 5f 73 77  c*)tcl_thread_sw
4e80: 61 70 20 20 20 20 20 20 20 7d 2c 0a 20 20 20 20  ap       },.    
4e90: 20 7b 20 22 74 68 72 65 61 64 5f 64 62 5f 67 65   { "thread_db_ge
4ea0: 74 22 2c 20 20 20 20 20 28 54 63 6c 5f 43 6d 64  t",     (Tcl_Cmd
4eb0: 50 72 6f 63 2a 29 74 63 6c 5f 74 68 72 65 61 64  Proc*)tcl_thread
4ec0: 5f 64 62 5f 67 65 74 20 20 20 20 20 7d 2c 0a 20  _db_get     },. 
4ed0: 20 20 20 20 7b 20 22 74 68 72 65 61 64 5f 73 74      { "thread_st
4ee0: 6d 74 5f 67 65 74 22 2c 20 20 20 28 54 63 6c 5f  mt_get",   (Tcl_
4ef0: 43 6d 64 50 72 6f 63 2a 29 74 63 6c 5f 74 68 72  CmdProc*)tcl_thr
4f00: 65 61 64 5f 73 74 6d 74 5f 67 65 74 20 20 20 7d  ead_stmt_get   }
4f10: 2c 0a 20 20 7d 3b 0a 20 20 69 6e 74 20 69 3b 0a  ,.  };.  int i;.
4f20: 0a 20 20 66 6f 72 28 69 3d 30 3b 20 69 3c 73 69  .  for(i=0; i<si
4f30: 7a 65 6f 66 28 61 43 6d 64 29 2f 73 69 7a 65 6f  zeof(aCmd)/sizeo
4f40: 66 28 61 43 6d 64 5b 30 5d 29 3b 20 69 2b 2b 29  f(aCmd[0]); i++)
4f50: 7b 0a 20 20 20 20 54 63 6c 5f 43 72 65 61 74 65  {.    Tcl_Create
4f60: 43 6f 6d 6d 61 6e 64 28 69 6e 74 65 72 70 2c 20  Command(interp, 
4f70: 61 43 6d 64 5b 69 5d 2e 7a 4e 61 6d 65 2c 20 61  aCmd[i].zName, a
4f80: 43 6d 64 5b 69 5d 2e 78 50 72 6f 63 2c 20 30 2c  Cmd[i].xProc, 0,
4f90: 20 30 29 3b 0a 20 20 7d 0a 20 20 72 65 74 75 72   0);.  }.  retur
4fa0: 6e 20 54 43 4c 5f 4f 4b 3b 0a 7d 0a 23 65 6c 73  n TCL_OK;.}.#els
4fb0: 65 0a 69 6e 74 20 53 71 6c 69 74 65 74 65 73 74  e.int Sqlitetest
4fc0: 34 5f 49 6e 69 74 28 54 63 6c 5f 49 6e 74 65 72  4_Init(Tcl_Inter
4fd0: 70 20 2a 69 6e 74 65 72 70 29 7b 20 72 65 74 75  p *interp){ retu
4fe0: 72 6e 20 54 43 4c 5f 4f 4b 3b 20 7d 0a 23 65 6e  rn TCL_OK; }.#en
4ff0: 64 69 66 20 2f 2a 20 4f 53 5f 55 4e 49 58 20 2a  dif /* OS_UNIX *
5000: 2f 0a                                            /.