mysql_statement.c 25 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949
  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: George Schlossnagle <george@omniti.com> |
  16. | Wez Furlong <wez@php.net> |
  17. | Johannes Schlueter <johannes@mysql.com> |
  18. +----------------------------------------------------------------------+
  19. */
  20. #ifdef HAVE_CONFIG_H
  21. #include "config.h"
  22. #endif
  23. #include "php.h"
  24. #include "php_ini.h"
  25. #include "ext/standard/info.h"
  26. #include "pdo/php_pdo.h"
  27. #include "pdo/php_pdo_driver.h"
  28. #include "php_pdo_mysql.h"
  29. #include "php_pdo_mysql_int.h"
  30. #ifdef PDO_USE_MYSQLND
  31. # define pdo_mysql_stmt_execute_prepared(stmt) pdo_mysql_stmt_execute_prepared_mysqlnd(stmt)
  32. # define pdo_free_bound_result(res) zval_ptr_dtor(res.zv)
  33. # define pdo_mysql_stmt_close(stmt) mysqlnd_stmt_close(stmt, 0)
  34. #else
  35. # define pdo_mysql_stmt_execute_prepared(stmt) pdo_mysql_stmt_execute_prepared_libmysql(stmt)
  36. # define pdo_free_bound_result(res) efree(res.buffer)
  37. # define pdo_mysql_stmt_close(stmt) mysql_stmt_close(stmt)
  38. #endif
  39. static int pdo_mysql_stmt_dtor(pdo_stmt_t *stmt) /* {{{ */
  40. {
  41. pdo_mysql_stmt *S = (pdo_mysql_stmt*)stmt->driver_data;
  42. PDO_DBG_ENTER("pdo_mysql_stmt_dtor");
  43. PDO_DBG_INF_FMT("stmt=%p", S->stmt);
  44. if (S->result) {
  45. /* free the resource */
  46. mysql_free_result(S->result);
  47. S->result = NULL;
  48. }
  49. if (S->einfo.errmsg) {
  50. pefree(S->einfo.errmsg, stmt->dbh->is_persistent);
  51. S->einfo.errmsg = NULL;
  52. }
  53. if (S->stmt) {
  54. pdo_mysql_stmt_close(S->stmt);
  55. S->stmt = NULL;
  56. }
  57. #ifndef PDO_USE_MYSQLND
  58. if (S->params) {
  59. efree(S->params);
  60. }
  61. if (S->in_null) {
  62. efree(S->in_null);
  63. }
  64. if (S->in_length) {
  65. efree(S->in_length);
  66. }
  67. if (S->bound_result)
  68. {
  69. int i;
  70. for (i = 0; i < stmt->column_count; i++) {
  71. pdo_free_bound_result(S->bound_result[i]);
  72. }
  73. efree(S->bound_result);
  74. efree(S->out_null);
  75. efree(S->out_length);
  76. }
  77. #endif
  78. if (!Z_ISUNDEF(stmt->database_object_handle)
  79. && IS_OBJ_VALID(EG(objects_store).object_buckets[Z_OBJ_HANDLE(stmt->database_object_handle)])
  80. && (!(OBJ_FLAGS(Z_OBJ(stmt->database_object_handle)) & IS_OBJ_FREE_CALLED))) {
  81. while (mysql_more_results(S->H->server)) {
  82. MYSQL_RES *res;
  83. if (mysql_next_result(S->H->server) != 0) {
  84. break;
  85. }
  86. res = mysql_store_result(S->H->server);
  87. if (res) {
  88. mysql_free_result(res);
  89. }
  90. }
  91. }
  92. #if PDO_USE_MYSQLND
  93. if (!S->stmt && S->current_data) {
  94. mnd_free(S->current_data);
  95. }
  96. #endif /* PDO_USE_MYSQLND */
  97. efree(S);
  98. PDO_DBG_RETURN(1);
  99. }
  100. /* }}} */
  101. static void pdo_mysql_stmt_set_row_count(pdo_stmt_t *stmt) /* {{{ */
  102. {
  103. zend_long row_count;
  104. pdo_mysql_stmt *S = stmt->driver_data;
  105. row_count = (zend_long) mysql_stmt_affected_rows(S->stmt);
  106. if (row_count != (zend_long)-1) {
  107. stmt->row_count = row_count;
  108. }
  109. }
  110. /* }}} */
  111. static int pdo_mysql_fill_stmt_from_result(pdo_stmt_t *stmt) /* {{{ */
  112. {
  113. pdo_mysql_stmt *S = (pdo_mysql_stmt*)stmt->driver_data;
  114. pdo_mysql_db_handle *H = S->H;
  115. my_ulonglong row_count;
  116. PDO_DBG_ENTER("pdo_mysql_fill_stmt_from_result");
  117. row_count = mysql_affected_rows(H->server);
  118. if (row_count == (my_ulonglong)-1) {
  119. /* we either have a query that returned a result set or an error occurred
  120. lets see if we have access to a result set */
  121. if (!H->buffered) {
  122. S->result = mysql_use_result(H->server);
  123. } else {
  124. S->result = mysql_store_result(H->server);
  125. }
  126. if (NULL == S->result) {
  127. pdo_mysql_error_stmt(stmt);
  128. PDO_DBG_RETURN(0);
  129. }
  130. stmt->row_count = (zend_long) mysql_num_rows(S->result);
  131. stmt->column_count = (int) mysql_num_fields(S->result);
  132. S->fields = mysql_fetch_fields(S->result);
  133. } else {
  134. /* this was a DML or DDL query (INSERT, UPDATE, DELETE, ... */
  135. stmt->row_count = (zend_long) row_count;
  136. }
  137. PDO_DBG_RETURN(1);
  138. }
  139. /* }}} */
  140. #ifndef PDO_USE_MYSQLND
  141. static int pdo_mysql_stmt_execute_prepared_libmysql(pdo_stmt_t *stmt) /* {{{ */
  142. {
  143. pdo_mysql_stmt *S = stmt->driver_data;
  144. pdo_mysql_db_handle *H = S->H;
  145. PDO_DBG_ENTER("pdo_mysql_stmt_execute_prepared_libmysql");
  146. /* (re)bind the parameters */
  147. if (mysql_stmt_bind_param(S->stmt, S->params) || mysql_stmt_execute(S->stmt)) {
  148. if (S->params) {
  149. memset(S->params, 0, S->num_params * sizeof(MYSQL_BIND));
  150. }
  151. pdo_mysql_error_stmt(stmt);
  152. if (mysql_stmt_errno(S->stmt) == 2057) {
  153. /* CR_NEW_STMT_METADATA makes the statement unusable */
  154. S->stmt = NULL;
  155. }
  156. PDO_DBG_RETURN(0);
  157. }
  158. if (!S->result) {
  159. int i;
  160. /* figure out the result set format, if any */
  161. S->result = mysql_stmt_result_metadata(S->stmt);
  162. if (S->result) {
  163. int calc_max_length = H->buffered && S->max_length == 1;
  164. S->fields = mysql_fetch_fields(S->result);
  165. if (S->bound_result) {
  166. int i;
  167. for (i = 0; i < stmt->column_count; i++) {
  168. efree(S->bound_result[i].buffer);
  169. }
  170. efree(S->bound_result);
  171. efree(S->out_null);
  172. efree(S->out_length);
  173. }
  174. stmt->column_count = (int)mysql_num_fields(S->result);
  175. S->bound_result = ecalloc(stmt->column_count, sizeof(MYSQL_BIND));
  176. S->out_null = ecalloc(stmt->column_count, sizeof(my_bool));
  177. S->out_length = ecalloc(stmt->column_count, sizeof(zend_ulong));
  178. /* summon memory to hold the row */
  179. for (i = 0; i < stmt->column_count; i++) {
  180. if (calc_max_length && S->fields[i].type == FIELD_TYPE_BLOB) {
  181. my_bool on = 1;
  182. mysql_stmt_attr_set(S->stmt, STMT_ATTR_UPDATE_MAX_LENGTH, &on);
  183. calc_max_length = 0;
  184. }
  185. switch (S->fields[i].type) {
  186. case FIELD_TYPE_INT24:
  187. S->bound_result[i].buffer_length = MAX_MEDIUMINT_WIDTH + 1;
  188. break;
  189. case FIELD_TYPE_LONG:
  190. S->bound_result[i].buffer_length = MAX_INT_WIDTH + 1;
  191. break;
  192. case FIELD_TYPE_LONGLONG:
  193. S->bound_result[i].buffer_length = MAX_BIGINT_WIDTH + 1;
  194. break;
  195. case FIELD_TYPE_TINY:
  196. S->bound_result[i].buffer_length = MAX_TINYINT_WIDTH + 1;
  197. break;
  198. case FIELD_TYPE_SHORT:
  199. S->bound_result[i].buffer_length = MAX_SMALLINT_WIDTH + 1;
  200. break;
  201. default:
  202. S->bound_result[i].buffer_length =
  203. S->fields[i].max_length? S->fields[i].max_length:
  204. S->fields[i].length;
  205. /* work-around for longtext and alike */
  206. if (S->bound_result[i].buffer_length > H->max_buffer_size) {
  207. S->bound_result[i].buffer_length = H->max_buffer_size;
  208. }
  209. }
  210. /* there are cases where the length reported by mysql is too short.
  211. * eg: when describing a table that contains an enum column. Since
  212. * we have no way of knowing the true length either, we'll bump up
  213. * our buffer size to a reasonable size, just in case */
  214. if (S->fields[i].max_length == 0 && S->bound_result[i].buffer_length < 128 && MYSQL_TYPE_VAR_STRING) {
  215. S->bound_result[i].buffer_length = 128;
  216. }
  217. S->out_length[i] = 0;
  218. S->bound_result[i].buffer = emalloc(S->bound_result[i].buffer_length);
  219. S->bound_result[i].is_null = &S->out_null[i];
  220. S->bound_result[i].length = &S->out_length[i];
  221. S->bound_result[i].buffer_type = MYSQL_TYPE_STRING;
  222. }
  223. if (mysql_stmt_bind_result(S->stmt, S->bound_result)) {
  224. pdo_mysql_error_stmt(stmt);
  225. PDO_DBG_RETURN(0);
  226. }
  227. /* if buffered, pre-fetch all the data */
  228. if (H->buffered) {
  229. mysql_stmt_store_result(S->stmt);
  230. }
  231. }
  232. }
  233. pdo_mysql_stmt_set_row_count(stmt);
  234. PDO_DBG_RETURN(1);
  235. }
  236. /* }}} */
  237. #endif
  238. #ifdef PDO_USE_MYSQLND
  239. static int pdo_mysql_stmt_execute_prepared_mysqlnd(pdo_stmt_t *stmt) /* {{{ */
  240. {
  241. pdo_mysql_stmt *S = stmt->driver_data;
  242. pdo_mysql_db_handle *H = S->H;
  243. int i;
  244. PDO_DBG_ENTER("pdo_mysql_stmt_execute_prepared_mysqlnd");
  245. if (mysql_stmt_execute(S->stmt)) {
  246. pdo_mysql_error_stmt(stmt);
  247. PDO_DBG_RETURN(0);
  248. }
  249. if (S->result) {
  250. /* TODO: add a test to check if we really have zvals here... */
  251. mysql_free_result(S->result);
  252. S->result = NULL;
  253. }
  254. /* for SHOW/DESCRIBE and others the column/field count is not available before execute */
  255. stmt->column_count = mysql_stmt_field_count(S->stmt);
  256. for (i = 0; i < stmt->column_count; i++) {
  257. mysqlnd_stmt_bind_one_result(S->stmt, i);
  258. }
  259. S->result = mysqlnd_stmt_result_metadata(S->stmt);
  260. if (S->result) {
  261. S->fields = mysql_fetch_fields(S->result);
  262. /* if buffered, pre-fetch all the data */
  263. if (H->buffered) {
  264. if (mysql_stmt_store_result(S->stmt)) {
  265. PDO_DBG_RETURN(0);
  266. }
  267. }
  268. }
  269. pdo_mysql_stmt_set_row_count(stmt);
  270. PDO_DBG_RETURN(1);
  271. }
  272. /* }}} */
  273. #endif
  274. static int pdo_mysql_stmt_execute(pdo_stmt_t *stmt) /* {{{ */
  275. {
  276. pdo_mysql_stmt *S = (pdo_mysql_stmt*)stmt->driver_data;
  277. pdo_mysql_db_handle *H = S->H;
  278. PDO_DBG_ENTER("pdo_mysql_stmt_execute");
  279. PDO_DBG_INF_FMT("stmt=%p", S->stmt);
  280. if (S->stmt) {
  281. PDO_DBG_RETURN(pdo_mysql_stmt_execute_prepared(stmt));
  282. }
  283. /* ensure that we free any previous unfetched results */
  284. if (S->result) {
  285. mysql_free_result(S->result);
  286. S->result = NULL;
  287. }
  288. if (mysql_real_query(H->server, stmt->active_query_string, stmt->active_query_stringlen) != 0) {
  289. pdo_mysql_error_stmt(stmt);
  290. PDO_DBG_RETURN(0);
  291. }
  292. PDO_DBG_RETURN(pdo_mysql_fill_stmt_from_result(stmt));
  293. }
  294. /* }}} */
  295. static int pdo_mysql_stmt_next_rowset(pdo_stmt_t *stmt) /* {{{ */
  296. {
  297. pdo_mysql_stmt *S = (pdo_mysql_stmt*)stmt->driver_data;
  298. pdo_mysql_db_handle *H = S->H;
  299. #if PDO_USE_MYSQLND
  300. zend_long row_count;
  301. #endif
  302. PDO_DBG_ENTER("pdo_mysql_stmt_next_rowset");
  303. PDO_DBG_INF_FMT("stmt=%p", S->stmt);
  304. #if PDO_USE_MYSQLND
  305. if (!H->emulate_prepare) {
  306. if (!mysqlnd_stmt_more_results(S->stmt)) {
  307. PDO_DBG_RETURN(0);
  308. }
  309. if (mysqlnd_stmt_next_result(S->stmt)) {
  310. PDO_DBG_RETURN(0);
  311. }
  312. if (!mysqlnd_stmt_more_results(S->stmt)) {
  313. /*
  314. MySQL gives us n + 1 result sets for
  315. CALL proc() and n result sets returned by the proc itself.
  316. Result set n + 1 is about the procedure call itself.
  317. As the PDO emulation does not return it, we skip it as well
  318. */
  319. PDO_DBG_RETURN(0);
  320. }
  321. /* TODO - this code is stolen from execute() - see above */
  322. if (S->result) {
  323. mysql_free_result(S->result);
  324. S->result = NULL;
  325. }
  326. {
  327. /* for SHOW/DESCRIBE and others the column/field count is not available before execute */
  328. int i;
  329. stmt->column_count = mysql_stmt_field_count(S->stmt);
  330. for (i = 0; i < stmt->column_count; i++) {
  331. mysqlnd_stmt_bind_one_result(S->stmt, i);
  332. }
  333. }
  334. S->result = mysqlnd_stmt_result_metadata(S->stmt);
  335. if (S->result) {
  336. S->fields = mysql_fetch_fields(S->result);
  337. /* if buffered, pre-fetch all the data */
  338. if (H->buffered) {
  339. if (mysql_stmt_store_result(S->stmt)) {
  340. PDO_DBG_RETURN(1);
  341. }
  342. }
  343. }
  344. row_count = (zend_long) mysql_stmt_affected_rows(S->stmt);
  345. if (row_count != (zend_long)-1) {
  346. stmt->row_count = row_count;
  347. }
  348. PDO_DBG_RETURN(1);
  349. }
  350. #endif
  351. /* ensure that we free any previous unfetched results */
  352. #ifndef PDO_USE_MYSQLND
  353. if (S->stmt) {
  354. if (S->result) {
  355. stmt->column_count = (int)mysql_num_fields(S->result);
  356. }
  357. mysql_stmt_free_result(S->stmt);
  358. }
  359. #endif
  360. if (S->result) {
  361. mysql_free_result(S->result);
  362. S->result = NULL;
  363. }
  364. if (!mysql_more_results(H->server)) {
  365. /* No more results */
  366. PDO_DBG_RETURN(0);
  367. }
  368. #if PDO_USE_MYSQLND
  369. if (mysql_next_result(H->server) == FAIL) {
  370. pdo_mysql_error_stmt(stmt);
  371. PDO_DBG_RETURN(0);
  372. } else {
  373. PDO_DBG_RETURN(pdo_mysql_fill_stmt_from_result(stmt));
  374. }
  375. #else
  376. if (mysql_next_result(H->server) > 0) {
  377. pdo_mysql_error_stmt(stmt);
  378. PDO_DBG_RETURN(0);
  379. } else {
  380. PDO_DBG_RETURN(pdo_mysql_fill_stmt_from_result(stmt));
  381. }
  382. #endif
  383. }
  384. /* }}} */
  385. static const char * const pdo_param_event_names[] =
  386. {
  387. "PDO_PARAM_EVT_ALLOC",
  388. "PDO_PARAM_EVT_FREE",
  389. "PDO_PARAM_EVT_EXEC_PRE",
  390. "PDO_PARAM_EVT_EXEC_POST",
  391. "PDO_PARAM_EVT_FETCH_PRE",
  392. "PDO_PARAM_EVT_FETCH_POST",
  393. "PDO_PARAM_EVT_NORMALIZE",
  394. };
  395. static int pdo_mysql_stmt_param_hook(pdo_stmt_t *stmt, struct pdo_bound_param_data *param, enum pdo_param_event event_type) /* {{{ */
  396. {
  397. zval *parameter;
  398. #ifndef PDO_USE_MYSQLND
  399. PDO_MYSQL_PARAM_BIND *b;
  400. #endif
  401. pdo_mysql_stmt *S = (pdo_mysql_stmt*)stmt->driver_data;
  402. PDO_DBG_ENTER("pdo_mysql_stmt_param_hook");
  403. PDO_DBG_INF_FMT("stmt=%p", S->stmt);
  404. PDO_DBG_INF_FMT("event = %s", pdo_param_event_names[event_type]);
  405. if (S->stmt && param->is_param) {
  406. switch (event_type) {
  407. case PDO_PARAM_EVT_ALLOC:
  408. /* sanity check parameter number range */
  409. if (param->paramno < 0 || param->paramno >= S->num_params) {
  410. strcpy(stmt->error_code, "HY093");
  411. PDO_DBG_RETURN(0);
  412. }
  413. S->params_given++;
  414. #ifndef PDO_USE_MYSQLND
  415. b = &S->params[param->paramno];
  416. param->driver_data = b;
  417. b->is_null = &S->in_null[param->paramno];
  418. b->length = &S->in_length[param->paramno];
  419. /* recall how many parameters have been provided */
  420. #endif
  421. PDO_DBG_RETURN(1);
  422. case PDO_PARAM_EVT_EXEC_PRE:
  423. if (S->params_given < (unsigned int) S->num_params) {
  424. /* too few parameter bound */
  425. PDO_DBG_ERR("too few parameters bound");
  426. strcpy(stmt->error_code, "HY093");
  427. PDO_DBG_RETURN(0);
  428. }
  429. if (!Z_ISREF(param->parameter)) {
  430. parameter = &param->parameter;
  431. } else {
  432. parameter = Z_REFVAL(param->parameter);
  433. }
  434. #if PDO_USE_MYSQLND
  435. if (PDO_PARAM_TYPE(param->param_type) == PDO_PARAM_NULL || (Z_TYPE_P(parameter) == IS_NULL)) {
  436. mysqlnd_stmt_bind_one_param(S->stmt, param->paramno, parameter, MYSQL_TYPE_NULL);
  437. PDO_DBG_RETURN(1);
  438. }
  439. #else
  440. b = (PDO_MYSQL_PARAM_BIND*)param->driver_data;
  441. *b->is_null = 0;
  442. if (PDO_PARAM_TYPE(param->param_type) == PDO_PARAM_NULL || Z_TYPE_P(parameter) == IS_NULL) {
  443. *b->is_null = 1;
  444. b->buffer_type = MYSQL_TYPE_STRING;
  445. b->buffer = NULL;
  446. b->buffer_length = 0;
  447. *b->length = 0;
  448. PDO_DBG_RETURN(1);
  449. }
  450. #endif /* PDO_USE_MYSQLND */
  451. switch (PDO_PARAM_TYPE(param->param_type)) {
  452. case PDO_PARAM_STMT:
  453. PDO_DBG_RETURN(0);
  454. case PDO_PARAM_LOB:
  455. PDO_DBG_INF("PDO_PARAM_LOB");
  456. if (!Z_ISREF(param->parameter)) {
  457. parameter = &param->parameter;
  458. } else {
  459. parameter = Z_REFVAL(param->parameter);
  460. }
  461. if (Z_TYPE_P(parameter) == IS_RESOURCE) {
  462. php_stream *stm = NULL;
  463. php_stream_from_zval_no_verify(stm, parameter);
  464. if (stm) {
  465. zend_string *mem = php_stream_copy_to_mem(stm, PHP_STREAM_COPY_ALL, 0);
  466. zval_ptr_dtor(parameter);
  467. ZVAL_STR(parameter, mem ? mem : ZSTR_EMPTY_ALLOC());
  468. } else {
  469. pdo_raise_impl_error(stmt->dbh, stmt, "HY105", "Expected a stream resource");
  470. return 0;
  471. }
  472. }
  473. /* fall through */
  474. default:
  475. ;
  476. }
  477. #if PDO_USE_MYSQLND
  478. /* Is it really correct to check the zval's type? - But well, that's what the old code below does, too */
  479. PDO_DBG_INF_FMT("param->parameter->type=%d", Z_TYPE(param->parameter));
  480. if (!Z_ISREF(param->parameter)) {
  481. parameter = &param->parameter;
  482. } else {
  483. parameter = Z_REFVAL(param->parameter);
  484. }
  485. switch (Z_TYPE_P(parameter)) {
  486. case IS_STRING:
  487. mysqlnd_stmt_bind_one_param(S->stmt, param->paramno, parameter, MYSQL_TYPE_VAR_STRING);
  488. break;
  489. case IS_LONG:
  490. #if SIZEOF_ZEND_LONG==8
  491. mysqlnd_stmt_bind_one_param(S->stmt, param->paramno, parameter, MYSQL_TYPE_LONGLONG);
  492. #elif SIZEOF_ZEND_LONG==4
  493. mysqlnd_stmt_bind_one_param(S->stmt, param->paramno, parameter, MYSQL_TYPE_LONG);
  494. #endif /* SIZEOF_LONG */
  495. break;
  496. case IS_TRUE:
  497. case IS_FALSE:
  498. mysqlnd_stmt_bind_one_param(S->stmt, param->paramno, parameter, MYSQL_TYPE_TINY);
  499. break;
  500. case IS_DOUBLE:
  501. mysqlnd_stmt_bind_one_param(S->stmt, param->paramno, parameter, MYSQL_TYPE_DOUBLE);
  502. break;
  503. default:
  504. PDO_DBG_RETURN(0);
  505. }
  506. PDO_DBG_RETURN(1);
  507. #else
  508. PDO_DBG_INF_FMT("param->parameter->type=%d", Z_TYPE(param->parameter));
  509. if (!Z_ISREF(param->parameter)) {
  510. parameter = &param->parameter;
  511. } else {
  512. parameter = Z_REFVAL(param->parameter);
  513. }
  514. switch (Z_TYPE_P(parameter)) {
  515. case IS_STRING:
  516. b->buffer_type = MYSQL_TYPE_STRING;
  517. b->buffer = Z_STRVAL_P(parameter);
  518. b->buffer_length = Z_STRLEN_P(parameter);
  519. *b->length = Z_STRLEN_P(parameter);
  520. PDO_DBG_RETURN(1);
  521. case IS_LONG:
  522. b->buffer_type = MYSQL_TYPE_LONG;
  523. b->buffer = &Z_LVAL_P(parameter);
  524. PDO_DBG_RETURN(1);
  525. case IS_DOUBLE:
  526. b->buffer_type = MYSQL_TYPE_DOUBLE;
  527. b->buffer = &Z_DVAL_P(parameter);
  528. PDO_DBG_RETURN(1);
  529. default:
  530. PDO_DBG_RETURN(0);
  531. }
  532. #endif /* PDO_USE_MYSQLND */
  533. case PDO_PARAM_EVT_FREE:
  534. case PDO_PARAM_EVT_EXEC_POST:
  535. case PDO_PARAM_EVT_FETCH_PRE:
  536. case PDO_PARAM_EVT_FETCH_POST:
  537. case PDO_PARAM_EVT_NORMALIZE:
  538. /* do nothing */
  539. break;
  540. }
  541. }
  542. PDO_DBG_RETURN(1);
  543. }
  544. /* }}} */
  545. static int pdo_mysql_stmt_fetch(pdo_stmt_t *stmt, enum pdo_fetch_orientation ori, zend_long offset) /* {{{ */
  546. {
  547. pdo_mysql_stmt *S = (pdo_mysql_stmt*)stmt->driver_data;
  548. #if PDO_USE_MYSQLND
  549. zend_bool fetched_anything;
  550. PDO_DBG_ENTER("pdo_mysql_stmt_fetch");
  551. PDO_DBG_INF_FMT("stmt=%p", S->stmt);
  552. if (S->stmt) {
  553. if (FAIL == mysqlnd_stmt_fetch(S->stmt, &fetched_anything) || fetched_anything == FALSE) {
  554. PDO_DBG_RETURN(0);
  555. }
  556. PDO_DBG_RETURN(1);
  557. }
  558. #else
  559. int ret;
  560. if (S->stmt) {
  561. ret = mysql_stmt_fetch(S->stmt);
  562. # ifdef MYSQL_DATA_TRUNCATED
  563. if (ret == MYSQL_DATA_TRUNCATED) {
  564. ret = 0;
  565. }
  566. # endif
  567. if (ret) {
  568. if (ret != MYSQL_NO_DATA) {
  569. pdo_mysql_error_stmt(stmt);
  570. }
  571. PDO_DBG_RETURN(0);
  572. }
  573. PDO_DBG_RETURN(1);
  574. }
  575. #endif /* PDO_USE_MYSQLND */
  576. if (!S->result) {
  577. strcpy(stmt->error_code, "HY000");
  578. PDO_DBG_RETURN(0);
  579. }
  580. #if PDO_USE_MYSQLND
  581. if (!S->stmt && S->current_data) {
  582. mnd_free(S->current_data);
  583. }
  584. #endif /* PDO_USE_MYSQLND */
  585. if ((S->current_data = mysql_fetch_row(S->result)) == NULL) {
  586. #if PDO_USE_MYSQLND
  587. if (S->result->unbuf && !S->result->unbuf->eof_reached && mysql_errno(S->H->server)) {
  588. #else
  589. if (!S->result->eof && mysql_errno(S->H->server)) {
  590. #endif
  591. pdo_mysql_error_stmt(stmt);
  592. }
  593. PDO_DBG_RETURN(0);
  594. }
  595. S->current_lengths = mysql_fetch_lengths(S->result);
  596. PDO_DBG_RETURN(1);
  597. }
  598. /* }}} */
  599. static int pdo_mysql_stmt_describe(pdo_stmt_t *stmt, int colno) /* {{{ */
  600. {
  601. pdo_mysql_stmt *S = (pdo_mysql_stmt*)stmt->driver_data;
  602. struct pdo_column_data *cols = stmt->columns;
  603. int i;
  604. PDO_DBG_ENTER("pdo_mysql_stmt_describe");
  605. PDO_DBG_INF_FMT("stmt=%p", S->stmt);
  606. if (!S->result) {
  607. PDO_DBG_RETURN(0);
  608. }
  609. if (colno >= stmt->column_count) {
  610. /* error invalid column */
  611. PDO_DBG_RETURN(0);
  612. }
  613. /* fetch all on demand, this seems easiest
  614. ** if we've been here before bail out
  615. */
  616. if (cols[0].name) {
  617. PDO_DBG_RETURN(1);
  618. }
  619. for (i = 0; i < stmt->column_count; i++) {
  620. if (S->H->fetch_table_names) {
  621. cols[i].name = strpprintf(0, "%s.%s", S->fields[i].table, S->fields[i].name);
  622. } else {
  623. cols[i].name = zend_string_init(S->fields[i].name, S->fields[i].name_length, 0);
  624. }
  625. cols[i].precision = S->fields[i].decimals;
  626. cols[i].maxlen = S->fields[i].length;
  627. #ifdef PDO_USE_MYSQLND
  628. if (S->stmt) {
  629. cols[i].param_type = PDO_PARAM_ZVAL;
  630. } else
  631. #endif
  632. {
  633. cols[i].param_type = PDO_PARAM_STR;
  634. }
  635. }
  636. PDO_DBG_RETURN(1);
  637. }
  638. /* }}} */
  639. static int pdo_mysql_stmt_get_col(pdo_stmt_t *stmt, int colno, char **ptr, size_t *len, int *caller_frees) /* {{{ */
  640. {
  641. pdo_mysql_stmt *S = (pdo_mysql_stmt*)stmt->driver_data;
  642. PDO_DBG_ENTER("pdo_mysql_stmt_get_col");
  643. PDO_DBG_INF_FMT("stmt=%p", S->stmt);
  644. if (!S->result) {
  645. PDO_DBG_RETURN(0);
  646. }
  647. /* With mysqlnd data is stored inside mysqlnd, not S->current_data */
  648. if (!S->stmt) {
  649. if (S->current_data == NULL || !S->result) {
  650. PDO_DBG_RETURN(0);
  651. }
  652. }
  653. if (colno >= stmt->column_count) {
  654. /* error invalid column */
  655. PDO_DBG_RETURN(0);
  656. }
  657. #if PDO_USE_MYSQLND
  658. if (S->stmt) {
  659. Z_TRY_ADDREF(S->stmt->data->result_bind[colno].zv);
  660. *ptr = (char*)&S->stmt->data->result_bind[colno].zv;
  661. *len = sizeof(zval);
  662. PDO_DBG_RETURN(1);
  663. }
  664. #else
  665. if (S->stmt) {
  666. if (S->out_null[colno]) {
  667. *ptr = NULL;
  668. *len = 0;
  669. PDO_DBG_RETURN(1);
  670. }
  671. *ptr = S->bound_result[colno].buffer;
  672. if (S->out_length[colno] > S->bound_result[colno].buffer_length) {
  673. /* mysql lied about the column width */
  674. strcpy(stmt->error_code, "01004"); /* truncated */
  675. S->out_length[colno] = S->bound_result[colno].buffer_length;
  676. *len = S->out_length[colno];
  677. PDO_DBG_RETURN(0);
  678. }
  679. *len = S->out_length[colno];
  680. PDO_DBG_RETURN(1);
  681. }
  682. #endif
  683. *ptr = S->current_data[colno];
  684. *len = S->current_lengths[colno];
  685. PDO_DBG_RETURN(1);
  686. } /* }}} */
  687. static char *type_to_name_native(int type) /* {{{ */
  688. {
  689. #define PDO_MYSQL_NATIVE_TYPE_NAME(x) case FIELD_TYPE_##x: return #x;
  690. switch (type) {
  691. PDO_MYSQL_NATIVE_TYPE_NAME(STRING)
  692. PDO_MYSQL_NATIVE_TYPE_NAME(VAR_STRING)
  693. #ifdef FIELD_TYPE_TINY
  694. PDO_MYSQL_NATIVE_TYPE_NAME(TINY)
  695. #endif
  696. #ifdef FIELD_TYPE_BIT
  697. PDO_MYSQL_NATIVE_TYPE_NAME(BIT)
  698. #endif
  699. PDO_MYSQL_NATIVE_TYPE_NAME(SHORT)
  700. PDO_MYSQL_NATIVE_TYPE_NAME(LONG)
  701. PDO_MYSQL_NATIVE_TYPE_NAME(LONGLONG)
  702. PDO_MYSQL_NATIVE_TYPE_NAME(INT24)
  703. PDO_MYSQL_NATIVE_TYPE_NAME(FLOAT)
  704. PDO_MYSQL_NATIVE_TYPE_NAME(DOUBLE)
  705. PDO_MYSQL_NATIVE_TYPE_NAME(DECIMAL)
  706. #ifdef FIELD_TYPE_NEWDECIMAL
  707. PDO_MYSQL_NATIVE_TYPE_NAME(NEWDECIMAL)
  708. #endif
  709. #ifdef FIELD_TYPE_GEOMETRY
  710. PDO_MYSQL_NATIVE_TYPE_NAME(GEOMETRY)
  711. #endif
  712. PDO_MYSQL_NATIVE_TYPE_NAME(TIMESTAMP)
  713. #ifdef FIELD_TYPE_YEAR
  714. PDO_MYSQL_NATIVE_TYPE_NAME(YEAR)
  715. #endif
  716. PDO_MYSQL_NATIVE_TYPE_NAME(SET)
  717. PDO_MYSQL_NATIVE_TYPE_NAME(ENUM)
  718. PDO_MYSQL_NATIVE_TYPE_NAME(DATE)
  719. #ifdef FIELD_TYPE_NEWDATE
  720. PDO_MYSQL_NATIVE_TYPE_NAME(NEWDATE)
  721. #endif
  722. PDO_MYSQL_NATIVE_TYPE_NAME(TIME)
  723. PDO_MYSQL_NATIVE_TYPE_NAME(DATETIME)
  724. PDO_MYSQL_NATIVE_TYPE_NAME(TINY_BLOB)
  725. PDO_MYSQL_NATIVE_TYPE_NAME(MEDIUM_BLOB)
  726. PDO_MYSQL_NATIVE_TYPE_NAME(LONG_BLOB)
  727. PDO_MYSQL_NATIVE_TYPE_NAME(BLOB)
  728. PDO_MYSQL_NATIVE_TYPE_NAME(NULL)
  729. default:
  730. return NULL;
  731. }
  732. #undef PDO_MYSQL_NATIVE_TYPE_NAME
  733. } /* }}} */
  734. static int pdo_mysql_stmt_col_meta(pdo_stmt_t *stmt, zend_long colno, zval *return_value) /* {{{ */
  735. {
  736. pdo_mysql_stmt *S = (pdo_mysql_stmt*)stmt->driver_data;
  737. const MYSQL_FIELD *F;
  738. zval flags;
  739. char *str;
  740. PDO_DBG_ENTER("pdo_mysql_stmt_col_meta");
  741. PDO_DBG_INF_FMT("stmt=%p", S->stmt);
  742. if (!S->result) {
  743. PDO_DBG_RETURN(FAILURE);
  744. }
  745. if (colno >= stmt->column_count) {
  746. /* error invalid column */
  747. PDO_DBG_RETURN(FAILURE);
  748. }
  749. array_init(return_value);
  750. array_init(&flags);
  751. F = S->fields + colno;
  752. if (F->def) {
  753. add_assoc_string(return_value, "mysql:def", F->def);
  754. }
  755. if (IS_NOT_NULL(F->flags)) {
  756. add_next_index_string(&flags, "not_null");
  757. }
  758. if (IS_PRI_KEY(F->flags)) {
  759. add_next_index_string(&flags, "primary_key");
  760. }
  761. if (F->flags & MULTIPLE_KEY_FLAG) {
  762. add_next_index_string(&flags, "multiple_key");
  763. }
  764. if (F->flags & UNIQUE_KEY_FLAG) {
  765. add_next_index_string(&flags, "unique_key");
  766. }
  767. if (IS_BLOB(F->flags)) {
  768. add_next_index_string(&flags, "blob");
  769. }
  770. str = type_to_name_native(F->type);
  771. if (str) {
  772. add_assoc_string(return_value, "native_type", str);
  773. }
  774. #ifdef PDO_USE_MYSQLND
  775. switch (F->type) {
  776. case MYSQL_TYPE_BIT:
  777. case MYSQL_TYPE_YEAR:
  778. case MYSQL_TYPE_TINY:
  779. case MYSQL_TYPE_SHORT:
  780. case MYSQL_TYPE_INT24:
  781. case MYSQL_TYPE_LONG:
  782. #if SIZEOF_ZEND_LONG==8
  783. case MYSQL_TYPE_LONGLONG:
  784. #endif
  785. add_assoc_long(return_value, "pdo_type", PDO_PARAM_INT);
  786. break;
  787. default:
  788. add_assoc_long(return_value, "pdo_type", PDO_PARAM_STR);
  789. break;
  790. }
  791. #endif
  792. add_assoc_zval(return_value, "flags", &flags);
  793. add_assoc_string(return_value, "table", (char *) (F->table?F->table : ""));
  794. PDO_DBG_RETURN(SUCCESS);
  795. } /* }}} */
  796. static int pdo_mysql_stmt_cursor_closer(pdo_stmt_t *stmt) /* {{{ */
  797. {
  798. pdo_mysql_stmt *S = (pdo_mysql_stmt*)stmt->driver_data;
  799. PDO_DBG_ENTER("pdo_mysql_stmt_cursor_closer");
  800. PDO_DBG_INF_FMT("stmt=%p", S->stmt);
  801. if (S->result) {
  802. mysql_free_result(S->result);
  803. S->result = NULL;
  804. }
  805. if (S->stmt) {
  806. int retval;
  807. retval = mysql_stmt_free_result(S->stmt);
  808. PDO_DBG_RETURN(retval ? 0 : 1);
  809. }
  810. while (mysql_more_results(S->H->server)) {
  811. MYSQL_RES *res;
  812. if (mysql_next_result(S->H->server) != 0) {
  813. break;
  814. }
  815. res = mysql_store_result(S->H->server);
  816. if (res) {
  817. mysql_free_result(res);
  818. }
  819. }
  820. PDO_DBG_RETURN(1);
  821. }
  822. /* }}} */
  823. const struct pdo_stmt_methods mysql_stmt_methods = {
  824. pdo_mysql_stmt_dtor,
  825. pdo_mysql_stmt_execute,
  826. pdo_mysql_stmt_fetch,
  827. pdo_mysql_stmt_describe,
  828. pdo_mysql_stmt_get_col,
  829. pdo_mysql_stmt_param_hook,
  830. NULL, /* set_attr */
  831. NULL, /* get_attr */
  832. pdo_mysql_stmt_col_meta,
  833. pdo_mysql_stmt_next_rowset,
  834. pdo_mysql_stmt_cursor_closer
  835. };
  836. /*
  837. * Local variables:
  838. * tab-width: 4
  839. * c-basic-offset: 4
  840. * End:
  841. * vim600: noet sw=4 ts=4 fdm=marker
  842. * vim<600: noet sw=4 ts=4
  843. */