os/persistentdata/persistentstorage/sql/SQLite364/delete.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
** This file contains C code routines that are called by the parser
sl@0
    13
** in order to generate code for DELETE FROM statements.
sl@0
    14
**
sl@0
    15
** $Id: delete.c,v 1.182 2008/10/10 23:48:26 drh Exp $
sl@0
    16
*/
sl@0
    17
#include "sqliteInt.h"
sl@0
    18
sl@0
    19
/*
sl@0
    20
** Look up every table that is named in pSrc.  If any table is not found,
sl@0
    21
** add an error message to pParse->zErrMsg and return NULL.  If all tables
sl@0
    22
** are found, return a pointer to the last table.
sl@0
    23
*/
sl@0
    24
Table *sqlite3SrcListLookup(Parse *pParse, SrcList *pSrc){
sl@0
    25
  struct SrcList_item *pItem = pSrc->a;
sl@0
    26
  Table *pTab;
sl@0
    27
  assert( pItem && pSrc->nSrc==1 );
sl@0
    28
  pTab = sqlite3LocateTable(pParse, 0, pItem->zName, pItem->zDatabase);
sl@0
    29
  sqlite3DeleteTable(pItem->pTab);
sl@0
    30
  pItem->pTab = pTab;
sl@0
    31
  if( pTab ){
sl@0
    32
    pTab->nRef++;
sl@0
    33
  }
sl@0
    34
  if( sqlite3IndexedByLookup(pParse, pItem) ){
sl@0
    35
    pTab = 0;
sl@0
    36
  }
sl@0
    37
  return pTab;
sl@0
    38
}
sl@0
    39
sl@0
    40
/*
sl@0
    41
** Check to make sure the given table is writable.  If it is not
sl@0
    42
** writable, generate an error message and return 1.  If it is
sl@0
    43
** writable return 0;
sl@0
    44
*/
sl@0
    45
int sqlite3IsReadOnly(Parse *pParse, Table *pTab, int viewOk){
sl@0
    46
  if( ((pTab->tabFlags & TF_Readonly)!=0
sl@0
    47
        && (pParse->db->flags & SQLITE_WriteSchema)==0
sl@0
    48
        && pParse->nested==0) 
sl@0
    49
#ifndef SQLITE_OMIT_VIRTUALTABLE
sl@0
    50
      || (pTab->pMod && pTab->pMod->pModule->xUpdate==0)
sl@0
    51
#endif
sl@0
    52
  ){
sl@0
    53
    sqlite3ErrorMsg(pParse, "table %s may not be modified", pTab->zName);
sl@0
    54
    return 1;
sl@0
    55
  }
sl@0
    56
#ifndef SQLITE_OMIT_VIEW
sl@0
    57
  if( !viewOk && pTab->pSelect ){
sl@0
    58
    sqlite3ErrorMsg(pParse,"cannot modify %s because it is a view",pTab->zName);
sl@0
    59
    return 1;
sl@0
    60
  }
sl@0
    61
#endif
sl@0
    62
  return 0;
sl@0
    63
}
sl@0
    64
sl@0
    65
/*
sl@0
    66
** Generate code that will open a table for reading.
sl@0
    67
*/
sl@0
    68
void sqlite3OpenTable(
sl@0
    69
  Parse *p,       /* Generate code into this VDBE */
sl@0
    70
  int iCur,       /* The cursor number of the table */
sl@0
    71
  int iDb,        /* The database index in sqlite3.aDb[] */
sl@0
    72
  Table *pTab,    /* The table to be opened */
sl@0
    73
  int opcode      /* OP_OpenRead or OP_OpenWrite */
sl@0
    74
){
sl@0
    75
  Vdbe *v;
sl@0
    76
  if( IsVirtual(pTab) ) return;
sl@0
    77
  v = sqlite3GetVdbe(p);
sl@0
    78
  assert( opcode==OP_OpenWrite || opcode==OP_OpenRead );
sl@0
    79
  sqlite3TableLock(p, iDb, pTab->tnum, (opcode==OP_OpenWrite), pTab->zName);
sl@0
    80
  sqlite3VdbeAddOp2(v, OP_SetNumColumns, 0, pTab->nCol);
sl@0
    81
  sqlite3VdbeAddOp3(v, opcode, iCur, pTab->tnum, iDb);
sl@0
    82
  VdbeComment((v, "%s", pTab->zName));
sl@0
    83
}
sl@0
    84
sl@0
    85
sl@0
    86
#if !defined(SQLITE_OMIT_VIEW) && !defined(SQLITE_OMIT_TRIGGER)
sl@0
    87
/*
sl@0
    88
** Evaluate a view and store its result in an ephemeral table.  The
sl@0
    89
** pWhere argument is an optional WHERE clause that restricts the
sl@0
    90
** set of rows in the view that are to be added to the ephemeral table.
sl@0
    91
*/
sl@0
    92
