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