sl@0
|
1 |
/*
|
sl@0
|
2 |
** 2006 June 10
|
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 help implement virtual tables.
|
sl@0
|
13 |
**
|
sl@0
|
14 |
** $Id: vtab.c,v 1.74 2008/08/02 03:50:39 drh Exp $
|
sl@0
|
15 |
*/
|
sl@0
|
16 |
#ifndef SQLITE_OMIT_VIRTUALTABLE
|
sl@0
|
17 |
#include "sqliteInt.h"
|
sl@0
|
18 |
|
sl@0
|
19 |
static int createModule(
|
sl@0
|
20 |
sqlite3 *db, /* Database in which module is registered */
|
sl@0
|
21 |
const char *zName, /* Name assigned to this module */
|
sl@0
|
22 |
const sqlite3_module *pModule, /* The definition of the module */
|
sl@0
|
23 |
void *pAux, /* Context pointer for xCreate/xConnect */
|
sl@0
|
24 |
void (*xDestroy)(void *) /* Module destructor function */
|
sl@0
|
25 |
) {
|
sl@0
|
26 |
int rc, nName;
|
sl@0
|
27 |
Module *pMod;
|
sl@0
|
28 |
|
sl@0
|
29 |
sqlite3_mutex_enter(db->mutex);
|
sl@0
|
30 |
nName = strlen(zName);
|
sl@0
|
31 |
pMod = (Module *)sqlite3DbMallocRaw(db, sizeof(Module) + nName + 1);
|
sl@0
|
32 |
if( pMod ){
|
sl@0
|
33 |
Module *pDel;
|
sl@0
|
34 |
char *zCopy = (char *)(&pMod[1]);
|
sl@0
|
35 |
memcpy(zCopy, zName, nName+1);
|
sl@0
|
36 |
pMod->zName = zCopy;
|
sl@0
|
37 |
pMod->pModule = pModule;
|
sl@0
|
38 |
pMod->pAux = pAux;
|
sl@0
|
39 |
pMod->xDestroy = xDestroy;
|
sl@0
|
40 |
pDel = (Module *)sqlite3HashInsert(&db->aModule, zCopy, nName, (void*)pMod);
|
sl@0
|
41 |
if( pDel && pDel->xDestroy ){
|
sl@0
|
42 |
pDel->xDestroy(pDel->pAux);
|
sl@0
|
43 |
}
|
sl@0
|
44 |
sqlite3DbFree(db, pDel);
|
sl@0
|
45 |
if( pDel==pMod ){
|
sl@0
|
46 |
db->mallocFailed = 1;
|
sl@0
|
47 |
}
|
sl@0
|
48 |
sqlite3ResetInternalSchema(db, 0);
|
sl@0
|
49 |
}
|
sl@0
|
50 |
rc = sqlite3ApiExit(db, SQLITE_OK);
|
sl@0
|
51 |
sqlite3_mutex_leave(db->mutex);
|
sl@0
|
52 |
return rc;
|
sl@0
|
53 |
}
|
sl@0
|
54 |
|
sl@0
|
55 |
|
sl@0
|
56 |
/*
|
sl@0
|
57 |
** External API function used to create a new virtual-table module.
|
sl@0
|
58 |
*/
|
sl@0
|
59 |
int sqlite3_create_module(
|
sl@0
|
60 |
sqlite3 *db, /* Database in which module is registered */
|
sl@0
|
61 |
const char *zName, /* Name assigned to this module */
|
sl@0
|
62 |
const sqlite3_module *pModule, /* The definition of the module */
|
sl@0
|
63 |
void *pAux /* Context pointer for xCreate/xConnect */
|
sl@0
|
64 |
){
|
sl@0
|
65 |
return createModule(db, zName, pModule, pAux, 0);
|
sl@0
|
66 |
}
|
sl@0
|
67 |
|
sl@0
|
68 |
/*
|
sl@0
|
69 |
** External API function used to create a new virtual-table module.
|
sl@0
|
70 |
*/
|
sl@0
|
71 |
int sqlite3_create_module_v2(
|
sl@0
|
72 |
sqlite3 *db, /* Database in which module is registered */
|
sl@0
|
73 |
const char *zName, /* Name assigned to this module */
|
sl@0
|
74 |
const sqlite3_module *pModule, /* The definition of the module */
|
sl@0
|
75 |
void *pAux, /* Context pointer for xCreate/xConnect */
|
sl@0
|
76 |
void (*xDestroy)(void *) /* Module destructor function */
|
sl@0
|
77 |
){
|
sl@0
|
78 |
return createModule(db, zName, pModule, pAux, xDestroy);
|
sl@0
|
79 |
}
|
sl@0
|
80 |
|
sl@0
|
81 |
/*
|
sl@0
|
82 |
** Lock the virtual table so that it cannot be disconnected.
|
sl@0
|
83 |
** Locks nest. Every lock should have a corresponding unlock.
|
sl@0
|
84 |
** If an unlock is omitted, resources leaks will occur.
|
sl@0
|
85 |
**
|
sl@0
|
86 |
** If a disconnect is attempted while a virtual table is locked,
|
sl@0
|
87 |
** the disconnect is deferred until all locks have been removed.
|
sl@0
|
88 |
*/
|
sl@0
|
89 |
void sqlite3VtabLock(sqlite3_vtab *pVtab){
|
sl@0
|
90 |
pVtab->nRef++;
|
sl@0
|
91 |
}
|
sl@0
|
92 |
|
sl@0
|
93 |
/*
|
sl@0
|
94 |
** Unlock a virtual table. When the last lock is removed,
|
sl@0
|
95 |
** disconnect the virtual table.
|
sl@0
|
96 |
*/
|
sl@0
|
97 |
void sqlite3VtabUnlock(sqlite3 *db, sqlite3_vtab *pVtab){
|
sl@0
|
98 |
pVtab->nRef--;
|
sl@0
|
99 |
assert(db);
|
sl@0
|
100 |
assert( sqlite3SafetyCheckOk(db) );
|
sl@0
|
101 |
if( pVtab->nRef==0 ){
|
sl@0
|
102 |
if( db->magic==SQLITE_MAGIC_BUSY ){
|
sl@0
|
103 |
(void)sqlite3SafetyOff(db);
|
sl@0
|
104 |
pVtab->pModule->xDisconnect(pVtab);
|
sl@0
|
105 |
(void)sqlite3SafetyOn(db);
|
sl@0
|
106 |
} else {
|
sl@0
|
107 |
pVtab->pModule->xDisconnect(pVtab);
|
sl@0
|
108 |
}
|
sl@0
|
109 |
}
|
sl@0
|
110 |
}
|
sl@0
|
111 |
|
sl@0
|
112 |
/*
|
sl@0
|
113 |
** Clear any and all virtual-table information from the Table record.
|
sl@0
|
114 |
** This routine is called, for example, just before deleting the Table
|
sl@0
|
115 |
** record.
|
sl@0
|
116 |
*/
|
sl@0
|
117 |
void sqlite3VtabClear(Table *p){
|
sl@0
|
118 |
sqlite3_vtab *pVtab = p->pVtab;
|
sl@0
|
119 |
sqlite3 *db = p->db;
|
sl@0
|
120 |
if( pVtab ){
|
sl@0
|
121 |
assert( p->pMod && p->pMod->pModule );
|
sl@0
|
122 |
sqlite3VtabUnlock(db, pVtab);
|
sl@0
|
123 |
p->pVtab = 0;
|
sl@0
|
124 |
}
|
sl@0
|
125 |
if( p->azModuleArg ){
|
sl@0
|
126 |
int i;
|
sl@0
|
127 |
for(i=0; i<p->nModuleArg; i++){
|
sl@0
|
128 |
sqlite3DbFree(db, p->azModuleArg[i]);
|
sl@0
|
129 |
}
|
sl@0
|
130 |
sqlite3DbFree(db, p->azModuleArg);
|
sl@0
|
131 |
}
|
sl@0
|
132 |
}
|
sl@0
|
133 |
|
sl@0
|
134 |
/*
|
sl@0
|
135 |
** Add a new module argument to pTable->azModuleArg[].
|
sl@0
|
136 |
** The string is not copied - the pointer is stored. The
|
sl@0
|
137 |
** string will be freed automatically when the table is
|
sl@0
|
138 |
** deleted.
|
sl@0
|
139 |
*/
|
sl@0
|
140 |
static void addModuleArgument(sqlite3 *db, Table *pTable, char *zArg){
|
sl@0
|
141 |
int i = pTable->nModuleArg++;
|
sl@0
|
142 |
int nBytes = sizeof(char *)*(1+pTable->nModuleArg);
|
sl@0
|
143 |
char **azModuleArg;
|
sl@0
|
144 |
azModuleArg = sqlite3DbRealloc(db, pTable->azModuleArg, nBytes);
|
sl@0
|
145 |
if( azModuleArg==0 ){
|
sl@0
|
146 |
int j;
|
sl@0
|
147 |
for(j=0; j<i; j++){
|
sl@0
|
148 |
sqlite3DbFree(db, pTable->azModuleArg[j]);
|
sl@0
|
149 |
}
|
sl@0
|
150 |
sqlite3DbFree(db, zArg);
|
sl@0
|
151 |
sqlite3DbFree(db, pTable->azModuleArg);
|
sl@0
|
152 |
pTable->nModuleArg = 0;
|
sl@0
|
153 |
}else{
|
sl@0
|
154 |
azModuleArg[i] = zArg;
|
sl@0
|
155 |
azModuleArg[i+1] = 0;
|
sl@0
|
156 |
}
|
sl@0
|
157 |
pTable->azModuleArg = azModuleArg;
|
sl@0
|
158 |
}
|
sl@0
|
159 |
|
sl@0
|
160 |
/*
|
sl@0
|
161 |
** The parser calls this routine when it first sees a CREATE VIRTUAL TABLE
|
sl@0
|
162 |
** statement. The module name has been parsed, but the optional list
|
sl@0
|
163 |
** of parameters that follow the module name are still pending.
|
sl@0
|
164 |
*/
|
sl@0
|
165 |
void sqlite3VtabBeginParse(
|
sl@0
|
166 |
Parse *pParse, /* Parsing context */
|
sl@0
|
167 |
Token *pName1, /* Name of new table, or database name */
|
sl@0
|
168 |
Token *pName2, /* Name of new table or NULL */
|
sl@0
|
169 |
Token *pModuleName /* Name of the module for the virtual table */
|
sl@0
|
170 |
){
|
sl@0
|
171 |
int iDb; /* The database the table is being created in */
|
sl@0
|
172 |
Table *pTable; /* The new virtual table */
|
sl@0
|
173 |
sqlite3 *db; /* Database connection */
|
sl@0
|
174 |
|
sl@0
|
175 |
if( pParse->db->flags & SQLITE_SharedCache ){
|
sl@0
|
176 |
sqlite3ErrorMsg(pParse, "Cannot use virtual tables in shared-cache mode");
|
sl@0
|
177 |
return;
|
sl@0
|
178 |
}
|
sl@0
|
179 |
|
sl@0
|
180 |
sqlite3StartTable(pParse, pName1, pName2, 0, 0, 1, 0);
|
sl@0
|
181 |
pTable = pParse->pNewTable;
|
sl@0
|
182 |
if( pTable==0 || pParse->nErr ) return;
|
sl@0
|
183 |
assert( 0==pTable->pIndex );
|
sl@0
|
184 |
|
sl@0
|
185 |
db = pParse->db;
|
sl@0
|
186 |
iDb = sqlite3SchemaToIndex(db, pTable->pSchema);
|
sl@0
|
187 |
assert( iDb>=0 );
|
sl@0
|
188 |
|
sl@0
|
189 |
pTable->isVirtual = 1;
|
sl@0
|
190 |
pTable->nModuleArg = 0;
|
sl@0
|
191 |
addModuleArgument(db, pTable, sqlite3NameFromToken(db, pModuleName));
|
sl@0
|
192 |
addModuleArgument(db, pTable, sqlite3DbStrDup(db, db->aDb[iDb].zName));
|
sl@0
|
193 |
addModuleArgument(db, pTable, sqlite3DbStrDup(db, pTable->zName));
|
sl@0
|
194 |
pParse->sNameToken.n = pModuleName->z + pModuleName->n - pName1->z;
|
sl@0
|
195 |
|
sl@0
|
196 |
#ifndef SQLITE_OMIT_AUTHORIZATION
|
sl@0
|
197 |
/* Creating a virtual table invokes the authorization callback twice.
|
sl@0
|
198 |
** The first invocation, to obtain permission to INSERT a row into the
|
sl@0
|
199 |
** sqlite_master table, has already been made by sqlite3StartTable().
|
sl@0
|
200 |
** The second call, to obtain permission to create the table, is made now.
|
sl@0
|
201 |
*/
|
sl@0
|
202 |
if( pTable->azModuleArg ){
|
sl@0
|
203 |
sqlite3AuthCheck(pParse, SQLITE_CREATE_VTABLE, pTable->zName,
|
sl@0
|
204 |
pTable->azModuleArg[0], pParse->db->aDb[iDb].zName);
|
sl@0
|
205 |
}
|
sl@0
|
206 |
#endif
|
sl@0
|
207 |
}
|
sl@0
|
208 |
|
sl@0
|
209 |
/*
|
sl@0
|
210 |
** This routine takes the module argument that has been accumulating
|
sl@0
|
211 |
** in pParse->zArg[] and appends it to the list of arguments on the
|
sl@0
|
212 |
** virtual table currently under construction in pParse->pTable.
|
sl@0
|
213 |
*/
|
sl@0
|
214 |
static void addArgumentToVtab(Parse *pParse){
|
sl@0
|
215 |
if( pParse->sArg.z && pParse->pNewTable ){
|
sl@0
|
216 |
const char *z = (const char*)pParse->sArg.z;
|
sl@0
|
217 |
int n = pParse->sArg.n;
|
sl@0
|
218 |
sqlite3 *db = pParse->db;
|
sl@0
|
219 |
addModuleArgument(db, pParse->pNewTable, sqlite3DbStrNDup(db, z, n));
|
sl@0
|
220 |
}
|
sl@0
|
221 |
}
|
sl@0
|
222 |
|
sl@0
|
223 |
/*
|
sl@0
|
224 |
** The parser calls this routine after the CREATE VIRTUAL TABLE statement
|
sl@0
|
225 |
** has been completely parsed.
|
sl@0
|
226 |
*/
|
sl@0
|
227 |
void sqlite3VtabFinishParse(Parse *pParse, Token *pEnd){
|
sl@0
|
228 |
Table *pTab; /* The table being constructed */
|
sl@0
|
229 |
sqlite3 *db; /* The database connection */
|
sl@0
|
230 |
char *zModule; /* The module name of the table: USING modulename */
|
sl@0
|
231 |
Module *pMod = 0;
|
sl@0
|
232 |
|
sl@0
|
233 |
addArgumentToVtab(pParse);
|
sl@0
|
234 |
pParse->sArg.z = 0;
|
sl@0
|
235 |
|
sl@0
|
236 |
/* Lookup the module name. */
|
sl@0
|
237 |
pTab = pParse->pNewTable;
|
sl@0
|
238 |
if( pTab==0 ) return;
|
sl@0
|
239 |
db = pParse->db;
|
sl@0
|
240 |
if( pTab->nModuleArg<1 ) return;
|
sl@0
|
241 |
zModule = pTab->azModuleArg[0];
|
sl@0
|
242 |
pMod = (Module *)sqlite3HashFind(&db->aModule, zModule, strlen(zModule));
|
sl@0
|
243 |
pTab->pMod = pMod;
|
sl@0
|
244 |
|
sl@0
|
245 |
/* If the CREATE VIRTUAL TABLE statement is being entered for the
|
sl@0
|
246 |
** first time (in other words if the virtual table is actually being
|
sl@0
|
247 |
** created now instead of just being read out of sqlite_master) then
|
sl@0
|
248 |
** do additional initialization work and store the statement text
|
sl@0
|
249 |
** in the sqlite_master table.
|
sl@0
|
250 |
*/
|
sl@0
|
251 |
if( !db->init.busy ){
|
sl@0
|
252 |
char *zStmt;
|
sl@0
|
253 |
char *zWhere;
|
sl@0
|
254 |
int iDb;
|
sl@0
|
255 |
Vdbe *v;
|
sl@0
|
256 |
|
sl@0
|
257 |
/* Compute the complete text of the CREATE VIRTUAL TABLE statement */
|
sl@0
|
258 |
if( pEnd ){
|
sl@0
|
259 |
pParse->sNameToken.n = pEnd->z - pParse->sNameToken.z + pEnd->n;
|
sl@0
|
260 |
}
|
sl@0
|
261 |
zStmt = sqlite3MPrintf(db, "CREATE VIRTUAL TABLE %T", &pParse->sNameToken);
|
sl@0
|
262 |
|
sl@0
|
263 |
/* A slot for the record has already been allocated in the
|
sl@0
|
264 |
** SQLITE_MASTER table. We just need to update that slot with all
|
sl@0
|
265 |
** the information we've collected.
|
sl@0
|
266 |
**
|
sl@0
|
267 |
** The VM register number pParse->regRowid holds the rowid of an
|
sl@0
|
268 |
** entry in the sqlite_master table tht was created for this vtab
|
sl@0
|
269 |
** by sqlite3StartTable().
|
sl@0
|
270 |
*/
|
sl@0
|
271 |
iDb = sqlite3SchemaToIndex(db, pTab->pSchema);
|
sl@0
|
272 |
sqlite3NestedParse(pParse,
|
sl@0
|
273 |
"UPDATE %Q.%s "
|
sl@0
|
274 |
"SET type='table', name=%Q, tbl_name=%Q, rootpage=0, sql=%Q "
|
sl@0
|
275 |
"WHERE rowid=#%d",
|
sl@0
|
276 |
db->aDb[iDb].zName, SCHEMA_TABLE(iDb),
|
sl@0
|
277 |
pTab->zName,
|
sl@0
|
278 |
pTab->zName,
|
sl@0
|
279 |
zStmt,
|
sl@0
|
280 |
pParse->regRowid
|
sl@0
|
281 |
);
|
sl@0
|
282 |
sqlite3DbFree(db, zStmt);
|
sl@0
|
283 |
v = sqlite3GetVdbe(pParse);
|
sl@0
|
284 |
sqlite3ChangeCookie(pParse, iDb);
|
sl@0
|
285 |
|
sl@0
|
286 |
sqlite3VdbeAddOp2(v, OP_Expire, 0, 0);
|
sl@0
|
287 |
zWhere = sqlite3MPrintf(db, "name='%q'", pTab->zName);
|
sl@0
|
288 |
sqlite3VdbeAddOp4(v, OP_ParseSchema, iDb, 1, 0, zWhere, P4_DYNAMIC);
|
sl@0
|
289 |
sqlite3VdbeAddOp4(v, OP_VCreate, iDb, 0, 0,
|
sl@0
|
290 |
pTab->zName, strlen(pTab->zName) + 1);
|
sl@0
|
291 |
}
|
sl@0
|
292 |
|
sl@0
|
293 |
/* If we are rereading the sqlite_master table create the in-memory
|
sl@0
|
294 |
** record of the table. If the module has already been registered,
|
sl@0
|
295 |
** also call the xConnect method here.
|
sl@0
|
296 |
*/
|
sl@0
|
297 |
else {
|
sl@0
|
298 |
Table *pOld;
|
sl@0
|
299 |
Schema *pSchema = pTab->pSchema;
|
sl@0
|
300 |
const char *zName = pTab->zName;
|
sl@0
|
301 |
int nName = strlen(zName) + 1;
|
sl@0
|
302 |
pOld = sqlite3HashInsert(&pSchema->tblHash, zName, nName, pTab);
|
sl@0
|
303 |
if( pOld ){
|
sl@0
|
304 |
db->mallocFailed = 1;
|
sl@0
|
305 |
assert( pTab==pOld ); /* Malloc must have failed inside HashInsert() */
|
sl@0
|
306 |
return;
|
sl@0
|
307 |
}
|
sl@0
|
308 |
pSchema->db = pParse->db;
|
sl@0
|
309 |
pParse->pNewTable = 0;
|
sl@0
|
310 |
}
|
sl@0
|
311 |
}
|
sl@0
|
312 |
|
sl@0
|
313 |
/*
|
sl@0
|
314 |
** The parser calls this routine when it sees the first token
|
sl@0
|
315 |
** of an argument to the module name in a CREATE VIRTUAL TABLE statement.
|
sl@0
|
316 |
*/
|
sl@0
|
317 |
void sqlite3VtabArgInit(Parse *pParse){
|
sl@0
|
318 |
addArgumentToVtab(pParse);
|
sl@0
|
319 |
pParse->sArg.z = 0;
|
sl@0
|
320 |
pParse->sArg.n = 0;
|
sl@0
|
321 |
}
|
sl@0
|
322 |
|
sl@0
|
323 |
/*
|
sl@0
|
324 |
** The parser calls this routine for each token after the first token
|
sl@0
|
325 |
** in an argument to the module name in a CREATE VIRTUAL TABLE statement.
|
sl@0
|
326 |
*/
|
sl@0
|
327 |
void sqlite3VtabArgExtend(Parse *pParse, Token *p){
|
sl@0
|
328 |
Token *pArg = &pParse->sArg;
|
sl@0
|
329 |
if( pArg->z==0 ){
|
sl@0
|
330 |
pArg->z = p->z;
|
sl@0
|
331 |
pArg->n = p->n;
|
sl@0
|
332 |
}else{
|
sl@0
|
333 |
assert(pArg->z < p->z);
|
sl@0
|
334 |
pArg->n = (p->z + p->n - pArg->z);
|
sl@0
|
335 |
}
|
sl@0
|
336 |
}
|
sl@0
|
337 |
|
sl@0
|
338 |
/*
|
sl@0
|
339 |
** Invoke a virtual table constructor (either xCreate or xConnect). The
|
sl@0
|
340 |
** pointer to the function to invoke is passed as the fourth parameter
|
sl@0
|
341 |
** to this procedure.
|
sl@0
|
342 |
*/
|
sl@0
|
343 |
static int vtabCallConstructor(
|
sl@0
|
344 |
sqlite3 *db,
|
sl@0
|
345 |
Table *pTab,
|
sl@0
|
346 |
Module *pMod,
|
sl@0
|
347 |
int (*xConstruct)(sqlite3*,void*,int,const char*const*,sqlite3_vtab**,char**),
|
sl@0
|
348 |
char **pzErr
|
sl@0
|
349 |
){
|
sl@0
|
350 |
int rc;
|
sl@0
|
351 |
int rc2;
|
sl@0
|
352 |
sqlite3_vtab *pVtab = 0;
|
sl@0
|
353 |
const char *const*azArg = (const char *const*)pTab->azModuleArg;
|
sl@0
|
354 |
int nArg = pTab->nModuleArg;
|
sl@0
|
355 |
char *zErr = 0;
|
sl@0
|
356 |
char *zModuleName = sqlite3MPrintf(db, "%s", pTab->zName);
|
sl@0
|
357 |
|
sl@0
|
358 |
if( !zModuleName ){
|
sl@0
|
359 |
return SQLITE_NOMEM;
|
sl@0
|
360 |
}
|
sl@0
|
361 |
|
sl@0
|
362 |
assert( !db->pVTab );
|
sl@0
|
363 |
assert( xConstruct );
|
sl@0
|
364 |
|
sl@0
|
365 |
db->pVTab = pTab;
|
sl@0
|
366 |
rc = sqlite3SafetyOff(db);
|
sl@0
|
367 |
assert( rc==SQLITE_OK );
|
sl@0
|
368 |
rc = xConstruct(db, pMod->pAux, nArg, azArg, &pVtab, &zErr);
|
sl@0
|
369 |
rc2 = sqlite3SafetyOn(db);
|
sl@0
|
370 |
if( rc==SQLITE_OK && pVtab ){
|
sl@0
|
371 |
pVtab->pModule = pMod->pModule;
|
sl@0
|
372 |
pVtab->nRef = 1;
|
sl@0
|
373 |
pTab->pVtab = pVtab;
|
sl@0
|
374 |
}
|
sl@0
|
375 |
|
sl@0
|
376 |
if( SQLITE_OK!=rc ){
|
sl@0
|
377 |
if( zErr==0 ){
|
sl@0
|
378 |
*pzErr = sqlite3MPrintf(db, "vtable constructor failed: %s", zModuleName);
|
sl@0
|
379 |
}else {
|
sl@0
|
380 |
*pzErr = sqlite3MPrintf(db, "%s", zErr);
|
sl@0
|
381 |
sqlite3DbFree(db, zErr);
|
sl@0
|
382 |
}
|
sl@0
|
383 |
}else if( db->pVTab ){
|
sl@0
|
384 |
const char *zFormat = "vtable constructor did not declare schema: %s";
|
sl@0
|
385 |
*pzErr = sqlite3MPrintf(db, zFormat, pTab->zName);
|
sl@0
|
386 |
rc = SQLITE_ERROR;
|
sl@0
|
387 |
}
|
sl@0
|
388 |
if( rc==SQLITE_OK ){
|
sl@0
|
389 |
rc = rc2;
|
sl@0
|
390 |
}
|
sl@0
|
391 |
db->pVTab = 0;
|
sl@0
|
392 |
sqlite3DbFree(db, zModuleName);
|
sl@0
|
393 |
|
sl@0
|
394 |
/* If everything went according to plan, loop through the columns
|
sl@0
|
395 |
** of the table to see if any of them contain the token "hidden".
|
sl@0
|
396 |
** If so, set the Column.isHidden flag and remove the token from
|
sl@0
|
397 |
** the type string.
|
sl@0
|
398 |
*/
|
sl@0
|
399 |
if( rc==SQLITE_OK ){
|
sl@0
|
400 |
int iCol;
|
sl@0
|
401 |
for(iCol=0; iCol<pTab->nCol; iCol++){
|
sl@0
|
402 |
char *zType = pTab->aCol[iCol].zType;
|
sl@0
|
403 |
int nType;
|
sl@0
|
404 |
int i = 0;
|
sl@0
|
405 |
if( !zType ) continue;
|
sl@0
|
406 |
nType = strlen(zType);
|
sl@0
|
407 |
if( sqlite3StrNICmp("hidden", zType, 6) || (zType[6] && zType[6]!=' ') ){
|
sl@0
|
408 |
for(i=0; i<nType; i++){
|
sl@0
|
409 |
if( (0==sqlite3StrNICmp(" hidden", &zType[i], 7))
|
sl@0
|
410 |
&& (zType[i+7]=='\0' || zType[i+7]==' ')
|
sl@0
|
411 |
){
|
sl@0
|
412 |
i++;
|
sl@0
|
413 |
break;
|
sl@0
|
414 |
}
|
sl@0
|
415 |
}
|
sl@0
|
416 |
}
|
sl@0
|
417 |
if( i<nType ){
|
sl@0
|
418 |
int j;
|
sl@0
|
419 |
int nDel = 6 + (zType[i+6] ? 1 : 0);
|
sl@0
|
420 |
for(j=i; (j+nDel)<=nType; j++){
|
sl@0
|
421 |
zType[j] = zType[j+nDel];
|
sl@0
|
422 |
}
|
sl@0
|
423 |
if( zType[i]=='\0' && i>0 ){
|
sl@0
|
424 |
assert(zType[i-1]==' ');
|
sl@0
|
425 |
zType[i-1] = '\0';
|
sl@0
|
426 |
}
|
sl@0
|
427 |
pTab->aCol[iCol].isHidden = 1;
|
sl@0
|
428 |
}
|
sl@0
|
429 |
}
|
sl@0
|
430 |
}
|
sl@0
|
431 |
return rc;
|
sl@0
|
432 |
}
|
sl@0
|
433 |
|
sl@0
|
434 |
/*
|
sl@0
|
435 |
** This function is invoked by the parser to call the xConnect() method
|
sl@0
|
436 |
** of the virtual table pTab. If an error occurs, an error code is returned
|
sl@0
|
437 |
** and an error left in pParse.
|
sl@0
|
438 |
**
|
sl@0
|
439 |
** This call is a no-op if table pTab is not a virtual table.
|
sl@0
|
440 |
*/
|
sl@0
|
441 |
int sqlite3VtabCallConnect(Parse *pParse, Table *pTab){
|
sl@0
|
442 |
Module *pMod;
|
sl@0
|
443 |
int rc = SQLITE_OK;
|
sl@0
|
444 |
|
sl@0
|
445 |
if( !pTab || !pTab->isVirtual || pTab->pVtab ){
|
sl@0
|
446 |
return SQLITE_OK;
|
sl@0
|
447 |
}
|
sl@0
|
448 |
|
sl@0
|
449 |
pMod = pTab->pMod;
|
sl@0
|
450 |
if( !pMod ){
|
sl@0
|
451 |
const char *zModule = pTab->azModuleArg[0];
|
sl@0
|
452 |
sqlite3ErrorMsg(pParse, "no such module: %s", zModule);
|
sl@0
|
453 |
rc = SQLITE_ERROR;
|
sl@0
|
454 |
} else {
|
sl@0
|
455 |
char *zErr = 0;
|
sl@0
|
456 |
sqlite3 *db = pParse->db;
|
sl@0
|
457 |
rc = vtabCallConstructor(db, pTab, pMod, pMod->pModule->xConnect, &zErr);
|
sl@0
|
458 |
if( rc!=SQLITE_OK ){
|
sl@0
|
459 |
sqlite3ErrorMsg(pParse, "%s", zErr);
|
sl@0
|
460 |
}
|
sl@0
|
461 |
sqlite3DbFree(db, zErr);
|
sl@0
|
462 |
}
|
sl@0
|
463 |
|
sl@0
|
464 |
return rc;
|
sl@0
|
465 |
}
|
sl@0
|
466 |
|
sl@0
|
467 |
/*
|
sl@0
|
468 |
** Add the virtual table pVtab to the array sqlite3.aVTrans[].
|
sl@0
|
469 |
*/
|
sl@0
|
470 |
static int addToVTrans(sqlite3 *db, sqlite3_vtab *pVtab){
|
sl@0
|
471 |
const int ARRAY_INCR = 5;
|
sl@0
|
472 |
|
sl@0
|
473 |
/* Grow the sqlite3.aVTrans array if required */
|
sl@0
|
474 |
if( (db->nVTrans%ARRAY_INCR)==0 ){
|
sl@0
|
475 |
sqlite3_vtab **aVTrans;
|
sl@0
|
476 |
int nBytes = sizeof(sqlite3_vtab *) * (db->nVTrans + ARRAY_INCR);
|
sl@0
|
477 |
aVTrans = sqlite3DbRealloc(db, (void *)db->aVTrans, nBytes);
|
sl@0
|
478 |
if( !aVTrans ){
|
sl@0
|
479 |
return SQLITE_NOMEM;
|
sl@0
|
480 |
}
|
sl@0
|
481 |
memset(&aVTrans[db->nVTrans], 0, sizeof(sqlite3_vtab *)*ARRAY_INCR);
|
sl@0
|
482 |
db->aVTrans = aVTrans;
|
sl@0
|
483 |
}
|
sl@0
|
484 |
|
sl@0
|
485 |
/* Add pVtab to the end of sqlite3.aVTrans */
|
sl@0
|
486 |
db->aVTrans[db->nVTrans++] = pVtab;
|
sl@0
|
487 |
sqlite3VtabLock(pVtab);
|
sl@0
|
488 |
return SQLITE_OK;
|
sl@0
|
489 |
}
|
sl@0
|
490 |
|
sl@0
|
491 |
/*
|
sl@0
|
492 |
** This function is invoked by the vdbe to call the xCreate method
|
sl@0
|
493 |
** of the virtual table named zTab in database iDb.
|
sl@0
|
494 |
**
|
sl@0
|
495 |
** If an error occurs, *pzErr is set to point an an English language
|
sl@0
|
496 |
** description of the error and an SQLITE_XXX error code is returned.
|
sl@0
|
497 |
** In this case the caller must call sqlite3DbFree(db, ) on *pzErr.
|
sl@0
|
498 |
*/
|
sl@0
|
499 |
int sqlite3VtabCallCreate(sqlite3 *db, int iDb, const char *zTab, char **pzErr){
|
sl@0
|
500 |
int rc = SQLITE_OK;
|
sl@0
|
501 |
Table *pTab;
|
sl@0
|
502 |
Module *pMod;
|
sl@0
|
503 |
const char *zModule;
|
sl@0
|
504 |
|
sl@0
|
505 |
pTab = sqlite3FindTable(db, zTab, db->aDb[iDb].zName);
|
sl@0
|
506 |
assert(pTab && pTab->isVirtual && !pTab->pVtab);
|
sl@0
|
507 |
pMod = pTab->pMod;
|
sl@0
|
508 |
zModule = pTab->azModuleArg[0];
|
sl@0
|
509 |
|
sl@0
|
510 |
/* If the module has been registered and includes a Create method,
|
sl@0
|
511 |
** invoke it now. If the module has not been registered, return an
|
sl@0
|
512 |
** error. Otherwise, do nothing.
|
sl@0
|
513 |
*/
|
sl@0
|
514 |
if( !pMod ){
|
sl@0
|
515 |
*pzErr = sqlite3MPrintf(db, "no such module: %s", zModule);
|
sl@0
|
516 |
rc = SQLITE_ERROR;
|
sl@0
|
517 |
}else{
|
sl@0
|
518 |
rc = vtabCallConstructor(db, pTab, pMod, pMod->pModule->xCreate, pzErr);
|
sl@0
|
519 |
}
|
sl@0
|
520 |
|
sl@0
|
521 |
if( rc==SQLITE_OK && pTab->pVtab ){
|
sl@0
|
522 |
rc = addToVTrans(db, pTab->pVtab);
|
sl@0
|
523 |
}
|
sl@0
|
524 |
|
sl@0
|
525 |
return rc;
|
sl@0
|
526 |
}
|
sl@0
|
527 |
|
sl@0
|
528 |
/*
|
sl@0
|
529 |
** This function is used to set the schema of a virtual table. It is only
|
sl@0
|
530 |
** valid to call this function from within the xCreate() or xConnect() of a
|
sl@0
|
531 |
** virtual table module.
|
sl@0
|
532 |
*/
|
sl@0
|
533 |
int sqlite3_declare_vtab(sqlite3 *db, const char *zCreateTable){
|
sl@0
|
534 |
Parse sParse;
|
sl@0
|
535 |
|
sl@0
|
536 |
int rc = SQLITE_OK;
|
sl@0
|
537 |
Table *pTab;
|
sl@0
|
538 |
char *zErr = 0;
|
sl@0
|
539 |
|
sl@0
|
540 |
sqlite3_mutex_enter(db->mutex);
|
sl@0
|
541 |
pTab = db->pVTab;
|
sl@0
|
542 |
if( !pTab ){
|
sl@0
|
543 |
sqlite3Error(db, SQLITE_MISUSE, 0);
|
sl@0
|
544 |
sqlite3_mutex_leave(db->mutex);
|
sl@0
|
545 |
return SQLITE_MISUSE;
|
sl@0
|
546 |
}
|
sl@0
|
547 |
assert(pTab->isVirtual && pTab->nCol==0 && pTab->aCol==0);
|
sl@0
|
548 |
|
sl@0
|
549 |
memset(&sParse, 0, sizeof(Parse));
|
sl@0
|
550 |
sParse.declareVtab = 1;
|
sl@0
|
551 |
sParse.db = db;
|
sl@0
|
552 |
|
sl@0
|
553 |
if(
|
sl@0
|
554 |
SQLITE_OK == sqlite3RunParser(&sParse, zCreateTable, &zErr) &&
|
sl@0
|
555 |
sParse.pNewTable &&
|
sl@0
|
556 |
!sParse.pNewTable->pSelect &&
|
sl@0
|
557 |
!sParse.pNewTable->isVirtual
|
sl@0
|
558 |
){
|
sl@0
|
559 |
pTab->aCol = sParse.pNewTable->aCol;
|
sl@0
|
560 |
pTab->nCol = sParse.pNewTable->nCol;
|
sl@0
|
561 |
sParse.pNewTable->nCol = 0;
|
sl@0
|
562 |
sParse.pNewTable->aCol = 0;
|
sl@0
|
563 |
db->pVTab = 0;
|
sl@0
|
564 |
} else {
|
sl@0
|
565 |
sqlite3Error(db, SQLITE_ERROR, zErr);
|
sl@0
|
566 |
sqlite3DbFree(db, zErr);
|
sl@0
|
567 |
rc = SQLITE_ERROR;
|
sl@0
|
568 |
}
|
sl@0
|
569 |
sParse.declareVtab = 0;
|
sl@0
|
570 |
|
sl@0
|
571 |
sqlite3_finalize((sqlite3_stmt*)sParse.pVdbe);
|
sl@0
|
572 |
sqlite3DeleteTable(sParse.pNewTable);
|
sl@0
|
573 |
sParse.pNewTable = 0;
|
sl@0
|
574 |
|
sl@0
|
575 |
assert( (rc&0xff)==rc );
|
sl@0
|
576 |
rc = sqlite3ApiExit(db, rc);
|
sl@0
|
577 |
sqlite3_mutex_leave(db->mutex);
|
sl@0
|
578 |
return rc;
|
sl@0
|
579 |
}
|
sl@0
|
580 |
|
sl@0
|
581 |
/*
|
sl@0
|
582 |
** This function is invoked by the vdbe to call the xDestroy method
|
sl@0
|
583 |
** of the virtual table named zTab in database iDb. This occurs
|
sl@0
|
584 |
** when a DROP TABLE is mentioned.
|
sl@0
|
585 |
**
|
sl@0
|
586 |
** This call is a no-op if zTab is not a virtual table.
|
sl@0
|
587 |
*/
|
sl@0
|
588 |
int sqlite3VtabCallDestroy(sqlite3 *db, int iDb, const char *zTab)
|
sl@0
|
589 |
{
|
sl@0
|
590 |
int rc = SQLITE_OK;
|
sl@0
|
591 |
Table *pTab;
|
sl@0
|
592 |
|
sl@0
|
593 |
pTab = sqlite3FindTable(db, zTab, db->aDb[iDb].zName);
|
sl@0
|
594 |
assert(pTab);
|
sl@0
|
595 |
if( pTab->pVtab ){
|
sl@0
|
596 |
int (*xDestroy)(sqlite3_vtab *pVTab) = pTab->pMod->pModule->xDestroy;
|
sl@0
|
597 |
rc = sqlite3SafetyOff(db);
|
sl@0
|
598 |
assert( rc==SQLITE_OK );
|
sl@0
|
599 |
if( xDestroy ){
|
sl@0
|
600 |
rc = xDestroy(pTab->pVtab);
|
sl@0
|
601 |
}
|
sl@0
|
602 |
(void)sqlite3SafetyOn(db);
|
sl@0
|
603 |
if( rc==SQLITE_OK ){
|
sl@0
|
604 |
int i;
|
sl@0
|
605 |
for(i=0; i<db->nVTrans; i++){
|
sl@0
|
606 |
if( db->aVTrans[i]==pTab->pVtab ){
|
sl@0
|
607 |
db->aVTrans[i] = db->aVTrans[--db->nVTrans];
|
sl@0
|
608 |
break;
|
sl@0
|
609 |
}
|
sl@0
|
610 |
}
|
sl@0
|
611 |
pTab->pVtab = 0;
|
sl@0
|
612 |
}
|
sl@0
|
613 |
}
|
sl@0
|
614 |
|
sl@0
|
615 |
return rc;
|
sl@0
|
616 |
}
|
sl@0
|
617 |
|
sl@0
|
618 |
/*
|
sl@0
|
619 |
** This function invokes either the xRollback or xCommit method
|
sl@0
|
620 |
** of each of the virtual tables in the sqlite3.aVTrans array. The method
|
sl@0
|
621 |
** called is identified by the second argument, "offset", which is
|
sl@0
|
622 |
** the offset of the method to call in the sqlite3_module structure.
|
sl@0
|
623 |
**
|
sl@0
|
624 |
** The array is cleared after invoking the callbacks.
|
sl@0
|
625 |
*/
|
sl@0
|
626 |
static void callFinaliser(sqlite3 *db, int offset){
|
sl@0
|
627 |
int i;
|
sl@0
|
628 |
if( db->aVTrans ){
|
sl@0
|
629 |
for(i=0; i<db->nVTrans && db->aVTrans[i]; i++){
|
sl@0
|
630 |
sqlite3_vtab *pVtab = db->aVTrans[i];
|
sl@0
|
631 |
int (*x)(sqlite3_vtab *);
|
sl@0
|
632 |
x = *(int (**)(sqlite3_vtab *))((char *)pVtab->pModule + offset);
|
sl@0
|
633 |
if( x ) x(pVtab);
|
sl@0
|
634 |
sqlite3VtabUnlock(db, pVtab);
|
sl@0
|
635 |
}
|
sl@0
|
636 |
sqlite3DbFree(db, db->aVTrans);
|
sl@0
|
637 |
db->nVTrans = 0;
|
sl@0
|
638 |
db->aVTrans = 0;
|
sl@0
|
639 |
}
|
sl@0
|
640 |
}
|
sl@0
|
641 |
|
sl@0
|
642 |
/*
|
sl@0
|
643 |
** Invoke the xSync method of all virtual tables in the sqlite3.aVTrans
|
sl@0
|
644 |
** array. Return the error code for the first error that occurs, or
|
sl@0
|
645 |
** SQLITE_OK if all xSync operations are successful.
|
sl@0
|
646 |
**
|
sl@0
|
647 |
** Set *pzErrmsg to point to a buffer that should be released using
|
sl@0
|
648 |
** sqlite3DbFree() containing an error message, if one is available.
|
sl@0
|
649 |
*/
|
sl@0
|
650 |
int sqlite3VtabSync(sqlite3 *db, char **pzErrmsg){
|
sl@0
|
651 |
int i;
|
sl@0
|
652 |
int rc = SQLITE_OK;
|
sl@0
|
653 |
int rcsafety;
|
sl@0
|
654 |
sqlite3_vtab **aVTrans = db->aVTrans;
|
sl@0
|
655 |
|
sl@0
|
656 |
rc = sqlite3SafetyOff(db);
|
sl@0
|
657 |
db->aVTrans = 0;
|
sl@0
|
658 |
for(i=0; rc==SQLITE_OK && i<db->nVTrans && aVTrans[i]; i++){
|
sl@0
|
659 |
sqlite3_vtab *pVtab = aVTrans[i];
|
sl@0
|
660 |
int (*x)(sqlite3_vtab *);
|
sl@0
|
661 |
x = pVtab->pModule->xSync;
|
sl@0
|
662 |
if( x ){
|
sl@0
|
663 |
rc = x(pVtab);
|
sl@0
|
664 |
sqlite3DbFree(db, *pzErrmsg);
|
sl@0
|
665 |
*pzErrmsg = pVtab->zErrMsg;
|
sl@0
|
666 |
pVtab->zErrMsg = 0;
|
sl@0
|
667 |
}
|
sl@0
|
668 |
}
|
sl@0
|
669 |
db->aVTrans = aVTrans;
|
sl@0
|
670 |
rcsafety = sqlite3SafetyOn(db);
|
sl@0
|
671 |
|
sl@0
|
672 |
if( rc==SQLITE_OK ){
|
sl@0
|
673 |
rc = rcsafety;
|
sl@0
|
674 |
}
|
sl@0
|
675 |
return rc;
|
sl@0
|
676 |
}
|
sl@0
|
677 |
|
sl@0
|
678 |
/*
|
sl@0
|
679 |
** Invoke the xRollback method of all virtual tables in the
|
sl@0
|
680 |
** sqlite3.aVTrans array. Then clear the array itself.
|
sl@0
|
681 |
*/
|
sl@0
|
682 |
int sqlite3VtabRollback(sqlite3 *db){
|
sl@0
|
683 |
callFinaliser(db, offsetof(sqlite3_module,xRollback));
|
sl@0
|
684 |
return SQLITE_OK;
|
sl@0
|
685 |
}
|
sl@0
|
686 |
|
sl@0
|
687 |
/*
|
sl@0
|
688 |
** Invoke the xCommit method of all virtual tables in the
|
sl@0
|
689 |
** sqlite3.aVTrans array. Then clear the array itself.
|
sl@0
|
690 |
*/
|
sl@0
|
691 |
int sqlite3VtabCommit(sqlite3 *db){
|
sl@0
|
692 |
callFinaliser(db, offsetof(sqlite3_module,xCommit));
|
sl@0
|
693 |
return SQLITE_OK;
|
sl@0
|
694 |
}
|
sl@0
|
695 |
|
sl@0
|
696 |
/*
|
sl@0
|
697 |
** If the virtual table pVtab supports the transaction interface
|
sl@0
|
698 |
** (xBegin/xRollback/xCommit and optionally xSync) and a transaction is
|
sl@0
|
699 |
** not currently open, invoke the xBegin method now.
|
sl@0
|
700 |
**
|
sl@0
|
701 |
** If the xBegin call is successful, place the sqlite3_vtab pointer
|
sl@0
|
702 |
** in the sqlite3.aVTrans array.
|
sl@0
|
703 |
*/
|
sl@0
|
704 |
int sqlite3VtabBegin(sqlite3 *db, sqlite3_vtab *pVtab){
|
sl@0
|
705 |
int rc = SQLITE_OK;
|
sl@0
|
706 |
const sqlite3_module *pModule;
|
sl@0
|
707 |
|
sl@0
|
708 |
/* Special case: If db->aVTrans is NULL and db->nVTrans is greater
|
sl@0
|
709 |
** than zero, then this function is being called from within a
|
sl@0
|
710 |
** virtual module xSync() callback. It is illegal to write to
|
sl@0
|
711 |
** virtual module tables in this case, so return SQLITE_LOCKED.
|
sl@0
|
712 |
*/
|
sl@0
|
713 |
if( 0==db->aVTrans && db->nVTrans>0 ){
|
sl@0
|
714 |
return SQLITE_LOCKED;
|
sl@0
|
715 |
}
|
sl@0
|
716 |
if( !pVtab ){
|
sl@0
|
717 |
return SQLITE_OK;
|
sl@0
|
718 |
}
|
sl@0
|
719 |
pModule = pVtab->pModule;
|
sl@0
|
720 |
|
sl@0
|
721 |
if( pModule->xBegin ){
|
sl@0
|
722 |
int i;
|
sl@0
|
723 |
|
sl@0
|
724 |
|
sl@0
|
725 |
/* If pVtab is already in the aVTrans array, return early */
|
sl@0
|
726 |
for(i=0; (i<db->nVTrans) && 0!=db->aVTrans[i]; i++){
|
sl@0
|
727 |
if( db->aVTrans[i]==pVtab ){
|
sl@0
|
728 |
return SQLITE_OK;
|
sl@0
|
729 |
}
|
sl@0
|
730 |
}
|
sl@0
|
731 |
|
sl@0
|
732 |
/* Invoke the xBegin method */
|
sl@0
|
733 |
rc = pModule->xBegin(pVtab);
|
sl@0
|
734 |
if( rc==SQLITE_OK ){
|
sl@0
|
735 |
rc = addToVTrans(db, pVtab);
|
sl@0
|
736 |
}
|
sl@0
|
737 |
}
|
sl@0
|
738 |
return rc;
|
sl@0
|
739 |
}
|
sl@0
|
740 |
|
sl@0
|
741 |
/*
|
sl@0
|
742 |
** The first parameter (pDef) is a function implementation. The
|
sl@0
|
743 |
** second parameter (pExpr) is the first argument to this function.
|
sl@0
|
744 |
** If pExpr is a column in a virtual table, then let the virtual
|
sl@0
|
745 |
** table implementation have an opportunity to overload the function.
|
sl@0
|
746 |
**
|
sl@0
|
747 |
** This routine is used to allow virtual table implementations to
|
sl@0
|
748 |
** overload MATCH, LIKE, GLOB, and REGEXP operators.
|
sl@0
|
749 |
**
|
sl@0
|
750 |
** Return either the pDef argument (indicating no change) or a
|
sl@0
|
751 |
** new FuncDef structure that is marked as ephemeral using the
|
sl@0
|
752 |
** SQLITE_FUNC_EPHEM flag.
|
sl@0
|
753 |
*/
|
sl@0
|
754 |
FuncDef *sqlite3VtabOverloadFunction(
|
sl@0
|
755 |
sqlite3 *db, /* Database connection for reporting malloc problems */
|
sl@0
|
756 |
FuncDef *pDef, /* Function to possibly overload */
|
sl@0
|
757 |
int nArg, /* Number of arguments to the function */
|
sl@0
|
758 |
Expr *pExpr /* First argument to the function */
|
sl@0
|
759 |
){
|
sl@0
|
760 |
Table *pTab;
|
sl@0
|
761 |
sqlite3_vtab *pVtab;
|
sl@0
|
762 |
sqlite3_module *pMod;
|
sl@0
|
763 |
void (*xFunc)(sqlite3_context*,int,sqlite3_value**);
|
sl@0
|
764 |
void *pArg;
|
sl@0
|
765 |
FuncDef *pNew;
|
sl@0
|
766 |
int rc = 0;
|
sl@0
|
767 |
char *zLowerName;
|
sl@0
|
768 |
unsigned char *z;
|
sl@0
|
769 |
|
sl@0
|
770 |
|
sl@0
|
771 |
/* Check to see the left operand is a column in a virtual table */
|
sl@0
|
772 |
if( pExpr==0 ) return pDef;
|
sl@0
|
773 |
if( pExpr->op!=TK_COLUMN ) return pDef;
|
sl@0
|
774 |
pTab = pExpr->pTab;
|
sl@0
|
775 |
if( pTab==0 ) return pDef;
|
sl@0
|
776 |
if( !pTab->isVirtual ) return pDef;
|
sl@0
|
777 |
pVtab = pTab->pVtab;
|
sl@0
|
778 |
assert( pVtab!=0 );
|
sl@0
|
779 |
assert( pVtab->pModule!=0 );
|
sl@0
|
780 |
pMod = (sqlite3_module *)pVtab->pModule;
|
sl@0
|
781 |
if( pMod->xFindFunction==0 ) return pDef;
|
sl@0
|
782 |
|
sl@0
|
783 |
/* Call the xFindFunction method on the virtual table implementation
|
sl@0
|
784 |
** to see if the implementation wants to overload this function
|
sl@0
|
785 |
*/
|
sl@0
|
786 |
zLowerName = sqlite3DbStrDup(db, pDef->zName);
|
sl@0
|
787 |
if( zLowerName ){
|
sl@0
|
788 |
for(z=(unsigned char*)zLowerName; *z; z++){
|
sl@0
|
789 |
*z = sqlite3UpperToLower[*z];
|
sl@0
|
790 |
}
|
sl@0
|
791 |
rc = pMod->xFindFunction(pVtab, nArg, zLowerName, &xFunc, &pArg);
|
sl@0
|
792 |
sqlite3DbFree(db, zLowerName);
|
sl@0
|
793 |
if( pVtab->zErrMsg ){
|
sl@0
|
794 |
sqlite3Error(db, rc, "%s", pVtab->zErrMsg);
|
sl@0
|
795 |
sqlite3DbFree(db, pVtab->zErrMsg);
|
sl@0
|
796 |
pVtab->zErrMsg = 0;
|
sl@0
|
797 |
}
|
sl@0
|
798 |
}
|
sl@0
|
799 |
if( rc==0 ){
|
sl@0
|
800 |
return pDef;
|
sl@0
|
801 |
}
|
sl@0
|
802 |
|
sl@0
|
803 |
/* Create a new ephemeral function definition for the overloaded
|
sl@0
|
804 |
** function */
|
sl@0
|
805 |
pNew = sqlite3DbMallocZero(db, sizeof(*pNew) + strlen(pDef->zName) );
|
sl@0
|
806 |
if( pNew==0 ){
|
sl@0
|
807 |
return pDef;
|
sl@0
|
808 |
}
|
sl@0
|
809 |
*pNew = *pDef;
|
sl@0
|
810 |
memcpy(pNew->zName, pDef->zName, strlen(pDef->zName)+1);
|
sl@0
|
811 |
pNew->xFunc = xFunc;
|
sl@0
|
812 |
pNew->pUserData = pArg;
|
sl@0
|
813 |
pNew->flags |= SQLITE_FUNC_EPHEM;
|
sl@0
|
814 |
return pNew;
|
sl@0
|
815 |
}
|
sl@0
|
816 |
|
sl@0
|
817 |
/*
|
sl@0
|
818 |
** Make sure virtual table pTab is contained in the pParse->apVirtualLock[]
|
sl@0
|
819 |
** array so that an OP_VBegin will get generated for it. Add pTab to the
|
sl@0
|
820 |
** array if it is missing. If pTab is already in the array, this routine
|
sl@0
|
821 |
** is a no-op.
|
sl@0
|
822 |
*/
|
sl@0
|
823 |
void sqlite3VtabMakeWritable(Parse *pParse, Table *pTab){
|
sl@0
|
824 |
int i, n;
|
sl@0
|
825 |
assert( IsVirtual(pTab) );
|
sl@0
|
826 |
for(i=0; i<pParse->nVtabLock; i++){
|
sl@0
|
827 |
if( pTab==pParse->apVtabLock[i] ) return;
|
sl@0
|
828 |
}
|
sl@0
|
829 |
n = (pParse->nVtabLock+1)*sizeof(pParse->apVtabLock[0]);
|
sl@0
|
830 |
pParse->apVtabLock = sqlite3_realloc(pParse->apVtabLock, n);
|
sl@0
|
831 |
if( pParse->apVtabLock ){
|
sl@0
|
832 |
pParse->apVtabLock[pParse->nVtabLock++] = pTab;
|
sl@0
|
833 |
}else{
|
sl@0
|
834 |
pParse->db->mallocFailed = 1;
|
sl@0
|
835 |
}
|
sl@0
|
836 |
}
|
sl@0
|
837 |
|
sl@0
|
838 |
#endif /* SQLITE_OMIT_VIRTUALTABLE */
|