void sqlite3MaterializeView(
sl@0
    93
  Parse *pParse,       /* Parsing context */
sl@0
    94
  Table *pView,        /* View definition */
sl@0
    95
  Expr *pWhere,        /* Optional WHERE clause to be added */
sl@0
    96
  int iCur             /* Cursor number for ephemerial table */
sl@0
    97
){
sl@0
    98
  SelectDest dest;
sl@0
    99
  Select *pDup;
sl@0
   100
  sqlite3 *db = pParse->db;
sl@0
   101
sl@0
   102
  pDup = sqlite3SelectDup(db, pView->pSelect);
sl@0
   103
  if( pWhere ){
sl@0
   104
    SrcList *pFrom;
sl@0
   105
    Token viewName;
sl@0
   106
    
sl@0
   107
    pWhere = sqlite3ExprDup(db, pWhere);
sl@0
   108
    viewName.z = (u8*)pView->zName;
sl@0
   109
    viewName.n = (unsigned int)strlen((const char*)viewName.z);
sl@0
   110
    pFrom = sqlite3SrcListAppendFromTerm(pParse, 0, 0, 0, &viewName, pDup, 0,0);
sl@0
   111
    pDup = sqlite3SelectNew(pParse, 0, pFrom, pWhere, 0, 0, 0, 0, 0, 0);
sl@0
   112
  }
sl@0
   113
  sqlite3SelectDestInit(&dest, SRT_EphemTab, iCur);
sl@0
   114
  sqlite3Select(pParse, pDup, &dest);
sl@0
   115
  sqlite3SelectDelete(db, pDup);
sl@0
   116
}
sl@0
   117
#endif /* !defined(SQLITE_OMIT_VIEW) && !defined(SQLITE_OMIT_TRIGGER) */
sl@0
   118
sl@0
   119
#if defined(SQLITE_ENABLE_UPDATE_DELETE_LIMIT) && !defined(SQLITE_OMIT_SUBQUERY)
sl@0
   120
/*
sl@0
   121
** Generate an expression tree to implement the WHERE, ORDER BY,
sl@0
   122
** and LIMIT/OFFSET portion of DELETE and UPDATE statements.
sl@0
   123
**
sl@0
   124
**     DELETE FROM table_wxyz WHERE a<5 ORDER BY a LIMIT 1;
sl@0
   125
**                            \__________________________/
sl@0
   126
**                               pLimitWhere (pInClause)
sl@0
   127
*/
sl@0
   128
