os/persistentdata/persistentstorage/sql/SQLite364/callback.c
author sl
Tue, 10 Jun 2014 14:32:02 +0200
changeset 1 260cb5ec6c19
permissions -rw-r--r--
Update contrib.
sl@0
     1
/*
sl@0
     2
** 2005 May 23 
sl@0
     3
**
sl@0
     4
** The author disclaims copyright to this source code.  In place of
sl@0
     5
** a legal notice, here is a blessing:
sl@0
     6
**
sl@0
     7
**    May you do good and not evil.
sl@0
     8
**    May you find forgiveness for yourself and forgive others.
sl@0
     9
**    May you share freely, never taking more than you give.
sl@0
    10
**
sl@0
    11
*************************************************************************
sl@0
    12
**
sl@0
    13
** This file contains functions used to access the internal hash tables
sl@0
    14
** of user defined functions and collation sequences.
sl@0
    15
**
sl@0
    16
** $Id: callback.c,v 1.32 2008/10/10 17:41:29 drh Exp $
sl@0
    17
*/
sl@0
    18
sl@0
    19
#include "sqliteInt.h"
sl@0
    20
sl@0
    21
/*
sl@0
    22
** Invoke the 'collation needed' callback to request a collation sequence
sl@0
    23
** in the database text encoding of name zName, length nName.
sl@0
    24
** If the collation sequence
sl@0
    25
*/
sl@0
    26
static void callCollNeeded(sqlite3 *db, const char *zName, int nName){
sl@0
    27
  assert( !db->xCollNeeded || !db->xCollNeeded16 );
sl@0
    28
  if( nName<0 ) nName = sqlite3Strlen(db, zName);
sl@0
    29
  if( db->xCollNeeded ){
sl@0
    30
    char *zExternal = sqlite3DbStrNDup(db, zName, nName);
sl@0
    31
    if( !zExternal ) return;
sl@0
    32
    db->xCollNeeded(db->pCollNeededArg, db, (int)ENC(db), zExternal);
sl@0
    33
    sqlite3DbFree(db, zExternal);
sl@0
    34
  }
sl@0
    35
#ifndef SQLITE_OMIT_UTF16
sl@0
    36
  if( db->xCollNeeded16 ){
sl@0
    37
    char const *zExternal;
sl@0
    38
    sqlite3_value *pTmp = sqlite3ValueNew(db);
sl@0
    39
    sqlite3ValueSetStr(pTmp, nName, zName, SQLITE_UTF8, SQLITE_STATIC);
sl@0
    40
    zExternal = sqlite3ValueText(pTmp, SQLITE_UTF16NATIVE);
sl@0
    41
    if( zExternal ){
sl@0
    42
      db->xCollNeeded16(db->pCollNeededArg, db, (int)ENC(db), zExternal);
sl@0
    43
    }
sl@0
    44
    sqlite3ValueFree(pTmp);
sl@0
    45
  }
sl@0
    46
#endif
sl@0
    47
}
sl@0
    48
sl@0
    49
/*
sl@0
    50
** This routine is called if the collation factory fails to deliver a
sl@0
    51
** collation function in the best encoding but there may be other versions
sl@0
    52
** of this collation function (for other text encodings) available. Use one
sl@0
    53
** of these instead if they exist. Avoid a UTF-8 <-> UTF-16 conversion if
sl@0
    54
** possible.
sl@0
    55
*/
sl@0
    56
static int synthCollSeq(sqlite3 *db, CollSeq *pColl){
sl@0
    57
  CollSeq *pColl2;
sl@0
    58
  char *z = pColl->zName;
sl@0
    59
  int n = strlen(z);
sl@0
    60
  int i;
sl@0
    61
  static const u8 aEnc[] = { SQLITE_UTF16BE, SQLITE_UTF16LE, SQLITE_UTF8 };
sl@0
    62
  for(i=0; i<3; i++){
sl@0
    63
    pColl2 = sqlite3FindCollSeq(db, aEnc[i], z, n, 0);
sl@0
    64
    if( pColl2->xCmp!=0 ){
sl@0
    65
      memcpy(pColl, pColl2, sizeof(CollSeq));
sl@0
    66
      pColl->xDel = 0;         /* Do not copy the destructor */
sl@0
    67
      return SQLITE_OK;
sl@0
    68
    }
sl@0
    69
  }
sl@0
    70
  return SQLITE_ERROR;
sl@0
    71
}
sl@0
    72
