zend_optimizer.c 20 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669
  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 "php.h"
  22. #include "Optimizer/zend_optimizer.h"
  23. #include "Optimizer/zend_optimizer_internal.h"
  24. #include "zend_API.h"
  25. #include "zend_constants.h"
  26. #include "zend_execute.h"
  27. #include "zend_vm.h"
  28. #define OPTIMIZATION_LEVEL \
  29. ZCG(accel_directives).optimization_level
  30. static void zend_optimizer_zval_dtor_wrapper(zval *zvalue)
  31. {
  32. zval_dtor(zvalue);
  33. }
  34. static void zend_optimizer_collect_constant(HashTable **constants, zval *name, zval* value)
  35. {
  36. zval val;
  37. if (!*constants) {
  38. *constants = emalloc(sizeof(HashTable));
  39. zend_hash_init(*constants, 16, NULL, (void (*)(void *))zend_optimizer_zval_dtor_wrapper, 0);
  40. }
  41. val = *value;
  42. zval_copy_ctor(&val);
  43. zend_hash_add(*constants, Z_STRVAL_P(name), Z_STRLEN_P(name)+1, (void**)&val, sizeof(zval), NULL);
  44. }
  45. static int zend_optimizer_get_collected_constant(HashTable *constants, zval *name, zval* value)
  46. {
  47. zval *val;
  48. if (zend_hash_find(constants, Z_STRVAL_P(name), Z_STRLEN_P(name)+1, (void**)&val) == SUCCESS) {
  49. *value = *val;
  50. zval_copy_ctor(value);
  51. return 1;
  52. }
  53. return 0;
  54. }
  55. #if ZEND_EXTENSION_API_NO >= PHP_5_5_X_API_NO
  56. static int zend_optimizer_lookup_cv(zend_op_array *op_array, char* name, int name_len)
  57. {
  58. int i = 0;
  59. ulong hash_value = zend_inline_hash_func(name, name_len+1);
  60. while (i < op_array->last_var) {
  61. if (op_array->vars[i].name == name ||
  62. (op_array->vars[i].hash_value == hash_value &&
  63. op_array->vars[i].name_len == name_len &&
  64. memcmp(op_array->vars[i].name, name, name_len) == 0)) {
  65. return i;
  66. }
  67. i++;
  68. }
  69. i = op_array->last_var;
  70. op_array->last_var++;
  71. op_array->vars = erealloc(op_array->vars, op_array->last_var * sizeof(zend_compiled_variable));
  72. if (IS_INTERNED(name)) {
  73. op_array->vars[i].name = name;
  74. } else {
  75. op_array->vars[i].name = estrndup(name, name_len);
  76. }
  77. op_array->vars[i].name_len = name_len;
  78. op_array->vars[i].hash_value = hash_value;
  79. return i;
  80. }
  81. #endif
  82. #if ZEND_EXTENSION_API_NO > PHP_5_3_X_API_NO
  83. int zend_optimizer_add_literal(zend_op_array *op_array, const zval *zv TSRMLS_DC)
  84. {
  85. int i = op_array->last_literal;
  86. op_array->last_literal++;
  87. op_array->literals = (zend_literal*)erealloc(op_array->literals, op_array->last_literal * sizeof(zend_literal));
  88. op_array->literals[i].constant = *zv;
  89. op_array->literals[i].hash_value = 0;
  90. op_array->literals[i].cache_slot = -1;
  91. Z_SET_REFCOUNT(op_array->literals[i].constant, 2);
  92. Z_SET_ISREF(op_array->literals[i].constant);
  93. return i;
  94. }
  95. # define LITERAL_LONG(op, val) do { \
  96. zval _c; \
  97. ZVAL_LONG(&_c, val); \
  98. op.constant = zend_optimizer_add_literal(op_array, &_c TSRMLS_CC); \
  99. } while (0)
  100. # define LITERAL_BOOL(op, val) do { \
  101. zval _c; \
  102. ZVAL_BOOL(&_c, val); \
  103. op.constant = zend_optimizer_add_literal(op_array, &_c TSRMLS_CC); \
  104. } while (0)
  105. # define literal_dtor(zv) do { \
  106. zval_dtor(zv); \
  107. Z_TYPE_P(zv) = IS_NULL; \
  108. } while (0)
  109. #define COPY_NODE(target, src) do { \
  110. target ## _type = src ## _type; \
  111. target = src; \
  112. } while (0)
  113. #else
  114. # define LITERAL_LONG(op, val) ZVAL_LONG(&op.u.constant, val)
  115. # define LITERAL_BOOL(op, val) ZVAL_BOOL(&op.u.constant, val)
  116. # define literal_dtor(zv) zval_dtor(zv)
  117. #define COPY_NODE(target, src) do { \
  118. target = src; \
  119. } while (0)
  120. #endif
  121. static void update_op1_const(zend_op_array *op_array,
  122. zend_op *opline,
  123. zval *val TSRMLS_DC)
  124. {
  125. if (opline->opcode == ZEND_FREE) {
  126. MAKE_NOP(opline);
  127. zval_dtor(val);
  128. } else {
  129. ZEND_OP1_TYPE(opline) = IS_CONST;
  130. #if ZEND_EXTENSION_API_NO > PHP_5_3_X_API_NO
  131. if (Z_TYPE_P(val) == IS_STRING) {
  132. switch (opline->opcode) {
  133. case ZEND_INIT_STATIC_METHOD_CALL:
  134. case ZEND_CATCH:
  135. case ZEND_FETCH_CONSTANT:
  136. opline->op1.constant = zend_optimizer_add_literal(op_array, val TSRMLS_CC);
  137. Z_HASH_P(&ZEND_OP1_LITERAL(opline)) = zend_hash_func(Z_STRVAL(ZEND_OP1_LITERAL(opline)), Z_STRLEN(ZEND_OP1_LITERAL(opline)) + 1);
  138. op_array->literals[opline->op1.constant].cache_slot = op_array->last_cache_slot++;
  139. Z_STRVAL_P(val) = zend_str_tolower_dup(Z_STRVAL_P(val), Z_STRLEN_P(val));
  140. zend_optimizer_add_literal(op_array, val TSRMLS_CC);
  141. op_array->literals[opline->op1.constant+1].hash_value = zend_hash_func(Z_STRVAL(op_array->literals[opline->op1.constant+1].constant), Z_STRLEN(op_array->literals[opline->op1.constant+1].constant) + 1);
  142. break;
  143. case ZEND_DO_FCALL:
  144. zend_str_tolower(Z_STRVAL_P(val), Z_STRLEN_P(val));
  145. opline->op1.constant = zend_optimizer_add_literal(op_array, val TSRMLS_CC);
  146. Z_HASH_P(&ZEND_OP1_LITERAL(opline)) = zend_hash_func(Z_STRVAL(ZEND_OP1_LITERAL(opline)), Z_STRLEN(ZEND_OP1_LITERAL(opline)) + 1);
  147. op_array->literals[opline->op1.constant].cache_slot = op_array->last_cache_slot++;
  148. break;
  149. default:
  150. opline->op1.constant = zend_optimizer_add_literal(op_array, val TSRMLS_CC);
  151. Z_HASH_P(&ZEND_OP1_LITERAL(opline)) = zend_hash_func(Z_STRVAL(ZEND_OP1_LITERAL(opline)), Z_STRLEN(ZEND_OP1_LITERAL(opline)) + 1);
  152. break;
  153. }
  154. } else {
  155. opline->op1.constant = zend_optimizer_add_literal(op_array, val TSRMLS_CC);
  156. }
  157. #else
  158. ZEND_OP1_LITERAL(opline) = *val;
  159. #endif
  160. }
  161. }
  162. static void update_op2_const(zend_op_array *op_array,
  163. zend_op *opline,
  164. zval *val TSRMLS_DC)
  165. {
  166. ZEND_OP2_TYPE(opline) = IS_CONST;
  167. #if ZEND_EXTENSION_API_NO > PHP_5_3_X_API_NO
  168. opline->op2.constant = zend_optimizer_add_literal(op_array, val TSRMLS_CC);
  169. if (Z_TYPE_P(val) == IS_STRING) {
  170. Z_HASH_P(&ZEND_OP2_LITERAL(opline)) = zend_hash_func(Z_STRVAL(ZEND_OP2_LITERAL(opline)), Z_STRLEN(ZEND_OP2_LITERAL(opline)) + 1);
  171. switch (opline->opcode) {
  172. case ZEND_FETCH_R:
  173. case ZEND_FETCH_W:
  174. case ZEND_FETCH_RW:
  175. case ZEND_FETCH_IS:
  176. case ZEND_FETCH_UNSET:
  177. case ZEND_FETCH_FUNC_ARG:
  178. case ZEND_FETCH_CLASS:
  179. case ZEND_INIT_FCALL_BY_NAME:
  180. /*case ZEND_INIT_NS_FCALL_BY_NAME:*/
  181. case ZEND_UNSET_VAR:
  182. case ZEND_ISSET_ISEMPTY_VAR:
  183. case ZEND_ADD_INTERFACE:
  184. case ZEND_ADD_TRAIT:
  185. op_array->literals[opline->op2.constant].cache_slot = op_array->last_cache_slot++;
  186. Z_STRVAL_P(val) = zend_str_tolower_dup(Z_STRVAL_P(val), Z_STRLEN_P(val));
  187. zend_optimizer_add_literal(op_array, val TSRMLS_CC);
  188. op_array->literals[opline->op2.constant+1].hash_value = zend_hash_func(Z_STRVAL(op_array->literals[opline->op2.constant+1].constant), Z_STRLEN(op_array->literals[opline->op2.constant+1].constant) + 1);
  189. break;
  190. case ZEND_INIT_METHOD_CALL:
  191. case ZEND_INIT_STATIC_METHOD_CALL:
  192. Z_STRVAL_P(val) = zend_str_tolower_dup(Z_STRVAL_P(val), Z_STRLEN_P(val));
  193. zend_optimizer_add_literal(op_array, val TSRMLS_CC);
  194. op_array->literals[opline->op2.constant+1].hash_value = zend_hash_func(Z_STRVAL(op_array->literals[opline->op2.constant+1].constant), Z_STRLEN(op_array->literals[opline->op2.constant+1].constant) + 1);
  195. /* break missing intentionally */
  196. /*case ZEND_FETCH_CONSTANT:*/
  197. case ZEND_ASSIGN_OBJ:
  198. case ZEND_FETCH_OBJ_R:
  199. case ZEND_FETCH_OBJ_W:
  200. case ZEND_FETCH_OBJ_RW:
  201. case ZEND_FETCH_OBJ_IS:
  202. case ZEND_FETCH_OBJ_UNSET:
  203. case ZEND_FETCH_OBJ_FUNC_ARG:
  204. case ZEND_UNSET_OBJ:
  205. case ZEND_PRE_INC_OBJ:
  206. case ZEND_PRE_DEC_OBJ:
  207. case ZEND_POST_INC_OBJ:
  208. case ZEND_POST_DEC_OBJ:
  209. case ZEND_ISSET_ISEMPTY_PROP_OBJ:
  210. op_array->literals[opline->op2.constant].cache_slot = op_array->last_cache_slot;
  211. op_array->last_cache_slot += 2;
  212. break;
  213. case ZEND_ASSIGN_ADD:
  214. case ZEND_ASSIGN_SUB:
  215. case ZEND_ASSIGN_MUL:
  216. case ZEND_ASSIGN_DIV:
  217. case ZEND_ASSIGN_MOD:
  218. case ZEND_ASSIGN_SL:
  219. case ZEND_ASSIGN_SR:
  220. case ZEND_ASSIGN_CONCAT:
  221. case ZEND_ASSIGN_BW_OR:
  222. case ZEND_ASSIGN_BW_AND:
  223. case ZEND_ASSIGN_BW_XOR:
  224. if (opline->extended_value == ZEND_ASSIGN_OBJ) {
  225. op_array->literals[opline->op2.constant].cache_slot = op_array->last_cache_slot;
  226. op_array->last_cache_slot += 2;
  227. }
  228. break;
  229. #if ZEND_EXTENSION_API_NO >= PHP_5_4_X_API_NO
  230. case ZEND_OP_DATA:
  231. if ((opline-1)->opcode == ZEND_ASSIGN_DIM ||
  232. ((opline-1)->extended_value == ZEND_ASSIGN_DIM &&
  233. ((opline-1)->opcode == ZEND_ASSIGN_ADD ||
  234. (opline-1)->opcode == ZEND_ASSIGN_SUB ||
  235. (opline-1)->opcode == ZEND_ASSIGN_MUL ||
  236. (opline-1)->opcode == ZEND_ASSIGN_DIV ||
  237. (opline-1)->opcode == ZEND_ASSIGN_MOD ||
  238. (opline-1)->opcode == ZEND_ASSIGN_SL ||
  239. (opline-1)->opcode == ZEND_ASSIGN_SR ||
  240. (opline-1)->opcode == ZEND_ASSIGN_CONCAT ||
  241. (opline-1)->opcode == ZEND_ASSIGN_BW_OR ||
  242. (opline-1)->opcode == ZEND_ASSIGN_BW_AND ||
  243. (opline-1)->opcode == ZEND_ASSIGN_BW_XOR))) {
  244. goto check_numeric;
  245. }
  246. break;
  247. case ZEND_ISSET_ISEMPTY_DIM_OBJ:
  248. case ZEND_ADD_ARRAY_ELEMENT:
  249. case ZEND_INIT_ARRAY:
  250. case ZEND_ASSIGN_DIM:
  251. case ZEND_UNSET_DIM:
  252. case ZEND_FETCH_DIM_R:
  253. case ZEND_FETCH_DIM_W:
  254. case ZEND_FETCH_DIM_RW:
  255. case ZEND_FETCH_DIM_IS:
  256. case ZEND_FETCH_DIM_FUNC_ARG:
  257. case ZEND_FETCH_DIM_UNSET:
  258. case ZEND_FETCH_DIM_TMP_VAR:
  259. check_numeric:
  260. {
  261. ulong index;
  262. int numeric = 0;
  263. ZEND_HANDLE_NUMERIC_EX(Z_STRVAL_P(val), Z_STRLEN_P(val)+1, index, numeric = 1);
  264. if (numeric) {
  265. zval_dtor(val);
  266. ZVAL_LONG(val, index);
  267. op_array->literals[opline->op2.constant].constant = *val;
  268. }
  269. }
  270. break;
  271. #endif
  272. default:
  273. break;
  274. }
  275. }
  276. #else
  277. ZEND_OP2_LITERAL(opline) = *val;
  278. #endif
  279. }
  280. static int replace_var_by_const(zend_op_array *op_array,
  281. zend_op *opline,
  282. zend_uint var,
  283. zval *val TSRMLS_DC)
  284. {
  285. zend_op *end = op_array->opcodes + op_array->last;
  286. while (opline < end) {
  287. if (ZEND_OP1_TYPE(opline) == IS_VAR &&
  288. ZEND_OP1(opline).var == var) {
  289. switch (opline->opcode) {
  290. case ZEND_FETCH_DIM_W:
  291. case ZEND_FETCH_DIM_RW:
  292. case ZEND_FETCH_DIM_FUNC_ARG:
  293. case ZEND_FETCH_DIM_UNSET:
  294. case ZEND_ASSIGN_DIM:
  295. #if ZEND_EXTENSION_API_NO > PHP_5_3_X_API_NO
  296. case ZEND_SEPARATE:
  297. #endif
  298. return 0;
  299. case ZEND_SEND_VAR_NO_REF:
  300. if (opline->extended_value & ZEND_ARG_COMPILE_TIME_BOUND) {
  301. if (opline->extended_value & ZEND_ARG_SEND_BY_REF) {
  302. return 0;
  303. }
  304. opline->extended_value = ZEND_DO_FCALL;
  305. } else {
  306. opline->extended_value = ZEND_DO_FCALL_BY_NAME;
  307. }
  308. opline->opcode = ZEND_SEND_VAL;
  309. break;
  310. case ZEND_SWITCH_FREE:
  311. case ZEND_CASE: {
  312. zend_op *m, *n;
  313. int brk = op_array->last_brk_cont;
  314. while (brk--) {
  315. if (op_array->brk_cont_array[brk].start <= (opline - op_array->opcodes) &&
  316. op_array->brk_cont_array[brk].brk > (opline - op_array->opcodes)) {
  317. break;
  318. }
  319. }
  320. m = opline;
  321. n = op_array->opcodes + op_array->brk_cont_array[brk].brk + 1;
  322. while (m < n) {
  323. if (ZEND_OP1_TYPE(m) == IS_VAR &&
  324. ZEND_OP1(m).var == var) {
  325. if (m->opcode == ZEND_CASE) {
  326. zval old_val;
  327. old_val = *val;
  328. zval_copy_ctor(val);
  329. update_op1_const(op_array, m, val TSRMLS_CC);
  330. *val = old_val;
  331. } else if (m->opcode == ZEND_SWITCH_FREE) {
  332. MAKE_NOP(m);
  333. } else {
  334. ZEND_ASSERT(0);
  335. }
  336. }
  337. m++;
  338. }
  339. zval_dtor(val);
  340. return 1;
  341. }
  342. case ZEND_FREE:
  343. MAKE_NOP(opline);
  344. zval_dtor(val);
  345. break;
  346. default:
  347. break;
  348. }
  349. update_op1_const(op_array, opline, val TSRMLS_CC);
  350. break;
  351. }
  352. if (ZEND_OP2_TYPE(opline) == IS_VAR &&
  353. ZEND_OP2(opline).var == var) {
  354. switch (opline->opcode) {
  355. case ZEND_ASSIGN_REF:
  356. return 0;
  357. default:
  358. break;
  359. }
  360. update_op2_const(op_array, opline, val TSRMLS_CC);
  361. break;
  362. }
  363. opline++;
  364. }
  365. return 1;
  366. }
  367. static void replace_tmp_by_const(zend_op_array *op_array,
  368. zend_op *opline,
  369. zend_uint var,
  370. zval *val
  371. TSRMLS_DC)
  372. {
  373. zend_op *end = op_array->opcodes + op_array->last;
  374. while (opline < end) {
  375. if (ZEND_OP1_TYPE(opline) == IS_TMP_VAR &&
  376. ZEND_OP1(opline).var == var) {
  377. /* In most cases IS_TMP_VAR operand may be used only once.
  378. * The operands are usually destroyed by the opcode handler.
  379. * ZEND_CASE is an exception, that keeps operand unchanged,
  380. * and allows its reuse. The number of ZEND_CASE instructions
  381. * usually terminated by ZEND_FREE that finally kills the value.
  382. */
  383. if (opline->opcode == ZEND_CASE || opline->opcode == ZEND_FREE) {
  384. zend_op *m, *n;
  385. int brk = op_array->last_brk_cont;
  386. zend_bool in_switch = 0;
  387. while (brk--) {
  388. if (op_array->brk_cont_array[brk].start <= (opline - op_array->opcodes) &&
  389. op_array->brk_cont_array[brk].brk > (opline - op_array->opcodes)) {
  390. in_switch = 1;
  391. break;
  392. }
  393. }
  394. if (!in_switch) {
  395. MAKE_NOP(opline);
  396. zval_dtor(val);
  397. break;
  398. }
  399. m = opline;
  400. n = op_array->opcodes + op_array->brk_cont_array[brk].brk + 1;
  401. while (m < n) {
  402. if (ZEND_OP1_TYPE(m) == IS_TMP_VAR &&
  403. ZEND_OP1(m).var == var) {
  404. if (m->opcode == ZEND_CASE) {
  405. zval old_val;
  406. old_val = *val;
  407. zval_copy_ctor(val);
  408. update_op1_const(op_array, m, val TSRMLS_CC);
  409. *val = old_val;
  410. } else if (m->opcode == ZEND_FREE) {
  411. MAKE_NOP(m);
  412. } else {
  413. ZEND_ASSERT(0);
  414. }
  415. }
  416. m++;
  417. }
  418. zval_dtor(val);
  419. break;
  420. } else {
  421. update_op1_const(op_array, opline, val TSRMLS_CC);
  422. break;
  423. }
  424. }
  425. if (ZEND_OP2_TYPE(opline) == IS_TMP_VAR &&
  426. ZEND_OP2(opline).var == var) {
  427. update_op2_const(op_array, opline, val TSRMLS_CC);
  428. /* TMP_VAR may be used only once */
  429. break;
  430. }
  431. opline++;
  432. }
  433. }
  434. #include "Optimizer/nop_removal.c"
  435. #include "Optimizer/block_pass.c"
  436. #include "Optimizer/optimize_temp_vars_5.c"
  437. #include "Optimizer/compact_literals.c"
  438. #include "Optimizer/optimize_func_calls.c"
  439. static void zend_optimize(zend_op_array *op_array,
  440. zend_persistent_script *script,
  441. HashTable **constants TSRMLS_DC)
  442. {
  443. if (op_array->type == ZEND_EVAL_CODE ||
  444. (op_array->fn_flags & ZEND_ACC_INTERACTIVE)) {
  445. return;
  446. }
  447. /* pass 1
  448. * - substitute persistent constants (true, false, null, etc)
  449. * - perform compile-time evaluation of constant binary and unary operations
  450. * - optimize series of ADD_STRING and/or ADD_CHAR
  451. * - convert CAST(IS_BOOL,x) into BOOL(x)
  452. */
  453. #include "Optimizer/pass1_5.c"
  454. /* pass 2:
  455. * - convert non-numeric constants to numeric constants in numeric operators
  456. * - optimize constant conditional JMPs
  457. * - optimize static BRKs and CONTs
  458. * - pre-evaluate constant function calls
  459. */
  460. #include "Optimizer/pass2.c"
  461. /* pass 3:
  462. * - optimize $i = $i+expr to $i+=expr
  463. * - optimize series of JMPs
  464. * - change $i++ to ++$i where possible
  465. */
  466. #include "Optimizer/pass3.c"
  467. #if ZEND_EXTENSION_API_NO > PHP_5_3_X_API_NO
  468. /* pass 4:
  469. * - INIT_FCALL_BY_NAME -> DO_FCALL
  470. */
  471. if (ZEND_OPTIMIZER_PASS_4 & OPTIMIZATION_LEVEL) {
  472. optimize_func_calls(op_array, script TSRMLS_CC);
  473. }
  474. #endif
  475. /* pass 5:
  476. * - CFG optimization
  477. */
  478. #include "Optimizer/pass5.c"
  479. /* pass 9:
  480. * - Optimize temp variables usage
  481. */
  482. #include "Optimizer/pass9.c"
  483. /* pass 10:
  484. * - remove NOPs
  485. */
  486. #include "Optimizer/pass10.c"
  487. #if ZEND_EXTENSION_API_NO > PHP_5_3_X_API_NO
  488. /* pass 11:
  489. * - Compact literals table
  490. */
  491. if (ZEND_OPTIMIZER_PASS_11 & OPTIMIZATION_LEVEL) {
  492. optimizer_compact_literals(op_array TSRMLS_CC);
  493. }
  494. #endif
  495. }
  496. static void zend_accel_optimize(zend_op_array *op_array,
  497. zend_persistent_script *script,
  498. HashTable **constants TSRMLS_DC)
  499. {
  500. zend_op *opline, *end;
  501. /* Revert pass_two() */
  502. opline = op_array->opcodes;
  503. end = opline + op_array->last;
  504. while (opline < end) {
  505. #if ZEND_EXTENSION_API_NO > PHP_5_3_X_API_NO
  506. if (opline->op1_type == IS_CONST) {
  507. opline->op1.constant = opline->op1.literal - op_array->literals;
  508. }
  509. if (opline->op2_type == IS_CONST) {
  510. opline->op2.constant = opline->op2.literal - op_array->literals;
  511. }
  512. #endif
  513. switch (opline->opcode) {
  514. case ZEND_JMP:
  515. #if ZEND_EXTENSION_API_NO > PHP_5_2_X_API_NO
  516. case ZEND_GOTO:
  517. #endif
  518. #if ZEND_EXTENSION_API_NO > PHP_5_4_X_API_NO
  519. case ZEND_FAST_CALL:
  520. #endif
  521. ZEND_OP1(opline).opline_num = ZEND_OP1(opline).jmp_addr - op_array->opcodes;
  522. break;
  523. case ZEND_JMPZ:
  524. case ZEND_JMPNZ:
  525. case ZEND_JMPZ_EX:
  526. case ZEND_JMPNZ_EX:
  527. #if ZEND_EXTENSION_API_NO > PHP_5_2_X_API_NO
  528. case ZEND_JMP_SET:
  529. #endif
  530. #if ZEND_EXTENSION_API_NO > PHP_5_3_X_API_NO
  531. case ZEND_JMP_SET_VAR:
  532. #endif
  533. ZEND_OP2(opline).opline_num = ZEND_OP2(opline).jmp_addr - op_array->opcodes;
  534. break;
  535. }
  536. opline++;
  537. }
  538. /* Do actual optimizations */
  539. zend_optimize(op_array, script, constants TSRMLS_CC);
  540. /* Redo pass_two() */
  541. opline = op_array->opcodes;
  542. end = opline + op_array->last;
  543. while (opline < end) {
  544. #if ZEND_EXTENSION_API_NO > PHP_5_3_X_API_NO
  545. if (opline->op1_type == IS_CONST) {
  546. opline->op1.zv = &op_array->literals[opline->op1.constant].constant;
  547. }
  548. if (opline->op2_type == IS_CONST) {
  549. opline->op2.zv = &op_array->literals[opline->op2.constant].constant;
  550. }
  551. #endif
  552. switch (opline->opcode) {
  553. case ZEND_JMP:
  554. #if ZEND_EXTENSION_API_NO > PHP_5_2_X_API_NO
  555. case ZEND_GOTO:
  556. #endif
  557. #if ZEND_EXTENSION_API_NO > PHP_5_4_X_API_NO
  558. case ZEND_FAST_CALL:
  559. #endif
  560. ZEND_OP1(opline).jmp_addr = &op_array->opcodes[ZEND_OP1(opline).opline_num];
  561. break;
  562. case ZEND_JMPZ:
  563. case ZEND_JMPNZ:
  564. case ZEND_JMPZ_EX:
  565. case ZEND_JMPNZ_EX:
  566. #if ZEND_EXTENSION_API_NO > PHP_5_2_X_API_NO
  567. case ZEND_JMP_SET:
  568. #endif
  569. #if ZEND_EXTENSION_API_NO > PHP_5_3_X_API_NO
  570. case ZEND_JMP_SET_VAR:
  571. #endif
  572. ZEND_OP2(opline).jmp_addr = &op_array->opcodes[ZEND_OP2(opline).opline_num];
  573. break;
  574. }
  575. ZEND_VM_SET_OPCODE_HANDLER(opline);
  576. opline++;
  577. }
  578. }
  579. int zend_accel_script_optimize(zend_persistent_script *script TSRMLS_DC)
  580. {
  581. Bucket *p, *q;
  582. HashTable *constants = NULL;
  583. zend_accel_optimize(&script->main_op_array, script, &constants TSRMLS_CC);
  584. p = script->function_table.pListHead;
  585. while (p) {
  586. zend_op_array *op_array = (zend_op_array*)p->pData;
  587. zend_accel_optimize(op_array, script, &constants TSRMLS_CC);
  588. p = p->pListNext;
  589. }
  590. p = script->class_table.pListHead;
  591. while (p) {
  592. zend_class_entry *ce = (zend_class_entry*)p->pDataPtr;
  593. q = ce->function_table.pListHead;
  594. while (q) {
  595. zend_op_array *op_array = (zend_op_array*)q->pData;
  596. if (op_array->scope == ce) {
  597. zend_accel_optimize(op_array, script, &constants TSRMLS_CC);
  598. } else if (op_array->type == ZEND_USER_FUNCTION) {
  599. zend_op_array *orig_op_array;
  600. if (zend_hash_find(&op_array->scope->function_table, q->arKey, q->nKeyLength, (void**)&orig_op_array) == SUCCESS) {
  601. HashTable *ht = op_array->static_variables;
  602. *op_array = *orig_op_array;
  603. op_array->static_variables = ht;
  604. }
  605. }
  606. q = q->pListNext;
  607. }
  608. p = p->pListNext;
  609. }
  610. if (constants) {
  611. zend_hash_destroy(constants);
  612. efree(constants);
  613. }
  614. return 1;
  615. }