This commit is contained in:
CismonX 2019-03-17 11:30:03 +08:00
parent 0bd7ee0163
commit 8842a2fa59
3 changed files with 74 additions and 46 deletions

0
EXPERIMENTAL Normal file
View File

View File

@ -9,6 +9,10 @@
#include "php_arma.hh" #include "php_arma.hh"
#include <complex>
typedef std::complex<double> cx_double;
namespace php_arma namespace php_arma
{ {
template <typename T> template <typename T>
@ -19,7 +23,9 @@ namespace php_arma
extern zend_class_entry *complex_ce; extern zend_class_entry *complex_ce;
template <typename T> template <typename T>
PHP_ARMA_DECLARE_CE_HANDLERS(complex); PHP_ARMA_DECLARE_CE_HANDLERS(complex);
/// Helper functions for handling cx_double types.
template <> zend_always_inline template <> zend_always_inline
bool zval_is_scalar<cx_double>(zval *zv) bool zval_is_scalar<cx_double>(zval *zv)
@ -27,11 +33,34 @@ namespace php_arma
return Z_TYPE_P(zv) == IS_OBJECT && instanceof_function(Z_OBJCE_P(zv), PHP_ARMA_CE(complex, double)); return Z_TYPE_P(zv) == IS_OBJECT && instanceof_function(Z_OBJCE_P(zv), PHP_ARMA_CE(complex, double));
} }
template <>
struct scalar_type<cx_double> {
static const int value = IS_OBJECT;
};
template <> zend_always_inline template <> zend_always_inline
const char *scalar_type_name<cx_double>() const char *scalar_type_name<cx_double>()
{ {
return ZSTR_VAL(PHP_ARMA_CE(complex, double)->name); return ZSTR_VAL(PHP_ARMA_CE(complex, double)->name);
} }
template <> zend_always_inline
cx_double zval_get_scalar<cx_double>(zval *zv)
{
auto zobj = Z_OBJ_P(zv);
return { Z_DVAL_P(OBJ_PROP_NUM(zobj, 0)), Z_DVAL_P(OBJ_PROP_NUM(zobj, 1)) };
}
zend_always_inline
void zval_set_scalar(zval *zv, cx_double val)
{
auto zobj = object_create(PHP_ARMA_CE(complex, double), PHP_ARMA_HANDLERS(complex, double));
ZVAL_DOUBLE(OBJ_PROP_NUM(zobj, 0), val.real());
ZVAL_DOUBLE(OBJ_PROP_NUM(zobj, 1), val.imag());
zval_ptr_dtor(zv);
ZVAL_OBJ(zv, zobj);
}
} }
#endif //!PHP_ARMA_COMPLEX_HH #endif //!PHP_ARMA_COMPLEX_HH

View File

