os/persistentdata/persistentstorage/sql/SQLite/complete.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
** 2001 September 15
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
** An tokenizer for SQL
sl@0
    13
**
sl@0
    14
** This file contains C code that implements the sqlite3_complete() API.
sl@0
    15
** This code used to be part of the tokenizer.c source file.  But by
sl@0
    16
** separating it out, the code will be automatically omitted from
sl@0
    17
** static links that do not use it.
sl@0
    18
**
sl@0
    19
** $Id: complete.c,v 1.7 2008/06/13 18:24:27 drh Exp $
sl@0
    20
*/
sl@0
    21
#include "sqliteInt.h"
sl@0
    22
#ifndef SQLITE_OMIT_COMPLETE
sl@0
    23
sl@0
    24
/*
sl@0
    25
** This is defined in tokenize.c.  We just have to import the definition.
sl@0
    26
*/
sl@0
    27
#ifndef SQLITE_AMALGAMATION
sl@0
    28
#ifdef SQLITE_ASCII
sl@0
    29
extern const char sqlite3IsAsciiIdChar[];
sl@0
    30
#define IdChar(C)  (((c=C)&0x80)!=0 || (c>0x1f && sqlite3IsAsciiIdChar[c-0x20]))
sl@0
    31
#endif
sl@0
    32
#ifdef SQLITE_EBCDIC
sl@0
    33
extern const char sqlite3IsEbcdicIdChar[];
sl@0
    34
#define IdChar(C)  (((c=C)>=0x42 && sqlite3IsEbcdicIdChar[c-0x40]))
sl@0
    35
#endif
sl@0
    36
#endif /* SQLITE_AMALGAMATION */
sl@0
    37
sl@0
    38
sl@0
    39
/*
sl@0
    40
** Token types used by the sqlite3_complete() routine.  See the header
sl@0
    41
** comments on that procedure for additional information.
sl@0
    42
*/
sl@0
    43
#define tkSEMI    0
sl@0
    44
#define tkWS      1
sl@0
    45
#define tkOTHER   2
sl@0
    46
#define tkEXPLAIN 3
sl@0
    47
#define tkCREATE  4
sl@0
    48
#define tkTEMP    5
sl@0
    49
#define tkTRIGGER 6
sl@0
    50
#define tkEND     7
sl@0
    51
sl@0
    52
/*
sl@0
    53
** Return TRUE if the given SQL string ends in a semicolon.
sl@0
    54
**
sl@0
    55
** Special handling is require for CREATE TRIGGER statements.
sl@0
    56
** Whenever the CREATE TRIGGER keywords are seen, the statement
sl@0
    57
** must end with ";END;".
sl@0
    58
**
sl@0
    59
** This implementation uses a state machine with 7 states:
sl@0
    60
**
sl@0
    61
**   (0) START     At the beginning or end of an SQL statement.  This routine
sl@0
    62
**                 returns 1 if it ends in the START state and 0 if it ends
sl@0
    63
**                 in any other state.
sl@0
    64
**
sl@0
    65
**   (1) NORMAL    We are in the middle of statement which ends with a single
sl@0
    66
**                 semicolon.
sl@0
    67
**
sl@0
    68
**   (2) EXPLAIN   The keyword EXPLAIN has been seen at the beginning of 
sl@0
    69
**                 a statement.
sl@0
    70
**
sl@0
    71
**   (3) CREATE    The keyword CREATE has been seen at the beginning of a
sl@0
    72
**                 statement, possibly preceeded by EXPLAIN and/or followed by
sl@0
    73
**                 TEMP or TEMPORARY
sl@0
    74
**
sl@0
    75
**   (4) TRIGGER   We are in the middle of a trigger definition that must be
sl@0
    76
**                 ended by a semicolon, the keyword END, and another semicolon.
sl@0
    77
**
sl@0
    78
**   (5) SEMI      We've seen the first semicolon in the ";END;" that occurs at
sl@0
    79
**                 the end of a trigger definition.
sl@0
    80
**
sl@0
    81
**   (6) END       We've seen the ";END" of the ";END;" that occurs at the end
sl@0
    82
**                 of a trigger difinition.
sl@0
    83
**
sl@0
    84
** Transitions between states above are determined by tokens extracted
sl@0
    85
** from the input.  The following tokens are significant:
sl@0
    86
**
sl@0
    87
**   (0) tkSEMI      A semicolon.
sl@0
    88
**   (1) tkWS        Whitespace
sl@0
    89
**   (2) tkOTHER     Any other SQL token.
sl@0
    90
**   (3) tkEXPLAIN   The "explain" keyword.
sl@0
    91
**   (4) tkCREATE    The "create" keyword.
sl@0
    92
**   (5) tkTEMP      The "temp" or "temporary" keyword.
sl@0
    93
**   (6) tkTRIGGER   The "trigger" keyword.
sl@0
    94
**   (7) tkEND       The "end" keyword.
sl@0
    95
**
sl@0
    96
** Whitespace never causes a state transition and is always ignored.
sl@0
    97
**
sl@0
    98
** If we compile with SQLITE_OMIT_TRIGGER, all of the computation needed
sl@0
    99
** to recognize the end of a trigger can be omitted.  All we have to do
sl@0
   100
** is look for a semicolon that is not part of an string or comment.
sl@0
   101
*/
sl@0
   102
