os/persistentdata/persistentstorage/sql/SQLite/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.26 2008/07/28 19:34:53 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
/*
sl@0
   226
** Locate a user function given a name, a number of arguments and a flag
sl@0
   227
** indicating whether the function prefers UTF-16 over UTF-8.  Return a
sl@0
   228
** pointer to the FuncDef structure that defines that function, or return
sl@0
   229
** NULL if the function does not exist.
sl@0
   230
**
sl@0
   231
** If the createFlag argument is true, then a new (blank) FuncDef
sl@0
   232
** structure is created and liked into the "db" structure if a
sl@0
   233
** no matching function previously existed.  When createFlag is true
sl@0
   234
** and the nArg parameter is -1, then only a function that accepts
sl@0
   235
** any number of arguments will be returned.
sl@0
   236
**
sl@0
   237
** If createFlag is false and nArg is -1, then the first valid
sl@0
   238
** function found is returned.  A function is valid if either xFunc
sl@0
   239
** or xStep is non-zero.
sl@0
   240
**
sl@0
   241
** If createFlag is false, then a function with the required name and
sl@0
   242
** number of arguments may be returned even if the eTextRep flag does not
sl@0
   243
** match that requested.
sl@0
   244
*/
sl@0
   245
FuncDef *sqlite3FindFunction(
sl@0
   246
  sqlite3 *db,       /* An open database */
sl@0
   247
  const char *zName, /* Name of the function.  Not null-terminated */
sl@0
   248
  int nName,         /* Number of characters in the name */
sl@0
   249
  int nArg,          /* Number of arguments.  -1 means any number */
sl@0
   250
  u8 enc,            /* Preferred text encoding */
sl@0
   251
  int createFlag     /* Create new entry if true and does not otherwise exist */
sl@0
   252
){
sl@0
   253
  FuncDef *p;         /* Iterator variable */
sl@0
   254
  FuncDef *pFirst;    /* First function with this name */
sl@0
   255
  FuncDef *pBest = 0; /* Best match found so far */
sl@0
   256
  int bestmatch = 0;  
sl@0
   257
sl@0
   258
sl@0
   259
  assert( enc==SQLITE_UTF8 || enc==SQLITE_UTF16LE || enc==SQLITE_UTF16BE );
sl@0
   260
  if( nArg<-1 ) nArg = -1;
sl@0
   261
sl@0
   262
  pFirst = (FuncDef*)sqlite3HashFind(&db->aFunc, zName, nName);
sl@0
   263
  for(p=pFirst; p; p=p->pNext){
sl@0
   264
    /* During the search for the best function definition, bestmatch is set
sl@0
   265
    ** as follows to indicate the quality of the match with the definition
sl@0
   266
    ** pointed to by pBest:
sl@0
   267
    **
sl@0
   268
    ** 0: pBest is NULL. No match has been found.
sl@0
   269
    ** 1: A variable arguments function that prefers UTF-8 when a UTF-16
sl@0
   270
    **    encoding is requested, or vice versa.
sl@0
   271
    ** 2: A variable arguments function that uses UTF-16BE when UTF-16LE is
sl@0
   272
    **    requested, or vice versa.
sl@0
   273
    ** 3: A variable arguments function using the same text encoding.
sl@0
   274
    ** 4: A function with the exact number of arguments requested that
sl@0
   275
    **    prefers UTF-8 when a UTF-16 encoding is requested, or vice versa.
sl@0
   276
    ** 5: A function with the exact number of arguments requested that
sl@0
   277
    **    prefers UTF-16LE when UTF-16BE is requested, or vice versa.
sl@0
   278
    ** 6: An exact match.
sl@0
   279
    **
sl@0
   280
    ** A larger value of 'matchqual' indicates a more desirable match.
sl@0
   281
    */
sl@0
   282
    if( p->nArg==-1 || p->nArg==nArg || nArg==-1 ){
sl@0
   283
      int match = 1;          /* Quality of this match */
sl@0
   284
      if( p->nArg==nArg || nArg==-1 ){
sl@0
   285
        match = 4;
sl@0
   286
      }
sl@0
   287
      if( enc==p->iPrefEnc ){
sl@0
   288
        match += 2;
sl@0
   289
      }
sl@0
   290
      else if( (enc==SQLITE_UTF16LE && p->iPrefEnc==SQLITE_UTF16BE) ||
sl@0
   291
               (enc==SQLITE_UTF16BE && p->iPrefEnc==SQLITE_UTF16LE) ){
sl@0
   292
        match += 1;
sl@0
   293
      }
sl@0
   294
sl@0
   295
      if( match>bestmatch ){
sl@0
   296
        pBest = p;
sl@0
   297
        bestmatch = match;
sl@0
   298
      }
sl@0
   299
    }
sl@0
   300
  }
sl@0
   301
sl@0
   302
  /* If the createFlag parameter is true, and the seach did not reveal an
sl@0
   303
  ** exact match for the name, number of arguments and encoding, then add a
sl@0
   304
  ** new entry to the hash table and return it.
sl@0
   305
  */
sl@0
   306
  if( createFlag && bestmatch<6 && 
sl@0
   307
      (pBest = sqlite3DbMallocZero(db, sizeof(*pBest)+nName))!=0 ){
sl@0
   308
    pBest->nArg = nArg;
sl@0
   309
    pBest->pNext = pFirst;
sl@0
   310
    pBest->iPrefEnc = enc;
sl@0
   311
    memcpy(pBest->zName, zName, nName);
sl@0
   312
    pBest->zName[nName] = 0;
sl@0
   313
    if( pBest==sqlite3HashInsert(&db->aFunc,pBest->zName,nName,(void*)pBest) ){
sl@0
   314
      db->mallocFailed = 1;
sl@0
   315
      sqlite3DbFree(db, pBest);
sl@0
   316
      return 0;
sl@0
   317
    }
sl@0
   318
  }
sl@0
   319
sl@0
   320
  if( pBest && (pBest->xStep || pBest->xFunc || createFlag) ){
sl@0
   321
    return pBest;
sl@0
   322
  }
sl@0
   323
  return 0;
sl@0
   324
}
sl@0
   325