@ -13,7 +13,8 @@
#include <php.h> #include <php.h>
#include <zend_exceptions.h> #include <zend_exceptions.h>
#include <complex>
#include <type_traits>
#define PHP_ARMA_VERSION "0.0.1" #define PHP_ARMA_VERSION "0.0.1"
@ -23,8 +24,6 @@
extern zend_module_entry arma_module_entry; extern zend_module_entry arma_module_entry;
typedef std::complex<double> cx_double;
/// Helper macros for method entry. /// Helper macros for method entry.
#define Z_OBJNAME_P(zval_p) \ #define Z_OBJNAME_P(zval_p) \
@ -81,6 +80,8 @@ typedef std::complex<double> cx_double;
namespace php_arma namespace php_arma
{ {
/// Helper functions for initializing class entry and object handlers.
template <typename F> template <typename F>
zend_class_entry *ce_init(const char *name, const zend_function_entry *methods, F init_func) zend_class_entry *ce_init(const char *name, const zend_function_entry *methods, F init_func)
{ {
@ -111,12 +112,11 @@ namespace php_arma
return ce_init(name, methods, zend_register_internal_class, interfaces...); return ce_init(name, methods, zend_register_internal_class, interfaces...);
} }
// Although methods are declared in interfaces as you see in the stubs,
// nothing is declared in the internal interface implementation.
template <typename> template <typename>
zend_class_entry *interface_register(const char *name) zend_class_entry *interface_register(const char *name)
{ {
// Although methods are declared in interfaces as you see in the stubs,
// nothing is declared in the internal interface implementation.
return ce_init(name, nullptr, zend_register_internal_class); return ce_init(name, nullptr, zend_register_internal_class);
} }
@ -156,7 +156,8 @@ namespace php_arma
return reinterpret_cast<T*>(zobj) - 1; return reinterpret_cast<T*>(zobj) - 1;
} }
/// Create a zend_object which wraps a native object. /// Helper functions for creating objects.
template <typename T, typename F> template <typename T, typename F>
zend_object *object_create(zend_class_entry *ce, F init) zend_object *object_create(zend_class_entry *ce, F init)
{ {
@ -169,29 +170,18 @@ namespace php_arma
return zobj; return zobj;
} }
/// Create a normal zend_object. zend_always_inline
template <typename T> zend_object *object_create(zend_class_entry *ce, const zend_object_handlers *handlers = &std_object_handlers)
zend_object *object_create(zend_class_entry *ce)
{ {
auto zobj = reinterpret_cast<zend_object*>(ecalloc(1, auto zobj = reinterpret_cast<zend_object*>(ecalloc(1,
sizeof(zend_object) + zend_object_properties_size(ce))); sizeof(zend_object) + zend_object_properties_size(ce)));
zend_object_std_init(zobj, ce); zend_object_std_init(zobj, ce);
object_properties_init(zobj, ce); object_properties_init(zobj, ce);
zobj->handlers = &std_object_handlers; zobj->handlers = handlers;
return zobj; return zobj;
} }
zend_always_inline /// Helper functions for handling scalar types.
void zval_set_scalar(zval *zv, double val)
{
ZVAL_DOUBLE(zv, val);
}
zend_always_inline
void zval_set_scalar(zval *zv, zend_long val)
{
ZVAL_LONG(zv, val);
}
template <typename> template <typename>
struct false_type : std::false_type {}; struct false_type : std::false_type {};
@ -202,24 +192,6 @@ namespace php_arma
static_assert(false_type<T>::value, "Type should be zend_long, double or cx_double"); static_assert(false_type<T>::value, "Type should be zend_long, double or cx_double");
} }
template <typename T>
T zval_get_scalar(zval *zv)
{
assert_bad_type<T>();
}
template <> zend_always_inline
double zval_get_scalar<double>(zval *zv)
{
return Z_DVAL_P(zv);
}
template <> zend_always_inline
zend_long zval_get_scalar<zend_long>(zval *zv)
{
return Z_LVAL_P(zv);
}
template <typename T> template <typename T>
bool zval_is_scalar(zval *zv) bool zval_is_scalar(zval *zv)
{ {
@ -251,11 +223,6 @@ namespace php_arma
static const int value = IS_LONG; static const int value = IS_LONG;
}; };
template <>
struct scalar_type<cx_double> {
static const int value = IS_OBJECT;
};
template <typename T> template <typename T>
const char *scalar_type_name() const char *scalar_type_name()
{ {
@ -274,6 +241,36 @@ namespace php_arma
return zend_get_type_by_const(IS_LONG); return zend_get_type_by_const(IS_LONG);
} }
template <typename T>
T zval_get_scalar(zval *zv)
{
assert_bad_type<T>();
}
template <> zend_always_inline
double zval_get_scalar<double>(zval *zv)
{
return Z_DVAL_P(zv);
}
template <> zend_always_inline
zend_long zval_get_scalar<zend_long>(zval *zv)
{
return Z_LVAL_P(zv);
}
zend_always_inline
void zval_set_scalar(zval *zv, double val)
{
ZVAL_DOUBLE(zv, val);
}
zend_always_inline
void zval_set_scalar(zval *zv, zend_long val)
{
ZVAL_LONG(zv, val);
}
zend_always_inline zend_always_inline
const char *zval_get_type_name(zval *zv) const char *zval_get_type_name(zval *zv)
{ {
@ -284,6 +281,8 @@ namespace php_arma
} }
} }
/// Helper functions for throwing exceptions when errors occur.
zend_always_inline zend_always_inline
void ex_bad_type(const char *expected, const char *got) void ex_bad_type(const char *expected, const char *got)
{ {