From:             
Operating system: Linux
PHP version:      Irrelevant
Package:          PDO related
Bug Type:         Bug
Bug description:PDO's mishandling of peristence, leading to a segfault

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 bug report at http://bugs.php.net/bug.php?id=53567&edit=1
-- 
Try a snapshot (PHP 5.2):            
http://bugs.php.net/fix.php?id=53567&r=trysnapshot52
Try a snapshot (PHP 5.3):            
http://bugs.php.net/fix.php?id=53567&r=trysnapshot53
Try a snapshot (trunk):              
http://bugs.php.net/fix.php?id=53567&r=trysnapshottrunk
Fixed in SVN:                        
http://bugs.php.net/fix.php?id=53567&r=fixed
Fixed in SVN and need be documented: 
http://bugs.php.net/fix.php?id=53567&r=needdocs
Fixed in release:                    
http://bugs.php.net/fix.php?id=53567&r=alreadyfixed
Need backtrace:                      
http://bugs.php.net/fix.php?id=53567&r=needtrace
Need Reproduce Script:               
http://bugs.php.net/fix.php?id=53567&r=needscript
Try newer version:                   
http://bugs.php.net/fix.php?id=53567&r=oldversion
Not developer issue:                 
http://bugs.php.net/fix.php?id=53567&r=support
Expected behavior:                   
http://bugs.php.net/fix.php?id=53567&r=notwrong
Not enough info:                     
http://bugs.php.net/fix.php?id=53567&r=notenoughinfo
Submitted twice:                     
http://bugs.php.net/fix.php?id=53567&r=submittedtwice
register_globals:                    
http://bugs.php.net/fix.php?id=53567&r=globals
PHP 4 support discontinued:          http://bugs.php.net/fix.php?id=53567&r=php4
Daylight Savings:                    http://bugs.php.net/fix.php?id=53567&r=dst
IIS Stability:                       
http://bugs.php.net/fix.php?id=53567&r=isapi
Install GNU Sed:                     
http://bugs.php.net/fix.php?id=53567&r=gnused
Floating point limitations:          
http://bugs.php.net/fix.php?id=53567&r=float
No Zend Extensions:                  
http://bugs.php.net/fix.php?id=53567&r=nozend
MySQL Configuration Error:           
http://bugs.php.net/fix.php?id=53567&r=mysqlcfg

Reply via email to