Expr *sqlite3LimitWhere(
sl@0
   129
  Parse *pParse,               /* The parser context */
sl@0
   130
  SrcList *pSrc,               /* the FROM clause -- which tables to scan */
sl@0
   131
  Expr *pWhere,                /* The WHERE clause.  May be null */
sl@0
   132
  ExprList *pOrderBy,          /* The ORDER BY clause.  May be null */
sl@0
   133
  Expr *pLimit,                /* The LIMIT clause.  May be null */
sl@0
   134
  Expr *pOffset,               /* The OFFSET clause.  May be null */
sl@0
   135
  char *zStmtType              /* Either DELETE or UPDATE.  For error messages. */
sl@0
   136
){
sl@0
   137
  Expr *pWhereRowid = NULL;    /* WHERE rowid .. */
sl@0
   138
  Expr *pInClause = NULL;      /* WHERE rowid IN ( select ) */
sl@0
   139
  Expr *pSelectRowid = NULL;   /* SELECT rowid ... */
sl@0
   140
  ExprList *pEList = NULL;     /* Expression list contaning only pSelectRowid */
sl@0
   141
  SrcList *pSelectSrc = NULL;  /* SELECT rowid FROM x ... (dup of pSrc) */
sl@0
   142
  Select *pSelect = NULL;      /* Complete SELECT tree */
sl@0
   143
sl@0
   144
  /* Check that there isn't an ORDER BY without a LIMIT clause.
sl@0
   145
  */
sl@0
   146
  if( pOrderBy && (pLimit == 0) ) {
sl@0
   147
    sqlite3ErrorMsg(pParse, "ORDER BY without LIMIT on %s", zStmtType);
sl@0
   148
    pParse->parseError = 1;
sl@0
   149
    goto limit_where_cleanup_2;
sl@0
   150
  }
sl@0
   151
sl@0
   152
  /* We only need to generate a select expression if there
sl@0
   153
  ** is a limit/offset term to enforce.
sl@0
   154
  */
sl@0
   155
  if( pLimit == 0 ) {
sl@0
   156
    /* if pLimit is null, pOffset will always be null as well. */
sl@0
   157
    assert( pOffset == 0 );
sl@0
   158
    return pWhere;
sl@0
   159
  }
sl@0
   160
sl@0
   161
  /* Generate a select expression tree to enforce the limit/offset 
sl@0
   162
  ** term for the DELETE or UPDATE statement.  For example:
sl@0
   163
  **   DELETE FROM table_a WHERE col1=1 ORDER BY col2 LIMIT 1 OFFSET 1
sl@0
   164
  ** becomes:
sl@0
   165
  **   DELETE FROM table_a WHERE rowid IN ( 
sl@0
   166
  **     SELECT rowid FROM table_a WHERE col1=1 ORDER BY col2 LIMIT 1 OFFSET 1
sl@0
   167
  **   );
sl@0
   168
  */
sl@0
   169
sl@0
   170
  pSelectRowid = sqlite3Expr(pParse->db, TK_ROW, 0, 0, 0);
sl@0
   171
  if( pSelectRowid == 0 ) goto limit_where_cleanup_2;
sl@0
   172
  pEList = sqlite3ExprListAppend(pParse, 0, pSelectRowid, 0);
sl@0
   173
  if( pEList == 0 ) goto limit_where_cleanup_2;
sl@0
   174
sl@0
   175
  /* duplicate the FROM clause as it is needed by both the DELETE/UPDATE tree
sl@0
   176
  ** and the SELECT subtree. */
sl@0
   177
  pSelectSrc = sqlite3SrcListDup(pParse->db, pSrc);
sl@0
   178
  if( pSelectSrc == 0 ) {
sl@0
   179
    sqlite3ExprListDelete(pParse->db, pEList);
sl@0
   180
    goto limit_where_cleanup_2;
sl@0
   181
  }
sl@0
   182
sl@0
   183
  /* generate the SELECT expression tree. */
sl@0
   184
  pSelect = sqlite3SelectNew(pParse,pEList,pSelectSrc,pWhere,0,0,pOrderBy,0,pLimit,pOffset);
sl@0
   185
  if( pSelect == 0 ) return 0;
sl@0
   186
sl@0
   187
  /* now generate the new WHERE rowid IN clause for the DELETE/UDPATE */
sl@0
   188
  pWhereRowid = sqlite3Expr(pParse->db, TK_ROW, 0, 0, 0);
sl@0
   189
  if( pWhereRowid == 0 ) goto limit_where_cleanup_1;
sl@0
   190
  pInClause = sqlite3PExpr(pParse, TK_IN, pWhereRowid, 0, 0);
sl@0
   191
  if( pInClause == 0 ) goto limit_where_cleanup_1;
sl@0
   192
sl@0
   193
  pInClause->pSelect = pSelect;
sl@0
   194
  sqlite3ExprSetHeight(pParse, pInClause);
sl@0
   195
  return pInClause;
sl@0
   196
sl@0
   197
  /* something went wrong. clean up anything allocated. */
sl@0
   198
limit_where_cleanup_1:
sl@0
   199
  sqlite3SelectDelete(pParse->db, pSelect);
sl@0
   200
  return 0;
sl@0
   201
sl@0
   202
limit_where_cleanup_2:
sl@0
   203
  sqlite3ExprDelete(pParse->db, pWhere);
sl@0
   204
  sqlite3ExprListDelete(pParse->db, pOrderBy);
sl@0
   205
  sqlite3ExprDelete(pParse->db, pLimit);
sl@0
   206
  sqlite3ExprDelete(pParse->db, pOffset);
sl@0
   207
  return 0;
sl@0
   208
}
sl@0
   209
#endif /* defined(SQLITE_ENABLE_UPDATE_DELETE_LIMIT) && !defined(SQLITE_OMIT_SUBQUERY) */
sl@0
   210
sl@0
   211
/*
sl@0
   212
** Generate code for a DELETE FROM statement.
sl@0
   213
**
sl@0
   214
**     DELETE FROM table_wxyz WHERE a<5 AND b NOT NULL;
sl@0
   215
**                 \________/       \________________/
sl@0
   216
**                  pTabList              pWhere
sl@0
   217
*/
sl@0
   218
