php-internal-docs 8.4.8
Unofficial docs for php/php-src
Loading...
Searching...
No Matches
mysql_driver.c
Go to the documentation of this file.
1/*
2 +----------------------------------------------------------------------+
3 | Copyright (c) The PHP Group |
4 +----------------------------------------------------------------------+
5 | This source file is subject to version 3.01 of the PHP license, |
6 | that is bundled with this package in the file LICENSE, and is |
7 | available through the world-wide-web at the following url: |
8 | https://www.php.net/license/3_01.txt |
9 | If you did not receive a copy of the PHP license and are unable to |
10 | obtain it through the world-wide-web, please send a note to |
11 | license@php.net so we can mail you a copy immediately. |
12 +----------------------------------------------------------------------+
13 | Author: George Schlossnagle <george@omniti.com> |
14 | Wez Furlong <wez@php.net> |
15 | Johannes Schlueter <johannes@mysql.com> |
16 +----------------------------------------------------------------------+
17*/
18
19#ifdef HAVE_CONFIG_H
20#include <config.h>
21#endif
22
23#include "php.h"
24#include "php_ini.h"
25#include "ext/standard/info.h"
26#include "ext/pdo/php_pdo.h"
28#include "php_pdo_mysql.h"
29#include "php_pdo_mysql_int.h"
30#ifndef PDO_USE_MYSQLND
31#include <mysqld_error.h>
32#endif
33#include "zend_exceptions.h"
34
35#ifdef PDO_USE_MYSQLND
36# define pdo_mysql_init(persistent) mysqlnd_init(MYSQLND_CLIENT_NO_FLAG, persistent)
37#else
38# define pdo_mysql_init(persistent) mysql_init(NULL)
39#endif
40
41/* {{{ _pdo_mysql_error */
42int _pdo_mysql_error(pdo_dbh_t *dbh, pdo_stmt_t *stmt, const char *file, int line)
43{
45 pdo_error_type *pdo_err;
48
49 PDO_DBG_ENTER("_pdo_mysql_error");
50 PDO_DBG_INF_FMT("file=%s line=%d", file, line);
51 if (stmt) {
53 pdo_err = &stmt->error_code;
54 einfo = &S->einfo;
55 } else {
56 pdo_err = &dbh->error_code;
57 einfo = &H->einfo;
58 }
59
60 if (S && S->stmt) {
61 einfo->errcode = mysql_stmt_errno(S->stmt);
62 } else {
63 einfo->errcode = mysql_errno(H->server);
64 }
65
66 einfo->file = file;
67 einfo->line = line;
68
69 if (einfo->errmsg) {
70 pefree(einfo->errmsg, dbh->is_persistent);
71 einfo->errmsg = NULL;
72 }
73
74 if (einfo->errcode) {
75 if (einfo->errcode == 2014) {
76 if (mysql_more_results(H->server)) {
77 einfo->errmsg = pestrdup(
78 "Cannot execute queries while there are pending result sets. "
79 "Consider unsetting the previous PDOStatement or calling "
80 "PDOStatement::closeCursor()",
81 dbh->is_persistent);
82 } else {
83 einfo->errmsg = pestrdup(
84 "Cannot execute queries while other unbuffered queries are active. "
85 "Consider using PDOStatement::fetchAll(). Alternatively, if your code "
86 "is only ever going to run against mysql, you may enable query "
87 "buffering by setting the PDO::MYSQL_ATTR_USE_BUFFERED_QUERY attribute.",
88 dbh->is_persistent);
89 }
90 } else if (einfo->errcode == 2057) {
91 einfo->errmsg = pestrdup(
92 "A stored procedure returning result sets of different size was called. "
93 "This is not supported by libmysql",
94 dbh->is_persistent);
95
96 } else {
97 if (S && S->stmt) {
98 einfo->errmsg = pestrdup(mysql_stmt_error(S->stmt), dbh->is_persistent);
99 } else {
100 einfo->errmsg = pestrdup(mysql_error(H->server), dbh->is_persistent);
101 }
102 }
103 } else { /* no error */
104 strcpy(*pdo_err, PDO_ERR_NONE);
106 }
107
108 if (S && S->stmt) {
109 strcpy(*pdo_err, mysql_stmt_sqlstate(S->stmt));
110 } else {
111 strcpy(*pdo_err, mysql_sqlstate(H->server));
112 }
113
114 if (!dbh->methods) {
115 PDO_DBG_INF("Throwing exception");
116 pdo_throw_exception(einfo->errcode, einfo->errmsg, pdo_err);
117 }
118
119 PDO_DBG_RETURN(einfo->errcode);
120}
121/* }}} */
122
123/* {{{ pdo_mysql_fetch_error_func */
124static void pdo_mysql_fetch_error_func(pdo_dbh_t *dbh, pdo_stmt_t *stmt, zval *info)
125{
127 pdo_mysql_error_info *einfo = &H->einfo;
128
129 PDO_DBG_ENTER("pdo_mysql_fetch_error_func");
130 PDO_DBG_INF_FMT("dbh=%p stmt=%p", dbh, stmt);
131 if (stmt) {
132 pdo_mysql_stmt *S = (pdo_mysql_stmt*)stmt->driver_data;
133 einfo = &S->einfo;
134 } else {
135 einfo = &H->einfo;
136 }
137
138 if (einfo->errcode) {
139 add_next_index_long(info, einfo->errcode);
140 add_next_index_string(info, einfo->errmsg);
141 }
142
144}
145/* }}} */
146
147/* {{{ mysql_handle_closer */
148static void mysql_handle_closer(pdo_dbh_t *dbh)
149{
151
152 PDO_DBG_ENTER("mysql_handle_closer");
153 PDO_DBG_INF_FMT("dbh=%p", dbh);
154 if (H) {
155 if (H->server) {
156 mysql_close(H->server);
157 H->server = NULL;
158 }
159 if (H->einfo.errmsg) {
160 pefree(H->einfo.errmsg, dbh->is_persistent);
161 H->einfo.errmsg = NULL;
162 }
163 pefree(H, dbh->is_persistent);
164 dbh->driver_data = NULL;
165 }
166}
167/* }}} */
168
169/* {{{ mysql_handle_preparer */
170static bool mysql_handle_preparer(pdo_dbh_t *dbh, zend_string *sql, pdo_stmt_t *stmt, zval *driver_options)
171{
173 pdo_mysql_stmt *S = ecalloc(1, sizeof(pdo_mysql_stmt));
174 zend_string *nsql = NULL;
175 int ret;
176 int server_version;
177
178 PDO_DBG_ENTER("mysql_handle_preparer");
179 PDO_DBG_INF_FMT("dbh=%p", dbh);
180 PDO_DBG_INF_FMT("sql=%.*s", (int) ZSTR_LEN(sql), ZSTR_VAL(sql));
181
182 S->H = H;
183 stmt->driver_data = S;
184 stmt->methods = &mysql_stmt_methods;
185
186 if (H->emulate_prepare) {
187 goto end;
188 }
189
190 server_version = mysql_get_server_version(H->server);
191 if (server_version < 40100) {
192 goto fallback;
193 }
194 stmt->supports_placeholders = PDO_PLACEHOLDER_POSITIONAL;
195 ret = pdo_parse_params(stmt, sql, &nsql);
196
197 if (ret == 1) {
198 /* query was rewritten */
199 sql = nsql;
200 } else if (ret == -1) {
201 /* failed to parse */
202 strcpy(dbh->error_code, stmt->error_code);
203 PDO_DBG_RETURN(false);
204 }
205
206 if (!(S->stmt = mysql_stmt_init(H->server))) {
207 pdo_mysql_error(dbh);
208 if (nsql) {
209 zend_string_release(nsql);
210 }
211 PDO_DBG_RETURN(false);
212 }
213
214 if (mysql_stmt_prepare(S->stmt, ZSTR_VAL(sql), ZSTR_LEN(sql))) {
215 if (nsql) {
216 zend_string_release(nsql);
217 }
218 /* TODO: might need to pull statement specific info here? */
219 /* if the query isn't supported by the protocol, fallback to emulation */
220 if (mysql_errno(H->server) == 1295) {
221 mysql_stmt_close(S->stmt);
222 S->stmt = NULL;
223 goto fallback;
224 }
225 pdo_mysql_error(dbh);
226 PDO_DBG_RETURN(false);
227 }
228 if (nsql) {
229 zend_string_release(nsql);
230 }
231
232 S->num_params = mysql_stmt_param_count(S->stmt);
233
234 if (S->num_params) {
235#ifdef PDO_USE_MYSQLND
236 S->params = NULL;
237#else
238 S->params = ecalloc(S->num_params, sizeof(MYSQL_BIND));
239 S->in_null = ecalloc(S->num_params, sizeof(my_bool));
240 S->in_length = ecalloc(S->num_params, sizeof(zend_ulong));
241#endif
242 }
243 dbh->alloc_own_columns = 1;
244
245 S->max_length = pdo_attr_lval(driver_options, PDO_ATTR_MAX_COLUMN_LEN, 0);
246
247 PDO_DBG_RETURN(true);
248
249fallback:
250end:
251 stmt->supports_placeholders = PDO_PLACEHOLDER_NONE;
252
253 PDO_DBG_RETURN(true);
254}
255/* }}} */
256
257/* {{{ mysql_handle_doer */
258static zend_long mysql_handle_doer(pdo_dbh_t *dbh, const zend_string *sql)
259{
261 PDO_DBG_ENTER("mysql_handle_doer");
262 PDO_DBG_INF_FMT("dbh=%p", dbh);
263 PDO_DBG_INF_FMT("sql=%.*s", (int)ZSTR_LEN(sql), ZSTR_VAL(sql));
264
265 if (mysql_real_query(H->server, ZSTR_VAL(sql), ZSTR_LEN(sql))) {
266 pdo_mysql_error(dbh);
267 PDO_DBG_RETURN(-1);
268 } else {
270 if (c == (my_ulonglong) -1) {
271 pdo_mysql_error(dbh);
272 PDO_DBG_RETURN(H->einfo.errcode ? -1 : 0);
273 } else {
274
275 /* MULTI_QUERY support - eat up all unfetched result sets */
277 while (mysql_more_results(H->server)) {
278 if (mysql_next_result(H->server)) {
279 pdo_mysql_error(dbh);
280 PDO_DBG_RETURN(-1);
281 }
282 result = mysql_store_result(H->server);
283 if (result) {
285 }
286 }
287 PDO_DBG_RETURN((int)c);
288 }
289 }
290}
291/* }}} */
292
293/* {{{ pdo_mysql_last_insert_id */
294static zend_string *pdo_mysql_last_insert_id(pdo_dbh_t *dbh, const zend_string *name)
295{
297 PDO_DBG_ENTER("pdo_mysql_last_insert_id");
299}
300/* }}} */
301
302#if defined(PDO_USE_MYSQLND) || MYSQL_VERSION_ID < 50707 || defined(MARIADB_BASE_VERSION)
303# define mysql_real_escape_string_quote(mysql, to, from, length, quote) \
304 mysql_real_escape_string(mysql, to, from, length)
305#endif
306
307/* {{{ mysql_handle_quoter */
308static zend_string* mysql_handle_quoter(pdo_dbh_t *dbh, const zend_string *unquoted, enum pdo_param_type paramtype )
309{
311 bool use_national_character_set = 0;
312 bool use_binary = 0;
313 size_t quotedlen;
314
315 if ((paramtype & PDO_PARAM_LOB) == PDO_PARAM_LOB) {
316 use_binary = 1;
317 } else {
318 if (H->assume_national_character_set_strings) {
319 use_national_character_set = 1;
320 }
321 if ((paramtype & PDO_PARAM_STR_NATL) == PDO_PARAM_STR_NATL) {
322 use_national_character_set = 1;
323 }
324 if ((paramtype & PDO_PARAM_STR_CHAR) == PDO_PARAM_STR_CHAR) {
325 use_national_character_set = 0;
326 }
327 }
328
329 PDO_DBG_ENTER("mysql_handle_quoter");
330 PDO_DBG_INF_FMT("dbh=%p", dbh);
331 PDO_DBG_INF_FMT("unquoted=%.*s", (int)ZSTR_LEN(unquoted), ZSTR_VAL(unquoted));
332
333 zend_string *quoted_str = zend_string_safe_alloc(2, ZSTR_LEN(unquoted),
334 3 + (use_national_character_set ? 1 : 0) + (use_binary ? 7 : 0), false);
335 char *quoted = ZSTR_VAL(quoted_str);
336
337 if (use_national_character_set) {
338 quotedlen = mysql_real_escape_string_quote(H->server, quoted + 2, ZSTR_VAL(unquoted), ZSTR_LEN(unquoted), '\'');
339 quoted[0] = 'N';
340 quoted[1] = '\'';
341
342 ++quotedlen; /* N prefix */
343 } else if (use_binary) {
344 quotedlen = mysql_real_escape_string_quote(H->server, quoted + 8, ZSTR_VAL(unquoted), ZSTR_LEN(unquoted), '\'');
345 memcpy(quoted, "_binary'", 8);
346
347 quotedlen += 7; /* _binary prefix */
348 } else {
349 quotedlen = mysql_real_escape_string_quote(H->server, quoted + 1, ZSTR_VAL(unquoted), ZSTR_LEN(unquoted), '\'');
350 quoted[0] = '\'';
351 }
352
353 quoted[++quotedlen] = '\'';
354 quoted[++quotedlen] = '\0';
355 PDO_DBG_INF_FMT("quoted=%.*s", (int)quotedlen, quoted);
356
357 quoted_str = zend_string_truncate(quoted_str, quotedlen, false);
358
359 PDO_DBG_RETURN(quoted_str);
360}
361/* }}} */
362
363/* {{{ mysql_handle_begin */
364static bool mysql_handle_begin(pdo_dbh_t *dbh)
365{
367 zend_string *command;
368
369 PDO_DBG_ENTER("mysql_handle_begin");
370 PDO_DBG_INF_FMT("dbh=%p", dbh);
371
372 command = ZSTR_INIT_LITERAL("START TRANSACTION", 0);
373 return_value = mysql_handle_doer(dbh, command);
374 zend_string_release_ex(command, 0);
376}
377/* }}} */
378
379/* {{{ mysql_handle_commit */
380static bool mysql_handle_commit(pdo_dbh_t *dbh)
381{
382 PDO_DBG_ENTER("mysql_handle_commit");
383 PDO_DBG_INF_FMT("dbh=%p", dbh);
384 if (mysql_commit(((pdo_mysql_db_handle *)dbh->driver_data)->server)) {
385 pdo_mysql_error(dbh);
386 PDO_DBG_RETURN(false);
387 }
388 PDO_DBG_RETURN(true);
389}
390/* }}} */
391
392/* {{{ mysql_handle_rollback */
393static bool mysql_handle_rollback(pdo_dbh_t *dbh)
394{
395 PDO_DBG_ENTER("mysql_handle_rollback");
396 PDO_DBG_INF_FMT("dbh=%p", dbh);
397 if (mysql_rollback(((pdo_mysql_db_handle *)dbh->driver_data)->server)) {
398 pdo_mysql_error(dbh);
399 PDO_DBG_RETURN(false);
400 }
401 PDO_DBG_RETURN(true);
402}
403/* }}} */
404
405/* {{{ mysql_handle_autocommit */
406static inline int mysql_handle_autocommit(pdo_dbh_t *dbh)
407{
408 PDO_DBG_ENTER("mysql_handle_autocommit");
409 PDO_DBG_INF_FMT("dbh=%p", dbh);
410 PDO_DBG_INF_FMT("dbh->autocommit=%d", dbh->auto_commit);
411 if (mysql_autocommit(((pdo_mysql_db_handle *)dbh->driver_data)->server, dbh->auto_commit)) {
412 pdo_mysql_error(dbh);
414 }
416}
417/* }}} */
418
419/* {{{ pdo_mysql_set_attribute */
420static bool pdo_mysql_set_attribute(pdo_dbh_t *dbh, zend_long attr, zval *val)
421{
422 zend_long lval;
423 bool bval;
424 PDO_DBG_ENTER("pdo_mysql_set_attribute");
425 PDO_DBG_INF_FMT("dbh=%p", dbh);
426 PDO_DBG_INF_FMT("attr=" ZEND_LONG_FMT, attr);
427
428 switch (attr) {
430 if (!pdo_get_bool_param(&bval, val)) {
431 PDO_DBG_RETURN(false);
432 }
433 /* ignore if the new value equals the old one */
434 if (dbh->auto_commit ^ bval) {
435 dbh->auto_commit = bval;
436 if (!mysql_handle_autocommit(dbh)) {
437 PDO_DBG_RETURN(false);
438 }
439 }
440 PDO_DBG_RETURN(true);
441
443 if (!pdo_get_long_param(&lval, val)) {
444 PDO_DBG_RETURN(false);
445 }
446 ((pdo_mysql_db_handle *)dbh->driver_data)->assume_national_character_set_strings = lval == PDO_PARAM_STR_NATL;
447 PDO_DBG_RETURN(true);
448
450 if (!pdo_get_bool_param(&bval, val)) {
451 PDO_DBG_RETURN(false);
452 }
453 /* ignore if the new value equals the old one */
454 ((pdo_mysql_db_handle *)dbh->driver_data)->buffered = bval;
455 PDO_DBG_RETURN(true);
456
459 if (!pdo_get_bool_param(&bval, val)) {
460 PDO_DBG_RETURN(false);
461 }
462 /* ignore if the new value equals the old one */
463 ((pdo_mysql_db_handle *)dbh->driver_data)->emulate_prepare = bval;
464 PDO_DBG_RETURN(true);
465
467 if (!pdo_get_bool_param(&bval, val)) {
468 PDO_DBG_RETURN(false);
469 }
470 ((pdo_mysql_db_handle *)dbh->driver_data)->fetch_table_names = bval;
471 PDO_DBG_RETURN(true);
472
473#ifdef PDO_USE_MYSQLND
475 if (!pdo_get_bool_param(&bval, val)) {
476 PDO_DBG_RETURN(false);
477 }
478 unsigned int int_and_float_native = !bval;
480 if (mysql_options(H->server, MYSQLND_OPT_INT_AND_FLOAT_NATIVE, (const char *) &int_and_float_native)) {
481 pdo_mysql_error(dbh);
482 PDO_DBG_RETURN(false);
483 }
484 PDO_DBG_RETURN(true);
485#else
487 if (!pdo_get_long_param(&lval, val)) {
488 PDO_DBG_RETURN(false);
489 }
490 if (lval < 0) {
491 /* TODO: Johannes, can we throw a warning here? */
492 ((pdo_mysql_db_handle *)dbh->driver_data)->max_buffer_size = 1024*1024;
493 PDO_DBG_INF_FMT("Adjusting invalid buffer size to =%l", ((pdo_mysql_db_handle *)dbh->driver_data)->max_buffer_size);
494 } else {
495 ((pdo_mysql_db_handle *)dbh->driver_data)->max_buffer_size = lval;
496 }
497 PDO_DBG_RETURN(true);
498 break;
499#endif
500
501 default:
502 PDO_DBG_RETURN(false);
503 }
504}
505/* }}} */
506
507/* {{{ pdo_mysql_get_attribute */
508static int pdo_mysql_get_attribute(pdo_dbh_t *dbh, zend_long attr, zval *return_value)
509{
511
512 PDO_DBG_ENTER("pdo_mysql_get_attribute");
513 PDO_DBG_INF_FMT("dbh=%p", dbh);
514 PDO_DBG_INF_FMT("attr=" ZEND_LONG_FMT, attr);
515 switch (attr) {
518 break;
519
522 break;
523
526 break;
528#ifdef PDO_USE_MYSQLND
529 zend_string *tmp;
530
531 if (mysqlnd_stat(H->server, &tmp) == PASS) {
533#else
534 char *tmp;
535 if ((tmp = (char *)mysql_stat(H->server))) {
537#endif
538 } else {
539 pdo_mysql_error(dbh);
540 PDO_DBG_RETURN(-1);
541 }
542 }
543 break;
544
547 break;
548
550 ZVAL_LONG(return_value, H->assume_national_character_set_strings ? PDO_PARAM_STR_NATL : PDO_PARAM_STR_CHAR);
551 break;
552
554 ZVAL_BOOL(return_value, H->buffered);
555 break;
556
559 ZVAL_BOOL(return_value, H->emulate_prepare);
560 break;
561
562#ifndef PDO_USE_MYSQLND
564 ZVAL_LONG(return_value, H->max_buffer_size);
565 break;
566#endif
567
569 ZVAL_BOOL(return_value, H->local_infile);
570 break;
571
572#if (MYSQL_VERSION_ID >= 80021 && !defined(MARIADB_BASE_VERSION)) || defined(PDO_USE_MYSQLND)
573 case PDO_MYSQL_ATTR_LOCAL_INFILE_DIRECTORY:
574 {
575 const char* local_infile_directory = NULL;
576#ifdef PDO_USE_MYSQLND
577 local_infile_directory = H->server->data->options->local_infile_directory;
578#else
579 mysql_get_option(H->server, MYSQL_OPT_LOAD_DATA_LOCAL_DIR, &local_infile_directory);
580#endif
583 } else {
585 }
586 break;
587 }
588#endif
589
591 ZVAL_BOOL(return_value, H->fetch_table_names);
592 break;
593
594 default:
596 }
597
599}
600/* }}} */
601
602/* {{{ pdo_mysql_check_liveness */
603static zend_result pdo_mysql_check_liveness(pdo_dbh_t *dbh)
604{
606
607 PDO_DBG_ENTER("pdo_mysql_check_liveness");
608 PDO_DBG_INF_FMT("dbh=%p", dbh);
609
610 if (mysql_ping(H->server)) {
612 }
614}
615/* }}} */
616
617/* {{{ pdo_mysql_request_shutdown */
618static void pdo_mysql_request_shutdown(pdo_dbh_t *dbh)
619{
620 PDO_DBG_ENTER("pdo_mysql_request_shutdown");
621 PDO_DBG_INF_FMT("dbh=%p", dbh);
622
623#ifdef PDO_USE_MYSQLND
625 if (H->server) {
626 mysqlnd_end_psession(H->server);
627 }
628#endif
629}
630/* }}} */
631
632#ifdef PDO_USE_MYSQLND
633# define pdo_mysql_get_server_status(m) mysqlnd_get_server_status(m)
634#else
635# define pdo_mysql_get_server_status(m) (m)->server_status
636#endif
637
638/* {{{ pdo_mysql_in_transaction */
639static bool pdo_mysql_in_transaction(pdo_dbh_t *dbh)
640{
642 PDO_DBG_ENTER("pdo_mysql_in_transaction");
644}
645/* }}} */
646
647/* {{{ mysql_methods */
648static const struct pdo_dbh_methods mysql_methods = {
649 mysql_handle_closer,
650 mysql_handle_preparer,
651 mysql_handle_doer,
652 mysql_handle_quoter,
653 mysql_handle_begin,
654 mysql_handle_commit,
655 mysql_handle_rollback,
656 pdo_mysql_set_attribute,
657 pdo_mysql_last_insert_id,
658 pdo_mysql_fetch_error_func,
659 pdo_mysql_get_attribute,
660 pdo_mysql_check_liveness,
661 NULL,
662 pdo_mysql_request_shutdown,
663 pdo_mysql_in_transaction,
664 NULL, /* get_gc */
666};
667/* }}} */
668
669#ifdef PHP_WIN32
670# define PDO_DEFAULT_MYSQL_UNIX_ADDR NULL
671#else
672# define PDO_DEFAULT_MYSQL_UNIX_ADDR PDO_MYSQL_G(default_socket)
673#endif
674
675/* {{{ pdo_mysql_handle_factory */
676static int pdo_mysql_handle_factory(pdo_dbh_t *dbh, zval *driver_options)
677{
679 size_t i;
680 int ret = 0;
681 char *host = NULL, *unix_socket = NULL;
682 unsigned int port = 3306;
683 char *dbname;
684 struct pdo_data_src_parser vars[] = {
685 { "charset", NULL, 0 },
686 { "dbname", "", 0 },
687 { "host", "localhost", 0 },
688 { "port", "3306", 0 },
689 { "unix_socket", PDO_DEFAULT_MYSQL_UNIX_ADDR, 0 },
690 { "user", NULL, 0 },
691 { "password", NULL, 0 },
692 };
693 int connect_opts = 0
694#ifdef CLIENT_MULTI_RESULTS
696#endif
697 ;
698#ifdef PDO_USE_MYSQLND
699 size_t dbname_len = 0;
700 size_t password_len = 0;
701#endif
702
703#ifdef CLIENT_MULTI_STATEMENTS
704 if (!driver_options) {
705 connect_opts |= CLIENT_MULTI_STATEMENTS;
706 } else if (pdo_attr_lval(driver_options, PDO_MYSQL_ATTR_MULTI_STATEMENTS, 1)) {
707 connect_opts |= CLIENT_MULTI_STATEMENTS;
708 }
709#endif
710
711 PDO_DBG_ENTER("pdo_mysql_handle_factory");
712 PDO_DBG_INF_FMT("dbh=%p", dbh);
713#ifdef CLIENT_MULTI_RESULTS
714 PDO_DBG_INF("multi results");
715#endif
716
718
719 H = pecalloc(1, sizeof(pdo_mysql_db_handle), dbh->is_persistent);
720
721 H->einfo.errcode = 0;
722 H->einfo.errmsg = NULL;
723
724 /* allocate an environment */
725
726 /* handle for the server */
727 if (!(H->server = pdo_mysql_init(dbh->is_persistent))) {
728 pdo_mysql_error(dbh);
729 goto cleanup;
730 }
731#ifdef PDO_USE_MYSQLND
732 if (dbh->is_persistent) {
734 }
735#endif
736
737 dbh->driver_data = H;
738
739 dbh->skip_param_evt =
740 1 << PDO_PARAM_EVT_FREE |
745
746#ifndef PDO_USE_MYSQLND
747 H->max_buffer_size = 1024*1024;
748#endif
749
750 H->assume_national_character_set_strings = 0;
751 H->buffered = H->emulate_prepare = 1;
752
753 /* handle MySQL options */
754 if (driver_options) {
755 zend_long connect_timeout = pdo_attr_lval(driver_options, PDO_ATTR_TIMEOUT, 30);
756 zend_string *init_cmd = NULL;
757#ifndef PDO_USE_MYSQLND
758 zend_string *default_file = NULL, *default_group = NULL;
759#endif
761 zend_string *ssl_key = NULL, *ssl_cert = NULL, *ssl_ca = NULL, *ssl_capath = NULL, *ssl_cipher = NULL;
762 H->buffered = pdo_attr_lval(driver_options, PDO_MYSQL_ATTR_USE_BUFFERED_QUERY, 1);
763
764 H->emulate_prepare = pdo_attr_lval(driver_options,
765 PDO_MYSQL_ATTR_DIRECT_QUERY, H->emulate_prepare);
766 H->emulate_prepare = pdo_attr_lval(driver_options,
767 PDO_ATTR_EMULATE_PREPARES, H->emulate_prepare);
768
769 H->assume_national_character_set_strings = pdo_attr_lval(driver_options,
771
772#ifndef PDO_USE_MYSQLND
773 H->max_buffer_size = pdo_attr_lval(driver_options, PDO_MYSQL_ATTR_MAX_BUFFER_SIZE, H->max_buffer_size);
774#endif
775
776 if (pdo_attr_lval(driver_options, PDO_MYSQL_ATTR_FOUND_ROWS, 0)) {
777 connect_opts |= CLIENT_FOUND_ROWS;
778 }
779
780 if (pdo_attr_lval(driver_options, PDO_MYSQL_ATTR_IGNORE_SPACE, 0)) {
781 connect_opts |= CLIENT_IGNORE_SPACE;
782 }
783
784 if (mysql_options(H->server, MYSQL_OPT_CONNECT_TIMEOUT, (const char *)&connect_timeout)) {
785 pdo_mysql_error(dbh);
786 goto cleanup;
787 }
788
789 if (pdo_attr_lval(driver_options, PDO_MYSQL_ATTR_LOCAL_INFILE, 0)) {
790 H->local_infile = 1;
791#ifndef PDO_USE_MYSQLND
792 if (PG(open_basedir) && PG(open_basedir)[0] != '\0') {
793 H->local_infile = 0;
794 }
795#endif
796 }
797
798#if (MYSQL_VERSION_ID >= 80021 && !defined(MARIADB_BASE_VERSION)) || defined(PDO_USE_MYSQLND)
799 zend_string *local_infile_directory = pdo_attr_strval(driver_options, PDO_MYSQL_ATTR_LOCAL_INFILE_DIRECTORY, NULL);
802 zend_string_release(local_infile_directory);
803 pdo_mysql_error(dbh);
804 goto cleanup;
805 }
806 zend_string_release(local_infile_directory);
807 }
808#endif
809#ifdef MYSQL_OPT_RECONNECT
810 /* since 5.0.3, the default for this option is 0 if not specified.
811 * we want the old behaviour
812 * mysqlnd doesn't support reconnect, thus we don't have "|| defined(PDO_USE_MYSQLND)"
813 */
814 {
815 zend_long reconnect = 1;
816 mysql_options(H->server, MYSQL_OPT_RECONNECT, (const char*)&reconnect);
817 }
818#endif
819 init_cmd = pdo_attr_strval(driver_options, PDO_MYSQL_ATTR_INIT_COMMAND, NULL);
820 if (init_cmd) {
821 if (mysql_options(H->server, MYSQL_INIT_COMMAND, (const char *)ZSTR_VAL(init_cmd))) {
822 zend_string_release_ex(init_cmd, 0);
823 pdo_mysql_error(dbh);
824 goto cleanup;
825 }
826 zend_string_release_ex(init_cmd, 0);
827 }
828#ifndef PDO_USE_MYSQLND
829 default_file = pdo_attr_strval(driver_options, PDO_MYSQL_ATTR_READ_DEFAULT_FILE, NULL);
830 if (default_file) {
831 if (mysql_options(H->server, MYSQL_READ_DEFAULT_FILE, (const char *)ZSTR_VAL(default_file))) {
832 zend_string_release_ex(default_file, 0);
833 pdo_mysql_error(dbh);
834 goto cleanup;
835 }
836 zend_string_release_ex(default_file, 0);
837 }
838
839 default_group = pdo_attr_strval(driver_options, PDO_MYSQL_ATTR_READ_DEFAULT_GROUP, NULL);
840 if (default_group) {
841 if (mysql_options(H->server, MYSQL_READ_DEFAULT_GROUP, (const char *)ZSTR_VAL(default_group))) {
842 zend_string_release_ex(default_group, 0);
843 pdo_mysql_error(dbh);
844 goto cleanup;
845 }
846 zend_string_release_ex(default_group, 0);
847 }
848#endif
849 compress = pdo_attr_lval(driver_options, PDO_MYSQL_ATTR_COMPRESS, 0);
850 if (compress) {
851 if (mysql_options(H->server, MYSQL_OPT_COMPRESS, 0)) {
852 pdo_mysql_error(dbh);
853 goto cleanup;
854 }
855 }
856
857 ssl_key = pdo_attr_strval(driver_options, PDO_MYSQL_ATTR_SSL_KEY, NULL);
858 ssl_cert = pdo_attr_strval(driver_options, PDO_MYSQL_ATTR_SSL_CERT, NULL);
859 ssl_ca = pdo_attr_strval(driver_options, PDO_MYSQL_ATTR_SSL_CA, NULL);
860 ssl_capath = pdo_attr_strval(driver_options, PDO_MYSQL_ATTR_SSL_CAPATH, NULL);
861 ssl_cipher = pdo_attr_strval(driver_options, PDO_MYSQL_ATTR_SSL_CIPHER, NULL);
862
863 if (ssl_key || ssl_cert || ssl_ca || ssl_capath || ssl_cipher) {
864 mysql_ssl_set(H->server,
865 ssl_key? ZSTR_VAL(ssl_key) : NULL,
866 ssl_cert? ZSTR_VAL(ssl_cert) : NULL,
867 ssl_ca? ZSTR_VAL(ssl_ca) : NULL,
868 ssl_capath? ZSTR_VAL(ssl_capath) : NULL,
869 ssl_cipher? ZSTR_VAL(ssl_cipher) : NULL);
870 if (ssl_key) {
871 zend_string_release_ex(ssl_key, 0);
872 }
873 if (ssl_cert) {
874 zend_string_release_ex(ssl_cert, 0);
875 }
876 if (ssl_ca) {
877 zend_string_release_ex(ssl_ca, 0);
878 }
879 if (ssl_capath) {
880 zend_string_release_ex(ssl_capath, 0);
881 }
882 if (ssl_cipher) {
883 zend_string_release_ex(ssl_cipher, 0);
884 }
885 }
886
887#if MYSQL_VERSION_ID > 50605 || defined(PDO_USE_MYSQLND)
888 {
889 zend_string *public_key = pdo_attr_strval(driver_options, PDO_MYSQL_ATTR_SERVER_PUBLIC_KEY, NULL);
890 if (public_key) {
891 if (mysql_options(H->server, MYSQL_SERVER_PUBLIC_KEY, ZSTR_VAL(public_key))) {
892 pdo_mysql_error(dbh);
893 zend_string_release_ex(public_key, 0);
894 goto cleanup;
895 }
896 zend_string_release_ex(public_key, 0);
897 }
898 }
899#endif
900
901#ifdef PDO_USE_MYSQLND
902 {
903 zend_long ssl_verify_cert = pdo_attr_lval(driver_options,
904 PDO_MYSQL_ATTR_SSL_VERIFY_SERVER_CERT, -1);
905 if (ssl_verify_cert != -1) {
906 connect_opts |= ssl_verify_cert ?
909 }
910 }
911#endif
912 }
913
914 /* Always explicitly set the LOCAL_INFILE option. */
915 unsigned int local_infile = H->local_infile;
916 if (mysql_options(H->server, MYSQL_OPT_LOCAL_INFILE, (const char *)&local_infile)) {
917 pdo_mysql_error(dbh);
918 goto cleanup;
919 }
920
921#ifdef PDO_USE_MYSQLND
922 unsigned int int_and_float_native = !pdo_attr_lval(driver_options, PDO_ATTR_STRINGIFY_FETCHES, dbh->stringify);
923 if (mysql_options(H->server, MYSQLND_OPT_INT_AND_FLOAT_NATIVE, (const char *) &int_and_float_native)) {
924 pdo_mysql_error(dbh);
925 goto cleanup;
926 }
927#endif
928
929 if (vars[0].optval && mysql_options(H->server, MYSQL_SET_CHARSET_NAME, vars[0].optval)) {
930 pdo_mysql_error(dbh);
931 goto cleanup;
932 }
933
934 dbname = vars[1].optval;
935 host = vars[2].optval;
936 if(vars[3].optval) {
937 port = atoi(vars[3].optval);
938 }
939
940#ifdef PHP_WIN32
941 if (vars[2].optval && !strcmp(".", vars[2].optval)) {
942#else
943 if (vars[2].optval && !strcmp("localhost", vars[2].optval)) {
944#endif
945 unix_socket = vars[4].optval;
946 }
947
948 if (!dbh->username && vars[5].optval) {
949 dbh->username = pestrdup(vars[5].optval, dbh->is_persistent);
950 }
951
952 if (!dbh->password && vars[6].optval) {
953 dbh->password = pestrdup(vars[6].optval, dbh->is_persistent);
954 }
955
956 /* TODO: - Check zval cache + ZTS */
957#ifdef PDO_USE_MYSQLND
958 if (dbname) {
959 dbname_len = strlen(dbname);
960 }
961
962 if (dbh->password) {
963 password_len = strlen(dbh->password);
964 }
965
966 if (mysqlnd_connect(H->server, host, dbh->username, dbh->password, password_len, dbname, dbname_len,
967 port, unix_socket, connect_opts, MYSQLND_CLIENT_NO_FLAG) == NULL) {
968#else
969 if (mysql_real_connect(H->server, host, dbh->username, dbh->password, dbname, port, unix_socket, connect_opts) == NULL) {
970#endif
971 pdo_mysql_error(dbh);
972 goto cleanup;
973 }
974
975 if (!dbh->auto_commit) {
976 mysql_handle_autocommit(dbh);
977 }
978
979 H->attached = 1;
980
981 dbh->alloc_own_columns = 1;
983 dbh->methods = &mysql_methods;
984
985 ret = 1;
986
987cleanup:
988 for (i = 0; i < sizeof(vars)/sizeof(vars[0]); i++) {
989 if (vars[i].freeme) {
990 efree(vars[i].optval);
991 }
992 }
993
994 dbh->methods = &mysql_methods;
995
997}
998/* }}} */
999
1001 PDO_DRIVER_HEADER(mysql),
1002 pdo_mysql_handle_factory
1003};
file(string $filename, int $flags=0, $context=null)
memcpy(ptr1, ptr2, size)
new_type attr
Definition ffi.c:4364
zval * val
Definition ffi.c:4262
PHPAPI int php_check_open_basedir(const char *path)
#define NULL
Definition gdcache.h:45
#define S(s, l, r)
Definition hash_gost.c:121
#define PASS(tables)
Definition hash_gost.c:193
#define SUCCESS
Definition hash_sha3.c:261
#define compress(passes)
Definition hash_tiger.c:83
#define H(x, y, z)
Definition md5.c:146
int _pdo_mysql_error(pdo_dbh_t *dbh, pdo_stmt_t *stmt, const char *file, int line)
#define pdo_mysql_get_server_status(m)
const pdo_driver_t pdo_mysql_driver
#define mysql_real_escape_string_quote(mysql, to, from, length, quote)
#define pdo_mysql_init(persistent)
#define PDO_DEFAULT_MYSQL_UNIX_ADDR
const struct pdo_stmt_methods mysql_stmt_methods
#define mysqlnd_restart_psession(conn)
Definition mysqlnd.h:71
#define mysqlnd_end_psession(conn)
Definition mysqlnd.h:72
#define mysqlnd_connect(conn, host, user, pass, pass_len, db, db_len, port, socket, mysql_flags, client_api_flags)
Definition mysqlnd.h:82
#define mysqlnd_stat(conn, msg)
Definition mysqlnd.h:200
#define CLIENT_SSL_DONT_VERIFY_SERVER_CERT
@ MYSQL_OPT_CONNECT_TIMEOUT
@ MYSQL_OPT_COMPRESS
@ MYSQL_OPT_LOAD_DATA_LOCAL_DIR
@ MYSQL_OPT_LOCAL_INFILE
@ MYSQL_SET_CHARSET_NAME
@ MYSQLND_OPT_INT_AND_FLOAT_NATIVE
@ MYSQL_SERVER_PUBLIC_KEY
@ MYSQL_READ_DEFAULT_GROUP
@ MYSQL_READ_DEFAULT_FILE
@ MYSQL_INIT_COMMAND
@ MYSQL_OPT_RECONNECT
#define CLIENT_MULTI_RESULTS
#define CLIENT_FOUND_ROWS
#define CLIENT_SSL_VERIFY_SERVER_CERT
#define CLIENT_MULTI_STATEMENTS
#define CLIENT_IGNORE_SPACE
#define SERVER_STATUS_IN_TRANS
#define MYSQLND_CLIENT_NO_FLAG
#define MYSQL_RES
#define mysql_stmt_param_count(s)
#define mysql_autocommit(r, m)
#define my_ulonglong
#define mysql_free_result(r)
#define mysql_ping(r)
#define mysql_insert_id(r)
#define mysql_stmt_error(s)
#define mysql_sqlstate(r)
#define my_bool
#define mysql_stmt_errno(s)
#define mysql_store_result(r)
#define mysql_more_results(r)
#define mysql_affected_rows(r)
#define mysql_ssl_set(c, key, cert, ca, capath, cipher)
#define mysql_stmt_close(s)
#define mysql_real_query(r, a, b)
#define mysql_options(c, a, v)
#define mysql_stmt_sqlstate(s)
#define mysql_error(r)
#define mysql_get_host_info(r)
#define mysql_close(r)
#define mysql_errno(r)
#define mysql_stmt_prepare(s, q, l)
#define mysql_rollback(r)
#define mysql_get_client_info()
#define mysql_stmt_init(r)
#define mysql_next_result(r)
#define mysql_get_server_info(r)
#define mysql_commit(r)
#define mysql_get_server_version(r)
PDO_API int php_pdo_parse_data_source(const char *data_source, zend_ulong data_source_len, struct pdo_data_src_parser *parsed, int nparams)
Definition pdo.c:157
PDO_API bool pdo_get_long_param(zend_long *lval, zval *value)
Definition pdo_dbh.c:774
PDO_API bool pdo_get_bool_param(bool *bval, zval *value)
Definition pdo_dbh.c:792
void pdo_throw_exception(unsigned int driver_errcode, char *driver_errmsg, pdo_error_type *pdo_error)
Definition pdo_dbh.c:41
int line
Definition php_ffi.h:54
unsigned const char * end
Definition php_ffi.h:51
#define PG(v)
Definition php_globals.h:31
char * local_infile_directory
PDO_API int pdo_parse_params(pdo_stmt_t *stmt, zend_string *inquery, zend_string **outquery)
struct _pdo_dbh_t pdo_dbh_t
#define PDO_ERR_NONE
#define PDO_DRIVER_HEADER(name)
struct _pdo_stmt_t pdo_stmt_t
pdo_param_type
@ PDO_PARAM_LOB
@ PDO_PARAM_STR_NATL
@ PDO_PARAM_STR_CHAR
@ PDO_PARAM_EVT_FETCH_POST
@ PDO_PARAM_EVT_NORMALIZE
@ PDO_PARAM_EVT_FETCH_PRE
@ PDO_PARAM_EVT_EXEC_POST
@ PDO_PARAM_EVT_FREE
char pdo_error_type[6]
@ PDO_PLACEHOLDER_NONE
@ PDO_PLACEHOLDER_POSITIONAL
@ PDO_ATTR_EMULATE_PREPARES
@ PDO_ATTR_DEFAULT_STR_PARAM
@ PDO_ATTR_MAX_COLUMN_LEN
@ PDO_ATTR_CONNECTION_STATUS
@ PDO_ATTR_SERVER_VERSION
@ PDO_ATTR_SERVER_INFO
@ PDO_ATTR_AUTOCOMMIT
@ PDO_ATTR_STRINGIFY_FETCHES
@ PDO_ATTR_FETCH_TABLE_NAMES
@ PDO_ATTR_TIMEOUT
@ PDO_ATTR_CLIENT_VERSION
#define PDO_DBG_RETURN(value)
int pdo_mysql_scanner(pdo_scanner_t *s)
#define PDO_DBG_VOID_RETURN
#define pdo_mysql_error(s)
@ PDO_MYSQL_ATTR_SSL_CIPHER
@ PDO_MYSQL_ATTR_READ_DEFAULT_FILE
@ PDO_MYSQL_ATTR_SSL_CERT
@ PDO_MYSQL_ATTR_FOUND_ROWS
@ PDO_MYSQL_ATTR_DIRECT_QUERY
@ PDO_MYSQL_ATTR_MAX_BUFFER_SIZE
@ PDO_MYSQL_ATTR_READ_DEFAULT_GROUP
@ PDO_MYSQL_ATTR_COMPRESS
@ PDO_MYSQL_ATTR_LOCAL_INFILE
@ PDO_MYSQL_ATTR_USE_BUFFERED_QUERY
@ PDO_MYSQL_ATTR_IGNORE_SPACE
@ PDO_MYSQL_ATTR_SSL_KEY
@ PDO_MYSQL_ATTR_INIT_COMMAND
@ PDO_MYSQL_ATTR_SSL_CAPATH
@ PDO_MYSQL_ATTR_SSL_CA
@ PDO_MYSQL_ATTR_MULTI_STATEMENTS
const char * data_source
unsigned max_escaped_char_length
unsigned is_persistent
size_t data_source_len
void * driver_data
unsigned stringify
pdo_error_type error_code
unsigned alloc_own_columns
unsigned auto_commit
const struct pdo_dbh_methods * methods
unsigned skip_param_evt
pdo_error_type error_code
ZEND_API zend_result add_next_index_long(zval *arg, zend_long n)
Definition zend_API.c:2132
ZEND_API zend_result add_next_index_string(zval *arg, const char *str)
Definition zend_API.c:2186
#define ZVAL_STRING(z, s)
Definition zend_API.h:956
#define pestrdup(s, persistent)
Definition zend_alloc.h:206
#define ecalloc(nmemb, size)
Definition zend_alloc.h:158
#define efree(ptr)
Definition zend_alloc.h:155
#define pefree(ptr, persistent)
Definition zend_alloc.h:191
#define pecalloc(nmemb, size, persistent)
Definition zend_alloc.h:200
struct _zval_struct zval
strlen(string $string)
strcmp(string $string1, string $string2)
zend_string_release_ex(func->internal_function.function_name, 0)
int32_t zend_long
Definition zend_long.h:42
uint32_t zend_ulong
Definition zend_long.h:43
#define ZEND_LONG_FMT
Definition zend_long.h:87
struct _zend_string zend_string
ZEND_API zend_string *ZEND_FASTCALL zend_u64_to_str(uint64_t num)
#define ZSTR_VAL(zstr)
Definition zend_string.h:68
#define ZSTR_INIT_LITERAL(s, persistent)
#define ZSTR_LEN(zstr)
Definition zend_string.h:69
#define ZVAL_STR(z, s)
#define ZVAL_NULL(z)
#define ZVAL_LONG(z, l)
@ FAILURE
Definition zend_types.h:61
ZEND_RESULT_CODE zend_result
Definition zend_types.h:64
#define ZVAL_BOOL(z, b)
zval * return_value
zend_string * name
bool result
zval * ret