/ Hex Artifact Content
Login

Artifact dcbbbb382626fd466a7c46907f74db35fc8bad64:


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 20 32 30 30 33 2f 31  4.c,v 1.2 2003/1
01e0: 32 2f 32 30 20 30 34 3a 30 30 3a 35 33 20 64 72  2/20 04:00:53 dr
01f0: 68 20 45 78 70 20 24 0a 2a 2f 0a 23 69 6e 63 6c  h Exp $.*/.#incl
0200: 75 64 65 20 22 73 71 6c 69 74 65 49 6e 74 2e 68  ude "sqliteInt.h
0210: 22 0a 23 69 6e 63 6c 75 64 65 20 22 74 63 6c 2e  ".#include "tcl.
0220: 68 22 0a 23 69 66 20 64 65 66 69 6e 65 64 28 4f  h".#if defined(O
0230: 53 5f 55 4e 49 58 29 20 26 26 20 4f 53 5f 55 4e  S_UNIX) && OS_UN
0240: 49 58 3d 3d 31 20 26 26 20 64 65 66 69 6e 65 64  IX==1 && defined
0250: 28 54 48 52 45 41 44 53 41 46 45 29 20 26 26 20  (THREADSAFE) && 
0260: 54 48 52 45 41 44 53 41 46 45 3d 3d 31 0a 23 69  THREADSAFE==1.#i
0270: 6e 63 6c 75 64 65 20 3c 73 74 64 6c 69 62 2e 68  nclude <stdlib.h
0280: 3e 0a 23 69 6e 63 6c 75 64 65 20 3c 73 74 72 69  >.#include <stri
0290: 6e 67 2e 68 3e 0a 23 69 6e 63 6c 75 64 65 20 3c  ng.h>.#include <
02a0: 70 74 68 72 65 61 64 2e 68 3e 0a 23 69 6e 63 6c  pthread.h>.#incl
02b0: 75 64 65 20 3c 73 63 68 65 64 2e 68 3e 0a 23 69  ude <sched.h>.#i
02c0: 6e 63 6c 75 64 65 20 3c 63 74 79 70 65 2e 68 3e  nclude <ctype.h>
02d0: 0a 0a 2f 2a 0a 2a 2a 20 45 61 63 68 20 74 68 72  ../*.** Each thr
02e0: 65 61 64 20 69 73 20 63 6f 6e 74 72 6f 6c 6c 65  ead is controlle
02f0: 64 20 62 79 20 61 6e 20 69 6e 73 74 61 6e 63 65  d by an instance
0300: 20 6f 66 20 74 68 65 20 66 6f 6c 6c 6f 77 69 6e   of the followin
0310: 67 0a 2a 2a 20 73 74 72 75 63 74 75 72 65 2e 0a  g.** structure..
0320: 2a 2f 0a 74 79 70 65 64 65 66 20 73 74 72 75 63  */.typedef struc
0330: 74 20 54 68 72 65 61 64 20 54 68 72 65 61 64 3b  t Thread Thread;
0340: 0a 73 74 72 75 63 74 20 54 68 72 65 61 64 20 7b  .struct Thread {
0350: 0a 20 20 2f 2a 20 54 68 65 20 66 69 72 73 74 20  .  /* The first 
0360: 67 72 6f 75 70 20 6f 66 20 66 69 65 6c 64 73 20  group of fields 
0370: 61 72 65 20 77 72 69 74 61 62 6c 65 20 62 79 20  are writable by 
0380: 74 68 65 20 6d 61 73 74 65 72 20 61 6e 64 20 72  the master and r
0390: 65 61 64 2d 6f 6e 6c 79 0a 20 20 2a 2a 20 74 6f  ead-only.  ** to
03a0: 20 74 68 65 20 74 68 72 65 61 64 2e 20 2a 2f 0a   the thread. */.
03b0: 20 20 63 68 61 72 20 2a 7a 46 69 6c 65 6e 61 6d    char *zFilenam
03c0: 65 3b 20 20 20 20 20 20 20 2f 2a 20 4e 61 6d 65  e;       /* Name
03d0: 20 6f 66 20 64 61 74 61 62 61 73 65 20 66 69 6c   of database fil
03e0: 65 20 2a 2f 0a 20 20 76 6f 69 64 20 28 2a 78 4f  e */.  void (*xO
03f0: 70 29 28 54 68 72 65 61 64 2a 29 3b 20 20 2f 2a  p)(Thread*);  /*
0400: 20 6e 65 78 74 20 6f 70 65 72 61 74 69 6f 6e 20   next operation 
0410: 74 6f 20 64 6f 20 2a 2f 0a 20 20 63 68 61 72 20  to do */.  char 
0420: 2a 7a 41 72 67 3b 20 20 20 20 20 20 20 20 20 20  *zArg;          
0430: 20 20 2f 2a 20 61 72 67 75 6d 65 6e 74 20 75 73    /* argument us
0440: 61 62 6c 65 20 62 79 20 78 4f 70 20 2a 2f 0a 20  able by xOp */. 
0450: 20 69 6e 74 20 6f 70 6e 75 6d 3b 20 20 20 20 20   int opnum;     
0460: 20 20 20 20 20 20 20 20 2f 2a 20 4f 70 65 72 61          /* Opera
0470: 74 69 6f 6e 20 6e 75 6d 62 65 72 20 2a 2f 0a 20  tion number */. 
0480: 20 69 6e 74 20 62 75 73 79 3b 20 20 20 20 20 20   int busy;      
0490: 20 20 20 20 20 20 20 20 2f 2a 20 54 72 75 65 20          /* True 
04a0: 69 66 20 74 68 69 73 20 74 68 72 65 61 64 20 69  if this thread i
04b0: 73 20 69 6e 20 75 73 65 20 2a 2f 0a 0a 20 20 2f  s in use */..  /
04c0: 2a 20 54 68 65 20 6e 65 78 74 20 67 72 6f 75 70  * The next group
04d0: 20 6f 66 20 66 69 65 6c 64 73 20 61 72 65 20 77   of fields are w
04e0: 72 69 74 61 62 6c 65 20 62 79 20 74 68 65 20 74  ritable by the t
04f0: 68 72 65 61 64 20 62 75 74 20 72 65 61 64 2d 6f  hread but read-o
0500: 6e 6c 79 20 74 6f 20 74 68 65 0a 20 20 2a 2a 20  nly to the.  ** 
0510: 6d 61 73 74 65 72 2e 20 2a 2f 0a 20 20 69 6e 74  master. */.  int
0520: 20 63 6f 6d 70 6c 65 74 65 64 3b 20 20 20 20 20   completed;     
0530: 20 20 20 2f 2a 20 4e 75 6d 62 65 72 20 6f 66 20     /* Number of 
0540: 6f 70 65 72 61 74 69 6f 6e 73 20 63 6f 6d 70 6c  operations compl
0550: 65 74 65 64 20 2a 2f 0a 20 20 73 71 6c 69 74 65  eted */.  sqlite
0560: 20 2a 64 62 3b 20 20 20 20 20 20 20 20 20 20 20   *db;           
0570: 2f 2a 20 4f 70 65 6e 20 64 61 74 61 62 61 73 65  /* Open database
0580: 20 2a 2f 0a 20 20 73 71 6c 69 74 65 5f 76 6d 20   */.  sqlite_vm 
0590: 2a 76 6d 3b 20 20 20 20 20 20 20 20 2f 2a 20 50  *vm;        /* P
05a0: 65 6e 64 69 6e 67 20 6f 70 65 72 61 74 69 6f 6e  ending operation
05b0: 20 2a 2f 0a 20 20 63 68 61 72 20 2a 7a 45 72 72   */.  char *zErr
05c0: 3b 20 20 20 20 20 20 20 20 20 20 20 2f 2a 20 6f  ;           /* o
05d0: 70 65 72 61 74 69 6f 6e 20 65 72 72 6f 72 20 2a  peration error *
05e0: 2f 0a 20 20 63 68 61 72 20 2a 7a 53 74 61 74 69  /.  char *zStati
05f0: 63 45 72 72 3b 20 20 20 20 20 2f 2a 20 53 74 61  cErr;     /* Sta
0600: 74 69 63 20 65 72 72 6f 72 20 6d 65 73 73 61 67  tic error messag
0610: 65 20 2a 2f 0a 20 20 69 6e 74 20 72 63 3b 20 20  e */.  int rc;  
0620: 20 20 20 20 20 20 20 20 20 20 20 20 20 2f 2a 20               /* 
0630: 6f 70 65 72 61 74 69 6f 6e 20 72 65 74 75 72 6e  operation return
0640: 20 63 6f 64 65 20 2a 2f 0a 20 20 69 6e 74 20 61   code */.  int a
0650: 72 67 63 3b 20 20 20 20 20 20 20 20 20 20 20 20  rgc;            
0660: 20 2f 2a 20 6e 75 6d 62 65 72 20 6f 66 20 63 6f   /* number of co
0670: 6c 75 6d 6e 73 20 69 6e 20 72 65 73 75 6c 74 20  lumns in result 
0680: 2a 2f 0a 20 20 63 6f 6e 73 74 20 63 68 61 72 20  */.  const char 
0690: 2a 2a 61 72 67 76 3b 20 20 20 20 2f 2a 20 72 65  **argv;    /* re
06a0: 73 75 6c 74 20 63 6f 6c 75 6d 6e 73 20 2a 2f 0a  sult columns */.
06b0: 20 20 63 6f 6e 73 74 20 63 68 61 72 20 2a 2a 63    const char **c
06c0: 6f 6c 76 3b 20 20 20 20 2f 2a 20 72 65 73 75 6c  olv;    /* resul
06d0: 74 20 63 6f 6c 75 6d 6e 20 6e 61 6d 65 73 20 2a  t column names *
06e0: 2f 0a 7d 3b 0a 0a 2f 2a 0a 2a 2a 20 54 68 65 72  /.};../*.** Ther
06f0: 65 20 63 61 6e 20 62 65 20 61 73 20 6d 61 6e 79  e can be as many
0700: 20 61 73 20 32 36 20 74 68 72 65 61 64 73 20 72   as 26 threads r
0710: 75 6e 6e 69 6e 67 20 61 74 20 6f 6e 63 65 2e 20  unning at once. 
0720: 20 45 61 63 68 20 69 73 20 6e 61 6d 65 64 0a 2a   Each is named.*
0730: 2a 20 62 79 20 61 20 63 61 70 69 74 61 6c 20 6c  * by a capital l
0740: 65 74 74 65 72 3a 20 41 2c 20 42 2c 20 43 2c 20  etter: A, B, C, 
0750: 2e 2e 2e 2c 20 59 2c 20 5a 2e 0a 2a 2f 0a 23 64  ..., Y, Z..*/.#d
0760: 65 66 69 6e 65 20 4e 5f 54 48 52 45 41 44 20 32  efine N_THREAD 2
0770: 36 0a 73 74 61 74 69 63 20 54 68 72 65 61 64 20  6.static Thread 
0780: 74 68 72 65 61 64 73 65 74 5b 4e 5f 54 48 52 45  threadset[N_THRE
0790: 41 44 5d 3b 0a 0a 0a 2f 2a 0a 2a 2a 20 54 68 65  AD];.../*.** The
07a0: 20 6d 61 69 6e 20 6c 6f 6f 70 20 66 6f 72 20 61   main loop for a
07b0: 20 74 68 72 65 61 64 2e 20 20 54 68 72 65 61 64   thread.  Thread
07c0: 73 20 75 73 65 20 62 75 73 79 20 77 61 69 74 69  s use busy waiti
07d0: 6e 67 2e 20 0a 2a 2f 0a 73 74 61 74 69 63 20 76  ng. .*/.static v
07e0: 6f 69 64 20 2a 74 68 72 65 61 64 5f 6d 61 69 6e  oid *thread_main
07f0: 28 76 6f 69 64 20 2a 70 41 72 67 29 7b 0a 20 20  (void *pArg){.  
0800: 54 68 72 65 61 64 20 2a 70 20 3d 20 28 54 68 72  Thread *p = (Thr
0810: 65 61 64 2a 29 70 41 72 67 3b 0a 20 20 69 66 28  ead*)pArg;.  if(
0820: 20 70 2d 3e 64 62 20 29 7b 0a 20 20 20 20 73 71   p->db ){.    sq
0830: 6c 69 74 65 5f 63 6c 6f 73 65 28 70 2d 3e 64 62  lite_close(p->db
0840: 29 3b 0a 20 20 7d 0a 20 20 70 2d 3e 64 62 20 3d  );.  }.  p->db =
0850: 20 73 71 6c 69 74 65 5f 6f 70 65 6e 28 70 2d 3e   sqlite_open(p->
0860: 7a 46 69 6c 65 6e 61 6d 65 2c 20 30 2c 20 26 70  zFilename, 0, &p
0870: 2d 3e 7a 45 72 72 29 3b 0a 20 20 70 2d 3e 76 6d  ->zErr);.  p->vm
0880: 20 3d 20 30 3b 0a 20 20 70 2d 3e 63 6f 6d 70 6c   = 0;.  p->compl
0890: 65 74 65 64 20 3d 20 31 3b 0a 20 20 77 68 69 6c  eted = 1;.  whil
08a0: 65 28 20 70 2d 3e 6f 70 6e 75 6d 3c 3d 70 2d 3e  e( p->opnum<=p->
08b0: 63 6f 6d 70 6c 65 74 65 64 20 29 20 73 63 68 65  completed ) sche
08c0: 64 5f 79 69 65 6c 64 28 29 3b 0a 20 20 77 68 69  d_yield();.  whi
08d0: 6c 65 28 20 70 2d 3e 78 4f 70 20 29 7b 0a 20 20  le( p->xOp ){.  
08e0: 20 20 69 66 28 20 70 2d 3e 7a 45 72 72 20 26 26    if( p->zErr &&
08f0: 20 70 2d 3e 7a 45 72 72 21 3d 70 2d 3e 7a 53 74   p->zErr!=p->zSt
0900: 61 74 69 63 45 72 72 20 29 7b 0a 20 20 20 20 20  aticErr ){.     
0910: 20 73 71 6c 69 74 65 5f 66 72 65 65 6d 65 6d 28   sqlite_freemem(
0920: 70 2d 3e 7a 45 72 72 29 3b 0a 20 20 20 20 20 20  p->zErr);.      
0930: 70 2d 3e 7a 45 72 72 20 3d 20 30 3b 0a 20 20 20  p->zErr = 0;.   
0940: 20 7d 0a 20 20 20 20 28 2a 70 2d 3e 78 4f 70 29   }.    (*p->xOp)
0950: 28 70 29 3b 0a 20 20 20 20 70 2d 3e 63 6f 6d 70  (p);.    p->comp
0960: 6c 65 74 65 64 2b 2b 3b 0a 20 20 20 20 77 68 69  leted++;.    whi
0970: 6c 65 28 20 70 2d 3e 6f 70 6e 75 6d 3c 3d 70 2d  le( p->opnum<=p-
0980: 3e 63 6f 6d 70 6c 65 74 65 64 20 29 20 73 63 68  >completed ) sch
0990: 65 64 5f 79 69 65 6c 64 28 29 3b 0a 20 20 7d 0a  ed_yield();.  }.
09a0: 20 20 69 66 28 20 70 2d 3e 76 6d 20 29 7b 0a 20    if( p->vm ){. 
09b0: 20 20 20 73 71 6c 69 74 65 5f 66 69 6e 61 6c 69     sqlite_finali
09c0: 7a 65 28 70 2d 3e 76 6d 2c 20 30 29 3b 0a 20 20  ze(p->vm, 0);.  
09d0: 20 20 70 2d 3e 76 6d 20 3d 20 30 3b 0a 20 20 7d    p->vm = 0;.  }
09e0: 0a 20 20 69 66 28 20 70 2d 3e 64 62 20 29 7b 0a  .  if( p->db ){.
09f0: 20 20 20 20 73 71 6c 69 74 65 5f 63 6c 6f 73 65      sqlite_close
0a00: 28 70 2d 3e 64 62 29 3b 0a 20 20 20 20 70 2d 3e  (p->db);.    p->
0a10: 64 62 20 3d 20 30 3b 0a 20 20 7d 0a 20 20 69 66  db = 0;.  }.  if
0a20: 28 20 70 2d 3e 7a 45 72 72 20 26 26 20 70 2d 3e  ( p->zErr && p->
0a30: 7a 45 72 72 21 3d 70 2d 3e 7a 53 74 61 74 69 63  zErr!=p->zStatic
0a40: 45 72 72 20 29 7b 0a 20 20 20 20 73 71 6c 69 74  Err ){.    sqlit
0a50: 65 5f 66 72 65 65 6d 65 6d 28 70 2d 3e 7a 45 72  e_freemem(p->zEr
0a60: 72 29 3b 0a 20 20 20 20 70 2d 3e 7a 45 72 72 20  r);.    p->zErr 
0a70: 3d 20 30 3b 0a 20 20 7d 0a 20 20 70 2d 3e 63 6f  = 0;.  }.  p->co
0a80: 6d 70 6c 65 74 65 64 2b 2b 3b 0a 20 20 72 65 74  mpleted++;.  ret
0a90: 75 72 6e 20 30 3b 0a 7d 0a 0a 2f 2a 0a 2a 2a 20  urn 0;.}../*.** 
0aa0: 47 65 74 20 61 20 74 68 72 65 61 64 20 49 44 20  Get a thread ID 
0ab0: 77 68 69 63 68 20 69 73 20 61 6e 20 75 70 70 65  which is an uppe
0ac0: 72 20 63 61 73 65 20 6c 65 74 74 65 72 2e 20 20  r case letter.  
0ad0: 52 65 74 75 72 6e 20 74 68 65 20 69 6e 64 65 78  Return the index
0ae0: 2e 0a 2a 2a 20 49 66 20 74 68 65 20 61 72 67 75  ..** If the argu
0af0: 6d 65 6e 74 20 69 73 20 6e 6f 74 20 61 20 76 61  ment is not a va
0b00: 6c 69 64 20 74 68 72 65 61 64 20 49 44 20 70 75  lid thread ID pu
0b10: 74 20 61 6e 20 65 72 72 6f 72 20 6d 65 73 73 61  t an error messa
0b20: 67 65 20 69 6e 0a 2a 2a 20 74 68 65 20 69 6e 74  ge in.** the int
0b30: 65 72 70 72 65 74 65 72 20 61 6e 64 20 72 65 74  erpreter and ret
0b40: 75 72 6e 20 2d 31 2e 0a 2a 2f 0a 73 74 61 74 69  urn -1..*/.stati
0b50: 63 20 69 6e 74 20 70 61 72 73 65 5f 74 68 72 65  c int parse_thre
0b60: 61 64 5f 69 64 28 54 63 6c 5f 49 6e 74 65 72 70  ad_id(Tcl_Interp
0b70: 20 2a 69 6e 74 65 72 70 2c 20 63 6f 6e 73 74 20   *interp, const 
0b80: 63 68 61 72 20 2a 7a 41 72 67 29 7b 0a 20 20 69  char *zArg){.  i
0b90: 66 28 20 7a 41 72 67 3d 3d 30 20 7c 7c 20 7a 41  f( zArg==0 || zA
0ba0: 72 67 5b 30 5d 3d 3d 30 20 7c 7c 20 7a 41 72 67  rg[0]==0 || zArg
0bb0: 5b 31 5d 21 3d 30 20 7c 7c 20 21 69 73 75 70 70  [1]!=0 || !isupp
0bc0: 65 72 28 7a 41 72 67 5b 30 5d 29 20 29 7b 0a 20  er(zArg[0]) ){. 
0bd0: 20 20 20 54 63 6c 5f 41 70 70 65 6e 64 52 65 73     Tcl_AppendRes
0be0: 75 6c 74 28 69 6e 74 65 72 70 2c 20 22 74 68 72  ult(interp, "thr
0bf0: 65 61 64 20 49 44 20 6d 75 73 74 20 62 65 20 61  ead ID must be a
0c00: 6e 20 75 70 70 65 72 20 63 61 73 65 20 6c 65 74  n upper case let
0c10: 74 65 72 22 2c 20 30 29 3b 0a 20 20 20 20 72 65  ter", 0);.    re
0c20: 74 75 72 6e 20 2d 31 3b 0a 20 20 7d 0a 20 20 72  turn -1;.  }.  r
0c30: 65 74 75 72 6e 20 7a 41 72 67 5b 30 5d 20 2d 20  eturn zArg[0] - 
0c40: 27 41 27 3b 0a 7d 0a 0a 2f 2a 0a 2a 2a 20 55 73  'A';.}../*.** Us
0c50: 61 67 65 3a 20 20 20 20 74 68 72 65 61 64 5f 63  age:    thread_c
0c60: 72 65 61 74 65 20 4e 41 4d 45 20 20 46 49 4c 45  reate NAME  FILE
0c70: 4e 41 4d 45 0a 2a 2a 0a 2a 2a 20 4e 41 4d 45 20  NAME.**.** NAME 
0c80: 73 68 6f 75 6c 64 20 62 65 20 61 6e 20 75 70 70  should be an upp
0c90: 65 72 20 63 61 73 65 20 6c 65 74 74 65 72 2e 20  er case letter. 
0ca0: 20 53 74 61 72 74 20 74 68 65 20 74 68 72 65 61   Start the threa
0cb0: 64 20 72 75 6e 6e 69 6e 67 20 77 69 74 68 0a 2a  d running with.*
0cc0: 2a 20 61 6e 20 6f 70 65 6e 20 63 6f 6e 6e 65 63  * an open connec
0cd0: 74 69 6f 6e 20 74 6f 20 74 68 65 20 67 69 76 65  tion to the give
0ce0: 6e 20 64 61 74 61 62 61 73 65 2e 0a 2a 2f 0a 73  n database..*/.s
0cf0: 74 61 74 69 63 20 69 6e 74 20 74 63 6c 5f 74 68  tatic int tcl_th
0d00: 72 65 61 64 5f 63 72 65 61 74 65 28 0a 20 20 76  read_create(.  v
0d10: 6f 69 64 20 2a 4e 6f 74 55 73 65 64 2c 0a 20 20  oid *NotUsed,.  
0d20: 54 63 6c 5f 49 6e 74 65 72 70 20 2a 69 6e 74 65  Tcl_Interp *inte
0d30: 72 70 2c 20 20 20 20 2f 2a 20 54 68 65 20 54 43  rp,    /* The TC
0d40: 4c 20 69 6e 74 65 72 70 72 65 74 65 72 20 74 68  L interpreter th
0d50: 61 74 20 69 6e 76 6f 6b 65 64 20 74 68 69 73 20  at invoked this 
0d60: 63 6f 6d 6d 61 6e 64 20 2a 2f 0a 20 20 69 6e 74  command */.  int
0d70: 20 61 72 67 63 2c 20 20 20 20 20 20 20 20 20 20   argc,          
0d80: 20 20 20 20 2f 2a 20 4e 75 6d 62 65 72 20 6f 66      /* Number of
0d90: 20 61 72 67 75 6d 65 6e 74 73 20 2a 2f 0a 20 20   arguments */.  
0da0: 63 6f 6e 73 74 20 63 68 61 72 20 2a 2a 61 72 67  const char **arg
0db0: 76 20 20 20 20 20 20 2f 2a 20 54 65 78 74 20 6f  v      /* Text o
0dc0: 66 20 65 61 63 68 20 61 72 67 75 6d 65 6e 74 20  f each argument 
0dd0: 2a 2f 0a 29 7b 0a 20 20 69 6e 74 20 69 3b 0a 20  */.){.  int i;. 
0de0: 20 70 74 68 72 65 61 64 5f 74 20 78 3b 0a 20 20   pthread_t x;.  
0df0: 69 6e 74 20 72 63 3b 0a 0a 20 20 69 66 28 20 61  int rc;..  if( a
0e00: 72 67 63 21 3d 33 20 29 7b 0a 20 20 20 20 54 63  rgc!=3 ){.    Tc
0e10: 6c 5f 41 70 70 65 6e 64 52 65 73 75 6c 74 28 69  l_AppendResult(i
0e20: 6e 74 65 72 70 2c 20 22 77 72 6f 6e 67 20 23 20  nterp, "wrong # 
0e30: 61 72 67 73 3a 20 73 68 6f 75 6c 64 20 62 65 20  args: should be 
0e40: 5c 22 22 2c 20 61 72 67 76 5b 30 5d 2c 0a 20 20  \"", argv[0],.  
0e50: 20 20 20 20 20 22 20 49 44 20 46 49 4c 45 4e 41       " ID FILENA
0e60: 4d 45 22 2c 20 30 29 3b 0a 20 20 20 20 72 65 74  ME", 0);.    ret
0e70: 75 72 6e 20 54 43 4c 5f 45 52 52 4f 52 3b 0a 20  urn TCL_ERROR;. 
0e80: 20 7d 0a 20 20 69 20 3d 20 70 61 72 73 65 5f 74   }.  i = parse_t
0e90: 68 72 65 61 64 5f 69 64 28 69 6e 74 65 72 70 2c  hread_id(interp,
0ea0: 20 61 72 67 76 5b 31 5d 29 3b 0a 20 20 69 66 28   argv[1]);.  if(
0eb0: 20 69 3c 30 20 29 20 72 65 74 75 72 6e 20 54 43   i<0 ) return TC
0ec0: 4c 5f 45 52 52 4f 52 3b 0a 20 20 69 66 28 20 74  L_ERROR;.  if( t
0ed0: 68 72 65 61 64 73 65 74 5b 69 5d 2e 62 75 73 79  hreadset[i].busy
0ee0: 20 29 7b 0a 20 20 20 20 54 63 6c 5f 41 70 70 65   ){.    Tcl_Appe
0ef0: 6e 64 52 65 73 75 6c 74 28 69 6e 74 65 72 70 2c  ndResult(interp,
0f00: 20 22 74 68 72 65 61 64 20 22 2c 20 61 72 67 76   "thread ", argv
0f10: 5b 31 5d 2c 20 22 20 69 73 20 61 6c 72 65 61 64  [1], " is alread
0f20: 79 20 72 75 6e 6e 69 6e 67 22 2c 20 30 29 3b 0a  y running", 0);.
0f30: 20 20 20 20 72 65 74 75 72 6e 20 54 43 4c 5f 45      return TCL_E
0f40: 52 52 4f 52 3b 0a 20 20 7d 0a 20 20 74 68 72 65  RROR;.  }.  thre
0f50: 61 64 73 65 74 5b 69 5d 2e 62 75 73 79 20 3d 20  adset[i].busy = 
0f60: 31 3b 0a 20 20 73 71 6c 69 74 65 46 72 65 65 28  1;.  sqliteFree(
0f70: 74 68 72 65 61 64 73 65 74 5b 69 5d 2e 7a 46 69  threadset[i].zFi
0f80: 6c 65 6e 61 6d 65 29 3b 0a 20 20 74 68 72 65 61  lename);.  threa
0f90: 64 73 65 74 5b 69 5d 2e 7a 46 69 6c 65 6e 61 6d  dset[i].zFilenam
0fa0: 65 20 3d 20 73 71 6c 69 74 65 53 74 72 44 75 70  e = sqliteStrDup
0fb0: 28 61 72 67 76 5b 32 5d 29 3b 0a 20 20 74 68 72  (argv[2]);.  thr
0fc0: 65 61 64 73 65 74 5b 69 5d 2e 6f 70 6e 75 6d 20  eadset[i].opnum 
0fd0: 3d 20 31 3b 0a 20 20 74 68 72 65 61 64 73 65 74  = 1;.  threadset
0fe0: 5b 69 5d 2e 63 6f 6d 70 6c 65 74 65 64 20 3d 20  [i].completed = 
0ff0: 30 3b 0a 20 20 72 63 20 3d 20 70 74 68 72 65 61  0;.  rc = pthrea
1000: 64 5f 63 72 65 61 74 65 28 26 78 2c 20 30 2c 20  d_create(&x, 0, 
1010: 74 68 72 65 61 64 5f 6d 61 69 6e 2c 20 26 74 68  thread_main, &th
1020: 72 65 61 64 73 65 74 5b 69 5d 29 3b 0a 20 20 69  readset[i]);.  i
1030: 66 28 20 72 63 20 29 7b 0a 20 20 20 20 54 63 6c  f( rc ){.    Tcl
1040: 5f 41 70 70 65 6e 64 52 65 73 75 6c 74 28 69 6e  _AppendResult(in
1050: 74 65 72 70 2c 20 22 66 61 69 6c 65 64 20 74 6f  terp, "failed to
1060: 20 63 72 65 61 74 65 20 74 68 65 20 74 68 72 65   create the thre
1070: 61 64 22 2c 20 30 29 3b 0a 20 20 20 20 73 71 6c  ad", 0);.    sql
1080: 69 74 65 46 72 65 65 28 74 68 72 65 61 64 73 65  iteFree(threadse
1090: 74 5b 69 5d 2e 7a 46 69 6c 65 6e 61 6d 65 29 3b  t[i].zFilename);
10a0: 0a 20 20 20 20 74 68 72 65 61 64 73 65 74 5b 69  .    threadset[i
10b0: 5d 2e 62 75 73 79 20 3d 20 30 3b 0a 20 20 20 20  ].busy = 0;.    
10c0: 72 65 74 75 72 6e 20 54 43 4c 5f 45 52 52 4f 52  return TCL_ERROR
10d0: 3b 0a 20 20 7d 0a 20 20 70 74 68 72 65 61 64 5f  ;.  }.  pthread_
10e0: 64 65 74 61 63 68 28 78 29 3b 0a 20 20 72 65 74  detach(x);.  ret
10f0: 75 72 6e 20 54 43 4c 5f 4f 4b 3b 0a 7d 0a 0a 2f  urn TCL_OK;.}../
1100: 2a 0a 2a 2a 20 57 61 69 74 20 66 6f 72 20 61 20  *.** Wait for a 
1110: 74 68 72 65 61 64 20 74 6f 20 72 65 61 63 68 20  thread to reach 
1120: 69 74 73 20 69 64 6c 65 20 73 74 61 74 65 2e 0a  its idle state..
1130: 2a 2f 0a 73 74 61 74 69 63 20 76 6f 69 64 20 74  */.static void t
1140: 68 72 65 61 64 5f 77 61 69 74 28 54 68 72 65 61  hread_wait(Threa
1150: 64 20 2a 70 29 7b 0a 20 20 77 68 69 6c 65 28 20  d *p){.  while( 
1160: 70 2d 3e 6f 70 6e 75 6d 3e 70 2d 3e 63 6f 6d 70  p->opnum>p->comp
1170: 6c 65 74 65 64 20 29 20 73 63 68 65 64 5f 79 69  leted ) sched_yi
1180: 65 6c 64 28 29 3b 0a 7d 0a 0a 2f 2a 0a 2a 2a 20  eld();.}../*.** 
1190: 55 73 61 67 65 3a 20 20 74 68 72 65 61 64 5f 77  Usage:  thread_w
11a0: 61 69 74 20 49 44 0a 2a 2a 0a 2a 2a 20 57 61 69  ait ID.**.** Wai
11b0: 74 20 6f 6e 20 74 68 72 65 61 64 20 49 44 20 74  t on thread ID t
11c0: 6f 20 72 65 61 63 68 20 69 74 73 20 69 64 6c 65  o reach its idle
11d0: 20 73 74 61 74 65 2e 0a 2a 2f 0a 73 74 61 74 69   state..*/.stati
11e0: 63 20 69 6e 74 20 74 63 6c 5f 74 68 72 65 61 64  c int tcl_thread
11f0: 5f 77 61 69 74 28 0a 20 20 76 6f 69 64 20 2a 4e  _wait(.  void *N
1200: 6f 74 55 73 65 64 2c 0a 20 20 54 63 6c 5f 49 6e  otUsed,.  Tcl_In
1210: 74 65 72 70 20 2a 69 6e 74 65 72 70 2c 20 20 20  terp *interp,   
1220: 20 2f 2a 20 54 68 65 20 54 43 4c 20 69 6e 74 65   /* The TCL inte
1230: 72 70 72 65 74 65 72 20 74 68 61 74 20 69 6e 76  rpreter that inv
1240: 6f 6b 65 64 20 74 68 69 73 20 63 6f 6d 6d 61 6e  oked this comman
1250: 64 20 2a 2f 0a 20 20 69 6e 74 20 61 72 67 63 2c  d */.  int argc,
1260: 20 20 20 20 20 20 20 20 20 20 20 20 20 20 2f 2a                /*
1270: 20 4e 75 6d 62 65 72 20 6f 66 20 61 72 67 75 6d   Number of argum
1280: 65 6e 74 73 20 2a 2f 0a 20 20 63 6f 6e 73 74 20  ents */.  const 
1290: 63 68 61 72 20 2a 2a 61 72 67 76 20 20 20 20 20  char **argv     
12a0: 20 2f 2a 20 54 65 78 74 20 6f 66 20 65 61 63 68   /* Text of each
12b0: 20 61 72 67 75 6d 65 6e 74 20 2a 2f 0a 29 7b 0a   argument */.){.
12c0: 20 20 69 6e 74 20 69 3b 0a 0a 20 20 69 66 28 20    int i;..  if( 
12d0: 61 72 67 63 21 3d 32 20 29 7b 0a 20 20 20 20 54  argc!=2 ){.    T
12e0: 63 6c 5f 41 70 70 65 6e 64 52 65 73 75 6c 74 28  cl_AppendResult(
12f0: 69 6e 74 65 72 70 2c 20 22 77 72 6f 6e 67 20 23  interp, "wrong #
1300: 20 61 72 67 73 3a 20 73 68 6f 75 6c 64 20 62 65   args: should be
1310: 20 5c 22 22 2c 20 61 72 67 76 5b 30 5d 2c 0a 20   \"", argv[0],. 
1320: 20 20 20 20 20 20 22 20 49 44 22 2c 20 30 29 3b        " ID", 0);
1330: 0a 20 20 20 20 72 65 74 75 72 6e 20 54 43 4c 5f  .    return TCL_
1340: 45 52 52 4f 52 3b 0a 20 20 7d 0a 20 20 69 20 3d  ERROR;.  }.  i =
1350: 20 70 61 72 73 65 5f 74 68 72 65 61 64 5f 69 64   parse_thread_id
1360: 28 69 6e 74 65 72 70 2c 20 61 72 67 76 5b 31 5d  (interp, argv[1]
1370: 29 3b 0a 20 20 69 66 28 20 69 3c 30 20 29 20 72  );.  if( i<0 ) r
1380: 65 74 75 72 6e 20 54 43 4c 5f 45 52 52 4f 52 3b  eturn TCL_ERROR;
1390: 0a 20 20 69 66 28 20 21 74 68 72 65 61 64 73 65  .  if( !threadse
13a0: 74 5b 69 5d 2e 62 75 73 79 20 29 7b 0a 20 20 20  t[i].busy ){.   
13b0: 20 54 63 6c 5f 41 70 70 65 6e 64 52 65 73 75 6c   Tcl_AppendResul
13c0: 74 28 69 6e 74 65 72 70 2c 20 22 6e 6f 20 73 75  t(interp, "no su
13d0: 63 68 20 74 68 72 65 61 64 22 2c 20 30 29 3b 0a  ch thread", 0);.
13e0: 20 20 20 20 72 65 74 75 72 6e 20 54 43 4c 5f 45      return TCL_E
13f0: 52 52 4f 52 3b 0a 20 20 7d 0a 20 20 74 68 72 65  RROR;.  }.  thre
1400: 61 64 5f 77 61 69 74 28 26 74 68 72 65 61 64 73  ad_wait(&threads
1410: 65 74 5b 69 5d 29 3b 0a 20 20 72 65 74 75 72 6e  et[i]);.  return
1420: 20 54 43 4c 5f 4f 4b 3b 0a 7d 0a 0a 2f 2a 0a 2a   TCL_OK;.}../*.*
1430: 2a 20 53 74 6f 70 20 61 20 74 68 72 65 61 64 2e  * Stop a thread.
1440: 0a 2a 2f 0a 73 74 61 74 69 63 20 76 6f 69 64 20  .*/.static void 
1450: 73 74 6f 70 5f 74 68 72 65 61 64 28 54 68 72 65  stop_thread(Thre
1460: 61 64 20 2a 70 29 7b 0a 20 20 74 68 72 65 61 64  ad *p){.  thread
1470: 5f 77 61 69 74 28 70 29 3b 0a 20 20 70 2d 3e 78  _wait(p);.  p->x
1480: 4f 70 20 3d 20 30 3b 0a 20 20 70 2d 3e 6f 70 6e  Op = 0;.  p->opn
1490: 75 6d 2b 2b 3b 0a 20 20 74 68 72 65 61 64 5f 77  um++;.  thread_w
14a0: 61 69 74 28 70 29 3b 0a 20 20 73 71 6c 69 74 65  ait(p);.  sqlite
14b0: 46 72 65 65 28 70 2d 3e 7a 41 72 67 29 3b 0a 20  Free(p->zArg);. 
14c0: 20 70 2d 3e 7a 41 72 67 20 3d 20 30 3b 0a 20 20   p->zArg = 0;.  
14d0: 73 71 6c 69 74 65 46 72 65 65 28 70 2d 3e 7a 46  sqliteFree(p->zF
14e0: 69 6c 65 6e 61 6d 65 29 3b 0a 20 20 70 2d 3e 7a  ilename);.  p->z
14f0: 46 69 6c 65 6e 61 6d 65 20 3d 20 30 3b 0a 20 20  Filename = 0;.  
1500: 70 2d 3e 62 75 73 79 20 3d 20 30 3b 0a 7d 0a 0a  p->busy = 0;.}..
1510: 2f 2a 0a 2a 2a 20 55 73 61 67 65 3a 20 20 74 68  /*.** Usage:  th
1520: 72 65 61 64 5f 68 61 6c 74 20 49 44 0a 2a 2a 0a  read_halt ID.**.
1530: 2a 2a 20 43 61 75 73 65 20 61 20 74 68 72 65 61  ** Cause a threa
1540: 64 20 74 6f 20 73 68 75 74 20 69 74 73 65 6c 66  d to shut itself
1550: 20 64 6f 77 6e 2e 20 20 57 61 69 74 20 66 6f 72   down.  Wait for
1560: 20 74 68 65 20 73 68 75 74 64 6f 77 6e 20 74 6f   the shutdown to
1570: 20 62 65 0a 2a 2a 20 63 6f 6d 70 6c 65 74 65 64   be.** completed
1580: 2e 20 20 49 66 20 49 44 20 69 73 20 22 2a 22 20  .  If ID is "*" 
1590: 74 68 65 6e 20 73 74 6f 70 20 61 6c 6c 20 74 68  then stop all th
15a0: 72 65 61 64 73 2e 0a 2a 2f 0a 73 74 61 74 69 63  reads..*/.static
15b0: 20 69 6e 74 20 74 63 6c 5f 74 68 72 65 61 64 5f   int tcl_thread_
15c0: 68 61 6c 74 28 0a 20 20 76 6f 69 64 20 2a 4e 6f  halt(.  void *No
15d0: 74 55 73 65 64 2c 0a 20 20 54 63 6c 5f 49 6e 74  tUsed,.  Tcl_Int
15e0: 65 72 70 20 2a 69 6e 74 65 72 70 2c 20 20 20 20  erp *interp,    
15f0: 2f 2a 20 54 68 65 20 54 43 4c 20 69 6e 74 65 72  /* The TCL inter
1600: 70 72 65 74 65 72 20 74 68 61 74 20 69 6e 76 6f  preter that invo
1610: 6b 65 64 20 74 68 69 73 20 63 6f 6d 6d 61 6e 64  ked this command
1620: 20 2a 2f 0a 20 20 69 6e 74 20 61 72 67 63 2c 20   */.  int argc, 
1630: 20 20 20 20 20 20 20 20 20 20 20 20 20 2f 2a 20               /* 
1640: 4e 75 6d 62 65 72 20 6f 66 20 61 72 67 75 6d 65  Number of argume
1650: 6e 74 73 20 2a 2f 0a 20 20 63 6f 6e 73 74 20 63  nts */.  const c
1660: 68 61 72 20 2a 2a 61 72 67 76 20 20 20 20 20 20  har **argv      
1670: 2f 2a 20 54 65 78 74 20 6f 66 20 65 61 63 68 20  /* Text of each 
1680: 61 72 67 75 6d 65 6e 74 20 2a 2f 0a 29 7b 0a 20  argument */.){. 
1690: 20 69 6e 74 20 69 3b 0a 0a 20 20 69 66 28 20 61   int i;..  if( a
16a0: 72 67 63 21 3d 32 20 29 7b 0a 20 20 20 20 54 63  rgc!=2 ){.    Tc
16b0: 6c 5f 41 70 70 65 6e 64 52 65 73 75 6c 74 28 69  l_AppendResult(i
16c0: 6e 74 65 72 70 2c 20 22 77 72 6f 6e 67 20 23 20  nterp, "wrong # 
16d0: 61 72 67 73 3a 20 73 68 6f 75 6c 64 20 62 65 20  args: should be 
16e0: 5c 22 22 2c 20 61 72 67 76 5b 30 5d 2c 0a 20 20  \"", argv[0],.  
16f0: 20 20 20 20 20 22 20 49 44 22 2c 20 30 29 3b 0a       " ID", 0);.
1700: 20 20 20 20 72 65 74 75 72 6e 20 54 43 4c 5f 45      return TCL_E
1710: 52 52 4f 52 3b 0a 20 20 7d 0a 20 20 69 66 28 20  RROR;.  }.  if( 
1720: 61 72 67 76 5b 31 5d 5b 30 5d 3d 3d 27 2a 27 20  argv[1][0]=='*' 
1730: 26 26 20 61 72 67 76 5b 31 5d 5b 31 5d 3d 3d 30  && argv[1][1]==0
1740: 20 29 7b 0a 20 20 20 20 66 6f 72 28 69 3d 30 3b   ){.    for(i=0;
1750: 20 69 3c 4e 5f 54 48 52 45 41 44 3b 20 69 2b 2b   i<N_THREAD; i++
1760: 29 7b 0a 20 20 20 20 20 20 69 66 28 20 74 68 72  ){.      if( thr
1770: 65 61 64 73 65 74 5b 69 5d 2e 62 75 73 79 20 29  eadset[i].busy )
1780: 20 73 74 6f 70 5f 74 68 72 65 61 64 28 26 74 68   stop_thread(&th
1790: 72 65 61 64 73 65 74 5b 69 5d 29 3b 0a 20 20 20  readset[i]);.   
17a0: 20 7d 0a 20 20 7d 65 6c 73 65 7b 0a 20 20 20 20   }.  }else{.    
17b0: 69 20 3d 20 70 61 72 73 65 5f 74 68 72 65 61 64  i = parse_thread
17c0: 5f 69 64 28 69 6e 74 65 72 70 2c 20 61 72 67 76  _id(interp, argv
17d0: 5b 31 5d 29 3b 0a 20 20 20 20 69 66 28 20 69 3c  [1]);.    if( i<
17e0: 30 20 29 20 72 65 74 75 72 6e 20 54 43 4c 5f 45  0 ) return TCL_E
17f0: 52 52 4f 52 3b 0a 20 20 20 20 69 66 28 20 21 74  RROR;.    if( !t
1800: 68 72 65 61 64 73 65 74 5b 69 5d 2e 62 75 73 79  hreadset[i].busy
1810: 20 29 7b 0a 20 20 20 20 20 20 54 63 6c 5f 41 70   ){.      Tcl_Ap
1820: 70 65 6e 64 52 65 73 75 6c 74 28 69 6e 74 65 72  pendResult(inter
1830: 70 2c 20 22 6e 6f 20 73 75 63 68 20 74 68 72 65  p, "no such thre
1840: 61 64 22 2c 20 30 29 3b 0a 20 20 20 20 20 20 72  ad", 0);.      r
1850: 65 74 75 72 6e 20 54 43 4c 5f 45 52 52 4f 52 3b  eturn TCL_ERROR;
1860: 0a 20 20 20 20 7d 0a 20 20 20 20 73 74 6f 70 5f  .    }.    stop_
1870: 74 68 72 65 61 64 28 26 74 68 72 65 61 64 73 65  thread(&threadse
1880: 74 5b 69 5d 29 3b 0a 20 20 7d 0a 20 20 72 65 74  t[i]);.  }.  ret
1890: 75 72 6e 20 54 43 4c 5f 4f 4b 3b 0a 7d 0a 0a 2f  urn TCL_OK;.}../
18a0: 2a 0a 2a 2a 20 55 73 61 67 65 3a 20 74 68 72 65  *.** Usage: thre
18b0: 61 64 5f 61 72 67 63 20 20 49 44 0a 2a 2a 0a 2a  ad_argc  ID.**.*
18c0: 2a 20 57 61 69 74 20 6f 6e 20 74 68 65 20 6d 6f  * Wait on the mo
18d0: 73 74 20 72 65 63 65 6e 74 20 74 68 72 65 61 64  st recent thread
18e0: 5f 73 74 65 70 20 74 6f 20 63 6f 6d 70 6c 65 74  _step to complet
18f0: 65 2c 20 74 68 65 6e 20 72 65 74 75 72 6e 20 74  e, then return t
1900: 68 65 0a 2a 2a 20 6e 75 6d 62 65 72 20 6f 66 20  he.** number of 
1910: 63 6f 6c 75 6d 6e 73 20 69 6e 20 74 68 65 20 72  columns in the r
1920: 65 73 75 6c 74 20 73 65 74 2e 0a 2a 2f 0a 73 74  esult set..*/.st
1930: 61 74 69 63 20 69 6e 74 20 74 63 6c 5f 74 68 72  atic int tcl_thr
1940: 65 61 64 5f 61 72 67 63 28 0a 20 20 76 6f 69 64  ead_argc(.  void
1950: 20 2a 4e 6f 74 55 73 65 64 2c 0a 20 20 54 63 6c   *NotUsed,.  Tcl
1960: 5f 49 6e 74 65 72 70 20 2a 69 6e 74 65 72 70 2c  _Interp *interp,
1970: 20 20 20 20 2f 2a 20 54 68 65 20 54 43 4c 20 69      /* The TCL i
1980: 6e 74 65 72 70 72 65 74 65 72 20 74 68 61 74 20  nterpreter that 
1990: 69 6e 76 6f 6b 65 64 20 74 68 69 73 20 63 6f 6d  invoked this com
19a0: 6d 61 6e 64 20 2a 2f 0a 20 20 69 6e 74 20 61 72  mand */.  int ar
19b0: 67 63 2c 20 20 20 20 20 20 20 20 20 20 20 20 20  gc,             
19c0: 20 2f 2a 20 4e 75 6d 62 65 72 20 6f 66 20 61 72   /* Number of ar
19d0: 67 75 6d 65 6e 74 73 20 2a 2f 0a 20 20 63 6f 6e  guments */.  con
19e0: 73 74 20 63 68 61 72 20 2a 2a 61 72 67 76 20 20  st char **argv  
19f0: 20 20 20 20 2f 2a 20 54 65 78 74 20 6f 66 20 65      /* Text of e
1a00: 61 63 68 20 61 72 67 75 6d 65 6e 74 20 2a 2f 0a  ach argument */.
1a10: 29 7b 0a 20 20 69 6e 74 20 69 3b 0a 20 20 63 68  ){.  int i;.  ch
1a20: 61 72 20 7a 42 75 66 5b 31 30 30 5d 3b 0a 0a 20  ar zBuf[100];.. 
1a30: 20 69 66 28 20 61 72 67 63 21 3d 32 20 29 7b 0a   if( argc!=2 ){.
1a40: 20 20 20 20 54 63 6c 5f 41 70 70 65 6e 64 52 65      Tcl_AppendRe
1a50: 73 75 6c 74 28 69 6e 74 65 72 70 2c 20 22 77 72  sult(interp, "wr
1a60: 6f 6e 67 20 23 20 61 72 67 73 3a 20 73 68 6f 75  ong # args: shou
1a70: 6c 64 20 62 65 20 5c 22 22 2c 20 61 72 67 76 5b  ld be \"", argv[
1a80: 30 5d 2c 0a 20 20 20 20 20 20 20 22 20 49 44 22  0],.       " ID"
1a90: 2c 20 30 29 3b 0a 20 20 20 20 72 65 74 75 72 6e  , 0);.    return
1aa0: 20 54 43 4c 5f 45 52 52 4f 52 3b 0a 20 20 7d 0a   TCL_ERROR;.  }.
1ab0: 20 20 69 20 3d 20 70 61 72 73 65 5f 74 68 72 65    i = parse_thre
1ac0: 61 64 5f 69 64 28 69 6e 74 65 72 70 2c 20 61 72  ad_id(interp, ar
1ad0: 67 76 5b 31 5d 29 3b 0a 20 20 69 66 28 20 69 3c  gv[1]);.  if( i<
1ae0: 30 20 29 20 72 65 74 75 72 6e 20 54 43 4c 5f 45  0 ) return TCL_E
1af0: 52 52 4f 52 3b 0a 20 20 69 66 28 20 21 74 68 72  RROR;.  if( !thr
1b00: 65 61 64 73 65 74 5b 69 5d 2e 62 75 73 79 20 29  eadset[i].busy )
1b10: 7b 0a 20 20 20 20 54 63 6c 5f 41 70 70 65 6e 64  {.    Tcl_Append
1b20: 52 65 73 75 6c 74 28 69 6e 74 65 72 70 2c 20 22  Result(interp, "
1b30: 6e 6f 20 73 75 63 68 20 74 68 72 65 61 64 22 2c  no such thread",
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 74 68 72 65 61 64 5f 77 61 69 74 28 26 74 68   thread_wait(&th
1b70: 72 65 61 64 73 65 74 5b 69 5d 29 3b 0a 20 20 73  readset[i]);.  s
1b80: 70 72 69 6e 74 66 28 7a 42 75 66 2c 20 22 25 64  printf(zBuf, "%d
1b90: 22 2c 20 74 68 72 65 61 64 73 65 74 5b 69 5d 2e  ", threadset[i].
1ba0: 61 72 67 63 29 3b 0a 20 20 54 63 6c 5f 41 70 70  argc);.  Tcl_App
1bb0: 65 6e 64 52 65 73 75 6c 74 28 69 6e 74 65 72 70  endResult(interp
1bc0: 2c 20 7a 42 75 66 2c 20 30 29 3b 0a 20 20 72 65  , zBuf, 0);.  re
1bd0: 74 75 72 6e 20 54 43 4c 5f 4f 4b 3b 0a 7d 0a 0a  turn TCL_OK;.}..
1be0: 2f 2a 0a 2a 2a 20 55 73 61 67 65 3a 20 74 68 72  /*.** Usage: thr
1bf0: 65 61 64 5f 61 72 67 76 20 20 49 44 20 20 20 4e  ead_argv  ID   N
1c00: 0a 2a 2a 0a 2a 2a 20 57 61 69 74 20 6f 6e 20 74  .**.** Wait on t
1c10: 68 65 20 6d 6f 73 74 20 72 65 63 65 6e 74 20 74  he most recent t
1c20: 68 72 65 61 64 5f 73 74 65 70 20 74 6f 20 63 6f  hread_step to co
1c30: 6d 70 6c 65 74 65 2c 20 74 68 65 6e 20 72 65 74  mplete, then ret
1c40: 75 72 6e 20 74 68 65 0a 2a 2a 20 76 61 6c 75 65  urn the.** value
1c50: 20 6f 66 20 74 68 65 20 4e 2d 74 68 20 63 6f 6c   of the N-th col
1c60: 75 6d 6e 73 20 69 6e 20 74 68 65 20 72 65 73 75  umns in the resu
1c70: 6c 74 20 73 65 74 2e 0a 2a 2f 0a 73 74 61 74 69  lt set..*/.stati
1c80: 63 20 69 6e 74 20 74 63 6c 5f 74 68 72 65 61 64  c int tcl_thread
1c90: 5f 61 72 67 76 28 0a 20 20 76 6f 69 64 20 2a 4e  _argv(.  void *N
1ca0: 6f 74 55 73 65 64 2c 0a 20 20 54 63 6c 5f 49 6e  otUsed,.  Tcl_In
1cb0: 74 65 72 70 20 2a 69 6e 74 65 72 70 2c 20 20 20  terp *interp,   
1cc0: 20 2f 2a 20 54 68 65 20 54 43 4c 20 69 6e 74 65   /* The TCL inte
1cd0: 72 70 72 65 74 65 72 20 74 68 61 74 20 69 6e 76  rpreter that inv
1ce0: 6f 6b 65 64 20 74 68 69 73 20 63 6f 6d 6d 61 6e  oked this comman
1cf0: 64 20 2a 2f 0a 20 20 69 6e 74 20 61 72 67 63 2c  d */.  int argc,
1d00: 20 20 20 20 20 20 20 20 20 20 20 20 20 20 2f 2a                /*
1d10: 20 4e 75 6d 62 65 72 20 6f 66 20 61 72 67 75 6d   Number of argum
1d20: 65 6e 74 73 20 2a 2f 0a 20 20 63 6f 6e 73 74 20  ents */.  const 
1d30: 63 68 61 72 20 2a 2a 61 72 67 76 20 20 20 20 20  char **argv     
1d40: 20 2f 2a 20 54 65 78 74 20 6f 66 20 65 61 63 68   /* Text of each
1d50: 20 61 72 67 75 6d 65 6e 74 20 2a 2f 0a 29 7b 0a   argument */.){.
1d60: 20 20 69 6e 74 20 69 3b 0a 20 20 69 6e 74 20 6e    int i;.  int n
1d70: 3b 0a 0a 20 20 69 66 28 20 61 72 67 63 21 3d 33  ;..  if( argc!=3
1d80: 20 29 7b 0a 20 20 20 20 54 63 6c 5f 41 70 70 65   ){.    Tcl_Appe
1d90: 6e 64 52 65 73 75 6c 74 28 69 6e 74 65 72 70 2c  ndResult(interp,
1da0: 20 22 77 72 6f 6e 67 20 23 20 61 72 67 73 3a 20   "wrong # args: 
1db0: 73 68 6f 75 6c 64 20 62 65 20 5c 22 22 2c 20 61  should be \"", a
1dc0: 72 67 76 5b 30 5d 2c 0a 20 20 20 20 20 20 20 22  rgv[0],.       "
1dd0: 20 49 44 20 4e 22 2c 20 30 29 3b 0a 20 20 20 20   ID N", 0);.    
1de0: 72 65 74 75 72 6e 20 54 43 4c 5f 45 52 52 4f 52  return TCL_ERROR
1df0: 3b 0a 20 20 7d 0a 20 20 69 20 3d 20 70 61 72 73  ;.  }.  i = pars
1e00: 65 5f 74 68 72 65 61 64 5f 69 64 28 69 6e 74 65  e_thread_id(inte
1e10: 72 70 2c 20 61 72 67 76 5b 31 5d 29 3b 0a 20 20  rp, argv[1]);.  
1e20: 69 66 28 20 69 3c 30 20 29 20 72 65 74 75 72 6e  if( i<0 ) return
1e30: 20 54 43 4c 5f 45 52 52 4f 52 3b 0a 20 20 69 66   TCL_ERROR;.  if
1e40: 28 20 21 74 68 72 65 61 64 73 65 74 5b 69 5d 2e  ( !threadset[i].
1e50: 62 75 73 79 20 29 7b 0a 20 20 20 20 54 63 6c 5f  busy ){.    Tcl_
1e60: 41 70 70 65 6e 64 52 65 73 75 6c 74 28 69 6e 74  AppendResult(int
1e70: 65 72 70 2c 20 22 6e 6f 20 73 75 63 68 20 74 68  erp, "no such th
1e80: 72 65 61 64 22 2c 20 30 29 3b 0a 20 20 20 20 72  read", 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 66 28 20 54 63 6c 5f 47  .  }.  if( Tcl_G
1eb0: 65 74 49 6e 74 28 69 6e 74 65 72 70 2c 20 61 72  etInt(interp, ar
1ec0: 67 76 5b 32 5d 2c 20 26 6e 29 20 29 20 72 65 74  gv[2], &n) ) ret
1ed0: 75 72 6e 20 54 43 4c 5f 45 52 52 4f 52 3b 0a 20  urn TCL_ERROR;. 
1ee0: 20 74 68 72 65 61 64 5f 77 61 69 74 28 26 74 68   thread_wait(&th
1ef0: 72 65 61 64 73 65 74 5b 69 5d 29 3b 0a 20 20 69  readset[i]);.  i
1f00: 66 28 20 6e 3c 30 20 7c 7c 20 6e 3e 3d 74 68 72  f( n<0 || n>=thr
1f10: 65 61 64 73 65 74 5b 69 5d 2e 61 72 67 63 20 29  eadset[i].argc )
1f20: 7b 0a 20 20 20 20 54 63 6c 5f 41 70 70 65 6e 64  {.    Tcl_Append
1f30: 52 65 73 75 6c 74 28 69 6e 74 65 72 70 2c 20 22  Result(interp, "
1f40: 63 6f 6c 75 6d 6e 20 6e 75 6d 62 65 72 20 6f 75  column number ou
1f50: 74 20 6f 66 20 72 61 6e 67 65 22 2c 20 30 29 3b  t of range", 0);
1f60: 0a 20 20 20 20 72 65 74 75 72 6e 20 54 43 4c 5f  .    return TCL_
1f70: 45 52 52 4f 52 3b 0a 20 20 7d 0a 20 20 54 63 6c  ERROR;.  }.  Tcl
1f80: 5f 41 70 70 65 6e 64 52 65 73 75 6c 74 28 69 6e  _AppendResult(in
1f90: 74 65 72 70 2c 20 74 68 72 65 61 64 73 65 74 5b  terp, threadset[
1fa0: 69 5d 2e 61 72 67 76 5b 6e 5d 2c 20 30 29 3b 0a  i].argv[n], 0);.
1fb0: 20 20 72 65 74 75 72 6e 20 54 43 4c 5f 4f 4b 3b    return TCL_OK;
1fc0: 0a 7d 0a 0a 2f 2a 0a 2a 2a 20 55 73 61 67 65 3a  .}../*.** Usage:
1fd0: 20 74 68 72 65 61 64 5f 63 6f 6c 6e 61 6d 65 20   thread_colname 
1fe0: 20 49 44 20 20 20 4e 0a 2a 2a 0a 2a 2a 20 57 61   ID   N.**.** Wa
1ff0: 69 74 20 6f 6e 20 74 68 65 20 6d 6f 73 74 20 72  it on the most r
2000: 65 63 65 6e 74 20 74 68 72 65 61 64 5f 73 74 65  ecent thread_ste
2010: 70 20 74 6f 20 63 6f 6d 70 6c 65 74 65 2c 20 74  p to complete, t
2020: 68 65 6e 20 72 65 74 75 72 6e 20 74 68 65 0a 2a  hen return the.*
2030: 2a 20 6e 61 6d 65 20 6f 66 20 74 68 65 20 4e 2d  * name of the N-
2040: 74 68 20 63 6f 6c 75 6d 6e 73 20 69 6e 20 74 68  th columns in th
2050: 65 20 72 65 73 75 6c 74 20 73 65 74 2e 0a 2a 2f  e result set..*/
2060: 0a 73 74 61 74 69 63 20 69 6e 74 20 74 63 6c 5f  .static int tcl_
2070: 74 68 72 65 61 64 5f 63 6f 6c 6e 61 6d 65 28 0a  thread_colname(.
2080: 20 20 76 6f 69 64 20 2a 4e 6f 74 55 73 65 64 2c    void *NotUsed,
2090: 0a 20 20 54 63 6c 5f 49 6e 74 65 72 70 20 2a 69  .  Tcl_Interp *i
20a0: 6e 74 65 72 70 2c 20 20 20 20 2f 2a 20 54 68 65  nterp,    /* The
20b0: 20 54 43 4c 20 69 6e 74 65 72 70 72 65 74 65 72   TCL interpreter
20c0: 20 74 68 61 74 20 69 6e 76 6f 6b 65 64 20 74 68   that invoked th
20d0: 69 73 20 63 6f 6d 6d 61 6e 64 20 2a 2f 0a 20 20  is command */.  
20e0: 69 6e 74 20 61 72 67 63 2c 20 20 20 20 20 20 20  int argc,       
20f0: 20 20 20 20 20 20 20 2f 2a 20 4e 75 6d 62 65 72         /* Number
2100: 20 6f 66 20 61 72 67 75 6d 65 6e 74 73 20 2a 2f   of arguments */
2110: 0a 20 20 63 6f 6e 73 74 20 63 68 61 72 20 2a 2a  .  const char **
2120: 61 72 67 76 20 20 20 20 20 20 2f 2a 20 54 65 78  argv      /* Tex
2130: 74 20 6f 66 20 65 61 63 68 20 61 72 67 75 6d 65  t of each argume
2140: 6e 74 20 2a 2f 0a 29 7b 0a 20 20 69 6e 74 20 69  nt */.){.  int i
2150: 3b 0a 20 20 69 6e 74 20 6e 3b 0a 0a 20 20 69 66  ;.  int n;..  if
2160: 28 20 61 72 67 63 21 3d 33 20 29 7b 0a 20 20 20  ( argc!=3 ){.   
2170: 20 54 63 6c 5f 41 70 70 65 6e 64 52 65 73 75 6c   Tcl_AppendResul
2180: 74 28 69 6e 74 65 72 70 2c 20 22 77 72 6f 6e 67  t(interp, "wrong
2190: 20 23 20 61 72 67 73 3a 20 73 68 6f 75 6c 64 20   # args: should 
21a0: 62 65 20 5c 22 22 2c 20 61 72 67 76 5b 30 5d 2c  be \"", argv[0],
21b0: 0a 20 20 20 20 20 20 20 22 20 49 44 20 4e 22 2c  .       " ID N",
21c0: 20 30 29 3b 0a 20 20 20 20 72 65 74 75 72 6e 20   0);.    return 
21d0: 54 43 4c 5f 45 52 52 4f 52 3b 0a 20 20 7d 0a 20  TCL_ERROR;.  }. 
21e0: 20 69 20 3d 20 70 61 72 73 65 5f 74 68 72 65 61   i = parse_threa
21f0: 64 5f 69 64 28 69 6e 74 65 72 70 2c 20 61 72 67  d_id(interp, arg
2200: 76 5b 31 5d 29 3b 0a 20 20 69 66 28 20 69 3c 30  v[1]);.  if( i<0
2210: 20 29 20 72 65 74 75 72 6e 20 54 43 4c 5f 45 52   ) return TCL_ER
2220: 52 4f 52 3b 0a 20 20 69 66 28 20 21 74 68 72 65  ROR;.  if( !thre
2230: 61 64 73 65 74 5b 69 5d 2e 62 75 73 79 20 29 7b  adset[i].busy ){
2240: 0a 20 20 20 20 54 63 6c 5f 41 70 70 65 6e 64 52  .    Tcl_AppendR
2250: 65 73 75 6c 74 28 69 6e 74 65 72 70 2c 20 22 6e  esult(interp, "n
2260: 6f 20 73 75 63 68 20 74 68 72 65 61 64 22 2c 20  o such thread", 
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 66 28 20 54 63 6c 5f 47 65 74 49 6e 74 28 69  if( Tcl_GetInt(i
22a0: 6e 74 65 72 70 2c 20 61 72 67 76 5b 32 5d 2c 20  nterp, argv[2], 
22b0: 26 6e 29 20 29 20 72 65 74 75 72 6e 20 54 43 4c  &n) ) return TCL
22c0: 5f 45 52 52 4f 52 3b 0a 20 20 74 68 72 65 61 64  _ERROR;.  thread
22d0: 5f 77 61 69 74 28 26 74 68 72 65 61 64 73 65 74  _wait(&threadset
22e0: 5b 69 5d 29 3b 0a 20 20 69 66 28 20 6e 3c 30 20  [i]);.  if( n<0 
22f0: 7c 7c 20 6e 3e 3d 74 68 72 65 61 64 73 65 74 5b  || n>=threadset[
2300: 69 5d 2e 61 72 67 63 20 29 7b 0a 20 20 20 20 54  i].argc ){.    T
2310: 63 6c 5f 41 70 70 65 6e 64 52 65 73 75 6c 74 28  cl_AppendResult(
2320: 69 6e 74 65 72 70 2c 20 22 63 6f 6c 75 6d 6e 20  interp, "column 
2330: 6e 75 6d 62 65 72 20 6f 75 74 20 6f 66 20 72 61  number out of ra
2340: 6e 67 65 22 2c 20 30 29 3b 0a 20 20 20 20 72 65  nge", 0);.    re
2350: 74 75 72 6e 20 54 43 4c 5f 45 52 52 4f 52 3b 0a  turn TCL_ERROR;.
2360: 20 20 7d 0a 20 20 54 63 6c 5f 41 70 70 65 6e 64    }.  Tcl_Append
2370: 52 65 73 75 6c 74 28 69 6e 74 65 72 70 2c 20 74  Result(interp, t
2380: 68 72 65 61 64 73 65 74 5b 69 5d 2e 63 6f 6c 76  hreadset[i].colv
2390: 5b 6e 5d 2c 20 30 29 3b 0a 20 20 72 65 74 75 72  [n], 0);.  retur
23a0: 6e 20 54 43 4c 5f 4f 4b 3b 0a 7d 0a 0a 2f 2a 0a  n TCL_OK;.}../*.
23b0: 2a 2a 20 55 73 61 67 65 3a 20 74 68 72 65 61 64  ** Usage: thread
23c0: 5f 72 65 73 75 6c 74 20 20 49 44 0a 2a 2a 0a 2a  _result  ID.**.*
23d0: 2a 20 57 61 69 74 20 6f 6e 20 74 68 65 20 6d 6f  * Wait on the mo
23e0: 73 74 20 72 65 63 65 6e 74 20 6f 70 65 72 61 74  st recent operat
23f0: 69 6f 6e 20 74 6f 20 63 6f 6d 70 6c 65 74 65 2c  ion to complete,
2400: 20 74 68 65 6e 20 72 65 74 75 72 6e 20 74 68 65   then return the
2410: 0a 2a 2a 20 72 65 73 75 6c 74 20 63 6f 64 65 20  .** result code 
2420: 66 72 6f 6d 20 74 68 61 74 20 6f 70 65 72 61 74  from that operat
2430: 69 6f 6e 2e 0a 2a 2f 0a 73 74 61 74 69 63 20 69  ion..*/.static i
2440: 6e 74 20 74 63 6c 5f 74 68 72 65 61 64 5f 72 65  nt tcl_thread_re
2450: 73 75 6c 74 28 0a 20 20 76 6f 69 64 20 2a 4e 6f  sult(.  void *No
2460: 74 55 73 65 64 2c 0a 20 20 54 63 6c 5f 49 6e 74  tUsed,.  Tcl_Int
2470: 65 72 70 20 2a 69 6e 74 65 72 70 2c 20 20 20 20  erp *interp,    
2480: 2f 2a 20 54 68 65 20 54 43 4c 20 69 6e 74 65 72  /* The TCL inter
2490: 70 72 65 74 65 72 20 74 68 61 74 20 69 6e 76 6f  preter that invo
24a0: 6b 65 64 20 74 68 69 73 20 63 6f 6d 6d 61 6e 64  ked this command
24b0: 20 2a 2f 0a 20 20 69 6e 74 20 61 72 67 63 2c 20   */.  int argc, 
24c0: 20 20 20 20 20 20 20 20 20 20 20 20 20 2f 2a 20               /* 
24d0: 4e 75 6d 62 65 72 20 6f 66 20 61 72 67 75 6d 65  Number of argume
24e0: 6e 74 73 20 2a 2f 0a 20 20 63 6f 6e 73 74 20 63  nts */.  const c
24f0: 68 61 72 20 2a 2a 61 72 67 76 20 20 20 20 20 20  har **argv      
2500: 2f 2a 20 54 65 78 74 20 6f 66 20 65 61 63 68 20  /* Text of each 
2510: 61 72 67 75 6d 65 6e 74 20 2a 2f 0a 29 7b 0a 20  argument */.){. 
2520: 20 69 6e 74 20 69 3b 0a 20 20 63 6f 6e 73 74 20   int i;.  const 
2530: 63 68 61 72 20 2a 7a 4e 61 6d 65 3b 0a 0a 20 20  char *zName;..  
2540: 69 66 28 20 61 72 67 63 21 3d 32 20 29 7b 0a 20  if( argc!=2 ){. 
2550: 20 20 20 54 63 6c 5f 41 70 70 65 6e 64 52 65 73     Tcl_AppendRes
2560: 75 6c 74 28 69 6e 74 65 72 70 2c 20 22 77 72 6f  ult(interp, "wro
2570: 6e 67 20 23 20 61 72 67 73 3a 20 73 68 6f 75 6c  ng # args: shoul
2580: 64 20 62 65 20 5c 22 22 2c 20 61 72 67 76 5b 30  d be \"", argv[0
2590: 5d 2c 0a 20 20 20 20 20 20 20 22 20 49 44 22 2c  ],.       " ID",
25a0: 20 30 29 3b 0a 20 20 20 20 72 65 74 75 72 6e 20   0);.    return 
25b0: 54 43 4c 5f 45 52 52 4f 52 3b 0a 20 20 7d 0a 20  TCL_ERROR;.  }. 
25c0: 20 69 20 3d 20 70 61 72 73 65 5f 74 68 72 65 61   i = parse_threa
25d0: 64 5f 69 64 28 69 6e 74 65 72 70 2c 20 61 72 67  d_id(interp, arg
25e0: 76 5b 31 5d 29 3b 0a 20 20 69 66 28 20 69 3c 30  v[1]);.  if( i<0
25f0: 20 29 20 72 65 74 75 72 6e 20 54 43 4c 5f 45 52   ) return TCL_ER
2600: 52 4f 52 3b 0a 20 20 69 66 28 20 21 74 68 72 65  ROR;.  if( !thre
2610: 61 64 73 65 74 5b 69 5d 2e 62 75 73 79 20 29 7b  adset[i].busy ){
2620: 0a 20 20 20 20 54 63 6c 5f 41 70 70 65 6e 64 52  .    Tcl_AppendR
2630: 65 73 75 6c 74 28 69 6e 74 65 72 70 2c 20 22 6e  esult(interp, "n
2640: 6f 20 73 75 63 68 20 74 68 72 65 61 64 22 2c 20  o such thread", 
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: 74 68 72 65 61 64 5f 77 61 69 74 28 26 74 68 72  thread_wait(&thr
2680: 65 61 64 73 65 74 5b 69 5d 29 3b 0a 20 20 73 77  eadset[i]);.  sw
2690: 69 74 63 68 28 20 74 68 72 65 61 64 73 65 74 5b  itch( threadset[
26a0: 69 5d 2e 72 63 20 29 7b 0a 20 20 20 20 63 61 73  i].rc ){.    cas
26b0: 65 20 53 51 4c 49 54 45 5f 4f 4b 3a 20 20 20 20  e SQLITE_OK:    
26c0: 20 20 20 20 20 7a 4e 61 6d 65 20 3d 20 22 53 51       zName = "SQ
26d0: 4c 49 54 45 5f 4f 4b 22 3b 20 20 20 20 20 20 20  LITE_OK";       
26e0: 20 20 20 62 72 65 61 6b 3b 0a 20 20 20 20 63 61     break;.    ca
26f0: 73 65 20 53 51 4c 49 54 45 5f 45 52 52 4f 52 3a  se SQLITE_ERROR:
2700: 20 20 20 20 20 20 7a 4e 61 6d 65 20 3d 20 22 53        zName = "S
2710: 51 4c 49 54 45 5f 45 52 52 4f 52 22 3b 20 20 20  QLITE_ERROR";   
2720: 20 20 20 20 62 72 65 61 6b 3b 0a 20 20 20 20 63      break;.    c
2730: 61 73 65 20 53 51 4c 49 54 45 5f 49 4e 54 45 52  ase SQLITE_INTER
2740: 4e 41 4c 3a 20 20 20 7a 4e 61 6d 65 20 3d 20 22  NAL:   zName = "
2750: 53 51 4c 49 54 45 5f 49 4e 54 45 52 4e 41 4c 22  SQLITE_INTERNAL"
2760: 3b 20 20 20 20 62 72 65 61 6b 3b 0a 20 20 20 20  ;    break;.    
2770: 63 61 73 65 20 53 51 4c 49 54 45 5f 50 45 52 4d  case SQLITE_PERM
2780: 3a 20 20 20 20 20 20 20 7a 4e 61 6d 65 20 3d 20  :       zName = 
2790: 22 53 51 4c 49 54 45 5f 50 45 52 4d 22 3b 20 20  "SQLITE_PERM";  
27a0: 20 20 20 20 20 20 62 72 65 61 6b 3b 0a 20 20 20        break;.   
27b0: 20 63 61 73 65 20 53 51 4c 49 54 45 5f 41 42 4f   case SQLITE_ABO
27c0: 52 54 3a 20 20 20 20 20 20 7a 4e 61 6d 65 20 3d  RT:      zName =
27d0: 20 22 53 51 4c 49 54 45 5f 41 42 4f 52 54 22 3b   "SQLITE_ABORT";
27e0: 20 20 20 20 20 20 20 62 72 65 61 6b 3b 0a 20 20         break;.  
27f0: 20 20 63 61 73 65 20 53 51 4c 49 54 45 5f 42 55    case SQLITE_BU
2800: 53 59 3a 20 20 20 20 20 20 20 7a 4e 61 6d 65 20  SY:       zName 
2810: 3d 20 22 53 51 4c 49 54 45 5f 42 55 53 59 22 3b  = "SQLITE_BUSY";
2820: 20 20 20 20 20 20 20 20 62 72 65 61 6b 3b 0a 20          break;. 
2830: 20 20 20 63 61 73 65 20 53 51 4c 49 54 45 5f 4c     case SQLITE_L
2840: 4f 43 4b 45 44 3a 20 20 20 20 20 7a 4e 61 6d 65  OCKED:     zName
2850: 20 3d 20 22 53 51 4c 49 54 45 5f 4c 4f 43 4b 45   = "SQLITE_LOCKE
2860: 44 22 3b 20 20 20 20 20 20 62 72 65 61 6b 3b 0a  D";      break;.
2870: 20 20 20 20 63 61 73 65 20 53 51 4c 49 54 45 5f      case SQLITE_
2880: 4e 4f 4d 45 4d 3a 20 20 20 20 20 20 7a 4e 61 6d  NOMEM:      zNam
2890: 65 20 3d 20 22 53 51 4c 49 54 45 5f 4e 4f 4d 45  e = "SQLITE_NOME
28a0: 4d 22 3b 20 20 20 20 20 20 20 62 72 65 61 6b 3b  M";       break;
28b0: 0a 20 20 20 20 63 61 73 65 20 53 51 4c 49 54 45  .    case SQLITE
28c0: 5f 52 45 41 44 4f 4e 4c 59 3a 20 20 20 7a 4e 61  _READONLY:   zNa
28d0: 6d 65 20 3d 20 22 53 51 4c 49 54 45 5f 52 45 41  me = "SQLITE_REA
28e0: 44 4f 4e 4c 59 22 3b 20 20 20 20 62 72 65 61 6b  DONLY";    break
28f0: 3b 0a 20 20 20 20 63 61 73 65 20 53 51 4c 49 54  ;.    case SQLIT
2900: 45 5f 49 4e 54 45 52 52 55 50 54 3a 20 20 7a 4e  E_INTERRUPT:  zN
2910: 61 6d 65 20 3d 20 22 53 51 4c 49 54 45 5f 49 4e  ame = "SQLITE_IN
2920: 54 45 52 52 55 50 54 22 3b 20 20 20 62 72 65 61  TERRUPT";   brea
2930: 6b 3b 0a 20 20 20 20 63 61 73 65 20 53 51 4c 49  k;.    case SQLI
2940: 54 45 5f 49 4f 45 52 52 3a 20 20 20 20 20 20 7a  TE_IOERR:      z
2950: 4e 61 6d 65 20 3d 20 22 53 51 4c 49 54 45 5f 49  Name = "SQLITE_I
2960: 4f 45 52 52 22 3b 20 20 20 20 20 20 20 62 72 65  OERR";       bre
2970: 61 6b 3b 0a 20 20 20 20 63 61 73 65 20 53 51 4c  ak;.    case SQL
2980: 49 54 45 5f 43 4f 52 52 55 50 54 3a 20 20 20 20  ITE_CORRUPT:    
2990: 7a 4e 61 6d 65 20 3d 20 22 53 51 4c 49 54 45 5f  zName = "SQLITE_
29a0: 43 4f 52 52 55 50 54 22 3b 20 20 20 20 20 62 72  CORRUPT";     br
29b0: 65 61 6b 3b 0a 20 20 20 20 63 61 73 65 20 53 51  eak;.    case SQ
29c0: 4c 49 54 45 5f 4e 4f 54 46 4f 55 4e 44 3a 20 20  LITE_NOTFOUND:  
29d0: 20 7a 4e 61 6d 65 20 3d 20 22 53 51 4c 49 54 45   zName = "SQLITE
29e0: 5f 4e 4f 54 46 4f 55 4e 44 22 3b 20 20 20 20 62  _NOTFOUND";    b
29f0: 72 65 61 6b 3b 0a 20 20 20 20 63 61 73 65 20 53  reak;.    case S
2a00: 51 4c 49 54 45 5f 46 55 4c 4c 3a 20 20 20 20 20  QLITE_FULL:     
2a10: 20 20 7a 4e 61 6d 65 20 3d 20 22 53 51 4c 49 54    zName = "SQLIT
2a20: 45 5f 46 55 4c 4c 22 3b 20 20 20 20 20 20 20 20  E_FULL";        
2a30: 62 72 65 61 6b 3b 0a 20 20 20 20 63 61 73 65 20  break;.    case 
2a40: 53 51 4c 49 54 45 5f 43 41 4e 54 4f 50 45 4e 3a  SQLITE_CANTOPEN:
2a50: 20 20 20 7a 4e 61 6d 65 20 3d 20 22 53 51 4c 49     zName = "SQLI
2a60: 54 45 5f 43 41 4e 54 4f 50 45 4e 22 3b 20 20 20  TE_CANTOPEN";   
2a70: 20 62 72 65 61 6b 3b 0a 20 20 20 20 63 61 73 65   break;.    case
2a80: 20 53 51 4c 49 54 45 5f 50 52 4f 54 4f 43 4f 4c   SQLITE_PROTOCOL
2a90: 3a 20 20 20 7a 4e 61 6d 65 20 3d 20 22 53 51 4c  :   zName = "SQL
2aa0: 49 54 45 5f 50 52 4f 54 4f 43 4f 4c 22 3b 20 20  ITE_PROTOCOL";  
2ab0: 20 20 62 72 65 61 6b 3b 0a 20 20 20 20 63 61 73    break;.    cas
2ac0: 65 20 53 51 4c 49 54 45 5f 45 4d 50 54 59 3a 20  e SQLITE_EMPTY: 
2ad0: 20 20 20 20 20 7a 4e 61 6d 65 20 3d 20 22 53 51       zName = "SQ
2ae0: 4c 49 54 45 5f 45 4d 50 54 59 22 3b 20 20 20 20  LITE_EMPTY";    
2af0: 20 20 20 62 72 65 61 6b 3b 0a 20 20 20 20 63 61     break;.    ca
2b00: 73 65 20 53 51 4c 49 54 45 5f 53 43 48 45 4d 41  se SQLITE_SCHEMA
2b10: 3a 20 20 20 20 20 7a 4e 61 6d 65 20 3d 20 22 53  :     zName = "S
2b20: 51 4c 49 54 45 5f 53 43 48 45 4d 41 22 3b 20 20  QLITE_SCHEMA";  
2b30: 20 20 20 20 62 72 65 61 6b 3b 0a 20 20 20 20 63      break;.    c
2b40: 61 73 65 20 53 51 4c 49 54 45 5f 54 4f 4f 42 49  ase SQLITE_TOOBI
2b50: 47 3a 20 20 20 20 20 7a 4e 61 6d 65 20 3d 20 22  G:     zName = "
2b60: 53 51 4c 49 54 45 5f 54 4f 4f 42 49 47 22 3b 20  SQLITE_TOOBIG"; 
2b70: 20 20 20 20 20 62 72 65 61 6b 3b 0a 20 20 20 20       break;.    
2b80: 63 61 73 65 20 53 51 4c 49 54 45 5f 43 4f 4e 53  case SQLITE_CONS
2b90: 54 52 41 49 4e 54 3a 20 7a 4e 61 6d 65 20 3d 20  TRAINT: zName = 
2ba0: 22 53 51 4c 49 54 45 5f 43 4f 4e 53 54 52 41 49  "SQLITE_CONSTRAI
2bb0: 4e 54 22 3b 20 20 62 72 65 61 6b 3b 0a 20 20 20  NT";  break;.   
2bc0: 20 63 61 73 65 20 53 51 4c 49 54 45 5f 4d 49 53   case SQLITE_MIS
2bd0: 4d 41 54 43 48 3a 20 20 20 7a 4e 61 6d 65 20 3d  MATCH:   zName =
2be0: 20 22 53 51 4c 49 54 45 5f 4d 49 53 4d 41 54 43   "SQLITE_MISMATC
2bf0: 48 22 3b 20 20 20 20 62 72 65 61 6b 3b 0a 20 20  H";    break;.  
2c00: 20 20 63 61 73 65 20 53 51 4c 49 54 45 5f 4d 49    case SQLITE_MI
2c10: 53 55 53 45 3a 20 20 20 20 20 7a 4e 61 6d 65 20  SUSE:     zName 
2c20: 3d 20 22 53 51 4c 49 54 45 5f 4d 49 53 55 53 45  = "SQLITE_MISUSE
2c30: 22 3b 20 20 20 20 20 20 62 72 65 61 6b 3b 0a 20  ";      break;. 
2c40: 20 20 20 63 61 73 65 20 53 51 4c 49 54 45 5f 4e     case SQLITE_N
2c50: 4f 4c 46 53 3a 20 20 20 20 20 20 7a 4e 61 6d 65  OLFS:      zName
2c60: 20 3d 20 22 53 51 4c 49 54 45 5f 4e 4f 4c 46 53   = "SQLITE_NOLFS
2c70: 22 3b 20 20 20 20 20 20 20 62 72 65 61 6b 3b 0a  ";       break;.
2c80: 20 20 20 20 63 61 73 65 20 53 51 4c 49 54 45 5f      case SQLITE_
2c90: 41 55 54 48 3a 20 20 20 20 20 20 20 7a 4e 61 6d  AUTH:       zNam
2ca0: 65 20 3d 20 22 53 51 4c 49 54 45 5f 41 55 54 48  e = "SQLITE_AUTH
2cb0: 22 3b 20 20 20 20 20 20 20 20 62 72 65 61 6b 3b  ";        break;
2cc0: 0a 20 20 20 20 63 61 73 65 20 53 51 4c 49 54 45  .    case SQLITE
2cd0: 5f 46 4f 52 4d 41 54 3a 20 20 20 20 20 7a 4e 61  _FORMAT:     zNa
2ce0: 6d 65 20 3d 20 22 53 51 4c 49 54 45 5f 46 4f 52  me = "SQLITE_FOR
2cf0: 4d 41 54 22 3b 20 20 20 20 20 20 62 72 65 61 6b  MAT";      break
2d00: 3b 0a 20 20 20 20 63 61 73 65 20 53 51 4c 49 54  ;.    case SQLIT
2d10: 45 5f 52 41 4e 47 45 3a 20 20 20 20 20 20 7a 4e  E_RANGE:      zN
2d20: 61 6d 65 20 3d 20 22 53 51 4c 49 54 45 5f 52 41  ame = "SQLITE_RA
2d30: 4e 47 45 22 3b 20 20 20 20 20 20 20 62 72 65 61  NGE";       brea
2d40: 6b 3b 0a 20 20 20 20 63 61 73 65 20 53 51 4c 49  k;.    case SQLI
2d50: 54 45 5f 52 4f 57 3a 20 20 20 20 20 20 20 20 7a  TE_ROW:        z
2d60: 4e 61 6d 65 20 3d 20 22 53 51 4c 49 54 45 5f 52  Name = "SQLITE_R
2d70: 4f 57 22 3b 20 20 20 20 20 20 20 20 20 62 72 65  OW";         bre
2d80: 61 6b 3b 0a 20 20 20 20 63 61 73 65 20 53 51 4c  ak;.    case SQL
2d90: 49 54 45 5f 44 4f 4e 45 3a 20 20 20 20 20 20 20  ITE_DONE:       
2da0: 7a 4e 61 6d 65 20 3d 20 22 53 51 4c 49 54 45 5f  zName = "SQLITE_
2db0: 44 4f 4e 45 22 3b 20 20 20 20 20 20 20 20 62 72  DONE";        br
2dc0: 65 61 6b 3b 0a 20 20 20 20 64 65 66 61 75 6c 74  eak;.    default
2dd0: 3a 20 20 20 20 20 20 20 20 20 20 20 20 20 20 20  :               
2de0: 20 7a 4e 61 6d 65 20 3d 20 22 53 51 4c 49 54 45   zName = "SQLITE
2df0: 5f 55 6e 6b 6e 6f 77 6e 22 3b 20 20 20 20 20 62  _Unknown";     b
2e00: 72 65 61 6b 3b 0a 20 20 7d 0a 20 20 54 63 6c 5f  reak;.  }.  Tcl_
2e10: 41 70 70 65 6e 64 52 65 73 75 6c 74 28 69 6e 74  AppendResult(int
2e20: 65 72 70 2c 20 7a 4e 61 6d 65 2c 20 30 29 3b 0a  erp, zName, 0);.
2e30: 20 20 72 65 74 75 72 6e 20 54 43 4c 5f 4f 4b 3b    return TCL_OK;
2e40: 0a 7d 0a 0a 2f 2a 0a 2a 2a 20 55 73 61 67 65 3a  .}../*.** Usage:
2e50: 20 74 68 72 65 61 64 5f 65 72 72 6f 72 20 20 49   thread_error  I
2e60: 44 0a 2a 2a 0a 2a 2a 20 57 61 69 74 20 6f 6e 20  D.**.** Wait on 
2e70: 74 68 65 20 6d 6f 73 74 20 72 65 63 65 6e 74 20  the most recent 
2e80: 6f 70 65 72 61 74 69 6f 6e 20 74 6f 20 63 6f 6d  operation to com
2e90: 70 6c 65 74 65 2c 20 74 68 65 6e 20 72 65 74 75  plete, then retu
2ea0: 72 6e 20 74 68 65 0a 2a 2a 20 65 72 72 6f 72 20  rn the.** error 
2eb0: 73 74 72 69 6e 67 2e 0a 2a 2f 0a 73 74 61 74 69  string..*/.stati
2ec0: 63 20 69 6e 74 20 74 63 6c 5f 74 68 72 65 61 64  c int tcl_thread
2ed0: 5f 65 72 72 6f 72 28 0a 20 20 76 6f 69 64 20 2a  _error(.  void *
2ee0: 4e 6f 74 55 73 65 64 2c 0a 20 20 54 63 6c 5f 49  NotUsed,.  Tcl_I
2ef0: 6e 74 65 72 70 20 2a 69 6e 74 65 72 70 2c 20 20  nterp *interp,  
2f00: 20 20 2f 2a 20 54 68 65 20 54 43 4c 20 69 6e 74    /* The TCL int
2f10: 65 72 70 72 65 74 65 72 20 74 68 61 74 20 69 6e  erpreter that in
2f20: 76 6f 6b 65 64 20 74 68 69 73 20 63 6f 6d 6d 61  voked this comma
2f30: 6e 64 20 2a 2f 0a 20 20 69 6e 74 20 61 72 67 63  nd */.  int argc
2f40: 2c 20 20 20 20 20 20 20 20 20 20 20 20 20 20 2f  ,              /
2f50: 2a 20 4e 75 6d 62 65 72 20 6f 66 20 61 72 67 75  * Number of argu
2f60: 6d 65 6e 74 73 20 2a 2f 0a 20 20 63 6f 6e 73 74  ments */.  const
2f70: 20 63 68 61 72 20 2a 2a 61 72 67 76 20 20 20 20   char **argv    
2f80: 20 20 2f 2a 20 54 65 78 74 20 6f 66 20 65 61 63    /* Text of eac
2f90: 68 20 61 72 67 75 6d 65 6e 74 20 2a 2f 0a 29 7b  h argument */.){
2fa0: 0a 20 20 69 6e 74 20 69 3b 0a 0a 20 20 69 66 28  .  int i;..  if(
2fb0: 20 61 72 67 63 21 3d 32 20 29 7b 0a 20 20 20 20   argc!=2 ){.    
2fc0: 54 63 6c 5f 41 70 70 65 6e 64 52 65 73 75 6c 74  Tcl_AppendResult
2fd0: 28 69 6e 74 65 72 70 2c 20 22 77 72 6f 6e 67 20  (interp, "wrong 
2fe0: 23 20 61 72 67 73 3a 20 73 68 6f 75 6c 64 20 62  # args: should b
2ff0: 65 20 5c 22 22 2c 20 61 72 67 76 5b 30 5d 2c 0a  e \"", argv[0],.
3000: 20 20 20 20 20 20 20 22 20 49 44 22 2c 20 30 29         " ID", 0)
3010: 3b 0a 20 20 20 20 72 65 74 75 72 6e 20 54 43 4c  ;.    return TCL
3020: 5f 45 52 52 4f 52 3b 0a 20 20 7d 0a 20 20 69 20  _ERROR;.  }.  i 
3030: 3d 20 70 61 72 73 65 5f 74 68 72 65 61 64 5f 69  = parse_thread_i
3040: 64 28 69 6e 74 65 72 70 2c 20 61 72 67 76 5b 31  d(interp, argv[1
3050: 5d 29 3b 0a 20 20 69 66 28 20 69 3c 30 20 29 20  ]);.  if( i<0 ) 
3060: 72 65 74 75 72 6e 20 54 43 4c 5f 45 52 52 4f 52  return TCL_ERROR
3070: 3b 0a 20 20 69 66 28 20 21 74 68 72 65 61 64 73  ;.  if( !threads
3080: 65 74 5b 69 5d 2e 62 75 73 79 20 29 7b 0a 20 20  et[i].busy ){.  
3090: 20 20 54 63 6c 5f 41 70 70 65 6e 64 52 65 73 75    Tcl_AppendResu
30a0: 6c 74 28 69 6e 74 65 72 70 2c 20 22 6e 6f 20 73  lt(interp, "no s
30b0: 75 63 68 20 74 68 72 65 61 64 22 2c 20 30 29 3b  uch thread", 0);
30c0: 0a 20 20 20 20 72 65 74 75 72 6e 20 54 43 4c 5f  .    return TCL_
30d0: 45 52 52 4f 52 3b 0a 20 20 7d 0a 20 20 74 68 72  ERROR;.  }.  thr
30e0: 65 61 64 5f 77 61 69 74 28 26 74 68 72 65 61 64  ead_wait(&thread
30f0: 73 65 74 5b 69 5d 29 3b 0a 20 20 54 63 6c 5f 41  set[i]);.  Tcl_A
3100: 70 70 65 6e 64 52 65 73 75 6c 74 28 69 6e 74 65  ppendResult(inte
3110: 72 70 2c 20 74 68 72 65 61 64 73 65 74 5b 69 5d  rp, threadset[i]
3120: 2e 7a 45 72 72 2c 20 30 29 3b 0a 20 20 72 65 74  .zErr, 0);.  ret
3130: 75 72 6e 20 54 43 4c 5f 4f 4b 3b 0a 7d 0a 0a 2f  urn TCL_OK;.}../
3140: 2a 0a 2a 2a 20 54 68 69 73 20 70 72 6f 63 65 64  *.** This proced
3150: 75 72 65 20 72 75 6e 73 20 69 6e 20 74 68 65 20  ure runs in the 
3160: 74 68 72 65 61 64 20 74 6f 20 63 6f 6d 70 69 6c  thread to compil
3170: 65 20 61 6e 20 53 51 4c 20 73 74 61 74 65 6d 65  e an SQL stateme
3180: 6e 74 2e 0a 2a 2f 0a 73 74 61 74 69 63 20 76 6f  nt..*/.static vo
3190: 69 64 20 64 6f 5f 63 6f 6d 70 69 6c 65 28 54 68  id do_compile(Th
31a0: 72 65 61 64 20 2a 70 29 7b 0a 20 20 69 66 28 20  read *p){.  if( 
31b0: 70 2d 3e 64 62 3d 3d 30 20 29 7b 0a 20 20 20 20  p->db==0 ){.    
31c0: 70 2d 3e 7a 45 72 72 20 3d 20 70 2d 3e 7a 53 74  p->zErr = p->zSt
31d0: 61 74 69 63 45 72 72 20 3d 20 22 6e 6f 20 64 61  aticErr = "no da
31e0: 74 61 62 61 73 65 20 69 73 20 6f 70 65 6e 22 3b  tabase is open";
31f0: 0a 20 20 20 20 70 2d 3e 72 63 20 3d 20 53 51 4c  .    p->rc = SQL
3200: 49 54 45 5f 45 52 52 4f 52 3b 0a 20 20 20 20 72  ITE_ERROR;.    r
3210: 65 74 75 72 6e 3b 0a 20 20 7d 0a 20 20 69 66 28  eturn;.  }.  if(
3220: 20 70 2d 3e 76 6d 20 29 7b 0a 20 20 20 20 73 71   p->vm ){.    sq
3230: 6c 69 74 65 5f 66 69 6e 61 6c 69 7a 65 28 70 2d  lite_finalize(p-
3240: 3e 76 6d 2c 20 30 29 3b 0a 20 20 20 20 70 2d 3e  >vm, 0);.    p->
3250: 76 6d 20 3d 20 30 3b 0a 20 20 7d 0a 20 20 70 2d  vm = 0;.  }.  p-
3260: 3e 72 63 20 3d 20 73 71 6c 69 74 65 5f 63 6f 6d  >rc = sqlite_com
3270: 70 69 6c 65 28 70 2d 3e 64 62 2c 20 70 2d 3e 7a  pile(p->db, p->z
3280: 41 72 67 2c 20 30 2c 20 26 70 2d 3e 76 6d 2c 20  Arg, 0, &p->vm, 
3290: 26 70 2d 3e 7a 45 72 72 29 3b 0a 7d 0a 0a 2f 2a  &p->zErr);.}../*
32a0: 0a 2a 2a 20 55 73 61 67 65 3a 20 74 68 72 65 61  .** Usage: threa
32b0: 64 5f 63 6f 6d 70 69 6c 65 20 49 44 20 53 51 4c  d_compile ID SQL
32c0: 0a 2a 2a 0a 2a 2a 20 43 6f 6d 70 69 6c 65 20 61  .**.** Compile a
32d0: 20 6e 65 77 20 76 69 72 74 75 61 6c 20 6d 61 63   new virtual mac
32e0: 68 69 6e 65 2e 0a 2a 2f 0a 73 74 61 74 69 63 20  hine..*/.static 
32f0: 69 6e 74 20 74 63 6c 5f 74 68 72 65 61 64 5f 63  int tcl_thread_c
3300: 6f 6d 70 69 6c 65 28 0a 20 20 76 6f 69 64 20 2a  ompile(.  void *
3310: 4e 6f 74 55 73 65 64 2c 0a 20 20 54 63 6c 5f 49  NotUsed,.  Tcl_I
3320: 6e 74 65 72 70 20 2a 69 6e 74 65 72 70 2c 20 20  nterp *interp,  
3330: 20 20 2f 2a 20 54 68 65 20 54 43 4c 20 69 6e 74    /* The TCL int
3340: 65 72 70 72 65 74 65 72 20 74 68 61 74 20 69 6e  erpreter that in
3350: 76 6f 6b 65 64 20 74 68 69 73 20 63 6f 6d 6d 61  voked this comma
3360: 6e 64 20 2a 2f 0a 20 20 69 6e 74 20 61 72 67 63  nd */.  int argc
3370: 2c 20 20 20 20 20 20 20 20 20 20 20 20 20 20 2f  ,              /
3380: 2a 20 4e 75 6d 62 65 72 20 6f 66 20 61 72 67 75  * Number of argu
3390: 6d 65 6e 74 73 20 2a 2f 0a 20 20 63 6f 6e 73 74  ments */.  const
33a0: 20 63 68 61 72 20 2a 2a 61 72 67 76 20 20 20 20   char **argv    
33b0: 20 20 2f 2a 20 54 65 78 74 20 6f 66 20 65 61 63    /* Text of eac
33c0: 68 20 61 72 67 75 6d 65 6e 74 20 2a 2f 0a 29 7b  h argument */.){
33d0: 0a 20 20 69 6e 74 20 69 3b 0a 20 20 69 66 28 20  .  int i;.  if( 
33e0: 61 72 67 63 21 3d 33 20 29 7b 0a 20 20 20 20 54  argc!=3 ){.    T
33f0: 63 6c 5f 41 70 70 65 6e 64 52 65 73 75 6c 74 28  cl_AppendResult(
3400: 69 6e 74 65 72 70 2c 20 22 77 72 6f 6e 67 20 23  interp, "wrong #
3410: 20 61 72 67 73 3a 20 73 68 6f 75 6c 64 20 62 65   args: should be
3420: 20 5c 22 22 2c 20 61 72 67 76 5b 30 5d 2c 0a 20   \"", argv[0],. 
3430: 20 20 20 20 20 20 22 20 49 44 20 53 51 4c 22 2c        " ID SQL",
3440: 20 30 29 3b 0a 20 20 20 20 72 65 74 75 72 6e 20   0);.    return 
3450: 54 43 4c 5f 45 52 52 4f 52 3b 0a 20 20 7d 0a 20  TCL_ERROR;.  }. 
3460: 20 69 20 3d 20 70 61 72 73 65 5f 74 68 72 65 61   i = parse_threa
3470: 64 5f 69 64 28 69 6e 74 65 72 70 2c 20 61 72 67  d_id(interp, arg
3480: 76 5b 31 5d 29 3b 0a 20 20 69 66 28 20 69 3c 30  v[1]);.  if( i<0
3490: 20 29 20 72 65 74 75 72 6e 20 54 43 4c 5f 45 52   ) return TCL_ER
34a0: 52 4f 52 3b 0a 20 20 69 66 28 20 21 74 68 72 65  ROR;.  if( !thre
34b0: 61 64 73 65 74 5b 69 5d 2e 62 75 73 79 20 29 7b  adset[i].busy ){
34c0: 0a 20 20 20 20 54 63 6c 5f 41 70 70 65 6e 64 52  .    Tcl_AppendR
34d0: 65 73 75 6c 74 28 69 6e 74 65 72 70 2c 20 22 6e  esult(interp, "n
34e0: 6f 20 73 75 63 68 20 74 68 72 65 61 64 22 2c 20  o such thread", 
34f0: 30 29 3b 0a 20 20 20 20 72 65 74 75 72 6e 20 54  0);.    return T
3500: 43 4c 5f 45 52 52 4f 52 3b 0a 20 20 7d 0a 20 20  CL_ERROR;.  }.  
3510: 74 68 72 65 61 64 5f 77 61 69 74 28 26 74 68 72  thread_wait(&thr
3520: 65 61 64 73 65 74 5b 69 5d 29 3b 0a 20 20 74 68  eadset[i]);.  th
3530: 72 65 61 64 73 65 74 5b 69 5d 2e 78 4f 70 20 3d  readset[i].xOp =
3540: 20 64 6f 5f 63 6f 6d 70 69 6c 65 3b 0a 20 20 73   do_compile;.  s
3550: 71 6c 69 74 65 46 72 65 65 28 74 68 72 65 61 64  qliteFree(thread
3560: 73 65 74 5b 69 5d 2e 7a 41 72 67 29 3b 0a 20 20  set[i].zArg);.  
3570: 74 68 72 65 61 64 73 65 74 5b 69 5d 2e 7a 41 72  threadset[i].zAr
3580: 67 20 3d 20 73 71 6c 69 74 65 53 74 72 44 75 70  g = sqliteStrDup
3590: 28 61 72 67 76 5b 32 5d 29 3b 0a 20 20 74 68 72  (argv[2]);.  thr
35a0: 65 61 64 73 65 74 5b 69 5d 2e 6f 70 6e 75 6d 2b  eadset[i].opnum+
35b0: 2b 3b 0a 20 20 72 65 74 75 72 6e 20 54 43 4c 5f  +;.  return TCL_
35c0: 4f 4b 3b 0a 7d 0a 0a 2f 2a 0a 2a 2a 20 54 68 69  OK;.}../*.** Thi
35d0: 73 20 70 72 6f 63 65 64 75 72 65 20 72 75 6e 73  s procedure runs
35e0: 20 69 6e 20 74 68 65 20 74 68 72 65 61 64 20 74   in the thread t
35f0: 6f 20 73 74 65 70 20 74 68 65 20 76 69 72 74 75  o step the virtu
3600: 61 6c 20 6d 61 63 68 69 6e 65 2e 0a 2a 2f 0a 73  al machine..*/.s
3610: 74 61 74 69 63 20 76 6f 69 64 20 64 6f 5f 73 74  tatic void do_st
3620: 65 70 28 54 68 72 65 61 64 20 2a 70 29 7b 0a 20  ep(Thread *p){. 
3630: 20 69 66 28 20 70 2d 3e 76 6d 3d 3d 30 20 29 7b   if( p->vm==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 5f 73 74 65 70 28 70   = sqlite_step(p
36c0: 2d 3e 76 6d 2c 20 26 70 2d 3e 61 72 67 63 2c 20  ->vm, &p->argc, 
36d0: 26 70 2d 3e 61 72 67 76 2c 20 26 70 2d 3e 63 6f  &p->argv, &p->co
36e0: 6c 76 29 3b 0a 7d 0a 0a 2f 2a 0a 2a 2a 20 55 73  lv);.}../*.** Us
36f0: 61 67 65 3a 20 74 68 72 65 61 64 5f 73 74 65 70  age: thread_step
3700: 20 49 44 0a 2a 2a 0a 2a 2a 20 41 64 76 61 6e 63   ID.**.** Advanc
3710: 65 20 74 68 65 20 76 69 72 74 75 61 6c 20 6d 61  e the virtual ma
3720: 63 68 69 6e 65 20 62 79 20 6f 6e 65 20 73 74 65  chine by one ste
3730: 70 0a 2a 2f 0a 73 74 61 74 69 63 20 69 6e 74 20  p.*/.static int 
3740: 74 63 6c 5f 74 68 72 65 61 64 5f 73 74 65 70 28  tcl_thread_step(
3750: 0a 20 20 76 6f 69 64 20 2a 4e 6f 74 55 73 65 64  .  void *NotUsed
3760: 2c 0a 20 20 54 63 6c 5f 49 6e 74 65 72 70 20 2a  ,.  Tcl_Interp *
3770: 69 6e 74 65 72 70 2c 20 20 20 20 2f 2a 20 54 68  interp,    /* Th
3780: 65 20 54 43 4c 20 69 6e 74 65 72 70 72 65 74 65  e TCL interprete
3790: 72 20 74 68 61 74 20 69 6e 76 6f 6b 65 64 20 74  r that invoked t
37a0: 68 69 73 20 63 6f 6d 6d 61 6e 64 20 2a 2f 0a 20  his command */. 
37b0: 20 69 6e 74 20 61 72 67 63 2c 20 20 20 20 20 20   int argc,      
37c0: 20 20 20 20 20 20 20 20 2f 2a 20 4e 75 6d 62 65          /* Numbe
37d0: 72 20 6f 66 20 61 72 67 75 6d 65 6e 74 73 20 2a  r of arguments *
37e0: 2f 0a 20 20 63 6f 6e 73 74 20 63 68 61 72 20 2a  /.  const char *
37f0: 2a 61 72 67 76 20 20 20 20 20 20 2f 2a 20 54 65  *argv      /* Te
3800: 78 74 20 6f 66 20 65 61 63 68 20 61 72 67 75 6d  xt of each argum
3810: 65 6e 74 20 2a 2f 0a 29 7b 0a 20 20 69 6e 74 20  ent */.){.  int 
3820: 69 3b 0a 20 20 69 66 28 20 61 72 67 63 21 3d 32  i;.  if( argc!=2
3830: 20 29 7b 0a 20 20 20 20 54 63 6c 5f 41 70 70 65   ){.    Tcl_Appe
3840: 6e 64 52 65 73 75 6c 74 28 69 6e 74 65 72 70 2c  ndResult(interp,
3850: 20 22 77 72 6f 6e 67 20 23 20 61 72 67 73 3a 20   "wrong # args: 
3860: 73 68 6f 75 6c 64 20 62 65 20 5c 22 22 2c 20 61  should be \"", a
3870: 72 67 76 5b 30 5d 2c 0a 20 20 20 20 20 20 20 22  rgv[0],.       "
3880: 20 49 44 4c 22 2c 20 30 29 3b 0a 20 20 20 20 72   IDL", 0);.    r
3890: 65 74 75 72 6e 20 54 43 4c 5f 45 52 52 4f 52 3b  eturn TCL_ERROR;
38a0: 0a 20 20 7d 0a 20 20 69 20 3d 20 70 61 72 73 65  .  }.  i = parse
38b0: 5f 74 68 72 65 61 64 5f 69 64 28 69 6e 74 65 72  _thread_id(inter
38c0: 70 2c 20 61 72 67 76 5b 31 5d 29 3b 0a 20 20 69  p, argv[1]);.  i
38d0: 66 28 20 69 3c 30 20 29 20 72 65 74 75 72 6e 20  f( i<0 ) return 
38e0: 54 43 4c 5f 45 52 52 4f 52 3b 0a 20 20 69 66 28  TCL_ERROR;.  if(
38f0: 20 21 74 68 72 65 61 64 73 65 74 5b 69 5d 2e 62   !threadset[i].b
3900: 75 73 79 20 29 7b 0a 20 20 20 20 54 63 6c 5f 41  usy ){.    Tcl_A
3910: 70 70 65 6e 64 52 65 73 75 6c 74 28 69 6e 74 65  ppendResult(inte
3920: 72 70 2c 20 22 6e 6f 20 73 75 63 68 20 74 68 72  rp, "no such thr
3930: 65 61 64 22 2c 20 30 29 3b 0a 20 20 20 20 72 65  ead", 0);.    re
3940: 74 75 72 6e 20 54 43 4c 5f 45 52 52 4f 52 3b 0a  turn TCL_ERROR;.
3950: 20 20 7d 0a 20 20 74 68 72 65 61 64 5f 77 61 69    }.  thread_wai
3960: 74 28 26 74 68 72 65 61 64 73 65 74 5b 69 5d 29  t(&threadset[i])
3970: 3b 0a 20 20 74 68 72 65 61 64 73 65 74 5b 69 5d  ;.  threadset[i]
3980: 2e 78 4f 70 20 3d 20 64 6f 5f 73 74 65 70 3b 0a  .xOp = do_step;.
3990: 20 20 74 68 72 65 61 64 73 65 74 5b 69 5d 2e 6f    threadset[i].o
39a0: 70 6e 75 6d 2b 2b 3b 0a 20 20 72 65 74 75 72 6e  pnum++;.  return
39b0: 20 54 43 4c 5f 4f 4b 3b 0a 7d 0a 0a 2f 2a 0a 2a   TCL_OK;.}../*.*
39c0: 2a 20 54 68 69 73 20 70 72 6f 63 65 64 75 72 65  * This procedure
39d0: 20 72 75 6e 73 20 69 6e 20 74 68 65 20 74 68 72   runs in the thr
39e0: 65 61 64 20 74 6f 20 66 69 6e 61 6c 69 7a 65 20  ead to finalize 
39f0: 61 20 76 69 72 74 75 61 6c 20 6d 61 63 68 69 6e  a virtual machin
3a00: 65 2e 0a 2a 2f 0a 73 74 61 74 69 63 20 76 6f 69  e..*/.static voi
3a10: 64 20 64 6f 5f 66 69 6e 61 6c 69 7a 65 28 54 68  d do_finalize(Th
3a20: 72 65 61 64 20 2a 70 29 7b 0a 20 20 69 66 28 20  read *p){.  if( 
3a30: 70 2d 3e 76 6d 3d 3d 30 20 29 7b 0a 20 20 20 20  p->vm==0 ){.    
3a40: 70 2d 3e 7a 45 72 72 20 3d 20 70 2d 3e 7a 53 74  p->zErr = p->zSt
3a50: 61 74 69 63 45 72 72 20 3d 20 22 6e 6f 20 76 69  aticErr = "no vi
3a60: 72 74 75 61 6c 20 6d 61 63 68 69 6e 65 20 61 76  rtual machine av
3a70: 61 69 6c 61 62 6c 65 22 3b 0a 20 20 20 20 70 2d  ailable";.    p-
3a80: 3e 72 63 20 3d 20 53 51 4c 49 54 45 5f 45 52 52  >rc = SQLITE_ERR
3a90: 4f 52 3b 0a 20 20 20 20 72 65 74 75 72 6e 3b 0a  OR;.    return;.
3aa0: 20 20 7d 0a 20 20 70 2d 3e 72 63 20 3d 20 73 71    }.  p->rc = sq
3ab0: 6c 69 74 65 5f 66 69 6e 61 6c 69 7a 65 28 70 2d  lite_finalize(p-
3ac0: 3e 76 6d 2c 20 26 70 2d 3e 7a 45 72 72 29 3b 0a  >vm, &p->zErr);.
3ad0: 20 20 70 2d 3e 76 6d 20 3d 20 30 3b 0a 7d 0a 0a    p->vm = 0;.}..
3ae0: 2f 2a 0a 2a 2a 20 55 73 61 67 65 3a 20 74 68 72  /*.** Usage: thr
3af0: 65 61 64 5f 66 69 6e 61 6c 69 7a 65 20 49 44 0a  ead_finalize ID.
3b00: 2a 2a 0a 2a 2a 20 46 69 6e 61 6c 69 7a 65 20 74  **.** Finalize t
3b10: 68 65 20 76 69 72 74 75 61 6c 20 6d 61 63 68 69  he virtual machi
3b20: 6e 65 2e 0a 2a 2f 0a 73 74 61 74 69 63 20 69 6e  ne..*/.static in
3b30: 74 20 74 63 6c 5f 74 68 72 65 61 64 5f 66 69 6e  t tcl_thread_fin
3b40: 61 6c 69 7a 65 28 0a 20 20 76 6f 69 64 20 2a 4e  alize(.  void *N
3b50: 6f 74 55 73 65 64 2c 0a 20 20 54 63 6c 5f 49 6e  otUsed,.  Tcl_In
3b60: 74 65 72 70 20 2a 69 6e 74 65 72 70 2c 20 20 20  terp *interp,   
3b70: 20 2f 2a 20 54 68 65 20 54 43 4c 20 69 6e 74 65   /* The TCL inte
3b80: 72 70 72 65 74 65 72 20 74 68 61 74 20 69 6e 76  rpreter that inv
3b90: 6f 6b 65 64 20 74 68 69 73 20 63 6f 6d 6d 61 6e  oked this comman
3ba0: 64 20 2a 2f 0a 20 20 69 6e 74 20 61 72 67 63 2c  d */.  int argc,
3bb0: 20 20 20 20 20 20 20 20 20 20 20 20 20 20 2f 2a                /*
3bc0: 20 4e 75 6d 62 65 72 20 6f 66 20 61 72 67 75 6d   Number of argum
3bd0: 65 6e 74 73 20 2a 2f 0a 20 20 63 6f 6e 73 74 20  ents */.  const 
3be0: 63 68 61 72 20 2a 2a 61 72 67 76 20 20 20 20 20  char **argv     
3bf0: 20 2f 2a 20 54 65 78 74 20 6f 66 20 65 61 63 68   /* Text of each
3c00: 20 61 72 67 75 6d 65 6e 74 20 2a 2f 0a 29 7b 0a   argument */.){.
3c10: 20 20 69 6e 74 20 69 3b 0a 20 20 69 66 28 20 61    int i;.  if( a
3c20: 72 67 63 21 3d 32 20 29 7b 0a 20 20 20 20 54 63  rgc!=2 ){.    Tc
3c30: 6c 5f 41 70 70 65 6e 64 52 65 73 75 6c 74 28 69  l_AppendResult(i
3c40: 6e 74 65 72 70 2c 20 22 77 72 6f 6e 67 20 23 20  nterp, "wrong # 
3c50: 61 72 67 73 3a 20 73 68 6f 75 6c 64 20 62 65 20  args: should be 
3c60: 5c 22 22 2c 20 61 72 67 76 5b 30 5d 2c 0a 20 20  \"", argv[0],.  
3c70: 20 20 20 20 20 22 20 49 44 4c 22 2c 20 30 29 3b       " IDL", 0);
3c80: 0a 20 20 20 20 72 65 74 75 72 6e 20 54 43 4c 5f  .    return TCL_
3c90: 45 52 52 4f 52 3b 0a 20 20 7d 0a 20 20 69 20 3d  ERROR;.  }.  i =
3ca0: 20 70 61 72 73 65 5f 74 68 72 65 61 64 5f 69 64   parse_thread_id
3cb0: 28 69 6e 74 65 72 70 2c 20 61 72 67 76 5b 31 5d  (interp, argv[1]
3cc0: 29 3b 0a 20 20 69 66 28 20 69 3c 30 20 29 20 72  );.  if( i<0 ) r
3cd0: 65 74 75 72 6e 20 54 43 4c 5f 45 52 52 4f 52 3b  eturn TCL_ERROR;
3ce0: 0a 20 20 69 66 28 20 21 74 68 72 65 61 64 73 65  .  if( !threadse
3cf0: 74 5b 69 5d 2e 62 75 73 79 20 29 7b 0a 20 20 20  t[i].busy ){.   
3d00: 20 54 63 6c 5f 41 70 70 65 6e 64 52 65 73 75 6c   Tcl_AppendResul
3d10: 74 28 69 6e 74 65 72 70 2c 20 22 6e 6f 20 73 75  t(interp, "no su
3d20: 63 68 20 74 68 72 65 61 64 22 2c 20 30 29 3b 0a  ch thread", 0);.
3d30: 20 20 20 20 72 65 74 75 72 6e 20 54 43 4c 5f 45      return TCL_E
3d40: 52 52 4f 52 3b 0a 20 20 7d 0a 20 20 74 68 72 65  RROR;.  }.  thre
3d50: 61 64 5f 77 61 69 74 28 26 74 68 72 65 61 64 73  ad_wait(&threads
3d60: 65 74 5b 69 5d 29 3b 0a 20 20 74 68 72 65 61 64  et[i]);.  thread
3d70: 73 65 74 5b 69 5d 2e 78 4f 70 20 3d 20 64 6f 5f  set[i].xOp = do_
3d80: 66 69 6e 61 6c 69 7a 65 3b 0a 20 20 73 71 6c 69  finalize;.  sqli
3d90: 74 65 46 72 65 65 28 74 68 72 65 61 64 73 65 74  teFree(threadset
3da0: 5b 69 5d 2e 7a 41 72 67 29 3b 0a 20 20 74 68 72  [i].zArg);.  thr
3db0: 65 61 64 73 65 74 5b 69 5d 2e 7a 41 72 67 20 3d  eadset[i].zArg =
3dc0: 20 30 3b 0a 20 20 74 68 72 65 61 64 73 65 74 5b   0;.  threadset[
3dd0: 69 5d 2e 6f 70 6e 75 6d 2b 2b 3b 0a 20 20 72 65  i].opnum++;.  re
3de0: 74 75 72 6e 20 54 43 4c 5f 4f 4b 3b 0a 7d 0a 0a  turn TCL_OK;.}..
3df0: 2f 2a 0a 2a 2a 20 52 65 67 69 73 74 65 72 20 63  /*.** Register c
3e00: 6f 6d 6d 61 6e 64 73 20 77 69 74 68 20 74 68 65  ommands with the
3e10: 20 54 43 4c 20 69 6e 74 65 72 70 72 65 74 65 72   TCL interpreter
3e20: 2e 0a 2a 2f 0a 69 6e 74 20 53 71 6c 69 74 65 74  ..*/.int Sqlitet
3e30: 65 73 74 34 5f 49 6e 69 74 28 54 63 6c 5f 49 6e  est4_Init(Tcl_In
3e40: 74 65 72 70 20 2a 69 6e 74 65 72 70 29 7b 0a 20  terp *interp){. 
3e50: 20 73 74 61 74 69 63 20 73 74 72 75 63 74 20 7b   static struct {
3e60: 0a 20 20 20 20 20 63 68 61 72 20 2a 7a 4e 61 6d  .     char *zNam
3e70: 65 3b 0a 20 20 20 20 20 54 63 6c 5f 43 6d 64 50  e;.     Tcl_CmdP
3e80: 72 6f 63 20 2a 78 50 72 6f 63 3b 0a 20 20 7d 20  roc *xProc;.  } 
3e90: 61 43 6d 64 5b 5d 20 3d 20 7b 0a 20 20 20 20 20  aCmd[] = {.     
3ea0: 7b 20 22 74 68 72 65 61 64 5f 63 72 65 61 74 65  { "thread_create
3eb0: 22 2c 20 20 20 20 20 28 54 63 6c 5f 43 6d 64 50  ",     (Tcl_CmdP
3ec0: 72 6f 63 2a 29 74 63 6c 5f 74 68 72 65 61 64 5f  roc*)tcl_thread_
3ed0: 63 72 65 61 74 65 20 20 20 20 20 7d 2c 0a 20 20  create     },.  
3ee0: 20 20 20 7b 20 22 74 68 72 65 61 64 5f 77 61 69     { "thread_wai
3ef0: 74 22 2c 20 20 20 20 20 20 20 28 54 63 6c 5f 43  t",       (Tcl_C
3f00: 6d 64 50 72 6f 63 2a 29 74 63 6c 5f 74 68 72 65  mdProc*)tcl_thre
3f10: 61 64 5f 77 61 69 74 20 20 20 20 20 20 20 7d 2c  ad_wait       },
3f20: 0a 20 20 20 20 20 7b 20 22 74 68 72 65 61 64 5f  .     { "thread_
3f30: 68 61 6c 74 22 2c 20 20 20 20 20 20 20 28 54 63  halt",       (Tc
3f40: 6c 5f 43 6d 64 50 72 6f 63 2a 29 74 63 6c 5f 74  l_CmdProc*)tcl_t
3f50: 68 72 65 61 64 5f 68 61 6c 74 20 20 20 20 20 20  hread_halt      
3f60: 20 7d 2c 0a 20 20 20 20 20 7b 20 22 74 68 72 65   },.     { "thre
3f70: 61 64 5f 61 72 67 63 22 2c 20 20 20 20 20 20 20  ad_argc",       
3f80: 28 54 63 6c 5f 43 6d 64 50 72 6f 63 2a 29 74 63  (Tcl_CmdProc*)tc
3f90: 6c 5f 74 68 72 65 61 64 5f 61 72 67 63 20 20 20  l_thread_argc   
3fa0: 20 20 20 20 7d 2c 0a 20 20 20 20 20 7b 20 22 74      },.     { "t
3fb0: 68 72 65 61 64 5f 61 72 67 76 22 2c 20 20 20 20  hread_argv",    
3fc0: 20 20 20 28 54 63 6c 5f 43 6d 64 50 72 6f 63 2a     (Tcl_CmdProc*
3fd0: 29 74 63 6c 5f 74 68 72 65 61 64 5f 61 72 67 76  )tcl_thread_argv
3fe0: 20 20 20 20 20 20 20 7d 2c 0a 20 20 20 20 20 7b         },.     {
3ff0: 20 22 74 68 72 65 61 64 5f 63 6f 6c 6e 61 6d 65   "thread_colname
4000: 22 2c 20 20 20 20 28 54 63 6c 5f 43 6d 64 50 72  ",    (Tcl_CmdPr
4010: 6f 63 2a 29 74 63 6c 5f 74 68 72 65 61 64 5f 63  oc*)tcl_thread_c
4020: 6f 6c 6e 61 6d 65 20 20 20 20 7d 2c 0a 20 20 20  olname    },.   
4030: 20 20 7b 20 22 74 68 72 65 61 64 5f 72 65 73 75    { "thread_resu
4040: 6c 74 22 2c 20 20 20 20 20 28 54 63 6c 5f 43 6d  lt",     (Tcl_Cm
4050: 64 50 72 6f 63 2a 29 74 63 6c 5f 74 68 72 65 61  dProc*)tcl_threa
4060: 64 5f 72 65 73 75 6c 74 20 20 20 20 20 7d 2c 0a  d_result     },.
4070: 20 20 20 20 20 7b 20 22 74 68 72 65 61 64 5f 65       { "thread_e
4080: 72 72 6f 72 22 2c 20 20 20 20 20 20 28 54 63 6c  rror",      (Tcl
4090: 5f 43 6d 64 50 72 6f 63 2a 29 74 63 6c 5f 74 68  _CmdProc*)tcl_th
40a0: 72 65 61 64 5f 65 72 72 6f 72 20 20 20 20 20 20  read_error      
40b0: 7d 2c 0a 20 20 20 20 20 7b 20 22 74 68 72 65 61  },.     { "threa
40c0: 64 5f 63 6f 6d 70 69 6c 65 22 2c 20 20 20 20 28  d_compile",    (
40d0: 54 63 6c 5f 43 6d 64 50 72 6f 63 2a 29 74 63 6c  Tcl_CmdProc*)tcl
40e0: 5f 74 68 72 65 61 64 5f 63 6f 6d 70 69 6c 65 20  _thread_compile 
40f0: 20 20 20 7d 2c 0a 20 20 20 20 20 7b 20 22 74 68     },.     { "th
4100: 72 65 61 64 5f 73 74 65 70 22 2c 20 20 20 20 20  read_step",     
4110: 20 20 28 54 63 6c 5f 43 6d 64 50 72 6f 63 2a 29    (Tcl_CmdProc*)
4120: 74 63 6c 5f 74 68 72 65 61 64 5f 73 74 65 70 20  tcl_thread_step 
4130: 20 20 20 20 20 20 7d 2c 0a 20 20 20 20 20 7b 20        },.     { 
4140: 22 74 68 72 65 61 64 5f 66 69 6e 61 6c 69 7a 65  "thread_finalize
4150: 22 2c 20 20 20 28 54 63 6c 5f 43 6d 64 50 72 6f  ",   (Tcl_CmdPro
4160: 63 2a 29 74 63 6c 5f 74 68 72 65 61 64 5f 66 69  c*)tcl_thread_fi
4170: 6e 61 6c 69 7a 65 20 20 20 7d 2c 0a 20 20 7d 3b  nalize   },.  };
4180: 0a 20 20 69 6e 74 20 69 3b 0a 0a 20 20 66 6f 72  .  int i;..  for
4190: 28 69 3d 30 3b 20 69 3c 73 69 7a 65 6f 66 28 61  (i=0; i<sizeof(a
41a0: 43 6d 64 29 2f 73 69 7a 65 6f 66 28 61 43 6d 64  Cmd)/sizeof(aCmd
41b0: 5b 30 5d 29 3b 20 69 2b 2b 29 7b 0a 20 20 20 20  [0]); i++){.    
41c0: 54 63 6c 5f 43 72 65 61 74 65 43 6f 6d 6d 61 6e  Tcl_CreateComman
41d0: 64 28 69 6e 74 65 72 70 2c 20 61 43 6d 64 5b 69  d(interp, aCmd[i
41e0: 5d 2e 7a 4e 61 6d 65 2c 20 61 43 6d 64 5b 69 5d  ].zName, aCmd[i]
41f0: 2e 78 50 72 6f 63 2c 20 30 2c 20 30 29 3b 0a 20  .xProc, 0, 0);. 
4200: 20 7d 0a 20 20 72 65 74 75 72 6e 20 54 43 4c 5f   }.  return TCL_
4210: 4f 4b 3b 0a 7d 0a 23 65 6c 73 65 0a 69 6e 74 20  OK;.}.#else.int 
4220: 53 71 6c 69 74 65 74 65 73 74 34 5f 49 6e 69 74  Sqlitetest4_Init
4230: 28 54 63 6c 5f 49 6e 74 65 72 70 20 2a 69 6e 74  (Tcl_Interp *int
4240: 65 72 70 29 7b 20 72 65 74 75 72 6e 20 54 43 4c  erp){ return TCL
4250: 5f 4f 4b 3b 20 7d 0a 23 65 6e 64 69 66 20 2f 2a  _OK; }.#endif /*
4260: 20 4f 53 5f 55 4e 49 58 20 2a 2f 0a               OS_UNIX */.