int sqlite3_complete(const char *zSql){
sl@0
   103
  u8 state = 0;   /* Current state, using numbers defined in header comment */
sl@0
   104
  u8 token;       /* Value of the next token */
sl@0
   105
sl@0
   106
#ifndef SQLITE_OMIT_TRIGGER
sl@0
   107
  /* A complex statement machine used to detect the end of a CREATE TRIGGER
sl@0
   108
  ** statement.  This is the normal case.
sl@0
   109
  */
sl@0
   110
  static const u8 trans[7][8] = {
sl@0
   111
                     /* Token:                                                */
sl@0
   112
     /* State:       **  SEMI  WS  OTHER EXPLAIN  CREATE  TEMP  TRIGGER  END  */
sl@0
   113
     /* 0   START: */ {    0,  0,     1,      2,      3,    1,       1,   1,  },
sl@0
   114
     /* 1  NORMAL: */ {    0,  1,     1,      1,      1,    1,       1,   1,  },
sl@0
   115
     /* 2 EXPLAIN: */ {    0,  2,     1,      1,      3,    1,       1,   1,  },
sl@0
   116
     /* 3  CREATE: */ {    0,  3,     1,      1,      1,    3,       4,   1,  },
sl@0
   117
     /* 4 TRIGGER: */ {    5,  4,     4,      4,      4,    4,       4,   4,  },
sl@0
   118
     /* 5    SEMI: */ {    5,  5,     4,      4,      4,    4,       4,   6,  },
sl@0
   119
     /* 6     END: */ {    0,  6,     4,      4,      4,    4,       4,   4,  },
sl@0
   120
  };
sl@0
   121
#else
sl@0
   122
  /* If triggers are not suppored by this compile then the statement machine
sl@0
   123
  ** used to detect the end of a statement is much simplier
sl@0
   124
  */
sl@0
   125
  static const u8 trans[2][3] = {
sl@0
   126
                     /* Token:           */
sl@0
   127
     /* State:       **  SEMI  WS  OTHER */
sl@0
   128
     /* 0   START: */ {    0,  0,     1, },
sl@0
   129
     /* 1  NORMAL: */ {    0,  1,     1, },
sl@0
   130
  };
sl@0
   131
#endif /* SQLITE_OMIT_TRIGGER */
sl@0
   132
sl@0
   133
  while( *zSql ){
sl@0
   134
    switch( *zSql ){
sl@0
   135
      case ';': {  /* A semicolon */
sl@0
   136
        token = tkSEMI;
sl@0
   137
        break;
sl@0
   138
      }
sl@0
   139
      case ' ':
sl@0
   140
      case '\r':
sl@0
   141
      case '\t':
sl@0
   142
      case '\n':
sl@0
   143
      case '\f': {  /* White space is ignored */
sl@0
   144
        token = tkWS;
sl@0
   145
        break;
sl@0
   146
      }
sl@0
   147
      case '/': {   /* C-style comments */
sl@0
   148
        if( zSql[1]!='*' ){
sl@0
   149
          token = tkOTHER;
sl@0
   150
          break;
sl@0
   151
        }
sl@0
   152
        zSql += 2;
sl@0
   153
        while( zSql[0] && (zSql[0]!='*' || zSql[1]!='/') ){ zSql++; }
sl@0
   154
        if( zSql[0]==0 ) return 0;
sl@0
   155
        zSql++;
sl@0
   156
        token = tkWS;
sl@0
   157
        break;
sl@0
   158
      }
sl@0
   159
      case '-': {   /* SQL-style comments from "--" to end of line */
sl@0
   160
        if( zSql[1]!='-' ){
sl@0
   161
          token = tkOTHER;
sl@0
   162
          break;
sl@0
   163
        }
sl@0
   164
        while( *zSql && *zSql!='\n' ){ zSql++; }
sl@0
   165
        if( *zSql==0 ) return state==0;
sl@0
   166
        token = tkWS;
sl@0
   167
        break;
sl@0
   168
      }
sl@0
   169
      case '[': {   /* Microsoft-style identifiers in [...] */
sl@0
   170
        zSql++;
sl@0
   171
        while( *zSql && *zSql!=']' ){ zSql++; }
sl@0
   172
        if( *zSql==0 ) return 0;
sl@0
   173
        token = tkOTHER;
sl@0
   174
        break;
sl@0
   175
      }
sl@0
   176
      case '`':     /* Grave-accent quoted symbols used by MySQL */
sl@0
   177
      case '"':     /* single- and double-quoted strings */
sl@0
   178
      case '\'': {
sl@0
   179
        int c = *zSql;
sl@0
   180
        zSql++;
sl@0
   181
        while( *zSql && *zSql!=c ){ zSql++; }
sl@0
   182
        if( *zSql==0 ) return 0;
sl@0
   183
        token = tkOTHER;
sl@0
   184
        break;
sl@0
   185
      }
sl@0
   186
      default: {
sl@0
   187
        int c;
sl@0
   188
        if( IdChar((u8)*zSql) ){
sl@0
   189
          /* Keywords and unquoted identifiers */
sl@0
   190
          int nId;
sl@0
   191
          for(nId=1; IdChar(zSql[nId]); nId++){}
sl@0
   192
#ifdef SQLITE_OMIT_TRIGGER
sl@0
   193
          token = tkOTHER;
sl@0
   194
#else
sl@0
   195
          switch( *zSql ){
sl@0
   196
            case 'c': case 'C': {
sl@0
   197
              if( nId==6 && sqlite3StrNICmp(zSql, "create", 6)==0 ){
sl@0
   198
                token = tkCREATE;
sl@0
   199
              }else{
sl@0
   200
                token = tkOTHER;
sl@0
   201
              }
sl@0
   202
              break;
sl@0
   203
            }
sl@0
   204
            case 't': case 'T': {
sl@0
   205
              if( nId==7 && sqlite3StrNICmp(zSql, "trigger", 7)==0 ){
sl@0
   206
                token = tkTRIGGER;
sl@0
   207
              }else if( nId==4 && sqlite3StrNICmp(zSql, "temp", 4)==0 ){
sl@0
   208
                token = tkTEMP;
sl@0
   209
              }else if( nId==9 && sqlite3StrNICmp(zSql, "temporary", 9)==0 ){
sl@0
   210
                token = tkTEMP;
sl@0
   211
              }else{
sl@0
   212
                token = tkOTHER;
sl@0
   213
              }
sl@0
   214
              break;
sl@0
   215
            }
sl@0
   216
            case 'e':  case 'E': {
sl@0
   217
              if( nId==3 && sqlite3StrNICmp(zSql, "end", 3)==0 ){
sl@0
   218
                token = tkEND;
sl@0
   219
              }else
sl@0
   220
#ifndef SQLITE_OMIT_EXPLAIN
sl@0
   221
              if( nId==7 && sqlite3StrNICmp(zSql, "explain", 7)==0 ){
sl@0
   222
                token = tkEXPLAIN;
sl@0
   223
              }else
sl@0
   224
#endif
sl@0
   225
              {
sl@0
   226
                token = tkOTHER;
sl@0
   227
              }
sl@0
   228
              break;
sl@0
   229
            }
sl@0
   230
            default: {
sl@0
   231
              token = tkOTHER;
sl@0
   232
              break;
sl@0
   233
            }
sl@0
   234
          }
sl@0
   235
#endif /* SQLITE_OMIT_TRIGGER */
sl@0
   236
          zSql += nId-1;
sl@0
   237
        }else{
sl@0
   238
          /* Operators and special symbols */
sl@0
   239
          token = tkOTHER;
sl@0
   240
        }
sl@0
   241
        break;
sl@0
   242
      }
sl@0
   243
    }
sl@0
   244
    state = trans[state][token];
sl@0
   245
    zSql++;
sl@0
   246
  }
sl@0
   247
  return state==0;
sl@0
   248
}
sl@0
   249
