zend_persist.c 29 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928
  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.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))
  31. #define zend_accel_memdup(p, size) \
  32. _zend_shared_memdup((void*)p, size, 0)
  33. #ifdef HAVE_OPCACHE_FILE_CACHE
  34. #define zend_set_str_gc_flags(str) do { \
  35. if (file_cache_only) { \
  36. GC_TYPE_INFO(str) = IS_STRING | (IS_STR_INTERNED << GC_FLAGS_SHIFT); \
  37. } else { \
  38. GC_TYPE_INFO(str) = IS_STRING | ((IS_STR_INTERNED | IS_STR_PERMANENT) << GC_FLAGS_SHIFT); \
  39. } \
  40. } while (0)
  41. #else
  42. #define zend_set_str_gc_flags(str) do {\
  43. GC_TYPE_INFO(str) = IS_STRING | ((IS_STR_INTERNED | IS_STR_PERMANENT) << GC_FLAGS_SHIFT); \
  44. } while (0)
  45. #endif
  46. #define zend_accel_store_string(str) do { \
  47. zend_string *new_str = zend_shared_alloc_get_xlat_entry(str); \
  48. if (new_str) { \
  49. zend_string_release_ex(str, 0); \
  50. str = new_str; \
  51. } else { \
  52. new_str = zend_accel_memdup((void*)str, _ZSTR_STRUCT_SIZE(ZSTR_LEN(str))); \
  53. zend_string_release_ex(str, 0); \
  54. str = new_str; \
  55. zend_string_hash_val(str); \
  56. zend_set_str_gc_flags(str); \
  57. } \
  58. } while (0)
  59. #define zend_accel_memdup_string(str) do { \
  60. str = zend_accel_memdup(str, _ZSTR_STRUCT_SIZE(ZSTR_LEN(str))); \
  61. zend_string_hash_val(str); \
  62. zend_set_str_gc_flags(str); \
  63. } while (0)
  64. #define zend_accel_store_interned_string(str) do { \
  65. if (!IS_ACCEL_INTERNED(str)) { \
  66. zend_accel_store_string(str); \
  67. } \
  68. } while (0)
  69. #define zend_accel_memdup_interned_string(str) do { \
  70. if (!IS_ACCEL_INTERNED(str)) { \
  71. zend_accel_memdup_string(str); \
  72. } \
  73. } while (0)
  74. typedef void (*zend_persist_func_t)(zval*);
  75. static void zend_persist_zval(zval *z);
  76. static const uint32_t uninitialized_bucket[-HT_MIN_MASK] =
  77. {HT_INVALID_IDX, HT_INVALID_IDX};
  78. static void zend_hash_persist(HashTable *ht, zend_persist_func_t pPersistElement)
  79. {
  80. uint32_t idx, nIndex;
  81. Bucket *p;
  82. HT_FLAGS(ht) |= HASH_FLAG_STATIC_KEYS;
  83. ht->pDestructor = NULL;
  84. if (!(HT_FLAGS(ht) & HASH_FLAG_INITIALIZED)) {
  85. if (EXPECTED(!ZCG(current_persistent_script)->corrupted)) {
  86. HT_SET_DATA_ADDR(ht, &ZCSG(uninitialized_bucket));
  87. } else {
  88. HT_SET_DATA_ADDR(ht, &uninitialized_bucket);
  89. }
  90. return;
  91. }
  92. if (ht->nNumUsed == 0) {
  93. efree(HT_GET_DATA_ADDR(ht));
  94. ht->nTableMask = HT_MIN_MASK;
  95. if (EXPECTED(!ZCG(current_persistent_script)->corrupted)) {
  96. HT_SET_DATA_ADDR(ht, &ZCSG(uninitialized_bucket));
  97. } else {
  98. HT_SET_DATA_ADDR(ht, &uninitialized_bucket);
  99. }
  100. HT_FLAGS(ht) &= ~HASH_FLAG_INITIALIZED;
  101. return;
  102. }
  103. if (HT_FLAGS(ht) & HASH_FLAG_PACKED) {
  104. void *data = HT_GET_DATA_ADDR(ht);
  105. zend_accel_store(data, HT_USED_SIZE(ht));
  106. HT_SET_DATA_ADDR(ht, data);
  107. } else if (ht->nNumUsed < (uint32_t)(-(int32_t)ht->nTableMask) / 4) {
  108. /* compact table */
  109. void *old_data = HT_GET_DATA_ADDR(ht);
  110. Bucket *old_buckets = ht->arData;
  111. uint32_t hash_size;
  112. if (ht->nNumUsed <= HT_MIN_SIZE) {
  113. hash_size = HT_MIN_SIZE * 2;
  114. } else {
  115. hash_size = (uint32_t)(-(int32_t)ht->nTableMask);
  116. while (hash_size >> 2 > ht->nNumUsed) {
  117. hash_size >>= 1;
  118. }
  119. }
  120. ht->nTableMask = (uint32_t)(-(int32_t)hash_size);
  121. ZEND_ASSERT(((zend_uintptr_t)ZCG(mem) & 0x7) == 0); /* should be 8 byte aligned */
  122. HT_SET_DATA_ADDR(ht, ZCG(mem));
  123. ZCG(mem) = (void*)((char*)ZCG(mem) + ZEND_ALIGNED_SIZE((hash_size * sizeof(uint32_t)) + (ht->nNumUsed * sizeof(Bucket))));
  124. HT_HASH_RESET(ht);
  125. memcpy(ht->arData, old_buckets, ht->nNumUsed * sizeof(Bucket));
  126. efree(old_data);
  127. for (idx = 0; idx < ht->nNumUsed; idx++) {
  128. p = ht->arData + idx;
  129. if (Z_TYPE(p->val) == IS_UNDEF) continue;
  130. /* persist bucket and key */
  131. if (p->key) {
  132. zend_accel_store_interned_string(p->key);
  133. }
  134. /* persist the data itself */
  135. pPersistElement(&p->val);
  136. nIndex = p->h | ht->nTableMask;
  137. Z_NEXT(p->val) = HT_HASH(ht, nIndex);
  138. HT_HASH(ht, nIndex) = HT_IDX_TO_HASH(idx);
  139. }
  140. return;
  141. } else {
  142. void *data = ZCG(mem);
  143. void *old_data = HT_GET_DATA_ADDR(ht);
  144. ZEND_ASSERT(((zend_uintptr_t)ZCG(mem) & 0x7) == 0); /* should be 8 byte aligned */
  145. ZCG(mem) = (void*)((char*)data + ZEND_ALIGNED_SIZE(HT_USED_SIZE(ht)));
  146. memcpy(data, old_data, HT_USED_SIZE(ht));
  147. efree(old_data);
  148. HT_SET_DATA_ADDR(ht, data);
  149. }
  150. for (idx = 0; idx < ht->nNumUsed; idx++) {
  151. p = ht->arData + idx;
  152. if (Z_TYPE(p->val) == IS_UNDEF) continue;
  153. /* persist bucket and key */
  154. if (p->key) {
  155. zend_accel_store_interned_string(p->key);
  156. }
  157. /* persist the data itself */
  158. pPersistElement(&p->val);
  159. }
  160. }
  161. static void zend_hash_persist_immutable(HashTable *ht)
  162. {
  163. uint32_t idx, nIndex;
  164. Bucket *p;
  165. HT_FLAGS(ht) |= HASH_FLAG_STATIC_KEYS;
  166. ht->pDestructor = NULL;
  167. if (!(HT_FLAGS(ht) & HASH_FLAG_INITIALIZED)) {
  168. if (EXPECTED(!ZCG(current_persistent_script)->corrupted)) {
  169. HT_SET_DATA_ADDR(ht, &ZCSG(uninitialized_bucket));
  170. } else {
  171. HT_SET_DATA_ADDR(ht, &uninitialized_bucket);
  172. }
  173. return;
  174. }
  175. if (ht->nNumUsed == 0) {
  176. efree(HT_GET_DATA_ADDR(ht));
  177. ht->nTableMask = HT_MIN_MASK;
  178. if (EXPECTED(!ZCG(current_persistent_script)->corrupted)) {
  179. HT_SET_DATA_ADDR(ht, &ZCSG(uninitialized_bucket));
  180. } else {
  181. HT_SET_DATA_ADDR(ht, &uninitialized_bucket);
  182. }
  183. HT_FLAGS(ht) &= ~HASH_FLAG_INITIALIZED;
  184. return;
  185. }
  186. if (HT_FLAGS(ht) & HASH_FLAG_PACKED) {
  187. HT_SET_DATA_ADDR(ht, zend_accel_memdup(HT_GET_DATA_ADDR(ht), HT_USED_SIZE(ht)));
  188. } else if (ht->nNumUsed < (uint32_t)(-(int32_t)ht->nTableMask) / 4) {
  189. /* compact table */
  190. void *old_data = HT_GET_DATA_ADDR(ht);
  191. Bucket *old_buckets = ht->arData;
  192. uint32_t hash_size;
  193. if (ht->nNumUsed <= HT_MIN_SIZE) {
  194. hash_size = HT_MIN_SIZE * 2;
  195. } else {
  196. hash_size = (uint32_t)(-(int32_t)ht->nTableMask);
  197. while (hash_size >> 2 > ht->nNumUsed) {
  198. hash_size >>= 1;
  199. }
  200. }
  201. ht->nTableMask = (uint32_t)(-(int32_t)hash_size);
  202. ZEND_ASSERT(((zend_uintptr_t)ZCG(mem) & 0x7) == 0); /* should be 8 byte aligned */
  203. HT_SET_DATA_ADDR(ht, ZCG(mem));
  204. ZCG(mem) = (void*)((char*)ZCG(mem) + (hash_size * sizeof(uint32_t)) + (ht->nNumUsed * sizeof(Bucket)));
  205. HT_HASH_RESET(ht);
  206. memcpy(ht->arData, old_buckets, ht->nNumUsed * sizeof(Bucket));
  207. efree(old_data);
  208. for (idx = 0; idx < ht->nNumUsed; idx++) {
  209. p = ht->arData + idx;
  210. if (Z_TYPE(p->val) == IS_UNDEF) continue;
  211. /* persist bucket and key */
  212. if (p->key) {
  213. zend_accel_memdup_interned_string(p->key);
  214. }
  215. /* persist the data itself */
  216. zend_persist_zval(&p->val);
  217. nIndex = p->h | ht->nTableMask;
  218. Z_NEXT(p->val) = HT_HASH(ht, nIndex);
  219. HT_HASH(ht, nIndex) = HT_IDX_TO_HASH(idx);
  220. }
  221. return;
  222. } else {
  223. void *data = ZCG(mem);
  224. ZEND_ASSERT(((zend_uintptr_t)ZCG(mem) & 0x7) == 0); /* should be 8 byte aligned */
  225. ZCG(mem) = (void*)((char*)data + ZEND_ALIGNED_SIZE(HT_USED_SIZE(ht)));
  226. memcpy(data, HT_GET_DATA_ADDR(ht), HT_USED_SIZE(ht));
  227. HT_SET_DATA_ADDR(ht, data);
  228. }
  229. for (idx = 0; idx < ht->nNumUsed; idx++) {
  230. p = ht->arData + idx;
  231. if (Z_TYPE(p->val) == IS_UNDEF) continue;
  232. /* persist bucket and key */
  233. if (p->key) {
  234. zend_accel_memdup_interned_string(p->key);
  235. }
  236. /* persist the data itself */
  237. zend_persist_zval(&p->val);
  238. }
  239. }
  240. static zend_ast *zend_persist_ast(zend_ast *ast)
  241. {
  242. uint32_t i;
  243. zend_ast *node;
  244. if (ast->kind == ZEND_AST_ZVAL || ast->kind == ZEND_AST_CONSTANT) {
  245. zend_ast_zval *copy = zend_accel_memdup(ast, sizeof(zend_ast_zval));
  246. zend_persist_zval(&copy->val);
  247. node = (zend_ast *) copy;
  248. } else if (zend_ast_is_list(ast)) {
  249. zend_ast_list *list = zend_ast_get_list(ast);
  250. zend_ast_list *copy = zend_accel_memdup(ast,
  251. sizeof(zend_ast_list) - sizeof(zend_ast *) + sizeof(zend_ast *) * list->children);
  252. for (i = 0; i < list->children; i++) {
  253. if (copy->child[i]) {
  254. copy->child[i] = zend_persist_ast(copy->child[i]);
  255. }
  256. }
  257. node = (zend_ast *) copy;
  258. } else {
  259. uint32_t children = zend_ast_get_num_children(ast);
  260. node = zend_accel_memdup(ast, sizeof(zend_ast) - sizeof(zend_ast *) + sizeof(zend_ast *) * children);
  261. for (i = 0; i < children; i++) {
  262. if (node->child[i]) {
  263. node->child[i] = zend_persist_ast(node->child[i]);
  264. }
  265. }
  266. }
  267. return node;
  268. }
  269. static void zend_persist_zval(zval *z)
  270. {
  271. void *new_ptr;
  272. switch (Z_TYPE_P(z)) {
  273. case IS_STRING:
  274. zend_accel_store_interned_string(Z_STR_P(z));
  275. Z_TYPE_FLAGS_P(z) = 0;
  276. break;
  277. case IS_ARRAY:
  278. new_ptr = zend_shared_alloc_get_xlat_entry(Z_ARR_P(z));
  279. if (new_ptr) {
  280. Z_ARR_P(z) = new_ptr;
  281. Z_TYPE_FLAGS_P(z) = 0;
  282. } else {
  283. if (!Z_REFCOUNTED_P(z)) {
  284. Z_ARR_P(z) = zend_accel_memdup(Z_ARR_P(z), sizeof(zend_array));
  285. zend_hash_persist_immutable(Z_ARRVAL_P(z));
  286. } else {
  287. GC_REMOVE_FROM_BUFFER(Z_ARR_P(z));
  288. zend_accel_store(Z_ARR_P(z), sizeof(zend_array));
  289. zend_hash_persist(Z_ARRVAL_P(z), zend_persist_zval);
  290. /* make immutable array */
  291. Z_TYPE_FLAGS_P(z) = 0;
  292. GC_SET_REFCOUNT(Z_COUNTED_P(z), 2);
  293. GC_ADD_FLAGS(Z_COUNTED_P(z), IS_ARRAY_IMMUTABLE);
  294. }
  295. }
  296. break;
  297. case IS_REFERENCE:
  298. new_ptr = zend_shared_alloc_get_xlat_entry(Z_REF_P(z));
  299. if (new_ptr) {
  300. Z_REF_P(z) = new_ptr;
  301. } else {
  302. zend_accel_store(Z_REF_P(z), sizeof(zend_reference));
  303. zend_persist_zval(Z_REFVAL_P(z));
  304. }
  305. break;
  306. case IS_CONSTANT_AST:
  307. new_ptr = zend_shared_alloc_get_xlat_entry(Z_AST_P(z));
  308. if (new_ptr) {
  309. Z_AST_P(z) = new_ptr;
  310. Z_TYPE_FLAGS_P(z) = 0;
  311. } else {
  312. zend_ast_ref *old_ref = Z_AST_P(z);
  313. Z_ARR_P(z) = zend_accel_memdup(Z_AST_P(z), sizeof(zend_ast_ref));
  314. zend_persist_ast(GC_AST(old_ref));
  315. Z_TYPE_FLAGS_P(z) = 0;
  316. GC_SET_REFCOUNT(Z_COUNTED_P(z), 1);
  317. efree(old_ref);
  318. }
  319. break;
  320. }
  321. }
  322. static void zend_persist_op_array_ex(zend_op_array *op_array, zend_persistent_script* main_persistent_script)
  323. {
  324. int already_stored = 0;
  325. zend_op *persist_ptr;
  326. zval *orig_literals = NULL;
  327. if (op_array->refcount && --(*op_array->refcount) == 0) {
  328. efree(op_array->refcount);
  329. }
  330. op_array->refcount = NULL;
  331. if (main_persistent_script) {
  332. zend_execute_data *orig_execute_data = EG(current_execute_data);
  333. zend_execute_data fake_execute_data;
  334. zval *offset;
  335. memset(&fake_execute_data, 0, sizeof(fake_execute_data));
  336. fake_execute_data.func = (zend_function*)op_array;
  337. EG(current_execute_data) = &fake_execute_data;
  338. if ((offset = zend_get_constant_str("__COMPILER_HALT_OFFSET__", sizeof("__COMPILER_HALT_OFFSET__") - 1)) != NULL) {
  339. main_persistent_script->compiler_halt_offset = Z_LVAL_P(offset);
  340. }
  341. EG(current_execute_data) = orig_execute_data;
  342. }
  343. if (op_array->static_variables) {
  344. HashTable *stored = zend_shared_alloc_get_xlat_entry(op_array->static_variables);
  345. if (stored) {
  346. op_array->static_variables = stored;
  347. } else {
  348. zend_hash_persist(op_array->static_variables, zend_persist_zval);
  349. zend_accel_store(op_array->static_variables, sizeof(HashTable));
  350. /* make immutable array */
  351. GC_SET_REFCOUNT(op_array->static_variables, 2);
  352. GC_TYPE_INFO(op_array->static_variables) = IS_ARRAY | (IS_ARRAY_IMMUTABLE << GC_FLAGS_SHIFT);
  353. }
  354. }
  355. if (zend_shared_alloc_get_xlat_entry(op_array->opcodes)) {
  356. already_stored = 1;
  357. }
  358. if (op_array->literals) {
  359. if (already_stored) {
  360. orig_literals = zend_shared_alloc_get_xlat_entry(op_array->literals);
  361. ZEND_ASSERT(orig_literals != NULL);
  362. op_array->literals = orig_literals;
  363. } else {
  364. zval *p = zend_accel_memdup(op_array->literals, sizeof(zval) * op_array->last_literal);
  365. zval *end = p + op_array->last_literal;
  366. orig_literals = op_array->literals;
  367. op_array->literals = p;
  368. while (p < end) {
  369. zend_persist_zval(p);
  370. p++;
  371. }
  372. #if ZEND_USE_ABS_CONST_ADDR
  373. efree(orig_literals);
  374. #endif
  375. }
  376. }
  377. if (already_stored) {
  378. persist_ptr = zend_shared_alloc_get_xlat_entry(op_array->opcodes);
  379. ZEND_ASSERT(persist_ptr != NULL);
  380. op_array->opcodes = persist_ptr;
  381. } else {
  382. zend_op *new_opcodes = zend_accel_memdup(op_array->opcodes, sizeof(zend_op) * op_array->last);
  383. zend_op *opline = new_opcodes;
  384. zend_op *end = new_opcodes + op_array->last;
  385. int offset = 0;
  386. for (; opline < end ; opline++, offset++) {
  387. #if ZEND_USE_ABS_CONST_ADDR
  388. if (opline->op1_type == IS_CONST) {
  389. opline->op1.zv = (zval*)((char*)opline->op1.zv + ((char*)op_array->literals - (char*)orig_literals));
  390. if (opline->opcode == ZEND_SEND_VAL
  391. || opline->opcode == ZEND_SEND_VAL_EX
  392. || opline->opcode == ZEND_QM_ASSIGN) {
  393. /* Update handlers to eliminate REFCOUNTED check */
  394. zend_vm_set_opcode_handler_ex(opline, 0, 0, 0);
  395. }
  396. }
  397. if (opline->op2_type == IS_CONST) {
  398. opline->op2.zv = (zval*)((char*)opline->op2.zv + ((char*)op_array->literals - (char*)orig_literals));
  399. }
  400. #else
  401. if (opline->op1_type == IS_CONST) {
  402. opline->op1.constant =
  403. (char*)(op_array->literals +
  404. ((zval*)((char*)(op_array->opcodes + (opline - new_opcodes)) +
  405. (int32_t)opline->op1.constant) - orig_literals)) -
  406. (char*)opline;
  407. if (opline->opcode == ZEND_SEND_VAL
  408. || opline->opcode == ZEND_SEND_VAL_EX
  409. || opline->opcode == ZEND_QM_ASSIGN) {
  410. zend_vm_set_opcode_handler_ex(opline, 0, 0, 0);
  411. }
  412. }
  413. if (opline->op2_type == IS_CONST) {
  414. opline->op2.constant =
  415. (char*)(op_array->literals +
  416. ((zval*)((char*)(op_array->opcodes + (opline - new_opcodes)) +
  417. (int32_t)opline->op2.constant) - orig_literals)) -
  418. (char*)opline;
  419. }
  420. #endif
  421. #if ZEND_USE_ABS_JMP_ADDR
  422. if (op_array->fn_flags & ZEND_ACC_DONE_PASS_TWO) {
  423. /* fix jumps to point to new array */
  424. switch (opline->opcode) {
  425. case ZEND_JMP:
  426. case ZEND_FAST_CALL:
  427. opline->op1.jmp_addr = &new_opcodes[opline->op1.jmp_addr - op_array->opcodes];
  428. break;
  429. case ZEND_JMPZNZ:
  430. /* relative extended_value don't have to be changed */
  431. /* break omitted intentionally */
  432. case ZEND_JMPZ:
  433. case ZEND_JMPNZ:
  434. case ZEND_JMPZ_EX:
  435. case ZEND_JMPNZ_EX:
  436. case ZEND_JMP_SET:
  437. case ZEND_COALESCE:
  438. case ZEND_FE_RESET_R:
  439. case ZEND_FE_RESET_RW:
  440. case ZEND_ASSERT_CHECK:
  441. opline->op2.jmp_addr = &new_opcodes[opline->op2.jmp_addr - op_array->opcodes];
  442. break;
  443. case ZEND_CATCH:
  444. if (!(opline->extended_value & ZEND_LAST_CATCH)) {
  445. opline->op2.jmp_addr = &new_opcodes[opline->op2.jmp_addr - op_array->opcodes];
  446. }
  447. break;
  448. case ZEND_DECLARE_ANON_CLASS:
  449. case ZEND_DECLARE_ANON_INHERITED_CLASS:
  450. case ZEND_FE_FETCH_R:
  451. case ZEND_FE_FETCH_RW:
  452. case ZEND_SWITCH_LONG:
  453. case ZEND_SWITCH_STRING:
  454. /* relative extended_value don't have to be changed */
  455. break;
  456. }
  457. }
  458. #endif
  459. }
  460. efree(op_array->opcodes);
  461. op_array->opcodes = new_opcodes;
  462. if (op_array->run_time_cache) {
  463. efree(op_array->run_time_cache);
  464. op_array->run_time_cache = NULL;
  465. }
  466. }
  467. if (op_array->function_name && !IS_ACCEL_INTERNED(op_array->function_name)) {
  468. zend_string *new_name;
  469. if (already_stored) {
  470. new_name = zend_shared_alloc_get_xlat_entry(op_array->function_name);
  471. ZEND_ASSERT(new_name != NULL);
  472. op_array->function_name = new_name;
  473. } else {
  474. zend_accel_store_interned_string(op_array->function_name);
  475. }
  476. }
  477. if (op_array->filename) {
  478. /* do not free! PHP has centralized filename storage, compiler will free it */
  479. zend_accel_memdup_string(op_array->filename);
  480. }
  481. if (op_array->arg_info) {
  482. zend_arg_info *arg_info = op_array->arg_info;
  483. uint32_t num_args = op_array->num_args;
  484. if (op_array->fn_flags & ZEND_ACC_HAS_RETURN_TYPE) {
  485. arg_info--;
  486. num_args++;
  487. }
  488. if (already_stored) {
  489. arg_info = zend_shared_alloc_get_xlat_entry(arg_info);
  490. ZEND_ASSERT(arg_info != NULL);
  491. } else {
  492. uint32_t i;
  493. if (op_array->fn_flags & ZEND_ACC_VARIADIC) {
  494. num_args++;
  495. }
  496. zend_accel_store(arg_info, sizeof(zend_arg_info) * num_args);
  497. for (i = 0; i < num_args; i++) {
  498. if (arg_info[i].name) {
  499. zend_accel_store_interned_string(arg_info[i].name);
  500. }
  501. if (ZEND_TYPE_IS_CLASS(arg_info[i].type)) {
  502. zend_string *type_name = ZEND_TYPE_NAME(arg_info[i].type);
  503. zend_bool allow_null = ZEND_TYPE_ALLOW_NULL(arg_info[i].type);
  504. zend_accel_store_interned_string(type_name);
  505. arg_info[i].type = ZEND_TYPE_ENCODE_CLASS(type_name, allow_null);
  506. }
  507. }
  508. }
  509. if (op_array->fn_flags & ZEND_ACC_HAS_RETURN_TYPE) {
  510. arg_info++;
  511. }
  512. op_array->arg_info = arg_info;
  513. }
  514. if (op_array->live_range) {
  515. zend_accel_store(op_array->live_range, sizeof(zend_live_range) * op_array->last_live_range);
  516. }
  517. if (op_array->scope) {
  518. op_array->scope = zend_shared_alloc_get_xlat_entry(op_array->scope);
  519. }
  520. if (op_array->doc_comment) {
  521. if (ZCG(accel_directives).save_comments) {
  522. if (already_stored) {
  523. op_array->doc_comment = zend_shared_alloc_get_xlat_entry(op_array->doc_comment);
  524. ZEND_ASSERT(op_array->doc_comment != NULL);
  525. } else {
  526. zend_accel_store_interned_string(op_array->doc_comment);
  527. }
  528. } else {
  529. if (!already_stored) {
  530. zend_string_release_ex(op_array->doc_comment, 0);
  531. }
  532. op_array->doc_comment = NULL;
  533. }
  534. }
  535. if (op_array->try_catch_array) {
  536. zend_accel_store(op_array->try_catch_array, sizeof(zend_try_catch_element) * op_array->last_try_catch);
  537. }
  538. if (op_array->vars) {
  539. if (already_stored) {
  540. persist_ptr = zend_shared_alloc_get_xlat_entry(op_array->vars);
  541. ZEND_ASSERT(persist_ptr != NULL);
  542. op_array->vars = (zend_string**)persist_ptr;
  543. } else {
  544. int i;
  545. zend_accel_store(op_array->vars, sizeof(zend_string*) * op_array->last_var);
  546. for (i = 0; i < op_array->last_var; i++) {
  547. zend_accel_store_interned_string(op_array->vars[i]);
  548. }
  549. }
  550. }
  551. /* "prototype" may be undefined if "scope" isn't set */
  552. if (op_array->scope && op_array->prototype) {
  553. if ((persist_ptr = zend_shared_alloc_get_xlat_entry(op_array->prototype))) {
  554. op_array->prototype = (union _zend_function*)persist_ptr;
  555. }
  556. } else {
  557. op_array->prototype = NULL;
  558. }
  559. ZCG(mem) = (void*)((char*)ZCG(mem) + ZEND_ALIGNED_SIZE(zend_extensions_op_array_persist(op_array, ZCG(mem))));
  560. }
  561. static void zend_persist_op_array(zval *zv)
  562. {
  563. zend_op_array *op_array = Z_PTR_P(zv);
  564. ZEND_ASSERT(op_array->type == ZEND_USER_FUNCTION);
  565. memcpy(ZCG(mem), Z_PTR_P(zv), sizeof(zend_op_array));
  566. Z_PTR_P(zv) = ZCG(mem);
  567. ZCG(mem) = (void*)((char*)ZCG(mem) + ZEND_ALIGNED_SIZE(sizeof(zend_op_array)));
  568. zend_persist_op_array_ex(Z_PTR_P(zv), NULL);
  569. ((zend_op_array*)Z_PTR_P(zv))->fn_flags |= ZEND_ACC_IMMUTABLE;
  570. }
  571. static void zend_persist_class_method(zval *zv)
  572. {
  573. zend_op_array *op_array = Z_PTR_P(zv);
  574. zend_op_array *old_op_array;
  575. ZEND_ASSERT(op_array->type == ZEND_USER_FUNCTION);
  576. old_op_array = zend_shared_alloc_get_xlat_entry(op_array);
  577. if (old_op_array) {
  578. Z_PTR_P(zv) = old_op_array;
  579. if (op_array->refcount && --(*op_array->refcount) == 0) {
  580. efree(op_array->refcount);
  581. }
  582. return;
  583. }
  584. memcpy(ZCG(arena_mem), Z_PTR_P(zv), sizeof(zend_op_array));
  585. zend_shared_alloc_register_xlat_entry(Z_PTR_P(zv), ZCG(arena_mem));
  586. Z_PTR_P(zv) = ZCG(arena_mem);
  587. ZCG(arena_mem) = (void*)((char*)ZCG(arena_mem) + ZEND_ALIGNED_SIZE(sizeof(zend_op_array)));
  588. zend_persist_op_array_ex(Z_PTR_P(zv), NULL);
  589. }
  590. static void zend_persist_property_info(zval *zv)
  591. {
  592. zend_property_info *prop = zend_shared_alloc_get_xlat_entry(Z_PTR_P(zv));
  593. if (prop) {
  594. Z_PTR_P(zv) = prop;
  595. return;
  596. }
  597. memcpy(ZCG(arena_mem), Z_PTR_P(zv), sizeof(zend_property_info));
  598. zend_shared_alloc_register_xlat_entry(Z_PTR_P(zv), ZCG(arena_mem));
  599. prop = Z_PTR_P(zv) = ZCG(arena_mem);
  600. ZCG(arena_mem) = (void*)((char*)ZCG(arena_mem) + ZEND_ALIGNED_SIZE(sizeof(zend_property_info)));
  601. prop->ce = zend_shared_alloc_get_xlat_entry(prop->ce);
  602. zend_accel_store_interned_string(prop->name);
  603. if (prop->doc_comment) {
  604. if (ZCG(accel_directives).save_comments) {
  605. zend_accel_store_interned_string(prop->doc_comment);
  606. } else {
  607. if (!zend_shared_alloc_get_xlat_entry(prop->doc_comment)) {
  608. zend_shared_alloc_register_xlat_entry(prop->doc_comment, prop->doc_comment);
  609. }
  610. zend_string_release_ex(prop->doc_comment, 0);
  611. prop->doc_comment = NULL;
  612. }
  613. }
  614. }
  615. static void zend_persist_class_constant(zval *zv)
  616. {
  617. zend_class_constant *c = zend_shared_alloc_get_xlat_entry(Z_PTR_P(zv));
  618. if (c) {
  619. Z_PTR_P(zv) = c;
  620. return;
  621. }
  622. memcpy(ZCG(arena_mem), Z_PTR_P(zv), sizeof(zend_class_constant));
  623. zend_shared_alloc_register_xlat_entry(Z_PTR_P(zv), ZCG(arena_mem));
  624. c = Z_PTR_P(zv) = ZCG(arena_mem);
  625. ZCG(arena_mem) = (void*)((char*)ZCG(arena_mem) + ZEND_ALIGNED_SIZE(sizeof(zend_class_constant)));
  626. zend_persist_zval(&c->value);
  627. c->ce = zend_shared_alloc_get_xlat_entry(c->ce);
  628. if (c->doc_comment) {
  629. if (ZCG(accel_directives).save_comments) {
  630. zend_string *doc_comment = zend_shared_alloc_get_xlat_entry(c->doc_comment);
  631. if (doc_comment) {
  632. c->doc_comment = doc_comment;
  633. } else {
  634. zend_accel_store_interned_string(c->doc_comment);
  635. }
  636. } else {
  637. zend_string *doc_comment = zend_shared_alloc_get_xlat_entry(c->doc_comment);
  638. if (!doc_comment) {
  639. zend_shared_alloc_register_xlat_entry(c->doc_comment, c->doc_comment);
  640. zend_string_release_ex(c->doc_comment, 0);
  641. }
  642. c->doc_comment = NULL;
  643. }
  644. }
  645. }
  646. static void zend_persist_class_entry(zval *zv)
  647. {
  648. zend_class_entry *ce = Z_PTR_P(zv);
  649. if (ce->type == ZEND_USER_CLASS) {
  650. memcpy(ZCG(arena_mem), Z_PTR_P(zv), sizeof(zend_class_entry));
  651. zend_shared_alloc_register_xlat_entry(Z_PTR_P(zv), ZCG(arena_mem));
  652. ce = Z_PTR_P(zv) = ZCG(arena_mem);
  653. ZCG(arena_mem) = (void*)((char*)ZCG(arena_mem) + ZEND_ALIGNED_SIZE(sizeof(zend_class_entry)));
  654. zend_accel_store_interned_string(ce->name);
  655. zend_hash_persist(&ce->function_table, zend_persist_class_method);
  656. if (ce->default_properties_table) {
  657. int i;
  658. zend_accel_store(ce->default_properties_table, sizeof(zval) * ce->default_properties_count);
  659. for (i = 0; i < ce->default_properties_count; i++) {
  660. zend_persist_zval(&ce->default_properties_table[i]);
  661. }
  662. }
  663. if (ce->default_static_members_table) {
  664. int i;
  665. zend_accel_store(ce->default_static_members_table, sizeof(zval) * ce->default_static_members_count);
  666. /* Persist only static properties in this class.
  667. * Static properties from parent classes will be handled in class_copy_ctor */
  668. i = ce->parent ? ce->parent->default_static_members_count : 0;
  669. for (; i < ce->default_static_members_count; i++) {
  670. zend_persist_zval(&ce->default_static_members_table[i]);
  671. }
  672. }
  673. ce->static_members_table = NULL;
  674. zend_hash_persist(&ce->constants_table, zend_persist_class_constant);
  675. if (ce->info.user.filename) {
  676. /* do not free! PHP has centralized filename storage, compiler will free it */
  677. zend_accel_memdup_string(ce->info.user.filename);
  678. }
  679. if (ce->info.user.doc_comment) {
  680. if (ZCG(accel_directives).save_comments) {
  681. zend_accel_store_interned_string(ce->info.user.doc_comment);
  682. } else {
  683. if (!zend_shared_alloc_get_xlat_entry(ce->info.user.doc_comment)) {
  684. zend_shared_alloc_register_xlat_entry(ce->info.user.doc_comment, ce->info.user.doc_comment);
  685. zend_string_release_ex(ce->info.user.doc_comment, 0);
  686. }
  687. ce->info.user.doc_comment = NULL;
  688. }
  689. }
  690. zend_hash_persist(&ce->properties_info, zend_persist_property_info);
  691. if (ce->num_interfaces && ce->interfaces) {
  692. efree(ce->interfaces);
  693. }
  694. ce->interfaces = NULL; /* will be filled in on fetch */
  695. if (ce->num_traits && ce->traits) {
  696. efree(ce->traits);
  697. }
  698. ce->traits = NULL;
  699. if (ce->trait_aliases) {
  700. int i = 0;
  701. while (ce->trait_aliases[i]) {
  702. if (ce->trait_aliases[i]->trait_method.method_name) {
  703. zend_accel_store_interned_string(ce->trait_aliases[i]->trait_method.method_name);
  704. }
  705. if (ce->trait_aliases[i]->trait_method.class_name) {
  706. zend_accel_store_interned_string(ce->trait_aliases[i]->trait_method.class_name);
  707. }
  708. if (ce->trait_aliases[i]->alias) {
  709. zend_accel_store_interned_string(ce->trait_aliases[i]->alias);
  710. }
  711. zend_accel_store(ce->trait_aliases[i], sizeof(zend_trait_alias));
  712. i++;
  713. }
  714. zend_accel_store(ce->trait_aliases, sizeof(zend_trait_alias*) * (i + 1));
  715. }
  716. if (ce->trait_precedences) {
  717. int i = 0;
  718. int j;
  719. while (ce->trait_precedences[i]) {
  720. zend_accel_store_interned_string(ce->trait_precedences[i]->trait_method.method_name);
  721. zend_accel_store_interned_string(ce->trait_precedences[i]->trait_method.class_name);
  722. for (j = 0; j < ce->trait_precedences[i]->num_excludes; j++) {
  723. zend_accel_store_interned_string(ce->trait_precedences[i]->exclude_class_names[j]);
  724. }
  725. zend_accel_store(ce->trait_precedences[i], sizeof(zend_trait_precedence) + (ce->trait_precedences[i]->num_excludes - 1) * sizeof(zend_string*));
  726. i++;
  727. }
  728. zend_accel_store(
  729. ce->trait_precedences, sizeof(zend_trait_precedence*) * (i + 1));
  730. }
  731. }
  732. }
  733. //static int zend_update_property_info_ce(zval *zv)
  734. //{
  735. // zend_property_info *prop = Z_PTR_P(zv);
  736. //
  737. // prop->ce = zend_shared_alloc_get_xlat_entry(prop->ce);
  738. // return 0;
  739. //}
  740. static int zend_update_parent_ce(zval *zv)
  741. {
  742. zend_class_entry *ce = Z_PTR_P(zv);
  743. if (ce->parent) {
  744. ce->parent = zend_shared_alloc_get_xlat_entry(ce->parent);
  745. }
  746. /* update methods */
  747. if (ce->constructor) {
  748. ce->constructor = zend_shared_alloc_get_xlat_entry(ce->constructor);
  749. }
  750. if (ce->destructor) {
  751. ce->destructor = zend_shared_alloc_get_xlat_entry(ce->destructor);
  752. }
  753. if (ce->clone) {
  754. ce->clone = zend_shared_alloc_get_xlat_entry(ce->clone);
  755. }
  756. if (ce->__get) {
  757. ce->__get = zend_shared_alloc_get_xlat_entry(ce->__get);
  758. }
  759. if (ce->__set) {
  760. ce->__set = zend_shared_alloc_get_xlat_entry(ce->__set);
  761. }
  762. if (ce->__call) {
  763. ce->__call = zend_shared_alloc_get_xlat_entry(ce->__call);
  764. }
  765. if (ce->serialize_func) {
  766. ce->serialize_func = zend_shared_alloc_get_xlat_entry(ce->serialize_func);
  767. }
  768. if (ce->unserialize_func) {
  769. ce->unserialize_func = zend_shared_alloc_get_xlat_entry(ce->unserialize_func);
  770. }
  771. if (ce->__isset) {
  772. ce->__isset = zend_shared_alloc_get_xlat_entry(ce->__isset);
  773. }
  774. if (ce->__unset) {
  775. ce->__unset = zend_shared_alloc_get_xlat_entry(ce->__unset);
  776. }
  777. if (ce->__tostring) {
  778. ce->__tostring = zend_shared_alloc_get_xlat_entry(ce->__tostring);
  779. }
  780. if (ce->__callstatic) {
  781. ce->__callstatic = zend_shared_alloc_get_xlat_entry(ce->__callstatic);
  782. }
  783. if (ce->__debugInfo) {
  784. ce->__debugInfo = zend_shared_alloc_get_xlat_entry(ce->__debugInfo);
  785. }
  786. // zend_hash_apply(&ce->properties_info, (apply_func_t) zend_update_property_info_ce);
  787. return 0;
  788. }
  789. static void zend_accel_persist_class_table(HashTable *class_table)
  790. {
  791. zend_hash_persist(class_table, zend_persist_class_entry);
  792. zend_hash_apply(class_table, (apply_func_t) zend_update_parent_ce);
  793. }
  794. zend_persistent_script *zend_accel_script_persist(zend_persistent_script *script, const char **key, unsigned int key_length, int for_shm)
  795. {
  796. script->mem = ZCG(mem);
  797. ZEND_ASSERT(((zend_uintptr_t)ZCG(mem) & 0x7) == 0); /* should be 8 byte aligned */
  798. zend_shared_alloc_clear_xlat_table();
  799. zend_accel_store(script, sizeof(zend_persistent_script));
  800. if (key && *key) {
  801. *key = zend_accel_memdup(*key, key_length + 1);
  802. }
  803. script->corrupted = 0;
  804. ZCG(current_persistent_script) = script;
  805. if (!for_shm) {
  806. /* script is not going to be saved in SHM */
  807. script->corrupted = 1;
  808. }
  809. zend_accel_store_interned_string(script->script.filename);
  810. #if defined(__AVX__) || defined(__SSE2__)
  811. /* Align to 64-byte boundary */
  812. ZCG(mem) = (void*)(((zend_uintptr_t)ZCG(mem) + 63L) & ~63L);
  813. #else
  814. ZEND_ASSERT(((zend_uintptr_t)ZCG(mem) & 0x7) == 0); /* should be 8 byte aligned */
  815. #endif
  816. script->arena_mem = ZCG(arena_mem) = ZCG(mem);
  817. ZCG(mem) = (void*)((char*)ZCG(mem) + script->arena_size);
  818. zend_accel_persist_class_table(&script->script.class_table);
  819. zend_hash_persist(&script->script.function_table, zend_persist_op_array);
  820. zend_persist_op_array_ex(&script->script.main_op_array, script);
  821. script->corrupted = 0;
  822. ZCG(current_persistent_script) = NULL;
  823. return script;
  824. }
  825. int zend_accel_script_persistable(zend_persistent_script *script)
  826. {
  827. return 1;
  828. }