File:  [ELWIX - Embedded LightWeight unIX -] / embedaddon / php / ext / sqlite / sqlite.c
Revision 1.1.1.1 (vendor branch): download - view: text, annotated - select for diffs - revision graph
Tue Feb 21 23:48:02 2012 UTC (13 years, 1 month ago) by misho
Branches: php, MAIN
CVS tags: v5_3_10, HEAD
php

    1: /*
    2:    +----------------------------------------------------------------------+
    3:    | PHP Version 5                                                        |
    4:    +----------------------------------------------------------------------+
    5:    | Copyright (c) 1997-2012 The PHP Group                                |
    6:    +----------------------------------------------------------------------+
    7:    | This source file is subject to version 3.01 of the PHP license,      |
    8:    | that is bundled with this package in the file LICENSE, and is        |
    9:    | available through the world-wide-web at the following url:           |
   10:    | http://www.php.net/license/3_01.txt                                  |
   11:    | If you did not receive a copy of the PHP license and are unable to   |
   12:    | obtain it through the world-wide-web, please send a note to          |
   13:    | license@php.net so we can mail you a copy immediately.               |
   14:    +----------------------------------------------------------------------+
   15:    | Authors: Wez Furlong <wez@thebrainroom.com>                          |
   16:    |          Tal Peer <tal@php.net>                                      |
   17:    |          Marcus Boerger <helly@php.net>                              |
   18:    +----------------------------------------------------------------------+
   19: 
   20:    $Id: sqlite.c,v 1.1.1.1 2012/02/21 23:48:02 misho Exp $
   21: */
   22: 
   23: #ifdef HAVE_CONFIG_H
   24: #include "config.h"
   25: #endif
   26: 
   27: #define PHP_SQLITE_MODULE_VERSION	"2.0-dev"
   28: 
   29: #include "php.h"
   30: #include "php_ini.h"
   31: #include "ext/standard/info.h"
   32: #if HAVE_PHP_SESSION && !defined(COMPILE_DL_SESSION)
   33: #include "ext/session/php_session.h"
   34: #endif
   35: #include "php_sqlite.h"
   36: 
   37: #if HAVE_TIME_H
   38: # include <time.h>
   39: #endif
   40: #if HAVE_UNISTD_H
   41: #include <unistd.h>
   42: #endif
   43: 
   44: #include <sqlite.h>
   45: 
   46: #include "zend_exceptions.h"
   47: #include "zend_interfaces.h"
   48: 
   49: #if defined(HAVE_SPL) && ((PHP_MAJOR_VERSION > 5) || (PHP_MAJOR_VERSION == 5 && PHP_MINOR_VERSION >= 1))
   50: extern PHPAPI zend_class_entry *spl_ce_RuntimeException;
   51: extern PHPAPI zend_class_entry *spl_ce_Countable;
   52: #endif
   53: 
   54: #if PHP_SQLITE2_HAVE_PDO
   55: # include "pdo/php_pdo.h"
   56: # include "pdo/php_pdo_driver.h"
   57: extern pdo_driver_t pdo_sqlite2_driver;
   58: #endif
   59: 
   60: #ifndef safe_emalloc
   61: # define safe_emalloc(a,b,c) emalloc((a)*(b)+(c))
   62: #endif
   63: 
   64: ZEND_DECLARE_MODULE_GLOBALS(sqlite)
   65: static PHP_GINIT_FUNCTION(sqlite);
   66: 
   67: #if HAVE_PHP_SESSION && !defined(COMPILE_DL_SESSION)
   68: extern ps_module ps_mod_sqlite;
   69: #define ps_sqlite_ptr &ps_mod_sqlite
   70: #endif
   71: 
   72: extern int sqlite_encode_binary(const unsigned char *in, int n, unsigned char *out);
   73: extern int sqlite_decode_binary(const unsigned char *in, unsigned char *out);
   74: 
   75: #define php_sqlite_encode_binary(in, n, out) sqlite_encode_binary((const unsigned char *)in, n, (unsigned char *)out)
   76: #define php_sqlite_decode_binary(in, out) in && *in ? sqlite_decode_binary((const unsigned char *)in, (unsigned char *)out) : 0
   77: 
   78: static int sqlite_count_elements(zval *object, long *count TSRMLS_DC);
   79: 
   80: static int le_sqlite_db, le_sqlite_result, le_sqlite_pdb;
   81: 
   82: static inline void php_sqlite_strtoupper(char *s)
   83: {
   84: 	while (*s!='\0') {
   85: 		*s = toupper(*s);
   86: 		s++;
   87: 	}
   88: }
   89: 
   90: static inline void php_sqlite_strtolower(char *s)
   91: {
   92: 	while (*s!='\0') {
   93: 		*s = tolower(*s);
   94: 		s++;
   95: 	}
   96: }
   97: 
   98: /* {{{ PHP_INI
   99:  */
  100: PHP_INI_BEGIN()
  101: STD_PHP_INI_ENTRY_EX("sqlite.assoc_case", "0", PHP_INI_ALL, OnUpdateLong, assoc_case, zend_sqlite_globals, sqlite_globals, display_link_numbers)
  102: PHP_INI_END()
  103: /* }}} */
  104: 
  105: #define DB_FROM_ZVAL(db, zv)	ZEND_FETCH_RESOURCE2(db, struct php_sqlite_db *, zv, -1, "sqlite database", le_sqlite_db, le_sqlite_pdb)
  106: 
  107: #define DB_FROM_OBJECT(db, object) \
  108: 	{ \
  109: 		sqlite_object *obj = (sqlite_object*) zend_object_store_get_object(object TSRMLS_CC); \
  110: 		db = obj->u.db; \
  111: 		if (!db) { \
  112: 			php_error_docref(NULL TSRMLS_CC, E_WARNING, "The database wasn't opened"); \
  113: 			RETURN_NULL(); \
  114: 		} \
  115: 	}
  116: 
  117: #define RES_FROM_OBJECT_RESTORE_ERH(res, object, error_handling) \
  118: 	{ \
  119: 		sqlite_object *obj = (sqlite_object*) zend_object_store_get_object(object TSRMLS_CC); \
  120: 		res = obj->u.res; \
  121: 		if (!res) { \
  122: 			php_error_docref(NULL TSRMLS_CC, E_WARNING, "No result set available"); \
  123: 			if (error_handling) \
  124: 				zend_restore_error_handling(error_handling TSRMLS_CC); \
  125: 			RETURN_NULL(); \
  126: 		} \
  127: 	}
  128: 
  129: #define RES_FROM_OBJECT(res, object) RES_FROM_OBJECT_RESTORE_ERH(res, object, NULL)
  130: 
  131: #define PHP_SQLITE_EMPTY_QUERY \
  132: 	if (!sql_len || !*sql) { \
  133: 		php_error_docref(NULL TSRMLS_CC, E_WARNING, "Cannot execute empty query."); \
  134: 		RETURN_FALSE; \
  135: 	}
  136: 
  137: struct php_sqlite_result {
  138: 	struct php_sqlite_db *db;
  139: 	sqlite_vm *vm;
  140: 	int buffered;
  141: 	int ncolumns;
  142: 	int nrows;
  143: 	int curr_row;
  144: 	char **col_names;
  145: 	int alloc_rows;
  146: 	int mode;
  147: 	char **table;
  148: };
  149: 
  150: struct php_sqlite_db {
  151: 	sqlite *db;
  152: 	int last_err_code;
  153: 	zend_bool is_persistent;
  154: 	long rsrc_id;
  155: 
  156: 	HashTable callbacks;
  157: };
  158: 
  159: struct php_sqlite_agg_functions {
  160: 	struct php_sqlite_db *db;
  161: 	int is_valid;
  162: 	zval *step;
  163: 	zval *fini;
  164: };
  165: 
  166: static void php_sqlite_fetch_array(struct php_sqlite_result *res, int mode, zend_bool decode_binary, int move_next, zval *return_value TSRMLS_DC);
  167: static int php_sqlite_fetch(struct php_sqlite_result *rres TSRMLS_DC);
  168: 
  169: enum { PHPSQLITE_ASSOC = 1, PHPSQLITE_NUM = 2, PHPSQLITE_BOTH = PHPSQLITE_ASSOC|PHPSQLITE_NUM };
  170: 
  171: /* {{{ arginfo */
  172: ZEND_BEGIN_ARG_INFO_EX(arginfo_sqlite_popen, 0, 0, 1)
  173: 	ZEND_ARG_INFO(0, filename)
  174: 	ZEND_ARG_INFO(0, mode)
  175: 	ZEND_ARG_INFO(1, error_message)
  176: ZEND_END_ARG_INFO()
  177: 
  178: ZEND_BEGIN_ARG_INFO_EX(arginfo_sqlite_open, 0, 0, 1)
  179: 	ZEND_ARG_INFO(0, filename)
  180: 	ZEND_ARG_INFO(0, mode)
  181: 	ZEND_ARG_INFO(1, error_message)
  182: ZEND_END_ARG_INFO()
  183: 
  184: ZEND_BEGIN_ARG_INFO_EX(arginfo_sqlite_factory, 0, 0, 1)
  185: 	ZEND_ARG_INFO(0, filename)
  186: 	ZEND_ARG_INFO(0, mode)
  187: 	ZEND_ARG_INFO(1, error_message)
  188: ZEND_END_ARG_INFO()
  189: 
  190: ZEND_BEGIN_ARG_INFO_EX(arginfo_sqlite_busy_timeout, 0, 0, 2)
  191: 	ZEND_ARG_INFO(0, db)
  192: 	ZEND_ARG_INFO(0, ms)
  193: ZEND_END_ARG_INFO()
  194: 
  195: ZEND_BEGIN_ARG_INFO_EX(arginfo_sqlite_method_busy_timeout, 0, 0, 1)
  196: 	ZEND_ARG_INFO(0, ms)
  197: ZEND_END_ARG_INFO()
  198: 
  199: ZEND_BEGIN_ARG_INFO_EX(arginfo_sqlite_close, 0, 0, 1)
  200: 	ZEND_ARG_INFO(0, db)
  201: ZEND_END_ARG_INFO()
  202: 
  203: ZEND_BEGIN_ARG_INFO_EX(arginfo_sqlite_unbuffered_query, 0, 0, 2)
  204: 	ZEND_ARG_INFO(0, query)
  205: 	ZEND_ARG_INFO(0, db)
  206: 	ZEND_ARG_INFO(0, result_type)
  207: 	ZEND_ARG_INFO(1, error_message)
  208: ZEND_END_ARG_INFO()
  209: 
  210: ZEND_BEGIN_ARG_INFO_EX(arginfo_sqlite_method_unbuffered_query, 0, 0, 1)
  211: 	ZEND_ARG_INFO(0, query)
  212: 	ZEND_ARG_INFO(0, result_type)
  213: 	ZEND_ARG_INFO(1, error_message)
  214: ZEND_END_ARG_INFO()
  215: 
  216: ZEND_BEGIN_ARG_INFO_EX(arginfo_sqlite_fetch_column_types, 0, 0, 2)
  217: 	ZEND_ARG_INFO(0, table_name)
  218: 	ZEND_ARG_INFO(0, db)
  219: 	ZEND_ARG_INFO(0, result_type)
  220: ZEND_END_ARG_INFO()
  221: 
  222: ZEND_BEGIN_ARG_INFO_EX(arginfo_sqlite_method_fetch_column_types, 0, 0, 1)
  223: 	ZEND_ARG_INFO(0, table_name)
  224: 	ZEND_ARG_INFO(0, result_type)
  225: ZEND_END_ARG_INFO()
  226: 
  227: ZEND_BEGIN_ARG_INFO_EX(arginfo_sqlite_query, 0, 0, 2)
  228: 	ZEND_ARG_INFO(0, query)
  229: 	ZEND_ARG_INFO(0, db)
  230: 	ZEND_ARG_INFO(0, result_type)
  231: 	ZEND_ARG_INFO(1, error_message)
  232: ZEND_END_ARG_INFO()
  233: 
  234: ZEND_BEGIN_ARG_INFO_EX(arginfo_sqlite_method_query, 0, 0, 1)
  235: 	ZEND_ARG_INFO(0, query)
  236: 	ZEND_ARG_INFO(0, result_type)
  237: 	ZEND_ARG_INFO(1, error_message)
  238: ZEND_END_ARG_INFO()
  239: 
  240: ZEND_BEGIN_ARG_INFO_EX(arginfo_sqlite_exec, 0, 0, 2)
  241: 	ZEND_ARG_INFO(0, query)
  242: 	ZEND_ARG_INFO(0, db)
  243: 	ZEND_ARG_INFO(1, error_message)
  244: ZEND_END_ARG_INFO()
  245: 
  246: ZEND_BEGIN_ARG_INFO_EX(arginfo_sqlite_method_exec, 0, 0, 1)
  247: 	ZEND_ARG_INFO(0, query)
  248: 	ZEND_ARG_INFO(1, error_message)
  249: ZEND_END_ARG_INFO()
  250: 
  251: ZEND_BEGIN_ARG_INFO_EX(arginfo_sqlite_fetch_all, 0, 0, 1)
  252: 	ZEND_ARG_INFO(0, result)
  253: 	ZEND_ARG_INFO(0, result_type)
  254: 	ZEND_ARG_INFO(0, decode_binary)
  255: ZEND_END_ARG_INFO()
  256: 
  257: ZEND_BEGIN_ARG_INFO_EX(arginfo_sqlite_method_fetch_all, 0, 0, 0)
  258: 	ZEND_ARG_INFO(0, result_type)
  259: 	ZEND_ARG_INFO(0, decode_binary)
  260: ZEND_END_ARG_INFO()
  261: 
  262: ZEND_BEGIN_ARG_INFO_EX(arginfo_sqlite_fetch_array, 0, 0, 1)
  263: 	ZEND_ARG_INFO(0, result)
  264: 	ZEND_ARG_INFO(0, result_type)
  265: 	ZEND_ARG_INFO(0, decode_binary)
  266: ZEND_END_ARG_INFO()
  267: 
  268: ZEND_BEGIN_ARG_INFO_EX(arginfo_sqlite_method_fetch_array, 0, 0, 0)
  269: 	ZEND_ARG_INFO(0, result_type)
  270: 	ZEND_ARG_INFO(0, decode_binary)
  271: ZEND_END_ARG_INFO()
  272: 
  273: ZEND_BEGIN_ARG_INFO_EX(arginfo_sqlite_fetch_object, 0, 0, 1)
  274: 	ZEND_ARG_INFO(0, result)
  275: 	ZEND_ARG_INFO(0, class_name)
  276: 	ZEND_ARG_INFO(0, ctor_params)
  277: 	ZEND_ARG_INFO(0, decode_binary)
  278: ZEND_END_ARG_INFO()
  279: 
  280: ZEND_BEGIN_ARG_INFO_EX(arginfo_sqlite_method_fetch_object, 0, 0, 0)
  281: 	ZEND_ARG_INFO(0, class_name)
  282: 	ZEND_ARG_INFO(0, ctor_params)
  283: 	ZEND_ARG_INFO(0, decode_binary)
  284: ZEND_END_ARG_INFO()
  285: 
  286: ZEND_BEGIN_ARG_INFO_EX(arginfo_sqlite_array_query, 0, 0, 2)
  287: 	ZEND_ARG_INFO(0, db)
  288: 	ZEND_ARG_INFO(0, query)
  289: 	ZEND_ARG_INFO(0, result_type)
  290: 	ZEND_ARG_INFO(0, decode_binary)
  291: ZEND_END_ARG_INFO()
  292: 
  293: ZEND_BEGIN_ARG_INFO_EX(arginfo_sqlite_method_array_query, 0, 0, 1)
  294: 	ZEND_ARG_INFO(0, query)
  295: 	ZEND_ARG_INFO(0, result_type)
  296: 	ZEND_ARG_INFO(0, decode_binary)
  297: ZEND_END_ARG_INFO()
  298: 
  299: ZEND_BEGIN_ARG_INFO_EX(arginfo_sqlite_single_query, 0, 0, 2)
  300: 	ZEND_ARG_INFO(0, db)
  301: 	ZEND_ARG_INFO(0, query)
  302: 	ZEND_ARG_INFO(0, first_row_only)
  303: 	ZEND_ARG_INFO(0, decode_binary)
  304: ZEND_END_ARG_INFO()
  305: 
  306: ZEND_BEGIN_ARG_INFO_EX(arginfo_sqlite_method_single_query, 0, 0, 1)
  307: 	ZEND_ARG_INFO(0, query)
  308: 	ZEND_ARG_INFO(0, first_row_only)
  309: 	ZEND_ARG_INFO(0, decode_binary)
  310: ZEND_END_ARG_INFO()
  311: 
  312: ZEND_BEGIN_ARG_INFO_EX(arginfo_sqlite_fetch_single, 0, 0, 1)
  313: 	ZEND_ARG_INFO(0, result)
  314: 	ZEND_ARG_INFO(0, decode_binary)
  315: ZEND_END_ARG_INFO()
  316: 
  317: ZEND_BEGIN_ARG_INFO_EX(arginfo_sqlite_method_fetch_single, 0, 0, 0)
  318: 	ZEND_ARG_INFO(0, decode_binary)
  319: ZEND_END_ARG_INFO()
  320: 
  321: ZEND_BEGIN_ARG_INFO_EX(arginfo_sqlite_current, 0, 0, 1)
  322: 	ZEND_ARG_INFO(0, result)
  323: 	ZEND_ARG_INFO(0, result_type)
  324: 	ZEND_ARG_INFO(0, decode_binary)
  325: ZEND_END_ARG_INFO()
  326: 
  327: ZEND_BEGIN_ARG_INFO_EX(arginfo_sqlite_method_current, 0, 0, 0)
  328: 	ZEND_ARG_INFO(0, result_type)
  329: 	ZEND_ARG_INFO(0, decode_binary)
  330: ZEND_END_ARG_INFO()
  331: 
  332: ZEND_BEGIN_ARG_INFO_EX(arginfo_sqlite_column, 0, 0, 2)
  333: 	ZEND_ARG_INFO(0, result)
  334: 	ZEND_ARG_INFO(0, index_or_name)
  335: 	ZEND_ARG_INFO(0, decode_binary)
  336: ZEND_END_ARG_INFO()
  337: 
  338: ZEND_BEGIN_ARG_INFO_EX(arginfo_sqlite_method_column, 0, 0, 1)
  339: 	ZEND_ARG_INFO(0, index_or_name)
  340: 	ZEND_ARG_INFO(0, decode_binary)
  341: ZEND_END_ARG_INFO()
  342: 
  343: ZEND_BEGIN_ARG_INFO(arginfo_sqlite_libversion, 0)
  344: ZEND_END_ARG_INFO()
  345: 
  346: ZEND_BEGIN_ARG_INFO(arginfo_sqlite_libencoding, 0)
  347: ZEND_END_ARG_INFO()
  348: 
  349: ZEND_BEGIN_ARG_INFO_EX(arginfo_sqlite_changes, 0, 0, 1)
  350: 	ZEND_ARG_INFO(0, db)
  351: ZEND_END_ARG_INFO()
  352: 
  353: ZEND_BEGIN_ARG_INFO(arginfo_sqlite_method_changes, 0)
  354: ZEND_END_ARG_INFO()
  355: 
  356: ZEND_BEGIN_ARG_INFO_EX(arginfo_sqlite_last_insert_rowid, 0, 0, 1)
  357: 	ZEND_ARG_INFO(0, db)
  358: ZEND_END_ARG_INFO()
  359: 
  360: ZEND_BEGIN_ARG_INFO(arginfo_sqlite_method_last_insert_rowid, 0)
  361: ZEND_END_ARG_INFO()
  362: 
  363: ZEND_BEGIN_ARG_INFO_EX(arginfo_sqlite_num_rows, 0, 0, 1)
  364: 	ZEND_ARG_INFO(0, result)
  365: ZEND_END_ARG_INFO()
  366: 
  367: ZEND_BEGIN_ARG_INFO(arginfo_sqlite_method_num_rows, 0)
  368: ZEND_END_ARG_INFO()
  369: 
  370: ZEND_BEGIN_ARG_INFO_EX(arginfo_sqlite_valid, 0, 0, 1)
  371: 	ZEND_ARG_INFO(0, result)
  372: ZEND_END_ARG_INFO()
  373: 
  374: ZEND_BEGIN_ARG_INFO(arginfo_sqlite_method_valid, 0)
  375: ZEND_END_ARG_INFO()
  376: 
  377: ZEND_BEGIN_ARG_INFO_EX(arginfo_sqlite_has_prev, 0, 0, 1)
  378: 	ZEND_ARG_INFO(0, result)
  379: ZEND_END_ARG_INFO()
  380: 
  381: ZEND_BEGIN_ARG_INFO(arginfo_sqlite_method_has_prev, 0)
  382: ZEND_END_ARG_INFO()
  383: 
  384: ZEND_BEGIN_ARG_INFO_EX(arginfo_sqlite_num_fields, 0, 0, 1)
  385: 	ZEND_ARG_INFO(0, result)
  386: ZEND_END_ARG_INFO()
  387: 
  388: ZEND_BEGIN_ARG_INFO(arginfo_sqlite_method_num_fields, 0)
  389: ZEND_END_ARG_INFO()
  390: 
  391: ZEND_BEGIN_ARG_INFO_EX(arginfo_sqlite_field_name, 0, 0, 2)
  392: 	ZEND_ARG_INFO(0, result)
  393: 	ZEND_ARG_INFO(0, field_index)
  394: ZEND_END_ARG_INFO()
  395: 
  396: ZEND_BEGIN_ARG_INFO_EX(arginfo_sqlite_method_field_name, 0, 0, 1)
  397: 	ZEND_ARG_INFO(0, field_index)
  398: ZEND_END_ARG_INFO()
  399: 
  400: ZEND_BEGIN_ARG_INFO_EX(arginfo_sqlite_seek, 0, 0, 2)
  401: 	ZEND_ARG_INFO(0, result)
  402: 	ZEND_ARG_INFO(0, row)
  403: ZEND_END_ARG_INFO()
  404: 
  405: ZEND_BEGIN_ARG_INFO_EX(arginfo_sqlite_method_seek, 0, 0, 1)
  406: 	ZEND_ARG_INFO(0, row)
  407: ZEND_END_ARG_INFO()
  408: 
  409: ZEND_BEGIN_ARG_INFO_EX(arginfo_sqlite_rewind, 0, 0, 1)
  410: 	ZEND_ARG_INFO(0, result)
  411: ZEND_END_ARG_INFO()
  412: 
  413: ZEND_BEGIN_ARG_INFO(arginfo_sqlite_method_rewind, 0)
  414: ZEND_END_ARG_INFO()
  415: 
  416: ZEND_BEGIN_ARG_INFO_EX(arginfo_sqlite_next, 0, 0, 1)
  417: 	ZEND_ARG_INFO(0, result)
  418: ZEND_END_ARG_INFO()
  419: 
  420: ZEND_BEGIN_ARG_INFO(arginfo_sqlite_method_next, 0)
  421: ZEND_END_ARG_INFO()
  422: 
  423: ZEND_BEGIN_ARG_INFO_EX(arginfo_sqlite_key, 0, 0, 1)
  424: 	ZEND_ARG_INFO(0, result)
  425: ZEND_END_ARG_INFO()
  426: 
  427: ZEND_BEGIN_ARG_INFO(arginfo_sqlite_method_key, 0)
  428: ZEND_END_ARG_INFO()
  429: 
  430: ZEND_BEGIN_ARG_INFO_EX(arginfo_sqlite_prev, 0, 0, 1)
  431: 	ZEND_ARG_INFO(0, result)
  432: ZEND_END_ARG_INFO()
  433: 
  434: ZEND_BEGIN_ARG_INFO(arginfo_sqlite_method_prev, 0)
  435: ZEND_END_ARG_INFO()
  436: 
  437: ZEND_BEGIN_ARG_INFO_EX(arginfo_sqlite_escape_string, 0, 0, 1)
  438: 	ZEND_ARG_INFO(0, item)
  439: ZEND_END_ARG_INFO()
  440: 
  441: ZEND_BEGIN_ARG_INFO_EX(arginfo_sqlite_last_error, 0, 0, 1)
  442: 	ZEND_ARG_INFO(0, db)
  443: ZEND_END_ARG_INFO()
  444: 
  445: ZEND_BEGIN_ARG_INFO(arginfo_sqlite_method_last_error, 0)
  446: ZEND_END_ARG_INFO()
  447: 
  448: ZEND_BEGIN_ARG_INFO_EX(arginfo_sqlite_error_string, 0, 0, 1)
  449: 	ZEND_ARG_INFO(0, error_code)
  450: ZEND_END_ARG_INFO()
  451: 
  452: ZEND_BEGIN_ARG_INFO_EX(arginfo_sqlite_create_aggregate, 0, 0, 4)
  453: 	ZEND_ARG_INFO(0, db)
  454: 	ZEND_ARG_INFO(0, funcname)
  455: 	ZEND_ARG_INFO(0, step_func)
  456: 	ZEND_ARG_INFO(0, finalize_func)
  457: 	ZEND_ARG_INFO(0, num_args)
  458: ZEND_END_ARG_INFO()
  459: 
  460: ZEND_BEGIN_ARG_INFO_EX(arginfo_sqlite_method_create_aggregate, 0, 0, 3)
  461: 	ZEND_ARG_INFO(0, funcname)
  462: 	ZEND_ARG_INFO(0, step_func)
  463: 	ZEND_ARG_INFO(0, finalize_func)
  464: 	ZEND_ARG_INFO(0, num_args)
  465: ZEND_END_ARG_INFO()
  466: 
  467: ZEND_BEGIN_ARG_INFO_EX(arginfo_sqlite_create_function, 0, 0, 3)
  468: 	ZEND_ARG_INFO(0, db)
  469: 	ZEND_ARG_INFO(0, funcname)
  470: 	ZEND_ARG_INFO(0, callback)
  471: 	ZEND_ARG_INFO(0, num_args)
  472: ZEND_END_ARG_INFO()
  473: 
  474: ZEND_BEGIN_ARG_INFO_EX(arginfo_sqlite_method_create_function, 0, 0, 2)
  475: 	ZEND_ARG_INFO(0, funcname)
  476: 	ZEND_ARG_INFO(0, callback)
  477: 	ZEND_ARG_INFO(0, num_args)
  478: ZEND_END_ARG_INFO()
  479: 
  480: ZEND_BEGIN_ARG_INFO_EX(arginfo_sqlite_udf_encode_binary, 0, 0, 1)
  481: 	ZEND_ARG_INFO(0, data)
  482: ZEND_END_ARG_INFO()
  483: 
  484: ZEND_BEGIN_ARG_INFO_EX(arginfo_sqlite_udf_decode_binary, 0, 0, 1)
  485: 	ZEND_ARG_INFO(0, data)
  486: ZEND_END_ARG_INFO()
  487: /* }}} */
  488: 
  489: const zend_function_entry sqlite_functions[] = {
  490: 	PHP_FE(sqlite_open, 				arginfo_sqlite_open)
  491: 	PHP_FE(sqlite_popen, 				arginfo_sqlite_popen)
  492: 	PHP_FE(sqlite_close, 				arginfo_sqlite_close)
  493: 	PHP_FE(sqlite_query, 				arginfo_sqlite_query)
  494: 	PHP_FE(sqlite_exec, 				arginfo_sqlite_exec)
  495: 	PHP_FE(sqlite_array_query, 			arginfo_sqlite_array_query)
  496: 	PHP_FE(sqlite_single_query, 		arginfo_sqlite_single_query)
  497: 	PHP_FE(sqlite_fetch_array, 			arginfo_sqlite_fetch_array)
  498: 	PHP_FE(sqlite_fetch_object, 		arginfo_sqlite_fetch_object)
  499: 	PHP_FE(sqlite_fetch_single, 		arginfo_sqlite_fetch_single)
  500: 	PHP_FALIAS(sqlite_fetch_string, sqlite_fetch_single, arginfo_sqlite_fetch_single)
  501: 	PHP_FE(sqlite_fetch_all, 			arginfo_sqlite_fetch_all)
  502: 	PHP_FE(sqlite_current, 				arginfo_sqlite_current)
  503: 	PHP_FE(sqlite_column, 				arginfo_sqlite_column)
  504: 	PHP_FE(sqlite_libversion, 			arginfo_sqlite_libversion)
  505: 	PHP_FE(sqlite_libencoding, 			arginfo_sqlite_libencoding)
  506: 	PHP_FE(sqlite_changes, 				arginfo_sqlite_changes)
  507: 	PHP_FE(sqlite_last_insert_rowid, 	arginfo_sqlite_last_insert_rowid)
  508: 	PHP_FE(sqlite_num_rows, 			arginfo_sqlite_num_rows)
  509: 	PHP_FE(sqlite_num_fields, 			arginfo_sqlite_num_fields)
  510: 	PHP_FE(sqlite_field_name, 			arginfo_sqlite_field_name)
  511: 	PHP_FE(sqlite_seek, 				arginfo_sqlite_seek)
  512: 	PHP_FE(sqlite_rewind, 				arginfo_sqlite_rewind)
  513: 	PHP_FE(sqlite_next, 				arginfo_sqlite_next)
  514: 	PHP_FE(sqlite_prev, 				arginfo_sqlite_prev)
  515: 	PHP_FE(sqlite_valid, 				arginfo_sqlite_valid)
  516: 	PHP_FALIAS(sqlite_has_more, sqlite_valid, arginfo_sqlite_valid)
  517: 	PHP_FE(sqlite_has_prev, 			arginfo_sqlite_has_prev)
  518: 	PHP_FE(sqlite_escape_string, 		arginfo_sqlite_escape_string)
  519: 	PHP_FE(sqlite_busy_timeout, 		arginfo_sqlite_busy_timeout)
  520: 	PHP_FE(sqlite_last_error, 			arginfo_sqlite_last_error)
  521: 	PHP_FE(sqlite_error_string, 		arginfo_sqlite_error_string)
  522: 	PHP_FE(sqlite_unbuffered_query, 	arginfo_sqlite_unbuffered_query)
  523: 	PHP_FE(sqlite_create_aggregate, 	arginfo_sqlite_create_aggregate)
  524: 	PHP_FE(sqlite_create_function, 		arginfo_sqlite_create_function)
  525: 	PHP_FE(sqlite_factory, 				arginfo_sqlite_factory)
  526: 	PHP_FE(sqlite_udf_encode_binary, 	arginfo_sqlite_udf_encode_binary)
  527: 	PHP_FE(sqlite_udf_decode_binary, 	arginfo_sqlite_udf_decode_binary)
  528: 	PHP_FE(sqlite_fetch_column_types,	arginfo_sqlite_fetch_column_types)
  529: 	{NULL, NULL, NULL}
  530: };
  531: 
  532: const zend_function_entry sqlite_funcs_db[] = {
  533: 	PHP_ME_MAPPING(__construct, sqlite_open, arginfo_sqlite_open, 0)
  534: /*	PHP_ME_MAPPING(close, sqlite_close, NULL, 0)*/
  535: 	PHP_ME_MAPPING(query, sqlite_query, arginfo_sqlite_method_query, 0)
  536: 	PHP_ME_MAPPING(queryExec, sqlite_exec, arginfo_sqlite_method_exec, 0)
  537: 	PHP_ME_MAPPING(arrayQuery, sqlite_array_query, arginfo_sqlite_method_array_query, 0)
  538: 	PHP_ME_MAPPING(singleQuery, sqlite_single_query, arginfo_sqlite_method_single_query, 0)
  539: 	PHP_ME_MAPPING(unbufferedQuery, sqlite_unbuffered_query, arginfo_sqlite_method_unbuffered_query, 0)
  540: 	PHP_ME_MAPPING(lastInsertRowid, sqlite_last_insert_rowid, arginfo_sqlite_method_last_insert_rowid, 0)
  541: 	PHP_ME_MAPPING(changes, sqlite_changes, arginfo_sqlite_method_changes, 0)
  542: 	PHP_ME_MAPPING(createAggregate, sqlite_create_aggregate, arginfo_sqlite_method_create_aggregate, 0)
  543: 	PHP_ME_MAPPING(createFunction, sqlite_create_function, arginfo_sqlite_method_create_function, 0)
  544: 	PHP_ME_MAPPING(busyTimeout, sqlite_busy_timeout, arginfo_sqlite_method_busy_timeout, 0)
  545: 	PHP_ME_MAPPING(lastError, sqlite_last_error, arginfo_sqlite_method_last_error, 0)
  546: 	PHP_ME_MAPPING(fetchColumnTypes, sqlite_fetch_column_types, arginfo_sqlite_method_fetch_column_types, 0)
  547: /*	PHP_ME_MAPPING(error_string, sqlite_error_string, NULL, 0) static */
  548: /*	PHP_ME_MAPPING(escape_string, sqlite_escape_string, NULL, 0) static */
  549: 	{NULL, NULL, NULL}
  550: };
  551: 
  552: const zend_function_entry sqlite_funcs_query[] = {
  553: 	PHP_ME_MAPPING(fetch, sqlite_fetch_array, arginfo_sqlite_method_fetch_array, 0)
  554: 	PHP_ME_MAPPING(fetchObject, sqlite_fetch_object, arginfo_sqlite_method_fetch_object, 0)
  555: 	PHP_ME_MAPPING(fetchSingle, sqlite_fetch_single, arginfo_sqlite_method_fetch_single, 0)
  556: 	PHP_ME_MAPPING(fetchAll, sqlite_fetch_all, arginfo_sqlite_method_fetch_all, 0)
  557: 	PHP_ME_MAPPING(column, sqlite_column, arginfo_sqlite_method_column, 0)
  558: 	PHP_ME_MAPPING(numFields, sqlite_num_fields, arginfo_sqlite_method_num_fields, 0)
  559: 	PHP_ME_MAPPING(fieldName, sqlite_field_name, arginfo_sqlite_method_field_name, 0)
  560: 	/* iterator */
  561: 	PHP_ME_MAPPING(current, sqlite_current, arginfo_sqlite_method_current, 0)
  562: 	PHP_ME_MAPPING(key, sqlite_key, arginfo_sqlite_method_key, 0)
  563: 	PHP_ME_MAPPING(next, sqlite_next, arginfo_sqlite_method_next, 0)
  564: 	PHP_ME_MAPPING(valid, sqlite_valid, arginfo_sqlite_method_valid, 0)
  565: 	PHP_ME_MAPPING(rewind, sqlite_rewind, arginfo_sqlite_method_rewind, 0)
  566: 	/* countable */
  567: 	PHP_ME_MAPPING(count, sqlite_num_rows, arginfo_sqlite_method_num_rows, 0)
  568: 	/* additional */
  569: 	PHP_ME_MAPPING(prev, sqlite_prev, arginfo_sqlite_method_prev, 0)
  570: 	PHP_ME_MAPPING(hasPrev, sqlite_has_prev, arginfo_sqlite_method_has_prev, 0)
  571: 	PHP_ME_MAPPING(numRows, sqlite_num_rows, arginfo_sqlite_method_num_rows, 0)
  572: 	PHP_ME_MAPPING(seek, sqlite_seek, arginfo_sqlite_method_seek, 0)
  573: 	{NULL, NULL, NULL}
  574: };
  575: 
  576: const zend_function_entry sqlite_funcs_ub_query[] = {
  577: 	PHP_ME_MAPPING(fetch, sqlite_fetch_array, arginfo_sqlite_method_fetch_array, 0)
  578: 	PHP_ME_MAPPING(fetchObject, sqlite_fetch_object, arginfo_sqlite_method_fetch_object, 0)
  579: 	PHP_ME_MAPPING(fetchSingle, sqlite_fetch_single, arginfo_sqlite_method_fetch_single, 0)
  580: 	PHP_ME_MAPPING(fetchAll, sqlite_fetch_all, arginfo_sqlite_method_fetch_all, 0)
  581: 	PHP_ME_MAPPING(column, sqlite_column, arginfo_sqlite_method_column, 0)
  582: 	PHP_ME_MAPPING(numFields, sqlite_num_fields, arginfo_sqlite_method_num_fields, 0)
  583: 	PHP_ME_MAPPING(fieldName, sqlite_field_name, arginfo_sqlite_method_field_name, 0)
  584: 	/* iterator */
  585: 	PHP_ME_MAPPING(current, sqlite_current, arginfo_sqlite_method_current, 0)
  586: 	PHP_ME_MAPPING(next, sqlite_next, arginfo_sqlite_method_next, 0)
  587: 	PHP_ME_MAPPING(valid, sqlite_valid, arginfo_sqlite_method_valid, 0)
  588: 	{NULL, NULL, NULL}
  589: };
  590: 
  591: const zend_function_entry sqlite_funcs_exception[] = {
  592: 	{NULL, NULL, NULL}
  593: };
  594: 
  595: /* Dependancies */
  596: static const zend_module_dep sqlite_deps[] = {
  597: #if defined(HAVE_SPL) && ((PHP_MAJOR_VERSION > 5) || (PHP_MAJOR_VERSION == 5 && PHP_MINOR_VERSION >= 1))
  598: 	ZEND_MOD_REQUIRED("spl")
  599: #endif
  600: #if HAVE_PHP_SESSION && !defined(COMPILE_DL_SESSION)
  601: 	ZEND_MOD_REQUIRED("session")
  602: #endif
  603: #ifdef PHP_SQLITE2_HAVE_PDO
  604: 	ZEND_MOD_REQUIRED("pdo")
  605: #endif
  606: 	{NULL, NULL, NULL}
  607: };
  608: 
  609: zend_module_entry sqlite_module_entry = {
  610: #if ZEND_MODULE_API_NO >= 20050922
  611: 	STANDARD_MODULE_HEADER_EX, NULL,
  612: 	sqlite_deps,
  613: #elif ZEND_MODULE_API_NO >= 20010901
  614: 	STANDARD_MODULE_HEADER,
  615: #endif
  616: 	"SQLite",
  617: 	sqlite_functions,
  618: 	PHP_MINIT(sqlite),
  619: 	PHP_MSHUTDOWN(sqlite),
  620: 	NULL,
  621: 	PHP_RSHUTDOWN(sqlite),
  622: 	PHP_MINFO(sqlite),
  623: #if ZEND_MODULE_API_NO >= 20010901
  624: 	PHP_SQLITE_MODULE_VERSION,
  625: #endif
  626: #if ZEND_MODULE_API_NO >= 20060613
  627: 	PHP_MODULE_GLOBALS(sqlite),
  628: 	PHP_GINIT(sqlite),
  629: 	NULL,
  630: 	NULL,
  631: 	STANDARD_MODULE_PROPERTIES_EX
  632: #else
  633: 	STANDARD_MODULE_PROPERTIES
  634: #endif
  635: };
  636: 
  637: 
  638: #ifdef COMPILE_DL_SQLITE
  639: ZEND_GET_MODULE(sqlite)
  640: #endif
  641: 
  642: static int php_sqlite_callback_invalidator(struct php_sqlite_agg_functions *funcs TSRMLS_DC)
  643: {
  644: 	if (!funcs->is_valid) {
  645: 		return 0;
  646: 	}
  647: 
  648: 	if (funcs->step) {
  649: 		zval_ptr_dtor(&funcs->step);
  650: 		funcs->step = NULL;
  651: 	}
  652: 
  653: 	if (funcs->fini) {
  654: 		zval_ptr_dtor(&funcs->fini);
  655: 		funcs->fini = NULL;
  656: 	}
  657: 
  658: 	funcs->is_valid = 0;
  659: 
  660: 	return 0;
  661: }
  662: 
  663: 
  664: static void php_sqlite_callback_dtor(void *pDest)
  665: {
  666: 	struct php_sqlite_agg_functions *funcs = (struct php_sqlite_agg_functions*)pDest;
  667: 
  668: 	if (funcs->is_valid) {
  669: 		TSRMLS_FETCH();
  670: 
  671: 		php_sqlite_callback_invalidator(funcs TSRMLS_CC);
  672: 	}
  673: }
  674: 
  675: static ZEND_RSRC_DTOR_FUNC(php_sqlite_db_dtor)
  676: {
  677: 	if (rsrc->ptr) {
  678: 		struct php_sqlite_db *db = (struct php_sqlite_db*)rsrc->ptr;
  679: 
  680: 		sqlite_close(db->db);
  681: 
  682: 		zend_hash_destroy(&db->callbacks);
  683: 
  684: 		pefree(db, db->is_persistent);
  685: 
  686: 		rsrc->ptr = NULL;
  687: 	}
  688: }
  689: 
  690: static void real_result_dtor(struct php_sqlite_result *res TSRMLS_DC)
  691: {
  692: 	int i, j, base;
  693: 
  694: 	if (res->vm) {
  695: 		sqlite_finalize(res->vm, NULL);
  696: 	}
  697: 
  698: 	if (res->table) {
  699: 		if (!res->buffered && res->nrows) {
  700: 			res->nrows = 1; /* only one row is stored */
  701: 		}
  702: 		for (i = 0; i < res->nrows; i++) {
  703: 			base = i * res->ncolumns;
  704: 			for (j = 0; j < res->ncolumns; j++) {
  705: 				if (res->table[base + j] != NULL) {
  706: 					efree(res->table[base + j]);
  707: 				}
  708: 			}
  709: 		}
  710: 		efree(res->table);
  711: 	}
  712: 	if (res->col_names) {
  713: 		for (j = 0; j < res->ncolumns; j++) {
  714: 			efree(res->col_names[j]);
  715: 		}
  716: 		efree(res->col_names);
  717: 	}
  718: 
  719: 	if (res->db) {
  720: 		zend_list_delete(res->db->rsrc_id);
  721: 	}
  722: 	efree(res);
  723: }
  724: 
  725: static int _clean_unfinished_results(zend_rsrc_list_entry *le, void *db TSRMLS_DC)
  726: {
  727: 	if (Z_TYPE_P(le) == le_sqlite_result) {
  728: 		struct php_sqlite_result *res = (struct php_sqlite_result *)le->ptr;
  729: 		if (res->db->rsrc_id == ((struct php_sqlite_db*)db)->rsrc_id) {
  730: 			return ZEND_HASH_APPLY_REMOVE;
  731: 		}
  732: 	}
  733: 	return ZEND_HASH_APPLY_KEEP;
  734: }
  735: 
  736: static ZEND_RSRC_DTOR_FUNC(php_sqlite_result_dtor)
  737: {
  738: 	struct php_sqlite_result *res = (struct php_sqlite_result *)rsrc->ptr;
  739: 	real_result_dtor(res TSRMLS_CC);
  740: }
  741: 
  742: static int php_sqlite_forget_persistent_id_numbers(zend_rsrc_list_entry *rsrc TSRMLS_DC)
  743: {
  744: 	struct php_sqlite_db *db = (struct php_sqlite_db*)rsrc->ptr;
  745: 
  746: 	if (Z_TYPE_P(rsrc) != le_sqlite_pdb) {
  747: 		return 0;
  748: 	}
  749: 
  750: 	/* prevent bad mojo if someone tries to use a previously registered function in the next request */
  751: 	zend_hash_apply(&db->callbacks, (apply_func_t)php_sqlite_callback_invalidator TSRMLS_CC);
  752: 
  753: 	db->rsrc_id = FAILURE;
  754: 
  755: 	/* don't leave pending commits hanging around */
  756: 	sqlite_exec(db->db, "ROLLBACK", NULL, NULL, NULL);
  757: 
  758: 	return 0;
  759: }
  760: 
  761: PHP_RSHUTDOWN_FUNCTION(sqlite)
  762: {
  763: 	zend_hash_apply(&EG(persistent_list), (apply_func_t)php_sqlite_forget_persistent_id_numbers TSRMLS_CC);
  764: 	return SUCCESS;
  765: }
  766: 
  767: /* {{{ PHP Function interface */
  768: static void php_sqlite_generic_function_callback(sqlite_func *func, int argc, const char **argv)
  769: {
  770: 	zval *retval = NULL;
  771: 	zval ***zargs = NULL;
  772: 	zval funcname;
  773: 	int i, res;
  774: 	char *callable = NULL, *errbuf=NULL;
  775: 	TSRMLS_FETCH();
  776: 
  777: 	/* sanity check the args */
  778: 	if (argc == 0) {
  779: 		sqlite_set_result_error(func, "not enough parameters", -1);
  780: 		return;
  781: 	}
  782: 
  783: 	ZVAL_STRING(&funcname, (char*)argv[0], 1);
  784: 
  785: 	if (!zend_make_callable(&funcname, &callable TSRMLS_CC)) {
  786: 		spprintf(&errbuf, 0, "function `%s' is not a function name", callable);
  787: 		sqlite_set_result_error(func, errbuf, -1);
  788: 		efree(errbuf);
  789: 		efree(callable);
  790: 		zval_dtor(&funcname);
  791: 		return;
  792: 	}
  793: 
  794: 	if (argc > 1) {
  795: 		zargs = (zval ***)safe_emalloc((argc - 1), sizeof(zval **), 0);
  796: 
  797: 		for (i = 0; i < argc-1; i++) {
  798: 			zargs[i] = emalloc(sizeof(zval *));
  799: 			MAKE_STD_ZVAL(*zargs[i]);
  800: 			ZVAL_STRING(*zargs[i], (char*)argv[i+1], 1);
  801: 		}
  802: 	}
  803: 
  804: 	res = call_user_function_ex(EG(function_table),
  805: 			NULL,
  806: 			&funcname,
  807: 			&retval,
  808: 			argc-1,
  809: 			zargs,
  810: 			0, NULL TSRMLS_CC);
  811: 
  812: 	zval_dtor(&funcname);
  813: 
  814: 	if (res == SUCCESS) {
  815: 		if (retval == NULL) {
  816: 			sqlite_set_result_string(func, NULL, 0);
  817: 		} else {
  818: 			switch (Z_TYPE_P(retval)) {
  819: 				case IS_STRING:
  820: 					sqlite_set_result_string(func, Z_STRVAL_P(retval), Z_STRLEN_P(retval));
  821: 					break;
  822: 				case IS_LONG:
  823: 				case IS_BOOL:
  824: 					sqlite_set_result_int(func, Z_LVAL_P(retval));
  825: 					break;
  826: 				case IS_DOUBLE:
  827: 					sqlite_set_result_double(func, Z_DVAL_P(retval));
  828: 					break;
  829: 				case IS_NULL:
  830: 				default:
  831: 					sqlite_set_result_string(func, NULL, 0);
  832: 			}
  833: 		}
  834: 	} else {
  835: 		char *errbuf;
  836: 		spprintf(&errbuf, 0, "call_user_function_ex failed for function %s()", callable);
  837: 		sqlite_set_result_error(func, errbuf, -1);
  838: 		efree(errbuf);
  839: 	}
  840: 
  841: 	efree(callable);
  842: 
  843: 	if (retval) {
  844: 		zval_ptr_dtor(&retval);
  845: 	}
  846: 
  847: 	if (zargs) {
  848: 		for (i = 0; i < argc-1; i++) {
  849: 			zval_ptr_dtor(zargs[i]);
  850: 			efree(zargs[i]);
  851: 		}
  852: 		efree(zargs);
  853: 	}
  854: }
  855: /* }}} */
  856: 
  857: /* {{{ callback for sqlite_create_function */
  858: static void php_sqlite_function_callback(sqlite_func *func, int argc, const char **argv)
  859: {
  860: 	zval *retval = NULL;
  861: 	zval ***zargs = NULL;
  862: 	int i, res;
  863: 	struct php_sqlite_agg_functions *funcs = sqlite_user_data(func);
  864: 	TSRMLS_FETCH();
  865: 
  866: 	if (!funcs->is_valid) {
  867: 		sqlite_set_result_error(func, "this function has not been correctly defined for this request", -1);
  868: 		return;
  869: 	}
  870: 
  871: 	if (argc > 0) {
  872: 		zargs = (zval ***)safe_emalloc(argc, sizeof(zval **), 0);
  873: 
  874: 		for (i = 0; i < argc; i++) {
  875: 			zargs[i] = emalloc(sizeof(zval *));
  876: 			MAKE_STD_ZVAL(*zargs[i]);
  877: 
  878: 			if (argv[i] == NULL) {
  879: 				ZVAL_NULL(*zargs[i]);
  880: 			} else {
  881: 				ZVAL_STRING(*zargs[i], (char*)argv[i], 1);
  882: 			}
  883: 		}
  884: 	}
  885: 
  886: 	res = call_user_function_ex(EG(function_table),
  887: 			NULL,
  888: 			funcs->step,
  889: 			&retval,
  890: 			argc,
  891: 			zargs,
  892: 			0, NULL TSRMLS_CC);
  893: 
  894: 	if (res == SUCCESS) {
  895: 		if (retval == NULL) {
  896: 			sqlite_set_result_string(func, NULL, 0);
  897: 		} else {
  898: 			switch (Z_TYPE_P(retval)) {
  899: 				case IS_STRING:
  900: 					/* TODO: for binary results, need to encode the string */
  901: 					sqlite_set_result_string(func, Z_STRVAL_P(retval), Z_STRLEN_P(retval));
  902: 					break;
  903: 				case IS_LONG:
  904: 				case IS_BOOL:
  905: 					sqlite_set_result_int(func, Z_LVAL_P(retval));
  906: 					break;
  907: 				case IS_DOUBLE:
  908: 					sqlite_set_result_double(func, Z_DVAL_P(retval));
  909: 					break;
  910: 				case IS_NULL:
  911: 				default:
  912: 					sqlite_set_result_string(func, NULL, 0);
  913: 			}
  914: 		}
  915: 	} else {
  916: 		sqlite_set_result_error(func, "call_user_function_ex failed", -1);
  917: 	}
  918: 
  919: 	if (retval) {
  920: 		zval_ptr_dtor(&retval);
  921: 	}
  922: 
  923: 	if (zargs) {
  924: 		for (i = 0; i < argc; i++) {
  925: 			zval_ptr_dtor(zargs[i]);
  926: 			efree(zargs[i]);
  927: 		}
  928: 		efree(zargs);
  929: 	}
  930: }
  931: /* }}} */
  932: 
  933: /* {{{ callback for sqlite_create_aggregate: step function */
  934: static void php_sqlite_agg_step_function_callback(sqlite_func *func, int argc, const char **argv)
  935: {
  936: 	zval *retval = NULL;
  937: 	zval ***zargs;
  938: 	zval **context_p;
  939: 	int i, res, zargc;
  940: 	struct php_sqlite_agg_functions *funcs = sqlite_user_data(func);
  941: 	TSRMLS_FETCH();
  942: 
  943: 	if (!funcs->is_valid) {
  944: 		sqlite_set_result_error(func, "this function has not been correctly defined for this request", -1);
  945: 		return;
  946: 	}
  947: 
  948: 	/* sanity check the args */
  949: 	if (argc < 1) {
  950: 		return;
  951: 	}
  952: 
  953: 	zargc = argc + 1;
  954: 	zargs = (zval ***)safe_emalloc(zargc, sizeof(zval **), 0);
  955: 
  956: 	/* first arg is always the context zval */
  957: 	context_p = (zval **)sqlite_aggregate_context(func, sizeof(*context_p));
  958: 
  959: 	if (*context_p == NULL) {
  960: 		MAKE_STD_ZVAL(*context_p);
  961: 		Z_SET_ISREF_PP(context_p);
  962: 		Z_TYPE_PP(context_p) = IS_NULL;
  963: 	}
  964: 
  965: 	zargs[0] = context_p;
  966: 
  967: 	/* copy the other args */
  968: 	for (i = 0; i < argc; i++) {
  969: 		zargs[i+1] = emalloc(sizeof(zval *));
  970: 		MAKE_STD_ZVAL(*zargs[i+1]);
  971: 		if (argv[i] == NULL) {
  972: 			ZVAL_NULL(*zargs[i+1]);
  973: 		} else {
  974: 			ZVAL_STRING(*zargs[i+1], (char*)argv[i], 1);
  975: 		}
  976: 	}
  977: 
  978: 	res = call_user_function_ex(EG(function_table),
  979: 			NULL,
  980: 			funcs->step,
  981: 			&retval,
  982: 			zargc,
  983: 			zargs,
  984: 			0, NULL TSRMLS_CC);
  985: 
  986: 	if (res != SUCCESS) {
  987: 		php_error_docref(NULL TSRMLS_CC, E_WARNING, "call_user_function_ex failed");
  988: 	}
  989: 
  990: 	if (retval) {
  991: 		zval_ptr_dtor(&retval);
  992: 	}
  993: 
  994: 	if (zargs) {
  995: 		for (i = 1; i < zargc; i++) {
  996: 			zval_ptr_dtor(zargs[i]);
  997: 			efree(zargs[i]);
  998: 		}
  999: 		efree(zargs);
 1000: 	}
 1001: }
 1002: /* }}} */
 1003: 
 1004: /* {{{ callback for sqlite_create_aggregate: finalize function */
 1005: static void php_sqlite_agg_fini_function_callback(sqlite_func *func)
 1006: {
 1007: 	zval *retval = NULL;
 1008: 	int res;
 1009: 	struct php_sqlite_agg_functions *funcs = sqlite_user_data(func);
 1010: 	zval **context_p;
 1011: 	TSRMLS_FETCH();
 1012: 
 1013: 	if (!funcs->is_valid) {
 1014: 		sqlite_set_result_error(func, "this function has not been correctly defined for this request", -1);
 1015: 		return;
 1016: 	}
 1017: 
 1018: 	context_p = (zval **)sqlite_aggregate_context(func, sizeof(*context_p));
 1019: 
 1020: 	res = call_user_function_ex(EG(function_table),
 1021: 			NULL,
 1022: 			funcs->fini,
 1023: 			&retval,
 1024: 			1,
 1025: 			&context_p,
 1026: 			0, NULL TSRMLS_CC);
 1027: 
 1028: 	if (res == SUCCESS) {
 1029: 		if (retval == NULL) {
 1030: 			sqlite_set_result_string(func, NULL, 0);
 1031: 		} else {
 1032: 			switch (Z_TYPE_P(retval)) {
 1033: 				case IS_STRING:
 1034: 					/* TODO: for binary results, need to encode the string */
 1035: 					sqlite_set_result_string(func, Z_STRVAL_P(retval), Z_STRLEN_P(retval));
 1036: 					break;
 1037: 				case IS_LONG:
 1038: 				case IS_BOOL:
 1039: 					sqlite_set_result_int(func, Z_LVAL_P(retval));
 1040: 					break;
 1041: 				case IS_DOUBLE:
 1042: 					sqlite_set_result_double(func, Z_DVAL_P(retval));
 1043: 					break;
 1044: 				case IS_NULL:
 1045: 				default:
 1046: 					sqlite_set_result_string(func, NULL, 0);
 1047: 			}
 1048: 		}
 1049: 	} else {
 1050: 		sqlite_set_result_error(func, "call_user_function_ex failed", -1);
 1051: 	}
 1052: 
 1053: 	if (retval) {
 1054: 		zval_ptr_dtor(&retval);
 1055: 	}
 1056: 
 1057: 	zval_ptr_dtor(context_p);
 1058: }
 1059: /* }}} */
 1060: 
 1061: /* {{{ Authorization Callback */
 1062: static int php_sqlite_authorizer(void *autharg, int access_type, const char *arg3, const char *arg4,
 1063: 		const char *arg5, const char *arg6)
 1064: {
 1065: 	switch (access_type) {
 1066: 		case SQLITE_COPY:
 1067: 			if (strncmp(arg4, ":memory:", sizeof(":memory:") - 1)) {
 1068: 				TSRMLS_FETCH();
 1069: 				if (PG(safe_mode) && (!php_checkuid(arg4, NULL, CHECKUID_CHECK_FILE_AND_DIR))) {
 1070: 					return SQLITE_DENY;
 1071: 				}
 1072: 
 1073: 				if (php_check_open_basedir(arg4 TSRMLS_CC)) {
 1074: 					return SQLITE_DENY;
 1075: 				}
 1076: 			}
 1077: 			return SQLITE_OK;
 1078: #ifdef SQLITE_ATTACH
 1079: 		case SQLITE_ATTACH:
 1080: 			if (strncmp(arg3, ":memory:", sizeof(":memory:") - 1)) {
 1081: 				TSRMLS_FETCH();
 1082: 				if (PG(safe_mode) && (!php_checkuid(arg3, NULL, CHECKUID_CHECK_FILE_AND_DIR))) {
 1083: 					return SQLITE_DENY;
 1084: 				}
 1085: 
 1086: 				if (php_check_open_basedir(arg3 TSRMLS_CC)) {
 1087: 					return SQLITE_DENY;
 1088: 				}
 1089: 			}
 1090: 			return SQLITE_OK;
 1091: #endif
 1092: 
 1093: 		default:
 1094: 			/* access allowed */
 1095: 			return SQLITE_OK;
 1096: 	}
 1097: }
 1098: /* }}} */
 1099: 
 1100: /* {{{ OO init/structure stuff */
 1101: #define REGISTER_SQLITE_CLASS(name, c_name, parent) \
 1102: 	{ \
 1103: 		zend_class_entry ce; \
 1104: 		INIT_CLASS_ENTRY(ce, "SQLite" # name, sqlite_funcs_ ## c_name); \
 1105: 		ce.create_object = sqlite_object_new_ ## c_name; \
 1106: 		sqlite_ce_ ## c_name = zend_register_internal_class_ex(&ce, parent, NULL TSRMLS_CC); \
 1107: 		memcpy(&sqlite_object_handlers_ ## c_name, zend_get_std_object_handlers(), sizeof(zend_object_handlers)); \
 1108: 		sqlite_object_handlers_ ## c_name.clone_obj = NULL; \
 1109: 		sqlite_ce_ ## c_name->ce_flags |= ZEND_ACC_FINAL_CLASS; \
 1110: 	}
 1111: 
 1112: zend_class_entry *sqlite_ce_db, *sqlite_ce_exception;
 1113: zend_class_entry *sqlite_ce_query, *sqlite_ce_ub_query;
 1114: 
 1115: static zend_object_handlers sqlite_object_handlers_db;
 1116: static zend_object_handlers sqlite_object_handlers_query;
 1117: static zend_object_handlers sqlite_object_handlers_ub_query;
 1118: static zend_object_handlers sqlite_object_handlers_exception;
 1119: 
 1120: typedef enum {
 1121: 	is_db,
 1122: 	is_result
 1123: } sqlite_obj_type;
 1124: 
 1125: typedef struct _sqlite_object {
 1126: 	zend_object       std;
 1127: 	sqlite_obj_type   type;
 1128: 	union {
 1129: 		struct php_sqlite_db     *db;
 1130: 		struct php_sqlite_result *res;
 1131: 		void *ptr;
 1132: 	} u;
 1133: } sqlite_object;
 1134: 
 1135: static int sqlite_free_persistent(zend_rsrc_list_entry *le, void *ptr TSRMLS_DC)
 1136: {
 1137: 	return le->ptr == ptr ? ZEND_HASH_APPLY_REMOVE : ZEND_HASH_APPLY_KEEP;
 1138: }
 1139: 
 1140: static void sqlite_object_free_storage(void *object TSRMLS_DC)
 1141: {
 1142: 	sqlite_object *intern = (sqlite_object *)object;
 1143: 
 1144: 	zend_object_std_dtor(&intern->std TSRMLS_CC);
 1145: 
 1146: 	if (intern->u.ptr) {
 1147: 		if (intern->type == is_db) {
 1148: 			if (intern->u.db->rsrc_id) {
 1149: 				zend_list_delete(intern->u.db->rsrc_id);
 1150: 				zend_hash_apply_with_argument(&EG(persistent_list), (apply_func_arg_t) sqlite_free_persistent, &intern->u.ptr TSRMLS_CC);
 1151: 			}
 1152: 		} else {
 1153: 			real_result_dtor(intern->u.res TSRMLS_CC);
 1154: 		}
 1155: 	}
 1156: 
 1157: 	efree(object);
 1158: }
 1159: 
 1160: static void sqlite_object_new(zend_class_entry *class_type, zend_object_handlers *handlers, zend_object_value *retval TSRMLS_DC)
 1161: {
 1162: 	sqlite_object *intern;
 1163: 	zval *tmp;
 1164: 
 1165: 	intern = emalloc(sizeof(sqlite_object));
 1166: 	memset(intern, 0, sizeof(sqlite_object));
 1167: 
 1168: 	zend_object_std_init(&intern->std, class_type TSRMLS_CC);
 1169: 	zend_hash_copy(intern->std.properties, &class_type->default_properties, (copy_ctor_func_t) zval_add_ref, (void *) &tmp, sizeof(zval *));
 1170: 
 1171: 	retval->handle = zend_objects_store_put(intern, (zend_objects_store_dtor_t)zend_objects_destroy_object, (zend_objects_free_object_storage_t) sqlite_object_free_storage, NULL TSRMLS_CC);
 1172: 	retval->handlers = handlers;
 1173: }
 1174: 
 1175: static zend_object_value sqlite_object_new_db(zend_class_entry *class_type TSRMLS_DC)
 1176: {
 1177: 	zend_object_value retval;
 1178: 
 1179: 	sqlite_object_new(class_type, &sqlite_object_handlers_db, &retval TSRMLS_CC);
 1180: 	return retval;
 1181: }
 1182: 
 1183: static zend_object_value sqlite_object_new_query(zend_class_entry *class_type TSRMLS_DC)
 1184: {
 1185: 	zend_object_value retval;
 1186: 
 1187: 	sqlite_object_new(class_type, &sqlite_object_handlers_query, &retval TSRMLS_CC);
 1188: 	return retval;
 1189: }
 1190: 
 1191: static zend_object_value sqlite_object_new_ub_query(zend_class_entry *class_type TSRMLS_DC)
 1192: {
 1193: 	zend_object_value retval;
 1194: 
 1195: 	sqlite_object_new(class_type, &sqlite_object_handlers_ub_query, &retval TSRMLS_CC);
 1196: 	return retval;
 1197: }
 1198: 
 1199: static zend_object_value sqlite_object_new_exception(zend_class_entry *class_type TSRMLS_DC)
 1200: {
 1201: 	zend_object_value retval;
 1202: 
 1203: 	sqlite_object_new(class_type, &sqlite_object_handlers_exception, &retval TSRMLS_CC);
 1204: 	return retval;
 1205: }
 1206: 
 1207: #define SQLITE_REGISTER_OBJECT(_type, _object, _ptr) \
 1208: 	{ \
 1209: 		sqlite_object *obj; \
 1210: 		obj = (sqlite_object*)zend_object_store_get_object(_object TSRMLS_CC); \
 1211: 		obj->type = is_ ## _type; \
 1212: 		obj->u._type = _ptr; \
 1213: 	}
 1214: 
 1215: static zend_class_entry *sqlite_get_ce_query(const zval *object TSRMLS_DC)
 1216: {
 1217: 	return sqlite_ce_query;
 1218: }
 1219: 
 1220: static zend_class_entry *sqlite_get_ce_ub_query(const zval *object TSRMLS_DC)
 1221: {
 1222: 	return sqlite_ce_ub_query;
 1223: }
 1224: 
 1225: static zval * sqlite_instanciate(zend_class_entry *pce, zval *object TSRMLS_DC)
 1226: {
 1227: 	if (!object) {
 1228: 		ALLOC_ZVAL(object);
 1229: 	}
 1230: 	Z_TYPE_P(object) = IS_OBJECT;
 1231: 	object_init_ex(object, pce);
 1232: 	Z_SET_REFCOUNT_P(object, 1);
 1233: 	Z_SET_ISREF_P(object);
 1234: 	return object;
 1235: }
 1236: 
 1237: typedef struct _sqlite_object_iterator {
 1238: 	zend_object_iterator     it;
 1239: 	struct php_sqlite_result *res;
 1240: 	zval *value;
 1241: } sqlite_object_iterator;
 1242: 
 1243: void sqlite_iterator_dtor(zend_object_iterator *iter TSRMLS_DC)
 1244: {
 1245: 	zval *object = (zval*)((sqlite_object_iterator*)iter)->it.data;
 1246: 
 1247: 	if (((sqlite_object_iterator*)iter)->value) {
 1248: 		zval_ptr_dtor(&((sqlite_object_iterator*)iter)->value);
 1249: 		((sqlite_object_iterator*)iter)->value = NULL;
 1250: 	}
 1251: 	zval_ptr_dtor(&object);
 1252: 	efree(iter);
 1253: }
 1254: 
 1255: void sqlite_iterator_rewind(zend_object_iterator *iter TSRMLS_DC)
 1256: {
 1257: 	struct php_sqlite_result *res = ((sqlite_object_iterator*)iter)->res;
 1258: 
 1259: 	if (((sqlite_object_iterator*)iter)->value) {
 1260: 		zval_ptr_dtor(&((sqlite_object_iterator*)iter)->value);
 1261: 		((sqlite_object_iterator*)iter)->value = NULL;
 1262: 	}
 1263: 	if (res) {
 1264: 		res->curr_row = 0;
 1265: 	}
 1266: }
 1267: 
 1268: int sqlite_iterator_valid(zend_object_iterator *iter TSRMLS_DC)
 1269: {
 1270: 	struct php_sqlite_result *res = ((sqlite_object_iterator*)iter)->res;
 1271: 
 1272: 	if (res && res->curr_row < res->nrows && res->nrows) { /* curr_row may be -1 */
 1273: 		return SUCCESS;
 1274: 	} else {
 1275: 		return FAILURE;
 1276: 	}
 1277: }
 1278: 
 1279: void sqlite_iterator_get_current_data(zend_object_iterator *iter, zval ***data TSRMLS_DC)
 1280: {
 1281: 	struct php_sqlite_result *res = ((sqlite_object_iterator*)iter)->res;
 1282: 
 1283: 	*data = &((sqlite_object_iterator*)iter)->value;
 1284: 	if (res && !**data) {
 1285: 		MAKE_STD_ZVAL(**data);
 1286: 		php_sqlite_fetch_array(res, res->mode, 1, 0, **data TSRMLS_CC);
 1287: 	}
 1288: 
 1289: }
 1290: 
 1291: int sqlite_iterator_get_current_key(zend_object_iterator *iter, char **str_key, uint *str_key_len, ulong *int_key TSRMLS_DC)
 1292: {
 1293: 	struct php_sqlite_result *res = ((sqlite_object_iterator*)iter)->res;
 1294: 
 1295: 	*str_key = NULL;
 1296: 	*str_key_len = 0;
 1297: 	*int_key = res ? res->curr_row : 0;
 1298: 	return HASH_KEY_IS_LONG;
 1299: }
 1300: 
 1301: void sqlite_iterator_move_forward(zend_object_iterator *iter TSRMLS_DC)
 1302: {
 1303: 	struct php_sqlite_result *res = ((sqlite_object_iterator*)iter)->res;
 1304: 
 1305: 	if (((sqlite_object_iterator*)iter)->value) {
 1306: 		zval_ptr_dtor(&((sqlite_object_iterator*)iter)->value);
 1307: 		((sqlite_object_iterator*)iter)->value = NULL;
 1308: 	}
 1309: 	if (res) {
 1310: 		if (!res->buffered && res->vm) {
 1311: 			php_sqlite_fetch(res TSRMLS_CC);
 1312: 		}
 1313: 		if (res->curr_row >= res->nrows) {
 1314: 			/* php_error_docref(NULL TSRMLS_CC, E_WARNING, "no more rows available"); */
 1315: 			return;
 1316: 		}
 1317: 
 1318: 		res->curr_row++;
 1319: 	}
 1320: }
 1321: 
 1322: zend_object_iterator_funcs sqlite_ub_query_iterator_funcs = {
 1323: 	sqlite_iterator_dtor,
 1324: 	sqlite_iterator_valid,
 1325: 	sqlite_iterator_get_current_data,
 1326: 	sqlite_iterator_get_current_key,
 1327: 	sqlite_iterator_move_forward,
 1328: 	NULL
 1329: };
 1330: 
 1331: zend_object_iterator_funcs sqlite_query_iterator_funcs = {
 1332: 	sqlite_iterator_dtor,
 1333: 	sqlite_iterator_valid,
 1334: 	sqlite_iterator_get_current_data,
 1335: 	sqlite_iterator_get_current_key,
 1336: 	sqlite_iterator_move_forward,
 1337: 	sqlite_iterator_rewind
 1338: };
 1339: 
 1340: zend_object_iterator *sqlite_get_iterator(zend_class_entry *ce, zval *object, int by_ref TSRMLS_DC)
 1341: {
 1342: 	sqlite_object_iterator *iterator = emalloc(sizeof(sqlite_object_iterator));
 1343: 
 1344: 	sqlite_object *obj = (sqlite_object*) zend_object_store_get_object(object TSRMLS_CC);
 1345: 
 1346: 	if (by_ref) {
 1347: 		zend_error(E_RECOVERABLE_ERROR, "An iterator cannot be used with foreach by reference");
 1348: 	}
 1349: 	Z_ADDREF_P(object);
 1350: 	iterator->it.data = (void*)object;
 1351: 	iterator->it.funcs = ce->iterator_funcs.funcs;
 1352: 	iterator->res = obj->u.res;
 1353: 	iterator->value = NULL;
 1354: 	return (zend_object_iterator*)iterator;
 1355: }
 1356: /* }}} */
 1357: 
 1358: static PHP_GINIT_FUNCTION(sqlite)
 1359: {
 1360: 	sqlite_globals->assoc_case = 0;
 1361: }
 1362: 
 1363: PHP_MINIT_FUNCTION(sqlite)
 1364: {
 1365: 	REGISTER_SQLITE_CLASS(Database,   db,        NULL);
 1366: 	REGISTER_SQLITE_CLASS(Result,     query,     NULL);
 1367: 	REGISTER_SQLITE_CLASS(Unbuffered, ub_query,  NULL);
 1368: #if defined(HAVE_SPL) && ((PHP_MAJOR_VERSION > 5) || (PHP_MAJOR_VERSION == 5 && PHP_MINOR_VERSION >= 1))
 1369: 	REGISTER_SQLITE_CLASS(Exception,  exception, spl_ce_RuntimeException);
 1370: #else
 1371: 	REGISTER_SQLITE_CLASS(Exception,  exception, zend_exception_get_default(TSRMLS_C));
 1372: #endif
 1373: 
 1374: 	sqlite_ce_db->ce_flags &= ~ZEND_ACC_FINAL_CLASS;
 1375: 	sqlite_ce_db->constructor->common.fn_flags |= ZEND_ACC_FINAL;
 1376: 
 1377: 	sqlite_object_handlers_query.get_class_entry = sqlite_get_ce_query;
 1378: 	sqlite_object_handlers_ub_query.get_class_entry = sqlite_get_ce_ub_query;
 1379: 	sqlite_object_handlers_ub_query.count_elements = sqlite_count_elements;
 1380: 
 1381: 	sqlite_ce_ub_query->get_iterator = sqlite_get_iterator;
 1382: 	sqlite_ce_ub_query->iterator_funcs.funcs = &sqlite_ub_query_iterator_funcs;
 1383: 
 1384: #if defined(HAVE_SPL) && ((PHP_MAJOR_VERSION > 5) || (PHP_MAJOR_VERSION == 5 && PHP_MINOR_VERSION >= 1))
 1385: 	zend_class_implements(sqlite_ce_query TSRMLS_CC, 2, zend_ce_iterator, spl_ce_Countable);
 1386: #else
 1387: 	zend_class_implements(sqlite_ce_query TSRMLS_CC, 1, zend_ce_iterator);
 1388: #endif
 1389: 	sqlite_ce_query->get_iterator = sqlite_get_iterator;
 1390: 	sqlite_ce_query->iterator_funcs.funcs = &sqlite_query_iterator_funcs;
 1391: 
 1392: 	REGISTER_INI_ENTRIES();
 1393: 
 1394: #if HAVE_PHP_SESSION && !defined(COMPILE_DL_SESSION)
 1395: 	php_session_register_module(ps_sqlite_ptr);
 1396: #endif
 1397: 
 1398: 	le_sqlite_db = zend_register_list_destructors_ex(php_sqlite_db_dtor, NULL, "sqlite database", module_number);
 1399: 	le_sqlite_pdb = zend_register_list_destructors_ex(NULL, php_sqlite_db_dtor, "sqlite database (persistent)", module_number);
 1400: 	le_sqlite_result = zend_register_list_destructors_ex(php_sqlite_result_dtor, NULL, "sqlite result", module_number);
 1401: 
 1402: 	REGISTER_LONG_CONSTANT("SQLITE_BOTH",	PHPSQLITE_BOTH, CONST_CS|CONST_PERSISTENT);
 1403: 	REGISTER_LONG_CONSTANT("SQLITE_NUM",	PHPSQLITE_NUM, CONST_CS|CONST_PERSISTENT);
 1404: 	REGISTER_LONG_CONSTANT("SQLITE_ASSOC",	PHPSQLITE_ASSOC, CONST_CS|CONST_PERSISTENT);
 1405: 
 1406: 	REGISTER_LONG_CONSTANT("SQLITE_OK",				SQLITE_OK, CONST_CS|CONST_PERSISTENT);
 1407: 	REGISTER_LONG_CONSTANT("SQLITE_ERROR",			SQLITE_ERROR, CONST_CS|CONST_PERSISTENT);
 1408: 	REGISTER_LONG_CONSTANT("SQLITE_INTERNAL",		SQLITE_INTERNAL, CONST_CS|CONST_PERSISTENT);
 1409: 	REGISTER_LONG_CONSTANT("SQLITE_PERM",			SQLITE_PERM, CONST_CS|CONST_PERSISTENT);
 1410: 	REGISTER_LONG_CONSTANT("SQLITE_ABORT",			SQLITE_ABORT, CONST_CS|CONST_PERSISTENT);
 1411: 	REGISTER_LONG_CONSTANT("SQLITE_BUSY",			SQLITE_BUSY, CONST_CS|CONST_PERSISTENT);
 1412: 	REGISTER_LONG_CONSTANT("SQLITE_LOCKED",			SQLITE_LOCKED, CONST_CS|CONST_PERSISTENT);
 1413: 	REGISTER_LONG_CONSTANT("SQLITE_NOMEM",			SQLITE_NOMEM, CONST_CS|CONST_PERSISTENT);
 1414: 	REGISTER_LONG_CONSTANT("SQLITE_READONLY",		SQLITE_READONLY, CONST_CS|CONST_PERSISTENT);
 1415: 	REGISTER_LONG_CONSTANT("SQLITE_INTERRUPT",		SQLITE_INTERRUPT, CONST_CS|CONST_PERSISTENT);
 1416: 	REGISTER_LONG_CONSTANT("SQLITE_IOERR",			SQLITE_IOERR, CONST_CS|CONST_PERSISTENT);
 1417: 	REGISTER_LONG_CONSTANT("SQLITE_CORRUPT",		SQLITE_CORRUPT, CONST_CS|CONST_PERSISTENT);
 1418: 	REGISTER_LONG_CONSTANT("SQLITE_NOTFOUND",		SQLITE_NOTFOUND, CONST_CS|CONST_PERSISTENT);
 1419: 	REGISTER_LONG_CONSTANT("SQLITE_FULL",			SQLITE_FULL, CONST_CS|CONST_PERSISTENT);
 1420: 	REGISTER_LONG_CONSTANT("SQLITE_CANTOPEN",		SQLITE_CANTOPEN, CONST_CS|CONST_PERSISTENT);
 1421: 	REGISTER_LONG_CONSTANT("SQLITE_PROTOCOL",		SQLITE_PROTOCOL, CONST_CS|CONST_PERSISTENT);
 1422: 	REGISTER_LONG_CONSTANT("SQLITE_EMPTY",			SQLITE_EMPTY, CONST_CS|CONST_PERSISTENT);
 1423: 	REGISTER_LONG_CONSTANT("SQLITE_SCHEMA",			SQLITE_SCHEMA, CONST_CS|CONST_PERSISTENT);
 1424: 	REGISTER_LONG_CONSTANT("SQLITE_TOOBIG",			SQLITE_TOOBIG, CONST_CS|CONST_PERSISTENT);
 1425: 	REGISTER_LONG_CONSTANT("SQLITE_CONSTRAINT",		SQLITE_CONSTRAINT, CONST_CS|CONST_PERSISTENT);
 1426: 	REGISTER_LONG_CONSTANT("SQLITE_MISMATCH",		SQLITE_MISMATCH, CONST_CS|CONST_PERSISTENT);
 1427: 	REGISTER_LONG_CONSTANT("SQLITE_MISUSE",			SQLITE_MISUSE, CONST_CS|CONST_PERSISTENT);
 1428: 	REGISTER_LONG_CONSTANT("SQLITE_NOLFS",			SQLITE_NOLFS, CONST_CS|CONST_PERSISTENT);
 1429: 	REGISTER_LONG_CONSTANT("SQLITE_AUTH",			SQLITE_AUTH, CONST_CS|CONST_PERSISTENT);
 1430: 	REGISTER_LONG_CONSTANT("SQLITE_NOTADB",			SQLITE_NOTADB, CONST_CS|CONST_PERSISTENT);
 1431: #ifdef SQLITE_FORMAT
 1432: 	REGISTER_LONG_CONSTANT("SQLITE_FORMAT",			SQLITE_FORMAT, CONST_CS|CONST_PERSISTENT);
 1433: #endif
 1434: 	REGISTER_LONG_CONSTANT("SQLITE_ROW",			SQLITE_ROW, CONST_CS|CONST_PERSISTENT);
 1435: 	REGISTER_LONG_CONSTANT("SQLITE_DONE",			SQLITE_DONE, CONST_CS|CONST_PERSISTENT);
 1436: 
 1437: #ifdef PHP_SQLITE2_HAVE_PDO
 1438:     if (FAILURE == php_pdo_register_driver(&pdo_sqlite2_driver)) {
 1439: 	return FAILURE;
 1440:     }
 1441: #endif
 1442: 
 1443: 	return SUCCESS;
 1444: }
 1445: 
 1446: PHP_MSHUTDOWN_FUNCTION(sqlite)
 1447: {
 1448: 	UNREGISTER_INI_ENTRIES();
 1449: 
 1450: #ifdef PHP_SQLITE2_HAVE_PDO
 1451:     php_pdo_unregister_driver(&pdo_sqlite2_driver);
 1452: #endif
 1453: 
 1454: 	return SUCCESS;
 1455: }
 1456: 
 1457: PHP_MINFO_FUNCTION(sqlite)
 1458: {
 1459: 	php_info_print_table_start();
 1460: 	php_info_print_table_header(2, "SQLite support", "enabled");
 1461: 	php_info_print_table_row(2, "PECL Module version", PHP_SQLITE_MODULE_VERSION " $Id: sqlite.c,v 1.1.1.1 2012/02/21 23:48:02 misho Exp $");
 1462: 	php_info_print_table_row(2, "SQLite Library", sqlite_libversion());
 1463: 	php_info_print_table_row(2, "SQLite Encoding", sqlite_libencoding());
 1464: 	php_info_print_table_end();
 1465: 
 1466: 	DISPLAY_INI_ENTRIES();
 1467: }
 1468: 
 1469: static struct php_sqlite_db *php_sqlite_open(char *filename, int mode, char *persistent_id, zval *return_value, zval *errmsg, zval *object TSRMLS_DC)
 1470: {
 1471: 	char *errtext = NULL;
 1472: 	sqlite *sdb = NULL;
 1473: 	struct php_sqlite_db *db = NULL;
 1474: 
 1475: 	sdb = sqlite_open(filename, mode, &errtext);
 1476: 
 1477: 	if (sdb == NULL) {
 1478: 
 1479: 		php_error_docref(NULL TSRMLS_CC, E_WARNING, "%s", errtext);
 1480: 
 1481: 		if (errmsg) {
 1482: 			ZVAL_STRING(errmsg, errtext, 1);
 1483: 		}
 1484: 
 1485: 		sqlite_freemem(errtext);
 1486: 
 1487: 		/* if object is not an object then we're called from the factory() function */
 1488: 		if (object && Z_TYPE_P(object) != IS_OBJECT) {
 1489: 			RETVAL_NULL();
 1490: 		} else {
 1491: 			RETVAL_FALSE;
 1492: 		}
 1493: 		return NULL;
 1494: 	}
 1495: 
 1496: 	db = (struct php_sqlite_db *)pemalloc(sizeof(struct php_sqlite_db), persistent_id ? 1 : 0);
 1497: 	db->is_persistent = persistent_id ? 1 : 0;
 1498: 	db->last_err_code = SQLITE_OK;
 1499: 	db->db = sdb;
 1500: 
 1501: 	zend_hash_init(&db->callbacks, 0, NULL, php_sqlite_callback_dtor, db->is_persistent);
 1502: 
 1503: 	/* register the PHP functions */
 1504: 	sqlite_create_function(sdb, "php", -1, php_sqlite_generic_function_callback, 0);
 1505: 
 1506: 	/* set default busy handler; keep retrying up until 1 minute has passed,
 1507: 	 * then fail with a busy status code */
 1508: 	sqlite_busy_timeout(sdb, 60000);
 1509: 
 1510: 	/* authorizer hook so we can enforce safe mode
 1511: 	 * Note: the declaration of php_sqlite_authorizer is correct for 2.8.2 of libsqlite,
 1512: 	 * and IS backwards binary compatible with earlier versions */
 1513: 	if (PG(safe_mode) || (PG(open_basedir) && *PG(open_basedir))) {
 1514: 		sqlite_set_authorizer(sdb, php_sqlite_authorizer, NULL);
 1515: 	}
 1516: 
 1517: 	db->rsrc_id = ZEND_REGISTER_RESOURCE(object ? NULL : return_value, db, persistent_id ? le_sqlite_pdb : le_sqlite_db);
 1518: 	if (object) {
 1519: 		/* if object is not an object then we're called from the factory() function */
 1520: 		if (Z_TYPE_P(object) != IS_OBJECT) {
 1521: 			sqlite_instanciate(sqlite_ce_db, object TSRMLS_CC);
 1522: 		}
 1523: 		/* and now register the object */
 1524: 		SQLITE_REGISTER_OBJECT(db, object, db)
 1525: 	}
 1526: 
 1527: 	if (persistent_id) {
 1528: 		zend_rsrc_list_entry le;
 1529: 
 1530: 		Z_TYPE(le) = le_sqlite_pdb;
 1531: 		le.ptr = db;
 1532: 
 1533: 		if (FAILURE == zend_hash_update(&EG(persistent_list), persistent_id,
 1534: 					strlen(persistent_id)+1,
 1535: 					(void *)&le, sizeof(le), NULL)) {
 1536: 			php_error_docref(NULL TSRMLS_CC, E_WARNING, "Failed to register persistent resource");
 1537: 		}
 1538: 	}
 1539: 
 1540: 	return db;
 1541: }
 1542: 
 1543: /* {{{ proto resource sqlite_popen(string filename [, int mode [, string &error_message]])
 1544:    Opens a persistent handle to a SQLite database. Will create the database if it does not exist. */
 1545: PHP_FUNCTION(sqlite_popen)
 1546: {
 1547: 	long mode = 0666;
 1548: 	char *filename, *fullpath, *hashkey;
 1549: 	int filename_len, hashkeylen;
 1550: 	zval *errmsg = NULL;
 1551: 	struct php_sqlite_db *db = NULL;
 1552: 	zend_rsrc_list_entry *le;
 1553: 
 1554: 	if (FAILURE == zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "s|lz/",
 1555: 				&filename, &filename_len, &mode, &errmsg)) {
 1556: 		return;
 1557: 	}
 1558: 	if (errmsg) {
 1559: 		zval_dtor(errmsg);
 1560: 		ZVAL_NULL(errmsg);
 1561: 	}
 1562: 
 1563: 	if (strlen(filename) != filename_len) {
 1564: 		RETURN_FALSE;
 1565: 	}
 1566: 	if (strncmp(filename, ":memory:", sizeof(":memory:") - 1)) {
 1567: 		/* resolve the fully-qualified path name to use as the hash key */
 1568: 		if (!(fullpath = expand_filepath(filename, NULL TSRMLS_CC))) {
 1569: 			RETURN_FALSE;
 1570: 		}
 1571: 
 1572: 		if ((PG(safe_mode) && (!php_checkuid(fullpath, NULL, CHECKUID_CHECK_FILE_AND_DIR))) || 
 1573: 				php_check_open_basedir(fullpath TSRMLS_CC)) {
 1574: 			efree(fullpath);
 1575: 			RETURN_FALSE;
 1576: 		}
 1577: 	} else {
 1578: 		fullpath = estrndup(filename, filename_len);
 1579: 	}
 1580: 
 1581: 	hashkeylen = spprintf(&hashkey, 0, "sqlite_pdb_%s:%ld", fullpath, mode);
 1582: 
 1583: 	/* do we have an existing persistent connection ? */
 1584: 	if (SUCCESS == zend_hash_find(&EG(persistent_list), hashkey, hashkeylen+1, (void*)&le)) {
 1585: 		if (Z_TYPE_P(le) == le_sqlite_pdb) {
 1586: 			db = (struct php_sqlite_db*)le->ptr;
 1587: 
 1588: 			if (db->rsrc_id == FAILURE) {
 1589: 				/* give it a valid resource id for this request */
 1590: 				db->rsrc_id = ZEND_REGISTER_RESOURCE(return_value, db, le_sqlite_pdb);
 1591: 			} else {
 1592: 				int type;
 1593: 				/* sanity check to ensure that the resource is still a valid regular resource
 1594: 				 * number */
 1595: 				if (zend_list_find(db->rsrc_id, &type) == db) {
 1596: 					/* already accessed this request; map it */
 1597: 					zend_list_addref(db->rsrc_id);
 1598: 					ZVAL_RESOURCE(return_value, db->rsrc_id);
 1599: 				} else {
 1600: 					db->rsrc_id = ZEND_REGISTER_RESOURCE(return_value, db, le_sqlite_pdb);
 1601: 				}
 1602: 			}
 1603: 
 1604: 			/* all set */
 1605: 			goto done;
 1606: 		}
 1607: 
 1608: 		php_error_docref(NULL TSRMLS_CC, E_WARNING, "Some other type of persistent resource is using this hash key!?");
 1609: 		RETVAL_FALSE;
 1610: 		goto done;
 1611: 	}
 1612: 
 1613: 	/* now we need to open the database */
 1614: 	php_sqlite_open(fullpath, (int)mode, hashkey, return_value, errmsg, NULL TSRMLS_CC);
 1615: done:
 1616: 	efree(fullpath);
 1617: 	efree(hashkey);
 1618: }
 1619: /* }}} */
 1620: 
 1621: /* {{{ proto resource sqlite_open(string filename [, int mode [, string &error_message]])
 1622:    Opens a SQLite database. Will create the database if it does not exist. */
 1623: PHP_FUNCTION(sqlite_open)
 1624: {
 1625: 	long mode = 0666;
 1626: 	char *filename, *fullpath = NULL;
 1627: 	int filename_len;
 1628: 	zval *errmsg = NULL;
 1629: 	zval *object = getThis();
 1630: 	zend_error_handling error_handling;
 1631: 
 1632: 	zend_replace_error_handling(object ? EH_THROW : EH_NORMAL, sqlite_ce_exception, &error_handling TSRMLS_CC);
 1633: 	if (FAILURE == zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "s|lz/",
 1634: 				&filename, &filename_len, &mode, &errmsg)) {
 1635: 		zend_restore_error_handling(&error_handling TSRMLS_CC);
 1636: 		return;
 1637: 	}
 1638: 	if (errmsg) {
 1639: 		zval_dtor(errmsg);
 1640: 		ZVAL_NULL(errmsg);
 1641: 	}
 1642: 
 1643: 	if (strlen(filename) != filename_len) {
 1644: 		zend_restore_error_handling(&error_handling TSRMLS_CC);
 1645: 		RETURN_FALSE;
 1646: 	}
 1647: 
 1648: 	if (strncmp(filename, ":memory:", sizeof(":memory:") - 1)) {
 1649: 		/* resolve the fully-qualified path name to use as the hash key */
 1650: 		if (!(fullpath = expand_filepath(filename, NULL TSRMLS_CC))) {
 1651: 			zend_restore_error_handling(&error_handling TSRMLS_CC);
 1652: 			if (object) {
 1653: 				RETURN_NULL();
 1654: 			} else {
 1655: 				RETURN_FALSE;
 1656: 			}
 1657: 		}
 1658: 
 1659: 		if ((PG(safe_mode) && (!php_checkuid(fullpath, NULL, CHECKUID_CHECK_FILE_AND_DIR))) ||
 1660: 				php_check_open_basedir(fullpath TSRMLS_CC)) {
 1661: 			efree(fullpath);
 1662: 			zend_restore_error_handling(&error_handling TSRMLS_CC);
 1663: 			if (object) {
 1664: 				RETURN_NULL();
 1665: 			} else {
 1666: 				RETURN_FALSE;
 1667: 			}
 1668: 		}
 1669: 	}
 1670: 
 1671: 	php_sqlite_open(fullpath ? fullpath : filename, (int)mode, NULL, return_value, errmsg, object TSRMLS_CC);
 1672: 
 1673: 	if (fullpath) {
 1674: 		efree(fullpath);
 1675: 	}
 1676: 	zend_restore_error_handling(&error_handling TSRMLS_CC);
 1677: }
 1678: /* }}} */
 1679: 
 1680: /* {{{ proto object sqlite_factory(string filename [, int mode [, string &error_message]])
 1681:    Opens a SQLite database and creates an object for it. Will create the database if it does not exist. */
 1682: PHP_FUNCTION(sqlite_factory)
 1683: {
 1684: 	long mode = 0666;
 1685: 	char *filename, *fullpath = NULL;
 1686: 	int filename_len;
 1687: 	zval *errmsg = NULL;
 1688: 	zend_error_handling error_handling;
 1689: 
 1690: 	zend_replace_error_handling(EH_THROW, sqlite_ce_exception, &error_handling TSRMLS_CC);
 1691: 	if (FAILURE == zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "s|lz/",
 1692: 				&filename, &filename_len, &mode, &errmsg)) {
 1693: 		zend_restore_error_handling(&error_handling TSRMLS_CC);
 1694: 		RETURN_NULL();
 1695: 	}
 1696: 	if (errmsg) {
 1697: 		zval_dtor(errmsg);
 1698: 		ZVAL_NULL(errmsg);
 1699: 	}
 1700: 
 1701: 	if (strlen(filename) != filename_len) {
 1702: 		zend_restore_error_handling(&error_handling TSRMLS_CC);
 1703: 		RETURN_FALSE;
 1704: 	}
 1705: 
 1706: 	if (strncmp(filename, ":memory:", sizeof(":memory:") - 1)) {
 1707: 		/* resolve the fully-qualified path name to use as the hash key */
 1708: 		if (!(fullpath = expand_filepath(filename, NULL TSRMLS_CC))) {
 1709: 			zend_restore_error_handling(&error_handling TSRMLS_CC);
 1710: 			RETURN_NULL();
 1711: 		}
 1712: 
 1713: 		if ((PG(safe_mode) && (!php_checkuid(fullpath, NULL, CHECKUID_CHECK_FILE_AND_DIR))) ||
 1714: 				php_check_open_basedir(fullpath TSRMLS_CC)) {
 1715: 			efree(fullpath);
 1716: 			zend_restore_error_handling(&error_handling TSRMLS_CC);
 1717: 			RETURN_NULL();
 1718: 		}
 1719: 	}
 1720: 
 1721: 	php_sqlite_open(fullpath ? fullpath : filename, (int)mode, NULL, return_value, errmsg, return_value TSRMLS_CC);
 1722: 	if (fullpath) {
 1723: 		efree(fullpath);
 1724: 	}
 1725: 	zend_restore_error_handling(&error_handling TSRMLS_CC);
 1726: }
 1727: /* }}} */
 1728: 
 1729: /* {{{ proto void sqlite_busy_timeout(resource db, int ms)
 1730:    Set busy timeout duration. If ms <= 0, all busy handlers are disabled. */
 1731: PHP_FUNCTION(sqlite_busy_timeout)
 1732: {
 1733: 	zval *zdb;
 1734: 	struct php_sqlite_db *db;
 1735: 	long ms;
 1736: 	zval *object = getThis();
 1737: 
 1738: 	if (object) {
 1739: 		if (FAILURE == zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "l", &ms)) {
 1740: 			return;
 1741: 		}
 1742: 		DB_FROM_OBJECT(db, object);
 1743: 	} else {
 1744: 		if (FAILURE == zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "rl", &zdb, &ms)) {
 1745: 			return;
 1746: 		}
 1747: 		DB_FROM_ZVAL(db, &zdb);
 1748: 	}
 1749: 
 1750: 	sqlite_busy_timeout(db->db, ms);
 1751: }
 1752: /* }}} */
 1753: 
 1754: /* {{{ proto void sqlite_close(resource db)
 1755:    Closes an open sqlite database. */
 1756: PHP_FUNCTION(sqlite_close)
 1757: {
 1758: 	zval *zdb;
 1759: 	struct php_sqlite_db *db;
 1760: 	zval *object = getThis();
 1761: 
 1762: 	if (object) {
 1763: 		php_error_docref(NULL TSRMLS_CC, E_NOTICE, "Ignored, you must destruct the object instead");
 1764: 	} else {
 1765: 		if (FAILURE == zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "r", &zdb)) {
 1766: 			return;
 1767: 		}
 1768: 		DB_FROM_ZVAL(db, &zdb);
 1769: 	}
 1770: 
 1771: 	zend_hash_apply_with_argument(&EG(regular_list),
 1772: 		(apply_func_arg_t) _clean_unfinished_results,
 1773: 		db TSRMLS_CC);
 1774: 
 1775: 	zend_list_delete(Z_RESVAL_P(zdb));
 1776: }
 1777: /* }}} */
 1778: 
 1779: /* {{{ php_sqlite_fetch */
 1780: static int php_sqlite_fetch(struct php_sqlite_result *rres TSRMLS_DC)
 1781: {
 1782: 	const char **rowdata, **colnames;
 1783: 	int ret, i, base;
 1784: 	char *errtext = NULL;
 1785: 
 1786: next_row:
 1787: 	ret = sqlite_step(rres->vm, &rres->ncolumns, &rowdata, &colnames);
 1788: 	if (!rres->nrows) {
 1789: 		/* first row - lets copy the column names */
 1790: 		rres->col_names = safe_emalloc(rres->ncolumns, sizeof(char *), 0);
 1791: 		for (i = 0; i < rres->ncolumns; i++) {
 1792: 			rres->col_names[i] = estrdup((char*)colnames[i]);
 1793: 
 1794: 			if (SQLITE_G(assoc_case) == 1) {
 1795: 				php_sqlite_strtoupper(rres->col_names[i]);
 1796: 			} else if (SQLITE_G(assoc_case) == 2) {
 1797: 				php_sqlite_strtolower(rres->col_names[i]);
 1798: 			}
 1799: 		}
 1800: 		if (!rres->buffered) {
 1801: 			/* non buffered mode - also fetch memory for on single row */
 1802: 			rres->table = safe_emalloc(rres->ncolumns, sizeof(char *), 0);
 1803: 		}
 1804: 	}
 1805: 
 1806: 	switch (ret) {
 1807: 		case SQLITE_ROW:
 1808: 			if (rres->buffered) {
 1809: 				/* add the row to our collection */
 1810: 				if (rres->nrows + 1 >= rres->alloc_rows) {
 1811: 					rres->alloc_rows = rres->alloc_rows ? rres->alloc_rows * 2 : 16;
 1812: 					rres->table = safe_erealloc(rres->table, rres->alloc_rows, rres->ncolumns*sizeof(char *), 0);
 1813: 				}
 1814: 				base = rres->nrows * rres->ncolumns;
 1815: 				for (i = 0; i < rres->ncolumns; i++) {
 1816: 					if (rowdata[i]) {
 1817: 						rres->table[base + i] = estrdup(rowdata[i]);
 1818: 					} else {
 1819: 						rres->table[base + i] = NULL;
 1820: 					}
 1821: 				}
 1822: 				rres->nrows++;
 1823: 				goto next_row;
 1824: 			} else {
 1825: 				/* non buffered: only fetch one row but first free data if not first row */
 1826: 				if (rres->nrows++) {
 1827: 					for (i = 0; i < rres->ncolumns; i++) {
 1828: 						if (rres->table[i]) {
 1829: 							efree(rres->table[i]);
 1830: 						}
 1831: 					}
 1832: 				}
 1833: 				for (i = 0; i < rres->ncolumns; i++) {
 1834: 					if (rowdata[i]) {
 1835: 						rres->table[i] = estrdup(rowdata[i]);
 1836: 					} else {
 1837: 						rres->table[i] = NULL;
 1838: 					}
 1839: 				}
 1840: 			}
 1841: 			ret = SQLITE_OK;
 1842: 			break;
 1843: 
 1844: 		case SQLITE_BUSY:
 1845: 		case SQLITE_ERROR:
 1846: 		case SQLITE_MISUSE:
 1847: 		case SQLITE_DONE:
 1848: 		default:
 1849: 			if (rres->vm) {
 1850: 				ret = sqlite_finalize(rres->vm, &errtext);
 1851: 			}
 1852: 			rres->vm = NULL;
 1853: 			if (ret != SQLITE_OK) {
 1854: 				php_error_docref(NULL TSRMLS_CC, E_WARNING, "%s", errtext);
 1855: 				sqlite_freemem(errtext);
 1856: 			}
 1857: 			break;
 1858: 	}
 1859: 	rres->db->last_err_code = ret;
 1860: 
 1861: 	return ret;
 1862: }
 1863: /* }}} */
 1864: 
 1865: /* {{{ sqlite_query */
 1866: void sqlite_query(zval *object, struct php_sqlite_db *db, char *sql, long sql_len, int mode, int buffered, zval *return_value, struct php_sqlite_result **prres, zval *errmsg TSRMLS_DC)
 1867: {
 1868: 	struct php_sqlite_result res, *rres;
 1869: 	int ret;
 1870: 	char *errtext = NULL;
 1871: 	const char *tail;
 1872: 
 1873: 	memset(&res, 0, sizeof(res));
 1874: 	res.buffered = buffered;
 1875: 	res.mode = mode;
 1876: 
 1877: 	ret = sqlite_compile(db->db, sql, &tail, &res.vm, &errtext);
 1878: 	db->last_err_code = ret;
 1879: 
 1880: 	if (ret != SQLITE_OK) {
 1881: 		php_error_docref(NULL TSRMLS_CC, E_WARNING, "%s", errtext);
 1882: 		if (errmsg) {
 1883: 			ZVAL_STRING(errmsg, errtext, 1);
 1884: 		}
 1885: 		sqlite_freemem(errtext);
 1886: 		goto terminate;
 1887: 	} else if (!res.vm) { /* empty query */
 1888: terminate:
 1889: 		if (return_value) {
 1890: 			RETURN_FALSE;
 1891: 		} else {
 1892: 			return;
 1893: 		}
 1894: 	}
 1895: 
 1896: 	if (!prres) {
 1897: 		rres = NULL;
 1898: 		prres = &rres;
 1899: 	}
 1900: 	if (!*prres) {
 1901: 		*prres = (struct php_sqlite_result*)emalloc(sizeof(**prres));
 1902: 	}
 1903: 	memcpy(*prres, &res, sizeof(**prres));
 1904: 	(*prres)->db = db;
 1905: 	zend_list_addref(db->rsrc_id);
 1906: 
 1907: 
 1908: 	/* now the result set is ready for stepping: get first row */
 1909: 	if (php_sqlite_fetch((*prres) TSRMLS_CC) != SQLITE_OK) {
 1910: 		real_result_dtor((*prres) TSRMLS_CC);
 1911: 		*prres = NULL;
 1912: 		if (return_value) {
 1913: 			RETURN_FALSE;
 1914: 		} else {
 1915: 			return;
 1916: 		}
 1917: 	}
 1918: 
 1919: 	(*prres)->curr_row = 0;
 1920: 
 1921: 	if (object) {
 1922: 		sqlite_object *obj;
 1923: 		if (buffered) {
 1924: 			sqlite_instanciate(sqlite_ce_query, return_value TSRMLS_CC);
 1925: 		} else {
 1926: 			sqlite_instanciate(sqlite_ce_ub_query, return_value TSRMLS_CC);
 1927: 		}
 1928: 		obj = (sqlite_object *) zend_object_store_get_object(return_value TSRMLS_CC);
 1929: 		obj->type = is_result;
 1930: 		obj->u.res = (*prres);
 1931: 	} else if (return_value) {
 1932: 		ZEND_REGISTER_RESOURCE(object ? NULL : return_value, (*prres), le_sqlite_result);
 1933: 	}
 1934: }
 1935: /* }}} */
 1936: 
 1937: /* {{{ proto resource sqlite_unbuffered_query(string query, resource db [ , int result_type [, string &error_message]])
 1938:    Executes a query that does not prefetch and buffer all data. */
 1939: PHP_FUNCTION(sqlite_unbuffered_query)
 1940: {
 1941: 	zval *zdb;
 1942: 	struct php_sqlite_db *db;
 1943: 	char *sql;
 1944: 	int sql_len;
 1945: 	long mode = PHPSQLITE_BOTH;
 1946: 	char *errtext = NULL;
 1947: 	zval *errmsg = NULL;
 1948: 	zval *object = getThis();
 1949: 
 1950: 	if (object) {
 1951: 		if (FAILURE == zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "s|lz/", &sql, &sql_len, &mode, &errmsg)) {
 1952: 			return;
 1953: 		}
 1954: 		DB_FROM_OBJECT(db, object);
 1955: 	} else {
 1956: 		if (FAILURE == zend_parse_parameters_ex(ZEND_PARSE_PARAMS_QUIET,
 1957: 				ZEND_NUM_ARGS() TSRMLS_CC, "sr|lz/", &sql, &sql_len, &zdb, &mode, &errmsg) &&
 1958: 			FAILURE == zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "rs|lz/", &zdb, &sql, &sql_len, &mode, &errmsg)) {
 1959: 			return;
 1960: 		}
 1961: 		DB_FROM_ZVAL(db, &zdb);
 1962: 	}
 1963: 
 1964: 	if (errmsg) {
 1965: 		zval_dtor(errmsg);
 1966: 		ZVAL_NULL(errmsg);
 1967: 	}
 1968: 
 1969: 	PHP_SQLITE_EMPTY_QUERY;
 1970: 
 1971: 	/* avoid doing work if we can */
 1972: 	if (!return_value_used) {
 1973: 		db->last_err_code = sqlite_exec(db->db, sql, NULL, NULL, &errtext);
 1974: 
 1975: 		if (db->last_err_code != SQLITE_OK) {
 1976: 			php_error_docref(NULL TSRMLS_CC, E_WARNING, "%s", errtext);
 1977: 			if (errmsg) {
 1978: 				ZVAL_STRING(errmsg, errtext, 1);
 1979: 			}
 1980: 			sqlite_freemem(errtext);
 1981: 		}
 1982: 		return;
 1983: 	}
 1984: 
 1985: 	sqlite_query(object, db, sql, sql_len, (int)mode, 0, return_value, NULL, errmsg TSRMLS_CC);
 1986: }
 1987: /* }}} */
 1988: 
 1989: /* {{{ proto resource sqlite_fetch_column_types(string table_name, resource db [, int result_type])
 1990:    Return an array of column types from a particular table. */
 1991: PHP_FUNCTION(sqlite_fetch_column_types)
 1992: {
 1993: 	zval *zdb;
 1994: 	struct php_sqlite_db *db;
 1995: 	char *tbl, *sql;
 1996: 	int tbl_len;
 1997: 	char *errtext = NULL;
 1998: 	zval *object = getThis();
 1999: 	struct php_sqlite_result res;
 2000: 	const char **rowdata, **colnames, *tail;
 2001: 	int i, ncols;
 2002: 	long result_type = PHPSQLITE_ASSOC;
 2003: 
 2004: 	if (object) {
 2005: 		if (FAILURE == zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "s|l", &tbl, &tbl_len, &result_type)) {
 2006: 			return;
 2007: 		}
 2008: 		DB_FROM_OBJECT(db, object);
 2009: 	} else {
 2010: 		if (FAILURE == zend_parse_parameters_ex(ZEND_PARSE_PARAMS_QUIET,
 2011: 				ZEND_NUM_ARGS() TSRMLS_CC, "sr|l", &tbl, &tbl_len, &zdb, &result_type) &&
 2012: 			FAILURE == zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "rs|l", &zdb, &tbl, &tbl_len, &result_type)) {
 2013: 			return;
 2014: 		}
 2015: 		DB_FROM_ZVAL(db, &zdb);
 2016: 	}
 2017: 
 2018: 	if (!(sql = sqlite_mprintf("SELECT * FROM '%q' LIMIT 1", tbl))) {
 2019: 		RETURN_FALSE;
 2020: 	}
 2021: 
 2022: 	sqlite_exec(db->db, "PRAGMA show_datatypes = ON", NULL, NULL, NULL);
 2023: 
 2024: 	db->last_err_code = sqlite_compile(db->db, sql, &tail, &res.vm, &errtext);
 2025: 
 2026: 	sqlite_freemem(sql);
 2027: 
 2028: 	if (db->last_err_code != SQLITE_OK) {
 2029: 		php_error_docref(NULL TSRMLS_CC, E_WARNING, "%s", errtext);
 2030: 		sqlite_freemem(errtext);
 2031: 		RETVAL_FALSE;
 2032: 		goto done;
 2033: 	}
 2034: 
 2035: 	sqlite_step(res.vm, &ncols, &rowdata, &colnames);
 2036: 
 2037: 	array_init(return_value);
 2038: 
 2039: 	for (i = 0; i < ncols; i++) {
 2040: 		if (result_type == PHPSQLITE_ASSOC) {
 2041: 			char *colname = estrdup((char *)colnames[i]);
 2042: 
 2043: 			if (SQLITE_G(assoc_case) == 1) {
 2044: 				php_sqlite_strtoupper(colname);
 2045: 			} else if (SQLITE_G(assoc_case) == 2) {
 2046: 				php_sqlite_strtolower(colname);
 2047: 			}
 2048: 
 2049: 			add_assoc_string(return_value, colname, colnames[ncols + i] ? (char *)colnames[ncols + i] : "", 1);
 2050: 			efree(colname);
 2051: 		}
 2052: 		if (result_type == PHPSQLITE_NUM) {
 2053: 			add_index_string(return_value, i, colnames[ncols + i] ? (char *)colnames[ncols + i] : "", 1);
 2054: 		}
 2055: 	}
 2056: 	if (res.vm) {
 2057: 		sqlite_finalize(res.vm, NULL);
 2058: 	}
 2059: done:
 2060: 	sqlite_exec(db->db, "PRAGMA show_datatypes = OFF", NULL, NULL, NULL);
 2061: }
 2062: /* }}} */
 2063: 
 2064: /* {{{ proto resource sqlite_query(string query, resource db [, int result_type [, string &error_message]])
 2065:    Executes a query against a given database and returns a result handle. */
 2066: PHP_FUNCTION(sqlite_query)
 2067: {
 2068: 	zval *zdb;
 2069: 	struct php_sqlite_db *db;
 2070: 	char *sql;
 2071: 	int sql_len;
 2072: 	long mode = PHPSQLITE_BOTH;
 2073: 	char *errtext = NULL;
 2074: 	zval *errmsg = NULL;
 2075: 	zval *object = getThis();
 2076: 
 2077: 	if (object) {
 2078: 		if (FAILURE == zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "s|lz/", &sql, &sql_len, &mode, &errmsg)) {
 2079: 			return;
 2080: 		}
 2081: 		DB_FROM_OBJECT(db, object);
 2082: 	} else {
 2083: 		if (FAILURE == zend_parse_parameters_ex(ZEND_PARSE_PARAMS_QUIET,
 2084: 				ZEND_NUM_ARGS() TSRMLS_CC, "sr|lz/", &sql, &sql_len, &zdb, &mode, &errmsg) &&
 2085: 			FAILURE == zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "rs|lz/", &zdb, &sql, &sql_len, &mode, &errmsg)) {
 2086: 			return;
 2087: 		}
 2088: 		DB_FROM_ZVAL(db, &zdb);
 2089: 	}
 2090: 
 2091: 	if (errmsg) {
 2092: 		zval_dtor(errmsg);
 2093: 		ZVAL_NULL(errmsg);
 2094: 	}
 2095: 
 2096: 	PHP_SQLITE_EMPTY_QUERY;
 2097: 
 2098: 	/* avoid doing work if we can */
 2099: 	if (!return_value_used) {
 2100: 		db->last_err_code = sqlite_exec(db->db, sql, NULL, NULL, &errtext);
 2101: 
 2102: 		if (db->last_err_code != SQLITE_OK) {
 2103: 			php_error_docref(NULL TSRMLS_CC, E_WARNING, "%s", errtext);
 2104: 			if (errmsg) {
 2105: 				ZVAL_STRING(errmsg, errtext, 1);
 2106: 			}
 2107: 			sqlite_freemem(errtext);
 2108: 		}
 2109: 		return;
 2110: 	}
 2111: 
 2112: 	sqlite_query(object, db, sql, sql_len, (int)mode, 1, return_value, NULL, errmsg TSRMLS_CC);
 2113: }
 2114: /* }}} */
 2115: 
 2116: /* {{{ proto boolean sqlite_exec(string query, resource db[, string &error_message])
 2117:    Executes a result-less query against a given database */
 2118: PHP_FUNCTION(sqlite_exec)
 2119: {
 2120: 	zval *zdb;
 2121: 	struct php_sqlite_db *db;
 2122: 	char *sql;
 2123: 	int sql_len;
 2124: 	char *errtext = NULL;
 2125: 	zval *errmsg = NULL;
 2126: 	zval *object = getThis();
 2127: 
 2128: 	if (object) {
 2129: 		if (FAILURE == zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "s|z/", &sql, &sql_len, &errmsg)) {
 2130: 			return;
 2131: 		}
 2132: 		DB_FROM_OBJECT(db, object);
 2133: 	} else {
 2134: 		if(FAILURE == zend_parse_parameters_ex(ZEND_PARSE_PARAMS_QUIET,
 2135: 			ZEND_NUM_ARGS() TSRMLS_CC, "sr", &sql, &sql_len, &zdb) &&
 2136: 		   FAILURE == zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "rs|z/", &zdb, &sql, &sql_len, &errmsg)) {
 2137: 			return;
 2138: 		}
 2139: 		DB_FROM_ZVAL(db, &zdb);
 2140: 	}
 2141: 
 2142: 	if (errmsg) {
 2143: 		zval_dtor(errmsg);
 2144: 		ZVAL_NULL(errmsg);
 2145: 	}
 2146: 
 2147: 	PHP_SQLITE_EMPTY_QUERY;
 2148: 
 2149: 	db->last_err_code = sqlite_exec(db->db, sql, NULL, NULL, &errtext);
 2150: 
 2151: 	if (db->last_err_code != SQLITE_OK) {
 2152: 		php_error_docref(NULL TSRMLS_CC, E_WARNING, "%s", errtext);
 2153: 		if (errmsg) {
 2154: 			ZVAL_STRING(errmsg, errtext, 1);
 2155: 		}
 2156: 		sqlite_freemem(errtext);
 2157: 		RETURN_FALSE;
 2158: 	}
 2159: 
 2160: 	RETURN_TRUE;
 2161: }
 2162: /* }}} */
 2163: 
 2164: /* {{{ php_sqlite_fetch_array */
 2165: static void php_sqlite_fetch_array(struct php_sqlite_result *res, int mode, zend_bool decode_binary, int move_next, zval *return_value TSRMLS_DC)
 2166: {
 2167: 	int j, n = res->ncolumns, buffered = res->buffered;
 2168: 	const char **rowdata, **colnames;
 2169: 
 2170: 	/* check range of the row */
 2171: 	if (res->curr_row >= res->nrows) {
 2172: 		/* no more */
 2173: 		RETURN_FALSE;
 2174: 	}
 2175: 	colnames = (const char**)res->col_names;
 2176: 	if (res->buffered) {
 2177: 		rowdata = (const char**)&res->table[res->curr_row * res->ncolumns];
 2178: 	} else {
 2179: 		rowdata = (const char**)res->table;
 2180: 	}
 2181: 
 2182: 	/* now populate the result */
 2183: 	array_init(return_value);
 2184: 
 2185: 	for (j = 0; j < n; j++) {
 2186: 		zval *decoded;
 2187: 		MAKE_STD_ZVAL(decoded);
 2188: 
 2189: 		if (rowdata[j] == NULL) {
 2190: 			ZVAL_NULL(decoded);
 2191: 		} else if (decode_binary && rowdata[j][0] == '\x01') {
 2192: 			Z_STRVAL_P(decoded) = emalloc(strlen(rowdata[j]));
 2193: 			Z_STRLEN_P(decoded) = php_sqlite_decode_binary(rowdata[j]+1, Z_STRVAL_P(decoded));
 2194: 			Z_STRVAL_P(decoded)[Z_STRLEN_P(decoded)] = '\0';
 2195: 			Z_TYPE_P(decoded) = IS_STRING;
 2196: 			if (!buffered) {
 2197: 				efree((char*)rowdata[j]);
 2198: 				rowdata[j] = NULL;
 2199: 			}
 2200: 		} else {
 2201: 			ZVAL_STRING(decoded, (char*)rowdata[j], buffered);
 2202: 			if (!buffered) {
 2203: 				rowdata[j] = NULL;
 2204: 			}
 2205: 		}
 2206: 
 2207: 		if (mode & PHPSQLITE_NUM) {
 2208: 			if (mode & PHPSQLITE_ASSOC) {
 2209: 				add_index_zval(return_value, j, decoded);
 2210: 				Z_ADDREF_P(decoded);
 2211: 				add_assoc_zval(return_value, (char*)colnames[j], decoded);
 2212: 			} else {
 2213: 				add_next_index_zval(return_value, decoded);
 2214: 			}
 2215: 		} else {
 2216: 			add_assoc_zval(return_value, (char*)colnames[j], decoded);
 2217: 		}
 2218: 	}
 2219: 
 2220: 	if (move_next) {
 2221: 		if (!res->buffered) {
 2222: 			/* non buffered: fetch next row */
 2223: 			php_sqlite_fetch(res TSRMLS_CC);
 2224: 		}
 2225: 		/* advance the row pointer */
 2226: 		res->curr_row++;
 2227: 	}
 2228: }
 2229: /* }}} */
 2230: 
 2231: /* {{{ php_sqlite_fetch_column */
 2232: static void php_sqlite_fetch_column(struct php_sqlite_result *res, zval *which, zend_bool decode_binary, zval *return_value TSRMLS_DC)
 2233: {
 2234: 	int j;
 2235: 	const char **rowdata, **colnames;
 2236: 
 2237: 	/* check range of the row */
 2238: 	if (res->curr_row >= res->nrows) {
 2239: 		/* no more */
 2240: 		RETURN_FALSE;
 2241: 	}
 2242: 	colnames = (const char**)res->col_names;
 2243: 
 2244: 	if (Z_TYPE_P(which) == IS_LONG) {
 2245: 		j = Z_LVAL_P(which);
 2246: 	} else {
 2247: 		convert_to_string_ex(&which);
 2248: 		for (j = 0; j < res->ncolumns; j++) {
 2249: 			if (!strcasecmp((char*)colnames[j], Z_STRVAL_P(which))) {
 2250: 				break;
 2251: 			}
 2252: 		}
 2253: 	}
 2254: 	if (j < 0 || j >= res->ncolumns) {
 2255: 		php_error_docref(NULL TSRMLS_CC, E_WARNING, "No such column %d", j);
 2256: 		RETURN_FALSE;
 2257: 	}
 2258: 
 2259: 	if (res->buffered) {
 2260: 		rowdata = (const char**)&res->table[res->curr_row * res->ncolumns];
 2261: 	} else {
 2262: 		rowdata = (const char**)res->table;
 2263: 	}
 2264: 
 2265: 	if (rowdata[j] == NULL) {
 2266: 		RETURN_NULL();
 2267: 	} else if (decode_binary && rowdata[j] != NULL && rowdata[j][0] == '\x01') {
 2268: 		int l = strlen(rowdata[j]);
 2269: 		char *decoded = emalloc(l);
 2270: 		l = php_sqlite_decode_binary(rowdata[j]+1, decoded);
 2271: 		decoded[l] = '\0';
 2272: 		RETVAL_STRINGL(decoded, l, 0);
 2273: 		if (!res->buffered) {
 2274: 			efree((char*)rowdata[j]);
 2275: 			rowdata[j] = NULL;
 2276: 		}
 2277: 	} else {
 2278: 		RETVAL_STRING((char*)rowdata[j], res->buffered);
 2279: 		if (!res->buffered) {
 2280: 			rowdata[j] = NULL;
 2281: 		}
 2282: 	}
 2283: }
 2284: /* }}} */
 2285: 
 2286: /* {{{ proto array sqlite_fetch_all(resource result [, int result_type [, bool decode_binary]])
 2287:    Fetches all rows from a result set as an array of arrays. */
 2288: PHP_FUNCTION(sqlite_fetch_all)
 2289: {
 2290: 	zval *zres, *ent;
 2291: 	long mode = PHPSQLITE_BOTH;
 2292: 	zend_bool decode_binary = 1;
 2293: 	struct php_sqlite_result *res;
 2294: 	zval *object = getThis();
 2295: 
 2296: 	if (object) {
 2297: 		if (FAILURE == zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "|lb", &mode, &decode_binary)) {
 2298: 			return;
 2299: 		}
 2300: 		RES_FROM_OBJECT(res, object);
 2301: 		if (!ZEND_NUM_ARGS()) {
 2302: 			mode = res->mode;
 2303: 		}
 2304: 	} else {
 2305: 		if (FAILURE == zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "r|lb", &zres, &mode, &decode_binary)) {
 2306: 			return;
 2307: 		}
 2308: 		ZEND_FETCH_RESOURCE(res, struct php_sqlite_result *, &zres, -1, "sqlite result", le_sqlite_result);
 2309: 		if (ZEND_NUM_ARGS() < 2) {
 2310: 			mode = res->mode;
 2311: 		}
 2312: 	}
 2313: 
 2314: 	if (res->curr_row >= res->nrows && res->nrows) {
 2315: 		if (!res->buffered) {
 2316: 			php_error_docref(NULL TSRMLS_CC, E_WARNING, "One or more rowsets were already returned; returning NULL this time");
 2317: 		} else {
 2318: 			res->curr_row = 0;
 2319: 		}
 2320: 	}
 2321: 
 2322: 	array_init(return_value);
 2323: 
 2324: 	while (res->curr_row < res->nrows) {
 2325: 		MAKE_STD_ZVAL(ent);
 2326: 		php_sqlite_fetch_array(res, mode, decode_binary, 1, ent TSRMLS_CC);
 2327: 		add_next_index_zval(return_value, ent);
 2328: 	}
 2329: }
 2330: /* }}} */
 2331: 
 2332: /* {{{ proto array sqlite_fetch_array(resource result [, int result_type [, bool decode_binary]])
 2333:    Fetches the next row from a result set as an array. */
 2334: PHP_FUNCTION(sqlite_fetch_array)
 2335: {
 2336: 	zval *zres;
 2337: 	long mode = PHPSQLITE_BOTH;
 2338: 	zend_bool decode_binary = 1;
 2339: 	struct php_sqlite_result *res;
 2340: 	zval *object = getThis();
 2341: 
 2342: 	if (object) {
 2343: 		if (FAILURE == zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "|lb", &mode, &decode_binary)) {
 2344: 			return;
 2345: 		}
 2346: 		RES_FROM_OBJECT(res, object);
 2347: 		if (!ZEND_NUM_ARGS()) {
 2348: 			mode = res->mode;
 2349: 		}
 2350: 	} else {
 2351: 		if (FAILURE == zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "r|lb", &zres, &mode, &decode_binary)) {
 2352: 			return;
 2353: 		}
 2354: 		ZEND_FETCH_RESOURCE(res, struct php_sqlite_result *, &zres, -1, "sqlite result", le_sqlite_result);
 2355: 		if (ZEND_NUM_ARGS() < 2) {
 2356: 			mode = res->mode;
 2357: 		}
 2358: 	}
 2359: 
 2360: 	php_sqlite_fetch_array(res, mode, decode_binary, 1, return_value TSRMLS_CC);
 2361: }
 2362: /* }}} */
 2363: 
 2364: /* {{{ proto object sqlite_fetch_object(resource result [, string class_name [, NULL|array ctor_params [, bool decode_binary]]])
 2365:    Fetches the next row from a result set as an object. */
 2366:    /* note that you can do array(&$val) for param ctor_params */
 2367: PHP_FUNCTION(sqlite_fetch_object)
 2368: {
 2369: 	zval *zres;
 2370: 	zend_bool decode_binary = 1;
 2371: 	struct php_sqlite_result *res;
 2372: 	zval *object = getThis();
 2373: 	char *class_name = NULL;
 2374: 	int class_name_len;
 2375: 	zend_class_entry *ce;
 2376: 	zval dataset;
 2377: 	zend_fcall_info fci;
 2378: 	zend_fcall_info_cache fcc;
 2379: 	zval *retval_ptr;
 2380: 	zval *ctor_params = NULL;
 2381: 	zend_error_handling error_handling;
 2382: 
 2383: 	zend_replace_error_handling(object ? EH_THROW : EH_NORMAL, sqlite_ce_exception, &error_handling TSRMLS_CC);
 2384: 	if (object) {
 2385: 		if (FAILURE == zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "|szb", &class_name, &class_name_len, &ctor_params, &decode_binary)) {
 2386: 			zend_restore_error_handling(&error_handling TSRMLS_CC);
 2387: 			return;
 2388: 		}
 2389: 		RES_FROM_OBJECT_RESTORE_ERH(res, object, &error_handling);
 2390: 		if (!class_name) {
 2391: 			ce = zend_standard_class_def;
 2392: 		} else {
 2393: 			ce = zend_fetch_class(class_name, class_name_len, ZEND_FETCH_CLASS_AUTO TSRMLS_CC);
 2394: 		}
 2395: 	} else {
 2396: 		if (FAILURE == zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "r|szb", &zres, &class_name, &class_name_len, &ctor_params, &decode_binary)) {
 2397: 			zend_restore_error_handling(&error_handling TSRMLS_CC);
 2398: 			return;
 2399: 		}
 2400: 		ZEND_FETCH_RESOURCE(res, struct php_sqlite_result *, &zres, -1, "sqlite result", le_sqlite_result);
 2401: 		if (!class_name) {
 2402: 			ce = zend_standard_class_def;
 2403: 		} else {
 2404: 			ce = zend_fetch_class(class_name, class_name_len, ZEND_FETCH_CLASS_AUTO TSRMLS_CC);
 2405: 		}
 2406: 	}
 2407: 
 2408: 	if (!ce) {
 2409: 		zend_throw_exception_ex(sqlite_ce_exception, 0 TSRMLS_CC, "Could not find class '%s'", class_name);
 2410: 		zend_restore_error_handling(&error_handling TSRMLS_CC);
 2411: 		return;
 2412: 	}
 2413: 
 2414: 	if (res->curr_row < res->nrows) {
 2415: 		php_sqlite_fetch_array(res, PHPSQLITE_ASSOC, decode_binary, 1, &dataset TSRMLS_CC);
 2416: 	} else {
 2417: 		zend_restore_error_handling(&error_handling TSRMLS_CC);
 2418: 		RETURN_FALSE;
 2419: 	}
 2420: 
 2421: 	object_and_properties_init(return_value, ce, NULL);
 2422: 	zend_merge_properties(return_value, Z_ARRVAL(dataset), 1 TSRMLS_CC);
 2423: 
 2424: 	zend_restore_error_handling(&error_handling TSRMLS_CC);
 2425: 
 2426: 	if (ce->constructor) {
 2427: 		fci.size = sizeof(fci);
 2428: 		fci.function_table = &ce->function_table;
 2429: 		fci.function_name = NULL;
 2430: 		fci.symbol_table = NULL;
 2431: 		fci.object_ptr = return_value;
 2432: 		fci.retval_ptr_ptr = &retval_ptr;
 2433: 		if (ctor_params && Z_TYPE_P(ctor_params) != IS_NULL) {
 2434: 			if (Z_TYPE_P(ctor_params) == IS_ARRAY) {
 2435: 				HashTable *ht = Z_ARRVAL_P(ctor_params);
 2436: 				Bucket *p;
 2437: 
 2438: 				fci.param_count = 0;
 2439: 				fci.params = safe_emalloc(sizeof(zval*), ht->nNumOfElements, 0);
 2440: 				p = ht->pListHead;
 2441: 				while (p != NULL) {
 2442: 					fci.params[fci.param_count++] = (zval**)p->pData;
 2443: 					p = p->pListNext;
 2444: 				}
 2445: 			} else {
 2446: 				/* Two problems why we throw exceptions here: PHP is typeless
 2447: 				 * and hence passing one argument that's not an array could be
 2448: 				 * by mistake and the other way round is possible, too. The
 2449: 				 * single value is an array. Also we'd have to make that one
 2450: 				 * argument passed by reference.
 2451: 				 */
 2452: 				zend_throw_exception(sqlite_ce_exception, "Parameter ctor_params must be an array", 0 TSRMLS_CC);
 2453: 				return;
 2454: 			}
 2455: 		} else {
 2456: 			fci.param_count = 0;
 2457: 			fci.params = NULL;
 2458: 		}
 2459: 		fci.no_separation = 1;
 2460: 
 2461: 		fcc.initialized = 1;
 2462: 		fcc.function_handler = ce->constructor;
 2463: 		fcc.calling_scope = EG(scope);
 2464: 		fcc.called_scope = Z_OBJCE_P(return_value);
 2465: 		fcc.object_ptr = return_value;
 2466: 
 2467: 		if (zend_call_function(&fci, &fcc TSRMLS_CC) == FAILURE) {
 2468: 			zend_throw_exception_ex(sqlite_ce_exception, 0 TSRMLS_CC, "Could not execute %s::%s()", class_name, ce->constructor->common.function_name);
 2469: 		} else {
 2470: 			if (retval_ptr) {
 2471: 				zval_ptr_dtor(&retval_ptr);
 2472: 			}
 2473: 		}
 2474: 		if (fci.params) {
 2475: 			efree(fci.params);
 2476: 		}
 2477: 	} else if (ctor_params && Z_TYPE_P(ctor_params) != IS_NULL) {
 2478: 		zend_throw_exception_ex(sqlite_ce_exception, 0 TSRMLS_CC, "Class %s does not have a constructor, use NULL for parameter ctor_params or omit it", class_name);
 2479: 	}
 2480: }
 2481: /* }}} */
 2482: 
 2483: /* {{{ proto array sqlite_array_query(resource db, string query [ , int result_type [, bool decode_binary]])
 2484:    Executes a query against a given database and returns an array of arrays. */
 2485: PHP_FUNCTION(sqlite_array_query)
 2486: {
 2487: 	zval *zdb, *ent;
 2488: 	struct php_sqlite_db *db;
 2489: 	struct php_sqlite_result *rres;
 2490: 	char *sql;
 2491: 	int sql_len;
 2492: 	long mode = PHPSQLITE_BOTH;
 2493: 	char *errtext = NULL;
 2494: 	zend_bool decode_binary = 1;
 2495: 	zval *object = getThis();
 2496: 
 2497: 	if (object) {
 2498: 		if (FAILURE == zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "s|lb", &sql, &sql_len, &mode, &decode_binary)) {
 2499: 			return;
 2500: 		}
 2501: 		DB_FROM_OBJECT(db, object);
 2502: 	} else {
 2503: 		if (FAILURE == zend_parse_parameters_ex(ZEND_PARSE_PARAMS_QUIET,
 2504: 				ZEND_NUM_ARGS() TSRMLS_CC, "sr|lb", &sql, &sql_len, &zdb, &mode, &decode_binary) &&
 2505: 			FAILURE == zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "rs|lb", &zdb, &sql, &sql_len, &mode, &decode_binary)) {
 2506: 			return;
 2507: 		}
 2508: 		DB_FROM_ZVAL(db, &zdb);
 2509: 	}
 2510: 
 2511: 	PHP_SQLITE_EMPTY_QUERY;
 2512: 
 2513: 	/* avoid doing work if we can */
 2514: 	if (!return_value_used) {
 2515: 		db->last_err_code = sqlite_exec(db->db, sql, NULL, NULL, &errtext);
 2516: 
 2517: 		if (db->last_err_code != SQLITE_OK) {
 2518: 			php_error_docref(NULL TSRMLS_CC, E_WARNING, "%s", errtext);
 2519: 			sqlite_freemem(errtext);
 2520: 		}
 2521: 		return;
 2522: 	}
 2523: 
 2524: 	rres = (struct php_sqlite_result *)ecalloc(1, sizeof(*rres));
 2525: 	sqlite_query(NULL, db, sql, sql_len, (int)mode, 0, NULL, &rres, NULL TSRMLS_CC);
 2526: 	if (db->last_err_code != SQLITE_OK) {
 2527: 		if (rres) {
 2528: 			efree(rres);
 2529: 		}
 2530: 		RETURN_FALSE;
 2531: 	}
 2532: 
 2533: 	array_init(return_value);
 2534: 
 2535: 	while (rres->curr_row < rres->nrows) {
 2536: 		MAKE_STD_ZVAL(ent);
 2537: 		php_sqlite_fetch_array(rres, mode, decode_binary, 1, ent TSRMLS_CC);
 2538: 		add_next_index_zval(return_value, ent);
 2539: 	}
 2540: 	real_result_dtor(rres TSRMLS_CC);
 2541: }
 2542: /* }}} */
 2543: 
 2544: /* {{{ php_sqlite_fetch_single */
 2545: static void php_sqlite_fetch_single(struct php_sqlite_result *res, zend_bool decode_binary, zval *return_value TSRMLS_DC)
 2546: {
 2547: 	const char **rowdata;
 2548: 	char *decoded;
 2549: 	int decoded_len;
 2550: 
 2551: 	/* check range of the row */
 2552: 	if (res->curr_row >= res->nrows) {
 2553: 		/* no more */
 2554: 		RETURN_FALSE;
 2555: 	}
 2556: 
 2557: 	if (res->buffered) {
 2558: 		rowdata = (const char**)&res->table[res->curr_row * res->ncolumns];
 2559: 	} else {
 2560: 		rowdata = (const char**)res->table;
 2561: 	}
 2562: 
 2563: 	if (decode_binary && rowdata[0] != NULL && rowdata[0][0] == '\x01') {
 2564: 		decoded = emalloc(strlen(rowdata[0]));
 2565: 		decoded_len = php_sqlite_decode_binary(rowdata[0]+1, decoded);
 2566: 		if (!res->buffered) {
 2567: 			efree((char*)rowdata[0]);
 2568: 			rowdata[0] = NULL;
 2569: 		}
 2570: 	} else if (rowdata[0]) {
 2571: 		decoded_len = strlen((char*)rowdata[0]);
 2572: 		if (res->buffered) {
 2573: 			decoded = estrndup((char*)rowdata[0], decoded_len);
 2574: 		} else {
 2575: 			decoded = (char*)rowdata[0];
 2576: 			rowdata[0] = NULL;
 2577: 		}
 2578: 	} else {
 2579: 		decoded = NULL;
 2580: 		decoded_len = 0;
 2581: 	}
 2582: 
 2583: 	if (!res->buffered) {
 2584: 		/* non buffered: fetch next row */
 2585: 		php_sqlite_fetch(res TSRMLS_CC);
 2586: 	}
 2587: 	/* advance the row pointer */
 2588: 	res->curr_row++;
 2589: 
 2590: 	if (decoded == NULL) {
 2591: 		RETURN_NULL();
 2592: 	} else {
 2593: 		RETURN_STRINGL(decoded, decoded_len, 0);
 2594: 	}
 2595: }
 2596: /* }}} */
 2597: 
 2598: 
 2599: /* {{{ proto array sqlite_single_query(resource db, string query [, bool first_row_only [, bool decode_binary]])
 2600:    Executes a query and returns either an array for one single column or the value of the first row. */
 2601: PHP_FUNCTION(sqlite_single_query)
 2602: {
 2603: 	zval *zdb, *ent;
 2604: 	struct php_sqlite_db *db;
 2605: 	struct php_sqlite_result *rres;
 2606: 	char *sql;
 2607: 	int sql_len;
 2608: 	char *errtext = NULL;
 2609: 	zend_bool decode_binary = 1;
 2610: 	zend_bool srow = 1;
 2611: 	zval *object = getThis();
 2612: 
 2613: 	if (object) {
 2614: 		if (FAILURE == zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "s|bb", &sql, &sql_len, &srow, &decode_binary)) {
 2615: 			return;
 2616: 		}
 2617: 		RES_FROM_OBJECT(db, object);
 2618: 	} else {
 2619: 		if (FAILURE == zend_parse_parameters_ex(ZEND_PARSE_PARAMS_QUIET,
 2620: 				ZEND_NUM_ARGS() TSRMLS_CC, "sr|bb", &sql, &sql_len, &zdb, &srow, &decode_binary) &&
 2621: 			FAILURE == zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "rs|bb", &zdb, &sql, &sql_len, &srow, &decode_binary)) {
 2622: 			return;
 2623: 		}
 2624: 		DB_FROM_ZVAL(db, &zdb);
 2625: 	}
 2626: 
 2627: 	PHP_SQLITE_EMPTY_QUERY;
 2628: 
 2629: 	/* avoid doing work if we can */
 2630: 	if (!return_value_used) {
 2631: 		db->last_err_code = sqlite_exec(db->db, sql, NULL, NULL, &errtext);
 2632: 
 2633: 		if (db->last_err_code != SQLITE_OK) {
 2634: 			php_error_docref(NULL TSRMLS_CC, E_WARNING, "%s", errtext);
 2635: 			sqlite_freemem(errtext);
 2636: 		}
 2637: 		return;
 2638: 	}
 2639: 
 2640: 	rres = (struct php_sqlite_result *)ecalloc(1, sizeof(*rres));
 2641: 	sqlite_query(NULL, db, sql, sql_len, PHPSQLITE_NUM, 0, NULL, &rres, NULL TSRMLS_CC);
 2642: 	if (db->last_err_code != SQLITE_OK) {
 2643: 		if (rres) {
 2644: 			efree(rres);
 2645: 		}
 2646: 		RETURN_FALSE;
 2647: 	}
 2648: 
 2649: 	if (!srow) {
 2650: 		array_init(return_value);
 2651: 	}
 2652: 
 2653: 	while (rres->curr_row < rres->nrows) {
 2654: 		MAKE_STD_ZVAL(ent);
 2655: 		php_sqlite_fetch_single(rres, decode_binary, ent TSRMLS_CC);
 2656: 
 2657: 		/* if set and we only have 1 row in the result set, return the result as a string. */
 2658: 		if (srow) {
 2659: 			if (rres->curr_row == 1 && rres->curr_row >= rres->nrows) {
 2660: 				*return_value = *ent;
 2661: 				zval_copy_ctor(return_value);
 2662: 				zval_dtor(ent);
 2663: 				FREE_ZVAL(ent);
 2664: 				break;
 2665: 			} else {
 2666: 				srow = 0;
 2667: 				array_init(return_value);
 2668: 			}
 2669: 		}
 2670: 		add_next_index_zval(return_value, ent);
 2671: 	}
 2672: 
 2673: 	real_result_dtor(rres TSRMLS_CC);
 2674: }
 2675: /* }}} */
 2676: 
 2677: 
 2678: /* {{{ proto string sqlite_fetch_single(resource result [, bool decode_binary])
 2679:    Fetches the first column of a result set as a string. */
 2680: PHP_FUNCTION(sqlite_fetch_single)
 2681: {
 2682: 	zval *zres;
 2683: 	zend_bool decode_binary = 1;
 2684: 	struct php_sqlite_result *res;
 2685: 	zval *object = getThis();
 2686: 
 2687: 	if (object) {
 2688: 		if (FAILURE == zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "|b", &decode_binary)) {
 2689: 			return;
 2690: 		}
 2691: 		RES_FROM_OBJECT(res, object);
 2692: 	} else {
 2693: 		if (FAILURE == zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "r|b", &zres, &decode_binary)) {
 2694: 			return;
 2695: 		}
 2696: 		ZEND_FETCH_RESOURCE(res, struct php_sqlite_result *, &zres, -1, "sqlite result", le_sqlite_result);
 2697: 	}
 2698: 
 2699: 	php_sqlite_fetch_single(res, decode_binary, return_value TSRMLS_CC);
 2700: }
 2701: /* }}} */
 2702: 
 2703: /* {{{ proto array sqlite_current(resource result [, int result_type [, bool decode_binary]])
 2704:    Fetches the current row from a result set as an array. */
 2705: PHP_FUNCTION(sqlite_current)
 2706: {
 2707: 	zval *zres;
 2708: 	long mode = PHPSQLITE_BOTH;
 2709: 	zend_bool decode_binary = 1;
 2710: 	struct php_sqlite_result *res;
 2711: 	zval *object = getThis();
 2712: 
 2713: 	if (object) {
 2714: 		if (ZEND_NUM_ARGS() && FAILURE == zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "|lb", &mode, &decode_binary)) {
 2715: 			return;
 2716: 		}
 2717: 		RES_FROM_OBJECT(res, object);
 2718: 		if (!ZEND_NUM_ARGS()) {
 2719: 			mode = res->mode;
 2720: 		}
 2721: 	} else {
 2722: 		if (FAILURE == zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "r|lb", &zres, &mode, &decode_binary)) {
 2723: 			return;
 2724: 		}
 2725: 		ZEND_FETCH_RESOURCE(res, struct php_sqlite_result *, &zres, -1, "sqlite result", le_sqlite_result);
 2726: 		if (ZEND_NUM_ARGS() < 2) {
 2727: 			mode = res->mode;
 2728: 		}
 2729: 	}
 2730: 
 2731: 	php_sqlite_fetch_array(res, mode, decode_binary, 0, return_value TSRMLS_CC);
 2732: }
 2733: /* }}} */
 2734: 
 2735: /* {{{ proto mixed sqlite_column(resource result, mixed index_or_name [, bool decode_binary])
 2736:    Fetches a column from the current row of a result set. */
 2737: PHP_FUNCTION(sqlite_column)
 2738: {
 2739: 	zval *zres;
 2740: 	zval *which;
 2741: 	zend_bool decode_binary = 1;
 2742: 	struct php_sqlite_result *res;
 2743: 	zval *object = getThis();
 2744: 
 2745: 	if (object) {
 2746: 		if (FAILURE == zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "z|b", &which, &decode_binary)) {
 2747: 			return;
 2748: 		}
 2749: 		RES_FROM_OBJECT(res, object);
 2750: 	} else {
 2751: 		if (FAILURE == zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "rz|b", &zres, &which, &decode_binary)) {
 2752: 			return;
 2753: 		}
 2754: 		ZEND_FETCH_RESOURCE(res, struct php_sqlite_result *, &zres, -1, "sqlite result", le_sqlite_result);
 2755: 	}
 2756: 
 2757: 	php_sqlite_fetch_column(res, which, decode_binary, return_value TSRMLS_CC);
 2758: }
 2759: /* }}} */
 2760: 
 2761: /* {{{ proto string sqlite_libversion()
 2762:    Returns the version of the linked SQLite library. */
 2763: PHP_FUNCTION(sqlite_libversion)
 2764: {
 2765: 	if (zend_parse_parameters_none() == FAILURE) {
 2766: 		return;
 2767: 	}
 2768: 	RETURN_STRING((char*)sqlite_libversion(), 1);
 2769: }
 2770: /* }}} */
 2771: 
 2772: /* {{{ proto string sqlite_libencoding()
 2773:    Returns the encoding (iso8859 or UTF-8) of the linked SQLite library. */
 2774: PHP_FUNCTION(sqlite_libencoding)
 2775: {
 2776: 	if (zend_parse_parameters_none() == FAILURE) {
 2777: 		return;
 2778: 	}
 2779: 	RETURN_STRING((char*)sqlite_libencoding(), 1);
 2780: }
 2781: /* }}} */
 2782: 
 2783: /* {{{ proto int sqlite_changes(resource db)
 2784:    Returns the number of rows that were changed by the most recent SQL statement. */
 2785: PHP_FUNCTION(sqlite_changes)
 2786: {
 2787: 	zval *zdb;
 2788: 	struct php_sqlite_db *db;
 2789: 	zval *object = getThis();
 2790: 
 2791: 	if (object) {
 2792: 		if (zend_parse_parameters_none() == FAILURE) {
 2793: 			return;
 2794: 		}
 2795: 		DB_FROM_OBJECT(db, object);
 2796: 	} else {
 2797: 		if (FAILURE == zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "r", &zdb)) {
 2798: 			return;
 2799: 		}
 2800: 		DB_FROM_ZVAL(db, &zdb);
 2801: 	}
 2802: 
 2803: 	RETURN_LONG(sqlite_changes(db->db));
 2804: }
 2805: /* }}} */
 2806: 
 2807: /* {{{ proto int sqlite_last_insert_rowid(resource db)
 2808:    Returns the rowid of the most recently inserted row. */
 2809: PHP_FUNCTION(sqlite_last_insert_rowid)
 2810: {
 2811: 	zval *zdb;
 2812: 	struct php_sqlite_db *db;
 2813: 	zval *object = getThis();
 2814: 
 2815: 	if (object) {
 2816: 		if (zend_parse_parameters_none() == FAILURE) {
 2817: 			return;
 2818: 		}
 2819: 		DB_FROM_OBJECT(db, object);
 2820: 	} else {
 2821: 		if (FAILURE == zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "r", &zdb)) {
 2822: 			return;
 2823: 		}
 2824: 		DB_FROM_ZVAL(db, &zdb);
 2825: 	}
 2826: 
 2827: 	RETURN_LONG(sqlite_last_insert_rowid(db->db));
 2828: }
 2829: /* }}} */
 2830: 
 2831: static int sqlite_count_elements(zval *object, long *count TSRMLS_DC) /* {{{ */
 2832: {
 2833: 	sqlite_object *obj = (sqlite_object*) zend_object_store_get_object(object TSRMLS_CC);
 2834: 
 2835: 	if (obj->u.res == NULL) {
 2836: 		zend_throw_exception(sqlite_ce_exception, "Row count is not available for this query", 0 TSRMLS_CC);
 2837: 		return FAILURE;
 2838: 	}
 2839: 
 2840: 	if (obj->u.res->buffered) {
 2841: 		* count = obj->u.res->nrows;
 2842: 		return SUCCESS;
 2843: 	} else {
 2844: 		zend_throw_exception(sqlite_ce_exception, "Row count is not available for unbuffered queries", 0 TSRMLS_CC);
 2845: 		return FAILURE;
 2846: 	}
 2847: } /* }}} */
 2848: 
 2849: /* {{{ proto int sqlite_num_rows(resource result)
 2850:    Returns the number of rows in a buffered result set. */
 2851: PHP_FUNCTION(sqlite_num_rows)
 2852: {
 2853: 	zval *zres;
 2854: 	struct php_sqlite_result *res;
 2855: 	zval *object = getThis();
 2856: 
 2857: 	if (object) {
 2858: 		if (zend_parse_parameters_none() == FAILURE) {
 2859: 			return;
 2860: 		}
 2861: 		RES_FROM_OBJECT(res, object);
 2862: 	} else {
 2863: 		if (FAILURE == zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "r", &zres)) {
 2864: 			return;
 2865: 		}
 2866: 		ZEND_FETCH_RESOURCE(res, struct php_sqlite_result *, &zres, -1, "sqlite result", le_sqlite_result);
 2867: 	}
 2868: 
 2869: 	if (res->buffered) {
 2870: 		RETURN_LONG(res->nrows);
 2871: 	} else {
 2872: 		php_error_docref(NULL TSRMLS_CC, E_WARNING, "Row count is not available for unbuffered queries");
 2873: 		RETURN_FALSE;
 2874: 	}
 2875: }
 2876: /* }}} */
 2877: 
 2878: /* {{{ proto bool sqlite_valid(resource result)
 2879:    Returns whether more rows are available. */
 2880: PHP_FUNCTION(sqlite_valid)
 2881: {
 2882: 	zval *zres;
 2883: 	struct php_sqlite_result *res;
 2884: 	zval *object = getThis();
 2885: 
 2886: 	if (object) {
 2887: 		if (zend_parse_parameters_none() == FAILURE) {
 2888: 			return;
 2889: 		}
 2890: 		RES_FROM_OBJECT(res, object);
 2891: 	} else {
 2892: 		if (FAILURE == zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "r", &zres)) {
 2893: 			return;
 2894: 		}
 2895: 		ZEND_FETCH_RESOURCE(res, struct php_sqlite_result *, &zres, -1, "sqlite result", le_sqlite_result);
 2896: 	}
 2897: 
 2898: 	RETURN_BOOL(res->curr_row < res->nrows && res->nrows); /* curr_row may be -1 */
 2899: }
 2900: /* }}} */
 2901: 
 2902: /* {{{ proto bool sqlite_has_prev(resource result)
 2903:  * Returns whether a previous row is available. */
 2904: PHP_FUNCTION(sqlite_has_prev)
 2905: {
 2906: 	zval *zres;
 2907: 	struct php_sqlite_result *res;
 2908: 	zval *object = getThis();
 2909: 
 2910: 	if (object) {
 2911: 		if (zend_parse_parameters_none() == FAILURE) {
 2912: 			return;
 2913: 		}
 2914: 		RES_FROM_OBJECT(res, object);
 2915: 	} else {
 2916: 		if (FAILURE == zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "r", &zres)) {
 2917: 			return;
 2918: 		}
 2919: 		ZEND_FETCH_RESOURCE(res, struct php_sqlite_result *, &zres, -1, "sqlite result", le_sqlite_result);
 2920: 	}
 2921: 
 2922: 	if(!res->buffered) {
 2923: 		php_error_docref(NULL TSRMLS_CC, E_WARNING, "you cannot use sqlite_has_prev on unbuffered querys");
 2924: 		RETURN_FALSE;
 2925: 	}
 2926: 
 2927: 	RETURN_BOOL(res->curr_row);
 2928: }
 2929: /* }}} */
 2930: 
 2931: /* {{{ proto int sqlite_num_fields(resource result)
 2932:    Returns the number of fields in a result set. */
 2933: PHP_FUNCTION(sqlite_num_fields)
 2934: {
 2935: 	zval *zres;
 2936: 	struct php_sqlite_result *res;
 2937: 	zval *object = getThis();
 2938: 
 2939: 	if (object) {
 2940: 		if (zend_parse_parameters_none() == FAILURE) {
 2941: 			return;
 2942: 		}
 2943: 		RES_FROM_OBJECT(res, object);
 2944: 	} else {
 2945: 		if (FAILURE == zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "r", &zres)) {
 2946: 			return;
 2947: 		}
 2948: 		ZEND_FETCH_RESOURCE(res, struct php_sqlite_result *, &zres, -1, "sqlite result", le_sqlite_result);
 2949: 	}
 2950: 
 2951: 	RETURN_LONG(res->ncolumns);
 2952: }
 2953: /* }}} */
 2954: 
 2955: /* {{{ proto string sqlite_field_name(resource result, int field_index)
 2956:    Returns the name of a particular field of a result set. */
 2957: PHP_FUNCTION(sqlite_field_name)
 2958: {
 2959: 	zval *zres;
 2960: 	struct php_sqlite_result *res;
 2961: 	long field;
 2962: 	zval *object = getThis();
 2963: 
 2964: 	if (object) {
 2965: 		if (FAILURE == zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "l", &field)) {
 2966: 			return;
 2967: 		}
 2968: 		RES_FROM_OBJECT(res, object);
 2969: 	} else {
 2970: 		if (FAILURE == zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "rl", &zres, &field)) {
 2971: 			return;
 2972: 		}
 2973: 		ZEND_FETCH_RESOURCE(res, struct php_sqlite_result *, &zres, -1, "sqlite result", le_sqlite_result);
 2974: 	}
 2975: 
 2976: 	if (field < 0 || field >= res->ncolumns) {
 2977: 		php_error_docref(NULL TSRMLS_CC, E_WARNING, "field %ld out of range", field);
 2978: 		RETURN_FALSE;
 2979: 	}
 2980: 
 2981: 	RETURN_STRING(res->col_names[field], 1);
 2982: }
 2983: /* }}} */
 2984: 
 2985: /* {{{ proto bool sqlite_seek(resource result, int row)
 2986:    Seek to a particular row number of a buffered result set. */
 2987: PHP_FUNCTION(sqlite_seek)
 2988: {
 2989: 	zval *zres;
 2990: 	struct php_sqlite_result *res;
 2991: 	long row;
 2992: 	zval *object = getThis();
 2993: 
 2994: 	if (object) {
 2995: 		if (FAILURE == zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "l", &row)) {
 2996: 			return;
 2997: 		}
 2998: 		RES_FROM_OBJECT(res, object);
 2999: 	} else {
 3000: 		if (FAILURE == zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "rl", &zres, &row)) {
 3001: 			return;
 3002: 		}
 3003: 		ZEND_FETCH_RESOURCE(res, struct php_sqlite_result *, &zres, -1, "sqlite result", le_sqlite_result);
 3004: 	}
 3005: 
 3006: 	if (!res->buffered) {
 3007: 		php_error_docref(NULL TSRMLS_CC, E_WARNING, "Cannot seek an unbuffered result set");
 3008: 		RETURN_FALSE;
 3009: 	}
 3010: 
 3011: 	if (row < 0 || row >= res->nrows) {
 3012: 		php_error_docref(NULL TSRMLS_CC, E_WARNING, "row %ld out of range", row);
 3013: 		RETURN_FALSE;
 3014: 	}
 3015: 
 3016: 	res->curr_row = row;
 3017: 	RETURN_TRUE;
 3018: }
 3019: /* }}} */
 3020: 
 3021: /* {{{ proto bool sqlite_rewind(resource result)
 3022:    Seek to the first row number of a buffered result set. */
 3023: PHP_FUNCTION(sqlite_rewind)
 3024: {
 3025: 	zval *zres;
 3026: 	struct php_sqlite_result *res;
 3027: 	zval *object = getThis();
 3028: 
 3029: 	if (object) {
 3030: 		if (zend_parse_parameters_none() == FAILURE) {
 3031: 			return;
 3032: 		}
 3033: 		RES_FROM_OBJECT(res, object);
 3034: 	} else {
 3035: 		if (FAILURE == zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "r", &zres)) {
 3036: 			return;
 3037: 		}
 3038: 		ZEND_FETCH_RESOURCE(res, struct php_sqlite_result *, &zres, -1, "sqlite result", le_sqlite_result);
 3039: 	}
 3040: 
 3041: 	if (!res->buffered) {
 3042: 		php_error_docref(NULL TSRMLS_CC, E_WARNING, "Cannot rewind an unbuffered result set");
 3043: 		RETURN_FALSE;
 3044: 	}
 3045: 
 3046: 	if (!res->nrows) {
 3047: 		php_error_docref(NULL TSRMLS_CC, E_NOTICE, "no rows received");
 3048: 		RETURN_FALSE;
 3049: 	}
 3050: 
 3051: 	res->curr_row = 0;
 3052: 	RETURN_TRUE;
 3053: }
 3054: /* }}} */
 3055: 
 3056: /* {{{ proto bool sqlite_next(resource result)
 3057:    Seek to the next row number of a result set. */
 3058: PHP_FUNCTION(sqlite_next)
 3059: {
 3060: 	zval *zres;
 3061: 	struct php_sqlite_result *res;
 3062: 	zval *object = getThis();
 3063: 
 3064: 	if (object) {
 3065: 		if (zend_parse_parameters_none() == FAILURE) {
 3066: 			return;
 3067: 		}
 3068: 		RES_FROM_OBJECT(res, object);
 3069: 	} else {
 3070: 		if (FAILURE == zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "r", &zres)) {
 3071: 			return;
 3072: 		}
 3073: 		ZEND_FETCH_RESOURCE(res, struct php_sqlite_result *, &zres, -1, "sqlite result", le_sqlite_result);
 3074: 	}
 3075: 
 3076: 	if (!res->buffered && res->vm) {
 3077: 		php_sqlite_fetch(res TSRMLS_CC);
 3078: 	}
 3079: 
 3080: 	if (res->curr_row >= res->nrows) {
 3081: 		php_error_docref(NULL TSRMLS_CC, E_WARNING, "no more rows available");
 3082: 		RETURN_FALSE;
 3083: 	}
 3084: 
 3085: 	res->curr_row++;
 3086: 
 3087: 	RETURN_TRUE;
 3088: }
 3089: /* }}} */
 3090: 
 3091: /* {{{ proto int sqlite_key(resource result)
 3092:    Return the current row index of a buffered result. */
 3093: PHP_FUNCTION(sqlite_key)
 3094: {
 3095: 	zval *zres;
 3096: 	struct php_sqlite_result *res;
 3097: 	zval *object = getThis();
 3098: 
 3099: 	if (object) {
 3100: 		if (zend_parse_parameters_none() == FAILURE) {
 3101: 			return;
 3102: 		}
 3103: 		RES_FROM_OBJECT(res, object);
 3104: 	} else {
 3105: 		if (FAILURE == zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "r", &zres)) {
 3106: 			return;
 3107: 		}
 3108: 		ZEND_FETCH_RESOURCE(res, struct php_sqlite_result *, &zres, -1, "sqlite result", le_sqlite_result);
 3109: 	}
 3110: 
 3111: 	RETURN_LONG(res->curr_row);
 3112: }
 3113: /* }}} */
 3114: 
 3115: /* {{{ proto bool sqlite_prev(resource result)
 3116:  * Seek to the previous row number of a result set. */
 3117: PHP_FUNCTION(sqlite_prev)
 3118: {
 3119: 	zval *zres;
 3120: 	struct php_sqlite_result *res;
 3121: 	zval *object = getThis();
 3122: 
 3123: 	if (object) {
 3124: 		if (zend_parse_parameters_none() == FAILURE) {
 3125: 			return;
 3126: 		}
 3127: 		RES_FROM_OBJECT(res, object);
 3128: 	} else {
 3129: 		if (FAILURE == zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "r", &zres)) {
 3130: 			return;
 3131: 		}
 3132: 		ZEND_FETCH_RESOURCE(res, struct php_sqlite_result *, &zres, -1, "sqlite result", le_sqlite_result);
 3133: 	}
 3134: 
 3135: 	if (!res->buffered) {
 3136: 		php_error_docref(NULL TSRMLS_CC, E_WARNING, "you cannot use sqlite_prev on unbuffered querys");
 3137: 		RETURN_FALSE;
 3138: 	}
 3139: 
 3140: 	if (res->curr_row <= 0) {
 3141: 		php_error_docref(NULL TSRMLS_CC, E_WARNING, "no previous row available");
 3142: 		RETURN_FALSE;
 3143: 	}
 3144: 
 3145: 	res->curr_row--;
 3146: 
 3147: 	RETURN_TRUE;
 3148: }
 3149: /* }}} */
 3150: 
 3151: /* {{{ proto string sqlite_escape_string(string item)
 3152:    Escapes a string for use as a query parameter. */
 3153: PHP_FUNCTION(sqlite_escape_string)
 3154: {
 3155: 	char *string = NULL;
 3156: 	int stringlen;
 3157: 	char *ret;
 3158: 
 3159: 	if (FAILURE == zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "s", &string, &stringlen)) {
 3160: 		return;
 3161: 	}
 3162: 
 3163: 	if (stringlen && (string[0] == '\x01' || memchr(string, '\0', stringlen) != NULL)) {
 3164: 		/* binary string */
 3165: 		int enclen;
 3166: 
 3167: 		ret = safe_emalloc(1 + stringlen / 254, 257, 3);
 3168: 		ret[0] = '\x01';
 3169: 		enclen = php_sqlite_encode_binary(string, stringlen, ret+1);
 3170: 		RETVAL_STRINGL(ret, enclen+1, 0);
 3171: 
 3172: 	} else if (stringlen) {
 3173: 		ret = sqlite_mprintf("%q", string);
 3174: 		if (ret) {
 3175: 			RETVAL_STRING(ret, 1);
 3176: 			sqlite_freemem(ret);
 3177: 		}
 3178: 	} else {
 3179: 		RETURN_EMPTY_STRING();
 3180: 	}
 3181: }
 3182: /* }}} */
 3183: 
 3184: /* {{{ proto int sqlite_last_error(resource db)
 3185:    Returns the error code of the last error for a database. */
 3186: PHP_FUNCTION(sqlite_last_error)
 3187: {
 3188: 	zval *zdb;
 3189: 	struct php_sqlite_db *db;
 3190: 	zval *object = getThis();
 3191: 
 3192: 	if (object) {
 3193: 		if (zend_parse_parameters_none() == FAILURE) {
 3194: 			return;
 3195: 		}
 3196: 		DB_FROM_OBJECT(db, object);
 3197: 	} else {
 3198: 		if (FAILURE == zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "r", &zdb)) {
 3199: 			return;
 3200: 		}
 3201: 		DB_FROM_ZVAL(db, &zdb);
 3202: 	}
 3203: 
 3204: 	RETURN_LONG(db->last_err_code);
 3205: }
 3206: /* }}} */
 3207: 
 3208: /* {{{ proto string sqlite_error_string(int error_code)
 3209:    Returns the textual description of an error code. */
 3210: PHP_FUNCTION(sqlite_error_string)
 3211: {
 3212: 	long code;
 3213: 	const char *msg;
 3214: 
 3215: 	if (FAILURE == zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "l", &code)) {
 3216: 		return;
 3217: 	}
 3218: 
 3219: 	msg = sqlite_error_string(code);
 3220: 
 3221: 	if (msg) {
 3222: 		RETURN_STRING((char*)msg, 1);
 3223: 	} else {
 3224: 		RETURN_NULL();
 3225: 	}
 3226: }
 3227: /* }}} */
 3228: 
 3229: /* manages duplicate registrations of a particular function, and
 3230:  * also handles the case where the db is using a persistent connection */
 3231: enum callback_prep_t { DO_REG, SKIP_REG, ERR };
 3232: 
 3233: static enum callback_prep_t prep_callback_struct(struct php_sqlite_db *db, int is_agg,
 3234: 		char *funcname,
 3235: 		zval *step, zval *fini, struct php_sqlite_agg_functions **funcs)
 3236: {
 3237: 	struct php_sqlite_agg_functions *alloc_funcs, func_tmp;
 3238: 	char *hashkey;
 3239: 	int hashkeylen;
 3240: 	enum callback_prep_t ret;
 3241: 
 3242: 	hashkeylen = spprintf(&hashkey, 0, "%s-%s", is_agg ? "agg" : "reg", funcname);
 3243: 
 3244: 	/* is it already registered ? */
 3245: 	if (SUCCESS == zend_hash_find(&db->callbacks, hashkey, hashkeylen+1, (void*)&alloc_funcs)) {
 3246: 		/* override the previous definition */
 3247: 
 3248: 		if (alloc_funcs->is_valid) {
 3249: 			/* release these */
 3250: 
 3251: 			if (alloc_funcs->step) {
 3252: 				zval_ptr_dtor(&alloc_funcs->step);
 3253: 				alloc_funcs->step = NULL;
 3254: 			}
 3255: 
 3256: 			if (alloc_funcs->fini) {
 3257: 				zval_ptr_dtor(&alloc_funcs->fini);
 3258: 				alloc_funcs->fini = NULL;
 3259: 			}
 3260: 		}
 3261: 
 3262: 		ret = SKIP_REG;
 3263: 	} else {
 3264: 		/* add a new one */
 3265: 		func_tmp.db = db;
 3266: 
 3267: 		ret = SUCCESS == zend_hash_update(&db->callbacks, hashkey, hashkeylen+1,
 3268: 				(void*)&func_tmp, sizeof(func_tmp), (void**)&alloc_funcs) ? DO_REG : ERR;
 3269: 	}
 3270: 
 3271: 	efree(hashkey);
 3272: 
 3273: 	MAKE_STD_ZVAL(alloc_funcs->step);
 3274: 	*(alloc_funcs->step)  = *step;
 3275: 	zval_copy_ctor(alloc_funcs->step);
 3276: 	INIT_PZVAL(alloc_funcs->step);
 3277: 
 3278: 	if (is_agg) {
 3279: 		MAKE_STD_ZVAL(alloc_funcs->fini);
 3280: 		*(alloc_funcs->fini) = *fini;
 3281: 		zval_copy_ctor(alloc_funcs->fini);
 3282: 		INIT_PZVAL(alloc_funcs->fini);
 3283: 	} else {
 3284: 		alloc_funcs->fini = NULL;
 3285: 	}
 3286: 	alloc_funcs->is_valid = 1;
 3287: 	*funcs = alloc_funcs;
 3288: 
 3289: 	return ret;
 3290: }
 3291: 
 3292: 
 3293: /* {{{ proto bool sqlite_create_aggregate(resource db, string funcname, mixed step_func, mixed finalize_func[, long num_args])
 3294:     Registers an aggregate function for queries. */
 3295: PHP_FUNCTION(sqlite_create_aggregate)
 3296: {
 3297: 	char *funcname = NULL;
 3298: 	int funcname_len;
 3299: 	zval *zstep, *zfinal, *zdb;
 3300: 	struct php_sqlite_db *db;
 3301: 	struct php_sqlite_agg_functions *funcs;
 3302: 	char *callable = NULL;
 3303: 	long num_args = -1;
 3304: 	zval *object = getThis();
 3305: 
 3306: 	if (object) {
 3307: 		if (FAILURE == zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "szz|l", &funcname, &funcname_len, &zstep, &zfinal, &num_args)) {
 3308: 			return;
 3309: 		}
 3310: 		DB_FROM_OBJECT(db, object);
 3311: 	} else {
 3312: 		if (FAILURE == zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "rszz|l", &zdb, &funcname, &funcname_len, &zstep, &zfinal, &num_args)) {
 3313: 			return;
 3314: 		}
 3315: 		DB_FROM_ZVAL(db, &zdb);
 3316: 	}
 3317: 
 3318: 	if (!zend_is_callable(zstep, 0, &callable TSRMLS_CC)) {
 3319: 		php_error_docref(NULL TSRMLS_CC, E_WARNING, "step function `%s' is not callable", callable);
 3320: 		efree(callable);
 3321: 		return;
 3322: 	}
 3323: 	efree(callable);
 3324: 
 3325: 	if (!zend_is_callable(zfinal, 0, &callable TSRMLS_CC)) {
 3326: 		php_error_docref(NULL TSRMLS_CC, E_WARNING, "finalize function `%s' is not callable", callable);
 3327: 		efree(callable);
 3328: 		return;
 3329: 	}
 3330: 	efree(callable);
 3331: 
 3332: 
 3333: 	if (prep_callback_struct(db, 1, funcname, zstep, zfinal, &funcs) == DO_REG) {
 3334: 		sqlite_create_aggregate(db->db, funcname, num_args,
 3335: 				php_sqlite_agg_step_function_callback,
 3336: 				php_sqlite_agg_fini_function_callback, funcs);
 3337: 	}
 3338: 
 3339: 
 3340: }
 3341: /* }}} */
 3342: 
 3343: /* {{{ proto bool sqlite_create_function(resource db, string funcname, mixed callback[, long num_args])
 3344:     Registers a "regular" function for queries. */
 3345: PHP_FUNCTION(sqlite_create_function)
 3346: {
 3347: 	char *funcname = NULL;
 3348: 	int funcname_len;
 3349: 	zval *zcall, *zdb;
 3350: 	struct php_sqlite_db *db;
 3351: 	struct php_sqlite_agg_functions *funcs;
 3352: 	char *callable = NULL;
 3353: 	long num_args = -1;
 3354: 
 3355: 	zval *object = getThis();
 3356: 
 3357: 	if (object) {
 3358: 		if (FAILURE == zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "sz|l", &funcname, &funcname_len, &zcall, &num_args)) {
 3359: 			return;
 3360: 		}
 3361: 		DB_FROM_OBJECT(db, object);
 3362: 	} else {
 3363: 		if (FAILURE == zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "rsz|l", &zdb, &funcname, &funcname_len, &zcall, &num_args)) {
 3364: 			return;
 3365: 		}
 3366: 		DB_FROM_ZVAL(db, &zdb);
 3367: 	}
 3368: 
 3369: 	if (!zend_is_callable(zcall, 0, &callable TSRMLS_CC)) {
 3370: 		php_error_docref(NULL TSRMLS_CC, E_WARNING, "function `%s' is not callable", callable);
 3371: 		efree(callable);
 3372: 		return;
 3373: 	}
 3374: 	efree(callable);
 3375: 
 3376: 	if (prep_callback_struct(db, 0, funcname, zcall, NULL, &funcs) == DO_REG) {
 3377: 		sqlite_create_function(db->db, funcname, num_args, php_sqlite_function_callback, funcs);
 3378: 	}
 3379: }
 3380: /* }}} */
 3381: 
 3382: /* {{{ proto string sqlite_udf_encode_binary(string data)
 3383:    Apply binary encoding (if required) to a string to return from an UDF. */
 3384: PHP_FUNCTION(sqlite_udf_encode_binary)
 3385: {
 3386: 	char *data = NULL;
 3387: 	int datalen;
 3388: 
 3389: 	if (FAILURE == zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "s!", &data, &datalen)) {
 3390: 		return;
 3391: 	}
 3392: 
 3393: 	if (data == NULL) {
 3394: 		RETURN_NULL();
 3395: 	}
 3396: 	if (datalen && (data[0] == '\x01' || memchr(data, '\0', datalen) != NULL)) {
 3397: 		/* binary string */
 3398: 		int enclen;
 3399: 		char *ret;
 3400: 
 3401: 		ret = safe_emalloc(1 + datalen / 254, 257, 3);
 3402: 		ret[0] = '\x01';
 3403: 		enclen = php_sqlite_encode_binary(data, datalen, ret+1);
 3404: 		RETVAL_STRINGL(ret, enclen+1, 0);
 3405: 	} else {
 3406: 		RETVAL_STRINGL(data, datalen, 1);
 3407: 	}
 3408: }
 3409: /* }}} */
 3410: 
 3411: /* {{{ proto string sqlite_udf_decode_binary(string data)
 3412:    Decode binary encoding on a string parameter passed to an UDF. */
 3413: PHP_FUNCTION(sqlite_udf_decode_binary)
 3414: {
 3415: 	char *data = NULL;
 3416: 	int datalen;
 3417: 
 3418: 	if (FAILURE == zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "s!", &data, &datalen)) {
 3419: 		return;
 3420: 	}
 3421: 
 3422: 	if (data == NULL) {
 3423: 		RETURN_NULL();
 3424: 	}
 3425: 	if (datalen && data[0] == '\x01') {
 3426: 		/* encoded string */
 3427: 		int enclen;
 3428: 		char *ret;
 3429: 
 3430: 		ret = emalloc(datalen);
 3431: 		enclen = php_sqlite_decode_binary(data+1, ret);
 3432: 		ret[enclen] = '\0';
 3433: 		RETVAL_STRINGL(ret, enclen, 0);
 3434: 	} else {
 3435: 		RETVAL_STRINGL(data, datalen, 1);
 3436: 	}
 3437: }
 3438: /* }}} */
 3439: 
 3440: 
 3441: /*
 3442:  * Local variables:
 3443:  * tab-width: 4
 3444:  * c-basic-offset: 4
 3445:  * End:
 3446:  * vim600: sw=4 ts=4 fdm=marker
 3447:  * vim<600: sw=4 ts=4
 3448:  */

FreeBSD-CVSweb <freebsd-cvsweb@FreeBSD.org>