zend_accelerator_util_funcs.c 30 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947
  1. /*
  2. +----------------------------------------------------------------------+
  3. | Zend OPcache |
  4. +----------------------------------------------------------------------+
  5. | Copyright (c) 1998-2018 The PHP Group |
  6. +----------------------------------------------------------------------+
  7. | This source file is subject to version 3.01 of the PHP license, |
  8. | that is bundled with this package in the file LICENSE, and is |
  9. | available through the world-wide-web at the following url: |
  10. | http://www.php.net/license/3_01.txt |
  11. | If you did not receive a copy of the PHP license and are unable to |
  12. | obtain it through the world-wide-web, please send a note to |
  13. | license@php.net so we can mail you a copy immediately. |
  14. +----------------------------------------------------------------------+
  15. | Authors: Andi Gutmans <andi@php.net> |
  16. | Zeev Suraski <zeev@php.net> |
  17. | Stanislav Malyshev <stas@zend.com> |
  18. | Dmitry Stogov <dmitry@php.net> |
  19. +----------------------------------------------------------------------+
  20. */
  21. #include "zend_API.h"
  22. #include "zend_constants.h"
  23. #include "zend_accelerator_util_funcs.h"
  24. #include "zend_persist.h"
  25. #include "zend_shared_alloc.h"
  26. #if SIZEOF_SIZE_T <= SIZEOF_ZEND_LONG
  27. /* If sizeof(void*) == sizeof(ulong) we can use zend_hash index functions */
  28. # define accel_xlat_set(old, new) zend_hash_index_add_new_ptr(&ZCG(bind_hash), (zend_ulong)(zend_uintptr_t)(old), (new))
  29. # define accel_xlat_get(old) zend_hash_index_find_ptr(&ZCG(bind_hash), (zend_ulong)(zend_uintptr_t)(old))
  30. #else
  31. # define accel_xlat_set(old, new) zend_hash_str_add_new_ptr(&ZCG(bind_hash), (char*)&(old), sizeof(void*), (new))
  32. # define accel_xlat_get(old) zend_hash_str_find_ptr(&ZCG(bind_hash), (char*)&(old), sizeof(void*))
  33. #endif
  34. #define ARENA_REALLOC(ptr) \
  35. (void*)(((char*)(ptr)) + ((char*)ZCG(arena_mem) - (char*)ZCG(current_persistent_script)->arena_mem))
  36. typedef int (*id_function_t)(void *, void *);
  37. typedef void (*unique_copy_ctor_func_t)(void *pElement);
  38. static void zend_accel_destroy_zend_function(zval *zv)
  39. {
  40. zend_function *function = Z_PTR_P(zv);
  41. if (function->type == ZEND_USER_FUNCTION) {
  42. if (function->op_array.static_variables) {
  43. if (!(GC_FLAGS(function->op_array.static_variables) & IS_ARRAY_IMMUTABLE)) {
  44. if (GC_DELREF(function->op_array.static_variables) == 0) {
  45. FREE_HASHTABLE(function->op_array.static_variables);
  46. }
  47. }
  48. function->op_array.static_variables = NULL;
  49. }
  50. }
  51. zend_function_dtor(zv);
  52. }
  53. static void zend_accel_destroy_zend_class(zval *zv)
  54. {
  55. zend_class_entry *ce = Z_PTR_P(zv);
  56. ce->function_table.pDestructor = zend_accel_destroy_zend_function;
  57. destroy_zend_class(zv);
  58. }
  59. zend_persistent_script* create_persistent_script(void)
  60. {
  61. zend_persistent_script *persistent_script = (zend_persistent_script *) emalloc(sizeof(zend_persistent_script));
  62. memset(persistent_script, 0, sizeof(zend_persistent_script));
  63. zend_hash_init(&persistent_script->script.function_table, 128, NULL, ZEND_FUNCTION_DTOR, 0);
  64. /* class_table is usually destroyed by free_persistent_script() that
  65. * overrides destructor. ZEND_CLASS_DTOR may be used by standard
  66. * PHP compiler
  67. */
  68. zend_hash_init(&persistent_script->script.class_table, 16, NULL, ZEND_CLASS_DTOR, 0);
  69. return persistent_script;
  70. }
  71. void free_persistent_script(zend_persistent_script *persistent_script, int destroy_elements)
  72. {
  73. if (destroy_elements) {
  74. persistent_script->script.function_table.pDestructor = zend_accel_destroy_zend_function;
  75. persistent_script->script.class_table.pDestructor = zend_accel_destroy_zend_class;
  76. } else {
  77. persistent_script->script.function_table.pDestructor = NULL;
  78. persistent_script->script.class_table.pDestructor = NULL;
  79. }
  80. zend_hash_destroy(&persistent_script->script.function_table);
  81. zend_hash_destroy(&persistent_script->script.class_table);
  82. if (persistent_script->script.filename) {
  83. zend_string_release_ex(persistent_script->script.filename, 0);
  84. }
  85. efree(persistent_script);
  86. }
  87. static int is_not_internal_function(zval *zv)
  88. {
  89. zend_function *function = Z_PTR_P(zv);
  90. return(function->type != ZEND_INTERNAL_FUNCTION);
  91. }
  92. void zend_accel_free_user_functions(HashTable *ht)
  93. {
  94. dtor_func_t orig_dtor = ht->pDestructor;
  95. ht->pDestructor = NULL;
  96. zend_hash_apply(ht, (apply_func_t) is_not_internal_function);
  97. ht->pDestructor = orig_dtor;
  98. }
  99. void zend_accel_move_user_functions(HashTable *src, HashTable *dst)
  100. {
  101. Bucket *p;
  102. dtor_func_t orig_dtor = src->pDestructor;
  103. src->pDestructor = NULL;
  104. zend_hash_extend(dst, dst->nNumUsed + src->nNumUsed, 0);
  105. ZEND_HASH_REVERSE_FOREACH_BUCKET(src, p) {
  106. zend_function *function = Z_PTR(p->val);
  107. if (EXPECTED(function->type == ZEND_USER_FUNCTION)) {
  108. _zend_hash_append_ptr(dst, p->key, function);
  109. zend_hash_del_bucket(src, p);
  110. } else {
  111. break;
  112. }
  113. } ZEND_HASH_FOREACH_END();
  114. src->pDestructor = orig_dtor;
  115. }
  116. void zend_accel_copy_internal_functions(void)
  117. {
  118. zend_string *key;
  119. zval *val;
  120. ZEND_HASH_FOREACH_STR_KEY_VAL(CG(function_table), key, val) {
  121. zend_internal_function *function = Z_PTR_P(val);
  122. if (function->type == ZEND_INTERNAL_FUNCTION) {
  123. zend_hash_add_new_ptr(&ZCG(function_table), key, function);
  124. }
  125. } ZEND_HASH_FOREACH_END();
  126. ZCG(internal_functions_count) = zend_hash_num_elements(&ZCG(function_table));
  127. }
  128. static inline void zend_clone_zval(zval *src)
  129. {
  130. void *ptr;
  131. if (Z_TYPE_P(src) == IS_REFERENCE) {
  132. ptr = accel_xlat_get(Z_REF_P(src));
  133. if (ptr != NULL) {
  134. Z_REF_P(src) = ptr;
  135. return;
  136. } else {
  137. zend_reference *old = Z_REF_P(src);
  138. ZVAL_NEW_REF(src, &old->val);
  139. Z_REF_P(src)->gc = old->gc;
  140. accel_xlat_set(old, Z_REF_P(src));
  141. src = Z_REFVAL_P(src);
  142. }
  143. }
  144. }
  145. static void zend_hash_clone_constants(HashTable *ht, HashTable *source)
  146. {
  147. Bucket *p, *q, *end;
  148. zend_ulong nIndex;
  149. zend_class_constant *c;
  150. ht->nTableSize = source->nTableSize;
  151. ht->nTableMask = source->nTableMask;
  152. ht->nNumUsed = 0;
  153. ht->nNumOfElements = source->nNumOfElements;
  154. ht->nNextFreeElement = source->nNextFreeElement;
  155. ht->pDestructor = NULL;
  156. HT_FLAGS(ht) = (HT_FLAGS(source) & (HASH_FLAG_INITIALIZED | HASH_FLAG_STATIC_KEYS));
  157. ht->nInternalPointer = 0;
  158. if (!(HT_FLAGS(ht) & HASH_FLAG_INITIALIZED)) {
  159. ht->arData = source->arData;
  160. return;
  161. }
  162. ZEND_ASSERT((HT_FLAGS(source) & HASH_FLAG_PACKED) == 0);
  163. HT_SET_DATA_ADDR(ht, emalloc(HT_SIZE(ht)));
  164. HT_HASH_RESET(ht);
  165. p = source->arData;
  166. end = p + source->nNumUsed;
  167. for (; p != end; p++) {
  168. ZEND_ASSERT(Z_TYPE(p->val) != IS_UNDEF);
  169. nIndex = p->h | ht->nTableMask;
  170. /* Insert into hash collision list */
  171. q = ht->arData + ht->nNumUsed;
  172. Z_NEXT(q->val) = HT_HASH(ht, nIndex);
  173. HT_HASH(ht, nIndex) = HT_IDX_TO_HASH(ht->nNumUsed++);
  174. /* Initialize key */
  175. q->h = p->h;
  176. q->key = p->key;
  177. /* Copy data */
  178. c = ARENA_REALLOC(Z_PTR(p->val));
  179. ZVAL_PTR(&q->val, c);
  180. zend_clone_zval(&c->value);
  181. if ((void*)c->ce >= ZCG(current_persistent_script)->arena_mem &&
  182. (void*)c->ce < (void*)((char*)ZCG(current_persistent_script)->arena_mem + ZCG(current_persistent_script)->arena_size)) {
  183. c->ce = ARENA_REALLOC(c->ce);
  184. }
  185. }
  186. }
  187. static void zend_hash_clone_methods(HashTable *ht, HashTable *source, zend_class_entry *old_ce, zend_class_entry *ce)
  188. {
  189. Bucket *p, *q, *end;
  190. zend_ulong nIndex;
  191. zend_op_array *new_entry;
  192. ht->nTableSize = source->nTableSize;
  193. ht->nTableMask = source->nTableMask;
  194. ht->nNumUsed = 0;
  195. ht->nNumOfElements = source->nNumOfElements;
  196. ht->nNextFreeElement = source->nNextFreeElement;
  197. ht->pDestructor = ZEND_FUNCTION_DTOR;
  198. HT_FLAGS(ht) = (HT_FLAGS(source) & (HASH_FLAG_INITIALIZED | HASH_FLAG_STATIC_KEYS));
  199. ht->nInternalPointer = 0;
  200. if (!(HT_FLAGS(ht) & HASH_FLAG_INITIALIZED)) {
  201. ht->arData = source->arData;
  202. return;
  203. }
  204. ZEND_ASSERT(!(HT_FLAGS(source) & HASH_FLAG_PACKED));
  205. HT_SET_DATA_ADDR(ht, emalloc(HT_SIZE(ht)));
  206. HT_HASH_RESET(ht);
  207. p = source->arData;
  208. end = p + source->nNumUsed;
  209. for (; p != end; p++) {
  210. ZEND_ASSERT(Z_TYPE(p->val) != IS_UNDEF);
  211. nIndex = p->h | ht->nTableMask;
  212. /* Insert into hash collision list */
  213. q = ht->arData + ht->nNumUsed;
  214. Z_NEXT(q->val) = HT_HASH(ht, nIndex);
  215. HT_HASH(ht, nIndex) = HT_IDX_TO_HASH(ht->nNumUsed++);
  216. /* Initialize key */
  217. q->h = p->h;
  218. ZEND_ASSERT(p->key != NULL);
  219. q->key = p->key;
  220. /* Copy data */
  221. ZVAL_PTR(&q->val, ARENA_REALLOC(Z_PTR(p->val)));
  222. new_entry = (zend_op_array*)Z_PTR(q->val);
  223. if ((void*)new_entry->scope >= ZCG(current_persistent_script)->arena_mem &&
  224. (void*)new_entry->scope < (void*)((char*)ZCG(current_persistent_script)->arena_mem + ZCG(current_persistent_script)->arena_size)) {
  225. new_entry->scope = ARENA_REALLOC(new_entry->scope);
  226. /* update prototype */
  227. if (new_entry->prototype) {
  228. new_entry->prototype = ARENA_REALLOC(new_entry->prototype);
  229. }
  230. }
  231. }
  232. }
  233. static void zend_hash_clone_prop_info(HashTable *ht, HashTable *source, zend_class_entry *old_ce)
  234. {
  235. Bucket *p, *q, *end;
  236. zend_ulong nIndex;
  237. zend_property_info *prop_info;
  238. ht->nTableSize = source->nTableSize;
  239. ht->nTableMask = source->nTableMask;
  240. ht->nNumUsed = 0;
  241. ht->nNumOfElements = source->nNumOfElements;
  242. ht->nNextFreeElement = source->nNextFreeElement;
  243. ht->pDestructor = NULL;
  244. HT_FLAGS(ht) = (HT_FLAGS(source) & (HASH_FLAG_INITIALIZED | HASH_FLAG_STATIC_KEYS));
  245. ht->nInternalPointer = 0;
  246. if (!(HT_FLAGS(ht) & HASH_FLAG_INITIALIZED)) {
  247. ht->arData = source->arData;
  248. return;
  249. }
  250. ZEND_ASSERT(!(HT_FLAGS(source) & HASH_FLAG_PACKED));
  251. HT_SET_DATA_ADDR(ht, emalloc(HT_SIZE(ht)));
  252. HT_HASH_RESET(ht);
  253. p = source->arData;
  254. end = p + source->nNumUsed;
  255. for (; p != end; p++) {
  256. ZEND_ASSERT(Z_TYPE(p->val) != IS_UNDEF);
  257. nIndex = p->h | ht->nTableMask;
  258. /* Insert into hash collision list */
  259. q = ht->arData + ht->nNumUsed;
  260. Z_NEXT(q->val) = HT_HASH(ht, nIndex);
  261. HT_HASH(ht, nIndex) = HT_IDX_TO_HASH(ht->nNumUsed++);
  262. /* Initialize key */
  263. q->h = p->h;
  264. ZEND_ASSERT(p->key != NULL);
  265. q->key = p->key;
  266. /* Copy data */
  267. prop_info = ARENA_REALLOC(Z_PTR(p->val));
  268. ZVAL_PTR(&q->val, prop_info);
  269. if ((void*)prop_info->ce >= ZCG(current_persistent_script)->arena_mem &&
  270. (void*)prop_info->ce < (void*)((char*)ZCG(current_persistent_script)->arena_mem + ZCG(current_persistent_script)->arena_size)) {
  271. prop_info->ce = ARENA_REALLOC(prop_info->ce);
  272. }
  273. }
  274. }
  275. #define zend_update_inherited_handler(handler) \
  276. { \
  277. if (ce->handler != NULL) { \
  278. ce->handler = ARENA_REALLOC(ce->handler); \
  279. } \
  280. }
  281. /* Protects class' refcount, copies default properties, functions and class name */
  282. static void zend_class_copy_ctor(zend_class_entry **pce)
  283. {
  284. zend_class_entry *ce = *pce;
  285. zend_class_entry *old_ce = ce;
  286. zval *src, *dst, *end;
  287. *pce = ce = ARENA_REALLOC(old_ce);
  288. ce->refcount = 1;
  289. if (ce->parent) {
  290. ce->parent = ARENA_REALLOC(ce->parent);
  291. }
  292. if (old_ce->default_properties_table) {
  293. ce->default_properties_table = emalloc(sizeof(zval) * old_ce->default_properties_count);
  294. src = old_ce->default_properties_table;
  295. end = src + old_ce->default_properties_count;
  296. dst = ce->default_properties_table;
  297. for (; src != end; src++, dst++) {
  298. ZVAL_COPY_VALUE(dst, src);
  299. zend_clone_zval(dst);
  300. }
  301. }
  302. zend_hash_clone_methods(&ce->function_table, &old_ce->function_table, old_ce, ce);
  303. /* static members */
  304. if (old_ce->default_static_members_table) {
  305. int i, end;
  306. zend_class_entry *parent = ce->parent;
  307. ce->default_static_members_table = emalloc(sizeof(zval) * old_ce->default_static_members_count);
  308. i = ce->default_static_members_count - 1;
  309. /* Copy static properties in this class */
  310. end = parent ? parent->default_static_members_count : 0;
  311. for (; i >= end; i--) {
  312. zval *p = &ce->default_static_members_table[i];
  313. ZVAL_COPY_VALUE(p, &old_ce->default_static_members_table[i]);
  314. zend_clone_zval(p);
  315. }
  316. /* Create indirections to static properties from parent classes */
  317. while (parent && parent->default_static_members_table) {
  318. end = parent->parent ? parent->parent->default_static_members_count : 0;
  319. for (; i >= end; i--) {
  320. zval *p = &ce->default_static_members_table[i];
  321. ZVAL_INDIRECT(p, &parent->default_static_members_table[i]);
  322. }
  323. parent = parent->parent;
  324. }
  325. }
  326. ce->static_members_table = ce->default_static_members_table;
  327. /* properties_info */
  328. zend_hash_clone_prop_info(&ce->properties_info, &old_ce->properties_info, old_ce);
  329. /* constants table */
  330. zend_hash_clone_constants(&ce->constants_table, &old_ce->constants_table);
  331. /* interfaces aren't really implemented, so we create a new table */
  332. if (ce->num_interfaces) {
  333. ce->interfaces = emalloc(sizeof(zend_class_entry *) * ce->num_interfaces);
  334. memset(ce->interfaces, 0, sizeof(zend_class_entry *) * ce->num_interfaces);
  335. } else {
  336. ce->interfaces = NULL;
  337. }
  338. zend_update_inherited_handler(constructor);
  339. zend_update_inherited_handler(destructor);
  340. zend_update_inherited_handler(clone);
  341. zend_update_inherited_handler(__get);
  342. zend_update_inherited_handler(__set);
  343. zend_update_inherited_handler(__call);
  344. /* 5.1 stuff */
  345. zend_update_inherited_handler(serialize_func);
  346. zend_update_inherited_handler(unserialize_func);
  347. zend_update_inherited_handler(__isset);
  348. zend_update_inherited_handler(__unset);
  349. /* 5.2 stuff */
  350. zend_update_inherited_handler(__tostring);
  351. /* 5.3 stuff */
  352. zend_update_inherited_handler(__callstatic);
  353. zend_update_inherited_handler(__debugInfo);
  354. /* 5.4 traits */
  355. if (ce->trait_aliases) {
  356. zend_trait_alias **trait_aliases;
  357. int i = 0;
  358. while (ce->trait_aliases[i]) {
  359. i++;
  360. }
  361. trait_aliases = emalloc(sizeof(zend_trait_alias*) * (i + 1));
  362. i = 0;
  363. while (ce->trait_aliases[i]) {
  364. trait_aliases[i] = emalloc(sizeof(zend_trait_alias));
  365. memcpy(trait_aliases[i], ce->trait_aliases[i], sizeof(zend_trait_alias));
  366. i++;
  367. }
  368. trait_aliases[i] = NULL;
  369. ce->trait_aliases = trait_aliases;
  370. }
  371. if (ce->trait_precedences) {
  372. zend_trait_precedence **trait_precedences;
  373. int i = 0;
  374. while (ce->trait_precedences[i]) {
  375. i++;
  376. }
  377. trait_precedences = emalloc(sizeof(zend_trait_precedence*) * (i + 1));
  378. i = 0;
  379. while (ce->trait_precedences[i]) {
  380. trait_precedences[i] = emalloc(sizeof(zend_trait_precedence) + (ce->trait_precedences[i]->num_excludes - 1) * sizeof(zend_string*));
  381. memcpy(trait_precedences[i], ce->trait_precedences[i], sizeof(zend_trait_precedence) + (ce->trait_precedences[i]->num_excludes - 1) * sizeof(zend_string*));
  382. i++;
  383. }
  384. trait_precedences[i] = NULL;
  385. ce->trait_precedences = trait_precedences;
  386. }
  387. }
  388. static void zend_accel_function_hash_copy(HashTable *target, HashTable *source)
  389. {
  390. zend_function *function1, *function2;
  391. Bucket *p, *end;
  392. zval *t;
  393. zend_hash_extend(target, target->nNumUsed + source->nNumUsed, 0);
  394. p = source->arData;
  395. end = p + source->nNumUsed;
  396. for (; p != end; p++) {
  397. ZEND_ASSERT(Z_TYPE(p->val) != IS_UNDEF);
  398. ZEND_ASSERT(p->key);
  399. t = zend_hash_find_ex(target, p->key, 1);
  400. if (UNEXPECTED(t != NULL)) {
  401. if (EXPECTED(ZSTR_LEN(p->key) > 0) && EXPECTED(ZSTR_VAL(p->key)[0] == 0)) {
  402. /* Mangled key */
  403. t = zend_hash_update(target, p->key, &p->val);
  404. } else {
  405. goto failure;
  406. }
  407. } else {
  408. _zend_hash_append_ptr(target, p->key, Z_PTR(p->val));
  409. }
  410. }
  411. target->nInternalPointer = 0;
  412. return;
  413. failure:
  414. function1 = Z_PTR(p->val);
  415. function2 = Z_PTR_P(t);
  416. CG(in_compilation) = 1;
  417. zend_set_compiled_filename(function1->op_array.filename);
  418. CG(zend_lineno) = function1->op_array.opcodes[0].lineno;
  419. if (function2->type == ZEND_USER_FUNCTION
  420. && function2->op_array.last > 0) {
  421. zend_error(E_ERROR, "Cannot redeclare %s() (previously declared in %s:%d)",
  422. ZSTR_VAL(function1->common.function_name),
  423. ZSTR_VAL(function2->op_array.filename),
  424. (int)function2->op_array.opcodes[0].lineno);
  425. } else {
  426. zend_error(E_ERROR, "Cannot redeclare %s()", ZSTR_VAL(function1->common.function_name));
  427. }
  428. }
  429. static void zend_accel_function_hash_copy_from_shm(HashTable *target, HashTable *source)
  430. {
  431. zend_function *function1, *function2;
  432. Bucket *p, *end;
  433. zval *t;
  434. zend_hash_extend(target, target->nNumUsed + source->nNumUsed, 0);
  435. p = source->arData;
  436. end = p + source->nNumUsed;
  437. for (; p != end; p++) {
  438. ZEND_ASSERT(Z_TYPE(p->val) != IS_UNDEF);
  439. ZEND_ASSERT(p->key);
  440. t = zend_hash_find_ex(target, p->key, 1);
  441. if (UNEXPECTED(t != NULL)) {
  442. if (EXPECTED(ZSTR_LEN(p->key) > 0) && EXPECTED(ZSTR_VAL(p->key)[0] == 0)) {
  443. /* Mangled key */
  444. zend_hash_update_ptr(target, p->key, Z_PTR(p->val));
  445. } else {
  446. goto failure;
  447. }
  448. } else {
  449. _zend_hash_append_ptr_ex(target, p->key, Z_PTR(p->val), 1);
  450. }
  451. }
  452. target->nInternalPointer = 0;
  453. return;
  454. failure:
  455. function1 = Z_PTR(p->val);
  456. function2 = Z_PTR_P(t);
  457. CG(in_compilation) = 1;
  458. zend_set_compiled_filename(function1->op_array.filename);
  459. CG(zend_lineno) = function1->op_array.opcodes[0].lineno;
  460. if (function2->type == ZEND_USER_FUNCTION
  461. && function2->op_array.last > 0) {
  462. zend_error(E_ERROR, "Cannot redeclare %s() (previously declared in %s:%d)",
  463. ZSTR_VAL(function1->common.function_name),
  464. ZSTR_VAL(function2->op_array.filename),
  465. (int)function2->op_array.opcodes[0].lineno);
  466. } else {
  467. zend_error(E_ERROR, "Cannot redeclare %s()", ZSTR_VAL(function1->common.function_name));
  468. }
  469. }
  470. static void zend_accel_class_hash_copy(HashTable *target, HashTable *source)
  471. {
  472. Bucket *p, *end;
  473. zval *t;
  474. zend_hash_extend(target, target->nNumUsed + source->nNumUsed, 0);
  475. p = source->arData;
  476. end = p + source->nNumUsed;
  477. for (; p != end; p++) {
  478. if (UNEXPECTED(Z_TYPE(p->val) == IS_UNDEF)) continue;
  479. ZEND_ASSERT(p->key);
  480. t = zend_hash_find_ex(target, p->key, 1);
  481. if (UNEXPECTED(t != NULL)) {
  482. if (EXPECTED(ZSTR_LEN(p->key) > 0) && EXPECTED(ZSTR_VAL(p->key)[0] == 0)) {
  483. /* Mangled key - ignore and wait for runtime */
  484. continue;
  485. } else if (UNEXPECTED(!ZCG(accel_directives).ignore_dups)) {
  486. zend_class_entry *ce1 = Z_PTR(p->val);
  487. if (!(ce1->ce_flags & ZEND_ACC_ANON_CLASS)) {
  488. CG(in_compilation) = 1;
  489. zend_set_compiled_filename(ce1->info.user.filename);
  490. CG(zend_lineno) = ce1->info.user.line_start;
  491. zend_error(E_ERROR,
  492. "Cannot declare %s %s, because the name is already in use",
  493. zend_get_object_type(ce1), ZSTR_VAL(ce1->name));
  494. return;
  495. }
  496. continue;
  497. }
  498. } else {
  499. t = _zend_hash_append_ptr(target, p->key, Z_PTR(p->val));
  500. }
  501. }
  502. target->nInternalPointer = 0;
  503. return;
  504. }
  505. static void zend_accel_class_hash_copy_from_shm(HashTable *target, HashTable *source)
  506. {
  507. Bucket *p, *end;
  508. zval *t;
  509. zend_hash_extend(target, target->nNumUsed + source->nNumUsed, 0);
  510. p = source->arData;
  511. end = p + source->nNumUsed;
  512. for (; p != end; p++) {
  513. ZEND_ASSERT(Z_TYPE(p->val) != IS_UNDEF);
  514. ZEND_ASSERT(p->key);
  515. t = zend_hash_find_ex(target, p->key, 1);
  516. if (UNEXPECTED(t != NULL)) {
  517. if (EXPECTED(ZSTR_LEN(p->key) > 0) && EXPECTED(ZSTR_VAL(p->key)[0] == 0)) {
  518. /* Mangled key - ignore and wait for runtime */
  519. continue;
  520. } else if (UNEXPECTED(!ZCG(accel_directives).ignore_dups)) {
  521. zend_class_entry *ce1 = Z_PTR(p->val);
  522. if (!(ce1->ce_flags & ZEND_ACC_ANON_CLASS)) {
  523. CG(in_compilation) = 1;
  524. zend_set_compiled_filename(ce1->info.user.filename);
  525. CG(zend_lineno) = ce1->info.user.line_start;
  526. zend_error(E_ERROR,
  527. "Cannot declare %s %s, because the name is already in use",
  528. zend_get_object_type(ce1), ZSTR_VAL(ce1->name));
  529. return;
  530. }
  531. continue;
  532. }
  533. } else {
  534. t = _zend_hash_append_ptr_ex(target, p->key, Z_PTR(p->val), 1);
  535. zend_class_copy_ctor((zend_class_entry**)&Z_PTR_P(t));
  536. }
  537. }
  538. target->nInternalPointer = 0;
  539. return;
  540. }
  541. #if defined(__AVX__)
  542. # include <nmmintrin.h>
  543. # if defined(__GNUC__) && defined(__i386__)
  544. static zend_always_inline void fast_memcpy(void *dest, const void *src, size_t size)
  545. {
  546. size_t delta = (char*)dest - (char*)src;
  547. __asm__ volatile (
  548. ".align 16\n\t"
  549. ".LL0%=:\n\t"
  550. "prefetchnta 0x40(%1)\n\t"
  551. "vmovaps (%1), %%ymm0\n\t"
  552. "vmovaps 0x20(%1), %%ymm1\n\t"
  553. "vmovaps %%ymm0, (%1,%2)\n\t"
  554. "vmovaps %%ymm1, 0x20(%1,%2)\n\t"
  555. "addl $0x40, %1\n\t"
  556. "subl $0x40, %0\n\t"
  557. "ja .LL0%="
  558. : "+r"(size),
  559. "+r"(src)
  560. : "r"(delta)
  561. : "cc", "memory", "%ymm0", "%ymm1");
  562. }
  563. # elif defined(__GNUC__) && defined(__x86_64__)
  564. static zend_always_inline void fast_memcpy(void *dest, const void *src, size_t size)
  565. {
  566. size_t delta = (char*)dest - (char*)src;
  567. __asm__ volatile (
  568. ".align 16\n\t"
  569. ".LL0%=:\n\t"
  570. "prefetchnta 0x40(%1)\n\t"
  571. "vmovaps (%1), %%ymm0\n\t"
  572. "vmovaps 0x20(%1), %%ymm1\n\t"
  573. "vmovaps %%ymm0, (%1,%2)\n\t"
  574. "vmovaps %%ymm1, 0x20(%1,%2)\n\t"
  575. "addq $0x40, %1\n\t"
  576. "subq $0x40, %0\n\t"
  577. "ja .LL0%="
  578. : "+r"(size),
  579. "+r"(src)
  580. : "r"(delta)
  581. : "cc", "memory", "%ymm0", "%ymm1");
  582. }
  583. # else
  584. static zend_always_inline void fast_memcpy(void *dest, const void *src, size_t size)
  585. {
  586. __m256 *dqdest = (__m256*)dest;
  587. const __m256 *dqsrc = (const __m256*)src;
  588. const __m256 *end = (const __m256*)((const char*)src + size);
  589. do {
  590. #ifdef PHP_WIN32
  591. _mm_prefetch((const char *)(dqsrc + 2), _MM_HINT_NTA);
  592. #else
  593. _mm_prefetch(dqsrc + 2, _MM_HINT_NTA);
  594. #endif
  595. __m256 ymm0 = _mm256_load_ps((const float *)(dqsrc + 0));
  596. __m256 ymm1 = _mm256_load_ps((const float *)(dqsrc + 1));
  597. dqsrc += 2;
  598. _mm256_store_ps((float *)(dqdest + 0), ymm0);
  599. _mm256_store_ps((float *)(dqdest + 1), ymm1);
  600. dqdest += 2;
  601. } while (dqsrc != end);
  602. }
  603. # endif
  604. #elif defined(__SSE2__)
  605. # include <emmintrin.h>
  606. # if defined(__GNUC__) && defined(__i386__)
  607. static zend_always_inline void fast_memcpy(void *dest, const void *src, size_t size)
  608. {
  609. size_t delta = (char*)dest - (char*)src;
  610. __asm__ volatile (
  611. ".align 16\n\t"
  612. ".LL0%=:\n\t"
  613. "prefetchnta 0x40(%1)\n\t"
  614. "movdqa (%1), %%xmm0\n\t"
  615. "movdqa 0x10(%1), %%xmm1\n\t"
  616. "movdqa 0x20(%1), %%xmm2\n\t"
  617. "movdqa 0x30(%1), %%xmm3\n\t"
  618. "movdqa %%xmm0, (%1,%2)\n\t"
  619. "movdqa %%xmm1, 0x10(%1,%2)\n\t"
  620. "movdqa %%xmm2, 0x20(%1,%2)\n\t"
  621. "movdqa %%xmm3, 0x30(%1,%2)\n\t"
  622. "addl $0x40, %1\n\t"
  623. "subl $0x40, %0\n\t"
  624. "ja .LL0%="
  625. : "+r"(size),
  626. "+r"(src)
  627. : "r"(delta)
  628. : "cc", "memory", "%xmm0", "%xmm1", "%xmm1", "%xmm2");
  629. }
  630. # elif defined(__GNUC__) && defined(__x86_64__) && !defined(__ILP32__)
  631. static zend_always_inline void fast_memcpy(void *dest, const void *src, size_t size)
  632. {
  633. size_t delta = (char*)dest - (char*)src;
  634. __asm__ volatile (
  635. ".align 16\n\t"
  636. ".LL0%=:\n\t"
  637. "prefetchnta 0x40(%1)\n\t"
  638. "movdqa (%1), %%xmm0\n\t"
  639. "movdqa 0x10(%1), %%xmm1\n\t"
  640. "movdqa 0x20(%1), %%xmm2\n\t"
  641. "movdqa 0x30(%1), %%xmm3\n\t"
  642. "movdqa %%xmm0, (%1,%2)\n\t"
  643. "movdqa %%xmm1, 0x10(%1,%2)\n\t"
  644. "movdqa %%xmm2, 0x20(%1,%2)\n\t"
  645. "movdqa %%xmm3, 0x30(%1,%2)\n\t"
  646. "addq $0x40, %1\n\t"
  647. "subq $0x40, %0\n\t"
  648. "ja .LL0%="
  649. : "+r"(size),
  650. "+r"(src)
  651. : "r"(delta)
  652. : "cc", "memory", "%xmm0", "%xmm1", "%xmm1", "%xmm2");
  653. }
  654. # else
  655. static zend_always_inline void fast_memcpy(void *dest, const void *src, size_t size)
  656. {
  657. __m128i *dqdest = (__m128i*)dest;
  658. const __m128i *dqsrc = (const __m128i*)src;
  659. const __m128i *end = (const __m128i*)((const char*)src + size);
  660. do {
  661. #ifdef PHP_WIN32
  662. _mm_prefetch((const char *)(dqsrc + 4), _MM_HINT_NTA);
  663. #else
  664. _mm_prefetch(dqsrc + 4, _MM_HINT_NTA);
  665. #endif
  666. __m128i xmm0 = _mm_load_si128(dqsrc + 0);
  667. __m128i xmm1 = _mm_load_si128(dqsrc + 1);
  668. __m128i xmm2 = _mm_load_si128(dqsrc + 2);
  669. __m128i xmm3 = _mm_load_si128(dqsrc + 3);
  670. dqsrc += 4;
  671. _mm_store_si128(dqdest + 0, xmm0);
  672. _mm_store_si128(dqdest + 1, xmm1);
  673. _mm_store_si128(dqdest + 2, xmm2);
  674. _mm_store_si128(dqdest + 3, xmm3);
  675. dqdest += 4;
  676. } while (dqsrc != end);
  677. }
  678. # endif
  679. #endif
  680. zend_op_array* zend_accel_load_script(zend_persistent_script *persistent_script, int from_shared_memory)
  681. {
  682. zend_op_array *op_array;
  683. op_array = (zend_op_array *) emalloc(sizeof(zend_op_array));
  684. *op_array = persistent_script->script.main_op_array;
  685. if (EXPECTED(from_shared_memory)) {
  686. zend_hash_init(&ZCG(bind_hash), 10, NULL, NULL, 0);
  687. ZCG(current_persistent_script) = persistent_script;
  688. ZCG(arena_mem) = NULL;
  689. if (EXPECTED(persistent_script->arena_size)) {
  690. #if defined(__AVX__) || defined(__SSE2__)
  691. /* Target address must be aligned to 64-byte boundary */
  692. _mm_prefetch(persistent_script->arena_mem, _MM_HINT_NTA);
  693. ZCG(arena_mem) = zend_arena_alloc(&CG(arena), persistent_script->arena_size + 64);
  694. ZCG(arena_mem) = (void*)(((zend_uintptr_t)ZCG(arena_mem) + 63L) & ~63L);
  695. fast_memcpy(ZCG(arena_mem), persistent_script->arena_mem, persistent_script->arena_size);
  696. #else
  697. ZCG(arena_mem) = zend_arena_alloc(&CG(arena), persistent_script->arena_size);
  698. memcpy(ZCG(arena_mem), persistent_script->arena_mem, persistent_script->arena_size);
  699. #endif
  700. }
  701. /* Copy all the necessary stuff from shared memory to regular memory, and protect the shared script */
  702. if (zend_hash_num_elements(&persistent_script->script.class_table) > 0) {
  703. zend_accel_class_hash_copy_from_shm(CG(class_table), &persistent_script->script.class_table);
  704. }
  705. /* we must first to copy all classes and then prepare functions, since functions may try to bind
  706. classes - which depend on pre-bind class entries existent in the class table */
  707. if (zend_hash_num_elements(&persistent_script->script.function_table) > 0) {
  708. zend_accel_function_hash_copy_from_shm(CG(function_table), &persistent_script->script.function_table);
  709. }
  710. /* Register __COMPILER_HALT_OFFSET__ constant */
  711. if (persistent_script->compiler_halt_offset != 0 &&
  712. persistent_script->script.filename) {
  713. zend_string *name;
  714. static const char haltoff[] = "__COMPILER_HALT_OFFSET__";
  715. name = zend_mangle_property_name(haltoff, sizeof(haltoff) - 1, ZSTR_VAL(persistent_script->script.filename), ZSTR_LEN(persistent_script->script.filename), 0);
  716. if (!zend_hash_exists(EG(zend_constants), name)) {
  717. zend_register_long_constant(ZSTR_VAL(name), ZSTR_LEN(name), persistent_script->compiler_halt_offset, CONST_CS, 0);
  718. }
  719. zend_string_release_ex(name, 0);
  720. }
  721. zend_hash_destroy(&ZCG(bind_hash));
  722. ZCG(current_persistent_script) = NULL;
  723. } else /* if (!from_shared_memory) */ {
  724. if (zend_hash_num_elements(&persistent_script->script.function_table) > 0) {
  725. zend_accel_function_hash_copy(CG(function_table), &persistent_script->script.function_table);
  726. }
  727. if (zend_hash_num_elements(&persistent_script->script.class_table) > 0) {
  728. zend_accel_class_hash_copy(CG(class_table), &persistent_script->script.class_table);
  729. }
  730. }
  731. if (persistent_script->script.first_early_binding_opline != (uint32_t)-1) {
  732. zend_string *orig_compiled_filename = CG(compiled_filename);
  733. CG(compiled_filename) = persistent_script->script.filename;
  734. zend_do_delayed_early_binding(op_array, persistent_script->script.first_early_binding_opline);
  735. CG(compiled_filename) = orig_compiled_filename;
  736. }
  737. if (UNEXPECTED(!from_shared_memory)) {
  738. free_persistent_script(persistent_script, 0); /* free only hashes */
  739. }
  740. return op_array;
  741. }
  742. /*
  743. * zend_adler32() is based on zlib implementation
  744. * Computes the Adler-32 checksum of a data stream
  745. *
  746. * Copyright (C) 1995-2005 Mark Adler
  747. * For conditions of distribution and use, see copyright notice in zlib.h
  748. *
  749. * Copyright (C) 1995-2005 Jean-loup Gailly and Mark Adler
  750. *
  751. * This software is provided 'as-is', without any express or implied
  752. * warranty. In no event will the authors be held liable for any damages
  753. * arising from the use of this software.
  754. *
  755. * Permission is granted to anyone to use this software for any purpose,
  756. * including commercial applications, and to alter it and redistribute it
  757. * freely, subject to the following restrictions:
  758. *
  759. * 1. The origin of this software must not be misrepresented; you must not
  760. * claim that you wrote the original software. If you use this software
  761. * in a product, an acknowledgment in the product documentation would be
  762. * appreciated but is not required.
  763. * 2. Altered source versions must be plainly marked as such, and must not be
  764. * misrepresented as being the original software.
  765. * 3. This notice may not be removed or altered from any source distribution.
  766. *
  767. */
  768. #define ADLER32_BASE 65521 /* largest prime smaller than 65536 */
  769. #define ADLER32_NMAX 5552
  770. /* NMAX is the largest n such that 255n(n+1)/2 + (n+1)(BASE-1) <= 2^32-1 */
  771. #define ADLER32_DO1(buf) {s1 += *(buf); s2 += s1;}
  772. #define ADLER32_DO2(buf, i) ADLER32_DO1(buf + i); ADLER32_DO1(buf + i + 1);
  773. #define ADLER32_DO4(buf, i) ADLER32_DO2(buf, i); ADLER32_DO2(buf, i + 2);
  774. #define ADLER32_DO8(buf, i) ADLER32_DO4(buf, i); ADLER32_DO4(buf, i + 4);
  775. #define ADLER32_DO16(buf) ADLER32_DO8(buf, 0); ADLER32_DO8(buf, 8);
  776. unsigned int zend_adler32(unsigned int checksum, unsigned char *buf, uint32_t len)
  777. {
  778. unsigned int s1 = checksum & 0xffff;
  779. unsigned int s2 = (checksum >> 16) & 0xffff;
  780. unsigned char *end;
  781. while (len >= ADLER32_NMAX) {
  782. len -= ADLER32_NMAX;
  783. end = buf + ADLER32_NMAX;
  784. do {
  785. ADLER32_DO16(buf);
  786. buf += 16;
  787. } while (buf != end);
  788. s1 %= ADLER32_BASE;
  789. s2 %= ADLER32_BASE;
  790. }
  791. if (len) {
  792. if (len >= 16) {
  793. end = buf + (len & 0xfff0);
  794. len &= 0xf;
  795. do {
  796. ADLER32_DO16(buf);
  797. buf += 16;
  798. } while (buf != end);
  799. }
  800. if (len) {
  801. end = buf + len;
  802. do {
  803. ADLER32_DO1(buf);
  804. buf++;
  805. } while (buf != end);
  806. }
  807. s1 %= ADLER32_BASE;
  808. s2 %= ADLER32_BASE;
  809. }
  810. return (s2 << 16) | s1;
  811. }
  812. unsigned int zend_accel_script_checksum(zend_persistent_script *persistent_script)
  813. {
  814. unsigned char *mem = (unsigned char*)persistent_script->mem;
  815. size_t size = persistent_script->size;
  816. size_t persistent_script_check_block_size = ((char *)&(persistent_script->dynamic_members)) - (char *)persistent_script;
  817. unsigned int checksum = ADLER32_INIT;
  818. if (mem < (unsigned char*)persistent_script) {
  819. checksum = zend_adler32(checksum, mem, (unsigned char*)persistent_script - mem);
  820. size -= (unsigned char*)persistent_script - mem;
  821. mem += (unsigned char*)persistent_script - mem;
  822. }
  823. zend_adler32(checksum, mem, persistent_script_check_block_size);
  824. mem += sizeof(*persistent_script);
  825. size -= sizeof(*persistent_script);
  826. if (size > 0) {
  827. checksum = zend_adler32(checksum, mem, size);
  828. }
  829. return checksum;
  830. }