os/persistentdata/persistentstorage/sql/SQLite/vacuum.c
author sl@SLION-WIN7.fritz.box
Fri, 15 Jun 2012 03:10:57 +0200
changeset 0 bde4ae8d615e
permissions -rw-r--r--
First public contribution.
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.81 2008/07/08 19:34:07 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
  Btree *pTemp;           /* The temporary database we vacuum into */
sl@0
    85
  char *zSql = 0;         /* SQL statements */
sl@0
    86
  int saved_flags;        /* Saved value of the db->flags */
sl@0
    87
  int saved_nChange;      /* Saved value of db->nChange */
sl@0
    88
  int saved_nTotalChange; /* Saved value of db->nTotalChange */
sl@0
    89
  Db *pDb = 0;            /* Database to detach at end of vacuum */
sl@0
    90
  int nRes;
sl@0
    91
sl@0
    92
  /* Save the current value of the write-schema flag before setting it. */
sl@0
    93
  saved_flags = db->flags;
sl@0
    94
  saved_nChange = db->nChange;
sl@0
    95
  saved_nTotalChange = db->nTotalChange;
sl@0
    96
  db->flags |= SQLITE_WriteSchema | SQLITE_IgnoreChecks;
sl@0
    97
sl@0
    98
  if( !db->autoCommit ){
sl@0
    99
    sqlite3SetString(pzErrMsg, db, "cannot VACUUM from within a transaction");
sl@0
   100
    rc = SQLITE_ERROR;
sl@0
   101
    goto end_of_vacuum;
sl@0
   102
  }
sl@0
   103
  pMain = db->aDb[0].pBt;
sl@0
   104
sl@0
   105
  /* Attach the temporary database as 'vacuum_db'. The synchronous pragma
sl@0
   106
  ** can be set to 'off' for this file, as it is not recovered if a crash
sl@0
   107
  ** occurs anyway. The integrity of the database is maintained by a
sl@0
   108
  ** (possibly synchronous) transaction opened on the main database before
sl@0
   109
  ** sqlite3BtreeCopyFile() is called.
sl@0
   110
  **
sl@0
   111
  ** An optimisation would be to use a non-journaled pager.
sl@0
   112
  ** (Later:) I tried setting "PRAGMA vacuum_db.journal_mode=OFF" but
sl@0
   113
  ** that actually made the VACUUM run slower.  Very little journalling
sl@0
   114
  ** actually occurs when doing a vacuum since the vacuum_db is initially
sl@0
   115
  ** empty.  Only the journal header is written.  Apparently it takes more
sl@0
   116
  ** time to parse and run the PRAGMA to turn journalling off than it does
sl@0
   117
  ** to write the journal header file.
sl@0
   118
  */
sl@0
   119
  zSql = "ATTACH '' AS vacuum_db;";
sl@0
   120
  rc = execSql(db, zSql);
sl@0
   121
  if( rc!=SQLITE_OK ) goto end_of_vacuum;
sl@0
   122
  pDb = &db->aDb[db->nDb-1];
sl@0
   123
  assert( strcmp(db->aDb[db->nDb-1].zName,"vacuum_db")==0 );
sl@0
   124
  pTemp = db->aDb[db->nDb-1].pBt;
sl@0
   125
sl@0
   126
  nRes = sqlite3BtreeGetReserve(pMain);
sl@0
   127
sl@0
   128
  /* A VACUUM cannot change the pagesize of an encrypted database. */
sl@0
   129
#ifdef SQLITE_HAS_CODEC
sl@0
   130
  if( db->nextPagesize ){
sl@0
   131
    extern void sqlite3CodecGetKey(sqlite3*, int, void**, int*);
sl@0
   132
    int nKey;
sl@0
   133
    char *zKey;
sl@0
   134
    sqlite3CodecGetKey(db, 0, (void**)&zKey, &nKey);
sl@0
   135
    if( nKey ) db->nextPagesize = 0;
sl@0
   136
  }
sl@0
   137
#endif
sl@0
   138
sl@0
   139
  if( sqlite3BtreeSetPageSize(pTemp, sqlite3BtreeGetPageSize(pMain), nRes)
sl@0
   140
   || sqlite3BtreeSetPageSize(pTemp, db->nextPagesize, nRes)
sl@0
   141
   || db->mallocFailed 
sl@0
   142
  ){
sl@0
   143
    rc = SQLITE_NOMEM;
sl@0
   144
    goto end_of_vacuum;
sl@0
   145
  }
sl@0
   146
  rc = execSql(db, "PRAGMA vacuum_db.synchronous=OFF");
sl@0
   147
  if( rc!=SQLITE_OK ){
sl@0
   148
    goto end_of_vacuum;
sl@0
   149
  }