void sqlite3DeleteFrom(
sl@0
   219
  Parse *pParse,         /* The parser context */
sl@0
   220
  SrcList *pTabList,     /* The table from which we should delete things */
sl@0
   221
  Expr *pWhere           /* The WHERE clause.  May be null */
sl@0
   222
){
sl@0
   223
  Vdbe *v;               /* The virtual database engine */
sl@0
   224
  Table *pTab;           /* The table from which records will be deleted */
sl@0
   225
  const char *zDb;       /* Name of database holding pTab */
sl@0
   226
  int end, addr = 0;     /* A couple addresses of generated code */
sl@0
   227
  int i;                 /* Loop counter */
sl@0
   228
  WhereInfo *pWInfo;     /* Information about the WHERE clause */
sl@0
   229
  Index *pIdx;           /* For looping over indices of the table */
sl@0
   230
  int iCur;              /* VDBE Cursor number for pTab */
sl@0
   231
  sqlite3 *db;           /* Main database structure */
sl@0
   232
  AuthContext sContext;  /* Authorization context */
sl@0
   233
  int oldIdx = -1;       /* Cursor for the OLD table of AFTER triggers */
sl@0
   234
  NameContext sNC;       /* Name context to resolve expressions in */
sl@0
   235
  int iDb;               /* Database number */
sl@0
   236
  int memCnt = 0;        /* Memory cell used for change counting */
sl@0
   237
sl@0
   238
#ifndef SQLITE_OMIT_TRIGGER
sl@0
   239
  int isView;                  /* True if attempting to delete from a view */
sl@0
   240
  int triggers_exist = 0;      /* True if any triggers exist */
sl@0
   241
#endif
sl@0
   242
  int iBeginAfterTrigger;      /* Address of after trigger program */
sl@0
   243
  int iEndAfterTrigger;        /* Exit of after trigger program */
sl@0
   244
  int iBeginBeforeTrigger;     /* Address of before trigger program */
sl@0
   245
  int iEndBeforeTrigger;       /* Exit of before trigger program */
sl@0
   246
  u32 old_col_mask = 0;        /* Mask of OLD.* columns in use */
sl@0
   247
sl@0
   248
  sContext.pParse = 0;
sl@0
   249
  db = pParse->db;
sl@0
   250
  if( pParse->nErr || db->mallocFailed ){
sl@0
   251
    goto delete_from_cleanup;
sl@0
   252
  }
sl@0
   253
  assert( pTabList->nSrc==1 );
sl@0
   254
sl@0
   255
  /* Locate the table which we want to delete.  This table has to be
sl@0
   256
  ** put in an SrcList structure because some of the subroutines we
sl@0
   257
  ** will be calling are designed to work with multiple tables and expect
sl@0
   258
  ** an SrcList* parameter instead of just a Table* parameter.
sl@0
   259
  */
sl@0
   260
  pTab = sqlite3SrcListLookup(pParse, pTabList);
sl@0
   261
  if( pTab==0 )  goto delete_from_cleanup;
sl@0
   262
sl@0
   263
  /* Figure out if we have any triggers and if the table being
sl@0
   264
  ** deleted from is a view
sl@0
   265
  */
sl@0
   266
#ifndef SQLITE_OMIT_TRIGGER
sl@0
   267
  triggers_exist = sqlite3TriggersExist(pParse, pTab, TK_DELETE, 0);
sl@0
   268
  isView = pTab->pSelect!=0;
sl@0
   269
#else
sl@0
   270
# define triggers_exist 0
sl@0
   271
# define isView 0
sl@0
   272
#endif
sl@0
   273
#ifdef SQLITE_OMIT_VIEW
sl@0
   274
# undef isView
sl@0
   275
# define isView 0
sl@0
   276
#endif
sl@0
   277
sl@0
   278
  if( sqlite3IsReadOnly(pParse, pTab, triggers_exist) ){
sl@0
   279
    goto delete_from_cleanup;
sl@0
   280
  }
sl@0
   281
  iDb = sqlite3SchemaToIndex(db, pTab->pSchema);
sl@0
   282
  assert( iDb<db->nDb );
sl@0
   283
  zDb = db->aDb[iDb].zName;
sl@0
   284
  if( sqlite3AuthCheck(pParse, SQLITE_DELETE, pTab->zName, 0, zDb) ){
sl@0
   285
    goto delete_from_cleanup;
sl@0
   286
  }
sl@0
   287
sl@0
   288
  /* If pTab is really a view, make sure it has been initialized.
sl@0
   289
  */
sl@0
   290
  if( sqlite3ViewGetColumnNames(pParse, pTab) ){
sl@0
   291
    goto delete_from_cleanup;
sl@0
   292
  }
sl@0
   293
sl@0
   294
  /* Allocate a cursor used to store the old.* data for a trigger.
sl@0
   295
  */
sl@0
   296
  if( triggers_exist ){ 
sl@0
   297
    oldIdx = pParse->nTab++;
sl@0
   298
  }
sl@0
   299
sl@0
   300
  /* Assign  cursor number to the table and all its indices.
sl@0
   301
  */
sl@0
   302
  assert( pTabList->nSrc==1 );
sl@0
   303
  iCur = pTabList->a[0].iCursor = pParse->nTab++;
sl@0
   304
  for(pIdx=pTab->pIndex; pIdx; pIdx=pIdx->pNext){
sl@0
   305
    pParse->nTab++;
sl@0
   306
  }
sl@0
   307
sl@0
   308
  /* Start the view context
sl@0
   309
  */
sl@0
   310
  if( isView ){
sl@0
   311
    sqlite3AuthContextPush(pParse, &sContext, pTab->zName);
sl@0
   312
  }
sl@0
   313
sl@0
   314
  /* Begin generating code.
sl@0
   315
  */
sl@0
   316
  v = sqlite3GetVdbe(pParse);
sl@0
   317
  if( v==0 ){
sl@0
   318
    goto delete_from_cleanup;
sl@0
   319
  }
sl@0
   320
  if( pParse->nested==0 ) sqlite3VdbeCountChanges(v);
sl@0
   321
  sqlite3BeginWriteOperation(pParse, triggers_exist, iDb);
sl@0
   322
sl@0
   323
  if( triggers_exist ){
sl@0
   324
    int orconf = ((pParse->trigStack)?pParse->trigStack->orconf:OE_Default);
sl@0
   325
    int iGoto = sqlite3VdbeAddOp0(v, OP_Goto);
sl@0
   326
    addr = sqlite3VdbeMakeLabel(v);
sl@0
   327
sl@0
   328
    iBeginBeforeTrigger = sqlite3VdbeCurrentAddr(v);
sl@0
   329
    (void)sqlite3CodeRowTrigger(pParse, TK_DELETE, 0, TRIGGER_BEFORE, pTab,
sl@0
   330
        -1, oldIdx, orconf, addr, &old_col_mask, 0);
sl@0
   331
    iEndBeforeTrigger = sqlite3VdbeAddOp0(v, OP_Goto);
sl@0
   332
sl@0
   333
    iBeginAfterTrigger = sqlite3VdbeCurrentAddr(v);
sl@0
   334
    (void)sqlite3CodeRowTrigger(pParse, TK_DELETE, 0, TRIGGER_AFTER, pTab, -1,
sl@0
   335
        oldIdx, orconf, addr, &old_col_mask, 0);
sl@0
   336
    iEndAfterTrigger = sqlite3VdbeAddOp0(v, OP_Goto);
sl@0
   337
sl@0
   338
    sqlite3VdbeJumpHere(v, iGoto);
sl@0
   339
  }
sl@0
   340
sl@0
   341
  /* If we are trying to delete from a view, realize that view into
sl@0
   342
  ** a ephemeral table.
sl@0
   343
  */
sl@0
   344
#if !defined(SQLITE_OMIT_VIEW) && !defined(SQLITE_OMIT_TRIGGER)
sl@0
   345
  if( isView ){
sl@0
   346
    sqlite3MaterializeView(pParse, pTab, pWhere, iCur);
sl@0
   347
  }
sl@0
   348
#endif
sl@0
   349
sl@0
   350
  /* Resolve the column names in the WHERE clause.
sl@0
   351
  */
sl@0
   352
  memset(&sNC, 0, sizeof(sNC));
sl@0
   353
  sNC.pParse = pParse;
sl@0
   354
  sNC.pSrcList = pTabList;
sl@0
   355
  if( sqlite3ResolveExprNames(&sNC, pWhere) ){
sl@0
   356
    goto delete_from_cleanup;
sl@0
   357
  }
sl@0
   358
sl@0
   359
  /* Initialize the counter of the number of rows deleted, if
sl@0
   360
  ** we are counting rows.
sl@0
   361
  */
sl@0
   362
  if( db->flags & SQLITE_CountRows ){
sl@0
   363
    memCnt = ++pParse->nMem;
sl@0
   364
    sqlite3VdbeAddOp2(v, OP_Integer, 0, memCnt);
sl@0
   365
  }
sl@0
   366
sl@0
   367
#ifndef SQLITE_OMIT_TRUNCATE_OPTIMIZATION
sl@0
   368
  /* Special case: A DELETE without a WHERE clause deletes everything.
sl@0
   369
  ** It is easier just to erase the whole table.  Note, however, that
sl@0
   370
  ** this means that the row change count will be incorrect.
sl@0
   371
  */
sl@0
   372
  if( pWhere==0 && !triggers_exist && !IsVirtual(pTab) ){
sl@0
   373
    if( db->flags & SQLITE_CountRows ){
sl@0
   374
      /* If counting rows deleted, just count the total number of
sl@0
   375
      ** entries in the table. */
sl@0
   376
      int addr2;
sl@0
   377
      if( !isView ){
sl@0
   378
        sqlite3OpenTable(pParse, iCur, iDb, pTab, OP_OpenRead);
sl@0
   379
      }
sl@0
   380
      sqlite3VdbeAddOp2(v, OP_Rewind, iCur, sqlite3VdbeCurrentAddr(v)+2);
sl@0
   381
      addr2 = sqlite3VdbeAddOp2(v, OP_AddImm, memCnt, 1);
sl@0
   382
      sqlite3VdbeAddOp2(v, OP_Next, iCur, addr2);
sl@0
   383
      sqlite3VdbeAddOp1(v, OP_Close, iCur);
sl@0
   384
    }
sl@0
   385
    if( !isView ){
sl@0
   386
      sqlite3VdbeAddOp2(v, OP_Clear, pTab->tnum, iDb);
sl@0
   387
      if( !pParse->nested ){
sl@0
   388
        sqlite3VdbeChangeP4(v, -1, pTab->zName, P4_STATIC);
sl@0
   389
      }
sl@0
   390
      for(pIdx=pTab->pIndex; pIdx; pIdx=pIdx->pNext){
sl@0
   391
        assert( pIdx->pSchema==pTab->pSchema );
sl@0
   392
        sqlite3VdbeAddOp2(v, OP_Clear, pIdx->tnum, iDb);
sl@0
   393
      }
sl@0
   394
    }
sl@0
   395
  }else
sl@0
   396
#endif /* SQLITE_OMIT_TRUNCATE_OPTIMIZATION */
sl@0
   397
  /* The usual case: There is a WHERE clause so we have to scan through
sl@0
   398
  ** the table and pick which records to delete.
sl@0
   399
  */
sl@0
   400
  {
sl@0
   401
    int iRowid = ++pParse->nMem;    /* Used for storing rowid values. */
sl@0
   402
sl@0
   403
    /* Begin the database scan
sl@0
   404
    */
sl@0
   405
    pWInfo = sqlite3WhereBegin(pParse, pTabList, pWhere, 0, 0);
sl@0
   406
    if( pWInfo==0 ) goto delete_from_cleanup;
sl@0
   407
sl@0
   408
    /* Remember the rowid of every item to be deleted.
sl@0
   409
    */
sl@0
   410
    sqlite3VdbeAddOp2(v, IsVirtual(pTab) ? OP_VRowid : OP_Rowid, iCur, iRowid);
sl@0
   411
    sqlite3VdbeAddOp1(v, OP_FifoWrite, iRowid);
sl@0
   412
    if( db->flags & SQLITE_CountRows ){
sl@0
   413
      sqlite3VdbeAddOp2(v, OP_AddImm, memCnt, 1);
sl@0
   414
    }
sl@0
   415
sl@0
   416
    /* End the database scan loop.
sl@0
   417
    */
sl@0
   418
    sqlite3WhereEnd(pWInfo);
sl@0
   419
sl@0
   420
    /* Open the pseudo-table used to store OLD if there are triggers.
sl@0
   421
    */
sl@0
   422
    if( triggers_exist ){
sl@0
   423
      sqlite3VdbeAddOp2(v, OP_SetNumColumns, 0, pTab->nCol);
sl@0
   424
      sqlite3VdbeAddOp1(v, OP_OpenPseudo, oldIdx);
sl@0
   425
    }
sl@0
   426
sl@0
   427
    /* Delete every item whose key was written to the list during the
sl@0
   428
    ** database scan.  We have to delete items after the scan is complete
sl@0
   429
    ** because deleting an item can change the scan order.
sl@0
   430
    */
sl@0
   431
    end = sqlite3VdbeMakeLabel(v);
sl@0
   432
sl@0
   433
    if( !isView ){
sl@0
   434
      /* Open cursors for the table we are deleting from and 
sl@0
   435
      ** all its indices.
sl@0
   436
      */
sl@0
   437
      sqlite3OpenTableAndIndices(pParse, pTab, iCur, OP_OpenWrite);
sl@0
   438
    }
sl@0
   439
sl@0
   440
    /* This is the beginning of the delete loop. If a trigger encounters
sl@0
   441
    ** an IGNORE constraint, it jumps back to here.
sl@0
   442
    */
sl@0
   443
    if( triggers_exist ){
sl@0
   444
      sqlite3VdbeResolveLabel(v, addr);
sl@0
   445
    }
sl@0
   446
    addr = sqlite3VdbeAddOp2(v, OP_FifoRead, iRowid, end);
sl@0
   447
sl@0
   448
    if( triggers_exist ){
sl@0
   449
      int iData = ++pParse->nMem;   /* For storing row data of OLD table */
sl@0
   450
sl@0
   451
      /* If the record is no longer present in the table, jump to the
sl@0
   452
      ** next iteration of the loop through the contents of the fifo.
sl@0
   453
      */
sl@0
   454
      sqlite3VdbeAddOp3(v, OP_NotExists, iCur, addr, iRowid);
sl@0
   455
sl@0
   456
      /* Populate the OLD.* pseudo-table */
sl@0
   457
      if( old_col_mask ){
sl@0
   458
        sqlite3VdbeAddOp2(v, OP_RowData, iCur, iData);
sl@0
   459
      }else{
sl@0
   460
        sqlite3VdbeAddOp2(v, OP_Null, 0, iData);
sl@0
   461
      }
sl@0
   462
      sqlite3VdbeAddOp3(v, OP_Insert, oldIdx, iData, iRowid);
sl@0
   463
sl@0
   464
      /* Jump back and run the BEFORE triggers */
sl@0
   465
      sqlite3VdbeAddOp2(v, OP_Goto, 0, iBeginBeforeTrigger);
sl@0
   466
      sqlite3VdbeJumpHere(v, iEndBeforeTrigger);
sl@0
   467
    }
sl@0
   468
sl@0
   469
    if( !isView ){
sl@0
   470
      /* Delete the row */
sl@0
   471
#ifndef SQLITE_OMIT_VIRTUALTABLE
sl@0
   472
      if( IsVirtual(pTab) ){
sl@0
   473
        const char *pVtab = (const char *)pTab->pVtab;
sl@0
   474
        sqlite3VtabMakeWritable(pParse, pTab);
sl@0
   475
        sqlite3VdbeAddOp4(v, OP_VUpdate, 0, 1, iRowid, pVtab, P4_VTAB);
sl@0
   476
      }else
sl@0
   477
#endif
sl@0
   478
      {
sl@0
   479
        sqlite3GenerateRowDelete(pParse, pTab, iCur, iRowid, pParse->nested==0);
sl@0
   480
      }
sl@0
   481
    }
sl@0
   482
sl@0
   483
    /* If there are row triggers, close all cursors then invoke
sl@0
   484
    ** the AFTER triggers
sl@0
   485
    */
sl@0
   486
    if( triggers_exist ){
sl@0
   487
      /* Jump back and run the AFTER triggers */
sl@0
   488
      sqlite3VdbeAddOp2(v, OP_Goto, 0, iBeginAfterTrigger);
sl@0
   489
      sqlite3VdbeJumpHere(v, iEndAfterTrigger);
sl@0
   490
    }
sl@0
   491
sl@0
   492
    /* End of the delete loop */
sl@0
   493
    sqlite3VdbeAddOp2(v, OP_Goto, 0, addr);
sl@0
   494
    sqlite3VdbeResolveLabel(v, end);
sl@0
   495
sl@0
   496
    /* Close the cursors after the loop if there are no row triggers */
sl@0
   497
    if( !isView  && !IsVirtual(pTab) ){
sl@0
   498
      for(i=1, pIdx=pTab->pIndex; pIdx; i++, pIdx=pIdx->pNext){
sl@0
   499
        sqlite3VdbeAddOp2(v, OP_Close, iCur + i, pIdx->tnum);
sl@0
   500
      }
sl@0
   501
      sqlite3VdbeAddOp1(v, OP_Close, iCur);
sl@0
   502
    }
sl@0
   503
  }
sl@0
   504
sl@0
   505
  /*
sl@0
   506
  ** Return the number of rows that were deleted. If this routine is 
sl@0
   507
  ** generating code because of a call to sqlite3NestedParse(), do not
sl@0
   508
  ** invoke the callback function.
sl@0
   509
  */
sl@0
   510
  if( db->flags & SQLITE_CountRows && pParse->nested==0 && !pParse->trigStack ){
sl@0
   511
    sqlite3VdbeAddOp2(v, OP_ResultRow, memCnt, 1);
sl@0
   512
    sqlite3VdbeSetNumCols(v, 1);
sl@0
   513
    sqlite3VdbeSetColName(v, 0, COLNAME_NAME, "rows deleted", P4_STATIC);
sl@0
   514
  }
sl@0
   515
sl@0
   516
delete_from_cleanup:
sl@0
   517
  sqlite3AuthContextPop(&sContext);
sl@0
   518
  sqlite3SrcListDelete(db, pTabList);
sl@0
   519
  sqlite3ExprDelete(db, pWhere);
sl@0
   520
  return;
sl@0
   521
}
sl@0
   522