sl@0
    73
/*
sl@0
    74
** This function is responsible for invoking the collation factory callback
sl@0
    75
** or substituting a collation sequence of a different encoding when the
sl@0
    76
** requested collation sequence is not available in the database native
sl@0
    77
** encoding.
sl@0
    78
** 
sl@0
    79
** If it is not NULL, then pColl must point to the database native encoding 
sl@0
    80
** collation sequence with name zName, length nName.
sl@0
    81
**
sl@0
    82
** The return value is either the collation sequence to be used in database
sl@0
    83
** db for collation type name zName, length nName, or NULL, if no collation
sl@0
    84
** sequence can be found.
sl@0
    85
*/
sl@0
    86
CollSeq *sqlite3GetCollSeq(
sl@0
    87
  sqlite3* db, 
sl@0
    88
  CollSeq *pColl, 
sl@0
    89
  const char *zName, 
sl@0
    90
  int nName
sl@0
    91
){
sl@0
    92
  CollSeq *p;
sl@0
    93
sl@0
    94
  p = pColl;
sl@0
    95
  if( !p ){
sl@0
    96
    p = sqlite3FindCollSeq(db, ENC(db), zName, nName, 0);
sl@0
    97
  }
sl@0
    98
  if( !p || !p->xCmp ){
sl@0
    99
    /* No collation sequence of this type for this encoding is registered.
sl@0
   100
    ** Call the collation factory to see if it can supply us with one.
sl@0
   101
    */
sl@0
   102
    callCollNeeded(db, zName, nName);
sl@0
   103
    p = sqlite3FindCollSeq(db, ENC(db), zName, nName, 0);
sl@0
   104
  }
sl@0
   105
  if( p && !p->xCmp && synthCollSeq(db, p) ){
sl@0
   106
    p = 0;
sl@0
   107
  }
sl@0
   108
  assert( !p || p->xCmp );
sl@0
   109
  return p;
sl@0
   110
}
sl@0
   111
sl@0
   112
/*
sl@0
   113
** This routine is called on a collation sequence before it is used to
sl@0
   114
** check that it is defined. An undefined collation sequence exists when
sl@0
   115
** a database is loaded that contains references to collation sequences
sl@0
   116
** that have not been defined by sqlite3_create_collation() etc.
sl@0
   117
**
sl@0
   118
** If required, this routine calls the 'collation needed' callback to
sl@0
   119
** request a definition of the collating sequence. If this doesn't work, 
sl@0
   120
** an equivalent collating sequence that uses a text encoding different
sl@0
   121
** from the main database is substituted, if one is available.
sl@0
   122
*/
sl@0
   123
int sqlite3CheckCollSeq(Parse *pParse, CollSeq *pColl){
sl@0
   124
  if( pColl ){
sl@0
   125
    const char *zName = pColl->zName;
sl@0
   126
    CollSeq *p = sqlite3GetCollSeq(pParse->db, pColl, zName, -1);
sl@0
   127
    if( !p ){
sl@0
   128
      if( pParse->nErr==0 ){
sl@0
   129
        sqlite3ErrorMsg(pParse, "no such collation sequence: %s", zName);
sl@0
   130
      }
sl@0
   131
      pParse->nErr++;
sl@0
   132
      return SQLITE_ERROR;
sl@0
   133
    }
sl@0
   134
    assert( p==pColl );
sl@0
   135
  }
sl@0
   136
  return SQLITE_OK;
sl@0
   137
}
sl@0
   138
sl@0
   139
sl@0
   140
sl@0
   141
/*
sl@0
   142
** Locate and return an entry from the db.aCollSeq hash table. If the entry
sl@0
   143
** specified by zName and nName is not found and parameter 'create' is
sl@0
   144
** true, then create a new entry. Otherwise return NULL.
sl@0
   145
**
sl@0
   146
** Each pointer stored in the sqlite3.aCollSeq hash table contains an
sl@0
   147
** array of three CollSeq structures. The first is the collation sequence
sl@0
   148
** prefferred for UTF-8, the second UTF-16le, and the third UTF-16be.
sl@0
   149
**
sl@0
   150
** Stored immediately after the three collation sequences is a copy of
sl@0
   151
** the collation sequence name. A pointer to this string is stored in
sl@0
   152
** each collation sequence structure.
sl@0
   153
*/
sl@0
   154
