os/persistentdata/persistentstorage/sqlite3api/SQLite/vacuum.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
** 2003 April 6
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 code used to implement the VACUUM command.
sl@0
    13
**
sl@0
    14
** Most of the code in this file may be omitted by defining the
sl@0
    15
** SQLITE_OMIT_VACUUM macro.
sl@0
    16
**
sl@0
    17
** $Id: vacuum.c,v 1.83 2008/08/26 21:07:27 drh Exp $
sl@0
    18
*/
sl@0
    19
#include "sqliteInt.h"
sl@0
    20
#include "vdbeInt.h"
sl@0
    21
sl@0
    22
#if !defined(SQLITE_OMIT_VACUUM) && !defined(SQLITE_OMIT_ATTACH)
sl@0
    23
/*
sl@0
    24
** Execute zSql on database db. Return an error code.
sl@0
    25
*/
sl@0
    26
static int execSql(sqlite3 *db, const char *zSql){
sl@0
    27
  sqlite3_stmt *pStmt;
sl@0
    28
  if( !zSql ){
sl@0
    29
    return SQLITE_NOMEM;
sl@0
    30
  }
sl@0
    31
  if( SQLITE_OK!=sqlite3_prepare(db, zSql, -1, &pStmt, 0) ){
sl@0
    32
    return sqlite3_errcode(db);
sl@0
    33
  }
sl@0
    34
  while( SQLITE_ROW==sqlite3_step(pStmt) ){}
sl@0
    35
  return sqlite3_finalize(pStmt);
sl@0
    36
}
sl@0
    37
sl@0
    38
/*
sl@0
    39
** Execute zSql on database db. The statement returns exactly
sl@0
    40
** one column. Execute this as SQL on the same database.
sl@0
    41
*/
sl@0
    42
static int execExecSql(sqlite3 *db, const char *zSql){
sl@0
    43
  sqlite3_stmt *pStmt;
sl@0
    44
  int rc;
sl@0
    45
sl@0
    46
  rc = sqlite3_prepare(db, zSql, -1, &pStmt, 0);
sl@0
    47
  if( rc!=SQLITE_OK ) return rc;
sl@0
    48
sl@0
    49
  while( SQLITE_ROW==sqlite3_step(pStmt) ){
sl@0
    50
    rc = execSql(db, (char*)sqlite3_column_text(pStmt, 0));
sl@0
    51
    if( rc!=SQLITE_OK ){
sl@0
    52
      sqlite3_finalize(pStmt);
sl@0
    53
      return rc;
sl@0
    54
    }
sl@0
    55
  }
sl@0
    56
sl@0
    57
  return sqlite3_finalize(pStmt);
sl@0
    58
}
sl@0
    59
sl@0
    60
/*
sl@0
    61
** The non-standard VACUUM command is used to clean up the database,
sl@0
    62
** collapse free space, etc.  It is modelled after the VACUUM command
sl@0
    63
** in PostgreSQL.
sl@0
    64
**
sl@0
    65
** In version 1.0.x of SQLite, the VACUUM command would call
sl@0
    66
** gdbm_reorganize() on all the database tables.  But beginning
sl@0
    67
** with 2.0.0, SQLite no longer uses GDBM so this command has
sl@0
    68
** become a no-op.
sl@0
    69
*/
sl@0
    70
void sqlite3Vacuum(Parse *pParse){
sl@0
    71
  Vdbe *v = sqlite3GetVdbe(pParse);
sl@0
    72
  if( v ){
sl@0
    73
    sqlite3VdbeAddOp2(v, OP_Vacuum, 0, 0);
sl@0
    74
  }
sl@0
    75
  return;
sl@0
    76
}
sl@0
    77
sl@0
    78
/*
sl@0
    79
** This routine implements the OP_Vacuum opcode of the VDBE.
sl@0
    80
*/
sl@0
    81
