summaryrefslogtreecommitdiff
path: root/ext/pdo
diff options
context:
space:
mode:
authorPierrick Charron <pierrick@php.net>2009-12-14 03:44:33 +0000
committerPierrick Charron <pierrick@php.net>2009-12-14 03:44:33 +0000
commit7d4643193da984be757e64b6dc8010167c10e8df (patch)
tree6dc2142d7ed7fdda5bbda1af83278459dce06731 /ext/pdo
parent35ad3308499bdaa2afc028129d962b3a3c456d1f (diff)
downloadphp-git-7d4643193da984be757e64b6dc8010167c10e8df.tar.gz
- Fixed bug #50458 (PDO::FETCH_FUNC fails with Closures)
Diffstat (limited to 'ext/pdo')
-rwxr-xr-xext/pdo/pdo_stmt.c91
-rw-r--r--ext/pdo/tests/bug_50458.phpt29
2 files changed, 38 insertions, 82 deletions
diff --git a/ext/pdo/pdo_stmt.c b/ext/pdo/pdo_stmt.c
index a8886bdc62..8a5bcfde42 100755
--- a/ext/pdo/pdo_stmt.c
+++ b/ext/pdo/pdo_stmt.c
@@ -784,95 +784,20 @@ static int do_fetch_class_prepare(pdo_stmt_t *stmt TSRMLS_DC) /* {{{ */
static int make_callable_ex(pdo_stmt_t *stmt, zval *callable, zend_fcall_info * fci, zend_fcall_info_cache * fcc, int num_args TSRMLS_DC) /* {{{ */
{
- zval *object = NULL, **method = NULL;
- char *fname = NULL, *cname;
- zend_class_entry * ce = NULL, **pce;
- zend_function *function_handler;
-
- if (Z_TYPE_P(callable) == IS_ARRAY) {
- if (Z_ARRVAL_P(callable)->nNumOfElements < 2) {
- pdo_raise_impl_error(stmt->dbh, stmt, "HY000", "user-supplied function must be a valid callback" TSRMLS_CC);
- return 0;
- }
- object = *(zval**)Z_ARRVAL_P(callable)->pListHead->pData;
- method = (zval**)Z_ARRVAL_P(callable)->pListHead->pListNext->pData;
+ char *is_callable_error = NULL;
- if (Z_TYPE_P(object) == IS_STRING) { /* static call */
- if (zend_lookup_class(Z_STRVAL_P(object), Z_STRLEN_P(object), &pce TSRMLS_CC) == FAILURE) {
- pdo_raise_impl_error(stmt->dbh, stmt, "HY000", "user-supplied class does not exist" TSRMLS_CC);
- return 0;
- } else {
- ce = *pce;
- }
- object = NULL;
- } else if (Z_TYPE_P(object) == IS_OBJECT) { /* object call */
- ce = Z_OBJCE_P(object);
+ if (zend_fcall_info_init(callable, 0, fci, fcc, NULL, &is_callable_error TSRMLS_CC) == FAILURE) {
+ if (is_callable_error) {
+ pdo_raise_impl_error(stmt->dbh, stmt, "HY000", is_callable_error TSRMLS_CC);
+ efree(is_callable_error);
} else {
- pdo_raise_impl_error(stmt->dbh, stmt, "HY000", "user-supplied function must be a valid callback; bogus object/class name" TSRMLS_CC);
- return 0;
- }
-
- if (Z_TYPE_PP(method) != IS_STRING) {
- pdo_raise_impl_error(stmt->dbh, stmt, "HY000", "user-supplied function must be a valid callback; bogus method name" TSRMLS_CC);
- return 0;
- }
- } else if (Z_TYPE_P(callable) == IS_STRING) {
- method = &callable;
- }
-
- if (!method || !zend_is_callable(callable, 0, &fname TSRMLS_CC)) {
- pdo_raise_impl_error(stmt->dbh, stmt, "HY000", "user-supplied function must be a valid callback" TSRMLS_CC);
- if (fname) {
- efree(fname);
+ pdo_raise_impl_error(stmt->dbh, stmt, "HY000", "user-supplied function must be a valid callback" TSRMLS_CC);
}
return 0;
}
- /* ATM we do not support array($obj, "CLASS::FUNC") or "CLASS_FUNC" */
- cname = fname;
- if ((fname = strstr(fname, "::")) == NULL) {
- fname = cname;
- cname = NULL;
- } else {
- *fname = '\0';
- fname += 2;
- }
- if (cname) {
- if (zend_lookup_class(cname, strlen(cname), &pce TSRMLS_CC) == FAILURE) {
- pdo_raise_impl_error(stmt->dbh, stmt, "HY000", "user-supplied class does not exist" TSRMLS_CC);
- return 0;
- } else {
- if (ce) {
- /* pce must be base of ce or ce itself */
- if (ce != *pce && !instanceof_function(ce, *pce TSRMLS_CC)) {
- pdo_raise_impl_error(stmt->dbh, stmt, "HY000", "user-supplied class has bogus lineage" TSRMLS_CC);
- return 0;
- }
- }
- ce = *pce;
- }
- }
-
- zend_str_tolower_copy(fname, fname, strlen(fname));
- fci->function_table = ce ? &ce->function_table : EG(function_table);
- if (zend_hash_find(fci->function_table, fname, strlen(fname)+1, (void **)&function_handler) == FAILURE) {
- pdo_raise_impl_error(stmt->dbh, stmt, "HY000", "user-supplied function does not exist" TSRMLS_CC);
- return 0;
- }
- efree(cname ? cname : fname);
-
- fci->size = sizeof(zend_fcall_info);
- fci->function_name = NULL;
- fci->symbol_table = NULL;
fci->param_count = num_args; /* probably less */
fci->params = safe_emalloc(sizeof(zval**), num_args, 0);
- fci->object_ptr = object;
-
- fcc->initialized = 1;
- fcc->function_handler = function_handler;
- fcc->calling_scope = EG(scope);
- fcc->called_scope = object ? Z_OBJCE_P(object) : NULL;
- fcc->object_ptr = object;
return 1;
}
@@ -1568,7 +1493,9 @@ static PHP_METHOD(PDOStatement, fetchAll)
case 3:
case 2:
stmt->fetch.func.function = arg2;
- do_fetch_func_prepare(stmt TSRMLS_CC);
+ if (do_fetch_func_prepare(stmt TSRMLS_CC) == 0) {
+ error = 1;
+ }
break;
}
break;
diff --git a/ext/pdo/tests/bug_50458.phpt b/ext/pdo/tests/bug_50458.phpt
new file mode 100644
index 0000000000..3deb2892ee
--- /dev/null
+++ b/ext/pdo/tests/bug_50458.phpt
@@ -0,0 +1,29 @@
+--TEST--
+PDO Common: Bug #50458 (PDO::FETCH_FUNC fails with Closures)
+--SKIPIF--
+<?php # vim:ft=php
+if (!extension_loaded('pdo')) die('skip');
+$dir = getenv('REDIR_TEST_DIR');
+if (false == $dir) die('skip no driver');
+require_once $dir . 'pdo_test.inc';
+PDOTest::skip();
+?>
+--FILE--
+<?php
+if (getenv('REDIR_TEST_DIR') === false) putenv('REDIR_TEST_DIR='.dirname(__FILE__) . '/../../pdo/tests/');
+require_once getenv('REDIR_TEST_DIR') . 'pdo_test.inc';
+
+$db = PDOTest::factory();
+$db->exec("CREATE TABLE test (a VARCHAR(10))");
+$db->exec("INSERT INTO test (a) VALUES ('xyz')");
+$res = $db->query("SELECT a FROM test");
+var_dump($res->fetchAll(PDO::FETCH_FUNC, function($a) { return strtoupper($a); }));
+
+?>
+===DONE===
+--EXPECTF--
+array(1) {
+ [0]=>
+ string(3) "XYZ"
+}
+===DONE===