static CollSeq *findCollSeqEntry(
sl@0
   155
  sqlite3 *db,
sl@0
   156
  const char *zName,
sl@0
   157
  int nName,
sl@0
   158
  int create
sl@0
   159
){
sl@0
   160
  CollSeq *pColl;
sl@0
   161
  if( nName<0 ) nName = sqlite3Strlen(db, zName);
sl@0
   162
  pColl = sqlite3HashFind(&db->aCollSeq, zName, nName);
sl@0
   163
sl@0
   164
  if( 0==pColl && create ){
sl@0
   165
    pColl = sqlite3DbMallocZero(db, 3*sizeof(*pColl) + nName + 1 );
sl@0
   166
    if( pColl ){
sl@0
   167
      CollSeq *pDel = 0;
sl@0
   168
      pColl[0].zName = (char*)&pColl[3];
sl@0
   169
      pColl[0].enc = SQLITE_UTF8;
sl@0
   170
      pColl[1].zName = (char*)&pColl[3];
sl@0
   171
      pColl[1].enc = SQLITE_UTF16LE;
sl@0
   172
      pColl[2].zName = (char*)&pColl[3];
sl@0
   173
      pColl[2].enc = SQLITE_UTF16BE;
sl@0
   174
      memcpy(pColl[0].zName, zName, nName);
sl@0
   175
      pColl[0].zName[nName] = 0;
sl@0
   176
      pDel = sqlite3HashInsert(&db->aCollSeq, pColl[0].zName, nName, pColl);
sl@0
   177
sl@0
   178
      /* If a malloc() failure occured in sqlite3HashInsert(), it will 
sl@0
   179
      ** return the pColl pointer to be deleted (because it wasn't added
sl@0
   180
      ** to the hash table).
sl@0
   181
      */
sl@0
   182
      assert( pDel==0 || pDel==pColl );
sl@0
   183
      if( pDel!=0 ){
sl@0
   184
        db->mallocFailed = 1;
sl@0
   185
        sqlite3DbFree(db, pDel);
sl@0
   186
        pColl = 0;
sl@0
   187
      }
sl@0
   188
    }
sl@0
   189
  }
sl@0
   190
  return pColl;
sl@0
   191
}
sl@0
   192
sl@0
   193
/*
sl@0
   194
** Parameter zName points to a UTF-8 encoded string nName bytes long.
sl@0
   195
** Return the CollSeq* pointer for the collation sequence named zName
sl@0
   196
** for the encoding 'enc' from the database 'db'.
sl@0
   197
**
sl@0
   198
** If the entry specified is not found and 'create' is true, then create a
sl@0
   199
** new entry.  Otherwise return NULL.
sl@0
   200
**
sl@0
   201
** A separate function sqlite3LocateCollSeq() is a wrapper around
sl@0
   202
** this routine.  sqlite3LocateCollSeq() invokes the collation factory
sl@0
   203
** if necessary and generates an error message if the collating sequence
sl@0
   204
** cannot be found.
sl@0
   205
*/
sl@0
   206
CollSeq *sqlite3FindCollSeq(
sl@0
   207
  sqlite3 *db,
sl@0
   208
  u8 enc,
sl@0
   209
  const char *zName,
sl@0
   210
  int nName,
sl@0
   211
  int create
sl@0
   212
){
sl@0
   213
  CollSeq *pColl;
sl@0
   214
  if( zName ){
sl@0
   215
    pColl = findCollSeqEntry(db, zName, nName, create);
sl@0
   216
  }else{
sl@0
   217
    pColl = db->pDfltColl;
sl@0
   218
  }
sl@0
   219
  assert( SQLITE_UTF8==1 && SQLITE_UTF16LE==2 && SQLITE_UTF16BE==3 );
sl@0
   220
  assert( enc>=SQLITE_UTF8 && enc<=SQLITE_UTF16BE );
sl@0
   221
  if( pColl ) pColl += enc-1;
sl@0
   222
  return pColl;
sl@0
   223
}
sl@0
   224
sl@0
   225
