我不能在线程内部发送任何数量的内存,而不会在https://github.com/php/php-src/blob/master/Zend/zend_alloc.c#L2409触发段错误。
只有在线程安全的PHP(ZTS)版本上运行时,才会出现(具有讽刺意味)段错误。
在纯NTS版本上运行时,一切正常。
这里有一些可以用来重现问题的代码(我使用php-cpp来简化扩展的创建)。
void* test(void* wrapper){
emalloc(sizeof(Php::Value));
return NULL;
}
void VoIP::__construct()
{
pthread_t a;
pthread_create(&a, NULL, test, this);
}
extern "C" {
PHPCPP_EXPORT void *get_module()
{
static Php::Extension extension("php-libtgvoip", "1.0");
Php::Class<VoIP> voip("VoIP");
voip.method<&VoIP::__construct>("__construct", Php::Public | Php::Final);
Php::Namespace danog("danog");
Php::Namespace MadelineProto("MadelineProto");
MadelineProto.add(std::move(voip));
danog.add(std::move(MadelineProto));
extension.add(std::move(danog));
return extension;
}
}
标题:
#include <php.h>
#include <php_ini.h>
#include <ext/standard/info.h>
#include <phpcpp.h>
class VoIP : public Php::Base {
public:
void __construct();
}
\ danog \ MadelineProto \ VoIP类的实例化会引发由test()中的emalloc引起的段错误:
Thread 2 "php" received signal SIGSEGV, Segmentation fault.
[Switching to Thread 0x7fffe91ff700 (LWP 4267)]
0x0000555555cfc7ed in _emalloc (size=32, __zend_filename=0x7fffed9c88a8 "main.cpp", __zend_lineno=30, __zend_orig_filename=0x0, __zend_orig_lineno=0)
at /root/php-src/Zend/zend_alloc.c:2409
2409 if (UNEXPECTED(AG(mm_heap)->use_custom_heap)) {
(gdb) backtrace
#0 0x0000555555cfc7ed in _emalloc (size=32, __zend_filename=0x7fffed9c88a8 "main.cpp", __zend_lineno=30, __zend_orig_filename=0x0, __zend_orig_lineno=0)
at /root/php-src/Zend/zend_alloc.c:2409
#1 0x00007fffed94e80e in test (wrapper=0x555556a73310) at main.cpp:30
#2 0x00007ffff572a494 in start_thread (arg=0x7fffe91ff700) at pthread_create.c:333
#3 0x00007ffff206eaff in clone () at ../sysdeps/unix/sysv/linux/x86_64/clone.S:97
(gdb)
完整的源代码可以在@ https://github.com/danog/php-libtgvoip
找到答案 0 :(得分:2)
我的回答可能相当明显,但不要使用emalloc
以及php线程之外的所有其他php / zend方法。无论ZTS
是打开还是关闭,您的代码都会失败。看一下应用程序崩溃的行:
if (UNEXPECTED(AG(mm_heap)->use_custom_heap))
AG被声明为
#ifdef ZTS
static int alloc_globals_id;
#define AG(v) ZEND_TSRMG(alloc_globals_id, zend_alloc_globals *, v)
#else
#define AG(v) (alloc_globals.v)
static zend_alloc_globals alloc_globals;
#endif
当ZTS打开时,会出现直接的段错误,因为调用线程未正确初始化为php线程使用TSRMG(线程安全资源管理器),当ZTS关闭时,将存在修改全局变量的竞争条件。