sl@0
   523
/*
sl@0
   524
** This routine generates VDBE code that causes a single row of a
sl@0
   525
** single table to be deleted.
sl@0
   526
**
sl@0
   527
** The VDBE must be in a particular state when this routine is called.
sl@0
   528
** These are the requirements:
sl@0
   529
**
sl@0
   530
**   1.  A read/write cursor pointing to pTab, the table containing the row
sl@0
   531
**       to be deleted, must be opened as cursor number "base".
sl@0
   532
**
sl@0
   533
**   2.  Read/write cursors for all indices of pTab must be open as
sl@0
   534
**       cursor number base+i for the i-th index.
sl@0
   535
**
sl@0
   536
**   3.  The record number of the row to be deleted must be stored in
sl@0
   537
**       memory cell iRowid.
sl@0
   538
**
sl@0
   539
** This routine pops the top of the stack to remove the record number
sl@0
   540
** and then generates code to remove both the table record and all index
sl@0
   541
** entries that point to that record.
sl@0
   542
*/
sl@0
   543
void sqlite3GenerateRowDelete(
sl@0
   544
  Parse *pParse,     /* Parsing context */
sl@0
   545
  Table *pTab,       /* Table containing the row to be deleted */
sl@0
   546
  int iCur,          /* Cursor number for the table */
sl@0
   547
  int iRowid,        /* Memory cell that contains the rowid to delete */
sl@0
   548
  int count          /* Increment the row change counter */
sl@0
   549
){
sl@0
   550
  int addr;
sl@0
   551
  Vdbe *v;
sl@0
   552
sl@0
   553
  v = pParse->pVdbe;
sl@0
   554
  addr = sqlite3VdbeAddOp3(v, OP_NotExists, iCur, 0, iRowid);
sl@0
   555
  sqlite3GenerateRowIndexDelete(pParse, pTab, iCur, 0);
sl@0
   556
  sqlite3VdbeAddOp2(v, OP_Delete, iCur, (count?OPFLAG_NCHANGE:0));
sl@0
   557
  if( count ){
sl@0
   558
    sqlite3VdbeChangeP4(v, -1, pTab->zName, P4_STATIC);
sl@0
   559
  }
sl@0
   560
  sqlite3VdbeJumpHere(v, addr);
sl@0
   561
}
sl@0
   562