/* During the search for the best function definition, this procedure
sl@0
   226
** is called to test how well the function passed as the first argument
sl@0
   227
** matches the request for a function with nArg arguments in a system
sl@0
   228
** that uses encoding enc. The value returned indicates how well the
sl@0
   229
** request is matched. A higher value indicates a better match.
sl@0
   230
**
sl@0
   231
** The returned value is always between 1 and 6, as follows:
sl@0
   232
**
sl@0
   233
** 1: A variable arguments function that prefers UTF-8 when a UTF-16
sl@0
   234
**    encoding is requested, or vice versa.
sl@0
   235
** 2: A variable arguments function that uses UTF-16BE when UTF-16LE is
sl@0
   236
**    requested, or vice versa.
sl@0
   237
** 3: A variable arguments function using the same text encoding.
sl@0
   238
** 4: A function with the exact number of arguments requested that
sl@0
   239
**    prefers UTF-8 when a UTF-16 encoding is requested, or vice versa.
sl@0
   240
** 5: A function with the exact number of arguments requested that
sl@0
   241
**    prefers UTF-16LE when UTF-16BE is requested, or vice versa.
sl@0
   242
** 6: An exact match.
sl@0
   243
**
sl@0
   244
*/
sl@0
   245
static int matchQuality(FuncDef *p, int nArg, u8 enc){
sl@0
   246
  int match = 0;
sl@0
   247
  if( p->nArg==-1 || p->nArg==nArg || nArg==-1 ){
sl@0
   248
    match = 1;
sl@0
   249
    if( p->nArg==nArg || nArg==-1 ){
sl@0
   250
      match = 4;
sl@0
   251
    }
sl@0
   252
    if( enc==p->iPrefEnc ){
sl@0
   253
      match += 2;
sl@0
   254
    }
sl@0
   255
    else if( (enc==SQLITE_UTF16LE && p->iPrefEnc==SQLITE_UTF16BE) ||
sl@0
   256
             (enc==SQLITE_UTF16BE && p->iPrefEnc==SQLITE_UTF16LE) ){
sl@0
   257
      match += 1;
sl@0
   258
    }
sl@0
   259
  }
sl@0
   260
  return match;
sl@0
   261
}
sl@0
   262
sl@0
   263
/*
sl@0
   264
** Search a FuncDefHash for a function with the given name.  Return
sl@0
   265
** a pointer to the matching FuncDef if found, or 0 if there is no match.
sl@0
   266
*/
sl@0
   267
static FuncDef *functionSearch(
sl@0
   268
  FuncDefHash *pHash,  /* Hash table to search */
sl@0
   269
  int h,               /* Hash of the name */
sl@0
   270
  const char *zFunc,   /* Name of function */
sl@0
   271
  int nFunc            /* Number of bytes in zFunc */
sl@0
   272
){
sl@0
   273
  FuncDef *p;
sl@0
   274
  for(p=pHash->a[h]; p; p=p->pHash){
sl@0
   275
    if( sqlite3StrNICmp(p->zName, zFunc, nFunc)==0 && p->zName[nFunc]==0 ){
sl@0
   276
      return p;
sl@0
   277
    }
sl@0
   278
  }
sl@0
   279
  return 0;
sl@0
   280
}
sl@0
   281
sl@0
   282
/*
sl@0
   283
** Insert a new FuncDef into a FuncDefHash hash table.
sl@0
   284
*/
sl@0
   285
void sqlite3FuncDefInsert(
sl@0
   286
  FuncDefHash *pHash,  /* The hash table into which to insert */
sl@0
   287
  FuncDef *pDef        /* The function definition to insert */
sl@0
   288
){
sl@0
   289
  FuncDef *pOther;
sl@0
   290
  int nName = strlen(pDef->zName);
sl@0
   291
  u8 c1 = (u8)pDef->zName[0];
sl@0
   292
  int h = (sqlite3UpperToLower[c1] + nName) % ArraySize(pHash->a);
sl@0
   293
  pOther = functionSearch(pHash, h, pDef->zName, nName);
sl@0
   294
  if( pOther ){
sl@0
   295
    pDef->pNext = pOther->pNext;
sl@0
   296
    pOther->pNext = pDef;
sl@0
   297
  }else{
sl@0
   298
    pDef->pNext = 0;
sl@0
   299
    pDef->pHash = pHash->a[h];
sl@0
   300
    pHash->a[h] = pDef;
sl@0
   301
  }
sl@0
   302
}
sl@0
   303
  
sl@0
   304
  
sl@0
   305