sl@0
   326
/*
sl@0
   327
** Free all resources held by the schema structure. The void* argument points
sl@0
   328
** at a Schema struct. This function does not call sqlite3DbFree(db, ) on the 
sl@0
   329
** pointer itself, it just cleans up subsiduary resources (i.e. the contents
sl@0
   330
** of the schema hash tables).
sl@0
   331
**
sl@0
   332
** The Schema.cache_size variable is not cleared.
sl@0
   333
*/
sl@0
   334
void sqlite3SchemaFree(void *p){
sl@0
   335
  Hash temp1;
sl@0
   336
  Hash temp2;
sl@0
   337
  HashElem *pElem;
sl@0
   338
  Schema *pSchema = (Schema *)p;
sl@0
   339
sl@0
   340
  temp1 = pSchema->tblHash;
sl@0
   341
  temp2 = pSchema->trigHash;
sl@0
   342
  sqlite3HashInit(&pSchema->trigHash, SQLITE_HASH_STRING, 0);
sl@0
   343
  sqlite3HashClear(&pSchema->aFKey);
sl@0
   344
  sqlite3HashClear(&pSchema->idxHash);
sl@0
   345
  for(pElem=sqliteHashFirst(&temp2); pElem; pElem=sqliteHashNext(pElem)){
sl@0
   346
    sqlite3DeleteTrigger(0, (Trigger*)sqliteHashData(pElem));
sl@0
   347
  }
sl@0
   348
  sqlite3HashClear(&temp2);
sl@0
   349
  sqlite3HashInit(&pSchema->tblHash, SQLITE_HASH_STRING, 0);
sl@0
   350
  for(pElem=sqliteHashFirst(&temp1); pElem; pElem=sqliteHashNext(pElem)){
sl@0
   351
    Table *pTab = sqliteHashData(pElem);
sl@0
   352
    sqlite3DeleteTable(pTab);
sl@0
   353
  }
sl@0
   354
  sqlite3HashClear(&temp1);
sl@0
   355
  pSchema->pSeqTab = 0;
sl@0
   356
  pSchema->flags &= ~DB_SchemaLoaded;
sl@0
   357
}
sl@0
   358
sl@0
   359
/*
sl@0
   360
** Find and return the schema associated with a BTree.  Create
sl@0
   361
** a new one if necessary.
sl@0
   362
*/
sl@0
   363
Schema *sqlite3SchemaGet(sqlite3 *db, Btree *pBt){
sl@0
   364
  Schema * p;
sl@0
   365
  if( pBt ){
sl@0
   366
    p = (Schema *)sqlite3BtreeSchema(pBt, sizeof(Schema), sqlite3SchemaFree);
sl@0
   367
  }else{
sl@0
   368
    p = (Schema *)sqlite3MallocZero(sizeof(Schema));
sl@0
   369
  }
sl@0
   370
  if( !p ){
sl@0
   371
    db->mallocFailed = 1;
sl@0
   372
  }else if ( 0==p->file_format ){
sl@0
   373
    sqlite3HashInit(&p->tblHash, SQLITE_HASH_STRING, 0);
sl@0
   374
    sqlite3HashInit(&p->idxHash, SQLITE_HASH_STRING, 0);
sl@0
   375
    sqlite3HashInit(&p->trigHash, SQLITE_HASH_STRING, 0);
sl@0
   376
    sqlite3HashInit(&p->aFKey, SQLITE_HASH_STRING, 1);
sl@0
   377
    p->enc = SQLITE_UTF8;
sl@0
   378
  }
sl@0
   379
  return p;
sl@0
   380
}