Edit report at http://bugs.php.net/bug.php?id=53567&edit=1
ID: 53567
User updated by: algernon at balabit dot hu
Reported by: algernon at balabit dot hu
Summary: PDO's mishandling of peristence, leading to a
segfault
Status: Open
Type: Bug
Package: PDO related
Operating System: Linux
PHP Version: Irrelevant
Block user comment: N
Private report: N
New Comment:
You're right - that's indeed a scenario I didn't consider.
I don't have the source code before me at the moment, but an option
would be to
free and clear ->query_stmt and ->query_stmt_zval before putting the DBH
object
into the cache. However, I have a feeling that would have a few unwanted
side
effects too.
Thinking further... would it be possible to decouple ->query_stmt and -
>query_stmt zval from the DBH object, and store them somewhere else? If
they
never reach the cache, then there is no problem.
Actually, if we just copy DBH before putting it into the cache, with
these two
fields blanked out, that might be enough, perhaps. Though, that might
lead to
another kind of memory leak - without being able to have a quick look,
I'm not
quite sure.
Previous Comments:
------------------------------------------------------------------------
[2010-12-18 17:26:36] [email protected]
I'm unsure about the attached patch.
If there's the exception and PHP bails out, zeroing the zval field is of
course correct. But what if the PDO error doesn't generate exceptions
(no PDO::ERRMODE_EXCEPTION) and you create another object? You'll be
zeroing the field from cache and leaking the object associated with the
zval because there's no call to zval_dtor, right?
Leaked objects are reclaimed on request shutdown, but another solution
would be preferable.
------------------------------------------------------------------------
[2010-12-17 22:06:46] algernon at balabit dot hu
Description:
------------
We observed a segmentation fault in PHP, which we tracked back to PDO,
and in
particular, the way it handles persistence. I will go into detail just a
tiny
bit later - we'll need a suitable environment to reproduce the bug
first.
For the record, we used the stock PHP5 packages that come with Ubuntu
Lucid, but
a quick look at SVN trunk as of yesterday suggests that the issue is
present
even in trunk. We did not test that, though.
We were using lighttpd with PHP over FastCGI: example configs can be
found all
around, there's nothing irregular in our setup. The key here, is FastCGI
(though, other scenarios where the zend_hash_* stuff is used might also
be
affected).
When PDO notices persistence is enabled, it will put the whole dbh
structure
into the persistent cache, near the end of the constructor. The next
time the
constructor is called, it will grab the dbh structure from the cache
as-is.
This would work, except that the dbh structure contains ->query_stmt and
-
>query_stmt_zval, which can potentially point to free'd memory: the last
statement is stored in these variables, but in case we issue an invalid
SQL
query, and get an uncaught exception back, PHP will bail out, without
calling
the PDO destructor, thus, ->query_stmt and ->query_stmt_zval will point
to
garbage.
This again wouldn't be a problem, if dbh wouldn't get reused. But
remember: it
has been stored in the cache, and the next time the PDO constructor gets
called,
it will grab dbh out of the cache, with two variables pointing to
garbage.
And then all hell breaks loose.
To reproduce the problem, the test script below can be used. For it to
work, one
will need any kind of database (the script uses postgres, but any driver
should
work) - the database can be empty, neither query requires any tables.
The script
will make two queries: a correct one, and a bad one. We need the bad one
to
trigger an exception.
Once the script is in place, I used the three terminals with a wget loop
each
(there might be easier ways to trigger the bug, but this ain't that bad,
either):
On the first terminal:
$ while true ; do wget -O /dev/null 'http://localhost/?sleep=10' ; sleep
2 ;
done
On the second and third:
$ while true ; do wget -O /dev/null 'http://localhost/?foo=bar' ; sleep
2 ; done
Within a couple of seconds, I started to get HTTP 500 errors and core
files.
The fix (see the patch attached), is to zero out ->query_stmt and -
>query_stmt_zval after retrieving a DBH object from the persistent
storage.
(Many thanks to my collegues Attila Magyar <[email protected]> & Balázs
Scheidler
<[email protected]> for doing the bulk of debugging & researching the
issue)
Test script:
---------------
<?php
class A
{
public function __construct() {
$this->pdo = new PDO('pgsql:dbname=sftest;', 'user', 's1krl+',
array(PDO::ATTR_PERSISTENT => true));
$this->pdo->setAttribute(PDO::ATTR_ERRMODE,
PDO::ERRMODE_EXCEPTION);
}
public function __destruct() {
unset($this->pdo);
}
public function trigger() {
$result = $this->pdo->query('SELECT 42 AS fortytwo;');
sleep ($_GET['sleep']);
$result = $this->pdo->query ('SELECT ICanHasFail();');
}
private $pdo;
}
$a = new A();
$a->trigger();
?>
Expected result:
----------------
The expected result would be anything but core files and segfaults.
Actual result:
--------------
Core was generated by `/usr/bin/php-cgi'.
Program terminated with signal 11, Segmentation fault.
#0 0x00000000006bb7a2 in zend_objects_store_del_ref_by_handle_ex
(handle=0,
handlers=0x2a66400) at /build/buildd/php5-5.3.2/Zend/zend_gc.h:190
190 root->prev->next = root->next;
(gdb) bt
#0 0x00000000006bb7a2 in zend_objects_store_del_ref_by_handle_ex
(handle=0,
handlers=0x2a66400) at /build/buildd/php5-5.3.2/Zend/zend_gc.h:190
#1 0x00000000006bb813 in zend_objects_store_del_ref (zobject=0x2a8d1e8)
at
/build/buildd/php5-5.3.2/Zend/zend_objects_API.c:172
#2 0x00007fe3bbc877b3 in pdo_dbh_attribute_set (dbh=0x2a8d140, attr=12,
value=0x2a68c90) at /build/buildd/php5-5.3.2/ext/pdo/pdo_dbh.c:825
#3 0x00007fe3bbc88275 in zim_PDO_dbh_constructor (ht=4,
return_value=0x2a66400,
return_value_ptr=0xd92340, this_ptr=0x7fffae890f68, return_value_used=0)
at
/build/buildd/php5-5.3.2/ext/pdo/pdo_dbh.c:409
#4 0x00000000006e719a in zend_do_fcall_common_helper_SPEC
(execute_data=0x2a96318) at
/build/buildd/php5-5.3.2/Zend/zend_vm_execute.h:313
#5 0x00000000006be480 in execute (op_array=0x2a68f20) at
/build/buildd/php5-
5.3.2/Zend/zend_vm_execute.h:104
#6 0x00000000006961ad in zend_execute_scripts (type=0,
retval=0x7fffae891390,
file_count=3) at /build/buildd/php5-5.3.2/Zend/zend.c:1266
#7 0x0000000000641df8 in php_execute_script (primary_file=0x2) at
/build/buildd/php5-5.3.2/main/main.c:2288
#8 0x0000000000723d44 in main (argc=32767, argv=0x0) at
/build/buildd/php5-
5.3.2/sapi/cgi/cgi_main.c:2110
(gdb) fr 3
#3 0x00007fe3bbc88275 in zim_PDO_dbh_constructor (ht=4,
return_value=0x2a66400,
return_value_ptr=0xd92340, this_ptr=0x7fffae890f68, return_value_used=0)
at
/build/buildd/php5-5.3.2/ext/pdo/pdo_dbh.c:409
409 && HASH_KEY_IS_LONG ==
zend_hash_get_current_key(Z_ARRVAL_P(options), &str_key, &long_key, 0))
{
(gdb) print dbh->query_stmt_zval
$2 = {value = {lval = 4, dval = 1.9762625833649862e-323, str = {val =
0x4
<Address 0x4 out of bounds>, len = -1142328224}, ht = 0x4, obj = {handle
= 4,
handlers = 0x7fe3bbe97460}}, refcount__gc = 2,
type = 5 '\005', is_ref__gc = 1 '\001'}
------------------------------------------------------------------------
--
Edit this bug report at http://bugs.php.net/bug.php?id=53567&edit=1