zend_persist.c 23 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721
  1. /*
  2. +----------------------------------------------------------------------+
  3. | Zend OPcache |
  4. +----------------------------------------------------------------------+
  5. | Copyright (c) 1998-2016 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@zend.com> |
  16. | Zeev Suraski <zeev@zend.com> |
  17. | Stanislav Malyshev <stas@zend.com> |
  18. | Dmitry Stogov <dmitry@zend.com> |
  19. +----------------------------------------------------------------------+
  20. */
  21. #include "zend.h"
  22. #include "ZendAccelerator.h"
  23. #include "zend_persist.h"
  24. #include "zend_extensions.h"
  25. #include "zend_shared_alloc.h"
  26. #include "zend_vm.h"
  27. #include "zend_constants.h"
  28. #include "zend_operators.h"
  29. #define zend_accel_store(p, size) \
  30. (p = _zend_shared_memdup((void*)p, size, 1 TSRMLS_CC))
  31. #define zend_accel_memdup(p, size) \
  32. _zend_shared_memdup((void*)p, size, 0 TSRMLS_CC)
  33. #if ZEND_EXTENSION_API_NO > PHP_5_3_X_API_NO
  34. # define zend_accel_memdup_interned_string(str, len) \
  35. IS_INTERNED(str) ? str : zend_accel_memdup(str, len)
  36. # define zend_accel_store_interned_string(str, len) do { \
  37. if (!IS_INTERNED(str)) { zend_accel_store(str, len); } \
  38. } while (0)
  39. #else
  40. # define zend_accel_memdup_interned_string(str, len) \
  41. zend_accel_memdup(str, len)
  42. # define zend_accel_store_interned_string(str, len) \
  43. zend_accel_store(str, len)
  44. #endif
  45. typedef void (*zend_persist_func_t)(void * TSRMLS_DC);
  46. static void zend_persist_zval_ptr(zval **zp TSRMLS_DC);
  47. static void zend_persist_zval(zval *z TSRMLS_DC);
  48. #if ZEND_EXTENSION_API_NO > PHP_5_3_X_API_NO
  49. static const Bucket *uninitialized_bucket = NULL;
  50. #endif
  51. static void zend_hash_persist(HashTable *ht, void (*pPersistElement)(void *pElement TSRMLS_DC), size_t el_size TSRMLS_DC)
  52. {
  53. Bucket *p = ht->pListHead;
  54. uint i;
  55. while (p) {
  56. Bucket *q = p;
  57. /* persist bucket and key */
  58. #if ZEND_EXTENSION_API_NO > PHP_5_3_X_API_NO
  59. p = zend_accel_memdup(p, sizeof(Bucket));
  60. if (p->nKeyLength) {
  61. p->arKey = zend_accel_memdup_interned_string(p->arKey, p->nKeyLength);
  62. }
  63. #else
  64. p = zend_accel_memdup(p, sizeof(Bucket) - 1 + p->nKeyLength);
  65. #endif
  66. /* persist data pointer in bucket */
  67. if (!p->pDataPtr) {
  68. zend_accel_store(p->pData, el_size);
  69. } else {
  70. /* Update p->pData to point to the new p->pDataPtr address, after the bucket relocation */
  71. p->pData = &p->pDataPtr;
  72. }
  73. /* persist the data itself */
  74. if (pPersistElement) {
  75. pPersistElement(p->pData TSRMLS_CC);
  76. }
  77. /* update linked lists */
  78. if (p->pLast) {
  79. p->pLast->pNext = p;
  80. }
  81. if (p->pNext) {
  82. p->pNext->pLast = p;
  83. }
  84. if (p->pListLast) {
  85. p->pListLast->pListNext = p;
  86. }
  87. if (p->pListNext) {
  88. p->pListNext->pListLast = p;
  89. }
  90. p = p->pListNext;
  91. /* delete the old non-persistent bucket */
  92. efree(q);
  93. }
  94. /* update linked lists */
  95. if (ht->pListHead) {
  96. ht->pListHead = zend_shared_alloc_get_xlat_entry(ht->pListHead);
  97. }
  98. if (ht->pListTail) {
  99. ht->pListTail = zend_shared_alloc_get_xlat_entry(ht->pListTail);
  100. }
  101. if (ht->pInternalPointer) {
  102. ht->pInternalPointer = zend_shared_alloc_get_xlat_entry(ht->pInternalPointer);
  103. }
  104. #if ZEND_EXTENSION_API_NO > PHP_5_3_X_API_NO
  105. /* Check if HastTable is initialized */
  106. if (ht->nTableMask) {
  107. #endif
  108. if (ht->nNumOfElements) {
  109. /* update hash table */
  110. for (i = 0; i < ht->nTableSize; i++) {
  111. if (ht->arBuckets[i]) {
  112. ht->arBuckets[i] = zend_shared_alloc_get_xlat_entry(ht->arBuckets[i]);
  113. }
  114. }
  115. }
  116. zend_accel_store(ht->arBuckets, sizeof(Bucket*) * ht->nTableSize);
  117. #if ZEND_EXTENSION_API_NO > PHP_5_3_X_API_NO
  118. } else {
  119. ht->arBuckets = (Bucket**)&uninitialized_bucket;
  120. }
  121. #endif
  122. }
  123. #if ZEND_EXTENSION_API_NO > PHP_5_5_X_API_NO
  124. static zend_ast *zend_persist_ast(zend_ast *ast TSRMLS_DC)
  125. {
  126. int i;
  127. zend_ast *node;
  128. if (ast->kind == ZEND_CONST) {
  129. node = zend_accel_memdup(ast, sizeof(zend_ast) + sizeof(zval));
  130. node->u.val = (zval*)(node + 1);
  131. zend_persist_zval(node->u.val TSRMLS_CC);
  132. } else {
  133. node = zend_accel_memdup(ast, sizeof(zend_ast) + sizeof(zend_ast*) * (ast->children - 1));
  134. for (i = 0; i < ast->children; i++) {
  135. if ((&node->u.child)[i]) {
  136. (&node->u.child)[i] = zend_persist_ast((&node->u.child)[i] TSRMLS_CC);
  137. }
  138. }
  139. }
  140. efree(ast);
  141. return node;
  142. }
  143. #endif
  144. static void zend_persist_zval(zval *z TSRMLS_DC)
  145. {
  146. #if ZEND_EXTENSION_API_NO >= PHP_5_3_X_API_NO
  147. switch (z->type & IS_CONSTANT_TYPE_MASK) {
  148. #else
  149. switch (z->type & ~IS_CONSTANT_INDEX) {
  150. #endif
  151. case IS_STRING:
  152. case IS_CONSTANT:
  153. zend_accel_store_interned_string(z->value.str.val, z->value.str.len + 1);
  154. break;
  155. case IS_ARRAY:
  156. #if ZEND_EXTENSION_API_NO <= PHP_5_5_API_NO
  157. case IS_CONSTANT_ARRAY:
  158. #endif
  159. zend_accel_store(z->value.ht, sizeof(HashTable));
  160. zend_hash_persist(z->value.ht, (zend_persist_func_t) zend_persist_zval_ptr, sizeof(zval**) TSRMLS_CC);
  161. break;
  162. #if ZEND_EXTENSION_API_NO > PHP_5_5_X_API_NO
  163. case IS_CONSTANT_AST:
  164. Z_AST_P(z) = zend_persist_ast(Z_AST_P(z) TSRMLS_CC);
  165. break;
  166. #endif
  167. }
  168. }
  169. static void zend_persist_zval_ptr(zval **zp TSRMLS_DC)
  170. {
  171. zval *new_ptr = zend_shared_alloc_get_xlat_entry(*zp);
  172. if (new_ptr) {
  173. *zp = new_ptr;
  174. } else {
  175. /* Attempt to store only if we didn't store this zval_ptr yet */
  176. zend_accel_store(*zp, sizeof(zval));
  177. zend_persist_zval(*zp TSRMLS_CC);
  178. }
  179. }
  180. static void zend_protect_zval(zval *z TSRMLS_DC)
  181. {
  182. PZ_SET_ISREF_P(z);
  183. PZ_SET_REFCOUNT_P(z, 2);
  184. }
  185. static void zend_persist_op_array_ex(zend_op_array *op_array, zend_persistent_script* main_persistent_script TSRMLS_DC)
  186. {
  187. zend_op *persist_ptr;
  188. #if ZEND_EXTENSION_API_NO < PHP_5_3_X_API_NO
  189. int has_jmp = 0;
  190. #endif
  191. #if ZEND_EXTENSION_API_NO > PHP_5_3_X_API_NO
  192. zend_literal *orig_literals = NULL;
  193. #endif
  194. if (op_array->type != ZEND_USER_FUNCTION) {
  195. return;
  196. }
  197. #if ZEND_EXTENSION_API_NO <= PHP_5_3_X_API_NO
  198. op_array->size = op_array->last;
  199. #endif
  200. if (--(*op_array->refcount) == 0) {
  201. efree(op_array->refcount);
  202. }
  203. op_array->refcount = NULL;
  204. if (op_array->filename) {
  205. /* do not free! PHP has centralized filename storage, compiler will free it */
  206. op_array->filename = zend_accel_memdup(op_array->filename, strlen(op_array->filename) + 1);
  207. }
  208. if (main_persistent_script) {
  209. zend_bool orig_in_execution = EG(in_execution);
  210. zend_op_array *orig_op_array = EG(active_op_array);
  211. zval offset;
  212. #if ZEND_EXTENSION_API_NO < PHP_5_3_X_API_NO
  213. main_persistent_script->early_binding = -1;
  214. #endif
  215. EG(in_execution) = 1;
  216. EG(active_op_array) = op_array;
  217. if (zend_get_constant("__COMPILER_HALT_OFFSET__", sizeof("__COMPILER_HALT_OFFSET__") - 1, &offset TSRMLS_CC)) {
  218. main_persistent_script->compiler_halt_offset = Z_LVAL(offset);
  219. }
  220. EG(active_op_array) = orig_op_array;
  221. EG(in_execution) = orig_in_execution;
  222. }
  223. #if ZEND_EXTENSION_API_NO > PHP_5_3_X_API_NO
  224. if (op_array->literals) {
  225. orig_literals = zend_shared_alloc_get_xlat_entry(op_array->literals);
  226. if (orig_literals) {
  227. op_array->literals = orig_literals;
  228. } else {
  229. zend_literal *p = zend_accel_memdup(op_array->literals, sizeof(zend_literal) * op_array->last_literal);
  230. zend_literal *end = p + op_array->last_literal;
  231. orig_literals = op_array->literals;
  232. op_array->literals = p;
  233. while (p < end) {
  234. zend_persist_zval(&p->constant TSRMLS_CC);
  235. zend_protect_zval(&p->constant TSRMLS_CC);
  236. p++;
  237. }
  238. efree(orig_literals);
  239. }
  240. }
  241. #endif
  242. if ((persist_ptr = zend_shared_alloc_get_xlat_entry(op_array->opcodes))) {
  243. op_array->opcodes = persist_ptr;
  244. } else {
  245. zend_op *new_opcodes = zend_accel_memdup(op_array->opcodes, sizeof(zend_op) * op_array->last);
  246. zend_op *opline = new_opcodes;
  247. zend_op *end = new_opcodes + op_array->last;
  248. int offset = 0;
  249. for (; opline < end ; opline++, offset++) {
  250. if (ZEND_OP1_TYPE(opline) == IS_CONST) {
  251. #if ZEND_EXTENSION_API_NO > PHP_5_3_X_API_NO
  252. opline->op1.zv = (zval*)((char*)opline->op1.zv + ((char*)op_array->literals - (char*)orig_literals));
  253. #else
  254. zend_persist_zval(&opline->op1.u.constant TSRMLS_CC);
  255. zend_protect_zval(&opline->op1.u.constant TSRMLS_CC);
  256. #endif
  257. }
  258. if (ZEND_OP2_TYPE(opline) == IS_CONST) {
  259. #if ZEND_EXTENSION_API_NO > PHP_5_3_X_API_NO
  260. opline->op2.zv = (zval*)((char*)opline->op2.zv + ((char*)op_array->literals - (char*)orig_literals));
  261. #else
  262. zend_persist_zval(&opline->op2.u.constant TSRMLS_CC);
  263. zend_protect_zval(&opline->op2.u.constant TSRMLS_CC);
  264. #endif
  265. }
  266. #if ZEND_EXTENSION_API_NO < PHP_5_3_X_API_NO
  267. switch (opline->opcode) {
  268. case ZEND_JMP:
  269. has_jmp = 1;
  270. if (ZEND_DONE_PASS_TWO(op_array)) {
  271. ZEND_OP1(opline).jmp_addr = &new_opcodes[ZEND_OP1(opline).jmp_addr - op_array->opcodes];
  272. }
  273. break;
  274. case ZEND_JMPZ:
  275. case ZEND_JMPNZ:
  276. case ZEND_JMPZ_EX:
  277. case ZEND_JMPNZ_EX:
  278. has_jmp = 1;
  279. if (ZEND_DONE_PASS_TWO(op_array)) {
  280. ZEND_OP2(opline).jmp_addr = &new_opcodes[ZEND_OP2(opline).jmp_addr - op_array->opcodes];
  281. }
  282. break;
  283. case ZEND_JMPZNZ:
  284. case ZEND_BRK:
  285. case ZEND_CONT:
  286. has_jmp = 1;
  287. break;
  288. case ZEND_DECLARE_INHERITED_CLASS:
  289. if (main_persistent_script && ZCG(accel_directives).inherited_hack) {
  290. if (!has_jmp &&
  291. ((opline + 2) >= end ||
  292. (opline + 1)->opcode != ZEND_FETCH_CLASS ||
  293. (opline + 2)->opcode != ZEND_ADD_INTERFACE)) {
  294. zend_uint *opline_num = &main_persistent_script->early_binding;
  295. while ((int)*opline_num != -1) {
  296. opline_num = &new_opcodes[*opline_num].result.u.opline_num;
  297. }
  298. *opline_num = opline - new_opcodes;
  299. opline->result.op_type = IS_UNUSED;
  300. opline->result.u.opline_num = -1;
  301. opline->opcode = ZEND_DECLARE_INHERITED_CLASS_DELAYED;
  302. ZEND_VM_SET_OPCODE_HANDLER(opline);
  303. }
  304. break;
  305. }
  306. }
  307. #else /* if ZEND_EXTENSION_API_NO >= PHP_5_3_X_API_NO */
  308. if (ZEND_DONE_PASS_TWO(op_array)) {
  309. /* fix jumps to point to new array */
  310. switch (opline->opcode) {
  311. case ZEND_JMP:
  312. case ZEND_GOTO:
  313. #if ZEND_EXTENSION_API_NO > PHP_5_4_X_API_NO
  314. case ZEND_FAST_CALL:
  315. #endif
  316. ZEND_OP1(opline).jmp_addr = &new_opcodes[ZEND_OP1(opline).jmp_addr - op_array->opcodes];
  317. break;
  318. case ZEND_JMPZ:
  319. case ZEND_JMPNZ:
  320. case ZEND_JMPZ_EX:
  321. case ZEND_JMPNZ_EX:
  322. case ZEND_JMP_SET:
  323. #if ZEND_EXTENSION_API_NO > PHP_5_3_X_API_NO
  324. case ZEND_JMP_SET_VAR:
  325. #endif
  326. ZEND_OP2(opline).jmp_addr = &new_opcodes[ZEND_OP2(opline).jmp_addr - op_array->opcodes];
  327. break;
  328. }
  329. }
  330. #endif /* if ZEND_EXTENSION_API_NO >= PHP_5_3_X_API_NO */
  331. }
  332. efree(op_array->opcodes);
  333. op_array->opcodes = new_opcodes;
  334. #if ZEND_EXTENSION_API_NO > PHP_5_3_X_API_NO
  335. if (op_array->run_time_cache) {
  336. efree(op_array->run_time_cache);
  337. op_array->run_time_cache = NULL;
  338. }
  339. #endif
  340. }
  341. if (op_array->function_name) {
  342. char *new_name;
  343. if ((new_name = zend_shared_alloc_get_xlat_entry(op_array->function_name))) {
  344. op_array->function_name = new_name;
  345. } else {
  346. zend_accel_store(op_array->function_name, strlen(op_array->function_name) + 1);
  347. }
  348. }
  349. if (op_array->arg_info) {
  350. zend_arg_info *new_ptr;
  351. if ((new_ptr = zend_shared_alloc_get_xlat_entry(op_array->arg_info))) {
  352. op_array->arg_info = new_ptr;
  353. } else {
  354. zend_uint i;
  355. zend_accel_store(op_array->arg_info, sizeof(zend_arg_info) * op_array->num_args);
  356. for (i = 0; i < op_array->num_args; i++) {
  357. if (op_array->arg_info[i].name) {
  358. zend_accel_store_interned_string(op_array->arg_info[i].name, op_array->arg_info[i].name_len + 1);
  359. }
  360. if (op_array->arg_info[i].class_name) {
  361. zend_accel_store_interned_string(op_array->arg_info[i].class_name, op_array->arg_info[i].class_name_len + 1);
  362. }
  363. }
  364. }
  365. }
  366. if (op_array->brk_cont_array) {
  367. zend_accel_store(op_array->brk_cont_array, sizeof(zend_brk_cont_element) * op_array->last_brk_cont);
  368. }
  369. if (op_array->static_variables) {
  370. zend_hash_persist(op_array->static_variables, (zend_persist_func_t) zend_persist_zval_ptr, sizeof(zval**) TSRMLS_CC);
  371. zend_accel_store(op_array->static_variables, sizeof(HashTable));
  372. }
  373. if (op_array->scope) {
  374. op_array->scope = zend_shared_alloc_get_xlat_entry(op_array->scope);
  375. }
  376. if (op_array->doc_comment) {
  377. if (ZCG(accel_directives).save_comments) {
  378. zend_accel_store(op_array->doc_comment, op_array->doc_comment_len + 1);
  379. } else {
  380. if (!zend_shared_alloc_get_xlat_entry(op_array->doc_comment)) {
  381. zend_shared_alloc_register_xlat_entry(op_array->doc_comment, op_array->doc_comment);
  382. efree((char*)op_array->doc_comment);
  383. }
  384. op_array->doc_comment = NULL;
  385. op_array->doc_comment_len = 0;
  386. }
  387. }
  388. if (op_array->try_catch_array) {
  389. zend_accel_store(op_array->try_catch_array, sizeof(zend_try_catch_element) * op_array->last_try_catch);
  390. }
  391. if (op_array->vars) {
  392. if ((persist_ptr = zend_shared_alloc_get_xlat_entry(op_array->vars))) {
  393. op_array->vars = (zend_compiled_variable*)persist_ptr;
  394. } else {
  395. int i;
  396. zend_accel_store(op_array->vars, sizeof(zend_compiled_variable) * op_array->last_var);
  397. for (i = 0; i < op_array->last_var; i++) {
  398. zend_accel_store_interned_string(op_array->vars[i].name, op_array->vars[i].name_len + 1);
  399. }
  400. }
  401. }
  402. /* "prototype" may be undefined if "scope" isn't set */
  403. if (op_array->scope && op_array->prototype) {
  404. if ((persist_ptr = zend_shared_alloc_get_xlat_entry(op_array->prototype))) {
  405. op_array->prototype = (union _zend_function*)persist_ptr;
  406. /* we use refcount to show that op_array is referenced from several places */
  407. op_array->prototype->op_array.refcount++;
  408. }
  409. } else {
  410. op_array->prototype = NULL;
  411. }
  412. }
  413. static void zend_persist_op_array(zend_op_array *op_array TSRMLS_DC)
  414. {
  415. zend_persist_op_array_ex(op_array, NULL TSRMLS_CC);
  416. }
  417. static void zend_persist_property_info(zend_property_info *prop TSRMLS_DC)
  418. {
  419. zend_accel_store_interned_string(prop->name, prop->name_length + 1);
  420. if (prop->doc_comment) {
  421. if (ZCG(accel_directives).save_comments) {
  422. zend_accel_store(prop->doc_comment, prop->doc_comment_len + 1);
  423. } else {
  424. if (!zend_shared_alloc_get_xlat_entry(prop->doc_comment)) {
  425. zend_shared_alloc_register_xlat_entry(prop->doc_comment, prop->doc_comment);
  426. efree((char*)prop->doc_comment);
  427. }
  428. prop->doc_comment = NULL;
  429. prop->doc_comment_len = 0;
  430. }
  431. }
  432. }
  433. static void zend_persist_class_entry(zend_class_entry **pce TSRMLS_DC)
  434. {
  435. zend_class_entry *ce = *pce;
  436. if (ce->type == ZEND_USER_CLASS) {
  437. *pce = zend_accel_store(ce, sizeof(zend_class_entry));
  438. zend_accel_store_interned_string(ce->name, ce->name_length + 1);
  439. zend_hash_persist(&ce->function_table, (zend_persist_func_t) zend_persist_op_array, sizeof(zend_op_array) TSRMLS_CC);
  440. #if ZEND_EXTENSION_API_NO > PHP_5_3_X_API_NO
  441. if (ce->default_properties_table) {
  442. int i;
  443. zend_accel_store(ce->default_properties_table, sizeof(zval*) * ce->default_properties_count);
  444. for (i = 0; i < ce->default_properties_count; i++) {
  445. if (ce->default_properties_table[i]) {
  446. zend_persist_zval_ptr(&ce->default_properties_table[i] TSRMLS_CC);
  447. }
  448. }
  449. }
  450. if (ce->default_static_members_table) {
  451. int i;
  452. zend_accel_store(ce->default_static_members_table, sizeof(zval*) * ce->default_static_members_count);
  453. for (i = 0; i < ce->default_static_members_count; i++) {
  454. if (ce->default_static_members_table[i]) {
  455. zend_persist_zval_ptr(&ce->default_static_members_table[i] TSRMLS_CC);
  456. }
  457. }
  458. }
  459. ce->static_members_table = NULL;
  460. #else
  461. zend_hash_persist(&ce->default_properties, (zend_persist_func_t) zend_persist_zval_ptr, sizeof(zval**) TSRMLS_CC);
  462. zend_hash_persist(&ce->default_static_members, (zend_persist_func_t) zend_persist_zval_ptr, sizeof(zval**) TSRMLS_CC);
  463. ce->static_members = NULL;
  464. #endif
  465. zend_hash_persist(&ce->constants_table, (zend_persist_func_t) zend_persist_zval_ptr, sizeof(zval**) TSRMLS_CC);
  466. if (ZEND_CE_FILENAME(ce)) {
  467. /* do not free! PHP has centralized filename storage, compiler will free it */
  468. ZEND_CE_FILENAME(ce) = zend_accel_memdup(ZEND_CE_FILENAME(ce), strlen(ZEND_CE_FILENAME(ce)) + 1);
  469. }
  470. if (ZEND_CE_DOC_COMMENT(ce)) {
  471. if (ZCG(accel_directives).save_comments) {
  472. zend_accel_store(ZEND_CE_DOC_COMMENT(ce), ZEND_CE_DOC_COMMENT_LEN(ce) + 1);
  473. } else {
  474. if (!zend_shared_alloc_get_xlat_entry(ZEND_CE_DOC_COMMENT(ce))) {
  475. zend_shared_alloc_register_xlat_entry(ZEND_CE_DOC_COMMENT(ce), ZEND_CE_DOC_COMMENT(ce));
  476. efree((char*)ZEND_CE_DOC_COMMENT(ce));
  477. }
  478. ZEND_CE_DOC_COMMENT(ce) = NULL;
  479. ZEND_CE_DOC_COMMENT_LEN(ce) = 0;
  480. }
  481. }
  482. zend_hash_persist(&ce->properties_info, (zend_persist_func_t) zend_persist_property_info, sizeof(zend_property_info) TSRMLS_CC);
  483. if (ce->num_interfaces && ce->interfaces) {
  484. efree(ce->interfaces);
  485. }
  486. ce->interfaces = NULL; /* will be filled in on fetch */
  487. #if ZEND_EXTENSION_API_NO > PHP_5_3_X_API_NO
  488. if (ce->num_traits && ce->traits) {
  489. efree(ce->traits);
  490. }
  491. ce->traits = NULL;
  492. if (ce->trait_aliases) {
  493. int i = 0;
  494. while (ce->trait_aliases[i]) {
  495. if (ce->trait_aliases[i]->trait_method) {
  496. if (ce->trait_aliases[i]->trait_method->method_name) {
  497. zend_accel_store(ce->trait_aliases[i]->trait_method->method_name,
  498. ce->trait_aliases[i]->trait_method->mname_len + 1);
  499. }
  500. if (ce->trait_aliases[i]->trait_method->class_name) {
  501. zend_accel_store(ce->trait_aliases[i]->trait_method->class_name,
  502. ce->trait_aliases[i]->trait_method->cname_len + 1);
  503. }
  504. ce->trait_aliases[i]->trait_method->ce = NULL;
  505. zend_accel_store(ce->trait_aliases[i]->trait_method,
  506. sizeof(zend_trait_method_reference));
  507. }
  508. if (ce->trait_aliases[i]->alias) {
  509. zend_accel_store(ce->trait_aliases[i]->alias,
  510. ce->trait_aliases[i]->alias_len + 1);
  511. }
  512. #if ZEND_EXTENSION_API_NO <= PHP_5_4_X_API_NO
  513. ce->trait_aliases[i]->function = NULL;
  514. #endif
  515. zend_accel_store(ce->trait_aliases[i], sizeof(zend_trait_alias));
  516. i++;
  517. }
  518. zend_accel_store(ce->trait_aliases, sizeof(zend_trait_alias*) * (i + 1));
  519. }
  520. if (ce->trait_precedences) {
  521. int i = 0;
  522. while (ce->trait_precedences[i]) {
  523. zend_accel_store(ce->trait_precedences[i]->trait_method->method_name,
  524. ce->trait_precedences[i]->trait_method->mname_len + 1);
  525. zend_accel_store(ce->trait_precedences[i]->trait_method->class_name,
  526. ce->trait_precedences[i]->trait_method->cname_len + 1);
  527. ce->trait_precedences[i]->trait_method->ce = NULL;
  528. zend_accel_store(ce->trait_precedences[i]->trait_method,
  529. sizeof(zend_trait_method_reference));
  530. if (ce->trait_precedences[i]->exclude_from_classes) {
  531. int j = 0;
  532. while (ce->trait_precedences[i]->exclude_from_classes[j]) {
  533. zend_accel_store(ce->trait_precedences[i]->exclude_from_classes[j],
  534. strlen((char*)ce->trait_precedences[i]->exclude_from_classes[j]) + 1);
  535. j++;
  536. }
  537. zend_accel_store(ce->trait_precedences[i]->exclude_from_classes,
  538. sizeof(zend_class_entry*) * (j + 1));
  539. }
  540. #if ZEND_EXTENSION_API_NO <= PHP_5_4_X_API_NO
  541. ce->trait_precedences[i]->function = NULL;
  542. #endif
  543. zend_accel_store(ce->trait_precedences[i], sizeof(zend_trait_precedence));
  544. i++;
  545. }
  546. zend_accel_store(
  547. ce->trait_precedences, sizeof(zend_trait_precedence*) * (i + 1));
  548. }
  549. #endif
  550. }
  551. }
  552. static int zend_update_property_info_ce(zend_property_info *prop TSRMLS_DC)
  553. {
  554. prop->ce = zend_shared_alloc_get_xlat_entry(prop->ce);
  555. return 0;
  556. }
  557. static int zend_update_parent_ce(zend_class_entry **pce TSRMLS_DC)
  558. {
  559. zend_class_entry *ce = *pce;
  560. if (ce->parent) {
  561. ce->parent = zend_shared_alloc_get_xlat_entry(ce->parent);
  562. /* We use refcount to show if the class is used as a parent */
  563. ce->parent->refcount++;
  564. }
  565. /* update methods */
  566. if (ce->constructor) {
  567. ce->constructor = zend_shared_alloc_get_xlat_entry(ce->constructor);
  568. /* we use refcount to show that op_array is referenced from several places */
  569. ce->constructor->op_array.refcount++;
  570. }
  571. if (ce->destructor) {
  572. ce->destructor = zend_shared_alloc_get_xlat_entry(ce->destructor);
  573. ce->destructor->op_array.refcount++;
  574. }
  575. if (ce->clone) {
  576. ce->clone = zend_shared_alloc_get_xlat_entry(ce->clone);
  577. ce->clone->op_array.refcount++;
  578. }
  579. if (ce->__get) {
  580. ce->__get = zend_shared_alloc_get_xlat_entry(ce->__get);
  581. ce->__get->op_array.refcount++;
  582. }
  583. if (ce->__set) {
  584. ce->__set = zend_shared_alloc_get_xlat_entry(ce->__set);
  585. ce->__set->op_array.refcount++;
  586. }
  587. if (ce->__call) {
  588. ce->__call = zend_shared_alloc_get_xlat_entry(ce->__call);
  589. ce->__call->op_array.refcount++;
  590. }
  591. if (ce->serialize_func) {
  592. ce->serialize_func = zend_shared_alloc_get_xlat_entry(ce->serialize_func);
  593. ce->serialize_func->op_array.refcount++;
  594. }
  595. if (ce->unserialize_func) {
  596. ce->unserialize_func = zend_shared_alloc_get_xlat_entry(ce->unserialize_func);
  597. ce->unserialize_func->op_array.refcount++;
  598. }
  599. if (ce->__isset) {
  600. ce->__isset = zend_shared_alloc_get_xlat_entry(ce->__isset);
  601. ce->__isset->op_array.refcount++;
  602. }
  603. if (ce->__unset) {
  604. ce->__unset = zend_shared_alloc_get_xlat_entry(ce->__unset);
  605. ce->__unset->op_array.refcount++;
  606. }
  607. if (ce->__tostring) {
  608. ce->__tostring = zend_shared_alloc_get_xlat_entry(ce->__tostring);
  609. ce->__tostring->op_array.refcount++;
  610. }
  611. #if ZEND_EXTENSION_API_NO >= PHP_5_3_X_API_NO
  612. if (ce->__callstatic) {
  613. ce->__callstatic = zend_shared_alloc_get_xlat_entry(ce->__callstatic);
  614. ce->__callstatic->op_array.refcount++;
  615. }
  616. #endif
  617. #if ZEND_EXTENSION_API_NO >= PHP_5_6_X_API_NO
  618. if (ce->__debugInfo) {
  619. ce->__debugInfo = zend_shared_alloc_get_xlat_entry(ce->__debugInfo);
  620. ce->__debugInfo->op_array.refcount++;
  621. }
  622. #endif
  623. zend_hash_apply(&ce->properties_info, (apply_func_t) zend_update_property_info_ce TSRMLS_CC);
  624. return 0;
  625. }
  626. static void zend_accel_persist_class_table(HashTable *class_table TSRMLS_DC)
  627. {
  628. zend_hash_persist(class_table, (zend_persist_func_t) zend_persist_class_entry, sizeof(zend_class_entry*) TSRMLS_CC);
  629. zend_hash_apply(class_table, (apply_func_t) zend_update_parent_ce TSRMLS_CC);
  630. }
  631. zend_persistent_script *zend_accel_script_persist(zend_persistent_script *script, char **key, unsigned int key_length TSRMLS_DC)
  632. {
  633. zend_shared_alloc_clear_xlat_table();
  634. zend_hash_persist(&script->function_table, (zend_persist_func_t) zend_persist_op_array, sizeof(zend_op_array) TSRMLS_CC);
  635. zend_accel_persist_class_table(&script->class_table TSRMLS_CC);
  636. zend_persist_op_array_ex(&script->main_op_array, script TSRMLS_CC);
  637. *key = zend_accel_memdup(*key, key_length + 1);
  638. zend_accel_store(script->full_path, script->full_path_len + 1);
  639. zend_accel_store(script, sizeof(zend_persistent_script));
  640. return script;
  641. }
  642. int zend_accel_script_persistable(zend_persistent_script *script)
  643. {
  644. return 1;
  645. }