4 ** The author disclaims copyright to this source code. In place of
5 ** a legal notice, here is a blessing:
7 ** May you do good and not evil.
8 ** May you find forgiveness for yourself and forgive others.
9 ** May you share freely, never taking more than you give.
11 *************************************************************************
13 ** This module implements the sqlite3_status() interface and related
16 #include "sqliteInt.h"
20 ** Variables in which to record status information.
22 typedef struct sqlite3StatType sqlite3StatType
;
23 static SQLITE_WSD
struct sqlite3StatType
{
24 int nowValue
[10]; /* Current value */
25 int mxValue
[10]; /* Maximum value */
26 } sqlite3Stat
= { {0,}, {0,} };
29 /* The "wsdStat" macro will resolve to the status information
30 ** state vector. If writable static data is unsupported on the target,
31 ** we have to locate the state vector at run-time. In the more common
32 ** case where writable static data is supported, wsdStat can refer directly
33 ** to the "sqlite3Stat" state vector declared above.
35 #ifdef SQLITE_OMIT_WSD
36 # define wsdStatInit sqlite3StatType *x = &GLOBAL(sqlite3StatType,sqlite3Stat)
40 # define wsdStat sqlite3Stat
44 ** Return the current value of a status parameter.
46 int sqlite3StatusValue(int op
){
48 assert( op
>=0 && op
<ArraySize(wsdStat
.nowValue
) );
49 return wsdStat
.nowValue
[op
];
53 ** Add N to the value of a status record. It is assumed that the
54 ** caller holds appropriate locks.
56 void sqlite3StatusAdd(int op
, int N
){
58 assert( op
>=0 && op
<ArraySize(wsdStat
.nowValue
) );
59 wsdStat
.nowValue
[op
] += N
;
60 if( wsdStat
.nowValue
[op
]>wsdStat
.mxValue
[op
] ){
61 wsdStat
.mxValue
[op
] = wsdStat
.nowValue
[op
];
66 ** Set the value of a status to X.
68 void sqlite3StatusSet(int op
, int X
){
70 assert( op
>=0 && op
<ArraySize(wsdStat
.nowValue
) );
71 wsdStat
.nowValue
[op
] = X
;
72 if( wsdStat
.nowValue
[op
]>wsdStat
.mxValue
[op
] ){
73 wsdStat
.mxValue
[op
] = wsdStat
.nowValue
[op
];
78 ** Query status information.
80 ** This implementation assumes that reading or writing an aligned
81 ** 32-bit integer is an atomic operation. If that assumption is not true,
82 ** then this routine is not threadsafe.
84 int sqlite3_status(int op
, int *pCurrent
, int *pHighwater
, int resetFlag
){
86 if( op
<0 || op
>=ArraySize(wsdStat
.nowValue
) ){
87 return SQLITE_MISUSE_BKPT
;
89 #ifdef SQLITE_ENABLE_API_ARMOR
90 if( pCurrent
==0 || pHighwater
==0 ) return SQLITE_MISUSE_BKPT
;
92 *pCurrent
= wsdStat
.nowValue
[op
];
93 *pHighwater
= wsdStat
.mxValue
[op
];
95 wsdStat
.mxValue
[op
] = wsdStat
.nowValue
[op
];
101 ** Query status information for a single database connection
103 int sqlite3_db_status(
104 sqlite3
*db
, /* The database connection whose status is desired */
105 int op
, /* Status verb */
106 int *pCurrent
, /* Write current value here */
107 int *pHighwater
, /* Write high-water mark here */
108 int resetFlag
/* Reset high-water mark if true */
110 int rc
= SQLITE_OK
; /* Return code */
111 #ifdef SQLITE_ENABLE_API_ARMOR
112 if( !sqlite3SafetyCheckOk(db
) || pCurrent
==0|| pHighwater
==0 ){
113 return SQLITE_MISUSE_BKPT
;
116 sqlite3_mutex_enter(db
->mutex
);
118 case SQLITE_DBSTATUS_LOOKASIDE_USED
: {
119 *pCurrent
= db
->lookaside
.nOut
;
120 *pHighwater
= db
->lookaside
.mxOut
;
122 db
->lookaside
.mxOut
= db
->lookaside
.nOut
;
127 case SQLITE_DBSTATUS_LOOKASIDE_HIT
:
128 case SQLITE_DBSTATUS_LOOKASIDE_MISS_SIZE
:
129 case SQLITE_DBSTATUS_LOOKASIDE_MISS_FULL
: {
130 testcase( op
==SQLITE_DBSTATUS_LOOKASIDE_HIT
);
131 testcase( op
==SQLITE_DBSTATUS_LOOKASIDE_MISS_SIZE
);
132 testcase( op
==SQLITE_DBSTATUS_LOOKASIDE_MISS_FULL
);
133 assert( (op
-SQLITE_DBSTATUS_LOOKASIDE_HIT
)>=0 );
134 assert( (op
-SQLITE_DBSTATUS_LOOKASIDE_HIT
)<3 );
136 *pHighwater
= db
->lookaside
.anStat
[op
- SQLITE_DBSTATUS_LOOKASIDE_HIT
];
138 db
->lookaside
.anStat
[op
- SQLITE_DBSTATUS_LOOKASIDE_HIT
] = 0;
144 ** Return an approximation for the amount of memory currently used
145 ** by all pagers associated with the given database connection. The
146 ** highwater mark is meaningless and is returned as zero.
148 case SQLITE_DBSTATUS_CACHE_USED
: {
151 sqlite3BtreeEnterAll(db
);
152 for(i
=0; i
<db
->nDb
; i
++){
153 Btree
*pBt
= db
->aDb
[i
].pBt
;
155 Pager
*pPager
= sqlite3BtreePager(pBt
);
156 totalUsed
+= sqlite3PagerMemUsed(pPager
);
159 sqlite3BtreeLeaveAll(db
);
160 *pCurrent
= totalUsed
;
166 ** *pCurrent gets an accurate estimate of the amount of memory used
167 ** to store the schema for all databases (main, temp, and any ATTACHed
168 ** databases. *pHighwater is set to zero.
170 case SQLITE_DBSTATUS_SCHEMA_USED
: {
171 int i
; /* Used to iterate through schemas */
172 int nByte
= 0; /* Used to accumulate return value */
174 sqlite3BtreeEnterAll(db
);
175 db
->pnBytesFreed
= &nByte
;
176 for(i
=0; i
<db
->nDb
; i
++){
177 Schema
*pSchema
= db
->aDb
[i
].pSchema
;
178 if( ALWAYS(pSchema
!=0) ){
181 nByte
+= sqlite3GlobalConfig
.m
.xRoundup(sizeof(HashElem
)) * (
182 pSchema
->tblHash
.count
183 + pSchema
->trigHash
.count
184 + pSchema
->idxHash
.count
185 + pSchema
->fkeyHash
.count
187 nByte
+= sqlite3MallocSize(pSchema
->tblHash
.ht
);
188 nByte
+= sqlite3MallocSize(pSchema
->trigHash
.ht
);
189 nByte
+= sqlite3MallocSize(pSchema
->idxHash
.ht
);
190 nByte
+= sqlite3MallocSize(pSchema
->fkeyHash
.ht
);
192 for(p
=sqliteHashFirst(&pSchema
->trigHash
); p
; p
=sqliteHashNext(p
)){
193 sqlite3DeleteTrigger(db
, (Trigger
*)sqliteHashData(p
));
195 for(p
=sqliteHashFirst(&pSchema
->tblHash
); p
; p
=sqliteHashNext(p
)){
196 sqlite3DeleteTable(db
, (Table
*)sqliteHashData(p
));
200 db
->pnBytesFreed
= 0;
201 sqlite3BtreeLeaveAll(db
);
209 ** *pCurrent gets an accurate estimate of the amount of memory used
210 ** to store all prepared statements.
211 ** *pHighwater is set to zero.
213 case SQLITE_DBSTATUS_STMT_USED
: {
214 struct Vdbe
*pVdbe
; /* Used to iterate through VMs */
215 int nByte
= 0; /* Used to accumulate return value */
217 db
->pnBytesFreed
= &nByte
;
218 for(pVdbe
=db
->pVdbe
; pVdbe
; pVdbe
=pVdbe
->pNext
){
219 sqlite3VdbeClearObject(db
, pVdbe
);
220 sqlite3DbFree(db
, pVdbe
);
222 db
->pnBytesFreed
= 0;
224 *pHighwater
= 0; /* IMP: R-64479-57858 */
231 ** Set *pCurrent to the total cache hits or misses encountered by all
232 ** pagers the database handle is connected to. *pHighwater is always set
235 case SQLITE_DBSTATUS_CACHE_HIT
:
236 case SQLITE_DBSTATUS_CACHE_MISS
:
237 case SQLITE_DBSTATUS_CACHE_WRITE
:{
240 assert( SQLITE_DBSTATUS_CACHE_MISS
==SQLITE_DBSTATUS_CACHE_HIT
+1 );
241 assert( SQLITE_DBSTATUS_CACHE_WRITE
==SQLITE_DBSTATUS_CACHE_HIT
+2 );
243 for(i
=0; i
<db
->nDb
; i
++){
244 if( db
->aDb
[i
].pBt
){
245 Pager
*pPager
= sqlite3BtreePager(db
->aDb
[i
].pBt
);
246 sqlite3PagerCacheStat(pPager
, op
, resetFlag
, &nRet
);
249 *pHighwater
= 0; /* IMP: R-42420-56072 */
250 /* IMP: R-54100-20147 */
251 /* IMP: R-29431-39229 */
256 /* Set *pCurrent to non-zero if there are unresolved deferred foreign
257 ** key constraints. Set *pCurrent to zero if all foreign key constraints
258 ** have been satisfied. The *pHighwater is always set to zero.
260 case SQLITE_DBSTATUS_DEFERRED_FKS
: {
261 *pHighwater
= 0; /* IMP: R-11967-56545 */
262 *pCurrent
= db
->nDeferredImmCons
>0 || db
->nDeferredCons
>0;
270 sqlite3_mutex_leave(db
->mutex
);