Этот вопрос лучше всего подойдет тому, кто является экспертом в Linux C ++ (и C) MAKE FILES, но также должен иметь представление о расширениях PHP и о том, как их создавать в Linux.
Я пишу многопоточное расширение PHP, относящееся к низкоуровневому TCP-сокету, в C ++ — для Linux (CentOS 6.5 64-bit). Как вы, вероятно, знаете, расширения PHP по существу «расширяют функциональность» пользовательского PHP-кода.
Я решил написать основные функциональные возможности и логику в стандартном C ++, используя стандартный make-файл C ++ (сгенерированный Eclipse CDT), и он создает стандартный исполняемый двоичный файл Linux с именем DC.a
Теперь мне нужно иметь возможность вызывать некоторые классы (и функции) из расширения PHP (и его экспортируемых функций), которое я создаю.
Написание PHP-расширения может быть немного сложнее, так как вам нужно собрать PHP из исходного кода, затем использовать «phpize», «./configure» и т. Д. Вместе с файлом config.m4.
Этот вызов phpize AUTO GENERATES MAKEFILE с целым набором настроек make-file, что ошеломляет меня.
Теперь мне нужно «перенести» мои стандартные классы (код) C ++ в каталог проекта расширения PHP и заставить их компилировать, собирать и связывать с расширением PHP, что, как вы знаете, создается как общий объект (.SO) (в Linux) — и .DLL (в Windows).
Эта проблема, по сути, заключается в том, чтобы ** изменить ** основной MAKEFILE, автоматически сгенерированный вызовом «phpize», чтобы иметь возможность компилировать и связывать дополнительный код C ++ ** (НЕ код C).
По сути, мне нужно добавить мясо (хранящееся в моем проекте на C ++) в скелетное расширение PHP, которое я уже разработал и работаю.
Любой, кто имеет знания в этой области — пожалуйста, не стесняйтесь мне помочь, и спасибо от всего сердца .., заранее. 🙂
Вот как мы это делаем:
Мы компилируем наш код C ++ в библиотеки (как вы уже упоминали), а затем связываем его. Кроме того, мы говорим phpize для активации компилятора C ++.
PHP_ARG_ENABLE(myextension, whether to enable myextension support,
[ --enable-myextension Enable myextension support])
if test "$PHP_MYEXTENSION" = "yes"; then
AC_DEFINE(HAVE_MYEXTENSION, 1, [Whether you have myExtension])
PHP_REQUIRE_CXX()
PHP_SUBST(MYEXTENSION_SHARED_LIBADD)
PHP_ADD_INCLUDE(../../../include)
PHP_ADD_INCLUDE(/usr/include)
PHP_ADD_INCLUDE(/opt/local/include)
PHP_ADD_LIBRARY(stdc++, 1, MYEXTENSION_SHARED_LIBADD)
PHP_ADD_LIBRARY_WITH_PATH(boost_thread-mt, /opt/local/lib, MYEXTENSION_SHARED_LIBADD)
PHP_ADD_LIBRARY_WITH_PATH(boost_system-mt, /opt/local/lib, MYEXTENSION_SHARED_LIBADD)
PHP_ADD_LIBRARY_WITH_PATH(customlib, /path/to/custom/lib, MYEXTENSION_SHARED_LIBADD)
AC_DEFINE(_ADDITIONAL_DEFINES_,1,[_ADDITIONAL_DEFINES_])
PHP_NEW_EXTENSION(MYEXTENSION, src/php.cpp, $ext_shared)
fi
В php.cpp мы должны убедиться, что что-то есть в C:
#ifdef HAVE_CONFIG_H
#include "../config.h"#endif
#ifdef WIN32
/* PHP Extension headers */
/* include zend win32 config first */
#include "zend_config.w32.h"#endif
extern "C" {
#include "php.h"#include "php_ini.h"
#ifdef ZTS
#include "TSRM.h"#endif
#include "zend_exceptions.h"
}
и мы также должны обернуть наш класс C ++:
zend_object_handlers myExtension_object_handlers;
struct myextension_object {
zend_object std;
YourNS::Extension::myExtension *extension;
};
void myExtension_free_storage(void *object TSRMLS_DC)
{
myextension_object *obj = (myextension_object *)object;
delete obj->myExtension;
zend_hash_destroy(obj->std.properties);
FREE_HASHTABLE(obj->std.properties);
efree(obj);
}
zend_object_value myExtension_create_handler(zend_class_entry *type TSRMLS_DC)
{
zval *tmp;
zend_object_value retval;
myextension_object *obj = (myextension_object *)emalloc(sizeof(myextension_object ));
memset(obj, 0, sizeof(myextension_object ));
obj->std.ce = type;
ALLOC_HASHTABLE(obj->std.properties);
zend_hash_init(obj->std.properties, 0, NULL, ZVAL_PTR_DTOR, 0);
#if PHP_VERSION_ID < 50399
zend_hash_copy(obj->std.properties, &type->default_properties,
(copy_ctor_func_t)zval_add_ref, (void *)&tmp, sizeof(zval *));
#else
object_properties_init(&obj->std, type);
#endif
retval.handle = zend_objects_store_put(obj, NULL,
myExtension_free_storage, NULL TSRMLS_CC);
retval.handlers = &myExtension_object_handlers;
return retval;
}
Позже вы можете использовать все это довольно удобно:
PHP_METHOD(myExtension, __construct)
{
YourNS::Extension::myExtension *myExt = NULL;
zval *object = getThis();
//if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "l", &maxGear) == FAILURE) {
// RETURN_NULL();
//}
myExt = new YourNS::Extension::myExtension();
myextension_object *obj = (myextension_object *)zend_object_store_get_object(object TSRMLS_CC);
obj->extension = myExt;
}
PHP_METHOD(myExtension, connect)
{
char* host, *user, *secret, *db, *dbtype, *baseschema;
int host_len, user_len, secret_len, db_len, dbtype_len, baseschema_len;
long port;
YourNS::Extension::myExtension *ext;
myextension_object *obj = (myextension_object *)zend_object_store_get_object(getThis() TSRMLS_CC);
ext= obj->extension;
if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "slsssss", &host, &host_len, &port, &user, &user_len, &secret, &secret_len, &db, &db_len, &dbtype, &dbtype_len, &baseschema, &baseschema_len) == FAILURE) {
RETURN_NULL();
}
if ( ext != NULL) {
try {
ext->connect(host, port, user, secret, db, dbtype, baseschema);
}
catch (const YourNS::Extension::RuntimeException& e) {
char* msg;
msg = estrdup(e.getMessage().c_str());
zend_throw_exception(zend_exception_get_default(TSRMLS_C), msg, 0 TSRMLS_CC);
efree(msg);
}
}
}
Других решений пока нет …