pdo_dbh.c 48 KB

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