123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407140814091410141114121413141414151416141714181419142014211422142314241425142614271428142914301431143214331434143514361437143814391440144114421443144414451446144714481449145014511452145314541455145614571458145914601461146214631464146514661467146814691470147114721473147414751476147714781479148014811482148314841485148614871488148914901491149214931494149514961497149814991500150115021503150415051506150715081509151015111512151315141515151615171518151915201521152215231524152515261527152815291530153115321533153415351536153715381539154015411542154315441545154615471548154915501551155215531554155515561557155815591560156115621563156415651566156715681569157015711572157315741575157615771578157915801581158215831584158515861587158815891590159115921593159415951596159715981599160016011602160316041605160616071608160916101611 |
- /*
- +----------------------------------------------------------------------+
- | PHP Version 7 |
- +----------------------------------------------------------------------+
- | Copyright (c) 1997-2018 The PHP Group |
- +----------------------------------------------------------------------+
- | This source file is subject to version 3.01 of the PHP license, |
- | that is bundled with this package in the file LICENSE, and is |
- | available through the world-wide-web at the following url: |
- | http://www.php.net/license/3_01.txt |
- | If you did not receive a copy of the PHP license and are unable to |
- | obtain it through the world-wide-web, please send a note to |
- | license@php.net so we can mail you a copy immediately. |
- +----------------------------------------------------------------------+
- | Author: Wez Furlong <wez@php.net> |
- | Marcus Boerger <helly@php.net> |
- | Sterling Hughes <sterling@php.net> |
- +----------------------------------------------------------------------+
- */
- /* The PDO Database Handle Class */
- #ifdef HAVE_CONFIG_H
- #include "config.h"
- #endif
- #include "php.h"
- #include "php_ini.h"
- #include "ext/standard/info.h"
- #include "php_pdo.h"
- #include "php_pdo_driver.h"
- #include "php_pdo_int.h"
- #include "zend_exceptions.h"
- #include "zend_object_handlers.h"
- #include "zend_hash.h"
- static int pdo_dbh_attribute_set(pdo_dbh_t *dbh, zend_long attr, zval *value);
- void pdo_throw_exception(unsigned int driver_errcode, char *driver_errmsg, pdo_error_type *pdo_error)
- {
- zval error_info,pdo_exception;
- char *pdo_exception_message;
- object_init_ex(&pdo_exception, php_pdo_get_exception());
- array_init(&error_info);
- add_next_index_string(&error_info, *pdo_error);
- add_next_index_long(&error_info, driver_errcode);
- add_next_index_string(&error_info, driver_errmsg);
- spprintf(&pdo_exception_message, 0,"SQLSTATE[%s] [%d] %s",*pdo_error, driver_errcode, driver_errmsg);
- zend_update_property(php_pdo_get_exception(), &pdo_exception, "errorInfo", sizeof("errorInfo")-1, &error_info);
- zend_update_property_long(php_pdo_get_exception(), &pdo_exception, "code", sizeof("code")-1, driver_errcode);
- zend_update_property_string(
- php_pdo_get_exception(),
- &pdo_exception,
- "message",
- sizeof("message")-1,
- pdo_exception_message
- );
- efree(pdo_exception_message);
- zval_ptr_dtor(&error_info);
- zend_throw_exception_object(&pdo_exception);
- }
- void pdo_raise_impl_error(pdo_dbh_t *dbh, pdo_stmt_t *stmt, const char *sqlstate, const char *supp) /* {{{ */
- {
- pdo_error_type *pdo_err = &dbh->error_code;
- char *message = NULL;
- const char *msg;
- if (dbh && dbh->error_mode == PDO_ERRMODE_SILENT) {
- #if 0
- /* BUG: if user is running in silent mode and hits an error at the driver level
- * when they use the PDO methods to call up the error information, they may
- * get bogus information */
- return;
- #endif
- }
- if (stmt) {
- pdo_err = &stmt->error_code;
- }
- strncpy(*pdo_err, sqlstate, 6);
- /* hash sqlstate to error messages */
- msg = pdo_sqlstate_state_to_description(*pdo_err);
- if (!msg) {
- msg = "<<Unknown error>>";
- }
- if (supp) {
- spprintf(&message, 0, "SQLSTATE[%s]: %s: %s", *pdo_err, msg, supp);
- } else {
- spprintf(&message, 0, "SQLSTATE[%s]: %s", *pdo_err, msg);
- }
- if (dbh && dbh->error_mode != PDO_ERRMODE_EXCEPTION) {
- php_error_docref(NULL, E_WARNING, "%s", message);
- } else {
- zval ex, info;
- zend_class_entry *def_ex = php_pdo_get_exception_base(1), *pdo_ex = php_pdo_get_exception();
- object_init_ex(&ex, pdo_ex);
- zend_update_property_string(def_ex, &ex, "message", sizeof("message")-1, message);
- zend_update_property_string(def_ex, &ex, "code", sizeof("code")-1, *pdo_err);
- array_init(&info);
- add_next_index_string(&info, *pdo_err);
- add_next_index_long(&info, 0);
- zend_update_property(pdo_ex, &ex, "errorInfo", sizeof("errorInfo")-1, &info);
- zval_ptr_dtor(&info);
- zend_throw_exception_object(&ex);
- }
- if (message) {
- efree(message);
- }
- }
- /* }}} */
- PDO_API void pdo_handle_error(pdo_dbh_t *dbh, pdo_stmt_t *stmt) /* {{{ */
- {
- pdo_error_type *pdo_err = &dbh->error_code;
- const char *msg = "<<Unknown>>";
- char *supp = NULL;
- zend_long native_code = 0;
- zend_string *message = NULL;
- zval info;
- if (dbh == NULL || dbh->error_mode == PDO_ERRMODE_SILENT) {
- return;
- }
- if (stmt) {
- pdo_err = &stmt->error_code;
- }
- /* hash sqlstate to error messages */
- msg = pdo_sqlstate_state_to_description(*pdo_err);
- if (!msg) {
- msg = "<<Unknown error>>";
- }
- ZVAL_UNDEF(&info);
- if (dbh->methods->fetch_err) {
- array_init(&info);
- add_next_index_string(&info, *pdo_err);
- if (dbh->methods->fetch_err(dbh, stmt, &info)) {
- zval *item;
- if ((item = zend_hash_index_find(Z_ARRVAL(info), 1)) != NULL) {
- native_code = Z_LVAL_P(item);
- }
- if ((item = zend_hash_index_find(Z_ARRVAL(info), 2)) != NULL) {
- supp = estrndup(Z_STRVAL_P(item), Z_STRLEN_P(item));
- }
- }
- }
- if (supp) {
- message = strpprintf(0, "SQLSTATE[%s]: %s: " ZEND_LONG_FMT " %s", *pdo_err, msg, native_code, supp);
- } else {
- message = strpprintf(0, "SQLSTATE[%s]: %s", *pdo_err, msg);
- }
- if (dbh->error_mode == PDO_ERRMODE_WARNING) {
- php_error_docref(NULL, E_WARNING, "%s", ZSTR_VAL(message));
- } else if (EG(exception) == NULL) {
- zval ex;
- zend_class_entry *def_ex = php_pdo_get_exception_base(1), *pdo_ex = php_pdo_get_exception();
- object_init_ex(&ex, pdo_ex);
- zend_update_property_str(def_ex, &ex, "message", sizeof("message") - 1, message);
- zend_update_property_string(def_ex, &ex, "code", sizeof("code") - 1, *pdo_err);
- if (!Z_ISUNDEF(info)) {
- zend_update_property(pdo_ex, &ex, "errorInfo", sizeof("errorInfo") - 1, &info);
- }
- zend_throw_exception_object(&ex);
- }
- if (!Z_ISUNDEF(info)) {
- zval_ptr_dtor(&info);
- }
- if (message) {
- zend_string_release_ex(message, 0);
- }
- if (supp) {
- efree(supp);
- }
- }
- /* }}} */
- static char *dsn_from_uri(char *uri, char *buf, size_t buflen) /* {{{ */
- {
- php_stream *stream;
- char *dsn = NULL;
- stream = php_stream_open_wrapper(uri, "rb", REPORT_ERRORS, NULL);
- if (stream) {
- dsn = php_stream_get_line(stream, buf, buflen, NULL);
- php_stream_close(stream);
- }
- return dsn;
- }
- /* }}} */
- /* {{{ proto PDO::__construct(string dsn[, string username[, string passwd [, array options]]])
- */
- static PHP_METHOD(PDO, dbh_constructor)
- {
- zval *object = getThis();
- pdo_dbh_t *dbh = NULL;
- zend_bool is_persistent = 0;
- char *data_source;
- size_t data_source_len;
- char *colon;
- char *username=NULL, *password=NULL;
- size_t usernamelen, passwordlen;
- pdo_driver_t *driver = NULL;
- zval *options = NULL;
- char alt_dsn[512];
- int call_factory = 1;
- zend_error_handling zeh;
- ZEND_PARSE_PARAMETERS_START_EX(ZEND_PARSE_PARAMS_THROW, 1, 4)
- Z_PARAM_STRING(data_source, data_source_len)
- Z_PARAM_OPTIONAL
- Z_PARAM_STRING_EX(username, usernamelen, 1, 0)
- Z_PARAM_STRING_EX(password, passwordlen, 1, 0)
- Z_PARAM_ARRAY_EX(options, 1, 0)
- ZEND_PARSE_PARAMETERS_END();
- /* parse the data source name */
- colon = strchr(data_source, ':');
- if (!colon) {
- /* let's see if this string has a matching dsn in the php.ini */
- char *ini_dsn = NULL;
- snprintf(alt_dsn, sizeof(alt_dsn), "pdo.dsn.%s", data_source);
- if (FAILURE == cfg_get_string(alt_dsn, &ini_dsn)) {
- zend_throw_exception_ex(php_pdo_get_exception(), 0, "invalid data source name");
- return;
- }
- data_source = ini_dsn;
- colon = strchr(data_source, ':');
- if (!colon) {
- zend_throw_exception_ex(php_pdo_get_exception(), 0, "invalid data source name (via INI: %s)", alt_dsn);
- return;
- }
- }
- if (!strncmp(data_source, "uri:", sizeof("uri:")-1)) {
- /* the specified URI holds connection details */
- data_source = dsn_from_uri(data_source + sizeof("uri:")-1, alt_dsn, sizeof(alt_dsn));
- if (!data_source) {
- zend_throw_exception_ex(php_pdo_get_exception(), 0, "invalid data source URI");
- return;
- }
- colon = strchr(data_source, ':');
- if (!colon) {
- zend_throw_exception_ex(php_pdo_get_exception(), 0, "invalid data source name (via URI)");
- return;
- }
- }
- driver = pdo_find_driver(data_source, colon - data_source);
- if (!driver) {
- /* NB: don't want to include the data_source in the error message as
- * it might contain a password */
- zend_throw_exception_ex(php_pdo_get_exception(), 0, "could not find driver");
- return;
- }
- dbh = Z_PDO_DBH_P(object);
- /* is this supposed to be a persistent connection ? */
- if (options) {
- int plen = 0;
- char *hashkey = NULL;
- zend_resource *le;
- pdo_dbh_t *pdbh = NULL;
- zval *v;
- if ((v = zend_hash_index_find_deref(Z_ARRVAL_P(options), PDO_ATTR_PERSISTENT)) != NULL) {
- if (Z_TYPE_P(v) == IS_STRING &&
- !is_numeric_string(Z_STRVAL_P(v), Z_STRLEN_P(v), NULL, NULL, 0) && Z_STRLEN_P(v) > 0) {
- /* user specified key */
- plen = spprintf(&hashkey, 0, "PDO:DBH:DSN=%s:%s:%s:%s", data_source,
- username ? username : "",
- password ? password : "",
- Z_STRVAL_P(v));
- is_persistent = 1;
- } else {
- is_persistent = zval_get_long(v) ? 1 : 0;
- plen = spprintf(&hashkey, 0, "PDO:DBH:DSN=%s:%s:%s", data_source,
- username ? username : "",
- password ? password : "");
- }
- }
- if (is_persistent) {
- /* let's see if we have one cached.... */
- if ((le = zend_hash_str_find_ptr(&EG(persistent_list), hashkey, plen)) != NULL) {
- if (le->type == php_pdo_list_entry()) {
- pdbh = (pdo_dbh_t*)le->ptr;
- /* is the connection still alive ? */
- if (pdbh->methods->check_liveness && FAILURE == (pdbh->methods->check_liveness)(pdbh)) {
- /* nope... need to kill it */
- pdbh->refcount--;
- zend_list_close(le);
- pdbh = NULL;
- }
- }
- }
- if (pdbh) {
- call_factory = 0;
- } else {
- /* need a brand new pdbh */
- pdbh = pecalloc(1, sizeof(*pdbh), 1);
- pdbh->refcount = 1;
- pdbh->is_persistent = 1;
- pdbh->persistent_id = pemalloc(plen + 1, 1);
- memcpy((char *)pdbh->persistent_id, hashkey, plen+1);
- pdbh->persistent_id_len = plen;
- pdbh->def_stmt_ce = dbh->def_stmt_ce;
- }
- }
- if (pdbh) {
- efree(dbh);
- /* switch over to the persistent one */
- Z_PDO_OBJECT_P(object)->inner = pdbh;
- pdbh->refcount++;
- dbh = pdbh;
- }
- if (hashkey) {
- efree(hashkey);
- }
- }
- if (call_factory) {
- dbh->data_source_len = strlen(colon + 1);
- dbh->data_source = (const char*)pestrdup(colon + 1, is_persistent);
- dbh->username = username ? pestrdup(username, is_persistent) : NULL;
- dbh->password = password ? pestrdup(password, is_persistent) : NULL;
- dbh->default_fetch_type = PDO_FETCH_BOTH;
- }
- dbh->auto_commit = pdo_attr_lval(options, PDO_ATTR_AUTOCOMMIT, 1);
- if (!dbh->data_source || (username && !dbh->username) || (password && !dbh->password)) {
- php_error_docref(NULL, E_ERROR, "out of memory");
- }
- zend_replace_error_handling(EH_THROW, pdo_exception_ce, &zeh);
- if (!call_factory) {
- /* we got a persistent guy from our cache */
- goto options;
- }
- if (driver->db_handle_factory(dbh, options)) {
- /* all set */
- if (is_persistent) {
- /* register in the persistent list etc. */
- /* we should also need to replace the object store entry,
- since it was created with emalloc */
- if ((zend_register_persistent_resource(
- (char*)dbh->persistent_id, dbh->persistent_id_len, dbh, php_pdo_list_entry())) == NULL) {
- php_error_docref(NULL, E_ERROR, "Failed to register persistent entry");
- }
- }
- dbh->driver = driver;
- options:
- if (options) {
- zval *attr_value;
- zend_ulong long_key;
- zend_string *str_key = NULL;
- ZEND_HASH_FOREACH_KEY_VAL(Z_ARRVAL_P(options), long_key, str_key, attr_value) {
- if (str_key) {
- continue;
- }
- ZVAL_DEREF(attr_value);
- pdo_dbh_attribute_set(dbh, long_key, attr_value);
- } ZEND_HASH_FOREACH_END();
- }
- zend_restore_error_handling(&zeh);
- return;
- }
- /* the connection failed; things will tidy up in free_storage */
- /* XXX raise exception */
- zend_restore_error_handling(&zeh);
- if (!EG(exception)) {
- zend_throw_exception(pdo_exception_ce, "Constructor failed", 0);
- }
- }
- /* }}} */
- static zval *pdo_stmt_instantiate(pdo_dbh_t *dbh, zval *object, zend_class_entry *dbstmt_ce, zval *ctor_args) /* {{{ */
- {
- if (!Z_ISUNDEF_P(ctor_args)) {
- if (Z_TYPE_P(ctor_args) != IS_ARRAY) {
- pdo_raise_impl_error(dbh, NULL, "HY000", "constructor arguments must be passed as an array");
- return NULL;
- }
- if (!dbstmt_ce->constructor) {
- pdo_raise_impl_error(dbh, NULL, "HY000", "user-supplied statement does not accept constructor arguments");
- return NULL;
- }
- }
- if (UNEXPECTED(object_init_ex(object, dbstmt_ce) != SUCCESS)) {
- return NULL;
- }
- return object;
- } /* }}} */
- static void pdo_stmt_construct(zend_execute_data *execute_data, pdo_stmt_t *stmt, zval *object, zend_class_entry *dbstmt_ce, zval *ctor_args) /* {{{ */
- {
- zval query_string;
- zval z_key;
- ZVAL_STRINGL(&query_string, stmt->query_string, stmt->query_stringlen);
- ZVAL_STRINGL(&z_key, "queryString", sizeof("queryString") - 1);
- zend_std_write_property(object, &z_key, &query_string, NULL);
- zval_ptr_dtor(&query_string);
- zval_ptr_dtor(&z_key);
- if (dbstmt_ce->constructor) {
- zend_fcall_info fci;
- zend_fcall_info_cache fcc;
- zval retval;
- fci.size = sizeof(zend_fcall_info);
- ZVAL_UNDEF(&fci.function_name);
- fci.object = Z_OBJ_P(object);
- fci.retval = &retval;
- fci.param_count = 0;
- fci.params = NULL;
- fci.no_separation = 1;
- zend_fcall_info_args(&fci, ctor_args);
- fcc.function_handler = dbstmt_ce->constructor;
- fcc.called_scope = Z_OBJCE_P(object);
- fcc.object = Z_OBJ_P(object);
- if (zend_call_function(&fci, &fcc) != FAILURE) {
- zval_ptr_dtor(&retval);
- }
- zend_fcall_info_args_clear(&fci, 1);
- }
- }
- /* }}} */
- /* {{{ proto object PDO::prepare(string statement [, array options])
- Prepares a statement for execution and returns a statement object */
- static PHP_METHOD(PDO, prepare)
- {
- pdo_stmt_t *stmt;
- char *statement;
- size_t statement_len;
- zval *options = NULL, *opt, *item, ctor_args;
- zend_class_entry *dbstmt_ce, *pce;
- pdo_dbh_object_t *dbh_obj = Z_PDO_OBJECT_P(getThis());
- pdo_dbh_t *dbh = dbh_obj->inner;
- ZEND_PARSE_PARAMETERS_START(1, 2)
- Z_PARAM_STRING(statement, statement_len)
- Z_PARAM_OPTIONAL
- Z_PARAM_ARRAY(options)
- ZEND_PARSE_PARAMETERS_END_EX(RETURN_FALSE);
- PDO_DBH_CLEAR_ERR();
- PDO_CONSTRUCT_CHECK;
- if (ZEND_NUM_ARGS() > 1 && (opt = zend_hash_index_find(Z_ARRVAL_P(options), PDO_ATTR_STATEMENT_CLASS)) != NULL) {
- if (Z_TYPE_P(opt) != IS_ARRAY || (item = zend_hash_index_find(Z_ARRVAL_P(opt), 0)) == NULL
- || Z_TYPE_P(item) != IS_STRING
- || (pce = zend_lookup_class(Z_STR_P(item))) == NULL
- ) {
- pdo_raise_impl_error(dbh, NULL, "HY000",
- "PDO::ATTR_STATEMENT_CLASS requires format array(classname, array(ctor_args)); "
- "the classname must be a string specifying an existing class"
- );
- PDO_HANDLE_DBH_ERR();
- RETURN_FALSE;
- }
- dbstmt_ce = pce;
- if (!instanceof_function(dbstmt_ce, pdo_dbstmt_ce)) {
- pdo_raise_impl_error(dbh, NULL, "HY000",
- "user-supplied statement class must be derived from PDOStatement");
- PDO_HANDLE_DBH_ERR();
- RETURN_FALSE;
- }
- if (dbstmt_ce->constructor && !(dbstmt_ce->constructor->common.fn_flags & (ZEND_ACC_PRIVATE|ZEND_ACC_PROTECTED))) {
- pdo_raise_impl_error(dbh, NULL, "HY000",
- "user-supplied statement class cannot have a public constructor");
- PDO_HANDLE_DBH_ERR();
- RETURN_FALSE;
- }
- if ((item = zend_hash_index_find(Z_ARRVAL_P(opt), 1)) != NULL) {
- if (Z_TYPE_P(item) != IS_ARRAY) {
- pdo_raise_impl_error(dbh, NULL, "HY000",
- "PDO::ATTR_STATEMENT_CLASS requires format array(classname, ctor_args); "
- "ctor_args must be an array"
- );
- PDO_HANDLE_DBH_ERR();
- RETURN_FALSE;
- }
- ZVAL_COPY_VALUE(&ctor_args, item);
- } else {
- ZVAL_UNDEF(&ctor_args);
- }
- } else {
- dbstmt_ce = dbh->def_stmt_ce;
- ZVAL_COPY_VALUE(&ctor_args, &dbh->def_stmt_ctor_args);
- }
- if (!pdo_stmt_instantiate(dbh, return_value, dbstmt_ce, &ctor_args)) {
- if (EXPECTED(!EG(exception))) {
- pdo_raise_impl_error(dbh, NULL, "HY000",
- "failed to instantiate user-supplied statement class"
- );
- }
- PDO_HANDLE_DBH_ERR();
- RETURN_FALSE;
- }
- stmt = Z_PDO_STMT_P(return_value);
- /* unconditionally keep this for later reference */
- stmt->query_string = estrndup(statement, statement_len);
- stmt->query_stringlen = statement_len;
- stmt->default_fetch_type = dbh->default_fetch_type;
- stmt->dbh = dbh;
- /* give it a reference to me */
- ZVAL_OBJ(&stmt->database_object_handle, &dbh_obj->std);
- Z_ADDREF(stmt->database_object_handle);
- /* we haven't created a lazy object yet */
- ZVAL_UNDEF(&stmt->lazy_object_ref);
- if (dbh->methods->preparer(dbh, statement, statement_len, stmt, options)) {
- pdo_stmt_construct(execute_data, stmt, return_value, dbstmt_ce, &ctor_args);
- return;
- }
- PDO_HANDLE_DBH_ERR();
- /* kill the object handle for the stmt here */
- zval_ptr_dtor(return_value);
- RETURN_FALSE;
- }
- /* }}} */
- /* {{{ proto bool PDO::beginTransaction()
- Initiates a transaction */
- static PHP_METHOD(PDO, beginTransaction)
- {
- pdo_dbh_t *dbh = Z_PDO_DBH_P(getThis());
- if (zend_parse_parameters_none() == FAILURE) {
- return;
- }
- PDO_CONSTRUCT_CHECK;
- if (dbh->in_txn) {
- zend_throw_exception_ex(php_pdo_get_exception(), 0, "There is already an active transaction");
- RETURN_FALSE;
- }
- if (!dbh->methods->begin) {
- /* TODO: this should be an exception; see the auto-commit mode
- * comments below */
- zend_throw_exception_ex(php_pdo_get_exception(), 0, "This driver doesn't support transactions");
- RETURN_FALSE;
- }
- if (dbh->methods->begin(dbh)) {
- dbh->in_txn = 1;
- RETURN_TRUE;
- }
- PDO_HANDLE_DBH_ERR();
- RETURN_FALSE;
- }
- /* }}} */
- /* {{{ proto bool PDO::commit()
- Commit a transaction */
- static PHP_METHOD(PDO, commit)
- {
- pdo_dbh_t *dbh = Z_PDO_DBH_P(getThis());
- if (zend_parse_parameters_none() == FAILURE) {
- return;
- }
- PDO_CONSTRUCT_CHECK;
- if (!dbh->in_txn) {
- zend_throw_exception_ex(php_pdo_get_exception(), 0, "There is no active transaction");
- RETURN_FALSE;
- }
- if (dbh->methods->commit(dbh)) {
- dbh->in_txn = 0;
- RETURN_TRUE;
- }
- PDO_HANDLE_DBH_ERR();
- RETURN_FALSE;
- }
- /* }}} */
- /* {{{ proto bool PDO::rollBack()
- roll back a transaction */
- static PHP_METHOD(PDO, rollBack)
- {
- pdo_dbh_t *dbh = Z_PDO_DBH_P(getThis());
- if (zend_parse_parameters_none() == FAILURE) {
- return;
- }
- PDO_CONSTRUCT_CHECK;
- if (!dbh->in_txn) {
- zend_throw_exception_ex(php_pdo_get_exception(), 0, "There is no active transaction");
- RETURN_FALSE;
- }
- if (dbh->methods->rollback(dbh)) {
- dbh->in_txn = 0;
- RETURN_TRUE;
- }
- PDO_HANDLE_DBH_ERR();
- RETURN_FALSE;
- }
- /* }}} */
- /* {{{ proto bool PDO::inTransaction()
- determine if inside a transaction */
- static PHP_METHOD(PDO, inTransaction)
- {
- pdo_dbh_t *dbh = Z_PDO_DBH_P(getThis());
- if (zend_parse_parameters_none() == FAILURE) {
- return;
- }
- PDO_CONSTRUCT_CHECK;
- if (!dbh->methods->in_transaction) {
- RETURN_BOOL(dbh->in_txn);
- }
- RETURN_BOOL(dbh->methods->in_transaction(dbh));
- }
- /* }}} */
- static int pdo_dbh_attribute_set(pdo_dbh_t *dbh, zend_long attr, zval *value) /* {{{ */
- {
- zend_long lval;
- #define PDO_LONG_PARAM_CHECK \
- if (Z_TYPE_P(value) != IS_LONG && Z_TYPE_P(value) != IS_STRING && Z_TYPE_P(value) != IS_FALSE && Z_TYPE_P(value) != IS_TRUE) { \
- pdo_raise_impl_error(dbh, NULL, "HY000", "attribute value must be an integer"); \
- PDO_HANDLE_DBH_ERR(); \
- return FAILURE; \
- } \
- switch (attr) {
- case PDO_ATTR_ERRMODE:
- PDO_LONG_PARAM_CHECK;
- lval = zval_get_long(value);
- switch (lval) {
- case PDO_ERRMODE_SILENT:
- case PDO_ERRMODE_WARNING:
- case PDO_ERRMODE_EXCEPTION:
- dbh->error_mode = lval;
- return SUCCESS;
- default:
- pdo_raise_impl_error(dbh, NULL, "HY000", "invalid error mode");
- PDO_HANDLE_DBH_ERR();
- return FAILURE;
- }
- return FAILURE;
- case PDO_ATTR_CASE:
- PDO_LONG_PARAM_CHECK;
- lval = zval_get_long(value);
- switch (lval) {
- case PDO_CASE_NATURAL:
- case PDO_CASE_UPPER:
- case PDO_CASE_LOWER:
- dbh->desired_case = lval;
- return SUCCESS;
- default:
- pdo_raise_impl_error(dbh, NULL, "HY000", "invalid case folding mode");
- PDO_HANDLE_DBH_ERR();
- return FAILURE;
- }
- return FAILURE;
- case PDO_ATTR_ORACLE_NULLS:
- PDO_LONG_PARAM_CHECK;
- dbh->oracle_nulls = zval_get_long(value);
- return SUCCESS;
- case PDO_ATTR_DEFAULT_FETCH_MODE:
- if (Z_TYPE_P(value) == IS_ARRAY) {
- zval *tmp;
- if ((tmp = zend_hash_index_find(Z_ARRVAL_P(value), 0)) != NULL && Z_TYPE_P(tmp) == IS_LONG) {
- if (Z_LVAL_P(tmp) == PDO_FETCH_INTO || Z_LVAL_P(tmp) == PDO_FETCH_CLASS) {
- pdo_raise_impl_error(dbh, NULL, "HY000", "FETCH_INTO and FETCH_CLASS are not yet supported as default fetch modes");
- return FAILURE;
- }
- }
- } else {
- PDO_LONG_PARAM_CHECK;
- }
- lval = zval_get_long(value);
- if (lval == PDO_FETCH_USE_DEFAULT) {
- pdo_raise_impl_error(dbh, NULL, "HY000", "invalid fetch mode type");
- return FAILURE;
- }
- dbh->default_fetch_type = lval;
- return SUCCESS;
- case PDO_ATTR_STRINGIFY_FETCHES:
- PDO_LONG_PARAM_CHECK;
- dbh->stringify = zval_get_long(value) ? 1 : 0;
- return SUCCESS;
- case PDO_ATTR_STATEMENT_CLASS: {
- /* array(string classname, array(mixed ctor_args)) */
- zend_class_entry *pce;
- zval *item;
- if (dbh->is_persistent) {
- pdo_raise_impl_error(dbh, NULL, "HY000",
- "PDO::ATTR_STATEMENT_CLASS cannot be used with persistent PDO instances"
- );
- PDO_HANDLE_DBH_ERR();
- return FAILURE;
- }
- if (Z_TYPE_P(value) != IS_ARRAY
- || (item = zend_hash_index_find(Z_ARRVAL_P(value), 0)) == NULL
- || Z_TYPE_P(item) != IS_STRING
- || (pce = zend_lookup_class(Z_STR_P(item))) == NULL
- ) {
- pdo_raise_impl_error(dbh, NULL, "HY000",
- "PDO::ATTR_STATEMENT_CLASS requires format array(classname, array(ctor_args)); "
- "the classname must be a string specifying an existing class"
- );
- PDO_HANDLE_DBH_ERR();
- return FAILURE;
- }
- if (!instanceof_function(pce, pdo_dbstmt_ce)) {
- pdo_raise_impl_error(dbh, NULL, "HY000",
- "user-supplied statement class must be derived from PDOStatement");
- PDO_HANDLE_DBH_ERR();
- return FAILURE;
- }
- if (pce->constructor && !(pce->constructor->common.fn_flags & (ZEND_ACC_PRIVATE|ZEND_ACC_PROTECTED))) {
- pdo_raise_impl_error(dbh, NULL, "HY000",
- "user-supplied statement class cannot have a public constructor");
- PDO_HANDLE_DBH_ERR();
- return FAILURE;
- }
- dbh->def_stmt_ce = pce;
- if (!Z_ISUNDEF(dbh->def_stmt_ctor_args)) {
- zval_ptr_dtor(&dbh->def_stmt_ctor_args);
- ZVAL_UNDEF(&dbh->def_stmt_ctor_args);
- }
- if ((item = zend_hash_index_find(Z_ARRVAL_P(value), 1)) != NULL) {
- if (Z_TYPE_P(item) != IS_ARRAY) {
- pdo_raise_impl_error(dbh, NULL, "HY000",
- "PDO::ATTR_STATEMENT_CLASS requires format array(classname, array(ctor_args)); "
- "ctor_args must be an array"
- );
- PDO_HANDLE_DBH_ERR();
- return FAILURE;
- }
- ZVAL_COPY(&dbh->def_stmt_ctor_args, item);
- }
- return SUCCESS;
- }
- default:
- ;
- }
- if (!dbh->methods->set_attribute) {
- goto fail;
- }
- PDO_DBH_CLEAR_ERR();
- if (dbh->methods->set_attribute(dbh, attr, value)) {
- return SUCCESS;
- }
- fail:
- if (attr == PDO_ATTR_AUTOCOMMIT) {
- zend_throw_exception_ex(php_pdo_get_exception(), 0, "The auto-commit mode cannot be changed for this driver");
- } else if (!dbh->methods->set_attribute) {
- pdo_raise_impl_error(dbh, NULL, "IM001", "driver does not support setting attributes");
- } else {
- PDO_HANDLE_DBH_ERR();
- }
- return FAILURE;
- }
- /* }}} */
- /* {{{ proto bool PDO::setAttribute(int attribute, mixed value)
- Set an attribute */
- static PHP_METHOD(PDO, setAttribute)
- {
- pdo_dbh_t *dbh = Z_PDO_DBH_P(getThis());
- zend_long attr;
- zval *value;
- ZEND_PARSE_PARAMETERS_START(2, 2)
- Z_PARAM_LONG(attr)
- Z_PARAM_ZVAL(value)
- ZEND_PARSE_PARAMETERS_END_EX(RETURN_FALSE);
- PDO_DBH_CLEAR_ERR();
- PDO_CONSTRUCT_CHECK;
- if (pdo_dbh_attribute_set(dbh, attr, value) != FAILURE) {
- RETURN_TRUE;
- }
- RETURN_FALSE;
- }
- /* }}} */
- /* {{{ proto mixed PDO::getAttribute(int attribute)
- Get an attribute */
- static PHP_METHOD(PDO, getAttribute)
- {
- pdo_dbh_t *dbh = Z_PDO_DBH_P(getThis());
- zend_long attr;
- ZEND_PARSE_PARAMETERS_START(1, 1)
- Z_PARAM_LONG(attr)
- ZEND_PARSE_PARAMETERS_END_EX(RETURN_FALSE);
- PDO_DBH_CLEAR_ERR();
- PDO_CONSTRUCT_CHECK;
- /* handle generic PDO-level attributes */
- switch (attr) {
- case PDO_ATTR_PERSISTENT:
- RETURN_BOOL(dbh->is_persistent);
- case PDO_ATTR_CASE:
- RETURN_LONG(dbh->desired_case);
- case PDO_ATTR_ORACLE_NULLS:
- RETURN_LONG(dbh->oracle_nulls);
- case PDO_ATTR_ERRMODE:
- RETURN_LONG(dbh->error_mode);
- case PDO_ATTR_DRIVER_NAME:
- RETURN_STRINGL((char*)dbh->driver->driver_name, dbh->driver->driver_name_len);
- case PDO_ATTR_STATEMENT_CLASS:
- array_init(return_value);
- add_next_index_str(return_value, zend_string_copy(dbh->def_stmt_ce->name));
- if (!Z_ISUNDEF(dbh->def_stmt_ctor_args)) {
- Z_TRY_ADDREF(dbh->def_stmt_ctor_args);
- add_next_index_zval(return_value, &dbh->def_stmt_ctor_args);
- }
- return;
- case PDO_ATTR_DEFAULT_FETCH_MODE:
- RETURN_LONG(dbh->default_fetch_type);
- default:
- break;
- }
- if (!dbh->methods->get_attribute) {
- pdo_raise_impl_error(dbh, NULL, "IM001", "driver does not support getting attributes");
- RETURN_FALSE;
- }
- switch (dbh->methods->get_attribute(dbh, attr, return_value)) {
- case -1:
- PDO_HANDLE_DBH_ERR();
- RETURN_FALSE;
- case 0:
- pdo_raise_impl_error(dbh, NULL, "IM001", "driver does not support that attribute");
- RETURN_FALSE;
- default:
- return;
- }
- }
- /* }}} */
- /* {{{ proto int PDO::exec(string query)
- Execute a query that does not return a row set, returning the number of affected rows */
- static PHP_METHOD(PDO, exec)
- {
- pdo_dbh_t *dbh = Z_PDO_DBH_P(getThis());
- char *statement;
- size_t statement_len;
- zend_long ret;
- ZEND_PARSE_PARAMETERS_START(1, 1)
- Z_PARAM_STRING(statement, statement_len)
- ZEND_PARSE_PARAMETERS_END_EX(RETURN_FALSE);
- if (!statement_len) {
- pdo_raise_impl_error(dbh, NULL, "HY000", "trying to execute an empty query");
- RETURN_FALSE;
- }
- PDO_DBH_CLEAR_ERR();
- PDO_CONSTRUCT_CHECK;
- ret = dbh->methods->doer(dbh, statement, statement_len);
- if(ret == -1) {
- PDO_HANDLE_DBH_ERR();
- RETURN_FALSE;
- } else {
- RETURN_LONG(ret);
- }
- }
- /* }}} */
- /* {{{ proto string PDO::lastInsertId([string seqname])
- Returns the id of the last row that we affected on this connection. Some databases require a sequence or table name to be passed in. Not always meaningful. */
- static PHP_METHOD(PDO, lastInsertId)
- {
- pdo_dbh_t *dbh = Z_PDO_DBH_P(getThis());
- char *name = NULL;
- size_t namelen;
- ZEND_PARSE_PARAMETERS_START(0, 1)
- Z_PARAM_OPTIONAL
- Z_PARAM_STRING_EX(name, namelen, 1, 0)
- ZEND_PARSE_PARAMETERS_END_EX(RETURN_FALSE);
- PDO_DBH_CLEAR_ERR();
- PDO_CONSTRUCT_CHECK;
- if (!dbh->methods->last_id) {
- pdo_raise_impl_error(dbh, NULL, "IM001", "driver does not support lastInsertId()");
- RETURN_FALSE;
- } else {
- size_t id_len;
- char *id;
- id = dbh->methods->last_id(dbh, name, &id_len);
- if (!id) {
- PDO_HANDLE_DBH_ERR();
- RETURN_FALSE;
- } else {
- //??? use zend_string ?
- RETVAL_STRINGL(id, id_len);
- efree(id);
- }
- }
- }
- /* }}} */
- /* {{{ proto string PDO::errorCode()
- Fetch the error code associated with the last operation on the database handle */
- static PHP_METHOD(PDO, errorCode)
- {
- pdo_dbh_t *dbh = Z_PDO_DBH_P(getThis());
- if (zend_parse_parameters_none() == FAILURE) {
- return;
- }
- PDO_CONSTRUCT_CHECK;
- if (dbh->query_stmt) {
- RETURN_STRING(dbh->query_stmt->error_code);
- }
- if (dbh->error_code[0] == '\0') {
- RETURN_NULL();
- }
- /**
- * Making sure that we fallback to the default implementation
- * if the dbh->error_code is not null.
- */
- RETURN_STRING(dbh->error_code);
- }
- /* }}} */
- /* {{{ proto int PDO::errorInfo()
- Fetch extended error information associated with the last operation on the database handle */
- static PHP_METHOD(PDO, errorInfo)
- {
- int error_count;
- int error_count_diff = 0;
- int error_expected_count = 3;
- pdo_dbh_t *dbh = Z_PDO_DBH_P(getThis());
- if (zend_parse_parameters_none() == FAILURE) {
- return;
- }
- PDO_CONSTRUCT_CHECK;
- array_init(return_value);
- if (dbh->query_stmt) {
- add_next_index_string(return_value, dbh->query_stmt->error_code);
- if(!strncmp(dbh->query_stmt->error_code, PDO_ERR_NONE, sizeof(PDO_ERR_NONE))) goto fill_array;
- } else {
- add_next_index_string(return_value, dbh->error_code);
- if(!strncmp(dbh->error_code, PDO_ERR_NONE, sizeof(PDO_ERR_NONE))) goto fill_array;
- }
- if (dbh->methods->fetch_err) {
- dbh->methods->fetch_err(dbh, dbh->query_stmt, return_value);
- }
- fill_array:
- /**
- * In order to be consistent, we have to make sure we add the good amount
- * of nulls depending on the current number of elements. We make a simple
- * difference and add the needed elements
- */
- error_count = zend_hash_num_elements(Z_ARRVAL_P(return_value));
- if (error_expected_count > error_count) {
- int current_index;
- error_count_diff = error_expected_count - error_count;
- for (current_index = 0; current_index < error_count_diff; current_index++) {
- add_next_index_null(return_value);
- }
- }
- }
- /* }}} */
- /* {{{ proto object PDO::query(string sql [, PDOStatement::setFetchMode() args])
- Prepare and execute $sql; returns the statement object for iteration */
- static PHP_METHOD(PDO, query)
- {
- pdo_stmt_t *stmt;
- char *statement;
- size_t statement_len;
- pdo_dbh_object_t *dbh_obj = Z_PDO_OBJECT_P(getThis());
- pdo_dbh_t *dbh = dbh_obj->inner;
- /* Return a meaningful error when no parameters were passed */
- if (!ZEND_NUM_ARGS()) {
- zend_parse_parameters(0, "z|z", NULL, NULL);
- RETURN_FALSE;
- }
- if (FAILURE == zend_parse_parameters(1, "s", &statement,
- &statement_len)) {
- RETURN_FALSE;
- }
- PDO_DBH_CLEAR_ERR();
- PDO_CONSTRUCT_CHECK;
- if (!pdo_stmt_instantiate(dbh, return_value, dbh->def_stmt_ce, &dbh->def_stmt_ctor_args)) {
- if (EXPECTED(!EG(exception))) {
- pdo_raise_impl_error(dbh, NULL, "HY000", "failed to instantiate user supplied statement class");
- }
- return;
- }
- stmt = Z_PDO_STMT_P(return_value);
- /* unconditionally keep this for later reference */
- stmt->query_string = estrndup(statement, statement_len);
- stmt->query_stringlen = statement_len;
- stmt->default_fetch_type = dbh->default_fetch_type;
- stmt->active_query_string = stmt->query_string;
- stmt->active_query_stringlen = statement_len;
- stmt->dbh = dbh;
- /* give it a reference to me */
- ZVAL_OBJ(&stmt->database_object_handle, &dbh_obj->std);
- Z_ADDREF(stmt->database_object_handle);
- /* we haven't created a lazy object yet */
- ZVAL_UNDEF(&stmt->lazy_object_ref);
- if (dbh->methods->preparer(dbh, statement, statement_len, stmt, NULL)) {
- PDO_STMT_CLEAR_ERR();
- if (ZEND_NUM_ARGS() == 1 || SUCCESS == pdo_stmt_setup_fetch_mode(INTERNAL_FUNCTION_PARAM_PASSTHRU, stmt, 1)) {
- /* now execute the statement */
- PDO_STMT_CLEAR_ERR();
- if (stmt->methods->executer(stmt)) {
- int ret = 1;
- if (!stmt->executed) {
- if (stmt->dbh->alloc_own_columns) {
- ret = pdo_stmt_describe_columns(stmt);
- }
- stmt->executed = 1;
- }
- if (ret) {
- pdo_stmt_construct(execute_data, stmt, return_value, dbh->def_stmt_ce, &dbh->def_stmt_ctor_args);
- return;
- }
- }
- }
- /* something broke */
- dbh->query_stmt = stmt;
- ZVAL_COPY_VALUE(&dbh->query_stmt_zval, return_value);
- Z_DELREF(stmt->database_object_handle);
- ZVAL_UNDEF(&stmt->database_object_handle);
- PDO_HANDLE_STMT_ERR();
- } else {
- PDO_HANDLE_DBH_ERR();
- zval_ptr_dtor(return_value);
- }
- RETURN_FALSE;
- }
- /* }}} */
- /* {{{ proto string PDO::quote(string string [, int paramtype])
- quotes string for use in a query. The optional paramtype acts as a hint for drivers that have alternate quoting styles. The default value is PDO_PARAM_STR */
- static PHP_METHOD(PDO, quote)
- {
- pdo_dbh_t *dbh = Z_PDO_DBH_P(getThis());
- char *str;
- size_t str_len;
- zend_long paramtype = PDO_PARAM_STR;
- char *qstr;
- size_t qlen;
- ZEND_PARSE_PARAMETERS_START(1, 2)
- Z_PARAM_STRING(str, str_len)
- Z_PARAM_OPTIONAL
- Z_PARAM_LONG(paramtype)
- ZEND_PARSE_PARAMETERS_END_EX(RETURN_FALSE);
- PDO_DBH_CLEAR_ERR();
- PDO_CONSTRUCT_CHECK;
- if (!dbh->methods->quoter) {
- pdo_raise_impl_error(dbh, NULL, "IM001", "driver does not support quoting");
- RETURN_FALSE;
- }
- if (dbh->methods->quoter(dbh, str, str_len, &qstr, &qlen, paramtype)) {
- RETVAL_STRINGL(qstr, qlen);
- efree(qstr);
- return;
- }
- PDO_HANDLE_DBH_ERR();
- RETURN_FALSE;
- }
- /* }}} */
- /* {{{ proto PDO::__wakeup()
- Prevents use of a PDO instance that has been unserialized */
- static PHP_METHOD(PDO, __wakeup)
- {
- zend_throw_exception_ex(php_pdo_get_exception(), 0, "You cannot serialize or unserialize PDO instances");
- }
- /* }}} */
- /* {{{ proto int PDO::__sleep()
- Prevents serialization of a PDO instance */
- static PHP_METHOD(PDO, __sleep)
- {
- zend_throw_exception_ex(php_pdo_get_exception(), 0, "You cannot serialize or unserialize PDO instances");
- }
- /* }}} */
- /* {{{ proto array PDO::getAvailableDrivers()
- Return array of available PDO drivers */
- static PHP_METHOD(PDO, getAvailableDrivers)
- {
- pdo_driver_t *pdriver;
- if (zend_parse_parameters_none() == FAILURE) {
- return;
- }
- array_init(return_value);
- ZEND_HASH_FOREACH_PTR(&pdo_driver_hash, pdriver) {
- add_next_index_stringl(return_value, (char*)pdriver->driver_name, pdriver->driver_name_len);
- } ZEND_HASH_FOREACH_END();
- }
- /* }}} */
- /* {{{ arginfo */
- ZEND_BEGIN_ARG_INFO_EX(arginfo_pdo___construct, 0, 0, 1)
- ZEND_ARG_INFO(0, dsn)
- ZEND_ARG_INFO(0, username)
- ZEND_ARG_INFO(0, passwd)
- ZEND_ARG_INFO(0, options) /* array */
- ZEND_END_ARG_INFO()
- ZEND_BEGIN_ARG_INFO_EX(arginfo_pdo_prepare, 0, 0, 1)
- ZEND_ARG_INFO(0, statement)
- ZEND_ARG_INFO(0, options) /* array */
- ZEND_END_ARG_INFO()
- ZEND_BEGIN_ARG_INFO(arginfo_pdo_setattribute, 0)
- ZEND_ARG_INFO(0, attribute)
- ZEND_ARG_INFO(0, value)
- ZEND_END_ARG_INFO()
- ZEND_BEGIN_ARG_INFO(arginfo_pdo_getattribute, 0)
- ZEND_ARG_INFO(0, attribute)
- ZEND_END_ARG_INFO()
- ZEND_BEGIN_ARG_INFO(arginfo_pdo_exec, 0)
- ZEND_ARG_INFO(0, query)
- ZEND_END_ARG_INFO()
- ZEND_BEGIN_ARG_INFO_EX(arginfo_pdo_lastinsertid, 0, 0, 0)
- ZEND_ARG_INFO(0, seqname)
- ZEND_END_ARG_INFO()
- ZEND_BEGIN_ARG_INFO_EX(arginfo_pdo_quote, 0, 0, 1)
- ZEND_ARG_INFO(0, string)
- ZEND_ARG_INFO(0, paramtype)
- ZEND_END_ARG_INFO()
- ZEND_BEGIN_ARG_INFO(arginfo_pdo__void, 0)
- ZEND_END_ARG_INFO()
- /* }}} */
- const zend_function_entry pdo_dbh_functions[] = /* {{{ */ {
- ZEND_MALIAS(PDO, __construct, dbh_constructor, arginfo_pdo___construct, ZEND_ACC_PUBLIC)
- PHP_ME(PDO, prepare, arginfo_pdo_prepare, ZEND_ACC_PUBLIC)
- PHP_ME(PDO, beginTransaction, arginfo_pdo__void, ZEND_ACC_PUBLIC)
- PHP_ME(PDO, commit, arginfo_pdo__void, ZEND_ACC_PUBLIC)
- PHP_ME(PDO, rollBack, arginfo_pdo__void, ZEND_ACC_PUBLIC)
- PHP_ME(PDO, inTransaction, arginfo_pdo__void, ZEND_ACC_PUBLIC)
- PHP_ME(PDO, setAttribute, arginfo_pdo_setattribute, ZEND_ACC_PUBLIC)
- PHP_ME(PDO, exec, arginfo_pdo_exec, ZEND_ACC_PUBLIC)
- PHP_ME(PDO, query, NULL, ZEND_ACC_PUBLIC)
- PHP_ME(PDO, lastInsertId, arginfo_pdo_lastinsertid, ZEND_ACC_PUBLIC)
- PHP_ME(PDO, errorCode, arginfo_pdo__void, ZEND_ACC_PUBLIC)
- PHP_ME(PDO, errorInfo, arginfo_pdo__void, ZEND_ACC_PUBLIC)
- PHP_ME(PDO, getAttribute, arginfo_pdo_getattribute, ZEND_ACC_PUBLIC)
- PHP_ME(PDO, quote, arginfo_pdo_quote, ZEND_ACC_PUBLIC)
- PHP_ME(PDO, __wakeup, arginfo_pdo__void, ZEND_ACC_PUBLIC|ZEND_ACC_FINAL)
- PHP_ME(PDO, __sleep, arginfo_pdo__void, ZEND_ACC_PUBLIC|ZEND_ACC_FINAL)
- PHP_ME(PDO, getAvailableDrivers, arginfo_pdo__void, ZEND_ACC_PUBLIC|ZEND_ACC_STATIC)
- PHP_FE_END
- };
- /* }}} */
- static void cls_method_dtor(zval *el) /* {{{ */ {
- zend_function *func = (zend_function*)Z_PTR_P(el);
- if (func->common.function_name) {
- zend_string_release_ex(func->common.function_name, 0);
- }
- efree(func);
- }
- /* }}} */
- static void cls_method_pdtor(zval *el) /* {{{ */ {
- zend_function *func = (zend_function*)Z_PTR_P(el);
- if (func->common.function_name) {
- zend_string_release_ex(func->common.function_name, 1);
- }
- pefree(func, 1);
- }
- /* }}} */
- /* {{{ overloaded object handlers for PDO class */
- int pdo_hash_methods(pdo_dbh_object_t *dbh_obj, int kind)
- {
- const zend_function_entry *funcs;
- zend_internal_function func;
- size_t namelen;
- char *lc_name;
- pdo_dbh_t *dbh = dbh_obj->inner;
- if (!dbh || !dbh->methods || !dbh->methods->get_driver_methods) {
- return 0;
- }
- funcs = dbh->methods->get_driver_methods(dbh, kind);
- if (!funcs) {
- return 0;
- }
- dbh->cls_methods[kind] = pemalloc(sizeof(HashTable), dbh->is_persistent);
- zend_hash_init_ex(dbh->cls_methods[kind], 8, NULL,
- dbh->is_persistent? cls_method_pdtor : cls_method_dtor, dbh->is_persistent, 0);
- memset(&func, 0, sizeof(func));
- while (funcs->fname) {
- func.type = ZEND_INTERNAL_FUNCTION;
- func.handler = funcs->handler;
- func.function_name = zend_string_init(funcs->fname, strlen(funcs->fname), dbh->is_persistent);
- func.scope = dbh_obj->std.ce;
- func.prototype = NULL;
- if (funcs->flags) {
- func.fn_flags = funcs->flags | ZEND_ACC_NEVER_CACHE;
- } else {
- func.fn_flags = ZEND_ACC_PUBLIC | ZEND_ACC_NEVER_CACHE;
- }
- if (funcs->arg_info) {
- zend_internal_function_info *info = (zend_internal_function_info*)funcs->arg_info;
- func.arg_info = (zend_internal_arg_info*)funcs->arg_info + 1;
- func.num_args = funcs->num_args;
- if (info->required_num_args == (uint32_t)-1) {
- func.required_num_args = funcs->num_args;
- } else {
- func.required_num_args = info->required_num_args;
- }
- if (info->return_reference) {
- func.fn_flags |= ZEND_ACC_RETURN_REFERENCE;
- }
- if (funcs->arg_info[funcs->num_args].is_variadic) {
- func.fn_flags |= ZEND_ACC_VARIADIC;
- /* Don't count the variadic argument */
- func.num_args--;
- }
- } else {
- func.arg_info = NULL;
- func.num_args = 0;
- func.required_num_args = 0;
- }
- zend_set_function_arg_flags((zend_function*)&func);
- namelen = strlen(funcs->fname);
- lc_name = emalloc(namelen+1);
- zend_str_tolower_copy(lc_name, funcs->fname, namelen);
- zend_hash_str_add_mem(dbh->cls_methods[kind], lc_name, namelen, &func, sizeof(func));
- efree(lc_name);
- funcs++;
- }
- return 1;
- }
- static union _zend_function *dbh_method_get(zend_object **object, zend_string *method_name, const zval *key)
- {
- zend_function *fbc = NULL;
- pdo_dbh_object_t *dbh_obj = php_pdo_dbh_fetch_object(*object);
- zend_string *lc_method_name;
- if ((fbc = zend_std_get_method(object, method_name, key)) == NULL) {
- /* not a pre-defined method, nor a user-defined method; check
- * the driver specific methods */
- if (!dbh_obj->inner->cls_methods[PDO_DBH_DRIVER_METHOD_KIND_DBH]) {
- if (!pdo_hash_methods(dbh_obj,
- PDO_DBH_DRIVER_METHOD_KIND_DBH)
- || !dbh_obj->inner->cls_methods[PDO_DBH_DRIVER_METHOD_KIND_DBH]) {
- goto out;
- }
- }
- lc_method_name = zend_string_tolower(method_name);
- fbc = zend_hash_find_ptr(dbh_obj->inner->cls_methods[PDO_DBH_DRIVER_METHOD_KIND_DBH], lc_method_name);
- zend_string_release_ex(lc_method_name, 0);
- }
- out:
- return fbc;
- }
- static int dbh_compare(zval *object1, zval *object2)
- {
- return -1;
- }
- static HashTable *dbh_get_gc(zval *object, zval **gc_data, int *gc_count)
- {
- pdo_dbh_t *dbh = Z_PDO_DBH_P(object);
- *gc_data = &dbh->def_stmt_ctor_args;
- *gc_count = 1;
- return zend_std_get_properties(object);
- }
- static zend_object_handlers pdo_dbh_object_handlers;
- static void pdo_dbh_free_storage(zend_object *std);
- void pdo_dbh_init(void)
- {
- zend_class_entry ce;
- INIT_CLASS_ENTRY(ce, "PDO", pdo_dbh_functions);
- pdo_dbh_ce = zend_register_internal_class(&ce);
- pdo_dbh_ce->create_object = pdo_dbh_new;
- memcpy(&pdo_dbh_object_handlers, &std_object_handlers, sizeof(zend_object_handlers));
- pdo_dbh_object_handlers.offset = XtOffsetOf(pdo_dbh_object_t, std);
- pdo_dbh_object_handlers.dtor_obj = zend_objects_destroy_object;
- pdo_dbh_object_handlers.free_obj = pdo_dbh_free_storage;
- pdo_dbh_object_handlers.clone_obj = NULL;
- pdo_dbh_object_handlers.get_method = dbh_method_get;
- pdo_dbh_object_handlers.compare_objects = dbh_compare;
- pdo_dbh_object_handlers.get_gc = dbh_get_gc;
- REGISTER_PDO_CLASS_CONST_LONG("PARAM_BOOL", (zend_long)PDO_PARAM_BOOL);
- REGISTER_PDO_CLASS_CONST_LONG("PARAM_NULL", (zend_long)PDO_PARAM_NULL);
- REGISTER_PDO_CLASS_CONST_LONG("PARAM_INT", (zend_long)PDO_PARAM_INT);
- REGISTER_PDO_CLASS_CONST_LONG("PARAM_STR", (zend_long)PDO_PARAM_STR);
- REGISTER_PDO_CLASS_CONST_LONG("PARAM_LOB", (zend_long)PDO_PARAM_LOB);
- REGISTER_PDO_CLASS_CONST_LONG("PARAM_STMT", (zend_long)PDO_PARAM_STMT);
- REGISTER_PDO_CLASS_CONST_LONG("PARAM_INPUT_OUTPUT", (zend_long)PDO_PARAM_INPUT_OUTPUT);
- REGISTER_PDO_CLASS_CONST_LONG("PARAM_STR_NATL", (zend_long)PDO_PARAM_STR_NATL);
- REGISTER_PDO_CLASS_CONST_LONG("PARAM_STR_CHAR", (zend_long)PDO_PARAM_STR_CHAR);
- REGISTER_PDO_CLASS_CONST_LONG("PARAM_EVT_ALLOC", (zend_long)PDO_PARAM_EVT_ALLOC);
- REGISTER_PDO_CLASS_CONST_LONG("PARAM_EVT_FREE", (zend_long)PDO_PARAM_EVT_FREE);
- REGISTER_PDO_CLASS_CONST_LONG("PARAM_EVT_EXEC_PRE", (zend_long)PDO_PARAM_EVT_EXEC_PRE);
- REGISTER_PDO_CLASS_CONST_LONG("PARAM_EVT_EXEC_POST", (zend_long)PDO_PARAM_EVT_EXEC_POST);
- REGISTER_PDO_CLASS_CONST_LONG("PARAM_EVT_FETCH_PRE", (zend_long)PDO_PARAM_EVT_FETCH_PRE);
- REGISTER_PDO_CLASS_CONST_LONG("PARAM_EVT_FETCH_POST", (zend_long)PDO_PARAM_EVT_FETCH_POST);
- REGISTER_PDO_CLASS_CONST_LONG("PARAM_EVT_NORMALIZE", (zend_long)PDO_PARAM_EVT_NORMALIZE);
- REGISTER_PDO_CLASS_CONST_LONG("FETCH_LAZY", (zend_long)PDO_FETCH_LAZY);
- REGISTER_PDO_CLASS_CONST_LONG("FETCH_ASSOC", (zend_long)PDO_FETCH_ASSOC);
- REGISTER_PDO_CLASS_CONST_LONG("FETCH_NUM", (zend_long)PDO_FETCH_NUM);
- REGISTER_PDO_CLASS_CONST_LONG("FETCH_BOTH", (zend_long)PDO_FETCH_BOTH);
- REGISTER_PDO_CLASS_CONST_LONG("FETCH_OBJ", (zend_long)PDO_FETCH_OBJ);
- REGISTER_PDO_CLASS_CONST_LONG("FETCH_BOUND", (zend_long)PDO_FETCH_BOUND);
- REGISTER_PDO_CLASS_CONST_LONG("FETCH_COLUMN", (zend_long)PDO_FETCH_COLUMN);
- REGISTER_PDO_CLASS_CONST_LONG("FETCH_CLASS", (zend_long)PDO_FETCH_CLASS);
- REGISTER_PDO_CLASS_CONST_LONG("FETCH_INTO", (zend_long)PDO_FETCH_INTO);
- REGISTER_PDO_CLASS_CONST_LONG("FETCH_FUNC", (zend_long)PDO_FETCH_FUNC);
- REGISTER_PDO_CLASS_CONST_LONG("FETCH_GROUP", (zend_long)PDO_FETCH_GROUP);
- REGISTER_PDO_CLASS_CONST_LONG("FETCH_UNIQUE", (zend_long)PDO_FETCH_UNIQUE);
- REGISTER_PDO_CLASS_CONST_LONG("FETCH_KEY_PAIR", (zend_long)PDO_FETCH_KEY_PAIR);
- REGISTER_PDO_CLASS_CONST_LONG("FETCH_CLASSTYPE", (zend_long)PDO_FETCH_CLASSTYPE);
- REGISTER_PDO_CLASS_CONST_LONG("FETCH_SERIALIZE",(zend_long)PDO_FETCH_SERIALIZE);
- REGISTER_PDO_CLASS_CONST_LONG("FETCH_PROPS_LATE", (zend_long)PDO_FETCH_PROPS_LATE);
- REGISTER_PDO_CLASS_CONST_LONG("FETCH_NAMED", (zend_long)PDO_FETCH_NAMED);
- REGISTER_PDO_CLASS_CONST_LONG("ATTR_AUTOCOMMIT", (zend_long)PDO_ATTR_AUTOCOMMIT);
- REGISTER_PDO_CLASS_CONST_LONG("ATTR_PREFETCH", (zend_long)PDO_ATTR_PREFETCH);
- REGISTER_PDO_CLASS_CONST_LONG("ATTR_TIMEOUT", (zend_long)PDO_ATTR_TIMEOUT);
- REGISTER_PDO_CLASS_CONST_LONG("ATTR_ERRMODE", (zend_long)PDO_ATTR_ERRMODE);
- REGISTER_PDO_CLASS_CONST_LONG("ATTR_SERVER_VERSION", (zend_long)PDO_ATTR_SERVER_VERSION);
- REGISTER_PDO_CLASS_CONST_LONG("ATTR_CLIENT_VERSION", (zend_long)PDO_ATTR_CLIENT_VERSION);
- REGISTER_PDO_CLASS_CONST_LONG("ATTR_SERVER_INFO", (zend_long)PDO_ATTR_SERVER_INFO);
- REGISTER_PDO_CLASS_CONST_LONG("ATTR_CONNECTION_STATUS", (zend_long)PDO_ATTR_CONNECTION_STATUS);
- REGISTER_PDO_CLASS_CONST_LONG("ATTR_CASE", (zend_long)PDO_ATTR_CASE);
- REGISTER_PDO_CLASS_CONST_LONG("ATTR_CURSOR_NAME", (zend_long)PDO_ATTR_CURSOR_NAME);
- REGISTER_PDO_CLASS_CONST_LONG("ATTR_CURSOR", (zend_long)PDO_ATTR_CURSOR);
- REGISTER_PDO_CLASS_CONST_LONG("ATTR_ORACLE_NULLS", (zend_long)PDO_ATTR_ORACLE_NULLS);
- REGISTER_PDO_CLASS_CONST_LONG("ATTR_PERSISTENT", (zend_long)PDO_ATTR_PERSISTENT);
- REGISTER_PDO_CLASS_CONST_LONG("ATTR_STATEMENT_CLASS", (zend_long)PDO_ATTR_STATEMENT_CLASS);
- REGISTER_PDO_CLASS_CONST_LONG("ATTR_FETCH_TABLE_NAMES", (zend_long)PDO_ATTR_FETCH_TABLE_NAMES);
- REGISTER_PDO_CLASS_CONST_LONG("ATTR_FETCH_CATALOG_NAMES", (zend_long)PDO_ATTR_FETCH_CATALOG_NAMES);
- REGISTER_PDO_CLASS_CONST_LONG("ATTR_DRIVER_NAME", (zend_long)PDO_ATTR_DRIVER_NAME);
- REGISTER_PDO_CLASS_CONST_LONG("ATTR_STRINGIFY_FETCHES", (zend_long)PDO_ATTR_STRINGIFY_FETCHES);
- REGISTER_PDO_CLASS_CONST_LONG("ATTR_MAX_COLUMN_LEN", (zend_long)PDO_ATTR_MAX_COLUMN_LEN);
- REGISTER_PDO_CLASS_CONST_LONG("ATTR_EMULATE_PREPARES", (zend_long)PDO_ATTR_EMULATE_PREPARES);
- REGISTER_PDO_CLASS_CONST_LONG("ATTR_DEFAULT_FETCH_MODE", (zend_long)PDO_ATTR_DEFAULT_FETCH_MODE);
- REGISTER_PDO_CLASS_CONST_LONG("ATTR_DEFAULT_STR_PARAM", (zend_long)PDO_ATTR_DEFAULT_STR_PARAM);
- REGISTER_PDO_CLASS_CONST_LONG("ERRMODE_SILENT", (zend_long)PDO_ERRMODE_SILENT);
- REGISTER_PDO_CLASS_CONST_LONG("ERRMODE_WARNING", (zend_long)PDO_ERRMODE_WARNING);
- REGISTER_PDO_CLASS_CONST_LONG("ERRMODE_EXCEPTION", (zend_long)PDO_ERRMODE_EXCEPTION);
- REGISTER_PDO_CLASS_CONST_LONG("CASE_NATURAL", (zend_long)PDO_CASE_NATURAL);
- REGISTER_PDO_CLASS_CONST_LONG("CASE_LOWER", (zend_long)PDO_CASE_LOWER);
- REGISTER_PDO_CLASS_CONST_LONG("CASE_UPPER", (zend_long)PDO_CASE_UPPER);
- REGISTER_PDO_CLASS_CONST_LONG("NULL_NATURAL", (zend_long)PDO_NULL_NATURAL);
- REGISTER_PDO_CLASS_CONST_LONG("NULL_EMPTY_STRING", (zend_long)PDO_NULL_EMPTY_STRING);
- REGISTER_PDO_CLASS_CONST_LONG("NULL_TO_STRING", (zend_long)PDO_NULL_TO_STRING);
- REGISTER_PDO_CLASS_CONST_STRING("ERR_NONE", PDO_ERR_NONE);
- REGISTER_PDO_CLASS_CONST_LONG("FETCH_ORI_NEXT", (zend_long)PDO_FETCH_ORI_NEXT);
- REGISTER_PDO_CLASS_CONST_LONG("FETCH_ORI_PRIOR", (zend_long)PDO_FETCH_ORI_PRIOR);
- REGISTER_PDO_CLASS_CONST_LONG("FETCH_ORI_FIRST", (zend_long)PDO_FETCH_ORI_FIRST);
- REGISTER_PDO_CLASS_CONST_LONG("FETCH_ORI_LAST", (zend_long)PDO_FETCH_ORI_LAST);
- REGISTER_PDO_CLASS_CONST_LONG("FETCH_ORI_ABS", (zend_long)PDO_FETCH_ORI_ABS);
- REGISTER_PDO_CLASS_CONST_LONG("FETCH_ORI_REL", (zend_long)PDO_FETCH_ORI_REL);
- REGISTER_PDO_CLASS_CONST_LONG("CURSOR_FWDONLY", (zend_long)PDO_CURSOR_FWDONLY);
- REGISTER_PDO_CLASS_CONST_LONG("CURSOR_SCROLL", (zend_long)PDO_CURSOR_SCROLL);
- }
- static void dbh_free(pdo_dbh_t *dbh, zend_bool free_persistent)
- {
- int i;
- if (dbh->query_stmt) {
- zval_ptr_dtor(&dbh->query_stmt_zval);
- dbh->query_stmt = NULL;
- }
- if (dbh->is_persistent) {
- #if ZEND_DEBUG
- ZEND_ASSERT(!free_persistent || (dbh->refcount == 1));
- #endif
- if (!free_persistent && (--dbh->refcount)) {
- return;
- }
- }
- if (dbh->methods) {
- dbh->methods->closer(dbh);
- }
- if (dbh->data_source) {
- pefree((char *)dbh->data_source, dbh->is_persistent);
- }
- if (dbh->username) {
- pefree(dbh->username, dbh->is_persistent);
- }
- if (dbh->password) {
- pefree(dbh->password, dbh->is_persistent);
- }
- if (dbh->persistent_id) {
- pefree((char *)dbh->persistent_id, dbh->is_persistent);
- }
- if (!Z_ISUNDEF(dbh->def_stmt_ctor_args)) {
- zval_ptr_dtor(&dbh->def_stmt_ctor_args);
- }
- for (i = 0; i < PDO_DBH_DRIVER_METHOD_KIND__MAX; i++) {
- if (dbh->cls_methods[i]) {
- zend_hash_destroy(dbh->cls_methods[i]);
- pefree(dbh->cls_methods[i], dbh->is_persistent);
- }
- }
- pefree(dbh, dbh->is_persistent);
- }
- static void pdo_dbh_free_storage(zend_object *std)
- {
- pdo_dbh_t *dbh = php_pdo_dbh_fetch_inner(std);
- if (dbh->in_txn && dbh->methods && dbh->methods->rollback) {
- dbh->methods->rollback(dbh);
- dbh->in_txn = 0;
- }
- if (dbh->is_persistent && dbh->methods && dbh->methods->persistent_shutdown) {
- dbh->methods->persistent_shutdown(dbh);
- }
- zend_object_std_dtor(std);
- dbh_free(dbh, 0);
- }
- zend_object *pdo_dbh_new(zend_class_entry *ce)
- {
- pdo_dbh_object_t *dbh;
- dbh = zend_object_alloc(sizeof(pdo_dbh_object_t), ce);
- zend_object_std_init(&dbh->std, ce);
- object_properties_init(&dbh->std, ce);
- rebuild_object_properties(&dbh->std);
- dbh->inner = ecalloc(1, sizeof(pdo_dbh_t));
- dbh->inner->def_stmt_ce = pdo_dbstmt_ce;
- dbh->std.handlers = &pdo_dbh_object_handlers;
- return &dbh->std;
- }
- /* }}} */
- ZEND_RSRC_DTOR_FUNC(php_pdo_pdbh_dtor) /* {{{ */
- {
- if (res->ptr) {
- pdo_dbh_t *dbh = (pdo_dbh_t*)res->ptr;
- dbh_free(dbh, 1);
- res->ptr = NULL;
- }
- }
- /* }}} */
- /*
- * Local variables:
- * tab-width: 4
- * c-basic-offset: 4
- * End:
- * vim600: noet sw=4 ts=4 fdm=marker
- * vim<600: noet sw=4 ts=4
- */
|