int sqlite3RunVacuum(char **pzErrMsg, sqlite3 *db){
sl@0
    82
  int rc = SQLITE_OK;     /* Return code from service routines */
sl@0
    83
  Btree *pMain;           /* The database being vacuumed */
sl@0
    84
  Pager *pMainPager;      /* Pager for database being vacuumed */
sl@0
    85
  Btree *pTemp;           /* The temporary database we vacuum into */
sl@0
    86
  char *zSql = 0;         /* SQL statements */
sl@0
    87
  int saved_flags;        /* Saved value of the db->flags */
sl@0
    88
  int saved_nChange;      /* Saved value of db->nChange */
sl@0
    89
  int saved_nTotalChange; /* Saved value of db->nTotalChange */
sl@0
    90
  Db *pDb = 0;            /* Database to detach at end of vacuum */
sl@0
    91
  int isMemDb;            /* True is vacuuming a :memory: database */
sl@0
    92
  int nRes;
sl@0
    93
sl@0
    94
  /* Save the current value of the write-schema flag before setting it. */
sl@0
    95
  saved_flags = db->flags;
sl@0
    96
  saved_nChange = db->nChange;
sl@0
    97
  saved_nTotalChange = db->nTotalChange;
sl@0
    98
  db->flags |= SQLITE_WriteSchema | SQLITE_IgnoreChecks;
sl@0
    99
sl@0
   100
  if( !db->autoCommit ){
sl@0
   101
    sqlite3SetString(pzErrMsg, db, "cannot VACUUM from within a transaction");
sl@0
   102
    rc = SQLITE_ERROR;
sl@0
   103
    goto end_of_vacuum;
sl@0
   104
  }
sl@0
   105
  pMain = db->aDb[0].pBt;
sl@0
   106
  pMainPager = sqlite3BtreePager(pMain);
sl@0
   107
  isMemDb = sqlite3PagerFile(pMainPager)->pMethods==0;
sl@0
   108
sl@0
   109
  /* Attach the temporary database as 'vacuum_db'. The synchronous pragma
sl@0
   110
  ** can be set to 'off' for this file, as it is not recovered if a crash
sl@0
   111
  ** occurs anyway. The integrity of the database is maintained by a
sl@0
   112
  ** (possibly synchronous) transaction opened on the main database before
sl@0
   113
  ** sqlite3BtreeCopyFile() is called.
sl@0
   114
  **
sl@0
   115
  ** An optimisation would be to use a non-journaled pager.
sl@0
   116
  ** (Later:) I tried setting "PRAGMA vacuum_db.journal_mode=OFF" but
sl@0
   117
  ** that actually made the VACUUM run slower.  Very little journalling
sl@0
   118
  ** actually occurs when doing a vacuum since the vacuum_db is initially
sl@0
   119
  ** empty.  Only the journal header is written.  Apparently it takes more
sl@0
   120
  ** time to parse and run the PRAGMA to turn journalling off than it does
sl@0
   121
  ** to write the journal header file.
sl@0
   122
  */
sl@0
   123
  zSql = "ATTACH '' AS vacuum_db;";
sl@0
   124
  rc = execSql(db, zSql);
sl@0
   125
  if( rc!=SQLITE_OK ) goto end_of_vacuum;
sl@0
   126
  pDb = &db->aDb[db->nDb-1];
sl@0
   127
  assert( strcmp(db->aDb[db->nDb-1].zName,"vacuum_db")==0 );
sl@0
   128
  pTemp = db->aDb[db->nDb-1].pBt;
sl@0
   129
sl@0
   130
  nRes = sqlite3BtreeGetReserve(pMain);
sl@0
   131
sl@0
   132
  /* A VACUUM cannot change the pagesize of an encrypted database. */
sl@0
   133
#ifdef SQLITE_HAS_CODEC
sl@0
   134
  if( db->nextPagesize ){
sl@0
   135
    extern void sqlite3CodecGetKey(sqlite3*, int, void**, int*);
sl@0
   136
    int nKey;
sl@0
   137
    char *zKey;
sl@0
   138
    sqlite3CodecGetKey(db, 0, (void**)&zKey, &nKey);
sl@0
   139
    if( nKey ) db->nextPagesize = 0;
sl@0
   140
  }
sl@0
   141
#endif
sl@0
   142
sl@0
   143
  if( sqlite3BtreeSetPageSize(pTemp, sqlite3BtreeGetPageSize(pMain), nRes)
sl@0
   144
   || (!isMemDb && sqlite3BtreeSetPageSize(pTemp, db->nextPagesize, nRes))
sl@0
   145
   || db->mallocFailed 
sl@0
   146
  ){
sl@0
   147
    rc = SQLITE_NOMEM;
sl@0
   148
    goto end_of_vacuum;
sl@0
   149
  }
sl@0
   150
  rc = execSql(db, "PRAGMA vacuum_db.synchronous=OFF");
sl@0
   151
  if( rc!=SQLITE_OK ){
sl@0
   152
    goto end_of_vacuum;
sl@0
   153
  }
sl@0
   154
sl@0
   155
#ifndef SQLITE_OMIT_AUTOVACUUM
sl@0
   156
  sqlite3BtreeSetAutoVacuum(pTemp, db->nextAutovac>=0 ? db->nextAutovac :
sl@0
   157
                                           sqlite3BtreeGetAutoVacuum(pMain));
sl@0
   158
#endif
sl@0
   159
sl@0
   160
  /* Begin a transaction */
sl@0
   161
  rc = execSql(db, "BEGIN EXCLUSIVE;");
sl@0
   162
  if( rc!=SQLITE_OK ) goto end_of_vacuum;
sl@0
   163
sl@0
   164
  /* Query the schema of the main database. Create a mirror schema
sl@0
   165
  ** in the temporary database.
sl@0
   166
  */
sl@0
   167
  rc = execExecSql(db, 
sl@0
   168
      "SELECT 'CREATE TABLE vacuum_db.' || substr(sql,14) "
sl@0
   169
      "  FROM sqlite_master WHERE type='table' AND name!='sqlite_sequence'"
sl@0
   170
      "   AND rootpage>0"
sl@0
   171
  );
sl@0
   172
  if( rc!=SQLITE_OK ) goto end_of_vacuum;
sl@0
   173
  rc = execExecSql(db, 
sl@0
   174
      "SELECT 'CREATE INDEX vacuum_db.' || substr(sql,14)"
sl@0
   175
      "  FROM sqlite_master WHERE sql LIKE 'CREATE INDEX %' ");
sl@0
   176
  if( rc!=SQLITE_OK ) goto end_of_vacuum;
sl@0
   177
  rc = execExecSql(db, 
sl@0
   178
      "SELECT 'CREATE UNIQUE INDEX vacuum_db.' || substr(sql,21) "
sl@0
   179
      "  FROM sqlite_master WHERE sql LIKE 'CREATE UNIQUE INDEX %'");
sl@0
   180
  if( rc!=SQLITE_OK ) goto end_of_vacuum;
sl@0
   181
sl@0
   182
  /* Loop through the tables in the main database. For each, do
sl@0
   183
  ** an "INSERT INTO vacuum_db.xxx SELECT * FROM xxx;" to copy
sl@0
   184
  ** the contents to the temporary database.
sl@0
   185
  */
sl@0
   186
  rc = execExecSql(db, 
sl@0
   187
      "SELECT 'INSERT INTO vacuum_db.' || quote(name) "
sl@0
   188
      "|| ' SELECT * FROM ' || quote(name) || ';'"
sl@0
   189
      "FROM sqlite_master "
sl@0
   190
      "WHERE type = 'table' AND name!='sqlite_sequence' "
sl@0
   191
      "  AND rootpage>0"
sl@0
   192
sl@0
   193
  );
sl@0
   194
  if( rc!=SQLITE_OK ) goto end_of_vacuum;
sl@0
   195
sl@0
   196
  /* Copy over the sequence table
sl@0
   197
  */
sl@0
   198
  rc = execExecSql(db, 
sl@0
   199
      "SELECT 'DELETE FROM vacuum_db.' || quote(name) || ';' "
sl@0
   200
      "FROM vacuum_db.sqlite_master WHERE name='sqlite_sequence' "
sl@0
   201
  );
sl@0
   202
  if( rc!=SQLITE_OK ) goto end_of_vacuum;
sl@0
   203
  rc = execExecSql(db, 
sl@0
   204
      "SELECT 'INSERT INTO vacuum_db.' || quote(name) "
sl@0
   205
      "|| ' SELECT * FROM ' || quote(name) || ';' "
sl@0
   206
      "FROM vacuum_db.sqlite_master WHERE name=='sqlite_sequence';"
sl@0
   207
  );
sl@0
   208
  if( rc!=SQLITE_OK ) goto end_of_vacuum;
sl@0
   209
sl@0
   210
sl@0
   211
  /* Copy the triggers, views, and virtual tables from the main database
sl@0
   212
  ** over to the temporary database.  None of these objects has any
sl@0
   213
  ** associated storage, so all we have to do is copy their entries
sl@0
   214
  ** from the SQLITE_MASTER table.
sl@0
   215
  */
sl@0
   216
  rc = execSql(db,
sl@0
   217
      "INSERT INTO vacuum_db.sqlite_master "
sl@0
   218
      "  SELECT type, name, tbl_name, rootpage, sql"
sl@0
   219
      "    FROM sqlite_master"
sl@0
   220
      "   WHERE type='view' OR type='trigger'"
sl@0
   221
      "      OR (type='table' AND rootpage=0)"
sl@0
   222
  );
sl@0
   223
  if( rc ) goto end_of_vacuum;
sl@0
   224
sl@0
   225
  /* At this point, unless the main db was completely empty, there is now a
sl@0
   226
  ** transaction open on the vacuum database, but not on the main database.
sl@0
   227
  ** Open a btree level transaction on the main database. This allows a
sl@0
   228
  ** call to sqlite3BtreeCopyFile(). The main database btree level
sl@0
   229
  ** transaction is then committed, so the SQL level never knows it was
sl@0
   230
  ** opened for writing. This way, the SQL transaction used to create the
sl@0
   231
  ** temporary database never needs to be committed.
sl@0
   232
  */
sl@0
   233
  if( rc==SQLITE_OK ){
sl@0
   234
    u32 meta;
sl@0
   235
    int i;
sl@0
   236
sl@0
   237
    /* This array determines which meta meta values are preserved in the
sl@0
   238
    ** vacuum.  Even entries are the meta value number and odd entries
sl@0
   239
    ** are an increment to apply to the meta value after the vacuum.
sl@0
   240
    ** The increment is used to increase the schema cookie so that other
sl@0
   241
    ** connections to the same database will know to reread the schema.
sl@0
   242
    */
sl@0
   243
    static const unsigned char aCopy[] = {
sl@0
   244
       1, 1,    /* Add one to the old schema cookie */
sl@0
   245
       3, 0,    /* Preserve the default page cache size */
sl@0
   246
       5, 0,    /* Preserve the default text encoding */
sl@0
   247
       6, 0,    /* Preserve the user version */
sl@0
   248
    };
sl@0
   249
sl@0
   250
    assert( 1==sqlite3BtreeIsInTrans(pTemp) );
sl@0
   251
    assert( 1==sqlite3BtreeIsInTrans(pMain) );
sl@0
   252
sl@0
   253
    /* Copy Btree meta values */
sl@0
   254
    for(i=0; i<sizeof(aCopy)/sizeof(aCopy[0]); i+=2){
sl@0
   255
      rc = sqlite3BtreeGetMeta(pMain, aCopy[i], &meta);
sl@0
   256
      if( rc!=SQLITE_OK ) goto end_of_vacuum;
sl@0
   257
      rc = sqlite3BtreeUpdateMeta(pTemp, aCopy[i], meta+aCopy[i+1]);
sl@0
   258
      if( rc!=SQLITE_OK ) goto end_of_vacuum;
sl@0
   259
    }
sl@0
   260
sl@0
   261
    rc = sqlite3BtreeCopyFile(pMain, pTemp);
sl@0
   262
    if( rc!=SQLITE_OK ) goto end_of_vacuum;
sl@0
   263
    rc = sqlite3BtreeCommit(pTemp);
sl@0
   264
    if( rc!=SQLITE_OK ) goto end_of_vacuum;
sl@0
   265
#ifndef SQLITE_OMIT_AUTOVACUUM
sl@0
   266
    sqlite3BtreeSetAutoVacuum(pMain, sqlite3BtreeGetAutoVacuum(pTemp));
sl@0
   267
#endif
sl@0
   268
    rc = sqlite3BtreeCommit(pMain);
sl@0
   269
  }
sl@0
   270
sl@0
   271
  if( rc==SQLITE_OK ){
sl@0
   272
    rc = sqlite3BtreeSetPageSize(pMain, sqlite3BtreeGetPageSize(pTemp), nRes);
sl@0
   273
  }
sl@0
   274
sl@0
   275
end_of_vacuum:
sl@0
   276
  /* Restore the original value of db->flags */
sl@0
   277
  db->flags = saved_flags;
sl@0
   278
  db->nChange = saved_nChange;
sl@0
   279
  db->nTotalChange = saved_nTotalChange;
sl@0
   280
sl@0
   281
  /* Currently there is an SQL level transaction open on the vacuum
sl@0
   282
  ** database. No locks are held on any other files (since the main file
sl@0
   283
  ** was committed at the btree level). So it safe to end the transaction
sl@0
   284
  ** by manually setting the autoCommit flag to true and detaching the
sl@0
   285
  ** vacuum database. The vacuum_db journal file is deleted when the pager
sl@0
   286
  ** is closed by the DETACH.
sl@0
   287
  */
sl@0
   288
  db->autoCommit = 1;
sl@0
   289
sl@0
   290
  if( pDb ){
sl@0
   291
    sqlite3BtreeClose(pDb->pBt);
sl@0
   292
    pDb->pBt = 0;
sl@0
   293
    pDb->pSchema = 0;
sl@0
   294
  }
sl@0
   295
sl@0
   296
  sqlite3ResetInternalSchema(db, 0);
sl@0
   297
sl@0
   298
  return rc;
sl@0
   299
}
sl@0
   300
#endif  /* SQLITE_OMIT_VACUUM && SQLITE_OMIT_ATTACH */