Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

'zend_execute_data' pointer becomes NULL inside internal function handler of 'curl_exec()' #18216

Closed
sjayexec opened this issue Apr 1, 2025 · 4 comments

Comments

@sjayexec
Copy link

sjayexec commented Apr 1, 2025

Description

Description

We have a PHP extension that overwrites the function handler for curl_exec() in the function table CG(function_table) and during some testing we found that the execute_data pointer passed to the function handler becomes NULL midway, before the function handler finishes execution.

Our custom function handler wrapper_curl_exec() roughly does this:

void ZEND_FASTCALL wrapper_curl_exec(zend_execute_data *execute_data, zval *return_value)
{
     // access execute_data -- pointer is valid
     // call original function handler of curl_exec()
     // access execute_data  -- pointer is NULL, our extension crashes without a NULL check
     // return
}

This does not happen always, but it is frequent enough on PHP 8.4.5 with our particular test setup and test case.

Test Setup

OS: Windows 11
PHP 8.4.5
PHP-CGI SAPI
Ngnix with 5 php-cgi.exe processes as a pool

Test Case
We are running curl -v http://localhost/main.php in a loop with 0.5 secs delay from powershell CLI.
main.php sends a downstream HTTP request via curl_exec() and downstream.php sends another downstream request via curl_exec(). final.php just returns some dummy json data as a response. (See the reference scripts the end).

Observation
This curl loop from command line works for some time before seemingly taking too long for each request. At some point, we see our extension crashing because execute_data pointer became NULL after the original curl_exec() function returned (inside our extension) for one of the curl_exec() calls.

A related observation is that even with our extension disabled, this curl loop starts taking too long after a while i.e., the curl_exec() in main.php and downstream.php take a long time. Sometimes nginx throws 'Bad GatewayorPage not available` errors.

This issue also occurs with PHP 8.3 and PHP 8.2 but far less frequently.

Questions

  1. Why does execute_data pointer become NULL before the function handler returns?
  2. Are there any valid/predictable scenarios where we should expect execute_data pointer to be NULL or become null in a function handler? This will allow us to decide if we have to pre-emptively keep a NULL check for execute_data in all our custom function handlers.

Let me know if you need any further details.

Reference
(relevant snippets)

# file: main.php
<?php

$ch = curl_init('http://localhost/downstream.php');
curl_setopt($ch, CURLOPT_RETURNTRANSFER, true);
curl_setopt($ch, CURLOPT_HTTPHEADER, array_map(function($key, $value) {
    return "$key: $value";
}, array_keys($phpinfo_req_headers), $phpinfo_req_headers));
$response = curl_exec($ch);
curl_close($ch);
# file: downstream.php
<?php

$ch = curl_init('http://localhost/final.php');
curl_setopt($ch, CURLOPT_RETURNTRANSFER, true);
curl_setopt($ch, CURLOPT_HTTPHEADER, array_map(function($key, $value) {
    return "$key: $value";
}, array_keys($phpinfo_req_headers), $phpinfo_req_headers));
$response = curl_exec($ch);
curl_close($ch);

PHP Version

PHP 8.4.5, PHP 8.3.17, PHP 8.2.27

Operating System

Windows 11

@DanielEScherzer
Copy link
Member

We have a PHP extension

Can you please share the extension code?

OS: Windows 11

Have you tested this on any other operating systems?

@sjayexec
Copy link
Author

sjayexec commented Apr 1, 2025

Unfortunately, I can't share the extension code, but here's is exactly how we overwrite function handler for curl_exec():

void ZEND_FASTCALL wrapper_curl_exec(zend_execute_data *execute_data, zval *return_value);

// get original function object of curl_exec()
orig_func = zend_hash_str_find_ptr(CG(function_table), "curl_exec", 9);

// backup the pointer to original function handler
zif_handler orig_curl_exec_handler = orig_func->internal_function.handler;

// store our function's address in the function handler pointer for curl_exec() in CG(function_table)
orig_func->internal_function.handler = wrapper_curl_exec;


void ZEND_FASTCALL wrapper_curl_exec(zend_execute_data *execute_data, zval *return_value)
{
     // read access execute_data -- pointer is valid
	 
     // call origin curl_exec() function handler
     orig_curl_exec_handler(execute_data, return_value);	
     
     // read access execute_data  -- pointer is NULL, our extension crashes without a NULL check
     // return
}

And we only read from the execute_data pointer, we don't alter it in any way.

Have you tested this on any other operating systems?

Yes, this code has always worked before with Ubuntu and Alpine - so it is a stable implementation. We also overwrite function handlers for numerous other functions & methods in the same way. On Windows though, we see this issue sporadically in the mentioned test setup and only for curl_exec() handler. And it is clear for us that when curl_exec() seems to take a long time for no reason from the php scripts I shared, it is at this time we encounter this issue.

@iluuu1994
Copy link
Member

Your example looks suspicious. Unless you share a pointer to execute_data, it should be impossible for orig_curl_exec_handler() to change it. Hence, this indicates stack corruption. Have you tried running your tests with Valgrind or ASan?

@sjayexec
Copy link
Author

sjayexec commented Apr 4, 2025

@iluuu1994 Yeah, we cross checked some stuff and this is some issue arising from our code. You can close this issue, thanks!

@sjayexec sjayexec closed this as completed Apr 4, 2025
@TimWolla TimWolla closed this as not planned Won't fix, can't repro, duplicate, stale Apr 4, 2025
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Projects
None yet
Development

No branches or pull requests

4 participants