pdo_dbh.c 47 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407140814091410141114121413141414151416141714181419142014211422142314241425142614271428142914301431143214331434143514361437143814391440144114421443144414451446144714481449145014511452145314541455145614571458145914601461146214631464146514661467146814691470147114721473147414751476147714781479148014811482148314841485148614871488148914901491149214931494149514961497149814991500150115021503150415051506150715081509151015111512151315141515151615171518151915201521152215231524152515261527152815291530153115321533153415351536153715381539154015411542154315441545154615471548154915501551155215531554155515561557155815591560156115621563156415651566156715681569157015711572157315741575157615771578157915801581158215831584158515861587158815891590159115921593159415951596159715981599160016011602160316041605160616071608160916101611
  1. /*
  2. +----------------------------------------------------------------------+
  3. | PHP Version 7 |
  4. +----------------------------------------------------------------------+
  5. | Copyright (c) 1997-2018 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. | Author: Wez Furlong <wez@php.net> |
  16. | Marcus Boerger <helly@php.net> |
  17. | Sterling Hughes <sterling@php.net> |
  18. +----------------------------------------------------------------------+
  19. */
  20. /* The PDO Database Handle Class */
  21. #ifdef HAVE_CONFIG_H
  22. #include "config.h"
  23. #endif
  24. #include "php.h"
  25. #include "php_ini.h"
  26. #include "ext/standard/info.h"
  27. #include "php_pdo.h"
  28. #include "php_pdo_driver.h"
  29. #include "php_pdo_int.h"
  30. #include "zend_exceptions.h"
  31. #include "zend_object_handlers.h"
  32. #include "zend_hash.h"
  33. static int pdo_dbh_attribute_set(pdo_dbh_t *dbh, zend_long attr, zval *value);
  34. void pdo_throw_exception(unsigned int driver_errcode, char *driver_errmsg, pdo_error_type *pdo_error)
  35. {
  36. zval error_info,pdo_exception;
  37. char *pdo_exception_message;
  38. object_init_ex(&pdo_exception, php_pdo_get_exception());
  39. array_init(&error_info);
  40. add_next_index_string(&error_info, *pdo_error);
  41. add_next_index_long(&error_info, driver_errcode);
  42. add_next_index_string(&error_info, driver_errmsg);
  43. spprintf(&pdo_exception_message, 0,"SQLSTATE[%s] [%d] %s",*pdo_error, driver_errcode, driver_errmsg);
  44. zend_update_property(php_pdo_get_exception(), &pdo_exception, "errorInfo", sizeof("errorInfo")-1, &error_info);
  45. zend_update_property_long(php_pdo_get_exception(), &pdo_exception, "code", sizeof("code")-1, driver_errcode);
  46. zend_update_property_string(
  47. php_pdo_get_exception(),
  48. &pdo_exception,
  49. "message",
  50. sizeof("message")-1,
  51. pdo_exception_message
  52. );
  53. efree(pdo_exception_message);
  54. zval_ptr_dtor(&error_info);
  55. zend_throw_exception_object(&pdo_exception);
  56. }
  57. void pdo_raise_impl_error(pdo_dbh_t *dbh, pdo_stmt_t *stmt, const char *sqlstate, const char *supp) /* {{{ */
  58. {
  59. pdo_error_type *pdo_err = &dbh->error_code;
  60. char *message = NULL;
  61. const char *msg;
  62. if (dbh && dbh->error_mode == PDO_ERRMODE_SILENT) {
  63. #if 0
  64. /* BUG: if user is running in silent mode and hits an error at the driver level
  65. * when they use the PDO methods to call up the error information, they may
  66. * get bogus information */
  67. return;
  68. #endif
  69. }
  70. if (stmt) {
  71. pdo_err = &stmt->error_code;
  72. }
  73. strncpy(*pdo_err, sqlstate, 6);
  74. /* hash sqlstate to error messages */
  75. msg = pdo_sqlstate_state_to_description(*pdo_err);
  76. if (!msg) {
  77. msg = "<<Unknown error>>";
  78. }
  79. if (supp) {
  80. spprintf(&message, 0, "SQLSTATE[%s]: %s: %s", *pdo_err, msg, supp);
  81. } else {
  82. spprintf(&message, 0, "SQLSTATE[%s]: %s", *pdo_err, msg);
  83. }
  84. if (dbh && dbh->error_mode != PDO_ERRMODE_EXCEPTION) {
  85. php_error_docref(NULL, E_WARNING, "%s", message);
  86. } else {
  87. zval ex, info;
  88. zend_class_entry *def_ex = php_pdo_get_exception_base(1), *pdo_ex = php_pdo_get_exception();
  89. object_init_ex(&ex, pdo_ex);
  90. zend_update_property_string(def_ex, &ex, "message", sizeof("message")-1, message);
  91. zend_update_property_string(def_ex, &ex, "code", sizeof("code")-1, *pdo_err);
  92. array_init(&info);
  93. add_next_index_string(&info, *pdo_err);
  94. add_next_index_long(&info, 0);
  95. zend_update_property(pdo_ex, &ex, "errorInfo", sizeof("errorInfo")-1, &info);
  96. zval_ptr_dtor(&info);
  97. zend_throw_exception_object(&ex);
  98. }
  99. if (message) {
  100. efree(message);
  101. }
  102. }
  103. /* }}} */
  104. PDO_API void pdo_handle_error(pdo_dbh_t *dbh, pdo_stmt_t *stmt) /* {{{ */
  105. {
  106. pdo_error_type *pdo_err = &dbh->error_code;
  107. const char *msg = "<<Unknown>>";
  108. char *supp = NULL;
  109. zend_long native_code = 0;
  110. zend_string *message = NULL;
  111. zval info;
  112. if (dbh == NULL || dbh->error_mode == PDO_ERRMODE_SILENT) {
  113. return;
  114. }
  115. if (stmt) {
  116. pdo_err = &stmt->error_code;
  117. }
  118. /* hash sqlstate to error messages */
  119. msg = pdo_sqlstate_state_to_description(*pdo_err);
  120. if (!msg) {
  121. msg = "<<Unknown error>>";
  122. }
  123. ZVAL_UNDEF(&info);
  124. if (dbh->methods->fetch_err) {
  125. array_init(&info);
  126. add_next_index_string(&info, *pdo_err);
  127. if (dbh->methods->fetch_err(dbh, stmt, &info)) {
  128. zval *item;
  129. if ((item = zend_hash_index_find(Z_ARRVAL(info), 1)) != NULL) {
  130. native_code = Z_LVAL_P(item);
  131. }
  132. if ((item = zend_hash_index_find(Z_ARRVAL(info), 2)) != NULL) {
  133. supp = estrndup(Z_STRVAL_P(item), Z_STRLEN_P(item));
  134. }
  135. }
  136. }
  137. if (supp) {
  138. message = strpprintf(0, "SQLSTATE[%s]: %s: " ZEND_LONG_FMT " %s", *pdo_err, msg, native_code, supp);
  139. } else {
  140. message = strpprintf(0, "SQLSTATE[%s]: %s", *pdo_err, msg);
  141. }
  142. if (dbh->error_mode == PDO_ERRMODE_WARNING) {
  143. php_error_docref(NULL, E_WARNING, "%s", ZSTR_VAL(message));
  144. } else if (EG(exception) == NULL) {
  145. zval ex;
  146. zend_class_entry *def_ex = php_pdo_get_exception_base(1), *pdo_ex = php_pdo_get_exception();
  147. object_init_ex(&ex, pdo_ex);
  148. zend_update_property_str(def_ex, &ex, "message", sizeof("message") - 1, message);
  149. zend_update_property_string(def_ex, &ex, "code", sizeof("code") - 1, *pdo_err);
  150. if (!Z_ISUNDEF(info)) {
  151. zend_update_property(pdo_ex, &ex, "errorInfo", sizeof("errorInfo") - 1, &info);
  152. }
  153. zend_throw_exception_object(&ex);
  154. }
  155. if (!Z_ISUNDEF(info)) {
  156. zval_ptr_dtor(&info);
  157. }
  158. if (message) {
  159. zend_string_release_ex(message, 0);
  160. }
  161. if (supp) {
  162. efree(supp);
  163. }
  164. }
  165. /* }}} */
  166. static char *dsn_from_uri(char *uri, char *buf, size_t buflen) /* {{{ */
  167. {
  168. php_stream *stream;
  169. char *dsn = NULL;
  170. stream = php_stream_open_wrapper(uri, "rb", REPORT_ERRORS, NULL);
  171. if (stream) {
  172. dsn = php_stream_get_line(stream, buf, buflen, NULL);
  173. php_stream_close(stream);
  174. }
  175. return dsn;
  176. }
  177. /* }}} */
  178. /* {{{ proto PDO::__construct(string dsn[, string username[, string passwd [, array options]]])
  179. */
  180. static PHP_METHOD(PDO, dbh_constructor)
  181. {
  182. zval *object = getThis();
  183. pdo_dbh_t *dbh = NULL;
  184. zend_bool is_persistent = 0;
  185. char *data_source;
  186. size_t data_source_len;
  187. char *colon;
  188. char *username=NULL, *password=NULL;
  189. size_t usernamelen, passwordlen;
  190. pdo_driver_t *driver = NULL;
  191. zval *options = NULL;
  192. char alt_dsn[512];
  193. int call_factory = 1;
  194. zend_error_handling zeh;
  195. ZEND_PARSE_PARAMETERS_START_EX(ZEND_PARSE_PARAMS_THROW, 1, 4)
  196. Z_PARAM_STRING(data_source, data_source_len)
  197. Z_PARAM_OPTIONAL
  198. Z_PARAM_STRING_EX(username, usernamelen, 1, 0)
  199. Z_PARAM_STRING_EX(password, passwordlen, 1, 0)
  200. Z_PARAM_ARRAY_EX(options, 1, 0)
  201. ZEND_PARSE_PARAMETERS_END();
  202. /* parse the data source name */
  203. colon = strchr(data_source, ':');
  204. if (!colon) {
  205. /* let's see if this string has a matching dsn in the php.ini */
  206. char *ini_dsn = NULL;
  207. snprintf(alt_dsn, sizeof(alt_dsn), "pdo.dsn.%s", data_source);
  208. if (FAILURE == cfg_get_string(alt_dsn, &ini_dsn)) {
  209. zend_throw_exception_ex(php_pdo_get_exception(), 0, "invalid data source name");
  210. return;
  211. }
  212. data_source = ini_dsn;
  213. colon = strchr(data_source, ':');
  214. if (!colon) {
  215. zend_throw_exception_ex(php_pdo_get_exception(), 0, "invalid data source name (via INI: %s)", alt_dsn);
  216. return;
  217. }
  218. }
  219. if (!strncmp(data_source, "uri:", sizeof("uri:")-1)) {
  220. /* the specified URI holds connection details */
  221. data_source = dsn_from_uri(data_source + sizeof("uri:")-1, alt_dsn, sizeof(alt_dsn));
  222. if (!data_source) {
  223. zend_throw_exception_ex(php_pdo_get_exception(), 0, "invalid data source URI");
  224. return;
  225. }
  226. colon = strchr(data_source, ':');
  227. if (!colon) {
  228. zend_throw_exception_ex(php_pdo_get_exception(), 0, "invalid data source name (via URI)");
  229. return;
  230. }
  231. }
  232. driver = pdo_find_driver(data_source, colon - data_source);
  233. if (!driver) {
  234. /* NB: don't want to include the data_source in the error message as
  235. * it might contain a password */
  236. zend_throw_exception_ex(php_pdo_get_exception(), 0, "could not find driver");
  237. return;
  238. }
  239. dbh = Z_PDO_DBH_P(object);
  240. /* is this supposed to be a persistent connection ? */
  241. if (options) {
  242. int plen = 0;
  243. char *hashkey = NULL;
  244. zend_resource *le;
  245. pdo_dbh_t *pdbh = NULL;
  246. zval *v;
  247. if ((v = zend_hash_index_find_deref(Z_ARRVAL_P(options), PDO_ATTR_PERSISTENT)) != NULL) {
  248. if (Z_TYPE_P(v) == IS_STRING &&
  249. !is_numeric_string(Z_STRVAL_P(v), Z_STRLEN_P(v), NULL, NULL, 0) && Z_STRLEN_P(v) > 0) {
  250. /* user specified key */
  251. plen = spprintf(&hashkey, 0, "PDO:DBH:DSN=%s:%s:%s:%s", data_source,
  252. username ? username : "",
  253. password ? password : "",
  254. Z_STRVAL_P(v));
  255. is_persistent = 1;
  256. } else {
  257. is_persistent = zval_get_long(v) ? 1 : 0;
  258. plen = spprintf(&hashkey, 0, "PDO:DBH:DSN=%s:%s:%s", data_source,
  259. username ? username : "",
  260. password ? password : "");
  261. }
  262. }
  263. if (is_persistent) {
  264. /* let's see if we have one cached.... */
  265. if ((le = zend_hash_str_find_ptr(&EG(persistent_list), hashkey, plen)) != NULL) {
  266. if (le->type == php_pdo_list_entry()) {
  267. pdbh = (pdo_dbh_t*)le->ptr;
  268. /* is the connection still alive ? */
  269. if (pdbh->methods->check_liveness && FAILURE == (pdbh->methods->check_liveness)(pdbh)) {
  270. /* nope... need to kill it */
  271. pdbh->refcount--;
  272. zend_list_close(le);
  273. pdbh = NULL;
  274. }
  275. }
  276. }
  277. if (pdbh) {
  278. call_factory = 0;
  279. } else {
  280. /* need a brand new pdbh */
  281. pdbh = pecalloc(1, sizeof(*pdbh), 1);
  282. pdbh->refcount = 1;
  283. pdbh->is_persistent = 1;
  284. pdbh->persistent_id = pemalloc(plen + 1, 1);
  285. memcpy((char *)pdbh->persistent_id, hashkey, plen+1);
  286. pdbh->persistent_id_len = plen;
  287. pdbh->def_stmt_ce = dbh->def_stmt_ce;
  288. }
  289. }
  290. if (pdbh) {
  291. efree(dbh);
  292. /* switch over to the persistent one */
  293. Z_PDO_OBJECT_P(object)->inner = pdbh;
  294. pdbh->refcount++;
  295. dbh = pdbh;
  296. }
  297. if (hashkey) {
  298. efree(hashkey);
  299. }
  300. }
  301. if (call_factory) {
  302. dbh->data_source_len = strlen(colon + 1);
  303. dbh->data_source = (const char*)pestrdup(colon + 1, is_persistent);
  304. dbh->username = username ? pestrdup(username, is_persistent) : NULL;
  305. dbh->password = password ? pestrdup(password, is_persistent) : NULL;
  306. dbh->default_fetch_type = PDO_FETCH_BOTH;
  307. }
  308. dbh->auto_commit = pdo_attr_lval(options, PDO_ATTR_AUTOCOMMIT, 1);
  309. if (!dbh->data_source || (username && !dbh->username) || (password && !dbh->password)) {
  310. php_error_docref(NULL, E_ERROR, "out of memory");
  311. }
  312. zend_replace_error_handling(EH_THROW, pdo_exception_ce, &zeh);
  313. if (!call_factory) {
  314. /* we got a persistent guy from our cache */
  315. goto options;
  316. }
  317. if (driver->db_handle_factory(dbh, options)) {
  318. /* all set */
  319. if (is_persistent) {
  320. /* register in the persistent list etc. */
  321. /* we should also need to replace the object store entry,
  322. since it was created with emalloc */
  323. if ((zend_register_persistent_resource(
  324. (char*)dbh->persistent_id, dbh->persistent_id_len, dbh, php_pdo_list_entry())) == NULL) {
  325. php_error_docref(NULL, E_ERROR, "Failed to register persistent entry");
  326. }
  327. }
  328. dbh->driver = driver;
  329. options:
  330. if (options) {
  331. zval *attr_value;
  332. zend_ulong long_key;
  333. zend_string *str_key = NULL;
  334. ZEND_HASH_FOREACH_KEY_VAL(Z_ARRVAL_P(options), long_key, str_key, attr_value) {
  335. if (str_key) {
  336. continue;
  337. }
  338. ZVAL_DEREF(attr_value);
  339. pdo_dbh_attribute_set(dbh, long_key, attr_value);
  340. } ZEND_HASH_FOREACH_END();
  341. }
  342. zend_restore_error_handling(&zeh);
  343. return;
  344. }
  345. /* the connection failed; things will tidy up in free_storage */
  346. /* XXX raise exception */
  347. zend_restore_error_handling(&zeh);
  348. if (!EG(exception)) {
  349. zend_throw_exception(pdo_exception_ce, "Constructor failed", 0);
  350. }
  351. }
  352. /* }}} */
  353. static zval *pdo_stmt_instantiate(pdo_dbh_t *dbh, zval *object, zend_class_entry *dbstmt_ce, zval *ctor_args) /* {{{ */
  354. {
  355. if (!Z_ISUNDEF_P(ctor_args)) {
  356. if (Z_TYPE_P(ctor_args) != IS_ARRAY) {
  357. pdo_raise_impl_error(dbh, NULL, "HY000", "constructor arguments must be passed as an array");
  358. return NULL;
  359. }
  360. if (!dbstmt_ce->constructor) {
  361. pdo_raise_impl_error(dbh, NULL, "HY000", "user-supplied statement does not accept constructor arguments");
  362. return NULL;
  363. }
  364. }
  365. if (UNEXPECTED(object_init_ex(object, dbstmt_ce) != SUCCESS)) {
  366. return NULL;
  367. }
  368. return object;
  369. } /* }}} */
  370. static void pdo_stmt_construct(zend_execute_data *execute_data, pdo_stmt_t *stmt, zval *object, zend_class_entry *dbstmt_ce, zval *ctor_args) /* {{{ */
  371. {
  372. zval query_string;
  373. zval z_key;
  374. ZVAL_STRINGL(&query_string, stmt->query_string, stmt->query_stringlen);
  375. ZVAL_STRINGL(&z_key, "queryString", sizeof("queryString") - 1);
  376. zend_std_write_property(object, &z_key, &query_string, NULL);
  377. zval_ptr_dtor(&query_string);
  378. zval_ptr_dtor(&z_key);
  379. if (dbstmt_ce->constructor) {
  380. zend_fcall_info fci;
  381. zend_fcall_info_cache fcc;
  382. zval retval;
  383. fci.size = sizeof(zend_fcall_info);
  384. ZVAL_UNDEF(&fci.function_name);
  385. fci.object = Z_OBJ_P(object);
  386. fci.retval = &retval;
  387. fci.param_count = 0;
  388. fci.params = NULL;
  389. fci.no_separation = 1;
  390. zend_fcall_info_args(&fci, ctor_args);
  391. fcc.function_handler = dbstmt_ce->constructor;
  392. fcc.called_scope = Z_OBJCE_P(object);
  393. fcc.object = Z_OBJ_P(object);
  394. if (zend_call_function(&fci, &fcc) != FAILURE) {
  395. zval_ptr_dtor(&retval);
  396. }
  397. zend_fcall_info_args_clear(&fci, 1);
  398. }
  399. }
  400. /* }}} */
  401. /* {{{ proto object PDO::prepare(string statement [, array options])
  402. Prepares a statement for execution and returns a statement object */
  403. static PHP_METHOD(PDO, prepare)
  404. {
  405. pdo_stmt_t *stmt;
  406. char *statement;
  407. size_t statement_len;
  408. zval *options = NULL, *opt, *item, ctor_args;
  409. zend_class_entry *dbstmt_ce, *pce;
  410. pdo_dbh_object_t *dbh_obj = Z_PDO_OBJECT_P(getThis());
  411. pdo_dbh_t *dbh = dbh_obj->inner;
  412. ZEND_PARSE_PARAMETERS_START(1, 2)
  413. Z_PARAM_STRING(statement, statement_len)
  414. Z_PARAM_OPTIONAL
  415. Z_PARAM_ARRAY(options)
  416. ZEND_PARSE_PARAMETERS_END_EX(RETURN_FALSE);
  417. PDO_DBH_CLEAR_ERR();
  418. PDO_CONSTRUCT_CHECK;
  419. if (ZEND_NUM_ARGS() > 1 && (opt = zend_hash_index_find(Z_ARRVAL_P(options), PDO_ATTR_STATEMENT_CLASS)) != NULL) {
  420. if (Z_TYPE_P(opt) != IS_ARRAY || (item = zend_hash_index_find(Z_ARRVAL_P(opt), 0)) == NULL
  421. || Z_TYPE_P(item) != IS_STRING
  422. || (pce = zend_lookup_class(Z_STR_P(item))) == NULL
  423. ) {
  424. pdo_raise_impl_error(dbh, NULL, "HY000",
  425. "PDO::ATTR_STATEMENT_CLASS requires format array(classname, array(ctor_args)); "
  426. "the classname must be a string specifying an existing class"
  427. );
  428. PDO_HANDLE_DBH_ERR();
  429. RETURN_FALSE;
  430. }
  431. dbstmt_ce = pce;
  432. if (!instanceof_function(dbstmt_ce, pdo_dbstmt_ce)) {
  433. pdo_raise_impl_error(dbh, NULL, "HY000",
  434. "user-supplied statement class must be derived from PDOStatement");
  435. PDO_HANDLE_DBH_ERR();
  436. RETURN_FALSE;
  437. }
  438. if (dbstmt_ce->constructor && !(dbstmt_ce->constructor->common.fn_flags & (ZEND_ACC_PRIVATE|ZEND_ACC_PROTECTED))) {
  439. pdo_raise_impl_error(dbh, NULL, "HY000",
  440. "user-supplied statement class cannot have a public constructor");
  441. PDO_HANDLE_DBH_ERR();
  442. RETURN_FALSE;
  443. }
  444. if ((item = zend_hash_index_find(Z_ARRVAL_P(opt), 1)) != NULL) {
  445. if (Z_TYPE_P(item) != IS_ARRAY) {
  446. pdo_raise_impl_error(dbh, NULL, "HY000",
  447. "PDO::ATTR_STATEMENT_CLASS requires format array(classname, ctor_args); "
  448. "ctor_args must be an array"
  449. );
  450. PDO_HANDLE_DBH_ERR();
  451. RETURN_FALSE;
  452. }
  453. ZVAL_COPY_VALUE(&ctor_args, item);
  454. } else {
  455. ZVAL_UNDEF(&ctor_args);
  456. }
  457. } else {
  458. dbstmt_ce = dbh->def_stmt_ce;
  459. ZVAL_COPY_VALUE(&ctor_args, &dbh->def_stmt_ctor_args);
  460. }
  461. if (!pdo_stmt_instantiate(dbh, return_value, dbstmt_ce, &ctor_args)) {
  462. if (EXPECTED(!EG(exception))) {
  463. pdo_raise_impl_error(dbh, NULL, "HY000",
  464. "failed to instantiate user-supplied statement class"
  465. );
  466. }
  467. PDO_HANDLE_DBH_ERR();
  468. RETURN_FALSE;
  469. }
  470. stmt = Z_PDO_STMT_P(return_value);
  471. /* unconditionally keep this for later reference */
  472. stmt->query_string = estrndup(statement, statement_len);
  473. stmt->query_stringlen = statement_len;
  474. stmt->default_fetch_type = dbh->default_fetch_type;
  475. stmt->dbh = dbh;
  476. /* give it a reference to me */
  477. ZVAL_OBJ(&stmt->database_object_handle, &dbh_obj->std);
  478. Z_ADDREF(stmt->database_object_handle);
  479. /* we haven't created a lazy object yet */
  480. ZVAL_UNDEF(&stmt->lazy_object_ref);
  481. if (dbh->methods->preparer(dbh, statement, statement_len, stmt, options)) {
  482. pdo_stmt_construct(execute_data, stmt, return_value, dbstmt_ce, &ctor_args);
  483. return;
  484. }
  485. PDO_HANDLE_DBH_ERR();
  486. /* kill the object handle for the stmt here */
  487. zval_ptr_dtor(return_value);
  488. RETURN_FALSE;
  489. }
  490. /* }}} */
  491. /* {{{ proto bool PDO::beginTransaction()
  492. Initiates a transaction */
  493. static PHP_METHOD(PDO, beginTransaction)
  494. {
  495. pdo_dbh_t *dbh = Z_PDO_DBH_P(getThis());
  496. if (zend_parse_parameters_none() == FAILURE) {
  497. return;
  498. }
  499. PDO_CONSTRUCT_CHECK;
  500. if (dbh->in_txn) {
  501. zend_throw_exception_ex(php_pdo_get_exception(), 0, "There is already an active transaction");
  502. RETURN_FALSE;
  503. }
  504. if (!dbh->methods->begin) {
  505. /* TODO: this should be an exception; see the auto-commit mode
  506. * comments below */
  507. zend_throw_exception_ex(php_pdo_get_exception(), 0, "This driver doesn't support transactions");
  508. RETURN_FALSE;
  509. }
  510. if (dbh->methods->begin(dbh)) {
  511. dbh->in_txn = 1;
  512. RETURN_TRUE;
  513. }
  514. PDO_HANDLE_DBH_ERR();
  515. RETURN_FALSE;
  516. }
  517. /* }}} */
  518. /* {{{ proto bool PDO::commit()
  519. Commit a transaction */
  520. static PHP_METHOD(PDO, commit)
  521. {
  522. pdo_dbh_t *dbh = Z_PDO_DBH_P(getThis());
  523. if (zend_parse_parameters_none() == FAILURE) {
  524. return;
  525. }
  526. PDO_CONSTRUCT_CHECK;
  527. if (!dbh->in_txn) {
  528. zend_throw_exception_ex(php_pdo_get_exception(), 0, "There is no active transaction");
  529. RETURN_FALSE;
  530. }
  531. if (dbh->methods->commit(dbh)) {
  532. dbh->in_txn = 0;
  533. RETURN_TRUE;
  534. }
  535. PDO_HANDLE_DBH_ERR();
  536. RETURN_FALSE;
  537. }
  538. /* }}} */
  539. /* {{{ proto bool PDO::rollBack()
  540. roll back a transaction */
  541. static PHP_METHOD(PDO, rollBack)
  542. {
  543. pdo_dbh_t *dbh = Z_PDO_DBH_P(getThis());
  544. if (zend_parse_parameters_none() == FAILURE) {
  545. return;
  546. }
  547. PDO_CONSTRUCT_CHECK;
  548. if (!dbh->in_txn) {
  549. zend_throw_exception_ex(php_pdo_get_exception(), 0, "There is no active transaction");
  550. RETURN_FALSE;
  551. }
  552. if (dbh->methods->rollback(dbh)) {
  553. dbh->in_txn = 0;
  554. RETURN_TRUE;
  555. }
  556. PDO_HANDLE_DBH_ERR();
  557. RETURN_FALSE;
  558. }
  559. /* }}} */
  560. /* {{{ proto bool PDO::inTransaction()
  561. determine if inside a transaction */
  562. static PHP_METHOD(PDO, inTransaction)
  563. {
  564. pdo_dbh_t *dbh = Z_PDO_DBH_P(getThis());
  565. if (zend_parse_parameters_none() == FAILURE) {
  566. return;
  567. }
  568. PDO_CONSTRUCT_CHECK;
  569. if (!dbh->methods->in_transaction) {
  570. RETURN_BOOL(dbh->in_txn);
  571. }
  572. RETURN_BOOL(dbh->methods->in_transaction(dbh));
  573. }
  574. /* }}} */
  575. static int pdo_dbh_attribute_set(pdo_dbh_t *dbh, zend_long attr, zval *value) /* {{{ */
  576. {
  577. zend_long lval;
  578. #define PDO_LONG_PARAM_CHECK \
  579. 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) { \
  580. pdo_raise_impl_error(dbh, NULL, "HY000", "attribute value must be an integer"); \
  581. PDO_HANDLE_DBH_ERR(); \
  582. return FAILURE; \
  583. } \
  584. switch (attr) {
  585. case PDO_ATTR_ERRMODE:
  586. PDO_LONG_PARAM_CHECK;
  587. lval = zval_get_long(value);
  588. switch (lval) {
  589. case PDO_ERRMODE_SILENT:
  590. case PDO_ERRMODE_WARNING:
  591. case PDO_ERRMODE_EXCEPTION:
  592. dbh->error_mode = lval;
  593. return SUCCESS;
  594. default:
  595. pdo_raise_impl_error(dbh, NULL, "HY000", "invalid error mode");
  596. PDO_HANDLE_DBH_ERR();
  597. return FAILURE;
  598. }
  599. return FAILURE;
  600. case PDO_ATTR_CASE:
  601. PDO_LONG_PARAM_CHECK;
  602. lval = zval_get_long(value);
  603. switch (lval) {
  604. case PDO_CASE_NATURAL:
  605. case PDO_CASE_UPPER:
  606. case PDO_CASE_LOWER:
  607. dbh->desired_case = lval;
  608. return SUCCESS;
  609. default:
  610. pdo_raise_impl_error(dbh, NULL, "HY000", "invalid case folding mode");
  611. PDO_HANDLE_DBH_ERR();
  612. return FAILURE;
  613. }
  614. return FAILURE;
  615. case PDO_ATTR_ORACLE_NULLS:
  616. PDO_LONG_PARAM_CHECK;
  617. dbh->oracle_nulls = zval_get_long(value);
  618. return SUCCESS;
  619. case PDO_ATTR_DEFAULT_FETCH_MODE:
  620. if (Z_TYPE_P(value) == IS_ARRAY) {
  621. zval *tmp;
  622. if ((tmp = zend_hash_index_find(Z_ARRVAL_P(value), 0)) != NULL && Z_TYPE_P(tmp) == IS_LONG) {
  623. if (Z_LVAL_P(tmp) == PDO_FETCH_INTO || Z_LVAL_P(tmp) == PDO_FETCH_CLASS) {
  624. pdo_raise_impl_error(dbh, NULL, "HY000", "FETCH_INTO and FETCH_CLASS are not yet supported as default fetch modes");
  625. return FAILURE;
  626. }
  627. }
  628. } else {
  629. PDO_LONG_PARAM_CHECK;
  630. }
  631. lval = zval_get_long(value);
  632. if (lval == PDO_FETCH_USE_DEFAULT) {
  633. pdo_raise_impl_error(dbh, NULL, "HY000", "invalid fetch mode type");
  634. return FAILURE;
  635. }
  636. dbh->default_fetch_type = lval;
  637. return SUCCESS;
  638. case PDO_ATTR_STRINGIFY_FETCHES:
  639. PDO_LONG_PARAM_CHECK;
  640. dbh->stringify = zval_get_long(value) ? 1 : 0;
  641. return SUCCESS;
  642. case PDO_ATTR_STATEMENT_CLASS: {
  643. /* array(string classname, array(mixed ctor_args)) */
  644. zend_class_entry *pce;
  645. zval *item;
  646. if (dbh->is_persistent) {
  647. pdo_raise_impl_error(dbh, NULL, "HY000",
  648. "PDO::ATTR_STATEMENT_CLASS cannot be used with persistent PDO instances"
  649. );
  650. PDO_HANDLE_DBH_ERR();
  651. return FAILURE;
  652. }
  653. if (Z_TYPE_P(value) != IS_ARRAY
  654. || (item = zend_hash_index_find(Z_ARRVAL_P(value), 0)) == NULL
  655. || Z_TYPE_P(item) != IS_STRING
  656. || (pce = zend_lookup_class(Z_STR_P(item))) == NULL
  657. ) {
  658. pdo_raise_impl_error(dbh, NULL, "HY000",
  659. "PDO::ATTR_STATEMENT_CLASS requires format array(classname, array(ctor_args)); "
  660. "the classname must be a string specifying an existing class"
  661. );
  662. PDO_HANDLE_DBH_ERR();
  663. return FAILURE;
  664. }
  665. if (!instanceof_function(pce, pdo_dbstmt_ce)) {
  666. pdo_raise_impl_error(dbh, NULL, "HY000",
  667. "user-supplied statement class must be derived from PDOStatement");
  668. PDO_HANDLE_DBH_ERR();
  669. return FAILURE;
  670. }
  671. if (pce->constructor && !(pce->constructor->common.fn_flags & (ZEND_ACC_PRIVATE|ZEND_ACC_PROTECTED))) {
  672. pdo_raise_impl_error(dbh, NULL, "HY000",
  673. "user-supplied statement class cannot have a public constructor");
  674. PDO_HANDLE_DBH_ERR();
  675. return FAILURE;
  676. }
  677. dbh->def_stmt_ce = pce;
  678. if (!Z_ISUNDEF(dbh->def_stmt_ctor_args)) {
  679. zval_ptr_dtor(&dbh->def_stmt_ctor_args);
  680. ZVAL_UNDEF(&dbh->def_stmt_ctor_args);
  681. }
  682. if ((item = zend_hash_index_find(Z_ARRVAL_P(value), 1)) != NULL) {
  683. if (Z_TYPE_P(item) != IS_ARRAY) {
  684. pdo_raise_impl_error(dbh, NULL, "HY000",
  685. "PDO::ATTR_STATEMENT_CLASS requires format array(classname, array(ctor_args)); "
  686. "ctor_args must be an array"
  687. );
  688. PDO_HANDLE_DBH_ERR();
  689. return FAILURE;
  690. }
  691. ZVAL_COPY(&dbh->def_stmt_ctor_args, item);
  692. }
  693. return SUCCESS;
  694. }
  695. default:
  696. ;
  697. }
  698. if (!dbh->methods->set_attribute) {
  699. goto fail;
  700. }
  701. PDO_DBH_CLEAR_ERR();
  702. if (dbh->methods->set_attribute(dbh, attr, value)) {
  703. return SUCCESS;
  704. }
  705. fail:
  706. if (attr == PDO_ATTR_AUTOCOMMIT) {
  707. zend_throw_exception_ex(php_pdo_get_exception(), 0, "The auto-commit mode cannot be changed for this driver");
  708. } else if (!dbh->methods->set_attribute) {
  709. pdo_raise_impl_error(dbh, NULL, "IM001", "driver does not support setting attributes");
  710. } else {
  711. PDO_HANDLE_DBH_ERR();
  712. }
  713. return FAILURE;
  714. }
  715. /* }}} */
  716. /* {{{ proto bool PDO::setAttribute(int attribute, mixed value)
  717. Set an attribute */
  718. static PHP_METHOD(PDO, setAttribute)
  719. {
  720. pdo_dbh_t *dbh = Z_PDO_DBH_P(getThis());
  721. zend_long attr;
  722. zval *value;
  723. ZEND_PARSE_PARAMETERS_START(2, 2)
  724. Z_PARAM_LONG(attr)
  725. Z_PARAM_ZVAL(value)
  726. ZEND_PARSE_PARAMETERS_END_EX(RETURN_FALSE);
  727. PDO_DBH_CLEAR_ERR();
  728. PDO_CONSTRUCT_CHECK;
  729. if (pdo_dbh_attribute_set(dbh, attr, value) != FAILURE) {
  730. RETURN_TRUE;
  731. }
  732. RETURN_FALSE;
  733. }
  734. /* }}} */
  735. /* {{{ proto mixed PDO::getAttribute(int attribute)
  736. Get an attribute */
  737. static PHP_METHOD(PDO, getAttribute)
  738. {
  739. pdo_dbh_t *dbh = Z_PDO_DBH_P(getThis());
  740. zend_long attr;
  741. ZEND_PARSE_PARAMETERS_START(1, 1)
  742. Z_PARAM_LONG(attr)
  743. ZEND_PARSE_PARAMETERS_END_EX(RETURN_FALSE);
  744. PDO_DBH_CLEAR_ERR();
  745. PDO_CONSTRUCT_CHECK;
  746. /* handle generic PDO-level attributes */
  747. switch (attr) {
  748. case PDO_ATTR_PERSISTENT:
  749. RETURN_BOOL(dbh->is_persistent);
  750. case PDO_ATTR_CASE:
  751. RETURN_LONG(dbh->desired_case);
  752. case PDO_ATTR_ORACLE_NULLS:
  753. RETURN_LONG(dbh->oracle_nulls);
  754. case PDO_ATTR_ERRMODE:
  755. RETURN_LONG(dbh->error_mode);
  756. case PDO_ATTR_DRIVER_NAME:
  757. RETURN_STRINGL((char*)dbh->driver->driver_name, dbh->driver->driver_name_len);
  758. case PDO_ATTR_STATEMENT_CLASS:
  759. array_init(return_value);
  760. add_next_index_str(return_value, zend_string_copy(dbh->def_stmt_ce->name));
  761. if (!Z_ISUNDEF(dbh->def_stmt_ctor_args)) {
  762. Z_TRY_ADDREF(dbh->def_stmt_ctor_args);
  763. add_next_index_zval(return_value, &dbh->def_stmt_ctor_args);
  764. }
  765. return;
  766. case PDO_ATTR_DEFAULT_FETCH_MODE:
  767. RETURN_LONG(dbh->default_fetch_type);
  768. default:
  769. break;
  770. }
  771. if (!dbh->methods->get_attribute) {
  772. pdo_raise_impl_error(dbh, NULL, "IM001", "driver does not support getting attributes");
  773. RETURN_FALSE;
  774. }
  775. switch (dbh->methods->get_attribute(dbh, attr, return_value)) {
  776. case -1:
  777. PDO_HANDLE_DBH_ERR();
  778. RETURN_FALSE;
  779. case 0:
  780. pdo_raise_impl_error(dbh, NULL, "IM001", "driver does not support that attribute");
  781. RETURN_FALSE;
  782. default:
  783. return;
  784. }
  785. }
  786. /* }}} */
  787. /* {{{ proto int PDO::exec(string query)
  788. Execute a query that does not return a row set, returning the number of affected rows */
  789. static PHP_METHOD(PDO, exec)
  790. {
  791. pdo_dbh_t *dbh = Z_PDO_DBH_P(getThis());
  792. char *statement;
  793. size_t statement_len;
  794. zend_long ret;
  795. ZEND_PARSE_PARAMETERS_START(1, 1)
  796. Z_PARAM_STRING(statement, statement_len)
  797. ZEND_PARSE_PARAMETERS_END_EX(RETURN_FALSE);
  798. if (!statement_len) {
  799. pdo_raise_impl_error(dbh, NULL, "HY000", "trying to execute an empty query");
  800. RETURN_FALSE;
  801. }
  802. PDO_DBH_CLEAR_ERR();
  803. PDO_CONSTRUCT_CHECK;
  804. ret = dbh->methods->doer(dbh, statement, statement_len);
  805. if(ret == -1) {
  806. PDO_HANDLE_DBH_ERR();
  807. RETURN_FALSE;
  808. } else {
  809. RETURN_LONG(ret);
  810. }
  811. }
  812. /* }}} */
  813. /* {{{ proto string PDO::lastInsertId([string seqname])
  814. 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. */
  815. static PHP_METHOD(PDO, lastInsertId)
  816. {
  817. pdo_dbh_t *dbh = Z_PDO_DBH_P(getThis());
  818. char *name = NULL;
  819. size_t namelen;
  820. ZEND_PARSE_PARAMETERS_START(0, 1)
  821. Z_PARAM_OPTIONAL
  822. Z_PARAM_STRING_EX(name, namelen, 1, 0)
  823. ZEND_PARSE_PARAMETERS_END_EX(RETURN_FALSE);
  824. PDO_DBH_CLEAR_ERR();
  825. PDO_CONSTRUCT_CHECK;
  826. if (!dbh->methods->last_id) {
  827. pdo_raise_impl_error(dbh, NULL, "IM001", "driver does not support lastInsertId()");
  828. RETURN_FALSE;
  829. } else {
  830. size_t id_len;
  831. char *id;
  832. id = dbh->methods->last_id(dbh, name, &id_len);
  833. if (!id) {
  834. PDO_HANDLE_DBH_ERR();
  835. RETURN_FALSE;
  836. } else {
  837. //??? use zend_string ?
  838. RETVAL_STRINGL(id, id_len);
  839. efree(id);
  840. }
  841. }
  842. }
  843. /* }}} */
  844. /* {{{ proto string PDO::errorCode()
  845. Fetch the error code associated with the last operation on the database handle */
  846. static PHP_METHOD(PDO, errorCode)
  847. {
  848. pdo_dbh_t *dbh = Z_PDO_DBH_P(getThis());
  849. if (zend_parse_parameters_none() == FAILURE) {
  850. return;
  851. }
  852. PDO_CONSTRUCT_CHECK;
  853. if (dbh->query_stmt) {
  854. RETURN_STRING(dbh->query_stmt->error_code);
  855. }
  856. if (dbh->error_code[0] == '\0') {
  857. RETURN_NULL();
  858. }
  859. /**
  860. * Making sure that we fallback to the default implementation
  861. * if the dbh->error_code is not null.
  862. */
  863. RETURN_STRING(dbh->error_code);
  864. }
  865. /* }}} */
  866. /* {{{ proto int PDO::errorInfo()
  867. Fetch extended error information associated with the last operation on the database handle */
  868. static PHP_METHOD(PDO, errorInfo)
  869. {
  870. int error_count;
  871. int error_count_diff = 0;
  872. int error_expected_count = 3;
  873. pdo_dbh_t *dbh = Z_PDO_DBH_P(getThis());
  874. if (zend_parse_parameters_none() == FAILURE) {
  875. return;
  876. }
  877. PDO_CONSTRUCT_CHECK;
  878. array_init(return_value);
  879. if (dbh->query_stmt) {
  880. add_next_index_string(return_value, dbh->query_stmt->error_code);
  881. if(!strncmp(dbh->query_stmt->error_code, PDO_ERR_NONE, sizeof(PDO_ERR_NONE))) goto fill_array;
  882. } else {
  883. add_next_index_string(return_value, dbh->error_code);
  884. if(!strncmp(dbh->error_code, PDO_ERR_NONE, sizeof(PDO_ERR_NONE))) goto fill_array;
  885. }
  886. if (dbh->methods->fetch_err) {
  887. dbh->methods->fetch_err(dbh, dbh->query_stmt, return_value);
  888. }
  889. fill_array:
  890. /**
  891. * In order to be consistent, we have to make sure we add the good amount
  892. * of nulls depending on the current number of elements. We make a simple
  893. * difference and add the needed elements
  894. */
  895. error_count = zend_hash_num_elements(Z_ARRVAL_P(return_value));
  896. if (error_expected_count > error_count) {
  897. int current_index;
  898. error_count_diff = error_expected_count - error_count;
  899. for (current_index = 0; current_index < error_count_diff; current_index++) {
  900. add_next_index_null(return_value);
  901. }
  902. }
  903. }
  904. /* }}} */
  905. /* {{{ proto object PDO::query(string sql [, PDOStatement::setFetchMode() args])
  906. Prepare and execute $sql; returns the statement object for iteration */
  907. static PHP_METHOD(PDO, query)
  908. {
  909. pdo_stmt_t *stmt;
  910. char *statement;
  911. size_t statement_len;
  912. pdo_dbh_object_t *dbh_obj = Z_PDO_OBJECT_P(getThis());
  913. pdo_dbh_t *dbh = dbh_obj->inner;
  914. /* Return a meaningful error when no parameters were passed */
  915. if (!ZEND_NUM_ARGS()) {
  916. zend_parse_parameters(0, "z|z", NULL, NULL);
  917. RETURN_FALSE;
  918. }
  919. if (FAILURE == zend_parse_parameters(1, "s", &statement,
  920. &statement_len)) {
  921. RETURN_FALSE;
  922. }
  923. PDO_DBH_CLEAR_ERR();
  924. PDO_CONSTRUCT_CHECK;
  925. if (!pdo_stmt_instantiate(dbh, return_value, dbh->def_stmt_ce, &dbh->def_stmt_ctor_args)) {
  926. if (EXPECTED(!EG(exception))) {
  927. pdo_raise_impl_error(dbh, NULL, "HY000", "failed to instantiate user supplied statement class");
  928. }
  929. return;
  930. }
  931. stmt = Z_PDO_STMT_P(return_value);
  932. /* unconditionally keep this for later reference */
  933. stmt->query_string = estrndup(statement, statement_len);
  934. stmt->query_stringlen = statement_len;
  935. stmt->default_fetch_type = dbh->default_fetch_type;
  936. stmt->active_query_string = stmt->query_string;
  937. stmt->active_query_stringlen = statement_len;
  938. stmt->dbh = dbh;
  939. /* give it a reference to me */
  940. ZVAL_OBJ(&stmt->database_object_handle, &dbh_obj->std);
  941. Z_ADDREF(stmt->database_object_handle);
  942. /* we haven't created a lazy object yet */
  943. ZVAL_UNDEF(&stmt->lazy_object_ref);
  944. if (dbh->methods->preparer(dbh, statement, statement_len, stmt, NULL)) {
  945. PDO_STMT_CLEAR_ERR();
  946. if (ZEND_NUM_ARGS() == 1 || SUCCESS == pdo_stmt_setup_fetch_mode(INTERNAL_FUNCTION_PARAM_PASSTHRU, stmt, 1)) {
  947. /* now execute the statement */
  948. PDO_STMT_CLEAR_ERR();
  949. if (stmt->methods->executer(stmt)) {
  950. int ret = 1;
  951. if (!stmt->executed) {
  952. if (stmt->dbh->alloc_own_columns) {
  953. ret = pdo_stmt_describe_columns(stmt);
  954. }
  955. stmt->executed = 1;
  956. }
  957. if (ret) {
  958. pdo_stmt_construct(execute_data, stmt, return_value, dbh->def_stmt_ce, &dbh->def_stmt_ctor_args);
  959. return;
  960. }
  961. }
  962. }
  963. /* something broke */
  964. dbh->query_stmt = stmt;
  965. ZVAL_COPY_VALUE(&dbh->query_stmt_zval, return_value);
  966. Z_DELREF(stmt->database_object_handle);
  967. ZVAL_UNDEF(&stmt->database_object_handle);
  968. PDO_HANDLE_STMT_ERR();
  969. } else {
  970. PDO_HANDLE_DBH_ERR();
  971. zval_ptr_dtor(return_value);
  972. }
  973. RETURN_FALSE;
  974. }
  975. /* }}} */
  976. /* {{{ proto string PDO::quote(string string [, int paramtype])
  977. 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 */
  978. static PHP_METHOD(PDO, quote)
  979. {
  980. pdo_dbh_t *dbh = Z_PDO_DBH_P(getThis());
  981. char *str;
  982. size_t str_len;
  983. zend_long paramtype = PDO_PARAM_STR;
  984. char *qstr;
  985. size_t qlen;
  986. ZEND_PARSE_PARAMETERS_START(1, 2)
  987. Z_PARAM_STRING(str, str_len)
  988. Z_PARAM_OPTIONAL
  989. Z_PARAM_LONG(paramtype)
  990. ZEND_PARSE_PARAMETERS_END_EX(RETURN_FALSE);
  991. PDO_DBH_CLEAR_ERR();
  992. PDO_CONSTRUCT_CHECK;
  993. if (!dbh->methods->quoter) {
  994. pdo_raise_impl_error(dbh, NULL, "IM001", "driver does not support quoting");
  995. RETURN_FALSE;
  996. }
  997. if (dbh->methods->quoter(dbh, str, str_len, &qstr, &qlen, paramtype)) {
  998. RETVAL_STRINGL(qstr, qlen);
  999. efree(qstr);
  1000. return;
  1001. }
  1002. PDO_HANDLE_DBH_ERR();
  1003. RETURN_FALSE;
  1004. }
  1005. /* }}} */
  1006. /* {{{ proto PDO::__wakeup()
  1007. Prevents use of a PDO instance that has been unserialized */
  1008. static PHP_METHOD(PDO, __wakeup)
  1009. {
  1010. zend_throw_exception_ex(php_pdo_get_exception(), 0, "You cannot serialize or unserialize PDO instances");
  1011. }
  1012. /* }}} */
  1013. /* {{{ proto int PDO::__sleep()
  1014. Prevents serialization of a PDO instance */
  1015. static PHP_METHOD(PDO, __sleep)
  1016. {
  1017. zend_throw_exception_ex(php_pdo_get_exception(), 0, "You cannot serialize or unserialize PDO instances");
  1018. }
  1019. /* }}} */
  1020. /* {{{ proto array PDO::getAvailableDrivers()
  1021. Return array of available PDO drivers */
  1022. static PHP_METHOD(PDO, getAvailableDrivers)
  1023. {
  1024. pdo_driver_t *pdriver;
  1025. if (zend_parse_parameters_none() == FAILURE) {
  1026. return;
  1027. }
  1028. array_init(return_value);
  1029. ZEND_HASH_FOREACH_PTR(&pdo_driver_hash, pdriver) {
  1030. add_next_index_stringl(return_value, (char*)pdriver->driver_name, pdriver->driver_name_len);
  1031. } ZEND_HASH_FOREACH_END();
  1032. }
  1033. /* }}} */
  1034. /* {{{ arginfo */
  1035. ZEND_BEGIN_ARG_INFO_EX(arginfo_pdo___construct, 0, 0, 1)
  1036. ZEND_ARG_INFO(0, dsn)
  1037. ZEND_ARG_INFO(0, username)
  1038. ZEND_ARG_INFO(0, passwd)
  1039. ZEND_ARG_INFO(0, options) /* array */
  1040. ZEND_END_ARG_INFO()
  1041. ZEND_BEGIN_ARG_INFO_EX(arginfo_pdo_prepare, 0, 0, 1)
  1042. ZEND_ARG_INFO(0, statement)
  1043. ZEND_ARG_INFO(0, options) /* array */
  1044. ZEND_END_ARG_INFO()
  1045. ZEND_BEGIN_ARG_INFO(arginfo_pdo_setattribute, 0)
  1046. ZEND_ARG_INFO(0, attribute)
  1047. ZEND_ARG_INFO(0, value)
  1048. ZEND_END_ARG_INFO()
  1049. ZEND_BEGIN_ARG_INFO(arginfo_pdo_getattribute, 0)
  1050. ZEND_ARG_INFO(0, attribute)
  1051. ZEND_END_ARG_INFO()
  1052. ZEND_BEGIN_ARG_INFO(arginfo_pdo_exec, 0)
  1053. ZEND_ARG_INFO(0, query)
  1054. ZEND_END_ARG_INFO()
  1055. ZEND_BEGIN_ARG_INFO_EX(arginfo_pdo_lastinsertid, 0, 0, 0)
  1056. ZEND_ARG_INFO(0, seqname)
  1057. ZEND_END_ARG_INFO()
  1058. ZEND_BEGIN_ARG_INFO_EX(arginfo_pdo_quote, 0, 0, 1)
  1059. ZEND_ARG_INFO(0, string)
  1060. ZEND_ARG_INFO(0, paramtype)
  1061. ZEND_END_ARG_INFO()
  1062. ZEND_BEGIN_ARG_INFO(arginfo_pdo__void, 0)
  1063. ZEND_END_ARG_INFO()
  1064. /* }}} */
  1065. const zend_function_entry pdo_dbh_functions[] = /* {{{ */ {
  1066. ZEND_MALIAS(PDO, __construct, dbh_constructor, arginfo_pdo___construct, ZEND_ACC_PUBLIC)
  1067. PHP_ME(PDO, prepare, arginfo_pdo_prepare, ZEND_ACC_PUBLIC)
  1068. PHP_ME(PDO, beginTransaction, arginfo_pdo__void, ZEND_ACC_PUBLIC)
  1069. PHP_ME(PDO, commit, arginfo_pdo__void, ZEND_ACC_PUBLIC)
  1070. PHP_ME(PDO, rollBack, arginfo_pdo__void, ZEND_ACC_PUBLIC)
  1071. PHP_ME(PDO, inTransaction, arginfo_pdo__void, ZEND_ACC_PUBLIC)
  1072. PHP_ME(PDO, setAttribute, arginfo_pdo_setattribute, ZEND_ACC_PUBLIC)
  1073. PHP_ME(PDO, exec, arginfo_pdo_exec, ZEND_ACC_PUBLIC)
  1074. PHP_ME(PDO, query, NULL, ZEND_ACC_PUBLIC)
  1075. PHP_ME(PDO, lastInsertId, arginfo_pdo_lastinsertid, ZEND_ACC_PUBLIC)
  1076. PHP_ME(PDO, errorCode, arginfo_pdo__void, ZEND_ACC_PUBLIC)
  1077. PHP_ME(PDO, errorInfo, arginfo_pdo__void, ZEND_ACC_PUBLIC)
  1078. PHP_ME(PDO, getAttribute, arginfo_pdo_getattribute, ZEND_ACC_PUBLIC)
  1079. PHP_ME(PDO, quote, arginfo_pdo_quote, ZEND_ACC_PUBLIC)
  1080. PHP_ME(PDO, __wakeup, arginfo_pdo__void, ZEND_ACC_PUBLIC|ZEND_ACC_FINAL)
  1081. PHP_ME(PDO, __sleep, arginfo_pdo__void, ZEND_ACC_PUBLIC|ZEND_ACC_FINAL)
  1082. PHP_ME(PDO, getAvailableDrivers, arginfo_pdo__void, ZEND_ACC_PUBLIC|ZEND_ACC_STATIC)
  1083. PHP_FE_END
  1084. };
  1085. /* }}} */
  1086. static void cls_method_dtor(zval *el) /* {{{ */ {
  1087. zend_function *func = (zend_function*)Z_PTR_P(el);
  1088. if (func->common.function_name) {
  1089. zend_string_release_ex(func->common.function_name, 0);
  1090. }
  1091. efree(func);
  1092. }
  1093. /* }}} */
  1094. static void cls_method_pdtor(zval *el) /* {{{ */ {
  1095. zend_function *func = (zend_function*)Z_PTR_P(el);
  1096. if (func->common.function_name) {
  1097. zend_string_release_ex(func->common.function_name, 1);
  1098. }
  1099. pefree(func, 1);
  1100. }
  1101. /* }}} */
  1102. /* {{{ overloaded object handlers for PDO class */
  1103. int pdo_hash_methods(pdo_dbh_object_t *dbh_obj, int kind)
  1104. {
  1105. const zend_function_entry *funcs;
  1106. zend_internal_function func;
  1107. size_t namelen;
  1108. char *lc_name;
  1109. pdo_dbh_t *dbh = dbh_obj->inner;
  1110. if (!dbh || !dbh->methods || !dbh->methods->get_driver_methods) {
  1111. return 0;
  1112. }
  1113. funcs = dbh->methods->get_driver_methods(dbh, kind);
  1114. if (!funcs) {
  1115. return 0;
  1116. }
  1117. dbh->cls_methods[kind] = pemalloc(sizeof(HashTable), dbh->is_persistent);
  1118. zend_hash_init_ex(dbh->cls_methods[kind], 8, NULL,
  1119. dbh->is_persistent? cls_method_pdtor : cls_method_dtor, dbh->is_persistent, 0);
  1120. memset(&func, 0, sizeof(func));
  1121. while (funcs->fname) {
  1122. func.type = ZEND_INTERNAL_FUNCTION;
  1123. func.handler = funcs->handler;
  1124. func.function_name = zend_string_init(funcs->fname, strlen(funcs->fname), dbh->is_persistent);
  1125. func.scope = dbh_obj->std.ce;
  1126. func.prototype = NULL;
  1127. if (funcs->flags) {
  1128. func.fn_flags = funcs->flags | ZEND_ACC_NEVER_CACHE;
  1129. } else {
  1130. func.fn_flags = ZEND_ACC_PUBLIC | ZEND_ACC_NEVER_CACHE;
  1131. }
  1132. if (funcs->arg_info) {
  1133. zend_internal_function_info *info = (zend_internal_function_info*)funcs->arg_info;
  1134. func.arg_info = (zend_internal_arg_info*)funcs->arg_info + 1;
  1135. func.num_args = funcs->num_args;
  1136. if (info->required_num_args == (uint32_t)-1) {
  1137. func.required_num_args = funcs->num_args;
  1138. } else {
  1139. func.required_num_args = info->required_num_args;
  1140. }
  1141. if (info->return_reference) {
  1142. func.fn_flags |= ZEND_ACC_RETURN_REFERENCE;
  1143. }
  1144. if (funcs->arg_info[funcs->num_args].is_variadic) {
  1145. func.fn_flags |= ZEND_ACC_VARIADIC;
  1146. /* Don't count the variadic argument */
  1147. func.num_args--;
  1148. }
  1149. } else {
  1150. func.arg_info = NULL;
  1151. func.num_args = 0;
  1152. func.required_num_args = 0;
  1153. }
  1154. zend_set_function_arg_flags((zend_function*)&func);
  1155. namelen = strlen(funcs->fname);
  1156. lc_name = emalloc(namelen+1);
  1157. zend_str_tolower_copy(lc_name, funcs->fname, namelen);
  1158. zend_hash_str_add_mem(dbh->cls_methods[kind], lc_name, namelen, &func, sizeof(func));
  1159. efree(lc_name);
  1160. funcs++;
  1161. }
  1162. return 1;
  1163. }
  1164. static union _zend_function *dbh_method_get(zend_object **object, zend_string *method_name, const zval *key)
  1165. {
  1166. zend_function *fbc = NULL;
  1167. pdo_dbh_object_t *dbh_obj = php_pdo_dbh_fetch_object(*object);
  1168. zend_string *lc_method_name;
  1169. if ((fbc = zend_std_get_method(object, method_name, key)) == NULL) {
  1170. /* not a pre-defined method, nor a user-defined method; check
  1171. * the driver specific methods */
  1172. if (!dbh_obj->inner->cls_methods[PDO_DBH_DRIVER_METHOD_KIND_DBH]) {
  1173. if (!pdo_hash_methods(dbh_obj,
  1174. PDO_DBH_DRIVER_METHOD_KIND_DBH)
  1175. || !dbh_obj->inner->cls_methods[PDO_DBH_DRIVER_METHOD_KIND_DBH]) {
  1176. goto out;
  1177. }
  1178. }
  1179. lc_method_name = zend_string_tolower(method_name);
  1180. fbc = zend_hash_find_ptr(dbh_obj->inner->cls_methods[PDO_DBH_DRIVER_METHOD_KIND_DBH], lc_method_name);
  1181. zend_string_release_ex(lc_method_name, 0);
  1182. }
  1183. out:
  1184. return fbc;
  1185. }
  1186. static int dbh_compare(zval *object1, zval *object2)
  1187. {
  1188. return -1;
  1189. }
  1190. static HashTable *dbh_get_gc(zval *object, zval **gc_data, int *gc_count)
  1191. {
  1192. pdo_dbh_t *dbh = Z_PDO_DBH_P(object);
  1193. *gc_data = &dbh->def_stmt_ctor_args;
  1194. *gc_count = 1;
  1195. return zend_std_get_properties(object);
  1196. }
  1197. static zend_object_handlers pdo_dbh_object_handlers;
  1198. static void pdo_dbh_free_storage(zend_object *std);
  1199. void pdo_dbh_init(void)
  1200. {
  1201. zend_class_entry ce;
  1202. INIT_CLASS_ENTRY(ce, "PDO", pdo_dbh_functions);
  1203. pdo_dbh_ce = zend_register_internal_class(&ce);
  1204. pdo_dbh_ce->create_object = pdo_dbh_new;
  1205. memcpy(&pdo_dbh_object_handlers, &std_object_handlers, sizeof(zend_object_handlers));
  1206. pdo_dbh_object_handlers.offset = XtOffsetOf(pdo_dbh_object_t, std);
  1207. pdo_dbh_object_handlers.dtor_obj = zend_objects_destroy_object;
  1208. pdo_dbh_object_handlers.free_obj = pdo_dbh_free_storage;
  1209. pdo_dbh_object_handlers.clone_obj = NULL;
  1210. pdo_dbh_object_handlers.get_method = dbh_method_get;
  1211. pdo_dbh_object_handlers.compare_objects = dbh_compare;
  1212. pdo_dbh_object_handlers.get_gc = dbh_get_gc;
  1213. REGISTER_PDO_CLASS_CONST_LONG("PARAM_BOOL", (zend_long)PDO_PARAM_BOOL);
  1214. REGISTER_PDO_CLASS_CONST_LONG("PARAM_NULL", (zend_long)PDO_PARAM_NULL);
  1215. REGISTER_PDO_CLASS_CONST_LONG("PARAM_INT", (zend_long)PDO_PARAM_INT);
  1216. REGISTER_PDO_CLASS_CONST_LONG("PARAM_STR", (zend_long)PDO_PARAM_STR);
  1217. REGISTER_PDO_CLASS_CONST_LONG("PARAM_LOB", (zend_long)PDO_PARAM_LOB);
  1218. REGISTER_PDO_CLASS_CONST_LONG("PARAM_STMT", (zend_long)PDO_PARAM_STMT);
  1219. REGISTER_PDO_CLASS_CONST_LONG("PARAM_INPUT_OUTPUT", (zend_long)PDO_PARAM_INPUT_OUTPUT);
  1220. REGISTER_PDO_CLASS_CONST_LONG("PARAM_STR_NATL", (zend_long)PDO_PARAM_STR_NATL);
  1221. REGISTER_PDO_CLASS_CONST_LONG("PARAM_STR_CHAR", (zend_long)PDO_PARAM_STR_CHAR);
  1222. REGISTER_PDO_CLASS_CONST_LONG("PARAM_EVT_ALLOC", (zend_long)PDO_PARAM_EVT_ALLOC);
  1223. REGISTER_PDO_CLASS_CONST_LONG("PARAM_EVT_FREE", (zend_long)PDO_PARAM_EVT_FREE);
  1224. REGISTER_PDO_CLASS_CONST_LONG("PARAM_EVT_EXEC_PRE", (zend_long)PDO_PARAM_EVT_EXEC_PRE);
  1225. REGISTER_PDO_CLASS_CONST_LONG("PARAM_EVT_EXEC_POST", (zend_long)PDO_PARAM_EVT_EXEC_POST);
  1226. REGISTER_PDO_CLASS_CONST_LONG("PARAM_EVT_FETCH_PRE", (zend_long)PDO_PARAM_EVT_FETCH_PRE);
  1227. REGISTER_PDO_CLASS_CONST_LONG("PARAM_EVT_FETCH_POST", (zend_long)PDO_PARAM_EVT_FETCH_POST);
  1228. REGISTER_PDO_CLASS_CONST_LONG("PARAM_EVT_NORMALIZE", (zend_long)PDO_PARAM_EVT_NORMALIZE);
  1229. REGISTER_PDO_CLASS_CONST_LONG("FETCH_LAZY", (zend_long)PDO_FETCH_LAZY);
  1230. REGISTER_PDO_CLASS_CONST_LONG("FETCH_ASSOC", (zend_long)PDO_FETCH_ASSOC);
  1231. REGISTER_PDO_CLASS_CONST_LONG("FETCH_NUM", (zend_long)PDO_FETCH_NUM);
  1232. REGISTER_PDO_CLASS_CONST_LONG("FETCH_BOTH", (zend_long)PDO_FETCH_BOTH);
  1233. REGISTER_PDO_CLASS_CONST_LONG("FETCH_OBJ", (zend_long)PDO_FETCH_OBJ);
  1234. REGISTER_PDO_CLASS_CONST_LONG("FETCH_BOUND", (zend_long)PDO_FETCH_BOUND);
  1235. REGISTER_PDO_CLASS_CONST_LONG("FETCH_COLUMN", (zend_long)PDO_FETCH_COLUMN);
  1236. REGISTER_PDO_CLASS_CONST_LONG("FETCH_CLASS", (zend_long)PDO_FETCH_CLASS);
  1237. REGISTER_PDO_CLASS_CONST_LONG("FETCH_INTO", (zend_long)PDO_FETCH_INTO);
  1238. REGISTER_PDO_CLASS_CONST_LONG("FETCH_FUNC", (zend_long)PDO_FETCH_FUNC);
  1239. REGISTER_PDO_CLASS_CONST_LONG("FETCH_GROUP", (zend_long)PDO_FETCH_GROUP);
  1240. REGISTER_PDO_CLASS_CONST_LONG("FETCH_UNIQUE", (zend_long)PDO_FETCH_UNIQUE);
  1241. REGISTER_PDO_CLASS_CONST_LONG("FETCH_KEY_PAIR", (zend_long)PDO_FETCH_KEY_PAIR);
  1242. REGISTER_PDO_CLASS_CONST_LONG("FETCH_CLASSTYPE", (zend_long)PDO_FETCH_CLASSTYPE);
  1243. REGISTER_PDO_CLASS_CONST_LONG("FETCH_SERIALIZE",(zend_long)PDO_FETCH_SERIALIZE);
  1244. REGISTER_PDO_CLASS_CONST_LONG("FETCH_PROPS_LATE", (zend_long)PDO_FETCH_PROPS_LATE);
  1245. REGISTER_PDO_CLASS_CONST_LONG("FETCH_NAMED", (zend_long)PDO_FETCH_NAMED);
  1246. REGISTER_PDO_CLASS_CONST_LONG("ATTR_AUTOCOMMIT", (zend_long)PDO_ATTR_AUTOCOMMIT);
  1247. REGISTER_PDO_CLASS_CONST_LONG("ATTR_PREFETCH", (zend_long)PDO_ATTR_PREFETCH);
  1248. REGISTER_PDO_CLASS_CONST_LONG("ATTR_TIMEOUT", (zend_long)PDO_ATTR_TIMEOUT);
  1249. REGISTER_PDO_CLASS_CONST_LONG("ATTR_ERRMODE", (zend_long)PDO_ATTR_ERRMODE);
  1250. REGISTER_PDO_CLASS_CONST_LONG("ATTR_SERVER_VERSION", (zend_long)PDO_ATTR_SERVER_VERSION);
  1251. REGISTER_PDO_CLASS_CONST_LONG("ATTR_CLIENT_VERSION", (zend_long)PDO_ATTR_CLIENT_VERSION);
  1252. REGISTER_PDO_CLASS_CONST_LONG("ATTR_SERVER_INFO", (zend_long)PDO_ATTR_SERVER_INFO);
  1253. REGISTER_PDO_CLASS_CONST_LONG("ATTR_CONNECTION_STATUS", (zend_long)PDO_ATTR_CONNECTION_STATUS);
  1254. REGISTER_PDO_CLASS_CONST_LONG("ATTR_CASE", (zend_long)PDO_ATTR_CASE);
  1255. REGISTER_PDO_CLASS_CONST_LONG("ATTR_CURSOR_NAME", (zend_long)PDO_ATTR_CURSOR_NAME);
  1256. REGISTER_PDO_CLASS_CONST_LONG("ATTR_CURSOR", (zend_long)PDO_ATTR_CURSOR);
  1257. REGISTER_PDO_CLASS_CONST_LONG("ATTR_ORACLE_NULLS", (zend_long)PDO_ATTR_ORACLE_NULLS);
  1258. REGISTER_PDO_CLASS_CONST_LONG("ATTR_PERSISTENT", (zend_long)PDO_ATTR_PERSISTENT);
  1259. REGISTER_PDO_CLASS_CONST_LONG("ATTR_STATEMENT_CLASS", (zend_long)PDO_ATTR_STATEMENT_CLASS);
  1260. REGISTER_PDO_CLASS_CONST_LONG("ATTR_FETCH_TABLE_NAMES", (zend_long)PDO_ATTR_FETCH_TABLE_NAMES);
  1261. REGISTER_PDO_CLASS_CONST_LONG("ATTR_FETCH_CATALOG_NAMES", (zend_long)PDO_ATTR_FETCH_CATALOG_NAMES);
  1262. REGISTER_PDO_CLASS_CONST_LONG("ATTR_DRIVER_NAME", (zend_long)PDO_ATTR_DRIVER_NAME);
  1263. REGISTER_PDO_CLASS_CONST_LONG("ATTR_STRINGIFY_FETCHES", (zend_long)PDO_ATTR_STRINGIFY_FETCHES);
  1264. REGISTER_PDO_CLASS_CONST_LONG("ATTR_MAX_COLUMN_LEN", (zend_long)PDO_ATTR_MAX_COLUMN_LEN);
  1265. REGISTER_PDO_CLASS_CONST_LONG("ATTR_EMULATE_PREPARES", (zend_long)PDO_ATTR_EMULATE_PREPARES);
  1266. REGISTER_PDO_CLASS_CONST_LONG("ATTR_DEFAULT_FETCH_MODE", (zend_long)PDO_ATTR_DEFAULT_FETCH_MODE);
  1267. REGISTER_PDO_CLASS_CONST_LONG("ATTR_DEFAULT_STR_PARAM", (zend_long)PDO_ATTR_DEFAULT_STR_PARAM);
  1268. REGISTER_PDO_CLASS_CONST_LONG("ERRMODE_SILENT", (zend_long)PDO_ERRMODE_SILENT);
  1269. REGISTER_PDO_CLASS_CONST_LONG("ERRMODE_WARNING", (zend_long)PDO_ERRMODE_WARNING);
  1270. REGISTER_PDO_CLASS_CONST_LONG("ERRMODE_EXCEPTION", (zend_long)PDO_ERRMODE_EXCEPTION);
  1271. REGISTER_PDO_CLASS_CONST_LONG("CASE_NATURAL", (zend_long)PDO_CASE_NATURAL);
  1272. REGISTER_PDO_CLASS_CONST_LONG("CASE_LOWER", (zend_long)PDO_CASE_LOWER);
  1273. REGISTER_PDO_CLASS_CONST_LONG("CASE_UPPER", (zend_long)PDO_CASE_UPPER);
  1274. REGISTER_PDO_CLASS_CONST_LONG("NULL_NATURAL", (zend_long)PDO_NULL_NATURAL);
  1275. REGISTER_PDO_CLASS_CONST_LONG("NULL_EMPTY_STRING", (zend_long)PDO_NULL_EMPTY_STRING);
  1276. REGISTER_PDO_CLASS_CONST_LONG("NULL_TO_STRING", (zend_long)PDO_NULL_TO_STRING);
  1277. REGISTER_PDO_CLASS_CONST_STRING("ERR_NONE", PDO_ERR_NONE);
  1278. REGISTER_PDO_CLASS_CONST_LONG("FETCH_ORI_NEXT", (zend_long)PDO_FETCH_ORI_NEXT);
  1279. REGISTER_PDO_CLASS_CONST_LONG("FETCH_ORI_PRIOR", (zend_long)PDO_FETCH_ORI_PRIOR);
  1280. REGISTER_PDO_CLASS_CONST_LONG("FETCH_ORI_FIRST", (zend_long)PDO_FETCH_ORI_FIRST);
  1281. REGISTER_PDO_CLASS_CONST_LONG("FETCH_ORI_LAST", (zend_long)PDO_FETCH_ORI_LAST);
  1282. REGISTER_PDO_CLASS_CONST_LONG("FETCH_ORI_ABS", (zend_long)PDO_FETCH_ORI_ABS);
  1283. REGISTER_PDO_CLASS_CONST_LONG("FETCH_ORI_REL", (zend_long)PDO_FETCH_ORI_REL);
  1284. REGISTER_PDO_CLASS_CONST_LONG("CURSOR_FWDONLY", (zend_long)PDO_CURSOR_FWDONLY);
  1285. REGISTER_PDO_CLASS_CONST_LONG("CURSOR_SCROLL", (zend_long)PDO_CURSOR_SCROLL);
  1286. }
  1287. static void dbh_free(pdo_dbh_t *dbh, zend_bool free_persistent)
  1288. {
  1289. int i;
  1290. if (dbh->query_stmt) {
  1291. zval_ptr_dtor(&dbh->query_stmt_zval);
  1292. dbh->query_stmt = NULL;
  1293. }
  1294. if (dbh->is_persistent) {
  1295. #if ZEND_DEBUG
  1296. ZEND_ASSERT(!free_persistent || (dbh->refcount == 1));
  1297. #endif
  1298. if (!free_persistent && (--dbh->refcount)) {
  1299. return;
  1300. }
  1301. }
  1302. if (dbh->methods) {
  1303. dbh->methods->closer(dbh);
  1304. }
  1305. if (dbh->data_source) {
  1306. pefree((char *)dbh->data_source, dbh->is_persistent);
  1307. }
  1308. if (dbh->username) {
  1309. pefree(dbh->username, dbh->is_persistent);
  1310. }
  1311. if (dbh->password) {
  1312. pefree(dbh->password, dbh->is_persistent);
  1313. }
  1314. if (dbh->persistent_id) {
  1315. pefree((char *)dbh->persistent_id, dbh->is_persistent);
  1316. }
  1317. if (!Z_ISUNDEF(dbh->def_stmt_ctor_args)) {
  1318. zval_ptr_dtor(&dbh->def_stmt_ctor_args);
  1319. }
  1320. for (i = 0; i < PDO_DBH_DRIVER_METHOD_KIND__MAX; i++) {
  1321. if (dbh->cls_methods[i]) {
  1322. zend_hash_destroy(dbh->cls_methods[i]);
  1323. pefree(dbh->cls_methods[i], dbh->is_persistent);
  1324. }
  1325. }
  1326. pefree(dbh, dbh->is_persistent);
  1327. }
  1328. static void pdo_dbh_free_storage(zend_object *std)
  1329. {
  1330. pdo_dbh_t *dbh = php_pdo_dbh_fetch_inner(std);
  1331. if (dbh->in_txn && dbh->methods && dbh->methods->rollback) {
  1332. dbh->methods->rollback(dbh);
  1333. dbh->in_txn = 0;
  1334. }
  1335. if (dbh->is_persistent && dbh->methods && dbh->methods->persistent_shutdown) {
  1336. dbh->methods->persistent_shutdown(dbh);
  1337. }
  1338. zend_object_std_dtor(std);
  1339. dbh_free(dbh, 0);
  1340. }
  1341. zend_object *pdo_dbh_new(zend_class_entry *ce)
  1342. {
  1343. pdo_dbh_object_t *dbh;
  1344. dbh = zend_object_alloc(sizeof(pdo_dbh_object_t), ce);
  1345. zend_object_std_init(&dbh->std, ce);
  1346. object_properties_init(&dbh->std, ce);
  1347. rebuild_object_properties(&dbh->std);
  1348. dbh->inner = ecalloc(1, sizeof(pdo_dbh_t));
  1349. dbh->inner->def_stmt_ce = pdo_dbstmt_ce;
  1350. dbh->std.handlers = &pdo_dbh_object_handlers;
  1351. return &dbh->std;
  1352. }
  1353. /* }}} */
  1354. ZEND_RSRC_DTOR_FUNC(php_pdo_pdbh_dtor) /* {{{ */
  1355. {
  1356. if (res->ptr) {
  1357. pdo_dbh_t *dbh = (pdo_dbh_t*)res->ptr;
  1358. dbh_free(dbh, 1);
  1359. res->ptr = NULL;
  1360. }
  1361. }
  1362. /* }}} */
  1363. /*
  1364. * Local variables:
  1365. * tab-width: 4
  1366. * c-basic-offset: 4
  1367. * End:
  1368. * vim600: noet sw=4 ts=4 fdm=marker
  1369. * vim<600: noet sw=4 ts=4
  1370. */