sl@0
   306
/*
sl@0
   307
** Locate a user function given a name, a number of arguments and a flag
sl@0
   308
** indicating whether the function prefers UTF-16 over UTF-8.  Return a
sl@0
   309
** pointer to the FuncDef structure that defines that function, or return
sl@0
   310
** NULL if the function does not exist.
sl@0
   311
**
sl@0
   312
** If the createFlag argument is true, then a new (blank) FuncDef
sl@0
   313
** structure is created and liked into the "db" structure if a
sl@0
   314
** no matching function previously existed.  When createFlag is true
sl@0
   315
** and the nArg parameter is -1, then only a function that accepts
sl@0
   316
** any number of arguments will be returned.
sl@0
   317
**
sl@0
   318
** If createFlag is false and nArg is -1, then the first valid
sl@0
   319
** function found is returned.  A function is valid if either xFunc
sl@0
   320
** or xStep is non-zero.
sl@0
   321
**
sl@0
   322
** If createFlag is false, then a function with the required name and
sl@0
   323
** number of arguments may be returned even if the eTextRep flag does not
sl@0
   324
** match that requested.
sl@0
   325
*/
sl@0
   326
FuncDef *sqlite3FindFunction(
sl@0
   327
  sqlite3 *db,       /* An open database */
sl@0
   328
  const char *zName, /* Name of the function.  Not null-terminated */
sl@0
   329
  int nName,         /* Number of characters in the name */
sl@0
   330
  int nArg,          /* Number of arguments.  -1 means any number */
sl@0
   331
  u8 enc,            /* Preferred text encoding */
sl@0
   332
  int createFlag     /* Create new entry if true and does not otherwise exist */
sl@0
   333
){
sl@0
   334
  FuncDef *p;         /* Iterator variable */
sl@0
   335
  FuncDef *pBest = 0; /* Best match found so far */
sl@0
   336
  int bestScore = 0;  /* Score of best match */
sl@0
   337
  int h;              /* Hash value */
sl@0
   338
sl@0
   339
sl@0
   340
  assert( enc==SQLITE_UTF8 || enc==SQLITE_UTF16LE || enc==SQLITE_UTF16BE );
sl@0
   341
  if( nArg<-1 ) nArg = -1;
sl@0
   342
  h = (sqlite3UpperToLower[(u8)zName[0]] + nName) % ArraySize(db->aFunc.a);
sl@0
   343
sl@0
   344
  /* First search for a match amongst the application-defined functions.
sl@0
   345
  */
sl@0
   346
  p = functionSearch(&db->aFunc, h, zName, nName);
sl@0
   347
  while( p ){
sl@0
   348
    int score = matchQuality(p, nArg, enc);
sl@0
   349
    if( score>bestScore ){
sl@0
   350
      pBest = p;
sl@0
   351
      bestScore = score;
sl@0
   352
    }
sl@0
   353
    p = p->pNext;
sl@0
   354
  }
sl@0
   355
sl@0
   356
  /* If no match is found, search the built-in functions.
sl@0
   357
  **
sl@0
   358
  ** Except, if createFlag is true, that means that we are trying to
sl@0
   359
  ** install a new function.  Whatever FuncDef structure is returned will
sl@0
   360
  ** have fields overwritten with new information appropriate for the
sl@0
   361
  ** new function.  But the FuncDefs for built-in functions are read-only.
sl@0
   362
  ** So we must not search for built-ins when creating a new function.
sl@0
   363
  */ 
sl@0
   364
  if( !createFlag && !pBest ){
sl@0
   365
    FuncDefHash *pHash = &GLOBAL(FuncDefHash, sqlite3GlobalFunctions);
sl@0
   366
    p = functionSearch(pHash, h, zName, nName);
sl@0
   367
    while( p ){
sl@0
   368
      int score = matchQuality(p, nArg, enc);
sl@0
   369
      if( score>bestScore ){
sl@0
   370
        pBest = p;
sl@0
   371
        bestScore = score;
sl@0
   372
      }
sl@0
   373
      p = p->pNext;
sl@0
   374
    }
sl@0
   375
  }
sl@0
   376
sl@0
   377
  /* If the createFlag parameter is true and the search did not reveal an
sl@0
   378
  ** exact match for the name, number of arguments and encoding, then add a
sl@0
   379
  ** new entry to the hash table and return it.
sl@0
   380
  */
sl@0
   381
  if( createFlag && (bestScore<6 || pBest->nArg!=nArg) && 
sl@0
   382
      (pBest = sqlite3DbMallocZero(db, sizeof(*pBest)+nName+1))!=0 ){
sl@0
   383
    pBest->zName = (char *)&pBest[1];
sl@0
   384
    pBest->nArg = nArg;
sl@0
   385
    pBest->iPrefEnc = enc;
sl@0
   386
    memcpy(pBest->zName, zName, nName);
sl@0
   387
    pBest->zName[nName] = 0;
sl@0
   388
    sqlite3FuncDefInsert(&db->aFunc, pBest);
sl@0
   389
  }
sl@0
   390
sl@0
   391
  if( pBest && (pBest->xStep || pBest->xFunc || createFlag) ){
sl@0
   392
    return pBest;
sl@0
   393
  }
sl@0
   394
  return 0;
sl@0
   395
}
sl@0
   396