sl@0
   250
#ifndef SQLITE_OMIT_UTF16
sl@0
   251
/*
sl@0
   252
** This routine is the same as the sqlite3_complete() routine described
sl@0
   253
** above, except that the parameter is required to be UTF-16 encoded, not
sl@0
   254
** UTF-8.
sl@0
   255
*/
sl@0
   256
int sqlite3_complete16(const void *zSql){
sl@0
   257
  sqlite3_value *pVal;
sl@0
   258
  char const *zSql8;
sl@0
   259
  int rc = SQLITE_NOMEM;
sl@0
   260
sl@0
   261
#ifndef SQLITE_OMIT_AUTOINIT
sl@0
   262
  rc = sqlite3_initialize();
sl@0
   263
  if( rc ) return rc;
sl@0
   264
#endif
sl@0
   265
  pVal = sqlite3ValueNew(0);
sl@0
   266
  sqlite3ValueSetStr(pVal, -1, zSql, SQLITE_UTF16NATIVE, SQLITE_STATIC);
sl@0
   267
  zSql8 = sqlite3ValueText(pVal, SQLITE_UTF8);
sl@0
   268
  if( zSql8 ){
sl@0
   269
    rc = sqlite3_complete(zSql8);
sl@0
   270
  }else{
sl@0
   271
    rc = SQLITE_NOMEM;
sl@0
   272
  }
sl@0
   273
  sqlite3ValueFree(pVal);
sl@0
   274
  return sqlite3ApiExit(0, rc);
sl@0
   275
}
sl@0
   276
#endif /* SQLITE_OMIT_UTF16 */
sl@0
   277
#endif /* SQLITE_OMIT_COMPLETE */