optimize_temp_vars_5.c 6.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201
  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 "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. #include "zend_bitset.h"
  29. #define GET_AVAILABLE_T() \
  30. for (i = 0; i < T; i++) { \
  31. if (!zend_bitset_in(taken_T, i)) { \
  32. break; \
  33. } \
  34. } \
  35. zend_bitset_incl(taken_T, i); \
  36. if (i > max) { \
  37. max = i; \
  38. }
  39. void zend_optimize_temporary_variables(zend_op_array *op_array, zend_optimizer_ctx *ctx)
  40. {
  41. int T = op_array->T;
  42. int offset = op_array->last_var;
  43. uint32_t bitset_len;
  44. zend_bitset taken_T; /* T index in use */
  45. zend_op **start_of_T; /* opline where T is first used */
  46. zend_bitset valid_T; /* Is the map_T valid */
  47. int *map_T; /* Map's the T to its new index */
  48. zend_op *opline, *end;
  49. int currT;
  50. int i;
  51. int max = -1;
  52. int var_to_free = -1;
  53. void *checkpoint = zend_arena_checkpoint(ctx->arena);
  54. bitset_len = zend_bitset_len(T);
  55. taken_T = (zend_bitset) zend_arena_alloc(&ctx->arena, bitset_len * ZEND_BITSET_ELM_SIZE);
  56. start_of_T = (zend_op **) zend_arena_alloc(&ctx->arena, T * sizeof(zend_op *));
  57. valid_T = (zend_bitset) zend_arena_alloc(&ctx->arena, bitset_len * ZEND_BITSET_ELM_SIZE);
  58. map_T = (int *) zend_arena_alloc(&ctx->arena, T * sizeof(int));
  59. end = op_array->opcodes;
  60. opline = &op_array->opcodes[op_array->last - 1];
  61. /* Find T definition points */
  62. while (opline >= end) {
  63. if (opline->result_type & (IS_VAR | IS_TMP_VAR)) {
  64. start_of_T[VAR_NUM(opline->result.var) - offset] = opline;
  65. }
  66. opline--;
  67. }
  68. zend_bitset_clear(valid_T, bitset_len);
  69. zend_bitset_clear(taken_T, bitset_len);
  70. end = op_array->opcodes;
  71. opline = &op_array->opcodes[op_array->last - 1];
  72. while (opline >= end) {
  73. if ((opline->op1_type & (IS_VAR | IS_TMP_VAR))) {
  74. currT = VAR_NUM(opline->op1.var) - offset;
  75. if (opline->opcode == ZEND_ROPE_END) {
  76. int num = (((opline->extended_value + 1) * sizeof(zend_string*)) + (sizeof(zval) - 1)) / sizeof(zval);
  77. int var;
  78. var = max;
  79. while (var >= 0 && !zend_bitset_in(taken_T, var)) {
  80. var--;
  81. }
  82. max = MAX(max, var + num);
  83. var = var + 1;
  84. map_T[currT] = var;
  85. zend_bitset_incl(valid_T, currT);
  86. zend_bitset_incl(taken_T, var);
  87. opline->op1.var = NUM_VAR(var + offset);
  88. while (num > 1) {
  89. num--;
  90. zend_bitset_incl(taken_T, var + num);
  91. }
  92. } else {
  93. if (!zend_bitset_in(valid_T, currT)) {
  94. int use_new_var = 0;
  95. /* Code in "finally" blocks may modify temorary variables.
  96. * We allocate new temporaries for values that need to
  97. * relive FAST_CALLs.
  98. */
  99. if ((op_array->fn_flags & ZEND_ACC_HAS_FINALLY_BLOCK) &&
  100. (opline->opcode == ZEND_RETURN ||
  101. opline->opcode == ZEND_GENERATOR_RETURN ||
  102. opline->opcode == ZEND_RETURN_BY_REF ||
  103. opline->opcode == ZEND_FREE ||
  104. opline->opcode == ZEND_FE_FREE)) {
  105. zend_op *curr = opline;
  106. while (--curr >= end) {
  107. if (curr->opcode == ZEND_FAST_CALL) {
  108. use_new_var = 1;
  109. break;
  110. } else if (curr->opcode != ZEND_FREE &&
  111. curr->opcode != ZEND_FE_FREE &&
  112. curr->opcode != ZEND_VERIFY_RETURN_TYPE &&
  113. curr->opcode != ZEND_DISCARD_EXCEPTION) {
  114. break;
  115. }
  116. }
  117. }
  118. if (use_new_var) {
  119. i = ++max;
  120. zend_bitset_incl(taken_T, i);
  121. } else {
  122. GET_AVAILABLE_T();
  123. }
  124. map_T[currT] = i;
  125. zend_bitset_incl(valid_T, currT);
  126. }
  127. opline->op1.var = NUM_VAR(map_T[currT] + offset);
  128. }
  129. }
  130. if ((opline->op2_type & (IS_VAR | IS_TMP_VAR))) {
  131. currT = VAR_NUM(opline->op2.var) - offset;
  132. if (!zend_bitset_in(valid_T, currT)) {
  133. GET_AVAILABLE_T();
  134. map_T[currT] = i;
  135. zend_bitset_incl(valid_T, currT);
  136. }
  137. opline->op2.var = NUM_VAR(map_T[currT] + offset);
  138. }
  139. if (opline->result_type & (IS_VAR | IS_TMP_VAR)) {
  140. currT = VAR_NUM(opline->result.var) - offset;
  141. if (zend_bitset_in(valid_T, currT)) {
  142. if (start_of_T[currT] == opline) {
  143. /* ZEND_FAST_CALL can not share temporary var with others
  144. * since the fast_var could also be set by ZEND_HANDLE_EXCEPTION
  145. * which could be ahead of it */
  146. if (opline->opcode != ZEND_FAST_CALL) {
  147. zend_bitset_excl(taken_T, map_T[currT]);
  148. }
  149. }
  150. opline->result.var = NUM_VAR(map_T[currT] + offset);
  151. if (opline->opcode == ZEND_ROPE_INIT) {
  152. if (start_of_T[currT] == opline) {
  153. uint32_t num = ((opline->extended_value * sizeof(zend_string*)) + (sizeof(zval) - 1)) / sizeof(zval);
  154. while (num > 1) {
  155. num--;
  156. zend_bitset_excl(taken_T, map_T[currT]+num);
  157. }
  158. }
  159. }
  160. } else {
  161. /* Code which gets here is using a wrongly built opcode such as RECV() */
  162. GET_AVAILABLE_T();
  163. map_T[currT] = i;
  164. zend_bitset_incl(valid_T, currT);
  165. opline->result.var = NUM_VAR(i + offset);
  166. }
  167. }
  168. if (var_to_free >= 0) {
  169. zend_bitset_excl(taken_T, var_to_free);
  170. var_to_free = -1;
  171. }
  172. opline--;
  173. }
  174. if (op_array->live_range) {
  175. for (i = 0; i < op_array->last_live_range; i++) {
  176. op_array->live_range[i].var =
  177. NUM_VAR(map_T[VAR_NUM(op_array->live_range[i].var & ~ZEND_LIVE_MASK) - offset] + offset) |
  178. (op_array->live_range[i].var & ZEND_LIVE_MASK);
  179. }
  180. }
  181. zend_arena_release(&ctx->arena, checkpoint);
  182. op_array->T = max + 1;
  183. }