nop_removal.c 3.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126
  1. /* pass 10:
  2. * - remove NOPs
  3. */
  4. static void nop_removal(zend_op_array *op_array)
  5. {
  6. zend_op *end, *opline;
  7. zend_uint new_count, i, shift;
  8. int j;
  9. zend_uint *shiftlist;
  10. ALLOCA_FLAG(use_heap);
  11. shiftlist = (zend_uint *)DO_ALLOCA(sizeof(zend_uint) * op_array->last);
  12. i = new_count = shift = 0;
  13. end = op_array->opcodes + op_array->last;
  14. for (opline = op_array->opcodes; opline < end; opline++) {
  15. #if ZEND_EXTENSION_API_NO >= PHP_5_3_X_API_NO
  16. /* GOTO target is unresolved yet. We can't optimize. */
  17. if (opline->opcode == ZEND_GOTO &&
  18. Z_TYPE(ZEND_OP2_LITERAL(opline)) != IS_LONG) {
  19. /* TODO: in general we can avoid this restriction */
  20. FREE_ALLOCA(shiftlist);
  21. return;
  22. }
  23. #endif
  24. /* Kill JMP-over-NOP-s */
  25. if (opline->opcode == ZEND_JMP && ZEND_OP1(opline).opline_num > i) {
  26. /* check if there are only NOPs under the branch */
  27. zend_op *target = op_array->opcodes + ZEND_OP1(opline).opline_num - 1;
  28. while (target->opcode == ZEND_NOP) {
  29. target--;
  30. }
  31. if (target == opline) {
  32. /* only NOPs */
  33. opline->opcode = ZEND_NOP;
  34. }
  35. }
  36. shiftlist[i++] = shift;
  37. if (opline->opcode == ZEND_NOP) {
  38. shift++;
  39. } else {
  40. if (shift) {
  41. op_array->opcodes[new_count] = *opline;
  42. }
  43. new_count++;
  44. }
  45. }
  46. if (shift) {
  47. op_array->last = new_count;
  48. end = op_array->opcodes + op_array->last;
  49. /* update JMPs */
  50. for (opline = op_array->opcodes; opline<end; opline++) {
  51. switch (opline->opcode) {
  52. case ZEND_JMP:
  53. #if ZEND_EXTENSION_API_NO >= PHP_5_3_X_API_NO
  54. case ZEND_GOTO:
  55. #endif
  56. #if ZEND_EXTENSION_API_NO > PHP_5_4_X_API_NO
  57. case ZEND_FAST_CALL:
  58. #endif
  59. ZEND_OP1(opline).opline_num -= shiftlist[ZEND_OP1(opline).opline_num];
  60. break;
  61. case ZEND_JMPZ:
  62. case ZEND_JMPNZ:
  63. case ZEND_JMPZ_EX:
  64. case ZEND_JMPNZ_EX:
  65. case ZEND_FE_FETCH:
  66. case ZEND_FE_RESET:
  67. case ZEND_NEW:
  68. #if ZEND_EXTENSION_API_NO >= PHP_5_3_X_API_NO
  69. case ZEND_JMP_SET:
  70. #endif
  71. #if ZEND_EXTENSION_API_NO > PHP_5_3_X_API_NO
  72. case ZEND_JMP_SET_VAR:
  73. #endif
  74. ZEND_OP2(opline).opline_num -= shiftlist[ZEND_OP2(opline).opline_num];
  75. break;
  76. case ZEND_JMPZNZ:
  77. ZEND_OP2(opline).opline_num -= shiftlist[ZEND_OP2(opline).opline_num];
  78. opline->extended_value -= shiftlist[opline->extended_value];
  79. break;
  80. case ZEND_CATCH:
  81. opline->extended_value -= shiftlist[opline->extended_value];
  82. break;
  83. }
  84. }
  85. /* update brk/cont array */
  86. for (j = 0; j < op_array->last_brk_cont; j++) {
  87. op_array->brk_cont_array[j].brk -= shiftlist[op_array->brk_cont_array[j].brk];
  88. op_array->brk_cont_array[j].cont -= shiftlist[op_array->brk_cont_array[j].cont];
  89. op_array->brk_cont_array[j].start -= shiftlist[op_array->brk_cont_array[j].start];
  90. }
  91. /* update try/catch array */
  92. for (j = 0; j < op_array->last_try_catch; j++) {
  93. op_array->try_catch_array[j].try_op -= shiftlist[op_array->try_catch_array[j].try_op];
  94. op_array->try_catch_array[j].catch_op -= shiftlist[op_array->try_catch_array[j].catch_op];
  95. #if ZEND_EXTENSION_API_NO > PHP_5_4_X_API_NO
  96. if (op_array->try_catch_array[j].finally_op) {
  97. op_array->try_catch_array[j].finally_op -= shiftlist[op_array->try_catch_array[j].finally_op];
  98. op_array->try_catch_array[j].finally_end -= shiftlist[op_array->try_catch_array[j].finally_end];
  99. }
  100. #endif
  101. }
  102. #if ZEND_EXTENSION_API_NO >= PHP_5_3_X_API_NO
  103. /* update early binding list */
  104. if (op_array->early_binding != (zend_uint)-1) {
  105. zend_uint *opline_num = &op_array->early_binding;
  106. do {
  107. *opline_num -= shiftlist[*opline_num];
  108. opline_num = &ZEND_RESULT(&op_array->opcodes[*opline_num]).opline_num;
  109. } while (*opline_num != (zend_uint)-1);
  110. }
  111. #endif
  112. }
  113. FREE_ALLOCA(shiftlist);
  114. }