php_v8js_macros.h 9.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306
  1. /*
  2. +----------------------------------------------------------------------+
  3. | PHP Version 5 |
  4. +----------------------------------------------------------------------+
  5. | Copyright (c) 1997-2013 The PHP Group |
  6. +----------------------------------------------------------------------+
  7. | http://www.opensource.org/licenses/mit-license.php MIT License |
  8. +----------------------------------------------------------------------+
  9. | Author: Jani Taskinen <[email protected]> |
  10. | Author: Patrick Reilly <[email protected]> |
  11. +----------------------------------------------------------------------+
  12. */
  13. /* $Id$ */
  14. #ifndef PHP_V8JS_MACROS_H
  15. #define PHP_V8JS_MACROS_H
  16. #if __GNUC__ == 4 && __GNUC_MINOR__ == 4
  17. #define _GLIBCXX_USE_NANOSLEEP 1
  18. #endif
  19. extern "C" {
  20. #include "php.h"
  21. #include "php_v8js.h"
  22. }
  23. #include <v8.h>
  24. #include <chrono>
  25. #include <stack>
  26. #include <thread>
  27. #include <map>
  28. #include <list>
  29. #include <vector>
  30. #include <mutex>
  31. /* V8Js Version */
  32. #define V8JS_VERSION "0.1.5"
  33. /* V8 from 3.23.12 has most v8::Anything::New constructors expect isolates
  34. as their first argument. Older versions don't provide these. */
  35. #if PHP_V8_API_VERSION < 3023012
  36. #define V8JS_NEW(t, i, v...) t::New(v)
  37. #else
  38. #define V8JS_NEW(t, i, v...) t::New(i, v)
  39. #endif
  40. /* Helper macros */
  41. #define V8JS_SYM(v) v8::String::NewFromUtf8(isolate, v, v8::String::kInternalizedString, sizeof(v) - 1)
  42. #define V8JS_SYML(v, l) v8::String::NewFromUtf8(isolate, v, v8::String::kInternalizedString, l)
  43. #define V8JS_STR(v) v8::String::NewFromUtf8(isolate, v)
  44. #define V8JS_STRL(v, l) v8::String::NewFromUtf8(isolate, v, v8::String::kNormalString, l)
  45. #if PHP_V8_API_VERSION < 3024010
  46. #define V8JS_INT(v) v8::Integer::New(v, isolate)
  47. #define V8JS_UINT(v) v8::Integer::NewFromUnsigned(v, isolate)
  48. #else
  49. #define V8JS_INT(v) v8::Integer::New(isolate, v)
  50. #define V8JS_UINT(v) v8::Integer::NewFromUnsigned(isolate, v)
  51. #endif
  52. #define V8JS_FLOAT(v) v8::Number::New(isolate, v)
  53. #define V8JS_BOOL(v) ((v)?v8::True(isolate):v8::False(isolate))
  54. #if PHP_V8_API_VERSION <= 3023012
  55. #define V8JS_DATE(v) v8::Date::New(v)
  56. #else
  57. #define V8JS_DATE(v) v8::Date::New(isolate, v)
  58. #endif
  59. #define V8JS_NULL v8::Null(isolate)
  60. #define V8JS_UNDEFINED v8::Undefined(isolate)
  61. #define V8JS_MN(name) v8js_method_##name
  62. #define V8JS_METHOD(name) void V8JS_MN(name)(const v8::FunctionCallbackInfo<v8::Value>& info)
  63. #define V8JS_THROW(isolate, type, message, message_len) (isolate)->ThrowException(v8::Exception::type(V8JS_STRL(message, message_len)))
  64. #define V8JS_GLOBAL(isolate) ((isolate)->GetCurrentContext()->Global())
  65. #if PHP_V8_API_VERSION < 3022000
  66. /* CopyablePersistentTraits is only part of V8 from 3.22.0 on,
  67. to be compatible with lower versions add our own (compatible) version. */
  68. namespace v8 {
  69. template<class T>
  70. struct CopyablePersistentTraits {
  71. typedef Persistent<T, CopyablePersistentTraits<T> > CopyablePersistent;
  72. static const bool kResetInDestructor = true;
  73. template<class S, class M>
  74. #if PHP_V8_API_VERSION >= 3021015
  75. static V8_INLINE void Copy(const Persistent<S, M>& source,
  76. CopyablePersistent* dest)
  77. #else
  78. V8_INLINE(static void Copy(const Persistent<S, M>& source,
  79. CopyablePersistent* dest))
  80. #endif
  81. {
  82. // do nothing, just allow copy
  83. }
  84. };
  85. }
  86. #endif
  87. /* Abbreviate long type names */
  88. typedef v8::Persistent<v8::FunctionTemplate, v8::CopyablePersistentTraits<v8::FunctionTemplate> > v8js_tmpl_t;
  89. typedef v8::Persistent<v8::Object, v8::CopyablePersistentTraits<v8::Object> > v8js_persistent_obj_t;
  90. /* Hidden field name used to link JS wrappers with underlying PHP object */
  91. #define PHPJS_OBJECT_KEY "phpjs::object"
  92. /* Helper macros */
  93. #if PHP_V8_API_VERSION < 2005009
  94. # define V8JS_GET_CLASS_NAME(var, obj) \
  95. /* Hack to prevent calling possibly set user-defined __toString() messing class name */ \
  96. v8::Local<v8::Function> constructor = v8::Local<v8::Function>::Cast(obj->Get(V8JS_SYM("constructor"))); \
  97. v8::String::Utf8Value var(constructor->GetName());
  98. #else
  99. # define V8JS_GET_CLASS_NAME(var, obj) \
  100. v8::String::Utf8Value var(obj->GetConstructorName());
  101. #endif
  102. #if ZEND_MODULE_API_NO >= 20100409
  103. # define ZEND_HASH_KEY_DC , const zend_literal *key
  104. # define ZEND_HASH_KEY_CC , key
  105. # define ZEND_HASH_KEY_NULL , NULL
  106. #else
  107. # define ZEND_HASH_KEY_DC
  108. # define ZEND_HASH_KEY_CC
  109. # define ZEND_HASH_KEY_NULL
  110. #endif
  111. /* method signatures of zend_update_property and zend_read_property were
  112. * declared as 'char *' instead of 'const char *' before PHP 5.4 */
  113. #if ZEND_MODULE_API_NO >= 20100525
  114. # define V8JS_CONST
  115. #else
  116. # define V8JS_CONST (char *)
  117. #endif
  118. /* Global flags */
  119. #define V8JS_GLOBAL_SET_FLAGS(isolate,flags) V8JS_GLOBAL(isolate)->SetHiddenValue(V8JS_SYM("__php_flags__"), V8JS_INT(flags))
  120. #define V8JS_GLOBAL_GET_FLAGS(isolate) V8JS_GLOBAL(isolate)->GetHiddenValue(V8JS_SYM("__php_flags__"))->IntegerValue();
  121. /* Options */
  122. #define V8JS_FLAG_NONE (1<<0)
  123. #define V8JS_FLAG_FORCE_ARRAY (1<<1)
  124. #define V8JS_DEBUG_AUTO_BREAK_NEVER 0
  125. #define V8JS_DEBUG_AUTO_BREAK_ONCE 1
  126. #define V8JS_DEBUG_AUTO_BREAK_ALWAYS 2
  127. /* Extracts a C string from a V8 Utf8Value. */
  128. static inline const char * ToCString(const v8::String::Utf8Value &value) /* {{{ */
  129. {
  130. return *value ? *value : "<string conversion failed>";
  131. }
  132. /* }}} */
  133. /* Extern Class entries */
  134. extern zend_class_entry *php_ce_v8_object;
  135. extern zend_class_entry *php_ce_v8_function;
  136. /* Create PHP V8 object */
  137. void php_v8js_create_v8(zval *, v8::Handle<v8::Value>, int, v8::Isolate * TSRMLS_DC);
  138. /* Fetch V8 object properties */
  139. int php_v8js_v8_get_properties_hash(v8::Handle<v8::Value>, HashTable *, int, v8::Isolate * TSRMLS_DC);
  140. /* Convert zval into V8 value */
  141. v8::Handle<v8::Value> zval_to_v8js(zval *, v8::Isolate * TSRMLS_DC);
  142. /* Convert V8 value into zval */
  143. int v8js_to_zval(v8::Handle<v8::Value>, zval *, int, v8::Isolate * TSRMLS_DC);
  144. struct php_v8js_accessor_ctx
  145. {
  146. char *variable_name_string;
  147. uint variable_name_string_len;
  148. v8::Isolate *isolate;
  149. };
  150. void php_v8js_accessor_ctx_dtor(php_v8js_accessor_ctx * TSRMLS_DC);
  151. /* Register accessors into passed object */
  152. void php_v8js_register_accessors(std::vector<php_v8js_accessor_ctx*> *accessor_list, v8::Local<v8::FunctionTemplate>, zval *, v8::Isolate * TSRMLS_DC);
  153. struct php_v8js_object;
  154. /* {{{ Context container */
  155. struct php_v8js_ctx {
  156. zend_object std;
  157. v8::Persistent<v8::String> object_name;
  158. v8::Persistent<v8::Context> context;
  159. zend_bool report_uncaught;
  160. zval *pending_exception;
  161. int in_execution;
  162. v8::Isolate *isolate;
  163. bool time_limit_hit;
  164. bool memory_limit_hit;
  165. v8::Persistent<v8::FunctionTemplate> global_template;
  166. zval *module_loader;
  167. std::vector<char *> modules_stack;
  168. std::vector<char *> modules_base;
  169. std::map<const char *,v8js_tmpl_t> template_cache;
  170. std::map<zval *, v8js_persistent_obj_t> weak_objects;
  171. std::map<v8js_tmpl_t *, v8js_persistent_obj_t> weak_closures;
  172. std::list<php_v8js_object *> php_v8js_objects;
  173. std::vector<php_v8js_accessor_ctx *> accessor_list;
  174. char *tz;
  175. #ifdef ZTS
  176. void ***zts_ctx;
  177. #endif
  178. };
  179. /* }}} */
  180. #ifdef ZTS
  181. # if PHP_V8_API_VERSION <= 3023008
  182. /* Until V8 3.23.8 Isolate could only take one external pointer. */
  183. # define V8JS_TSRMLS_FETCH() TSRMLS_FETCH_FROM_CTX(((php_v8js_ctx *) isolate->GetData())->zts_ctx);
  184. # else
  185. # define V8JS_TSRMLS_FETCH() TSRMLS_FETCH_FROM_CTX(((php_v8js_ctx *) isolate->GetData(0))->zts_ctx);
  186. # endif
  187. #else
  188. # define V8JS_TSRMLS_FETCH()
  189. #endif
  190. // Timer context
  191. struct php_v8js_timer_ctx
  192. {
  193. long time_limit;
  194. long memory_limit;
  195. std::chrono::time_point<std::chrono::high_resolution_clock> time_point;
  196. php_v8js_ctx *v8js_ctx;
  197. bool killed;
  198. };
  199. /* {{{ Object container */
  200. struct php_v8js_object {
  201. zend_object std;
  202. v8::Persistent<v8::Value> v8obj;
  203. int flags;
  204. struct php_v8js_ctx *ctx;
  205. HashTable *properties;
  206. };
  207. /* }}} */
  208. /* Resource declaration */
  209. /* Module globals */
  210. ZEND_BEGIN_MODULE_GLOBALS(v8js)
  211. int v8_initialized;
  212. HashTable *extensions;
  213. /* Ini globals */
  214. char *v8_flags; /* V8 command line flags */
  215. bool use_date; /* Generate JS Date objects instead of PHP DateTime */
  216. // Timer thread globals
  217. std::stack<php_v8js_timer_ctx *> timer_stack;
  218. std::thread *timer_thread;
  219. std::mutex timer_mutex;
  220. bool timer_stop;
  221. std::map<char *, v8::Handle<v8::Object> > modules_loaded;
  222. // fatal error unwinding
  223. bool fatal_error_abort;
  224. int error_num;
  225. char *error_message;
  226. jmp_buf *unwind_env;
  227. void (*old_error_handler)(int, const char *, const uint, const char*, va_list);
  228. ZEND_END_MODULE_GLOBALS(v8js)
  229. extern zend_v8js_globals v8js_globals;
  230. ZEND_EXTERN_MODULE_GLOBALS(v8js)
  231. #ifdef ZTS
  232. # define V8JSG(v) TSRMG(v8js_globals_id, zend_v8js_globals *, v)
  233. #else
  234. # define V8JSG(v) (v8js_globals.v)
  235. #endif
  236. /* Register builtin methods into passed object */
  237. void php_v8js_register_methods(v8::Handle<v8::ObjectTemplate>, php_v8js_ctx *c);
  238. typedef struct _php_v8js_script {
  239. char *name;
  240. v8::Isolate *isolate;
  241. v8::Persistent<v8::Script, v8::CopyablePersistentTraits<v8::Script>> *script;
  242. } php_v8js_script;
  243. static void php_v8js_script_free(php_v8js_script *res);
  244. #endif /* PHP_V8JS_MACROS_H */
  245. /*
  246. * Local variables:
  247. * tab-width: 4
  248. * c-basic-offset: 4
  249. * End:
  250. * vim600: noet sw=4 ts=4 fdm=marker
  251. * vim<600: noet sw=4 ts=4
  252. */