]> granicus.if.org Git - php/commitdiff
fix pdo_mysql crash with empty bound stream param
authorAnatol Belski <ab@php.net>
Thu, 5 Nov 2015 20:32:31 +0000 (21:32 +0100)
committerAnatol Belski <ab@php.net>
Thu, 5 Nov 2015 20:32:31 +0000 (21:32 +0100)
rel #70862

ext/pdo_mysql/mysql_statement.c
ext/pdo_mysql/tests/bug70862.phpt [new file with mode: 0644]

index f55843072f222f57b6d194a7708c72102e19f90c..7c6d3272fd338cfedeb137a5df1d78238df310fd 100644 (file)
@@ -523,10 +523,11 @@ static int pdo_mysql_stmt_param_hook(pdo_stmt_t *stmt, struct pdo_bound_param_da
                                                        parameter = Z_REFVAL(param->parameter);
                                                }
                                                if (Z_TYPE_P(parameter) == IS_RESOURCE) {
-                                                       php_stream *stm;
+                                                       php_stream *stm = NULL;
                                                        php_stream_from_zval_no_verify(stm, parameter);
                                                        if (stm) {
-                                                               ZVAL_STR(parameter, php_stream_copy_to_mem(stm, PHP_STREAM_COPY_ALL, 0));
+                                                               zend_string *mem = php_stream_copy_to_mem(stm, PHP_STREAM_COPY_ALL, 0);
+                                                               ZVAL_STR(parameter, mem ? mem : STR_EMPTY_ALLOC());
                                                        } else {
                                                                pdo_raise_impl_error(stmt->dbh, stmt, "HY105", "Expected a stream resource");
                                                                return 0;
diff --git a/ext/pdo_mysql/tests/bug70862.phpt b/ext/pdo_mysql/tests/bug70862.phpt
new file mode 100644 (file)
index 0000000..4a3a618
--- /dev/null
@@ -0,0 +1,46 @@
+--TEST--
+MySQL Prepared Statements and BLOBs
+--SKIPIF--
+<?php
+require_once(dirname(__FILE__) . DIRECTORY_SEPARATOR . 'skipif.inc');
+require_once(dirname(__FILE__) . DIRECTORY_SEPARATOR . 'mysql_pdo_test.inc');
+MySQLPDOTest::skip();
+?>
+--FILE--
+<?php
+       require_once(dirname(__FILE__) . DIRECTORY_SEPARATOR . 'mysql_pdo_test.inc');
+       $db = MySQLPDOTest::factory();
+
+       $db->exec('DROP TABLE IF EXISTS test');
+       $db->exec(sprintf('CREATE TABLE test(id INT, label BLOB)'));
+
+       $db->setAttribute(PDO::ATTR_EMULATE_PREPARES, 0);
+       $db->setAttribute(PDO::ATTR_STRINGIFY_FETCHES, true);
+
+       class HelloWrapper {
+               public function stream_open() { return true; }
+               public function stream_eof() { return true; }
+               public function stream_read() { return NULL; }
+               public function stream_stat() { return array(); }
+       }
+       stream_wrapper_register("hello", "HelloWrapper");
+
+       $f = fopen("hello://there", "r");
+
+       $stmt = $db->prepare('INSERT INTO test(id, label) VALUES (1, :para)');
+       $stmt->bindParam(":para", $f, PDO::PARAM_LOB);
+       $stmt->execute();
+
+       var_dump($f);
+
+       print "done!";
+?>
+--CLEAN--
+<?php
+require dirname(__FILE__) . '/mysql_pdo_test.inc';
+$db = MySQLPDOTest::factory();
+$db->exec('DROP TABLE IF EXISTS test');
+?>
+--EXPECTF--
+string(0) ""
+done!