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