sl@0
   563
/*
sl@0
   564
** This routine generates VDBE code that causes the deletion of all
sl@0
   565
** index entries associated with a single row of a single table.
sl@0
   566
**
sl@0
   567
** The VDBE must be in a particular state when this routine is called.
sl@0
   568
** These are the requirements:
sl@0
   569
**
sl@0
   570
**   1.  A read/write cursor pointing to pTab, the table containing the row
sl@0
   571
**       to be deleted, must be opened as cursor number "iCur".
sl@0
   572
**
sl@0
   573
**   2.  Read/write cursors for all indices of pTab must be open as
sl@0
   574
**       cursor number iCur+i for the i-th index.
sl@0
   575
**
sl@0
   576
**   3.  The "iCur" cursor must be pointing to the row that is to be
sl@0
   577
**       deleted.
sl@0
   578
*/
sl@0
   579
void sqlite3GenerateRowIndexDelete(
sl@0
   580
  Parse *pParse,     /* Parsing and code generating context */
sl@0
   581
  Table *pTab,       /* Table containing the row to be deleted */
sl@0
   582
  int iCur,          /* Cursor number for the table */
sl@0
   583
  int *aRegIdx       /* Only delete if aRegIdx!=0 && aRegIdx[i]>0 */
sl@0
   584
){
sl@0
   585
  int i;
sl@0
   586
  Index *pIdx;
sl@0
   587
  int r1;
sl@0
   588
sl@0
   589
  for(i=1, pIdx=pTab->pIndex; pIdx; i++, pIdx=pIdx->pNext){
sl@0
   590
    if( aRegIdx!=0 && aRegIdx[i-1]==0 ) continue;
sl@0
   591
    r1 = sqlite3GenerateIndexKey(pParse, pIdx, iCur, 0, 0);
sl@0
   592
    sqlite3VdbeAddOp3(pParse->pVdbe, OP_IdxDelete, iCur+i, r1,pIdx->nColumn+1);
sl@0
   593
  }
sl@0
   594
}
sl@0
   595