sl@0
   397
/*
sl@0
   398
** Free all resources held by the schema structure. The void* argument points
sl@0
   399
** at a Schema struct. This function does not call sqlite3DbFree(db, ) on the 
sl@0
   400
** pointer itself, it just cleans up subsiduary resources (i.e. the contents
sl@0
   401
** of the schema hash tables).
sl@0
   402
**
sl@0
   403
** The Schema.cache_size variable is not cleared.
sl@0
   404
*/
sl@0
   405
void sqlite3SchemaFree(void *p){
sl@0
   406
  Hash temp1;
sl@0
   407
  Hash temp2;
sl@0
   408
  HashElem *pElem;
sl@0
   409
  Schema *pSchema = (Schema *)p;
sl@0
   410
sl@0
   411
  temp1 = pSchema->tblHash;
sl@0
   412
  temp2 = pSchema->trigHash;
sl@0
   413
  sqlite3HashInit(&pSchema->trigHash, 0);
sl@0
   414
  sqlite3HashClear(&pSchema->aFKey);
sl@0
   415
  sqlite3HashClear(&pSchema->idxHash);
sl@0
   416
  for(pElem=sqliteHashFirst(&temp2); pElem; pElem=sqliteHashNext(pElem)){
sl@0
   417
    sqlite3DeleteTrigger(0, (Trigger*)sqliteHashData(pElem));
sl@0
   418
  }
sl@0
   419
  sqlite3HashClear(&temp2);
sl@0
   420
  sqlite3HashInit(&pSchema->tblHash, 0);
sl@0
   421
  for(pElem=sqliteHashFirst(&temp1); pElem; pElem=sqliteHashNext(pElem)){
sl@0
   422
    Table *pTab = sqliteHashData(pElem);
sl@0
   423
    sqlite3DeleteTable(pTab);
sl@0
   424
  }
sl@0
   425
  sqlite3HashClear(&temp1);
sl@0
   426
  pSchema->pSeqTab = 0;
sl@0
   427
  pSchema->flags &= ~DB_SchemaLoaded;
sl@0
   428
}
sl@0
   429
sl@0
   430
/*
sl@0
   431
** Find and return the schema associated with a BTree.  Create
sl@0
   432
** a new one if necessary.
sl@0
   433
*/
sl@0
   434
Schema *sqlite3SchemaGet(sqlite3 *db, Btree *pBt){
sl@0
   435
  Schema * p;
sl@0
   436
  if( pBt ){
sl@0
   437
    p = (Schema *)sqlite3BtreeSchema(pBt, sizeof(Schema), sqlite3SchemaFree);
sl@0
   438
  }else{
sl@0
   439
    p = (Schema *)sqlite3MallocZero(sizeof(Schema));
sl@0
   440
  }
sl@0
   441
  if( !p ){
sl@0
   442
    db->mallocFailed = 1;
sl@0
   443
  }else if ( 0==p->file_format ){
sl@0
   444
    sqlite3HashInit(&p->tblHash, 0);
sl@0
   445
    sqlite3HashInit(&p->idxHash, 0);
sl@0
   446
    sqlite3HashInit(&p->trigHash, 0);
sl@0
   447
    sqlite3HashInit(&p->aFKey, 1);
sl@0
   448
    p->enc = SQLITE_UTF8;
sl@0
   449
  }
sl@0
   450
  return p;
sl@0
   451
}