/ Hex Artifact Content
Login

Artifact cc07e5ee19436c08ae2331e8476db0c968ade42528df68cfa40eb58314cd21e1:


0000: 2f 2a 0a 2a 2a 20 32 30 31 37 20 41 70 72 69 6c  /*.** 2017 April
0010: 20 32 34 0a 2a 2a 0a 2a 2a 20 54 68 65 20 61 75   24.**.** The au
0020: 74 68 6f 72 20 64 69 73 63 6c 61 69 6d 73 20 63  thor disclaims c
0030: 6f 70 79 72 69 67 68 74 20 74 6f 20 74 68 69 73  opyright to this
0040: 20 73 6f 75 72 63 65 20 63 6f 64 65 2e 20 20 49   source code.  I
0050: 6e 20 70 6c 61 63 65 20 6f 66 0a 2a 2a 20 61 20  n place of.** a 
0060: 6c 65 67 61 6c 20 6e 6f 74 69 63 65 2c 20 68 65  legal notice, he
0070: 72 65 20 69 73 20 61 20 62 6c 65 73 73 69 6e 67  re is a blessing
0080: 3a 0a 2a 2a 0a 2a 2a 20 20 20 20 4d 61 79 20 79  :.**.**    May y
0090: 6f 75 20 64 6f 20 67 6f 6f 64 20 61 6e 64 20 6e  ou do good and n
00a0: 6f 74 20 65 76 69 6c 2e 0a 2a 2a 20 20 20 20 4d  ot evil..**    M
00b0: 61 79 20 79 6f 75 20 66 69 6e 64 20 66 6f 72 67  ay you find forg
00c0: 69 76 65 6e 65 73 73 20 66 6f 72 20 79 6f 75 72  iveness for your
00d0: 73 65 6c 66 20 61 6e 64 20 66 6f 72 67 69 76 65  self and forgive
00e0: 20 6f 74 68 65 72 73 2e 0a 2a 2a 20 20 20 20 4d   others..**    M
00f0: 61 79 20 79 6f 75 20 73 68 61 72 65 20 66 72 65  ay you share fre
0100: 65 6c 79 2c 20 6e 65 76 65 72 20 74 61 6b 69 6e  ely, never takin
0110: 67 20 6d 6f 72 65 20 74 68 61 6e 20 79 6f 75 20  g more than you 
0120: 67 69 76 65 2e 0a 2a 2a 0a 2a 2a 2a 2a 2a 2a 2a  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 0a 2a 2f 0a 0a 23 69 6e 63 6c 75 64 65 20  **.*/..#include 
0180: 22 73 71 6c 69 74 65 49 6e 74 2e 68 22 0a 0a 23  "sqliteInt.h"..#
0190: 69 66 64 65 66 20 53 51 4c 49 54 45 5f 53 45 52  ifdef SQLITE_SER
01a0: 56 45 52 5f 45 44 49 54 49 4f 4e 0a 0a 2f 2a 0a  VER_EDITION../*.
01b0: 2a 2a 20 50 61 67 65 2d 6c 6f 63 6b 69 6e 67 20  ** Page-locking 
01c0: 73 6c 6f 74 20 66 6f 72 6d 61 74 3a 0a 2a 2a 0a  slot format:.**.
01d0: 2a 2a 20 20 20 41 73 73 75 6d 69 6e 67 20 48 4d  **   Assuming HM
01e0: 41 5f 4d 41 58 5f 54 52 41 4e 53 41 43 54 49 4f  A_MAX_TRANSACTIO
01f0: 4e 49 44 20 69 73 20 73 65 74 20 74 6f 20 31 36  NID is set to 16
0200: 2e 0a 2a 2a 0a 2a 2a 20 20 20 54 68 65 20 6c 65  ..**.**   The le
0210: 61 73 74 2d 73 69 67 6e 69 66 69 63 61 6e 74 20  ast-significant 
0220: 31 36 20 62 69 74 73 20 61 72 65 20 75 73 65 64  16 bits are used
0230: 20 66 6f 72 20 72 65 61 64 20 6c 6f 63 6b 73 2e   for read locks.
0240: 20 57 68 65 6e 20 61 20 72 65 61 64 0a 2a 2a 20   When a read.** 
0250: 20 20 6c 6f 63 6b 20 69 73 20 74 61 6b 65 6e 2c    lock is taken,
0260: 20 74 68 65 20 63 6c 69 65 6e 74 20 73 65 74 73   the client sets
0270: 20 74 68 65 20 62 69 74 20 61 73 73 6f 63 69 61   the bit associa
0280: 74 65 64 20 77 69 74 68 20 69 74 73 20 0a 2a 2a  ted with its .**
0290: 20 20 20 74 72 61 6e 73 61 63 74 69 6f 6e 2d 69     transaction-i
02a0: 64 2e 0a 2a 2a 0a 2a 2a 20 20 20 54 68 65 20 6e  d..**.**   The n
02b0: 65 78 74 20 35 20 62 69 74 73 20 61 72 65 20 73  ext 5 bits are s
02c0: 65 74 20 74 6f 20 30 20 69 66 20 6e 6f 20 63 6c  et to 0 if no cl
02d0: 69 65 6e 74 20 63 75 72 72 65 6e 74 6c 79 20 68  ient currently h
02e0: 6f 6c 64 73 20 61 20 77 72 69 74 65 0a 2a 2a 20  olds a write.** 
02f0: 20 20 6c 6f 63 6b 2e 20 4f 72 20 74 6f 20 28 74    lock. Or to (t
0300: 72 61 6e 73 61 63 74 69 6f 6e 2d 69 64 20 2b 20  ransaction-id + 
0310: 31 29 20 69 66 20 61 20 77 72 69 74 65 20 6c 6f  1) if a write lo
0320: 63 6b 20 69 73 20 68 65 6c 64 2e 0a 2a 2a 0a 2a  ck is held..**.*
0330: 2a 20 20 20 54 68 65 20 6e 65 78 74 20 38 20 62  *   The next 8 b
0340: 69 74 73 20 61 72 65 20 73 65 74 20 74 6f 20 74  its are set to t
0350: 68 65 20 6e 75 6d 62 65 72 20 6f 66 20 74 72 61  he number of tra
0360: 6e 73 69 65 6e 74 2d 72 65 61 64 20 6c 6f 63 6b  nsient-read lock
0370: 73 20 0a 2a 2a 20 20 20 63 75 72 72 65 6e 74 6c  s .**   currentl
0380: 79 20 68 65 6c 64 20 6f 6e 20 74 68 65 20 70 61  y held on the pa
0390: 67 65 2e 0a 2a 2f 0a 23 64 65 66 69 6e 65 20 48  ge..*/.#define H
03a0: 4d 41 5f 53 4c 4f 54 5f 52 4c 5f 42 49 54 53 20  MA_SLOT_RL_BITS 
03b0: 31 36 20 20 20 20 20 20 20 2f 2a 20 62 69 74 73  16       /* bits
03c0: 20 66 6f 72 20 52 65 61 64 20 4c 6f 63 6b 73 20   for Read Locks 
03d0: 2a 2f 0a 23 64 65 66 69 6e 65 20 48 4d 41 5f 53  */.#define HMA_S
03e0: 4c 4f 54 5f 57 4c 5f 42 49 54 53 20 35 20 20 20  LOT_WL_BITS 5   
03f0: 20 20 20 20 20 2f 2a 20 62 69 74 73 20 66 6f 72       /* bits for
0400: 20 57 72 69 74 65 20 4c 6f 63 6b 73 20 2a 2f 0a   Write Locks */.
0410: 23 64 65 66 69 6e 65 20 48 4d 41 5f 53 4c 4f 54  #define HMA_SLOT
0420: 5f 54 52 5f 42 49 54 53 20 38 20 20 20 20 20 20  _TR_BITS 8      
0430: 20 20 2f 2a 20 62 69 74 73 20 66 6f 72 20 54 72    /* bits for Tr
0440: 61 6e 73 69 65 6e 74 20 52 65 61 64 65 72 20 6c  ansient Reader l
0450: 6f 63 6b 73 20 2a 2f 0a 0a 23 64 65 66 69 6e 65  ocks */..#define
0460: 20 48 4d 41 5f 53 4c 4f 54 5f 52 4c 57 4c 5f 42   HMA_SLOT_RLWL_B
0470: 49 54 53 20 28 48 4d 41 5f 53 4c 4f 54 5f 52 4c  ITS (HMA_SLOT_RL
0480: 5f 42 49 54 53 20 2b 20 48 4d 41 5f 53 4c 4f 54  _BITS + HMA_SLOT
0490: 5f 57 4c 5f 42 49 54 53 29 0a 0a 0a 23 64 65 66  _WL_BITS)...#def
04a0: 69 6e 65 20 48 4d 41 5f 53 4c 4f 54 5f 52 4c 5f  ine HMA_SLOT_RL_
04b0: 4d 41 53 4b 20 28 28 31 20 3c 3c 20 48 4d 41 5f  MASK ((1 << HMA_
04c0: 53 4c 4f 54 5f 52 4c 5f 42 49 54 53 29 2d 31 29  SLOT_RL_BITS)-1)
04d0: 0a 23 64 65 66 69 6e 65 20 48 4d 41 5f 53 4c 4f  .#define HMA_SLO
04e0: 54 5f 57 4c 5f 4d 41 53 4b 20 28 28 28 31 20 3c  T_WL_MASK (((1 <
04f0: 3c 20 48 4d 41 5f 53 4c 4f 54 5f 57 4c 5f 42 49  < HMA_SLOT_WL_BI
0500: 54 53 29 2d 31 29 20 3c 3c 20 48 4d 41 5f 53 4c  TS)-1) << HMA_SL
0510: 4f 54 5f 52 4c 5f 42 49 54 53 29 0a 23 64 65 66  OT_RL_BITS).#def
0520: 69 6e 65 20 48 4d 41 5f 53 4c 4f 54 5f 54 52 5f  ine HMA_SLOT_TR_
0530: 4d 41 53 4b 20 28 28 28 31 20 3c 3c 20 48 4d 41  MASK (((1 << HMA
0540: 5f 53 4c 4f 54 5f 54 52 5f 42 49 54 53 29 2d 31  _SLOT_TR_BITS)-1
0550: 29 20 3c 3c 20 48 4d 41 5f 53 4c 4f 54 5f 52 4c  ) << HMA_SLOT_RL
0560: 57 4c 5f 42 49 54 53 29 0a 0a 0a 2f 2a 20 4e 75  WL_BITS).../* Nu
0570: 6d 62 65 72 20 6f 66 20 70 61 67 65 2d 6c 6f 63  mber of page-loc
0580: 6b 69 6e 67 20 73 6c 6f 74 73 20 2a 2f 0a 23 64  king slots */.#d
0590: 65 66 69 6e 65 20 48 4d 41 5f 50 41 47 45 4c 4f  efine HMA_PAGELO
05a0: 43 4b 5f 53 4c 4f 54 53 20 28 32 35 36 2a 31 30  CK_SLOTS (256*10
05b0: 32 34 29 0a 0a 2f 2a 20 4d 61 78 69 6d 75 6d 20  24)../* Maximum 
05c0: 63 6f 6e 63 75 72 72 65 6e 74 20 72 65 61 64 2f  concurrent read/
05d0: 77 72 69 74 65 20 74 72 61 6e 73 61 63 74 69 6f  write transactio
05e0: 6e 73 20 2a 2f 0a 23 64 65 66 69 6e 65 20 48 4d  ns */.#define HM
05f0: 41 5f 4d 41 58 5f 54 52 41 4e 53 41 43 54 49 4f  A_MAX_TRANSACTIO
0600: 4e 49 44 20 31 36 0a 0a 0a 23 64 65 66 69 6e 65  NID 16...#define
0610: 20 48 4d 41 5f 48 41 53 48 5f 53 49 5a 45 20 35   HMA_HASH_SIZE 5
0620: 31 32 0a 0a 2f 2a 0a 2a 2a 20 54 68 65 20 61 72  12../*.** The ar
0630: 67 75 6d 65 6e 74 20 74 6f 20 74 68 69 73 20 6d  gument to this m
0640: 61 63 72 6f 20 69 73 20 74 68 65 20 76 61 6c 75  acro is the valu
0650: 65 20 6f 66 20 61 20 6c 6f 63 6b 69 6e 67 20 73  e of a locking s
0660: 6c 6f 74 2e 20 49 74 20 72 65 74 75 72 6e 73 0a  lot. It returns.
0670: 2a 2a 20 2d 31 20 69 66 20 6e 6f 20 63 6c 69 65  ** -1 if no clie
0680: 6e 74 20 63 75 72 72 65 6e 74 6c 79 20 68 6f 6c  nt currently hol
0690: 64 73 20 74 68 65 20 77 72 69 74 65 20 6c 6f 63  ds the write loc
06a0: 6b 2c 20 6f 72 20 74 68 65 20 74 72 61 6e 73 61  k, or the transa
06b0: 63 74 69 6f 6e 2d 69 64 0a 2a 2a 20 6f 66 20 74  ction-id.** of t
06c0: 68 65 20 6c 6f 63 6b 65 72 20 6f 74 68 65 72 77  he locker otherw
06d0: 69 73 65 2e 0a 2a 2f 0a 23 64 65 66 69 6e 65 20  ise..*/.#define 
06e0: 73 6c 6f 74 47 65 74 57 72 69 74 65 72 28 76 29  slotGetWriter(v)
06f0: 20 28 28 28 28 69 6e 74 29 28 76 29 26 48 4d 41   ((((int)(v)&HMA
0700: 5f 53 4c 4f 54 5f 57 4c 5f 4d 41 53 4b 29 20 3e  _SLOT_WL_MASK) >
0710: 3e 20 48 4d 41 5f 53 4c 4f 54 5f 52 4c 5f 42 49  > HMA_SLOT_RL_BI
0720: 54 53 29 20 2d 20 31 29 0a 0a 2f 2a 0a 2a 2a 20  TS) - 1)../*.** 
0730: 54 68 65 20 61 72 67 75 6d 65 6e 74 20 74 6f 20  The argument to 
0740: 74 68 69 73 20 6d 61 63 72 6f 20 69 73 20 74 68  this macro is th
0750: 65 20 76 61 6c 75 65 20 6f 66 20 61 20 6c 6f 63  e value of a loc
0760: 6b 69 6e 67 20 73 6c 6f 74 2e 20 54 68 69 73 20  king slot. This 
0770: 6d 61 63 72 6f 0a 2a 2a 20 72 65 74 75 72 6e 73  macro.** returns
0780: 20 74 68 65 20 63 75 72 72 65 6e 74 20 6e 75 6d   the current num
0790: 62 65 72 20 6f 66 20 73 6c 6f 77 20 72 65 61 64  ber of slow read
07a0: 65 72 20 63 6c 69 65 6e 74 73 20 72 65 61 64 69  er clients readi
07b0: 6e 67 20 74 68 65 20 70 61 67 65 2e 0a 2a 2f 0a  ng the page..*/.
07c0: 23 64 65 66 69 6e 65 20 73 6c 6f 74 47 65 74 53  #define slotGetS
07d0: 6c 6f 77 52 65 61 64 65 72 73 28 76 29 20 28 28  lowReaders(v) ((
07e0: 28 76 29 20 26 20 48 4d 41 5f 53 4c 4f 54 5f 54  (v) & HMA_SLOT_T
07f0: 52 5f 4d 41 53 4b 29 20 3e 3e 20 48 4d 41 5f 53  R_MASK) >> HMA_S
0800: 4c 4f 54 5f 52 4c 57 4c 5f 42 49 54 53 29 0a 0a  LOT_RLWL_BITS)..
0810: 23 64 65 66 69 6e 65 20 73 6c 6f 74 52 65 61 64  #define slotRead
0820: 65 72 4d 61 73 6b 28 76 29 20 28 28 76 29 20 26  erMask(v) ((v) &
0830: 20 48 4d 41 5f 53 4c 4f 54 5f 52 4c 5f 4d 41 53   HMA_SLOT_RL_MAS
0840: 4b 29 0a 0a 23 69 6e 63 6c 75 64 65 20 22 75 6e  K)..#include "un
0850: 69 73 74 64 2e 68 22 0a 23 69 6e 63 6c 75 64 65  istd.h".#include
0860: 20 22 66 63 6e 74 6c 2e 68 22 0a 23 69 6e 63 6c   "fcntl.h".#incl
0870: 75 64 65 20 22 73 79 73 2f 6d 6d 61 6e 2e 68 22  ude "sys/mman.h"
0880: 0a 23 69 6e 63 6c 75 64 65 20 22 73 79 73 2f 74  .#include "sys/t
0890: 79 70 65 73 2e 68 22 0a 23 69 6e 63 6c 75 64 65  ypes.h".#include
08a0: 20 22 73 79 73 2f 73 74 61 74 2e 68 22 0a 23 69   "sys/stat.h".#i
08b0: 6e 63 6c 75 64 65 20 22 65 72 72 6e 6f 2e 68 22  nclude "errno.h"
08c0: 0a 0a 74 79 70 65 64 65 66 20 73 74 72 75 63 74  ..typedef struct
08d0: 20 53 65 72 76 65 72 44 62 20 53 65 72 76 65 72   ServerDb Server
08e0: 44 62 3b 0a 74 79 70 65 64 65 66 20 73 74 72 75  Db;.typedef stru
08f0: 63 74 20 53 65 72 76 65 72 4a 6f 75 72 6e 61 6c  ct ServerJournal
0900: 20 53 65 72 76 65 72 4a 6f 75 72 6e 61 6c 3b 0a   ServerJournal;.
0910: 0a 73 74 72 75 63 74 20 53 65 72 76 65 72 47 6c  .struct ServerGl
0920: 6f 62 61 6c 20 7b 0a 20 20 53 65 72 76 65 72 44  obal {.  ServerD
0930: 62 20 2a 70 44 62 3b 20 20 20 20 20 20 20 20 20  b *pDb;         
0940: 20 20 20 20 20 20 20 20 20 2f 2a 20 4c 69 6e 6b           /* Link
0950: 65 64 20 6c 69 73 74 20 6f 66 20 61 6c 6c 20 53  ed list of all S
0960: 65 72 76 65 72 48 4d 41 20 6f 62 6a 65 63 74 73  erverHMA objects
0970: 20 2a 2f 0a 7d 3b 0a 73 74 61 74 69 63 20 73 74   */.};.static st
0980: 72 75 63 74 20 53 65 72 76 65 72 47 6c 6f 62 61  ruct ServerGloba
0990: 6c 20 67 5f 73 65 72 76 65 72 3b 0a 0a 73 74 72  l g_server;..str
09a0: 75 63 74 20 53 65 72 76 65 72 4a 6f 75 72 6e 61  uct ServerJourna
09b0: 6c 20 7b 0a 20 20 63 68 61 72 20 2a 7a 4a 6f 75  l {.  char *zJou
09c0: 72 6e 61 6c 3b 0a 20 20 73 71 6c 69 74 65 33 5f  rnal;.  sqlite3_
09d0: 66 69 6c 65 20 2a 6a 66 64 3b 0a 7d 3b 0a 0a 2f  file *jfd;.};../
09e0: 2a 0a 2a 2a 20 54 68 65 72 65 20 69 73 20 6f 6e  *.** There is on
09f0: 65 20 69 6e 73 74 61 6e 63 65 20 6f 66 20 74 68  e instance of th
0a00: 65 20 66 6f 6c 6c 6f 77 69 6e 67 20 73 74 72 75  e following stru
0a10: 63 74 75 72 65 20 66 6f 72 20 65 61 63 68 20 64  cture for each d
0a20: 69 73 74 69 6e 63 74 20 0a 2a 2a 20 64 61 74 61  istinct .** data
0a30: 62 61 73 65 20 66 69 6c 65 20 6f 70 65 6e 65 64  base file opened
0a40: 20 69 6e 20 73 65 72 76 65 72 20 6d 6f 64 65 20   in server mode 
0a50: 62 79 20 74 68 69 73 20 70 72 6f 63 65 73 73 2e  by this process.
0a60: 0a 2a 2f 0a 73 74 72 75 63 74 20 53 65 72 76 65  .*/.struct Serve
0a70: 72 44 62 20 7b 0a 20 20 73 71 6c 69 74 65 33 5f  rDb {.  sqlite3_
0a80: 6d 75 74 65 78 20 2a 6d 75 74 65 78 3b 20 20 20  mutex *mutex;   
0a90: 20 20 20 20 20 20 20 20 2f 2a 20 4e 6f 6e 2d 72          /* Non-r
0aa0: 65 63 75 72 73 69 76 65 20 6d 75 74 65 78 20 2a  ecursive mutex *
0ab0: 2f 0a 20 20 69 6e 74 20 6e 43 6c 69 65 6e 74 3b  /.  int nClient;
0ac0: 20 20 20 20 20 20 20 20 20 20 20 20 20 20 20 20                  
0ad0: 20 20 20 20 2f 2a 20 43 75 72 72 65 6e 74 20 6e      /* Current n
0ae0: 75 6d 62 65 72 20 6f 66 20 63 6c 69 65 6e 74 73  umber of clients
0af0: 20 2a 2f 0a 20 20 69 6e 74 20 62 49 6e 69 74 3b   */.  int bInit;
0b00: 20 20 20 20 20 20 20 20 20 20 20 20 20 20 20 20                  
0b10: 20 20 20 20 20 20 2f 2a 20 54 72 75 65 20 6f 6e        /* True on
0b20: 63 65 20 69 6e 69 74 69 61 6c 69 7a 65 64 20 2a  ce initialized *
0b30: 2f 0a 20 20 75 33 32 20 74 72 61 6e 73 6d 61 73  /.  u32 transmas
0b40: 6b 3b 20 20 20 20 20 20 20 20 20 20 20 20 20 20  k;              
0b50: 20 20 20 20 2f 2a 20 42 69 74 6d 61 73 6b 20 6f      /* Bitmask o
0b60: 66 20 74 61 6b 65 6e 20 74 72 61 6e 73 61 63 74  f taken transact
0b70: 69 6f 6e 20 69 64 73 20 2a 2f 0a 20 20 75 33 32  ion ids */.  u32
0b80: 20 2a 61 53 6c 6f 74 3b 20 20 20 20 20 20 20 20   *aSlot;        
0b90: 20 20 20 20 20 20 20 20 20 20 20 20 20 2f 2a 20               /* 
0ba0: 41 72 72 61 79 20 6f 66 20 70 61 67 65 20 6c 6f  Array of page lo
0bb0: 63 6b 69 6e 67 20 73 6c 6f 74 73 20 2a 2f 0a 20  cking slots */. 
0bc0: 20 69 36 34 20 61 46 69 6c 65 49 64 5b 32 5d 3b   i64 aFileId[2];
0bd0: 20 20 20 20 20 20 20 20 20 20 20 20 20 20 20 20                  
0be0: 20 2f 2a 20 4f 70 61 71 75 65 20 56 46 53 20 66   /* Opaque VFS f
0bf0: 69 6c 65 2d 69 64 20 2a 2f 0a 20 20 53 65 72 76  ile-id */.  Serv
0c00: 65 72 44 62 20 2a 70 4e 65 78 74 3b 20 20 20 20  erDb *pNext;    
0c10: 20 20 20 20 20 20 20 20 20 20 20 20 2f 2a 20 4e              /* N
0c20: 65 78 74 20 64 62 20 69 6e 20 74 68 69 73 20 70  ext db in this p
0c30: 72 6f 63 65 73 73 20 2a 2f 0a 0a 20 20 73 71 6c  rocess */..  sql
0c40: 69 74 65 33 5f 76 66 73 20 2a 70 56 66 73 3b 0a  ite3_vfs *pVfs;.
0c50: 20 20 53 65 72 76 65 72 4a 6f 75 72 6e 61 6c 20    ServerJournal 
0c60: 61 4a 72 6e 6c 5b 48 4d 41 5f 4d 41 58 5f 54 52  aJrnl[HMA_MAX_TR
0c70: 41 4e 53 41 43 54 49 4f 4e 49 44 5d 3b 0a 20 20  ANSACTIONID];.  
0c80: 75 38 20 2a 61 4a 72 6e 6c 46 64 53 70 61 63 65  u8 *aJrnlFdSpace
0c90: 3b 0a 0a 20 20 69 6e 74 20 69 4e 65 78 74 43 6f  ;..  int iNextCo
0ca0: 6d 6d 69 74 3b 20 20 20 20 20 20 20 20 20 20 20  mmit;           
0cb0: 20 20 20 20 20 2f 2a 20 43 6f 6d 6d 69 74 20 69       /* Commit i
0cc0: 64 20 66 6f 72 20 6e 65 78 74 20 70 72 65 2d 63  d for next pre-c
0cd0: 6f 6d 6d 69 74 20 63 61 6c 6c 20 2a 2f 20 0a 20  ommit call */ . 
0ce0: 20 53 65 72 76 65 72 20 2a 70 43 6f 6d 6d 69 74   Server *pCommit
0cf0: 3b 20 20 20 20 20 20 20 20 20 20 20 20 20 20 20  ;               
0d00: 20 2f 2a 20 4c 69 73 74 20 6f 66 20 63 6f 6e 6e   /* List of conn
0d10: 65 63 74 69 6f 6e 73 20 63 75 72 72 65 6e 74 6c  ections currentl
0d20: 79 20 63 6f 6d 6d 69 74 69 6e 67 20 2a 2f 0a 20  y commiting */. 
0d30: 20 53 65 72 76 65 72 20 2a 70 52 65 61 64 65 72   Server *pReader
0d40: 3b 20 20 20 20 20 20 20 20 20 20 20 20 20 20 20  ;               
0d50: 20 2f 2a 20 43 6f 6e 6e 65 63 74 69 6f 6e 73 20   /* Connections 
0d60: 69 6e 20 73 6c 6f 77 65 72 2d 72 65 61 64 65 72  in slower-reader
0d70: 20 74 72 61 6e 73 61 63 74 69 6f 6e 20 2a 2f 0a   transaction */.
0d80: 20 20 53 65 72 76 65 72 50 61 67 65 20 2a 70 50    ServerPage *pP
0d90: 67 46 69 72 73 74 3b 20 20 20 20 20 20 20 20 20  gFirst;         
0da0: 20 20 2f 2a 20 46 69 72 73 74 20 28 6f 6c 64 65    /* First (olde
0db0: 73 74 29 20 69 6e 20 6c 69 73 74 20 6f 66 20 70  st) in list of p
0dc0: 61 67 65 73 20 2a 2f 0a 20 20 53 65 72 76 65 72  ages */.  Server
0dd0: 50 61 67 65 20 2a 70 50 67 4c 61 73 74 3b 20 20  Page *pPgLast;  
0de0: 20 20 20 20 20 20 20 20 20 20 2f 2a 20 4c 61 73            /* Las
0df0: 74 20 28 6e 65 77 65 73 74 29 20 69 6e 20 6c 69  t (newest) in li
0e00: 73 74 20 6f 66 20 70 61 67 65 73 20 2a 2f 0a 20  st of pages */. 
0e10: 20 53 65 72 76 65 72 50 61 67 65 20 2a 61 70 50   ServerPage *apP
0e20: 67 5b 48 4d 41 5f 48 41 53 48 5f 53 49 5a 45 5d  g[HMA_HASH_SIZE]
0e30: 3b 0a 0a 20 20 53 65 72 76 65 72 50 61 67 65 20  ;..  ServerPage 
0e40: 2a 70 46 72 65 65 3b 20 20 20 20 20 20 20 20 20  *pFree;         
0e50: 20 20 20 20 20 2f 2a 20 4c 69 73 74 20 6f 66 20       /* List of 
0e60: 66 72 65 65 20 70 61 67 65 20 62 75 66 66 65 72  free page buffer
0e70: 73 20 2a 2f 0a 7d 3b 0a 0a 2f 2a 0a 2a 2a 20 4f  s */.};../*.** O
0e80: 6e 63 65 20 69 6e 73 74 61 6e 63 65 20 66 6f 72  nce instance for
0e90: 20 65 61 63 68 20 63 6c 69 65 6e 74 20 63 6f 6e   each client con
0ea0: 6e 65 63 74 69 6f 6e 20 6f 70 65 6e 20 6f 6e 20  nection open on 
0eb0: 61 20 73 65 72 76 65 72 20 6d 6f 64 65 20 64 61  a server mode da
0ec0: 74 61 62 61 73 65 0a 2a 2a 20 69 6e 20 74 68 69  tabase.** in thi
0ed0: 73 20 70 72 6f 63 65 73 73 2e 0a 2a 2f 0a 73 74  s process..*/.st
0ee0: 72 75 63 74 20 53 65 72 76 65 72 20 7b 0a 20 20  ruct Server {.  
0ef0: 53 65 72 76 65 72 44 62 20 2a 70 44 62 3b 20 20  ServerDb *pDb;  
0f00: 20 20 20 20 20 20 20 20 20 20 20 20 20 20 20 20                  
0f10: 2f 2a 20 44 61 74 61 62 61 73 65 20 6f 62 6a 65  /* Database obje
0f20: 63 74 20 2a 2f 0a 20 20 50 61 67 65 72 20 2a 70  ct */.  Pager *p
0f30: 50 61 67 65 72 3b 20 20 20 20 20 20 20 20 20 20  Pager;          
0f40: 20 20 20 20 20 20 20 20 2f 2a 20 41 73 73 6f 63          /* Assoc
0f50: 69 61 74 65 64 20 70 61 67 65 72 20 6f 62 6a 65  iated pager obje
0f60: 63 74 20 2a 2f 0a 20 20 69 6e 74 20 65 54 72 61  ct */.  int eTra
0f70: 6e 73 3b 20 20 20 20 20 20 20 20 20 20 20 20 20  ns;             
0f80: 20 20 20 20 20 20 20 20 2f 2a 20 4f 6e 65 20 6f          /* One o
0f90: 66 20 74 68 65 20 53 45 52 56 45 52 5f 54 52 41  f the SERVER_TRA
0fa0: 4e 53 5f 78 78 78 20 76 61 6c 75 65 73 20 2a 2f  NS_xxx values */
0fb0: 20 0a 20 20 69 6e 74 20 69 54 72 61 6e 73 49 64   .  int iTransId
0fc0: 3b 20 20 20 20 20 20 20 20 20 20 20 20 20 20 20  ;               
0fd0: 20 20 20 20 2f 2a 20 43 75 72 72 65 6e 74 20 74      /* Current t
0fe0: 72 61 6e 73 61 63 74 69 6f 6e 20 69 64 20 28 6f  ransaction id (o
0ff0: 72 20 2d 31 29 20 2a 2f 0a 20 20 69 6e 74 20 69  r -1) */.  int i
1000: 43 6f 6d 6d 69 74 49 64 3b 20 20 20 20 20 20 20  CommitId;       
1010: 20 20 20 20 20 20 20 20 20 20 20 2f 2a 20 43 75             /* Cu
1020: 72 72 65 6e 74 20 63 6f 6d 6d 69 74 20 69 64 20  rrent commit id 
1030: 28 6f 72 20 30 29 20 2a 2f 0a 20 20 69 6e 74 20  (or 0) */.  int 
1040: 6e 41 6c 6c 6f 63 3b 20 20 20 20 20 20 20 20 20  nAlloc;         
1050: 20 20 20 20 20 20 20 20 20 20 20 20 2f 2a 20 41              /* A
1060: 6c 6c 6f 63 61 74 65 64 20 73 69 7a 65 20 6f 66  llocated size of
1070: 20 61 4c 6f 63 6b 5b 5d 20 61 72 72 61 79 20 2a   aLock[] array *
1080: 2f 0a 20 20 69 6e 74 20 6e 4c 6f 63 6b 3b 20 20  /.  int nLock;  
1090: 20 20 20 20 20 20 20 20 20 20 20 20 20 20 20 20                  
10a0: 20 20 20 20 2f 2a 20 4e 75 6d 62 65 72 20 6f 66      /* Number of
10b0: 20 65 6e 74 72 69 65 73 20 69 6e 20 61 4c 6f 63   entries in aLoc
10c0: 6b 5b 5d 20 2a 2f 0a 20 20 75 33 32 20 2a 61 4c  k[] */.  u32 *aL
10d0: 6f 63 6b 3b 20 20 20 20 20 20 20 20 20 20 20 20  ock;            
10e0: 20 20 20 20 20 20 20 20 20 2f 2a 20 4d 61 70 70           /* Mapp
10f0: 65 64 20 6c 6f 63 6b 20 66 69 6c 65 20 2a 2f 0a  ed lock file */.
1100: 20 20 53 65 72 76 65 72 20 2a 70 4e 65 78 74 3b    Server *pNext;
1110: 20 20 20 20 20 20 20 20 20 20 20 20 20 20 20 20                  
1120: 20 20 2f 2a 20 4e 65 78 74 20 69 6e 20 70 43 6f    /* Next in pCo
1130: 6d 6d 69 74 20 6f 72 20 70 52 65 61 64 65 72 20  mmit or pReader 
1140: 6c 69 73 74 20 2a 2f 0a 7d 3b 0a 0a 2f 2a 0a 2a  list */.};../*.*
1150: 2a 20 50 6f 73 73 69 62 6c 65 20 76 61 6c 75 65  * Possible value
1160: 73 20 66 6f 72 20 53 65 72 76 65 72 2e 65 54 72  s for Server.eTr
1170: 61 6e 73 2e 0a 2a 2f 0a 23 64 65 66 69 6e 65 20  ans..*/.#define 
1180: 53 45 52 56 45 52 5f 54 52 41 4e 53 5f 4e 4f 4e  SERVER_TRANS_NON
1190: 45 20 20 20 20 20 20 30 0a 23 64 65 66 69 6e 65  E      0.#define
11a0: 20 53 45 52 56 45 52 5f 54 52 41 4e 53 5f 52 45   SERVER_TRANS_RE
11b0: 41 44 4f 4e 4c 59 20 20 31 0a 23 64 65 66 69 6e  ADONLY  1.#defin
11c0: 65 20 53 45 52 56 45 52 5f 54 52 41 4e 53 5f 52  e SERVER_TRANS_R
11d0: 45 41 44 57 52 49 54 45 20 32 0a 0a 23 64 65 66  EADWRITE 2..#def
11e0: 69 6e 65 20 53 45 52 56 45 52 5f 57 52 49 54 45  ine SERVER_WRITE
11f0: 5f 4c 4f 43 4b 20 33 0a 23 64 65 66 69 6e 65 20  _LOCK 3.#define 
1200: 53 45 52 56 45 52 5f 52 45 41 44 5f 4c 4f 43 4b  SERVER_READ_LOCK
1210: 20 20 32 0a 23 64 65 66 69 6e 65 20 53 45 52 56    2.#define SERV
1220: 45 52 5f 4e 4f 5f 4c 4f 43 4b 20 20 20 20 31 0a  ER_NO_LOCK    1.
1230: 0a 2f 2a 0a 2a 2a 20 47 6c 6f 62 61 6c 20 6d 75  ./*.** Global mu
1240: 74 65 78 20 66 75 6e 63 74 69 6f 6e 73 20 75 73  tex functions us
1250: 65 64 20 62 79 20 63 6f 64 65 20 69 6e 20 74 68  ed by code in th
1260: 69 73 20 66 69 6c 65 2e 0a 2a 2f 0a 73 74 61 74  is file..*/.stat
1270: 69 63 20 76 6f 69 64 20 73 65 72 76 65 72 45 6e  ic void serverEn
1280: 74 65 72 4d 75 74 65 78 28 76 6f 69 64 29 7b 0a  terMutex(void){.
1290: 20 20 73 71 6c 69 74 65 33 5f 6d 75 74 65 78 5f    sqlite3_mutex_
12a0: 65 6e 74 65 72 28 73 71 6c 69 74 65 33 4d 75 74  enter(sqlite3Mut
12b0: 65 78 41 6c 6c 6f 63 28 53 51 4c 49 54 45 5f 4d  exAlloc(SQLITE_M
12c0: 55 54 45 58 5f 53 54 41 54 49 43 5f 41 50 50 31  UTEX_STATIC_APP1
12d0: 29 29 3b 0a 7d 0a 73 74 61 74 69 63 20 76 6f 69  ));.}.static voi
12e0: 64 20 73 65 72 76 65 72 4c 65 61 76 65 4d 75 74  d serverLeaveMut
12f0: 65 78 28 76 6f 69 64 29 7b 0a 20 20 73 71 6c 69  ex(void){.  sqli
1300: 74 65 33 5f 6d 75 74 65 78 5f 6c 65 61 76 65 28  te3_mutex_leave(
1310: 73 71 6c 69 74 65 33 4d 75 74 65 78 41 6c 6c 6f  sqlite3MutexAllo
1320: 63 28 53 51 4c 49 54 45 5f 4d 55 54 45 58 5f 53  c(SQLITE_MUTEX_S
1330: 54 41 54 49 43 5f 41 50 50 31 29 29 3b 0a 7d 0a  TATIC_APP1));.}.
1340: 73 74 61 74 69 63 20 76 6f 69 64 20 73 65 72 76  static void serv
1350: 65 72 41 73 73 65 72 74 4d 75 74 65 78 48 65 6c  erAssertMutexHel
1360: 64 28 76 6f 69 64 29 7b 0a 20 20 61 73 73 65 72  d(void){.  asser
1370: 74 28 20 73 71 6c 69 74 65 33 5f 6d 75 74 65 78  t( sqlite3_mutex
1380: 5f 68 65 6c 64 28 73 71 6c 69 74 65 33 4d 75 74  _held(sqlite3Mut
1390: 65 78 41 6c 6c 6f 63 28 53 51 4c 49 54 45 5f 4d  exAlloc(SQLITE_M
13a0: 55 54 45 58 5f 53 54 41 54 49 43 5f 41 50 50 31  UTEX_STATIC_APP1
13b0: 29 29 20 29 3b 0a 7d 0a 0a 73 74 61 74 69 63 20  )) );.}..static 
13c0: 69 6e 74 20 73 65 72 76 65 72 46 69 6e 64 44 61  int serverFindDa
13d0: 74 61 62 61 73 65 28 53 65 72 76 65 72 20 2a 70  tabase(Server *p
13e0: 4e 65 77 2c 20 69 36 34 20 2a 61 46 69 6c 65 49  New, i64 *aFileI
13f0: 64 29 7b 0a 20 20 53 65 72 76 65 72 44 62 20 2a  d){.  ServerDb *
1400: 70 3b 0a 20 20 69 6e 74 20 72 63 20 3d 20 53 51  p;.  int rc = SQ
1410: 4c 49 54 45 5f 4f 4b 3b 0a 20 20 73 65 72 76 65  LITE_OK;.  serve
1420: 72 45 6e 74 65 72 4d 75 74 65 78 28 29 3b 0a 20  rEnterMutex();. 
1430: 20 66 6f 72 28 70 3d 67 5f 73 65 72 76 65 72 2e   for(p=g_server.
1440: 70 44 62 3b 20 70 3b 20 70 3d 70 2d 3e 70 4e 65  pDb; p; p=p->pNe
1450: 78 74 29 7b 0a 20 20 20 20 69 66 28 20 70 2d 3e  xt){.    if( p->
1460: 61 46 69 6c 65 49 64 5b 30 5d 3d 3d 61 46 69 6c  aFileId[0]==aFil
1470: 65 49 64 5b 30 5d 20 26 26 20 70 2d 3e 61 46 69  eId[0] && p->aFi
1480: 6c 65 49 64 5b 31 5d 3d 3d 61 46 69 6c 65 49 64  leId[1]==aFileId
1490: 5b 31 5d 20 29 7b 0a 20 20 20 20 20 20 62 72 65  [1] ){.      bre
14a0: 61 6b 3b 0a 20 20 20 20 7d 0a 20 20 7d 0a 20 20  ak;.    }.  }.  
14b0: 69 66 28 20 70 3d 3d 30 20 29 7b 0a 20 20 20 20  if( p==0 ){.    
14c0: 70 20 3d 20 28 53 65 72 76 65 72 44 62 2a 29 73  p = (ServerDb*)s
14d0: 71 6c 69 74 65 33 4d 61 6c 6c 6f 63 5a 65 72 6f  qlite3MallocZero
14e0: 28 73 69 7a 65 6f 66 28 53 65 72 76 65 72 44 62  (sizeof(ServerDb
14f0: 29 29 3b 0a 20 20 20 20 69 66 28 20 70 20 29 7b  ));.    if( p ){
1500: 0a 20 20 20 20 20 20 70 2d 3e 61 53 6c 6f 74 20  .      p->aSlot 
1510: 3d 20 28 75 33 32 2a 29 73 71 6c 69 74 65 33 4d  = (u32*)sqlite3M
1520: 61 6c 6c 6f 63 5a 65 72 6f 28 73 69 7a 65 6f 66  allocZero(sizeof
1530: 28 75 33 32 29 2a 48 4d 41 5f 50 41 47 45 4c 4f  (u32)*HMA_PAGELO
1540: 43 4b 5f 53 4c 4f 54 53 29 3b 0a 20 20 20 20 20  CK_SLOTS);.     
1550: 20 69 66 28 20 70 2d 3e 61 53 6c 6f 74 3d 3d 30   if( p->aSlot==0
1560: 20 29 7b 0a 20 20 20 20 20 20 20 20 72 63 20 3d   ){.        rc =
1570: 20 53 51 4c 49 54 45 5f 4e 4f 4d 45 4d 5f 42 4b   SQLITE_NOMEM_BK
1580: 50 54 3b 0a 20 20 20 20 20 20 7d 65 6c 73 65 7b  PT;.      }else{
1590: 0a 20 20 20 20 20 20 20 20 70 2d 3e 6d 75 74 65  .        p->mute
15a0: 78 20 3d 20 73 71 6c 69 74 65 33 5f 6d 75 74 65  x = sqlite3_mute
15b0: 78 5f 61 6c 6c 6f 63 28 53 51 4c 49 54 45 5f 4d  x_alloc(SQLITE_M
15c0: 55 54 45 58 5f 46 41 53 54 29 3b 0a 23 69 66 20  UTEX_FAST);.#if 
15d0: 53 51 4c 49 54 45 5f 54 48 52 45 41 44 53 41 46  SQLITE_THREADSAF
15e0: 45 21 3d 30 0a 20 20 20 20 20 20 20 20 69 66 28  E!=0.        if(
15f0: 20 70 2d 3e 6d 75 74 65 78 3d 3d 30 20 29 20 72   p->mutex==0 ) r
1600: 63 20 3d 20 53 51 4c 49 54 45 5f 4e 4f 4d 45 4d  c = SQLITE_NOMEM
1610: 5f 42 4b 50 54 3b 0a 23 65 6e 64 69 66 0a 20 20  _BKPT;.#endif.  
1620: 20 20 20 20 7d 0a 0a 20 20 20 20 20 20 69 66 28      }..      if(
1630: 20 72 63 3d 3d 53 51 4c 49 54 45 5f 4e 4f 4d 45   rc==SQLITE_NOME
1640: 4d 20 29 7b 0a 20 20 20 20 20 20 20 20 73 71 6c  M ){.        sql
1650: 69 74 65 33 5f 66 72 65 65 28 70 2d 3e 61 53 6c  ite3_free(p->aSl
1660: 6f 74 29 3b 0a 20 20 20 20 20 20 20 20 73 71 6c  ot);.        sql
1670: 69 74 65 33 5f 66 72 65 65 28 70 29 3b 0a 20 20  ite3_free(p);.  
1680: 20 20 20 20 20 20 70 20 3d 20 30 3b 0a 20 20 20        p = 0;.   
1690: 20 20 20 7d 65 6c 73 65 7b 0a 20 20 20 20 20 20     }else{.      
16a0: 20 20 70 2d 3e 6e 43 6c 69 65 6e 74 20 3d 20 31    p->nClient = 1
16b0: 3b 0a 20 20 20 20 20 20 20 20 70 2d 3e 69 4e 65  ;.        p->iNe
16c0: 78 74 43 6f 6d 6d 69 74 20 3d 20 31 3b 0a 20 20  xtCommit = 1;.  
16d0: 20 20 20 20 20 20 70 2d 3e 61 46 69 6c 65 49 64        p->aFileId
16e0: 5b 30 5d 20 3d 20 61 46 69 6c 65 49 64 5b 30 5d  [0] = aFileId[0]
16f0: 3b 0a 20 20 20 20 20 20 20 20 70 2d 3e 61 46 69  ;.        p->aFi
1700: 6c 65 49 64 5b 31 5d 20 3d 20 61 46 69 6c 65 49  leId[1] = aFileI
1710: 64 5b 31 5d 3b 0a 20 20 20 20 20 20 20 20 70 2d  d[1];.        p-
1720: 3e 70 4e 65 78 74 20 3d 20 67 5f 73 65 72 76 65  >pNext = g_serve
1730: 72 2e 70 44 62 3b 0a 20 20 20 20 20 20 20 20 67  r.pDb;.        g
1740: 5f 73 65 72 76 65 72 2e 70 44 62 20 3d 20 70 3b  _server.pDb = p;
1750: 0a 20 20 20 20 20 20 7d 0a 20 20 20 20 7d 65 6c  .      }.    }el
1760: 73 65 7b 0a 20 20 20 20 20 20 72 63 20 3d 20 53  se{.      rc = S
1770: 51 4c 49 54 45 5f 4e 4f 4d 45 4d 5f 42 4b 50 54  QLITE_NOMEM_BKPT
1780: 3b 0a 20 20 20 20 7d 0a 20 20 7d 65 6c 73 65 7b  ;.    }.  }else{
1790: 0a 20 20 20 20 70 2d 3e 6e 43 6c 69 65 6e 74 2b  .    p->nClient+
17a0: 2b 3b 0a 20 20 7d 0a 20 20 70 4e 65 77 2d 3e 70  +;.  }.  pNew->p
17b0: 44 62 20 3d 20 70 3b 0a 20 20 73 65 72 76 65 72  Db = p;.  server
17c0: 4c 65 61 76 65 4d 75 74 65 78 28 29 3b 0a 20 20  LeaveMutex();.  
17d0: 72 65 74 75 72 6e 20 72 63 3b 0a 7d 0a 0a 2f 2a  return rc;.}../*
17e0: 0a 2a 2a 20 46 72 65 65 20 61 6c 6c 20 72 65 73  .** Free all res
17f0: 6f 75 72 63 65 73 20 61 6c 6c 6f 63 61 74 65 64  ources allocated
1800: 20 62 79 20 73 65 72 76 65 72 49 6e 69 74 44 61   by serverInitDa
1810: 74 61 62 61 73 65 28 29 20 61 73 73 6f 63 69 61  tabase() associa
1820: 74 65 64 20 77 69 74 68 20 74 68 65 0a 2a 2a 20  ted with the.** 
1830: 6f 62 6a 65 63 74 20 70 61 73 73 65 64 20 61 73  object passed as
1840: 20 74 68 65 20 6f 6e 6c 79 20 61 72 67 75 6d 65   the only argume
1850: 6e 74 2e 0a 2a 2f 0a 73 74 61 74 69 63 20 76 6f  nt..*/.static vo
1860: 69 64 20 73 65 72 76 65 72 53 68 75 74 64 6f 77  id serverShutdow
1870: 6e 44 61 74 61 62 61 73 65 28 53 65 72 76 65 72  nDatabase(Server
1880: 44 62 20 2a 70 44 62 29 7b 0a 20 20 69 6e 74 20  Db *pDb){.  int 
1890: 69 3b 0a 0a 20 20 66 6f 72 28 69 3d 30 3b 20 69  i;..  for(i=0; i
18a0: 3c 48 4d 41 5f 4d 41 58 5f 54 52 41 4e 53 41 43  <HMA_MAX_TRANSAC
18b0: 54 49 4f 4e 49 44 3b 20 69 2b 2b 29 7b 0a 20 20  TIONID; i++){.  
18c0: 20 20 53 65 72 76 65 72 4a 6f 75 72 6e 61 6c 20    ServerJournal 
18d0: 2a 70 4a 20 3d 20 26 70 44 62 2d 3e 61 4a 72 6e  *pJ = &pDb->aJrn
18e0: 6c 5b 69 5d 3b 0a 20 20 20 20 69 66 28 20 70 4a  l[i];.    if( pJ
18f0: 2d 3e 6a 66 64 20 29 7b 0a 20 20 20 20 20 20 73  ->jfd ){.      s
1900: 71 6c 69 74 65 33 4f 73 43 6c 6f 73 65 28 70 4a  qlite3OsClose(pJ
1910: 2d 3e 6a 66 64 29 3b 0a 20 20 20 20 20 20 73 71  ->jfd);.      sq
1920: 6c 69 74 65 33 4f 73 44 65 6c 65 74 65 28 70 44  lite3OsDelete(pD
1930: 62 2d 3e 70 56 66 73 2c 20 70 4a 2d 3e 7a 4a 6f  b->pVfs, pJ->zJo
1940: 75 72 6e 61 6c 2c 20 30 29 3b 0a 20 20 20 20 7d  urnal, 0);.    }
1950: 0a 20 20 20 20 73 71 6c 69 74 65 33 5f 66 72 65  .    sqlite3_fre
1960: 65 28 70 4a 2d 3e 7a 4a 6f 75 72 6e 61 6c 29 3b  e(pJ->zJournal);
1970: 0a 20 20 7d 0a 20 20 6d 65 6d 73 65 74 28 70 44  .  }.  memset(pD
1980: 62 2d 3e 61 4a 72 6e 6c 2c 20 30 2c 20 73 69 7a  b->aJrnl, 0, siz
1990: 65 6f 66 28 53 65 72 76 65 72 4a 6f 75 72 6e 61  eof(ServerJourna
19a0: 6c 29 2a 48 4d 41 5f 4d 41 58 5f 54 52 41 4e 53  l)*HMA_MAX_TRANS
19b0: 41 43 54 49 4f 4e 49 44 29 3b 0a 0a 20 20 69 66  ACTIONID);..  if
19c0: 28 20 70 44 62 2d 3e 61 4a 72 6e 6c 46 64 53 70  ( pDb->aJrnlFdSp
19d0: 61 63 65 20 29 7b 0a 20 20 20 20 73 71 6c 69 74  ace ){.    sqlit
19e0: 65 33 5f 66 72 65 65 28 70 44 62 2d 3e 61 4a 72  e3_free(pDb->aJr
19f0: 6e 6c 46 64 53 70 61 63 65 29 3b 0a 20 20 20 20  nlFdSpace);.    
1a00: 70 44 62 2d 3e 61 4a 72 6e 6c 46 64 53 70 61 63  pDb->aJrnlFdSpac
1a10: 65 20 3d 20 30 3b 0a 20 20 7d 0a 0a 20 20 73 71  e = 0;.  }..  sq
1a20: 6c 69 74 65 33 5f 66 72 65 65 28 70 44 62 2d 3e  lite3_free(pDb->
1a30: 61 53 6c 6f 74 29 3b 0a 20 20 70 44 62 2d 3e 62  aSlot);.  pDb->b
1a40: 49 6e 69 74 20 3d 20 30 3b 0a 7d 0a 0a 2f 2a 0a  Init = 0;.}../*.
1a50: 2a 2a 20 54 68 69 73 20 66 75 6e 63 74 69 6f 6e  ** This function
1a60: 20 69 73 20 63 61 6c 6c 65 64 20 77 68 65 6e 20   is called when 
1a70: 74 68 65 20 76 65 72 79 20 66 69 72 73 74 20 63  the very first c
1a80: 6f 6e 6e 65 63 74 69 6f 6e 20 74 6f 20 61 20 64  onnection to a d
1a90: 61 74 61 62 61 73 65 0a 2a 2a 20 69 73 20 65 73  atabase.** is es
1aa0: 74 61 62 6c 69 73 68 65 64 2e 20 49 74 20 69 73  tablished. It is
1ab0: 20 72 65 73 70 6f 6e 73 69 62 6c 65 20 66 6f 72   responsible for
1ac0: 20 72 6f 6c 6c 69 6e 67 20 62 61 63 6b 20 61 6e   rolling back an
1ad0: 79 20 68 6f 74 20 6a 6f 75 72 6e 61 6c 0a 2a 2a  y hot journal.**
1ae0: 20 66 69 6c 65 73 20 66 6f 75 6e 64 20 69 6e 20   files found in 
1af0: 74 68 65 20 66 69 6c 65 2d 73 79 73 74 65 6d 2e  the file-system.
1b00: 0a 2a 2f 0a 73 74 61 74 69 63 20 69 6e 74 20 73  .*/.static int s
1b10: 65 72 76 65 72 49 6e 69 74 44 61 74 61 62 61 73  erverInitDatabas
1b20: 65 28 53 65 72 76 65 72 20 2a 70 4e 65 77 29 7b  e(Server *pNew){
1b30: 0a 20 20 69 6e 74 20 6e 42 79 74 65 3b 0a 20 20  .  int nByte;.  
1b40: 69 6e 74 20 72 63 20 3d 20 53 51 4c 49 54 45 5f  int rc = SQLITE_
1b50: 4f 4b 3b 0a 20 20 53 65 72 76 65 72 44 62 20 2a  OK;.  ServerDb *
1b60: 70 44 62 20 3d 20 70 4e 65 77 2d 3e 70 44 62 3b  pDb = pNew->pDb;
1b70: 0a 20 20 73 71 6c 69 74 65 33 5f 76 66 73 20 2a  .  sqlite3_vfs *
1b80: 70 56 66 73 3b 0a 20 20 63 6f 6e 73 74 20 63 68  pVfs;.  const ch
1b90: 61 72 20 2a 7a 46 69 6c 65 6e 61 6d 65 20 3d 20  ar *zFilename = 
1ba0: 73 71 6c 69 74 65 33 50 61 67 65 72 46 69 6c 65  sqlite3PagerFile
1bb0: 6e 61 6d 65 28 70 4e 65 77 2d 3e 70 50 61 67 65  name(pNew->pPage
1bc0: 72 2c 20 30 29 3b 0a 0a 20 20 61 73 73 65 72 74  r, 0);..  assert
1bd0: 28 20 7a 46 69 6c 65 6e 61 6d 65 20 29 3b 0a 20  ( zFilename );. 
1be0: 20 70 56 66 73 20 3d 20 70 44 62 2d 3e 70 56 66   pVfs = pDb->pVf
1bf0: 73 20 3d 20 73 71 6c 69 74 65 33 50 61 67 65 72  s = sqlite3Pager
1c00: 56 66 73 28 70 4e 65 77 2d 3e 70 50 61 67 65 72  Vfs(pNew->pPager
1c10: 29 3b 0a 20 20 6e 42 79 74 65 20 3d 20 52 4f 55  );.  nByte = ROU
1c20: 4e 44 38 28 70 56 66 73 2d 3e 73 7a 4f 73 46 69  ND8(pVfs->szOsFi
1c30: 6c 65 29 20 2a 20 48 4d 41 5f 4d 41 58 5f 54 52  le) * HMA_MAX_TR
1c40: 41 4e 53 41 43 54 49 4f 4e 49 44 3b 0a 20 20 70  ANSACTIONID;.  p
1c50: 44 62 2d 3e 61 4a 72 6e 6c 46 64 53 70 61 63 65  Db->aJrnlFdSpace
1c60: 20 3d 20 28 75 38 2a 29 73 71 6c 69 74 65 33 4d   = (u8*)sqlite3M
1c70: 61 6c 6c 6f 63 5a 65 72 6f 28 6e 42 79 74 65 29  allocZero(nByte)
1c80: 3b 0a 20 20 69 66 28 20 70 44 62 2d 3e 61 4a 72  ;.  if( pDb->aJr
1c90: 6e 6c 46 64 53 70 61 63 65 3d 3d 30 20 29 7b 0a  nlFdSpace==0 ){.
1ca0: 20 20 20 20 72 63 20 3d 20 53 51 4c 49 54 45 5f      rc = SQLITE_
1cb0: 4e 4f 4d 45 4d 5f 42 4b 50 54 3b 0a 20 20 7d 65  NOMEM_BKPT;.  }e
1cc0: 6c 73 65 7b 0a 20 20 20 20 75 38 20 2a 61 20 3d  lse{.    u8 *a =
1cd0: 20 70 44 62 2d 3e 61 4a 72 6e 6c 46 64 53 70 61   pDb->aJrnlFdSpa
1ce0: 63 65 3b 0a 20 20 20 20 69 6e 74 20 69 3b 0a 20  ce;.    int i;. 
1cf0: 20 20 20 66 6f 72 28 69 3d 30 3b 20 72 63 3d 3d     for(i=0; rc==
1d00: 53 51 4c 49 54 45 5f 4f 4b 20 26 26 20 69 3c 48  SQLITE_OK && i<H
1d10: 4d 41 5f 4d 41 58 5f 54 52 41 4e 53 41 43 54 49  MA_MAX_TRANSACTI
1d20: 4f 4e 49 44 3b 20 69 2b 2b 29 7b 0a 20 20 20 20  ONID; i++){.    
1d30: 20 20 69 6e 74 20 62 45 78 69 73 74 73 20 3d 20    int bExists = 
1d40: 30 3b 0a 20 20 20 20 20 20 53 65 72 76 65 72 4a  0;.      ServerJ
1d50: 6f 75 72 6e 61 6c 20 2a 70 4a 20 3d 20 26 70 44  ournal *pJ = &pD
1d60: 62 2d 3e 61 4a 72 6e 6c 5b 69 5d 3b 0a 20 20 20  b->aJrnl[i];.   
1d70: 20 20 20 70 4a 2d 3e 6a 66 64 20 3d 20 28 73 71     pJ->jfd = (sq
1d80: 6c 69 74 65 33 5f 66 69 6c 65 2a 29 26 61 5b 52  lite3_file*)&a[R
1d90: 4f 55 4e 44 38 28 70 56 66 73 2d 3e 73 7a 4f 73  OUND8(pVfs->szOs
1da0: 46 69 6c 65 29 2a 69 5d 3b 0a 20 20 20 20 20 20  File)*i];.      
1db0: 70 4a 2d 3e 7a 4a 6f 75 72 6e 61 6c 20 3d 20 73  pJ->zJournal = s
1dc0: 71 6c 69 74 65 33 5f 6d 70 72 69 6e 74 66 28 22  qlite3_mprintf("
1dd0: 25 73 2d 6a 6f 75 72 6e 61 6c 2f 25 64 2d 6a 6f  %s-journal/%d-jo
1de0: 75 72 6e 61 6c 22 2c 20 7a 46 69 6c 65 6e 61 6d  urnal", zFilenam
1df0: 65 2c 20 69 29 3b 0a 20 20 20 20 20 20 69 66 28  e, i);.      if(
1e00: 20 70 4a 2d 3e 7a 4a 6f 75 72 6e 61 6c 3d 3d 30   pJ->zJournal==0
1e10: 20 29 7b 0a 20 20 20 20 20 20 20 20 72 63 20 3d   ){.        rc =
1e20: 20 53 51 4c 49 54 45 5f 4e 4f 4d 45 4d 5f 42 4b   SQLITE_NOMEM_BK
1e30: 50 54 3b 0a 20 20 20 20 20 20 20 20 62 72 65 61  PT;.        brea
1e40: 6b 3b 0a 20 20 20 20 20 20 7d 0a 0a 20 20 20 20  k;.      }..    
1e50: 20 20 72 63 20 3d 20 73 71 6c 69 74 65 33 4f 73    rc = sqlite3Os
1e60: 41 63 63 65 73 73 28 70 56 66 73 2c 20 70 4a 2d  Access(pVfs, pJ-
1e70: 3e 7a 4a 6f 75 72 6e 61 6c 2c 20 53 51 4c 49 54  >zJournal, SQLIT
1e80: 45 5f 41 43 43 45 53 53 5f 45 58 49 53 54 53 2c  E_ACCESS_EXISTS,
1e90: 20 26 62 45 78 69 73 74 73 29 3b 0a 20 20 20 20   &bExists);.    
1ea0: 20 20 69 66 28 20 72 63 3d 3d 53 51 4c 49 54 45    if( rc==SQLITE
1eb0: 5f 4f 4b 20 26 26 20 62 45 78 69 73 74 73 20 29  _OK && bExists )
1ec0: 7b 0a 20 20 20 20 20 20 20 20 69 6e 74 20 66 6c  {.        int fl
1ed0: 61 67 73 20 3d 20 53 51 4c 49 54 45 5f 4f 50 45  ags = SQLITE_OPE
1ee0: 4e 5f 52 45 41 44 57 52 49 54 45 7c 53 51 4c 49  N_READWRITE|SQLI
1ef0: 54 45 5f 4f 50 45 4e 5f 4d 41 49 4e 5f 4a 4f 55  TE_OPEN_MAIN_JOU
1f00: 52 4e 41 4c 3b 0a 20 20 20 20 20 20 20 20 72 63  RNAL;.        rc
1f10: 20 3d 20 73 71 6c 69 74 65 33 4f 73 4f 70 65 6e   = sqlite3OsOpen
1f20: 28 70 56 66 73 2c 20 70 4a 2d 3e 7a 4a 6f 75 72  (pVfs, pJ->zJour
1f30: 6e 61 6c 2c 20 70 4a 2d 3e 6a 66 64 2c 20 66 6c  nal, pJ->jfd, fl
1f40: 61 67 73 2c 20 26 66 6c 61 67 73 29 3b 0a 20 20  ags, &flags);.  
1f50: 20 20 20 20 20 20 69 66 28 20 72 63 3d 3d 53 51        if( rc==SQ
1f60: 4c 49 54 45 5f 4f 4b 20 29 7b 0a 20 20 20 20 20  LITE_OK ){.     
1f70: 20 20 20 20 20 72 63 20 3d 20 73 71 6c 69 74 65       rc = sqlite
1f80: 33 50 61 67 65 72 52 6f 6c 6c 62 61 63 6b 4a 6f  3PagerRollbackJo
1f90: 75 72 6e 61 6c 28 70 4e 65 77 2d 3e 70 50 61 67  urnal(pNew->pPag
1fa0: 65 72 2c 20 70 4a 2d 3e 6a 66 64 29 3b 0a 20 20  er, pJ->jfd);.  
1fb0: 20 20 20 20 20 20 7d 0a 20 20 20 20 20 20 7d 0a        }.      }.
1fc0: 20 20 20 20 7d 0a 20 20 7d 0a 0a 20 20 69 66 28      }.  }..  if(
1fd0: 20 72 63 3d 3d 53 51 4c 49 54 45 5f 4f 4b 20 29   rc==SQLITE_OK )
1fe0: 7b 0a 20 20 20 20 70 44 62 2d 3e 62 49 6e 69 74  {.    pDb->bInit
1ff0: 20 3d 20 31 3b 0a 20 20 7d 65 6c 73 65 7b 0a 20   = 1;.  }else{. 
2000: 20 20 20 73 65 72 76 65 72 53 68 75 74 64 6f 77     serverShutdow
2010: 6e 44 61 74 61 62 61 73 65 28 70 4e 65 77 2d 3e  nDatabase(pNew->
2020: 70 44 62 29 3b 0a 20 20 7d 0a 20 20 72 65 74 75  pDb);.  }.  retu
2030: 72 6e 20 72 63 3b 0a 7d 0a 0a 2f 2a 0a 2a 2a 20  rn rc;.}../*.** 
2040: 43 6c 6f 73 65 20 74 68 65 20 63 6f 6e 6e 65 63  Close the connec
2050: 74 69 6f 6e 2e 0a 2a 2f 0a 76 6f 69 64 20 73 71  tion..*/.void sq
2060: 6c 69 74 65 33 53 65 72 76 65 72 44 69 73 63 6f  lite3ServerDisco
2070: 6e 6e 65 63 74 28 53 65 72 76 65 72 20 2a 70 2c  nnect(Server *p,
2080: 20 73 71 6c 69 74 65 33 5f 66 69 6c 65 20 2a 64   sqlite3_file *d
2090: 62 66 64 29 7b 0a 20 20 53 65 72 76 65 72 44 62  bfd){.  ServerDb
20a0: 20 2a 70 44 62 20 3d 20 70 2d 3e 70 44 62 3b 0a   *pDb = p->pDb;.
20b0: 0a 20 20 73 65 72 76 65 72 45 6e 74 65 72 4d 75  .  serverEnterMu
20c0: 74 65 78 28 29 3b 0a 20 20 70 44 62 2d 3e 6e 43  tex();.  pDb->nC
20d0: 6c 69 65 6e 74 2d 2d 3b 0a 20 20 69 66 28 20 70  lient--;.  if( p
20e0: 44 62 2d 3e 6e 43 6c 69 65 6e 74 3d 3d 30 20 29  Db->nClient==0 )
20f0: 7b 0a 20 20 20 20 53 65 72 76 65 72 50 61 67 65  {.    ServerPage
2100: 20 2a 70 46 72 65 65 3b 0a 20 20 20 20 53 65 72   *pFree;.    Ser
2110: 76 65 72 44 62 20 2a 2a 70 70 3b 0a 20 20 20 20  verDb **pp;.    
2120: 73 65 72 76 65 72 53 68 75 74 64 6f 77 6e 44 61  serverShutdownDa
2130: 74 61 62 61 73 65 28 70 44 62 29 3b 0a 20 20 20  tabase(pDb);.   
2140: 20 66 6f 72 28 70 70 3d 26 67 5f 73 65 72 76 65   for(pp=&g_serve
2150: 72 2e 70 44 62 3b 20 2a 70 70 21 3d 70 44 62 3b  r.pDb; *pp!=pDb;
2160: 20 70 70 3d 26 28 28 2a 70 70 29 2d 3e 70 4e 65   pp=&((*pp)->pNe
2170: 78 74 29 29 3b 0a 20 20 20 20 2a 70 70 20 3d 20  xt));.    *pp = 
2180: 70 44 62 2d 3e 70 4e 65 78 74 3b 0a 20 20 20 20  pDb->pNext;.    
2190: 73 71 6c 69 74 65 33 5f 6d 75 74 65 78 5f 66 72  sqlite3_mutex_fr
21a0: 65 65 28 70 44 62 2d 3e 6d 75 74 65 78 29 3b 0a  ee(pDb->mutex);.
21b0: 20 20 20 20 77 68 69 6c 65 28 20 70 46 72 65 65      while( pFree
21c0: 3d 70 44 62 2d 3e 70 46 72 65 65 20 29 7b 0a 20  =pDb->pFree ){. 
21d0: 20 20 20 20 20 70 44 62 2d 3e 70 46 72 65 65 20       pDb->pFree 
21e0: 3d 20 70 46 72 65 65 2d 3e 70 4e 65 78 74 3b 0a  = pFree->pNext;.
21f0: 20 20 20 20 20 20 73 71 6c 69 74 65 33 5f 66 72        sqlite3_fr
2200: 65 65 28 70 46 72 65 65 29 3b 0a 20 20 20 20 7d  ee(pFree);.    }
2210: 0a 20 20 20 20 73 71 6c 69 74 65 33 5f 66 72 65  .    sqlite3_fre
2220: 65 28 70 44 62 29 3b 0a 20 20 7d 0a 20 20 73 65  e(pDb);.  }.  se
2230: 72 76 65 72 4c 65 61 76 65 4d 75 74 65 78 28 29  rverLeaveMutex()
2240: 3b 0a 0a 20 20 73 71 6c 69 74 65 33 5f 66 72 65  ;..  sqlite3_fre
2250: 65 28 70 2d 3e 61 4c 6f 63 6b 29 3b 0a 20 20 73  e(p->aLock);.  s
2260: 71 6c 69 74 65 33 5f 66 72 65 65 28 70 29 3b 0a  qlite3_free(p);.
2270: 7d 0a 0a 2f 2a 0a 2a 2a 20 43 6f 6e 6e 65 63 74  }../*.** Connect
2280: 20 74 6f 20 74 68 65 20 73 79 73 74 65 6d 2e 0a   to the system..
2290: 2a 2f 0a 69 6e 74 20 73 71 6c 69 74 65 33 53 65  */.int sqlite3Se
22a0: 72 76 65 72 43 6f 6e 6e 65 63 74 28 0a 20 20 50  rverConnect(.  P
22b0: 61 67 65 72 20 2a 70 50 61 67 65 72 2c 0a 20 20  ager *pPager,.  
22c0: 53 65 72 76 65 72 20 2a 2a 70 70 4f 75 74 20 20  Server **ppOut  
22d0: 20 20 20 20 20 20 20 20 20 20 20 20 20 20 20 20                  
22e0: 2f 2a 20 4f 55 54 3a 20 53 65 72 76 65 72 20 68  /* OUT: Server h
22f0: 61 6e 64 6c 65 20 2a 2f 0a 29 7b 0a 20 20 53 65  andle */.){.  Se
2300: 72 76 65 72 20 2a 70 4e 65 77 20 3d 20 30 3b 0a  rver *pNew = 0;.
2310: 20 20 73 71 6c 69 74 65 33 5f 66 69 6c 65 20 2a    sqlite3_file *
2320: 64 62 66 64 20 3d 20 73 71 6c 69 74 65 33 50 61  dbfd = sqlite3Pa
2330: 67 65 72 46 69 6c 65 28 70 50 61 67 65 72 29 3b  gerFile(pPager);
2340: 0a 20 20 69 36 34 20 61 46 69 6c 65 49 64 5b 32  .  i64 aFileId[2
2350: 5d 3b 0a 20 20 69 6e 74 20 72 63 3b 0a 0a 20 20  ];.  int rc;..  
2360: 72 63 20 3d 20 73 71 6c 69 74 65 33 4f 73 46 69  rc = sqlite3OsFi
2370: 6c 65 43 6f 6e 74 72 6f 6c 28 64 62 66 64 2c 20  leControl(dbfd, 
2380: 53 51 4c 49 54 45 5f 46 43 4e 54 4c 5f 46 49 4c  SQLITE_FCNTL_FIL
2390: 45 49 44 2c 20 28 76 6f 69 64 2a 29 61 46 69 6c  EID, (void*)aFil
23a0: 65 49 64 29 3b 0a 20 20 69 66 28 20 72 63 3d 3d  eId);.  if( rc==
23b0: 53 51 4c 49 54 45 5f 4f 4b 20 29 7b 0a 20 20 20  SQLITE_OK ){.   
23c0: 20 70 4e 65 77 20 3d 20 28 53 65 72 76 65 72 2a   pNew = (Server*
23d0: 29 73 71 6c 69 74 65 33 4d 61 6c 6c 6f 63 5a 65  )sqlite3MallocZe
23e0: 72 6f 28 73 69 7a 65 6f 66 28 53 65 72 76 65 72  ro(sizeof(Server
23f0: 29 29 3b 0a 20 20 20 20 69 66 28 20 70 4e 65 77  ));.    if( pNew
2400: 20 29 7b 0a 20 20 20 20 20 20 70 4e 65 77 2d 3e   ){.      pNew->
2410: 70 50 61 67 65 72 20 3d 20 70 50 61 67 65 72 3b  pPager = pPager;
2420: 0a 20 20 20 20 20 20 70 4e 65 77 2d 3e 69 54 72  .      pNew->iTr
2430: 61 6e 73 49 64 20 3d 20 2d 31 3b 0a 20 20 20 20  ansId = -1;.    
2440: 20 20 72 63 20 3d 20 73 65 72 76 65 72 46 69 6e    rc = serverFin
2450: 64 44 61 74 61 62 61 73 65 28 70 4e 65 77 2c 20  dDatabase(pNew, 
2460: 61 46 69 6c 65 49 64 29 3b 0a 20 20 20 20 20 20  aFileId);.      
2470: 69 66 28 20 72 63 21 3d 53 51 4c 49 54 45 5f 4f  if( rc!=SQLITE_O
2480: 4b 20 29 7b 0a 20 20 20 20 20 20 20 20 73 71 6c  K ){.        sql
2490: 69 74 65 33 5f 66 72 65 65 28 70 4e 65 77 29 3b  ite3_free(pNew);
24a0: 0a 20 20 20 20 20 20 20 20 70 4e 65 77 20 3d 20  .        pNew = 
24b0: 30 3b 0a 20 20 20 20 20 20 7d 65 6c 73 65 7b 0a  0;.      }else{.
24c0: 20 20 20 20 20 20 20 20 73 71 6c 69 74 65 33 5f          sqlite3_
24d0: 6d 75 74 65 78 5f 65 6e 74 65 72 28 70 4e 65 77  mutex_enter(pNew
24e0: 2d 3e 70 44 62 2d 3e 6d 75 74 65 78 29 3b 0a 20  ->pDb->mutex);. 
24f0: 20 20 20 20 20 20 20 69 66 28 20 70 4e 65 77 2d         if( pNew-
2500: 3e 70 44 62 2d 3e 62 49 6e 69 74 3d 3d 30 20 29  >pDb->bInit==0 )
2510: 7b 0a 20 20 20 20 20 20 20 20 20 20 72 63 20 3d  {.          rc =
2520: 20 73 65 72 76 65 72 49 6e 69 74 44 61 74 61 62   serverInitDatab
2530: 61 73 65 28 70 4e 65 77 29 3b 0a 20 20 20 20 20  ase(pNew);.     
2540: 20 20 20 7d 0a 20 20 20 20 20 20 20 20 73 71 6c     }.        sql
2550: 69 74 65 33 5f 6d 75 74 65 78 5f 6c 65 61 76 65  ite3_mutex_leave
2560: 28 70 4e 65 77 2d 3e 70 44 62 2d 3e 6d 75 74 65  (pNew->pDb->mute
2570: 78 29 3b 0a 20 20 20 20 20 20 7d 0a 20 20 20 20  x);.      }.    
2580: 7d 65 6c 73 65 7b 0a 20 20 20 20 20 20 72 63 20  }else{.      rc 
2590: 3d 20 53 51 4c 49 54 45 5f 4e 4f 4d 45 4d 5f 42  = SQLITE_NOMEM_B
25a0: 4b 50 54 3b 0a 20 20 20 20 7d 0a 20 20 7d 0a 0a  KPT;.    }.  }..
25b0: 20 20 2a 70 70 4f 75 74 20 3d 20 70 4e 65 77 3b    *ppOut = pNew;
25c0: 0a 20 20 72 65 74 75 72 6e 20 72 63 3b 0a 7d 0a  .  return rc;.}.
25d0: 0a 2f 2a 0a 2a 2a 20 42 65 67 69 6e 20 61 20 74  ./*.** Begin a t
25e0: 72 61 6e 73 61 63 74 69 6f 6e 2e 0a 2a 2f 0a 69  ransaction..*/.i
25f0: 6e 74 20 73 71 6c 69 74 65 33 53 65 72 76 65 72  nt sqlite3Server
2600: 42 65 67 69 6e 28 53 65 72 76 65 72 20 2a 70 2c  Begin(Server *p,
2610: 20 69 6e 74 20 62 52 65 61 64 6f 6e 6c 79 29 7b   int bReadonly){
2620: 0a 20 20 69 6e 74 20 72 63 20 3d 20 53 51 4c 49  .  int rc = SQLI
2630: 54 45 5f 4f 4b 3b 0a 0a 20 20 69 66 28 20 70 2d  TE_OK;..  if( p-
2640: 3e 65 54 72 61 6e 73 3d 3d 53 45 52 56 45 52 5f  >eTrans==SERVER_
2650: 54 52 41 4e 53 5f 4e 4f 4e 45 20 29 7b 0a 20 20  TRANS_NONE ){.  
2660: 20 20 69 6e 74 20 69 64 3b 0a 20 20 20 20 53 65    int id;.    Se
2670: 72 76 65 72 44 62 20 2a 70 44 62 20 3d 20 70 2d  rverDb *pDb = p-
2680: 3e 70 44 62 3b 0a 20 20 20 20 75 33 32 20 74 3b  >pDb;.    u32 t;
2690: 0a 0a 20 20 20 20 61 73 73 65 72 74 28 20 70 2d  ..    assert( p-
26a0: 3e 69 54 72 61 6e 73 49 64 3c 30 20 29 3b 0a 20  >iTransId<0 );. 
26b0: 20 20 20 61 73 73 65 72 74 28 20 70 2d 3e 70 4e     assert( p->pN
26c0: 65 78 74 3d 3d 30 20 29 3b 0a 20 20 20 20 73 71  ext==0 );.    sq
26d0: 6c 69 74 65 33 5f 6d 75 74 65 78 5f 65 6e 74 65  lite3_mutex_ente
26e0: 72 28 70 44 62 2d 3e 6d 75 74 65 78 29 3b 0a 0a  r(pDb->mutex);..
26f0: 20 20 20 20 69 66 28 20 62 52 65 61 64 6f 6e 6c      if( bReadonl
2700: 79 20 29 7b 0a 20 20 20 20 20 20 53 65 72 76 65  y ){.      Serve
2710: 72 20 2a 70 49 74 65 72 3b 0a 20 20 20 20 20 20  r *pIter;.      
2720: 70 2d 3e 69 43 6f 6d 6d 69 74 49 64 20 3d 20 70  p->iCommitId = p
2730: 44 62 2d 3e 69 4e 65 78 74 43 6f 6d 6d 69 74 3b  Db->iNextCommit;
2740: 0a 20 20 20 20 20 20 66 6f 72 28 70 49 74 65 72  .      for(pIter
2750: 3d 70 44 62 2d 3e 70 43 6f 6d 6d 69 74 3b 20 70  =pDb->pCommit; p
2760: 49 74 65 72 3b 20 70 49 74 65 72 3d 70 49 74 65  Iter; pIter=pIte
2770: 72 2d 3e 70 4e 65 78 74 29 7b 0a 20 20 20 20 20  r->pNext){.     
2780: 20 20 20 69 66 28 20 70 49 74 65 72 2d 3e 69 43     if( pIter->iC
2790: 6f 6d 6d 69 74 49 64 3c 70 2d 3e 69 43 6f 6d 6d  ommitId<p->iComm
27a0: 69 74 49 64 20 29 7b 0a 20 20 20 20 20 20 20 20  itId ){.        
27b0: 20 20 70 2d 3e 69 43 6f 6d 6d 69 74 49 64 20 3d    p->iCommitId =
27c0: 20 70 49 74 65 72 2d 3e 69 43 6f 6d 6d 69 74 49   pIter->iCommitI
27d0: 64 3b 0a 20 20 20 20 20 20 20 20 7d 0a 20 20 20  d;.        }.   
27e0: 20 20 20 7d 0a 20 20 20 20 20 20 70 2d 3e 70 4e     }.      p->pN
27f0: 65 78 74 20 3d 20 70 44 62 2d 3e 70 52 65 61 64  ext = pDb->pRead
2800: 65 72 3b 0a 20 20 20 20 20 20 70 44 62 2d 3e 70  er;.      pDb->p
2810: 52 65 61 64 65 72 20 3d 20 70 3b 0a 20 20 20 20  Reader = p;.    
2820: 20 20 70 2d 3e 65 54 72 61 6e 73 20 3d 20 53 45    p->eTrans = SE
2830: 52 56 45 52 5f 54 52 41 4e 53 5f 52 45 41 44 4f  RVER_TRANS_READO
2840: 4e 4c 59 3b 0a 20 20 20 20 7d 65 6c 73 65 7b 0a  NLY;.    }else{.
2850: 20 20 20 20 20 20 2f 2a 20 46 69 6e 64 20 61 20        /* Find a 
2860: 74 72 61 6e 73 61 63 74 69 6f 6e 20 69 64 20 74  transaction id t
2870: 6f 20 75 73 65 20 2a 2f 0a 20 20 20 20 20 20 72  o use */.      r
2880: 63 20 3d 20 53 51 4c 49 54 45 5f 42 55 53 59 3b  c = SQLITE_BUSY;
2890: 0a 20 20 20 20 20 20 74 20 3d 20 70 44 62 2d 3e  .      t = pDb->
28a0: 74 72 61 6e 73 6d 61 73 6b 3b 0a 20 20 20 20 20  transmask;.     
28b0: 20 66 6f 72 28 69 64 3d 30 3b 20 69 64 3c 48 4d   for(id=0; id<HM
28c0: 41 5f 4d 41 58 5f 54 52 41 4e 53 41 43 54 49 4f  A_MAX_TRANSACTIO
28d0: 4e 49 44 3b 20 69 64 2b 2b 29 7b 0a 20 20 20 20  NID; id++){.    
28e0: 20 20 20 20 69 66 28 20 28 74 20 26 20 28 31 20      if( (t & (1 
28f0: 3c 3c 20 69 64 29 29 3d 3d 30 20 29 7b 0a 20 20  << id))==0 ){.  
2900: 20 20 20 20 20 20 20 20 74 20 3d 20 74 20 7c 20          t = t | 
2910: 28 31 20 3c 3c 20 69 64 29 3b 0a 20 20 20 20 20  (1 << id);.     
2920: 20 20 20 20 20 72 63 20 3d 20 53 51 4c 49 54 45       rc = SQLITE
2930: 5f 4f 4b 3b 0a 20 20 20 20 20 20 20 20 20 20 62  _OK;.          b
2940: 72 65 61 6b 3b 0a 20 20 20 20 20 20 20 20 7d 0a  reak;.        }.
2950: 20 20 20 20 20 20 7d 0a 20 20 20 20 20 20 70 44        }.      pD
2960: 62 2d 3e 74 72 61 6e 73 6d 61 73 6b 20 3d 20 74  b->transmask = t
2970: 3b 0a 20 20 20 20 20 20 70 2d 3e 65 54 72 61 6e  ;.      p->eTran
2980: 73 20 3d 20 53 45 52 56 45 52 5f 54 52 41 4e 53  s = SERVER_TRANS
2990: 5f 52 45 41 44 57 52 49 54 45 3b 0a 20 20 20 20  _READWRITE;.    
29a0: 7d 0a 0a 20 20 20 20 73 71 6c 69 74 65 33 5f 6d  }..    sqlite3_m
29b0: 75 74 65 78 5f 6c 65 61 76 65 28 70 44 62 2d 3e  utex_leave(pDb->
29c0: 6d 75 74 65 78 29 3b 0a 0a 20 20 20 20 69 66 28  mutex);..    if(
29d0: 20 72 63 3d 3d 53 51 4c 49 54 45 5f 4f 4b 20 26   rc==SQLITE_OK &
29e0: 26 20 62 52 65 61 64 6f 6e 6c 79 3d 3d 30 20 29  & bReadonly==0 )
29f0: 7b 0a 20 20 20 20 20 20 53 65 72 76 65 72 4a 6f  {.      ServerJo
2a00: 75 72 6e 61 6c 20 2a 70 4a 72 6e 6c 20 3d 20 26  urnal *pJrnl = &
2a10: 70 44 62 2d 3e 61 4a 72 6e 6c 5b 69 64 5d 3b 0a  pDb->aJrnl[id];.
2a20: 20 20 20 20 20 20 73 71 6c 69 74 65 33 50 61 67        sqlite3Pag
2a30: 65 72 53 65 72 76 65 72 4a 6f 75 72 6e 61 6c 28  erServerJournal(
2a40: 70 2d 3e 70 50 61 67 65 72 2c 20 70 4a 72 6e 6c  p->pPager, pJrnl
2a50: 2d 3e 6a 66 64 2c 20 70 4a 72 6e 6c 2d 3e 7a 4a  ->jfd, pJrnl->zJ
2a60: 6f 75 72 6e 61 6c 29 3b 0a 20 20 20 20 20 20 70  ournal);.      p
2a70: 2d 3e 69 54 72 61 6e 73 49 64 20 3d 20 69 64 3b  ->iTransId = id;
2a80: 0a 20 20 20 20 7d 0a 20 20 7d 0a 0a 20 20 72 65  .    }.  }..  re
2a90: 74 75 72 6e 20 72 63 3b 0a 7d 0a 0a 73 74 61 74  turn rc;.}..stat
2aa0: 69 63 20 76 6f 69 64 20 73 65 72 76 65 72 52 65  ic void serverRe
2ab0: 6c 65 61 73 65 4c 6f 63 6b 73 28 53 65 72 76 65  leaseLocks(Serve
2ac0: 72 20 2a 70 29 7b 0a 20 20 53 65 72 76 65 72 44  r *p){.  ServerD
2ad0: 62 20 2a 70 44 62 20 3d 20 70 2d 3e 70 44 62 3b  b *pDb = p->pDb;
2ae0: 0a 20 20 69 6e 74 20 69 3b 0a 20 20 61 73 73 65  .  int i;.  asse
2af0: 72 74 28 20 73 71 6c 69 74 65 33 5f 6d 75 74 65  rt( sqlite3_mute
2b00: 78 5f 68 65 6c 64 28 70 44 62 2d 3e 6d 75 74 65  x_held(pDb->mute
2b10: 78 29 20 29 3b 0a 0a 20 20 66 6f 72 28 69 3d 30  x) );..  for(i=0
2b20: 3b 20 69 3c 70 2d 3e 6e 4c 6f 63 6b 3b 20 69 2b  ; i<p->nLock; i+
2b30: 2b 29 7b 0a 20 20 20 20 75 33 32 20 2a 70 53 6c  +){.    u32 *pSl
2b40: 6f 74 20 3d 20 26 70 44 62 2d 3e 61 53 6c 6f 74  ot = &pDb->aSlot
2b50: 5b 70 2d 3e 61 4c 6f 63 6b 5b 69 5d 20 25 20 48  [p->aLock[i] % H
2b60: 4d 41 5f 50 41 47 45 4c 4f 43 4b 5f 53 4c 4f 54  MA_PAGELOCK_SLOT
2b70: 53 5d 3b 0a 20 20 20 20 69 66 28 20 73 6c 6f 74  S];.    if( slot
2b80: 47 65 74 57 72 69 74 65 72 28 2a 70 53 6c 6f 74  GetWriter(*pSlot
2b90: 29 3d 3d 70 2d 3e 69 54 72 61 6e 73 49 64 20 29  )==p->iTransId )
2ba0: 7b 0a 20 20 20 20 20 20 2a 70 53 6c 6f 74 20 2d  {.      *pSlot -
2bb0: 3d 20 28 28 70 2d 3e 69 54 72 61 6e 73 49 64 20  = ((p->iTransId 
2bc0: 2b 20 31 29 20 3c 3c 20 48 4d 41 5f 4d 41 58 5f  + 1) << HMA_MAX_
2bd0: 54 52 41 4e 53 41 43 54 49 4f 4e 49 44 29 3b 0a  TRANSACTIONID);.
2be0: 20 20 20 20 7d 0a 20 20 20 20 2a 70 53 6c 6f 74      }.    *pSlot
2bf0: 20 26 3d 20 7e 28 28 75 33 32 29 31 20 3c 3c 20   &= ~((u32)1 << 
2c00: 70 2d 3e 69 54 72 61 6e 73 49 64 29 3b 0a 20 20  p->iTransId);.  
2c10: 7d 0a 0a 20 20 70 2d 3e 6e 4c 6f 63 6b 20 3d 20  }..  p->nLock = 
2c20: 30 3b 0a 7d 0a 0a 2f 2a 0a 2a 2a 20 45 6e 64 20  0;.}../*.** End 
2c30: 61 20 74 72 61 6e 73 61 63 74 69 6f 6e 20 28 61  a transaction (a
2c40: 6e 64 20 72 65 6c 65 61 73 65 20 61 6c 6c 20 6c  nd release all l
2c50: 6f 63 6b 73 29 2e 0a 2a 2f 0a 69 6e 74 20 73 71  ocks)..*/.int sq
2c60: 6c 69 74 65 33 53 65 72 76 65 72 45 6e 64 28 53  lite3ServerEnd(S
2c70: 65 72 76 65 72 20 2a 70 29 7b 0a 20 20 69 6e 74  erver *p){.  int
2c80: 20 72 63 20 3d 20 53 51 4c 49 54 45 5f 4f 4b 3b   rc = SQLITE_OK;
2c90: 0a 20 20 69 66 28 20 70 2d 3e 65 54 72 61 6e 73  .  if( p->eTrans
2ca0: 21 3d 53 45 52 56 45 52 5f 54 52 41 4e 53 5f 4e  !=SERVER_TRANS_N
2cb0: 4f 4e 45 20 29 7b 0a 20 20 20 20 53 65 72 76 65  ONE ){.    Serve
2cc0: 72 20 2a 2a 70 70 3b 0a 20 20 20 20 53 65 72 76  r **pp;.    Serv
2cd0: 65 72 44 62 20 2a 70 44 62 20 3d 20 70 2d 3e 70  erDb *pDb = p->p
2ce0: 44 62 3b 0a 20 20 20 20 53 65 72 76 65 72 50 61  Db;.    ServerPa
2cf0: 67 65 20 2a 70 50 67 20 3d 20 30 3b 0a 0a 20 20  ge *pPg = 0;..  
2d00: 20 20 73 71 6c 69 74 65 33 5f 6d 75 74 65 78 5f    sqlite3_mutex_
2d10: 65 6e 74 65 72 28 70 44 62 2d 3e 6d 75 74 65 78  enter(pDb->mutex
2d20: 29 3b 0a 0a 20 20 20 20 69 66 28 20 70 2d 3e 65  );..    if( p->e
2d30: 54 72 61 6e 73 3d 3d 53 45 52 56 45 52 5f 54 52  Trans==SERVER_TR
2d40: 41 4e 53 5f 52 45 41 44 4f 4e 4c 59 20 29 7b 0a  ANS_READONLY ){.
2d50: 20 20 20 20 20 20 2f 2a 20 52 65 6d 6f 76 65 20        /* Remove 
2d60: 74 68 65 20 63 6f 6e 6e 65 63 74 69 6f 6e 20 66  the connection f
2d70: 72 6f 6d 20 74 68 65 20 72 65 61 64 65 72 73 20  rom the readers 
2d80: 6c 69 73 74 20 2a 2f 0a 20 20 20 20 20 20 66 6f  list */.      fo
2d90: 72 28 70 70 3d 26 70 44 62 2d 3e 70 52 65 61 64  r(pp=&pDb->pRead
2da0: 65 72 3b 20 2a 70 70 21 3d 70 3b 20 70 70 20 3d  er; *pp!=p; pp =
2db0: 20 26 28 28 2a 70 70 29 2d 3e 70 4e 65 78 74 29   &((*pp)->pNext)
2dc0: 29 3b 0a 20 20 20 20 20 20 2a 70 70 20 3d 20 70  );.      *pp = p
2dd0: 2d 3e 70 4e 65 78 74 3b 0a 20 20 20 20 7d 65 6c  ->pNext;.    }el
2de0: 73 65 7b 0a 20 20 20 20 20 20 73 65 72 76 65 72  se{.      server
2df0: 52 65 6c 65 61 73 65 4c 6f 63 6b 73 28 70 29 3b  ReleaseLocks(p);
2e00: 0a 0a 20 20 20 20 20 20 2f 2a 20 43 6c 65 61 72  ..      /* Clear
2e10: 20 74 68 65 20 62 69 74 20 69 6e 20 74 68 65 20   the bit in the 
2e20: 74 72 61 6e 73 61 63 74 69 6f 6e 20 6d 61 73 6b  transaction mask
2e30: 2e 20 2a 2f 0a 20 20 20 20 20 20 70 44 62 2d 3e  . */.      pDb->
2e40: 74 72 61 6e 73 6d 61 73 6b 20 26 3d 20 7e 28 28  transmask &= ~((
2e50: 75 33 32 29 31 20 3c 3c 20 70 2d 3e 69 54 72 61  u32)1 << p->iTra
2e60: 6e 73 49 64 29 3b 0a 0a 20 20 20 20 20 20 2f 2a  nsId);..      /*
2e70: 20 49 66 20 74 68 69 73 20 63 6f 6e 6e 65 63 74   If this connect
2e80: 69 6f 6e 20 69 73 20 69 6e 20 74 68 65 20 63 6f  ion is in the co
2e90: 6d 6d 69 74 74 65 72 73 20 6c 69 73 74 2c 20 72  mmitters list, r
2ea0: 65 6d 6f 76 65 20 69 74 2e 20 2a 2f 0a 20 20 20  emove it. */.   
2eb0: 20 20 20 66 6f 72 28 70 70 3d 26 70 44 62 2d 3e     for(pp=&pDb->
2ec0: 70 43 6f 6d 6d 69 74 3b 20 2a 70 70 3b 20 70 70  pCommit; *pp; pp
2ed0: 20 3d 20 26 28 28 2a 70 70 29 2d 3e 70 4e 65 78   = &((*pp)->pNex
2ee0: 74 29 29 7b 0a 20 20 20 20 20 20 20 20 69 66 28  t)){.        if(
2ef0: 20 2a 70 70 3d 3d 70 20 29 7b 0a 20 20 20 20 20   *pp==p ){.     
2f00: 20 20 20 20 20 2a 70 70 20 3d 20 70 2d 3e 70 4e       *pp = p->pN
2f10: 65 78 74 3b 0a 20 20 20 20 20 20 20 20 20 20 62  ext;.          b
2f20: 72 65 61 6b 3b 0a 20 20 20 20 20 20 20 20 7d 0a  reak;.        }.
2f30: 20 20 20 20 20 20 7d 0a 20 20 20 20 7d 0a 0a 20        }.    }.. 
2f40: 20 20 20 2f 2a 20 53 65 65 20 69 66 20 69 74 20     /* See if it 
2f50: 69 73 20 70 6f 73 73 69 62 6c 65 20 74 6f 20 66  is possible to f
2f60: 72 65 65 20 61 6e 79 20 53 65 72 76 65 72 50 61  ree any ServerPa
2f70: 67 65 20 72 65 63 6f 72 64 73 2e 20 49 66 20 73  ge records. If s
2f80: 6f 2c 20 72 65 6d 6f 76 65 0a 20 20 20 20 2a 2a  o, remove.    **
2f90: 20 74 68 65 6d 20 66 72 6f 6d 20 74 68 65 20 6c   them from the l
2fa0: 69 6e 6b 65 64 20 6c 69 73 74 20 61 6e 64 20 68  inked list and h
2fb0: 61 73 68 20 74 61 62 6c 65 2c 20 62 75 74 20 64  ash table, but d
2fc0: 6f 20 6e 6f 74 20 63 61 6c 6c 20 73 71 6c 69 74  o not call sqlit
2fd0: 65 33 5f 66 72 65 65 28 29 0a 20 20 20 20 2a 2a  e3_free().    **
2fe0: 20 6f 6e 20 74 68 65 6d 20 75 6e 74 69 6c 20 74   on them until t
2ff0: 68 65 20 6d 75 74 65 78 20 68 61 73 20 62 65 65  he mutex has bee
3000: 6e 20 72 65 6c 65 61 73 65 64 2e 20 20 2a 2f 0a  n released.  */.
3010: 20 20 20 20 69 66 28 20 70 44 62 2d 3e 70 50 67      if( pDb->pPg
3020: 46 69 72 73 74 20 29 7b 0a 20 20 20 20 20 20 53  First ){.      S
3030: 65 72 76 65 72 50 61 67 65 20 2a 70 4c 61 73 74  erverPage *pLast
3040: 20 3d 20 30 3b 0a 20 20 20 20 20 20 53 65 72 76   = 0;.      Serv
3050: 65 72 20 2a 70 49 74 65 72 3b 0a 20 20 20 20 20  er *pIter;.     
3060: 20 69 6e 74 20 69 4f 6c 64 65 73 74 20 3d 20 30   int iOldest = 0
3070: 78 37 46 46 46 46 46 46 46 3b 0a 20 20 20 20 20  x7FFFFFFF;.     
3080: 20 66 6f 72 28 70 49 74 65 72 3d 70 44 62 2d 3e   for(pIter=pDb->
3090: 70 52 65 61 64 65 72 3b 20 70 49 74 65 72 3b 20  pReader; pIter; 
30a0: 70 49 74 65 72 3d 70 49 74 65 72 2d 3e 70 4e 65  pIter=pIter->pNe
30b0: 78 74 29 7b 0a 20 20 20 20 20 20 20 20 69 4f 6c  xt){.        iOl
30c0: 64 65 73 74 20 3d 20 4d 49 4e 28 69 4f 6c 64 65  dest = MIN(iOlde
30d0: 73 74 2c 20 70 49 74 65 72 2d 3e 69 43 6f 6d 6d  st, pIter->iComm
30e0: 69 74 49 64 29 3b 0a 20 20 20 20 20 20 7d 0a 20  itId);.      }. 
30f0: 20 20 20 20 20 66 6f 72 28 70 49 74 65 72 3d 70       for(pIter=p
3100: 44 62 2d 3e 70 43 6f 6d 6d 69 74 3b 20 70 49 74  Db->pCommit; pIt
3110: 65 72 3b 20 70 49 74 65 72 3d 70 49 74 65 72 2d  er; pIter=pIter-
3120: 3e 70 4e 65 78 74 29 7b 0a 20 20 20 20 20 20 20  >pNext){.       
3130: 20 69 4f 6c 64 65 73 74 20 3d 20 4d 49 4e 28 69   iOldest = MIN(i
3140: 4f 6c 64 65 73 74 2c 20 70 49 74 65 72 2d 3e 69  Oldest, pIter->i
3150: 43 6f 6d 6d 69 74 49 64 29 3b 0a 20 20 20 20 20  CommitId);.     
3160: 20 7d 0a 0a 20 20 20 20 20 20 66 6f 72 28 70 50   }..      for(pP
3170: 67 3d 70 44 62 2d 3e 70 50 67 46 69 72 73 74 3b  g=pDb->pPgFirst;
3180: 20 70 50 67 20 26 26 20 70 50 67 2d 3e 69 43 6f   pPg && pPg->iCo
3190: 6d 6d 69 74 49 64 3c 69 4f 6c 64 65 73 74 3b 20  mmitId<iOldest; 
31a0: 70 50 67 3d 70 50 67 2d 3e 70 4e 65 78 74 29 7b  pPg=pPg->pNext){
31b0: 0a 20 20 20 20 20 20 20 20 69 66 28 20 70 50 67  .        if( pPg
31c0: 2d 3e 70 48 61 73 68 50 72 65 76 20 29 7b 0a 20  ->pHashPrev ){. 
31d0: 20 20 20 20 20 20 20 20 20 70 50 67 2d 3e 70 48           pPg->pH
31e0: 61 73 68 50 72 65 76 2d 3e 70 48 61 73 68 4e 65  ashPrev->pHashNe
31f0: 78 74 20 3d 20 70 50 67 2d 3e 70 48 61 73 68 4e  xt = pPg->pHashN
3200: 65 78 74 3b 0a 20 20 20 20 20 20 20 20 7d 65 6c  ext;.        }el
3210: 73 65 7b 0a 20 20 20 20 20 20 20 20 20 20 69 6e  se{.          in
3220: 74 20 69 48 61 73 68 20 3d 20 70 50 67 2d 3e 70  t iHash = pPg->p
3230: 67 6e 6f 20 25 20 48 4d 41 5f 48 41 53 48 5f 53  gno % HMA_HASH_S
3240: 49 5a 45 3b 0a 20 20 20 20 20 20 20 20 20 20 61  IZE;.          a
3250: 73 73 65 72 74 28 20 70 44 62 2d 3e 61 70 50 67  ssert( pDb->apPg
3260: 5b 69 48 61 73 68 5d 3d 3d 70 50 67 20 29 3b 0a  [iHash]==pPg );.
3270: 20 20 20 20 20 20 20 20 20 20 70 44 62 2d 3e 61            pDb->a
3280: 70 50 67 5b 69 48 61 73 68 5d 20 3d 20 70 50 67  pPg[iHash] = pPg
3290: 2d 3e 70 48 61 73 68 4e 65 78 74 3b 0a 20 20 20  ->pHashNext;.   
32a0: 20 20 20 20 20 7d 0a 20 20 20 20 20 20 20 20 69       }.        i
32b0: 66 28 20 70 50 67 2d 3e 70 48 61 73 68 4e 65 78  f( pPg->pHashNex
32c0: 74 20 29 7b 0a 20 20 20 20 20 20 20 20 20 20 70  t ){.          p
32d0: 50 67 2d 3e 70 48 61 73 68 4e 65 78 74 2d 3e 70  Pg->pHashNext->p
32e0: 48 61 73 68 50 72 65 76 20 3d 20 70 50 67 2d 3e  HashPrev = pPg->
32f0: 70 48 61 73 68 50 72 65 76 3b 0a 20 20 20 20 20  pHashPrev;.     
3300: 20 20 20 7d 0a 20 20 20 20 20 20 20 20 70 4c 61     }.        pLa
3310: 73 74 20 3d 20 70 50 67 3b 0a 20 20 20 20 20 20  st = pPg;.      
3320: 7d 0a 0a 20 20 20 20 20 20 69 66 28 20 70 4c 61  }..      if( pLa
3330: 73 74 20 29 7b 0a 20 20 20 20 20 20 20 20 61 73  st ){.        as
3340: 73 65 72 74 28 20 70 4c 61 73 74 2d 3e 70 4e 65  sert( pLast->pNe
3350: 78 74 3d 3d 70 50 67 20 29 3b 0a 20 20 20 20 20  xt==pPg );.     
3360: 20 20 20 70 4c 61 73 74 2d 3e 70 4e 65 78 74 20     pLast->pNext 
3370: 3d 20 70 44 62 2d 3e 70 46 72 65 65 3b 0a 20 20  = pDb->pFree;.  
3380: 20 20 20 20 20 20 70 44 62 2d 3e 70 46 72 65 65        pDb->pFree
3390: 20 3d 20 70 44 62 2d 3e 70 50 67 46 69 72 73 74   = pDb->pPgFirst
33a0: 3b 0a 20 20 20 20 20 20 7d 0a 0a 20 20 20 20 20  ;.      }..     
33b0: 20 69 66 28 20 70 50 67 3d 3d 30 20 29 7b 0a 20   if( pPg==0 ){. 
33c0: 20 20 20 20 20 20 20 70 44 62 2d 3e 70 50 67 46         pDb->pPgF
33d0: 69 72 73 74 20 3d 20 70 44 62 2d 3e 70 50 67 4c  irst = pDb->pPgL
33e0: 61 73 74 20 3d 20 30 3b 0a 20 20 20 20 20 20 7d  ast = 0;.      }
33f0: 65 6c 73 65 7b 0a 20 20 20 20 20 20 20 20 70 44  else{.        pD
3400: 62 2d 3e 70 50 67 46 69 72 73 74 20 3d 20 70 50  b->pPgFirst = pP
3410: 67 3b 0a 20 20 20 20 20 20 7d 0a 20 20 20 20 7d  g;.      }.    }
3420: 0a 0a 20 20 20 20 73 71 6c 69 74 65 33 5f 6d 75  ..    sqlite3_mu
3430: 74 65 78 5f 6c 65 61 76 65 28 70 44 62 2d 3e 6d  tex_leave(pDb->m
3440: 75 74 65 78 29 3b 0a 0a 20 20 20 20 70 2d 3e 70  utex);..    p->p
3450: 4e 65 78 74 20 3d 20 30 3b 0a 20 20 20 20 70 2d  Next = 0;.    p-
3460: 3e 65 54 72 61 6e 73 20 3d 20 53 45 52 56 45 52  >eTrans = SERVER
3470: 5f 54 52 41 4e 53 5f 4e 4f 4e 45 3b 0a 20 20 20  _TRANS_NONE;.   
3480: 20 70 2d 3e 69 54 72 61 6e 73 49 64 20 3d 20 2d   p->iTransId = -
3490: 31 3b 0a 20 20 20 20 70 2d 3e 69 43 6f 6d 6d 69  1;.    p->iCommi
34a0: 74 49 64 20 3d 20 30 3b 0a 20 20 7d 0a 20 20 72  tId = 0;.  }.  r
34b0: 65 74 75 72 6e 20 72 63 3b 0a 7d 0a 0a 69 6e 74  eturn rc;.}..int
34c0: 20 73 71 6c 69 74 65 33 53 65 72 76 65 72 50 72   sqlite3ServerPr
34d0: 65 43 6f 6d 6d 69 74 28 53 65 72 76 65 72 20 2a  eCommit(Server *
34e0: 70 2c 20 53 65 72 76 65 72 50 61 67 65 20 2a 70  p, ServerPage *p
34f0: 50 67 29 7b 0a 20 20 53 65 72 76 65 72 44 62 20  Pg){.  ServerDb 
3500: 2a 70 44 62 20 3d 20 70 2d 3e 70 44 62 3b 0a 20  *pDb = p->pDb;. 
3510: 20 69 6e 74 20 72 63 20 3d 20 53 51 4c 49 54 45   int rc = SQLITE
3520: 5f 4f 4b 3b 0a 20 20 53 65 72 76 65 72 50 61 67  _OK;.  ServerPag
3530: 65 20 2a 70 49 74 65 72 3b 0a 0a 20 20 69 66 28  e *pIter;..  if(
3540: 20 70 50 67 3d 3d 30 20 29 20 72 65 74 75 72 6e   pPg==0 ) return
3550: 20 53 51 4c 49 54 45 5f 4f 4b 3b 0a 0a 20 20 73   SQLITE_OK;..  s
3560: 71 6c 69 74 65 33 5f 6d 75 74 65 78 5f 65 6e 74  qlite3_mutex_ent
3570: 65 72 28 70 44 62 2d 3e 6d 75 74 65 78 29 3b 0a  er(pDb->mutex);.
3580: 0a 20 20 2f 2a 20 41 73 73 69 67 6e 20 61 20 63  .  /* Assign a c
3590: 6f 6d 6d 69 74 20 69 64 20 74 6f 20 74 68 69 73  ommit id to this
35a0: 20 74 72 61 6e 73 61 63 74 69 6f 6e 20 2a 2f 0a   transaction */.
35b0: 20 20 61 73 73 65 72 74 28 20 70 2d 3e 69 43 6f    assert( p->iCo
35c0: 6d 6d 69 74 49 64 3d 3d 30 20 29 3b 0a 20 20 61  mmitId==0 );.  a
35d0: 73 73 65 72 74 28 20 70 2d 3e 65 54 72 61 6e 73  ssert( p->eTrans
35e0: 3d 3d 53 45 52 56 45 52 5f 54 52 41 4e 53 5f 52  ==SERVER_TRANS_R
35f0: 45 41 44 57 52 49 54 45 20 29 3b 0a 20 20 61 73  EADWRITE );.  as
3600: 73 65 72 74 28 20 70 2d 3e 69 54 72 61 6e 73 49  sert( p->iTransI
3610: 64 3e 3d 30 20 29 3b 0a 0a 20 20 70 2d 3e 69 43  d>=0 );..  p->iC
3620: 6f 6d 6d 69 74 49 64 20 3d 20 70 44 62 2d 3e 69  ommitId = pDb->i
3630: 4e 65 78 74 43 6f 6d 6d 69 74 2b 2b 3b 0a 0a 20  NextCommit++;.. 
3640: 20 2f 2a 20 49 74 65 72 61 74 65 20 74 68 72 6f   /* Iterate thro
3650: 75 67 68 20 61 6c 6c 20 70 61 67 65 73 2e 20 46  ugh all pages. F
3660: 6f 72 20 65 61 63 68 3a 0a 20 20 2a 2a 0a 20 20  or each:.  **.  
3670: 2a 2a 20 20 20 31 2e 20 53 65 74 20 74 68 65 20  **   1. Set the 
3680: 69 43 6f 6d 6d 69 74 49 64 20 66 69 65 6c 64 2e  iCommitId field.
3690: 0a 20 20 2a 2a 20 20 20 32 2e 20 41 64 64 20 74  .  **   2. Add t
36a0: 68 65 20 70 61 67 65 20 74 6f 20 74 68 65 20 68  he page to the h
36b0: 61 73 68 20 74 61 62 6c 65 2e 0a 20 20 2a 2a 20  ash table..  ** 
36c0: 20 20 33 2e 20 57 61 69 74 20 75 6e 74 69 6c 20    3. Wait until 
36d0: 61 6c 6c 20 73 6c 6f 77 2d 72 65 61 64 65 72 20  all slow-reader 
36e0: 6c 6f 63 6b 73 20 68 61 76 65 20 63 6c 65 61 72  locks have clear
36f0: 65 64 2e 0a 20 20 2a 2f 0a 20 20 66 6f 72 28 70  ed..  */.  for(p
3700: 49 74 65 72 3d 70 50 67 3b 20 70 49 74 65 72 3b  Iter=pPg; pIter;
3710: 20 70 49 74 65 72 3d 70 49 74 65 72 2d 3e 70 4e   pIter=pIter->pN
3720: 65 78 74 29 7b 0a 20 20 20 20 75 33 32 20 2a 70  ext){.    u32 *p
3730: 53 6c 6f 74 20 3d 20 26 70 44 62 2d 3e 61 53 6c  Slot = &pDb->aSl
3740: 6f 74 5b 70 49 74 65 72 2d 3e 70 67 6e 6f 20 25  ot[pIter->pgno %
3750: 20 48 4d 41 5f 50 41 47 45 4c 4f 43 4b 5f 53 4c   HMA_PAGELOCK_SL
3760: 4f 54 53 5d 3b 0a 20 20 20 20 69 6e 74 20 69 48  OTS];.    int iH
3770: 61 73 68 20 3d 20 70 49 74 65 72 2d 3e 70 67 6e  ash = pIter->pgn
3780: 6f 20 25 20 48 4d 41 5f 48 41 53 48 5f 53 49 5a  o % HMA_HASH_SIZ
3790: 45 3b 0a 20 20 20 20 70 49 74 65 72 2d 3e 69 43  E;.    pIter->iC
37a0: 6f 6d 6d 69 74 49 64 20 3d 20 70 2d 3e 69 43 6f  ommitId = p->iCo
37b0: 6d 6d 69 74 49 64 3b 0a 20 20 20 20 70 49 74 65  mmitId;.    pIte
37c0: 72 2d 3e 70 48 61 73 68 4e 65 78 74 20 3d 20 70  r->pHashNext = p
37d0: 44 62 2d 3e 61 70 50 67 5b 69 48 61 73 68 5d 3b  Db->apPg[iHash];
37e0: 0a 20 20 20 20 69 66 28 20 70 49 74 65 72 2d 3e  .    if( pIter->
37f0: 70 48 61 73 68 4e 65 78 74 20 29 7b 0a 20 20 20  pHashNext ){.   
3800: 20 20 20 70 49 74 65 72 2d 3e 70 48 61 73 68 4e     pIter->pHashN
3810: 65 78 74 2d 3e 70 48 61 73 68 50 72 65 76 20 3d  ext->pHashPrev =
3820: 20 70 49 74 65 72 3b 0a 20 20 20 20 7d 0a 20 20   pIter;.    }.  
3830: 20 20 70 44 62 2d 3e 61 70 50 67 5b 69 48 61 73    pDb->apPg[iHas
3840: 68 5d 20 3d 20 70 49 74 65 72 3b 0a 0a 20 20 20  h] = pIter;..   
3850: 20 2f 2a 20 54 4f 44 4f 3a 20 53 6f 6d 65 74 68   /* TODO: Someth
3860: 69 6e 67 20 62 65 74 74 65 72 20 74 68 61 6e 20  ing better than 
3870: 74 68 69 73 21 20 2a 2f 0a 20 20 20 20 77 68 69  this! */.    whi
3880: 6c 65 28 20 73 6c 6f 74 47 65 74 53 6c 6f 77 52  le( slotGetSlowR
3890: 65 61 64 65 72 73 28 2a 70 53 6c 6f 74 29 3e 30  eaders(*pSlot)>0
38a0: 20 29 7b 0a 20 20 20 20 20 20 73 71 6c 69 74 65   ){.      sqlite
38b0: 33 5f 6d 75 74 65 78 5f 6c 65 61 76 65 28 70 44  3_mutex_leave(pD
38c0: 62 2d 3e 6d 75 74 65 78 29 3b 0a 20 20 20 20 20  b->mutex);.     
38d0: 20 73 71 6c 69 74 65 33 5f 6d 75 74 65 78 5f 65   sqlite3_mutex_e
38e0: 6e 74 65 72 28 70 44 62 2d 3e 6d 75 74 65 78 29  nter(pDb->mutex)
38f0: 3b 0a 20 20 20 20 7d 0a 0a 20 20 20 20 2f 2a 20  ;.    }..    /* 
3900: 49 66 20 70 49 74 65 72 20 69 73 20 74 68 65 20  If pIter is the 
3910: 6c 61 73 74 20 65 6c 65 6d 65 6e 74 20 69 6e 20  last element in 
3920: 74 68 65 20 6c 69 73 74 2c 20 61 70 70 65 6e 64  the list, append
3930: 20 74 68 65 20 6e 65 77 20 6c 69 73 74 20 74 6f   the new list to
3940: 0a 20 20 20 20 2a 2a 20 74 68 65 20 53 65 72 76  .    ** the Serv
3950: 65 72 44 62 2e 70 50 67 46 69 72 73 74 2f 70 50  erDb.pPgFirst/pP
3960: 67 4c 61 73 74 20 6c 69 73 74 20 61 74 20 74 68  gLast list at th
3970: 69 73 20 70 6f 69 6e 74 2e 20 20 2a 2f 0a 20 20  is point.  */.  
3980: 20 20 69 66 28 20 70 49 74 65 72 2d 3e 70 4e 65    if( pIter->pNe
3990: 78 74 3d 3d 30 20 29 7b 0a 20 20 20 20 20 20 69  xt==0 ){.      i
39a0: 66 28 20 70 44 62 2d 3e 70 50 67 4c 61 73 74 20  f( pDb->pPgLast 
39b0: 29 7b 0a 20 20 20 20 20 20 20 20 61 73 73 65 72  ){.        asser
39c0: 74 28 20 70 44 62 2d 3e 70 50 67 46 69 72 73 74  t( pDb->pPgFirst
39d0: 20 29 3b 0a 20 20 20 20 20 20 20 20 70 44 62 2d   );.        pDb-
39e0: 3e 70 50 67 4c 61 73 74 2d 3e 70 4e 65 78 74 20  >pPgLast->pNext 
39f0: 3d 20 70 50 67 3b 0a 20 20 20 20 20 20 7d 65 6c  = pPg;.      }el
3a00: 73 65 7b 0a 20 20 20 20 20 20 20 20 61 73 73 65  se{.        asse
3a10: 72 74 28 20 70 44 62 2d 3e 70 50 67 46 69 72 73  rt( pDb->pPgFirs
3a20: 74 3d 3d 30 20 29 3b 0a 20 20 20 20 20 20 20 20  t==0 );.        
3a30: 70 44 62 2d 3e 70 50 67 46 69 72 73 74 20 3d 20  pDb->pPgFirst = 
3a40: 70 50 67 3b 0a 20 20 20 20 20 20 7d 0a 20 20 20  pPg;.      }.   
3a50: 20 20 20 70 44 62 2d 3e 70 50 67 4c 61 73 74 20     pDb->pPgLast 
3a60: 3d 20 70 49 74 65 72 3b 0a 20 20 20 20 7d 0a 20  = pIter;.    }. 
3a70: 20 7d 0a 0a 20 20 2f 2a 20 41 64 64 20 74 68 69   }..  /* Add thi
3a80: 73 20 63 6f 6e 6e 65 63 74 69 6f 6e 20 74 6f 20  s connection to 
3a90: 74 68 65 20 6c 69 73 74 20 6f 66 20 63 75 72 72  the list of curr
3aa0: 65 6e 74 20 63 6f 6d 6d 69 74 74 65 72 73 20 2a  ent committers *
3ab0: 2f 0a 20 20 61 73 73 65 72 74 28 20 70 2d 3e 70  /.  assert( p->p
3ac0: 4e 65 78 74 3d 3d 30 20 29 3b 0a 20 20 70 2d 3e  Next==0 );.  p->
3ad0: 70 4e 65 78 74 20 3d 20 70 44 62 2d 3e 70 43 6f  pNext = pDb->pCo
3ae0: 6d 6d 69 74 3b 0a 20 20 70 44 62 2d 3e 70 43 6f  mmit;.  pDb->pCo
3af0: 6d 6d 69 74 20 3d 20 70 3b 0a 0a 20 20 73 71 6c  mmit = p;..  sql
3b00: 69 74 65 33 5f 6d 75 74 65 78 5f 6c 65 61 76 65  ite3_mutex_leave
3b10: 28 70 44 62 2d 3e 6d 75 74 65 78 29 3b 0a 20 20  (pDb->mutex);.  
3b20: 72 65 74 75 72 6e 20 72 63 3b 0a 7d 0a 0a 2f 2a  return rc;.}../*
3b30: 0a 2a 2a 20 52 65 6c 65 61 73 65 20 61 6c 6c 20  .** Release all 
3b40: 77 72 69 74 65 2d 6c 6f 63 6b 73 2e 0a 2a 2f 0a  write-locks..*/.
3b50: 69 6e 74 20 73 71 6c 69 74 65 33 53 65 72 76 65  int sqlite3Serve
3b60: 72 52 65 6c 65 61 73 65 57 72 69 74 65 4c 6f 63  rReleaseWriteLoc
3b70: 6b 73 28 53 65 72 76 65 72 20 2a 70 29 7b 0a 20  ks(Server *p){. 
3b80: 20 69 6e 74 20 72 63 20 3d 20 53 51 4c 49 54 45   int rc = SQLITE
3b90: 5f 4f 4b 3b 0a 20 20 72 65 74 75 72 6e 20 72 63  _OK;.  return rc
3ba0: 3b 0a 7d 0a 0a 2f 2a 0a 2a 2a 20 4c 6f 63 6b 20  ;.}../*.** Lock 
3bb0: 70 61 67 65 20 70 67 6e 6f 20 66 6f 72 20 72 65  page pgno for re
3bc0: 61 64 69 6e 67 20 28 62 57 72 69 74 65 3d 3d 30  ading (bWrite==0
3bd0: 29 20 6f 72 20 77 72 69 74 69 6e 67 20 28 62 57  ) or writing (bW
3be0: 72 69 74 65 3d 3d 31 29 2e 0a 2a 2a 0a 2a 2a 20  rite==1)..**.** 
3bf0: 49 66 20 70 61 72 61 6d 65 74 65 72 20 62 42 6c  If parameter bBl
3c00: 6f 63 6b 20 69 73 20 6e 6f 6e 2d 7a 65 72 6f 2c  ock is non-zero,
3c10: 20 74 68 65 6e 20 6d 61 6b 65 20 74 68 69 73 20   then make this 
3c20: 61 20 62 6c 6f 63 6b 69 6e 67 20 6c 6f 63 6b 20  a blocking lock 
3c30: 69 66 0a 2a 2a 20 70 6f 73 73 69 62 6c 65 2e 0a  if.** possible..
3c40: 2a 2f 0a 69 6e 74 20 73 71 6c 69 74 65 33 53 65  */.int sqlite3Se
3c50: 72 76 65 72 4c 6f 63 6b 28 53 65 72 76 65 72 20  rverLock(Server 
3c60: 2a 70 2c 20 50 67 6e 6f 20 70 67 6e 6f 2c 20 69  *p, Pgno pgno, i
3c70: 6e 74 20 62 57 72 69 74 65 2c 20 69 6e 74 20 62  nt bWrite, int b
3c80: 42 6c 6f 63 6b 29 7b 0a 20 20 69 6e 74 20 72 63  Block){.  int rc
3c90: 20 3d 20 53 51 4c 49 54 45 5f 4f 4b 3b 0a 0a 20   = SQLITE_OK;.. 
3ca0: 20 61 73 73 65 72 74 28 20 70 2d 3e 65 54 72 61   assert( p->eTra
3cb0: 6e 73 3d 3d 53 45 52 56 45 52 5f 54 52 41 4e 53  ns==SERVER_TRANS
3cc0: 5f 52 45 41 44 57 52 49 54 45 20 0a 20 20 20 20  _READWRITE .    
3cd0: 20 20 20 7c 7c 20 70 2d 3e 65 54 72 61 6e 73 3d     || p->eTrans=
3ce0: 3d 53 45 52 56 45 52 5f 54 52 41 4e 53 5f 52 45  =SERVER_TRANS_RE
3cf0: 41 44 4f 4e 4c 59 20 0a 20 20 29 3b 0a 20 20 69  ADONLY .  );.  i
3d00: 66 28 20 70 2d 3e 65 54 72 61 6e 73 3d 3d 53 45  f( p->eTrans==SE
3d10: 52 56 45 52 5f 54 52 41 4e 53 5f 52 45 41 44 57  RVER_TRANS_READW
3d20: 52 49 54 45 20 29 7b 0a 20 20 20 20 53 65 72 76  RITE ){.    Serv
3d30: 65 72 44 62 20 2a 70 44 62 20 3d 20 70 2d 3e 70  erDb *pDb = p->p
3d40: 44 62 3b 0a 20 20 20 20 69 6e 74 20 69 57 72 69  Db;.    int iWri
3d50: 74 65 72 3b 0a 20 20 20 20 69 6e 74 20 62 53 6b  ter;.    int bSk
3d60: 69 70 20 3d 20 30 3b 0a 20 20 20 20 75 33 32 20  ip = 0;.    u32 
3d70: 2a 70 53 6c 6f 74 3b 0a 0a 20 20 20 20 61 73 73  *pSlot;..    ass
3d80: 65 72 74 28 20 70 2d 3e 69 54 72 61 6e 73 49 64  ert( p->iTransId
3d90: 3e 3d 30 20 29 3b 0a 20 20 20 20 61 73 73 65 72  >=0 );.    asser
3da0: 74 28 20 70 2d 3e 6e 4c 6f 63 6b 3c 3d 70 2d 3e  t( p->nLock<=p->
3db0: 6e 41 6c 6c 6f 63 20 29 3b 0a 20 20 20 20 69 66  nAlloc );.    if
3dc0: 28 20 70 2d 3e 6e 4c 6f 63 6b 3d 3d 70 2d 3e 6e  ( p->nLock==p->n
3dd0: 41 6c 6c 6f 63 20 29 7b 0a 20 20 20 20 20 20 69  Alloc ){.      i
3de0: 6e 74 20 6e 4e 65 77 20 3d 20 70 2d 3e 6e 4c 6f  nt nNew = p->nLo
3df0: 63 6b 20 3f 20 70 2d 3e 6e 4c 6f 63 6b 2a 32 20  ck ? p->nLock*2 
3e00: 3a 20 32 35 36 3b 0a 20 20 20 20 20 20 75 33 32  : 256;.      u32
3e10: 20 2a 61 4e 65 77 20 3d 20 73 71 6c 69 74 65 33   *aNew = sqlite3
3e20: 5f 72 65 61 6c 6c 6f 63 28 70 2d 3e 61 4c 6f 63  _realloc(p->aLoc
3e30: 6b 2c 20 6e 4e 65 77 2a 73 69 7a 65 6f 66 28 75  k, nNew*sizeof(u
3e40: 33 32 29 29 3b 0a 20 20 20 20 20 20 69 66 28 20  32));.      if( 
3e50: 61 4e 65 77 3d 3d 30 20 29 20 72 65 74 75 72 6e  aNew==0 ) return
3e60: 20 53 51 4c 49 54 45 5f 4e 4f 4d 45 4d 5f 42 4b   SQLITE_NOMEM_BK
3e70: 50 54 3b 0a 20 20 20 20 20 20 6d 65 6d 73 65 74  PT;.      memset
3e80: 28 26 61 4e 65 77 5b 70 2d 3e 6e 4c 6f 63 6b 5d  (&aNew[p->nLock]
3e90: 2c 20 30 2c 20 73 69 7a 65 6f 66 28 75 33 32 29  , 0, sizeof(u32)
3ea0: 20 2a 20 28 6e 4e 65 77 20 2d 20 70 2d 3e 6e 4c   * (nNew - p->nL
3eb0: 6f 63 6b 29 29 3b 0a 20 20 20 20 20 20 70 2d 3e  ock));.      p->
3ec0: 6e 41 6c 6c 6f 63 20 3d 20 6e 4e 65 77 3b 0a 20  nAlloc = nNew;. 
3ed0: 20 20 20 20 20 70 2d 3e 61 4c 6f 63 6b 20 3d 20       p->aLock = 
3ee0: 61 4e 65 77 3b 0a 20 20 20 20 7d 0a 0a 20 20 20  aNew;.    }..   
3ef0: 20 73 71 6c 69 74 65 33 5f 6d 75 74 65 78 5f 65   sqlite3_mutex_e
3f00: 6e 74 65 72 28 70 44 62 2d 3e 6d 75 74 65 78 29  nter(pDb->mutex)
3f10: 3b 0a 0a 20 20 20 20 70 53 6c 6f 74 20 3d 20 26  ;..    pSlot = &
3f20: 70 44 62 2d 3e 61 53 6c 6f 74 5b 70 67 6e 6f 20  pDb->aSlot[pgno 
3f30: 25 20 48 4d 41 5f 50 41 47 45 4c 4f 43 4b 5f 53  % HMA_PAGELOCK_S
3f40: 4c 4f 54 53 5d 3b 0a 20 20 20 20 61 73 73 65 72  LOTS];.    asser
3f50: 74 28 20 73 6c 6f 74 47 65 74 57 72 69 74 65 72  t( slotGetWriter
3f60: 28 2a 70 53 6c 6f 74 29 3c 30 20 0a 20 20 20 20  (*pSlot)<0 .    
3f70: 20 20 20 20 7c 7c 20 73 6c 6f 74 52 65 61 64 65      || slotReade
3f80: 72 4d 61 73 6b 28 2a 70 53 6c 6f 74 29 3d 3d 30  rMask(*pSlot)==0
3f90: 20 0a 20 20 20 20 20 20 20 20 7c 7c 20 73 6c 6f   .        || slo
3fa0: 74 52 65 61 64 65 72 4d 61 73 6b 28 2a 70 53 6c  tReaderMask(*pSl
3fb0: 6f 74 29 3d 3d 28 31 20 3c 3c 20 73 6c 6f 74 47  ot)==(1 << slotG
3fc0: 65 74 57 72 69 74 65 72 28 2a 70 53 6c 6f 74 29  etWriter(*pSlot)
3fd0: 29 0a 20 20 20 20 20 20 20 20 29 3b 0a 0a 20 20  ).        );..  
3fe0: 20 20 69 57 72 69 74 65 72 20 3d 20 73 6c 6f 74    iWriter = slot
3ff0: 47 65 74 57 72 69 74 65 72 28 2a 70 53 6c 6f 74  GetWriter(*pSlot
4000: 29 3b 0a 20 20 20 20 69 66 28 20 69 57 72 69 74  );.    if( iWrit
4010: 65 72 3d 3d 70 2d 3e 69 54 72 61 6e 73 49 64 20  er==p->iTransId 
4020: 7c 7c 20 28 62 57 72 69 74 65 3d 3d 30 20 26 26  || (bWrite==0 &&
4030: 20 28 2a 70 53 6c 6f 74 20 26 20 28 31 3c 3c 70   (*pSlot & (1<<p
4040: 2d 3e 69 54 72 61 6e 73 49 64 29 29 29 20 29 7b  ->iTransId))) ){
4050: 0a 20 20 20 20 20 20 62 53 6b 69 70 20 3d 20 31  .      bSkip = 1
4060: 3b 0a 20 20 20 20 7d 65 6c 73 65 20 69 66 28 20  ;.    }else if( 
4070: 69 57 72 69 74 65 72 3e 3d 30 20 29 7b 0a 20 20  iWriter>=0 ){.  
4080: 20 20 20 20 72 63 20 3d 20 53 51 4c 49 54 45 5f      rc = SQLITE_
4090: 42 55 53 59 5f 44 45 41 44 4c 4f 43 4b 3b 0a 20  BUSY_DEADLOCK;. 
40a0: 20 20 20 7d 65 6c 73 65 20 69 66 28 20 62 57 72     }else if( bWr
40b0: 69 74 65 20 29 7b 0a 20 20 20 20 20 20 69 66 28  ite ){.      if(
40c0: 20 28 73 6c 6f 74 52 65 61 64 65 72 4d 61 73 6b   (slotReaderMask
40d0: 28 2a 70 53 6c 6f 74 29 20 26 20 7e 28 31 20 3c  (*pSlot) & ~(1 <
40e0: 3c 20 70 2d 3e 69 54 72 61 6e 73 49 64 29 29 3d  < p->iTransId))=
40f0: 3d 30 20 29 7b 0a 20 20 20 20 20 20 20 20 2a 70  =0 ){.        *p
4100: 53 6c 6f 74 20 2b 3d 20 28 28 70 2d 3e 69 54 72  Slot += ((p->iTr
4110: 61 6e 73 49 64 20 2b 20 31 29 20 3c 3c 20 48 4d  ansId + 1) << HM
4120: 41 5f 4d 41 58 5f 54 52 41 4e 53 41 43 54 49 4f  A_MAX_TRANSACTIO
4130: 4e 49 44 29 3b 0a 20 20 20 20 20 20 7d 65 6c 73  NID);.      }els
4140: 65 7b 0a 20 20 20 20 20 20 20 20 72 63 20 3d 20  e{.        rc = 
4150: 53 51 4c 49 54 45 5f 42 55 53 59 5f 44 45 41 44  SQLITE_BUSY_DEAD
4160: 4c 4f 43 4b 3b 0a 20 20 20 20 20 20 7d 0a 20 20  LOCK;.      }.  
4170: 20 20 7d 65 6c 73 65 7b 0a 20 20 20 20 20 20 2a    }else{.      *
4180: 70 53 6c 6f 74 20 7c 3d 20 28 31 20 3c 3c 20 70  pSlot |= (1 << p
4190: 2d 3e 69 54 72 61 6e 73 49 64 29 3b 0a 20 20 20  ->iTransId);.   
41a0: 20 7d 0a 0a 20 20 20 20 61 73 73 65 72 74 28 20   }..    assert( 
41b0: 73 6c 6f 74 47 65 74 57 72 69 74 65 72 28 2a 70  slotGetWriter(*p
41c0: 53 6c 6f 74 29 3c 30 20 0a 20 20 20 20 20 20 20  Slot)<0 .       
41d0: 20 7c 7c 20 73 6c 6f 74 52 65 61 64 65 72 4d 61   || slotReaderMa
41e0: 73 6b 28 2a 70 53 6c 6f 74 29 3d 3d 30 20 0a 20  sk(*pSlot)==0 . 
41f0: 20 20 20 20 20 20 20 7c 7c 20 73 6c 6f 74 52 65         || slotRe
4200: 61 64 65 72 4d 61 73 6b 28 2a 70 53 6c 6f 74 29  aderMask(*pSlot)
4210: 3d 3d 28 31 20 3c 3c 20 73 6c 6f 74 47 65 74 57  ==(1 << slotGetW
4220: 72 69 74 65 72 28 2a 70 53 6c 6f 74 29 29 0a 20  riter(*pSlot)). 
4230: 20 20 20 20 20 20 20 29 3b 0a 0a 20 20 20 20 73         );..    s
4240: 71 6c 69 74 65 33 5f 6d 75 74 65 78 5f 6c 65 61  qlite3_mutex_lea
4250: 76 65 28 70 44 62 2d 3e 6d 75 74 65 78 29 3b 0a  ve(pDb->mutex);.
4260: 0a 20 20 20 20 69 66 28 20 62 53 6b 69 70 3d 3d  .    if( bSkip==
4270: 30 20 29 7b 0a 20 20 20 20 20 20 70 2d 3e 61 4c  0 ){.      p->aL
4280: 6f 63 6b 5b 70 2d 3e 6e 4c 6f 63 6b 2b 2b 5d 20  ock[p->nLock++] 
4290: 3d 20 70 67 6e 6f 3b 0a 20 20 20 20 7d 0a 20 20  = pgno;.    }.  
42a0: 7d 0a 0a 20 20 72 65 74 75 72 6e 20 72 63 3b 0a  }..  return rc;.
42b0: 7d 0a 0a 69 6e 74 20 73 71 6c 69 74 65 33 53 65  }..int sqlite3Se
42c0: 72 76 65 72 48 61 73 4c 6f 63 6b 28 53 65 72 76  rverHasLock(Serv
42d0: 65 72 20 2a 70 2c 20 50 67 6e 6f 20 70 67 6e 6f  er *p, Pgno pgno
42e0: 2c 20 69 6e 74 20 62 57 72 69 74 65 29 7b 0a 20  , int bWrite){. 
42f0: 20 61 73 73 65 72 74 28 20 30 20 29 3b 0a 20 20   assert( 0 );.  
4300: 72 65 74 75 72 6e 20 30 3b 0a 7d 0a 0a 73 74 61  return 0;.}..sta
4310: 74 69 63 20 76 6f 69 64 20 73 65 72 76 65 72 49  tic void serverI
4320: 6e 63 72 53 6c 6f 77 52 65 61 64 65 72 28 75 33  ncrSlowReader(u3
4330: 32 20 2a 70 53 6c 6f 74 2c 20 69 6e 74 20 6e 29  2 *pSlot, int n)
4340: 7b 0a 20 20 61 73 73 65 72 74 28 20 6e 3d 3d 31  {.  assert( n==1
4350: 20 7c 7c 20 6e 3d 3d 2d 31 20 29 3b 0a 20 20 2a   || n==-1 );.  *
4360: 70 53 6c 6f 74 20 2b 3d 20 28 6e 20 2a 20 28 31  pSlot += (n * (1
4370: 20 3c 3c 20 48 4d 41 5f 53 4c 4f 54 5f 52 4c 57   << HMA_SLOT_RLW
4380: 4c 5f 42 49 54 53 29 29 3b 0a 7d 0a 0a 76 6f 69  L_BITS));.}..voi
4390: 64 20 73 71 6c 69 74 65 33 53 65 72 76 65 72 52  d sqlite3ServerR
43a0: 65 61 64 50 61 67 65 28 53 65 72 76 65 72 20 2a  eadPage(Server *
43b0: 70 2c 20 50 67 6e 6f 20 70 67 6e 6f 2c 20 75 38  p, Pgno pgno, u8
43c0: 20 2a 2a 70 70 44 61 74 61 29 7b 0a 20 20 69 66   **ppData){.  if
43d0: 28 20 70 2d 3e 65 54 72 61 6e 73 3d 3d 53 45 52  ( p->eTrans==SER
43e0: 56 45 52 5f 54 52 41 4e 53 5f 52 45 41 44 4f 4e  VER_TRANS_READON
43f0: 4c 59 20 29 7b 0a 20 20 20 20 53 65 72 76 65 72  LY ){.    Server
4400: 44 62 20 2a 70 44 62 20 3d 20 70 2d 3e 70 44 62  Db *pDb = p->pDb
4410: 3b 0a 20 20 20 20 53 65 72 76 65 72 50 61 67 65  ;.    ServerPage
4420: 20 2a 70 49 74 65 72 3b 0a 20 20 20 20 53 65 72   *pIter;.    Ser
4430: 76 65 72 50 61 67 65 20 2a 70 42 65 73 74 20 3d  verPage *pBest =
4440: 20 30 3b 0a 20 20 20 20 69 6e 74 20 69 48 61 73   0;.    int iHas
4450: 68 20 3d 20 70 67 6e 6f 20 25 20 48 4d 41 5f 48  h = pgno % HMA_H
4460: 41 53 48 5f 53 49 5a 45 3b 0a 0a 20 20 20 20 73  ASH_SIZE;..    s
4470: 71 6c 69 74 65 33 5f 6d 75 74 65 78 5f 65 6e 74  qlite3_mutex_ent
4480: 65 72 28 70 44 62 2d 3e 6d 75 74 65 78 29 3b 0a  er(pDb->mutex);.
4490: 0a 20 20 20 20 2f 2a 20 53 65 61 72 63 68 20 74  .    /* Search t
44a0: 68 65 20 68 61 73 68 20 74 61 62 6c 65 20 66 6f  he hash table fo
44b0: 72 20 74 68 65 20 6f 6c 64 65 73 74 20 76 65 72  r the oldest ver
44c0: 73 69 6f 6e 20 6f 66 20 70 61 67 65 20 70 67 6e  sion of page pgn
44d0: 6f 20 77 69 74 68 0a 20 20 20 20 2a 2a 20 61 20  o with.    ** a 
44e0: 63 6f 6d 6d 69 74 2d 69 64 20 67 72 65 61 74 65  commit-id greate
44f0: 72 20 74 68 61 6e 20 6f 72 20 65 71 75 61 6c 20  r than or equal 
4500: 74 6f 20 53 65 72 76 65 72 2e 69 43 6f 6d 6d 69  to Server.iCommi
4510: 74 49 64 2e 20 20 2a 2f 0a 20 20 20 20 66 6f 72  tId.  */.    for
4520: 28 70 49 74 65 72 3d 70 44 62 2d 3e 61 70 50 67  (pIter=pDb->apPg
4530: 5b 69 48 61 73 68 5d 3b 20 70 49 74 65 72 3b 20  [iHash]; pIter; 
4540: 70 49 74 65 72 3d 70 49 74 65 72 2d 3e 70 48 61  pIter=pIter->pHa
4550: 73 68 4e 65 78 74 29 7b 0a 20 20 20 20 20 20 69  shNext){.      i
4560: 66 28 20 70 49 74 65 72 2d 3e 70 67 6e 6f 3d 3d  f( pIter->pgno==
4570: 70 67 6e 6f 20 0a 20 20 20 20 20 20 20 26 26 20  pgno .       && 
4580: 70 49 74 65 72 2d 3e 69 43 6f 6d 6d 69 74 49 64  pIter->iCommitId
4590: 3e 3d 70 2d 3e 69 43 6f 6d 6d 69 74 49 64 20 0a  >=p->iCommitId .
45a0: 20 20 20 20 20 20 20 26 26 20 28 70 42 65 73 74         && (pBest
45b0: 3d 3d 30 20 7c 7c 20 70 49 74 65 72 2d 3e 69 43  ==0 || pIter->iC
45c0: 6f 6d 6d 69 74 49 64 3c 70 42 65 73 74 2d 3e 69  ommitId<pBest->i
45d0: 43 6f 6d 6d 69 74 49 64 29 20 0a 20 20 20 20 20  CommitId) .     
45e0: 20 29 7b 0a 20 20 20 20 20 20 20 20 70 42 65 73   ){.        pBes
45f0: 74 20 3d 20 70 49 74 65 72 3b 0a 20 20 20 20 20  t = pIter;.     
4600: 20 7d 0a 20 20 20 20 7d 0a 0a 20 20 20 20 69 66   }.    }..    if
4610: 28 20 70 42 65 73 74 20 29 7b 0a 20 20 20 20 20  ( pBest ){.     
4620: 20 2a 70 70 44 61 74 61 20 3d 20 70 42 65 73 74   *ppData = pBest
4630: 2d 3e 61 44 61 74 61 3b 0a 20 20 20 20 7d 65 6c  ->aData;.    }el
4640: 73 65 7b 0a 20 20 20 20 20 20 75 33 32 20 2a 70  se{.      u32 *p
4650: 53 6c 6f 74 20 3d 20 26 70 44 62 2d 3e 61 53 6c  Slot = &pDb->aSl
4660: 6f 74 5b 70 67 6e 6f 20 25 20 48 4d 41 5f 50 41  ot[pgno % HMA_PA
4670: 47 45 4c 4f 43 4b 5f 53 4c 4f 54 53 5d 3b 0a 20  GELOCK_SLOTS];. 
4680: 20 20 20 20 20 73 65 72 76 65 72 49 6e 63 72 53       serverIncrS
4690: 6c 6f 77 52 65 61 64 65 72 28 70 53 6c 6f 74 2c  lowReader(pSlot,
46a0: 20 31 29 3b 0a 20 20 20 20 7d 0a 0a 20 20 20 20   1);.    }..    
46b0: 73 71 6c 69 74 65 33 5f 6d 75 74 65 78 5f 6c 65  sqlite3_mutex_le
46c0: 61 76 65 28 70 44 62 2d 3e 6d 75 74 65 78 29 3b  ave(pDb->mutex);
46d0: 0a 20 20 7d 0a 7d 0a 0a 76 6f 69 64 20 73 71 6c  .  }.}..void sql
46e0: 69 74 65 33 53 65 72 76 65 72 45 6e 64 52 65 61  ite3ServerEndRea
46f0: 64 50 61 67 65 28 53 65 72 76 65 72 20 2a 70 2c  dPage(Server *p,
4700: 20 50 67 6e 6f 20 70 67 6e 6f 29 7b 0a 20 20 69   Pgno pgno){.  i
4710: 66 28 20 70 2d 3e 65 54 72 61 6e 73 3d 3d 53 45  f( p->eTrans==SE
4720: 52 56 45 52 5f 54 52 41 4e 53 5f 52 45 41 44 4f  RVER_TRANS_READO
4730: 4e 4c 59 20 29 7b 0a 20 20 20 20 53 65 72 76 65  NLY ){.    Serve
4740: 72 44 62 20 2a 70 44 62 20 3d 20 70 2d 3e 70 44  rDb *pDb = p->pD
4750: 62 3b 0a 20 20 20 20 75 33 32 20 2a 70 53 6c 6f  b;.    u32 *pSlo
4760: 74 20 3d 20 26 70 44 62 2d 3e 61 53 6c 6f 74 5b  t = &pDb->aSlot[
4770: 70 67 6e 6f 20 25 20 48 4d 41 5f 50 41 47 45 4c  pgno % HMA_PAGEL
4780: 4f 43 4b 5f 53 4c 4f 54 53 5d 3b 0a 20 20 20 20  OCK_SLOTS];.    
4790: 73 71 6c 69 74 65 33 5f 6d 75 74 65 78 5f 65 6e  sqlite3_mutex_en
47a0: 74 65 72 28 70 44 62 2d 3e 6d 75 74 65 78 29 3b  ter(pDb->mutex);
47b0: 0a 20 20 20 20 73 65 72 76 65 72 49 6e 63 72 53  .    serverIncrS
47c0: 6c 6f 77 52 65 61 64 65 72 28 70 53 6c 6f 74 2c  lowReader(pSlot,
47d0: 20 2d 31 29 3b 0a 20 20 20 20 61 73 73 65 72 74   -1);.    assert
47e0: 28 20 73 6c 6f 74 47 65 74 53 6c 6f 77 52 65 61  ( slotGetSlowRea
47f0: 64 65 72 73 28 2a 70 53 6c 6f 74 29 3e 3d 30 20  ders(*pSlot)>=0 
4800: 29 3b 0a 20 20 20 20 73 71 6c 69 74 65 33 5f 6d  );.    sqlite3_m
4810: 75 74 65 78 5f 6c 65 61 76 65 28 70 44 62 2d 3e  utex_leave(pDb->
4820: 6d 75 74 65 78 29 3b 0a 20 20 7d 0a 7d 0a 0a 53  mutex);.  }.}..S
4830: 65 72 76 65 72 50 61 67 65 20 2a 73 71 6c 69 74  erverPage *sqlit
4840: 65 33 53 65 72 76 65 72 42 75 66 66 65 72 28 53  e3ServerBuffer(S
4850: 65 72 76 65 72 20 2a 70 29 7b 0a 20 20 53 65 72  erver *p){.  Ser
4860: 76 65 72 44 62 20 2a 70 44 62 20 3d 20 70 2d 3e  verDb *pDb = p->
4870: 70 44 62 3b 0a 20 20 53 65 72 76 65 72 50 61 67  pDb;.  ServerPag
4880: 65 20 2a 70 52 65 74 20 3d 20 30 3b 0a 20 20 73  e *pRet = 0;.  s
4890: 71 6c 69 74 65 33 5f 6d 75 74 65 78 5f 65 6e 74  qlite3_mutex_ent
48a0: 65 72 28 70 44 62 2d 3e 6d 75 74 65 78 29 3b 0a  er(pDb->mutex);.
48b0: 20 20 69 66 28 20 70 44 62 2d 3e 70 46 72 65 65    if( pDb->pFree
48c0: 20 29 7b 0a 20 20 20 20 70 52 65 74 20 3d 20 70   ){.    pRet = p
48d0: 44 62 2d 3e 70 46 72 65 65 3b 0a 20 20 20 20 70  Db->pFree;.    p
48e0: 44 62 2d 3e 70 46 72 65 65 20 3d 20 70 52 65 74  Db->pFree = pRet
48f0: 2d 3e 70 4e 65 78 74 3b 0a 20 20 20 20 70 52 65  ->pNext;.    pRe
4900: 74 2d 3e 70 4e 65 78 74 20 3d 20 30 3b 0a 20 20  t->pNext = 0;.  
4910: 7d 0a 20 20 73 71 6c 69 74 65 33 5f 6d 75 74 65  }.  sqlite3_mute
4920: 78 5f 6c 65 61 76 65 28 70 44 62 2d 3e 6d 75 74  x_leave(pDb->mut
4930: 65 78 29 3b 0a 20 20 72 65 74 75 72 6e 20 70 52  ex);.  return pR
4940: 65 74 3b 0a 7d 0a 0a 23 65 6e 64 69 66 20 2f 2a  et;.}..#endif /*
4950: 20 69 66 64 65 66 20 53 51 4c 49 54 45 5f 53 45   ifdef SQLITE_SE
4960: 52 56 45 52 5f 45 44 49 54 49 4f 4e 20 2a 2f 0a  RVER_EDITION */.