sl@0
   596
/*
sl@0
   597
** Generate code that will assemble an index key and put it in register
sl@0
   598
** regOut.  The key with be for index pIdx which is an index on pTab.
sl@0
   599
** iCur is the index of a cursor open on the pTab table and pointing to
sl@0
   600
** the entry that needs indexing.
sl@0
   601
**
sl@0
   602
** Return a register number which is the first in a block of
sl@0
   603
** registers that holds the elements of the index key.  The
sl@0
   604
** block of registers has already been deallocated by the time
sl@0
   605
** this routine returns.
sl@0
   606
*/
sl@0
   607
int sqlite3GenerateIndexKey(
sl@0
   608
  Parse *pParse,     /* Parsing context */
sl@0
   609
  Index *pIdx,       /* The index for which to generate a key */
sl@0
   610
  int iCur,          /* Cursor number for the pIdx->pTable table */
sl@0
   611
  int regOut,        /* Write the new index key to this register */
sl@0
   612
  int doMakeRec      /* Run the OP_MakeRecord instruction if true */
sl@0
   613
){
sl@0
   614
  Vdbe *v = pParse->pVdbe;
sl@0
   615
  int j;
sl@0
   616
  Table *pTab = pIdx->pTable;
sl@0
   617
  int regBase;
sl@0
   618
  int nCol;
sl@0
   619
sl@0
   620
  nCol = pIdx->nColumn;
sl@0
   621
  regBase = sqlite3GetTempRange(pParse, nCol+1);
sl@0
   622
  sqlite3VdbeAddOp2(v, OP_Rowid, iCur, regBase+nCol);
sl@0
   623
  for(j=0; j<nCol; j++){
sl@0
   624
    int idx = pIdx->aiColumn[j];
sl@0
   625
    if( idx==pTab->iPKey ){
sl@0
   626
      sqlite3VdbeAddOp2(v, OP_SCopy, regBase+nCol, regBase+j);
sl@0
   627
    }else{
sl@0
   628
      sqlite3VdbeAddOp3(v, OP_Column, iCur, idx, regBase+j);
sl@0
   629
      sqlite3ColumnDefault(v, pTab, idx);
sl@0
   630
    }
sl@0
   631
  }
sl@0
   632
  if( doMakeRec ){
sl@0
   633
    sqlite3VdbeAddOp3(v, OP_MakeRecord, regBase, nCol+1, regOut);
sl@0
   634
    sqlite3IndexAffinityStr(v, pIdx);
sl@0
   635
    sqlite3ExprCacheAffinityChange(pParse, regBase, nCol+1);
sl@0
   636
  }
sl@0
   637
  sqlite3ReleaseTempRange(pParse, regBase, nCol+1);
sl@0
   638
  return regBase;
sl@0
   639
}
sl@0
   640
sl@0
   641
/* Make sure "isView" gets undefined in case this file becomes part of
sl@0
   642
** the amalgamation - so that subsequent files do not see isView as a
sl@0
   643
** macro. */
sl@0
   644
#undef isView