mirror of
https://github.com/sqlite/sqlite.git
synced 2025-11-09 14:21:03 +03:00
Cleanup the printf code to make it smaller and more modular.
Fix a memory leak in the new OP_ContextPush opcode. (CVS 1258) FossilOrigin-Name: 2756f7af3382fa9d186ab99cf76f469fb891a3c3
This commit is contained in:
113
src/util.c
113
src/util.c
@@ -14,7 +14,7 @@
|
||||
** This file contains functions for allocating memory, comparing
|
||||
** strings, and stuff like that.
|
||||
**
|
||||
** $Id: util.c,v 1.72 2004/01/07 03:41:04 drh Exp $
|
||||
** $Id: util.c,v 1.73 2004/02/21 19:02:31 drh Exp $
|
||||
*/
|
||||
#include "sqliteInt.h"
|
||||
#include <stdarg.h>
|
||||
@@ -417,120 +417,11 @@ void sqliteSetNString(char **pz, ...){
|
||||
*/
|
||||
void sqliteErrorMsg(Parse *pParse, const char *zFormat, ...){
|
||||
va_list ap;
|
||||
int nByte;
|
||||
int i, j;
|
||||
char *z;
|
||||
static char zNull[] = "NULL";
|
||||
|
||||
pParse->nErr++;
|
||||
nByte = 1 + strlen(zFormat);
|
||||
va_start(ap, zFormat);
|
||||
for(i=0; zFormat[i]; i++){
|
||||
if( zFormat[i]!='%' || zFormat[i+1]==0 ) continue;
|
||||
i++;
|
||||
switch( zFormat[i] ){
|
||||
case 'd': {
|
||||
(void)va_arg(ap, int);
|
||||
nByte += 20;
|
||||
break;
|
||||
}
|
||||
case 'z':
|
||||
case 's': {
|
||||
char *z2 = va_arg(ap, char*);
|
||||
if( z2==0 ) z2 = zNull;
|
||||
nByte += strlen(z2);
|
||||
break;
|
||||
}
|
||||
case 'T': {
|
||||
Token *p = va_arg(ap, Token*);
|
||||
nByte += p->n;
|
||||
break;
|
||||
}
|
||||
case 'S': {
|
||||
SrcList *p = va_arg(ap, SrcList*);
|
||||
int k = va_arg(ap, int);
|
||||
assert( p->nSrc>k && k>=0 );
|
||||
nByte += strlen(p->a[k].zName);
|
||||
if( p->a[k].zDatabase && p->a[k].zDatabase[0] ){
|
||||
nByte += strlen(p->a[k].zDatabase)+1;
|
||||
}
|
||||
break;
|
||||
}
|
||||
default: {
|
||||
nByte++;
|
||||
break;
|
||||
}
|
||||
}
|
||||
}
|
||||
va_end(ap);
|
||||
z = sqliteMalloc( nByte );
|
||||
if( z==0 ) return;
|
||||
sqliteFree(pParse->zErrMsg);
|
||||
pParse->zErrMsg = z;
|
||||
va_start(ap, zFormat);
|
||||
for(i=j=0; zFormat[i]; i++){
|
||||
if( zFormat[i]!='%' || zFormat[i+1]==0 ) continue;
|
||||
if( i>j ){
|
||||
memcpy(z, &zFormat[j], i-j);
|
||||
z += i-j;
|
||||
}
|
||||
j = i+2;
|
||||
i++;
|
||||
switch( zFormat[i] ){
|
||||
case 'd': {
|
||||
int x = va_arg(ap, int);
|
||||
sprintf(z, "%d", x);
|
||||
z += strlen(z);
|
||||
break;
|
||||
}
|
||||
case 'z':
|
||||
case 's': {
|
||||
int len;
|
||||
char *z2 = va_arg(ap, char*);
|
||||
if( z2==0 ) z2 = zNull;
|
||||
len = strlen(z2);
|
||||
memcpy(z, z2, len);
|
||||
z += len;
|
||||
if( zFormat[i]=='z' && z2!=zNull ){
|
||||
sqliteFree(z2);
|
||||
}
|
||||
break;
|
||||
}
|
||||
case 'T': {
|
||||
Token *p = va_arg(ap, Token*);
|
||||
memcpy(z, p->z, p->n);
|
||||
z += p->n;
|
||||
break;
|
||||
}
|
||||
case 'S': {
|
||||
int len;
|
||||
SrcList *p = va_arg(ap, SrcList*);
|
||||
int k = va_arg(ap, int);
|
||||
assert( p->nSrc>k && k>=0 );
|
||||
if( p->a[k].zDatabase && p->a[k].zDatabase[0] ){
|
||||
len = strlen(p->a[k].zDatabase);
|
||||
memcpy(z, p->a[k].zDatabase, len);
|
||||
z += len;
|
||||
*(z++) = '.';
|
||||
}
|
||||
len = strlen(p->a[k].zName);
|
||||
memcpy(z, p->a[k].zName, len);
|
||||
z += len;
|
||||
break;
|
||||
}
|
||||
default: {
|
||||
*(z++) = zFormat[i];
|
||||
break;
|
||||
}
|
||||
}
|
||||
}
|
||||
pParse->zErrMsg = sqliteVMPrintf(zFormat, ap);
|
||||
va_end(ap);
|
||||
if( i>j ){
|
||||
memcpy(z, &zFormat[j], i-j);
|
||||
z += i-j;
|
||||
}
|
||||
assert( (z - pParse->zErrMsg) < nByte );
|
||||
*z = 0;
|
||||
}
|
||||
|
||||
/*
|
||||
|
||||
Reference in New Issue
Block a user