sl@0
   150
sl@0
   151
#ifndef SQLITE_OMIT_AUTOVACUUM
sl@0
   152
  sqlite3BtreeSetAutoVacuum(pTemp, db->nextAutovac>=0 ? db->nextAutovac :
sl@0
   153
                                           sqlite3BtreeGetAutoVacuum(pMain));
sl@0
   154
#endif
sl@0
   155
sl@0
   156
  /* Begin a transaction */
sl@0
   157
  rc = execSql(db, "BEGIN EXCLUSIVE;");
sl@0
   158
  if( rc!=SQLITE_OK ) goto end_of_vacuum;
sl@0
   159
sl@0
   160
  /* Query the schema of the main database. Create a mirror schema
sl@0
   161
  ** in the temporary database.
sl@0
   162
  */
sl@0
   163
  rc = execExecSql(db, 
sl@0
   164
      "SELECT 'CREATE TABLE vacuum_db.' || substr(sql,14) "
sl@0
   165
      "  FROM sqlite_master WHERE type='table' AND name!='sqlite_sequence'"
sl@0
   166
      "   AND rootpage>0"
sl@0
   167
  );
sl@0
   168
  if( rc!=SQLITE_OK ) goto end_of_vacuum;
sl@0
   169
  rc = execExecSql(db, 
sl@0
   170
      "SELECT 'CREATE INDEX vacuum_db.' || substr(sql,14)"
sl@0
   171
      "  FROM sqlite_master WHERE sql LIKE 'CREATE INDEX %' ");
sl@0
   172
  if( rc!=SQLITE_OK ) goto end_of_vacuum;
sl@0
   173
  rc = execExecSql(db, 
sl@0
   174
      "SELECT 'CREATE UNIQUE INDEX vacuum_db.' || substr(sql,21) "
sl@0
   175
      "  FROM sqlite_master WHERE sql LIKE 'CREATE UNIQUE INDEX %'");
sl@0
   176
  if( rc!=SQLITE_OK ) goto end_of_vacuum;
sl@0
   177
sl@0
   178
  /* Loop through the tables in the main database. For each, do
sl@0
   179
  ** an "INSERT INTO vacuum_db.xxx SELECT * FROM xxx;" to copy
sl@0
   180
  ** the contents to the temporary database.
sl@0
   181
  */
sl@0
   182
  rc = execExecSql(db, 
sl@0
   183
      "SELECT 'INSERT INTO vacuum_db.' || quote(name) "
sl@0
   184
      "|| ' SELECT * FROM ' || quote(name) || ';'"
sl@0
   185
      "FROM sqlite_master "
sl@0
   186
      "WHERE type = 'table' AND name!='sqlite_sequence' "
sl@0
   187
      "  AND rootpage>0"
sl@0
   188
sl@0
   189
  );
sl@0
   190
  if( rc!=SQLITE_OK ) goto end_of_vacuum;
sl@0
   191
sl@0
   192
  /* Copy over the sequence table
sl@0
   193
  */
sl@0
   194
  rc = execExecSql(db, 
sl@0
   195
      "SELECT 'DELETE FROM vacuum_db.' || quote(name) || ';' "
sl@0
   196
      "FROM vacuum_db.sqlite_master WHERE name='sqlite_sequence' "
sl@0
   197
  );
sl@0
   198
  if( rc!=SQLITE_OK ) goto end_of_vacuum;
sl@0
   199
  rc = execExecSql(db, 
sl@0
   200
      "SELECT 'INSERT INTO vacuum_db.' || quote(name) "
sl@0
   201
      "|| ' SELECT * FROM ' || quote(name) || ';' "
sl@0
   202
      "FROM vacuum_db.sqlite_master WHERE name=='sqlite_sequence';"
sl@0
   203
  );
sl@0
   204
  if( rc!=SQLITE_OK ) goto end_of_vacuum;
sl@0
   205
sl@0
   206
sl@0
   207
  /* Copy the triggers, views, and virtual tables from the main database
sl@0
   208
  ** over to the temporary database.  None of these objects has any
sl@0
   209
  ** associated storage, so all we have to do is copy their entries
sl@0
   210
  ** from the SQLITE_MASTER table.
sl@0
   211
  */
sl@0
   212
  rc = execSql(db,
sl@0
   213
      "INSERT INTO vacuum_db.sqlite_master "
sl@0
   214
      "  SELECT type, name, tbl_name, rootpage, sql"
sl@0
   215
      "    FROM sqlite_master"
sl@0
   216
      "   WHERE type='view' OR type='trigger'"
sl@0
   217
      "      OR (type='table' AND rootpage=0)"
sl@0
   218
  );
sl@0
   219
  if( rc ) goto end_of_vacuum;
sl@0
   220
sl@0
   221
  /* At this point, unless the main db was completely empty, there is now a
sl@0
   222
  ** transaction open on the vacuum database, but not on the main database.
sl@0
   223
  ** Open a btree level transaction on the main database. This allows a
sl@0
   224
  ** call to sqlite3BtreeCopyFile(). The main database btree level
sl@0
   225
  ** transaction is then committed, so the SQL level never knows it was
sl@0
   226
  ** opened for writing. This way, the SQL transaction used to create the
sl@0
   227
  ** temporary database never needs to be committed.
sl@0
   228
  */
sl@0
   229
  if( rc==SQLITE_OK ){
sl@0
   230
    u32 meta;
sl@0
   231
    int i;
sl@0
   232
sl@0
   233
    /* This array determines which meta meta values are preserved in the
sl@0
   234
    ** vacuum.  Even entries are the meta value number and odd entries
sl@0
   235
    ** are an increment to apply to the meta value after the vacuum.
sl@0
   236
    ** The increment is used to increase the schema cookie so that other
sl@0
   237
    ** connections to the same database will know to reread the schema.
sl@0
   238
    */
sl@0
   239
    static const unsigned char aCopy[] = {
sl@0
   240
       1, 1,    /* Add one to the old schema cookie */
sl@0
   241
       3, 0,    /* Preserve the default page cache size */
sl@0
   242
       5, 0,    /* Preserve the default text encoding */
sl@0
   243
       6, 0,    /* Preserve the user version */
sl@0
   244
    };
sl@0
   245
sl@0
   246
    assert( 1==sqlite3BtreeIsInTrans(pTemp) );
sl@0
   247
    assert( 1==sqlite3BtreeIsInTrans(pMain) );
sl@0
   248
sl@0
   249
    /* Copy Btree meta values */
sl@0
   250
    for(i=0; i<sizeof(aCopy)/sizeof(aCopy[0]); i+=2){
sl@0
   251
      rc = sqlite3BtreeGetMeta(pMain, aCopy[i], &meta);
sl@0
   252
      if( rc!=SQLITE_OK ) goto end_of_vacuum;
sl@0
   253
      rc = sqlite3BtreeUpdateMeta(pTemp, aCopy[i], meta+aCopy[i+1]);
sl@0
   254
      if( rc!=SQLITE_OK ) goto end_of_vacuum;
sl@0
   255
    }
sl@0
   256
sl@0
   257
    rc = sqlite3BtreeCopyFile(pMain, pTemp);
sl@0
   258
    if( rc!=SQLITE_OK ) goto end_of_vacuum;
sl@0
   259
    rc = sqlite3BtreeCommit(pTemp);
sl@0
   260
    if( rc!=SQLITE_OK ) goto end_of_vacuum;
sl@0
   261
    rc = sqlite3BtreeCommit(pMain);
sl@0
   262
  }
sl@0
   263
sl@0
   264
  if( rc==SQLITE_OK ){
sl@0
   265
    rc = sqlite3BtreeSetPageSize(pMain, sqlite3BtreeGetPageSize(pTemp), nRes);
sl@0
   266
  }
sl@0
   267
sl@0
   268
end_of_vacuum:
sl@0
   269
  /* Restore the original value of db->flags */
sl@0
   270
  db->flags = saved_flags;
sl@0
   271
  db->nChange = saved_nChange;
sl@0
   272
  db->nTotalChange = saved_nTotalChange;
sl@0
   273
sl@0
   274
  /* Currently there is an SQL level transaction open on the vacuum
sl@0
   275
  ** database. No locks are held on any other files (since the main file
sl@0
   276
  ** was committed at the btree level). So it safe to end the transaction
sl@0
   277
  ** by manually setting the autoCommit flag to true and detaching the
sl@0
   278
  ** vacuum database. The vacuum_db journal file is deleted when the pager
sl@0
   279
  ** is closed by the DETACH.
sl@0
   280
  */
sl@0
   281
  db->autoCommit = 1;
sl@0
   282
sl@0
   283
  if( pDb ){
sl@0
   284
    sqlite3BtreeClose(pDb->pBt);
sl@0
   285
    pDb->pBt = 0;
sl@0
   286
    pDb->pSchema = 0;
sl@0
   287
  }
sl@0
   288
sl@0
   289
  sqlite3ResetInternalSchema(db, 0);
sl@0
   290
sl@0
   291
  return rc;
sl@0
   292
}
sl@0
   293
#endif  /* SQLITE_OMIT_VACUUM && SQLITE_OMIT_ATTACH */