]> granicus.if.org Git - php/commitdiff
Refactor operator implementations
authorNikita Popov <nikita.ppv@gmail.com>
Tue, 31 Mar 2020 16:03:30 +0000 (18:03 +0200)
committerNikita Popov <nikita.ppv@gmail.com>
Tue, 31 Mar 2020 18:49:39 +0000 (20:49 +0200)
Instead of looping, use straight-line code with the following
layout:

1. Try to apply the base operation on the dereferenced operands.
2. Try overloaded object operations.
3. Try to convert operands to number, else error out.
4. Apply the base operation on the converted operands.

This makes the code easier to reason about and fixes some edge-case
bugs:

1. We should only try invoking operator overloading once prior to
   type conversion. Previously it was invoked both before and after
   type conversion.
2. We should not modify any values if an exception is thrown.
   Previously we sometimes modified the LHS of a compound assignment
   operator.
3. If conversion of the first operand fails, we no longer try to
   convert the second operand. I think the previous behavior here
   was fine as well, but this still seems a more typical.

This will also make some followup changes I have in mind simpler.

Zend/tests/add_002.phpt
Zend/tests/add_007.phpt
Zend/tests/compound_assign_failure.phpt
Zend/zend_operators.c

index 5d862ef729df59a18fe11506e27985e16829d1f7..da99499864078dd884a951596e6779172d75fd00 100644 (file)
@@ -20,12 +20,8 @@ var_dump($c);
 echo "Done\n";
 ?>
 --EXPECTF--
-Notice: Object of class stdClass could not be converted to number in %sadd_002.php on line %d
-
 Exception: Unsupported operand types
 
-Notice: Object of class stdClass could not be converted to number in %s on line %d
-
 Fatal error: Uncaught Error: Unsupported operand types in %s:%d
 Stack trace:
 #0 {main}
index e490826d8f0a8fbb2da8f653fb8e4faf2e8b11ad..e7e0de79baf88fab25db8f44ad88f2bafb6b3140 100644 (file)
@@ -19,12 +19,8 @@ var_dump($c);
 echo "Done\n";
 ?>
 --EXPECTF--
-Warning: A non-numeric value encountered in %s on line %d
-
 Exception: Unsupported operand types
 
-Warning: A non-numeric value encountered in %s on line %d
-
 Fatal error: Uncaught Error: Unsupported operand types in %s:%d
 Stack trace:
 #0 {main}
index b4563e341f034a334e167c9872ad55e90401637a..1780725d93dcece49b88050a400ecb98810d8a5b 100644 (file)
@@ -208,19 +208,19 @@ string(3) "foo"
 object(stdClass)#%d (0) {
 }
 int(1)
-int(0)
+string(3) "foo"
 object(stdClass)#%d (0) {
 }
 int(1)
-int(0)
+string(3) "foo"
 object(stdClass)#%d (0) {
 }
 int(1)
-int(0)
+string(3) "foo"
 object(stdClass)#%d (0) {
 }
 int(1)
-int(0)
+string(3) "foo"
 object(stdClass)#%d (0) {
 }
 int(1)
@@ -228,7 +228,7 @@ string(3) "foo"
 object(stdClass)#%d (0) {
 }
 int(1)
-int(0)
+string(3) "foo"
 object(stdClass)#%d (0) {
 }
 int(1)
index 54298a00a92cd16d21714424728c3135b68cf8cd..af21693f3b6c2ba1c57480b39f44a10d7ef60b4c 100644 (file)
@@ -145,7 +145,7 @@ ZEND_API zend_long ZEND_FASTCALL zend_atol(const char *str, size_t str_len) /* {
 
 /* }}} */
 
-static void ZEND_FASTCALL _convert_scalar_to_number(zval *op, zend_bool silent, zend_bool check) /* {{{ */
+ZEND_API void ZEND_FASTCALL convert_scalar_to_number(zval *op) /* {{{ */
 {
 try_again:
        switch (Z_TYPE_P(op)) {
@@ -157,11 +157,8 @@ try_again:
                                zend_string *str;
 
                                str = Z_STR_P(op);
-                               if ((Z_TYPE_INFO_P(op)=is_numeric_string(ZSTR_VAL(str), ZSTR_LEN(str), &Z_LVAL_P(op), &Z_DVAL_P(op), silent ? 1 : -1)) == 0) {
+                               if ((Z_TYPE_INFO_P(op)=is_numeric_string(ZSTR_VAL(str), ZSTR_LEN(str), &Z_LVAL_P(op), &Z_DVAL_P(op), 1)) == 0) {
                                        ZVAL_LONG(op, 0);
-                                       if (!silent) {
-                                               zend_error(E_WARNING, "A non-numeric value encountered");
-                                       }
                                }
                                zend_string_release_ex(str, 0);
                                break;
@@ -185,9 +182,6 @@ try_again:
                                zval dst;
 
                                convert_object_to_type(op, &dst, _IS_NUMBER);
-                               if (check && UNEXPECTED(EG(exception))) {
-                                       return;
-                               }
                                zval_ptr_dtor(op);
 
                                if (Z_TYPE(dst) == IS_LONG || Z_TYPE(dst) == IS_DOUBLE) {
@@ -201,14 +195,8 @@ try_again:
 }
 /* }}} */
 
-ZEND_API void ZEND_FASTCALL convert_scalar_to_number(zval *op) /* {{{ */
-{
-       _convert_scalar_to_number(op, 1, 0);
-}
-/* }}} */
-
-/* {{{ _zendi_convert_scalar_to_number_ex */
-static zend_always_inline zval* _zendi_convert_scalar_to_number_ex(zval *op, zval *holder, zend_bool silent) /* {{{ */
+/* {{{ _zendi_convert_scalar_to_number */
+static zend_never_inline zval* ZEND_FASTCALL _zendi_convert_scalar_to_number_silent(zval *op, zval *holder) /* {{{ */
 {
        switch (Z_TYPE_P(op)) {
                case IS_NULL:
@@ -219,11 +207,8 @@ static zend_always_inline zval* _zendi_convert_scalar_to_number_ex(zval *op, zva
                        ZVAL_LONG(holder, 1);
                        return holder;
                case IS_STRING:
-                       if ((Z_TYPE_INFO_P(holder) = is_numeric_string(Z_STRVAL_P(op), Z_STRLEN_P(op), &Z_LVAL_P(holder), &Z_DVAL_P(holder), silent ? 1 : -1)) == 0) {
+                       if ((Z_TYPE_INFO_P(holder) = is_numeric_string(Z_STRVAL_P(op), Z_STRLEN_P(op), &Z_LVAL_P(holder), &Z_DVAL_P(holder), 1)) == 0) {
                                ZVAL_LONG(holder, 0);
-                               if (!silent) {
-                                       zend_error(E_WARNING, "A non-numeric value encountered");
-                               }
                        }
                        return holder;
                case IS_RESOURCE:
@@ -244,25 +229,52 @@ static zend_always_inline zval* _zendi_convert_scalar_to_number_ex(zval *op, zva
 }
 /* }}} */
 
-/* {{{ _zendi_convert_scalar_to_number */
-static zend_never_inline zval* ZEND_FASTCALL _zendi_convert_scalar_to_number_silent(zval *op, zval *holder) /* {{{ */
+static zend_never_inline int ZEND_FASTCALL _zendi_try_convert_scalar_to_number(zval *op, zval *holder) /* {{{ */
 {
-       return _zendi_convert_scalar_to_number_ex(op, holder, 1);
+       switch (Z_TYPE_P(op)) {
+               case IS_NULL:
+               case IS_FALSE:
+                       ZVAL_LONG(holder, 0);
+                       return SUCCESS;
+               case IS_TRUE:
+                       ZVAL_LONG(holder, 1);
+                       return SUCCESS;
+               case IS_STRING:
+                       if ((Z_TYPE_INFO_P(holder) = is_numeric_string(Z_STRVAL_P(op), Z_STRLEN_P(op), &Z_LVAL_P(holder), &Z_DVAL_P(holder), -1)) == 0) {
+                               ZVAL_LONG(holder, 0);
+                               zend_error(E_WARNING, "A non-numeric value encountered");
+                               if (UNEXPECTED(EG(exception))) {
+                                       return FAILURE;
+                               }
+                       }
+                       return SUCCESS;
+               case IS_RESOURCE:
+                       ZVAL_LONG(holder, Z_RES_HANDLE_P(op));
+                       return SUCCESS;
+               case IS_OBJECT:
+                       convert_object_to_type(op, holder, _IS_NUMBER);
+                       if (UNEXPECTED(EG(exception))) {
+                               return FAILURE;
+                       }
+                       if (UNEXPECTED(Z_TYPE_P(holder) != IS_LONG && Z_TYPE_P(holder) != IS_DOUBLE)) {
+                               ZVAL_LONG(holder, 1);
+                       }
+                       return SUCCESS;
+               default:
+                       return FAILURE;
+       }
 }
 /* }}} */
 
-/* {{{ _zendi_convert_scalar_to_number_noisy */
-static zend_never_inline zval* ZEND_FASTCALL _zendi_convert_scalar_to_number_noisy(zval *op, zval *holder) /* {{{ */
+static zend_always_inline int zendi_try_convert_scalar_to_number(zval *op, zval *holder)
 {
-       return _zendi_convert_scalar_to_number_ex(op, holder, 0);
+       if (Z_TYPE_P(op) == IS_LONG || Z_TYPE_P(op) == IS_DOUBLE) {
+               ZVAL_COPY_VALUE(holder, op);
+               return SUCCESS;
+       } else {
+               return _zendi_try_convert_scalar_to_number(op, holder);
+       }
 }
-/* }}} */
-
-#define zendi_convert_scalar_to_number_noisy(op, holder, result) \
-       ((Z_TYPE_P(op) == IS_LONG || Z_TYPE_P(op) == IS_DOUBLE) ? (op) : \
-               (((op) == result) \
-                       ? (_convert_scalar_to_number((op), /* silent */ 0, 1), (op)) : \
-                       _zendi_convert_scalar_to_number_noisy((op), holder)))
 
 #define convert_op1_op2_long(op1, op1_lval, op2, op2_lval, result, op, op_func) \
        do {                                                                                                                            \
@@ -941,42 +953,35 @@ static zend_always_inline int add_function_fast(zval *result, zval *op1, zval *o
 
 static zend_never_inline int ZEND_FASTCALL add_function_slow(zval *result, zval *op1, zval *op2) /* {{{ */
 {
-       zval op1_copy, op2_copy;
-       int converted = 0;
+       ZVAL_DEREF(op1);
+       ZVAL_DEREF(op2);
+       if (add_function_fast(result, op1, op2) == SUCCESS) {
+               return SUCCESS;
+       }
 
-       while (1) {
-               if (Z_ISREF_P(op1)) {
-                       op1 = Z_REFVAL_P(op1);
-               } else if (Z_ISREF_P(op2)) {
-                       op2 = Z_REFVAL_P(op2);
-               } else if (!converted) {
-                       ZEND_TRY_BINARY_OBJECT_OPERATION(ZEND_ADD, add_function);
-
-                       if (EXPECTED(op1 != op2)) {
-                               op1 = zendi_convert_scalar_to_number_noisy(op1, &op1_copy, result);
-                               op2 = zendi_convert_scalar_to_number_noisy(op2, &op2_copy, result);
-                       } else {
-                               op1 = zendi_convert_scalar_to_number_noisy(op1, &op1_copy, result);
-                               op2 = op1;
-                       }
-                       if (EG(exception)) {
-                               if (result != op1) {
-                                       ZVAL_UNDEF(result);
-                               }
-                               return FAILURE;
-                       }
-                       converted = 1;
-               } else {
-                       if (result != op1) {
-                               ZVAL_UNDEF(result);
-                       }
+       ZEND_TRY_BINARY_OBJECT_OPERATION(ZEND_ADD, add_function);
+
+       zval op1_copy, op2_copy;
+       if (UNEXPECTED(zendi_try_convert_scalar_to_number(op1, &op1_copy) == FAILURE)
+                       || UNEXPECTED(zendi_try_convert_scalar_to_number(op2, &op2_copy) == FAILURE)) {
+               if (!EG(exception)) {
                        zend_throw_error(NULL, "Unsupported operand types");
-                       return FAILURE; /* unknown datatype */
                }
-               if (add_function_fast(result, op1, op2) == SUCCESS) {
-                       return SUCCESS;
+               if (result != op1) {
+                       ZVAL_UNDEF(result);
                }
+               return FAILURE;
        }
+
+       if (result == op1) {
+               zval_ptr_dtor(result);
+       }
+
+       if (add_function_fast(result, &op1_copy, &op2_copy) == SUCCESS) {
+               return SUCCESS;
+       }
+
+       ZEND_ASSERT(0 && "Operation must succeed");
 } /* }}} */
 
 ZEND_API int ZEND_FASTCALL add_function(zval *result, zval *op1, zval *op2) /* {{{ */
@@ -1013,42 +1018,35 @@ static zend_always_inline int sub_function_fast(zval *result, zval *op1, zval *o
 
 static zend_never_inline int ZEND_FASTCALL sub_function_slow(zval *result, zval *op1, zval *op2) /* {{{ */
 {
+       ZVAL_DEREF(op1);
+       ZVAL_DEREF(op2);
+       if (sub_function_fast(result, op1, op2) == SUCCESS) {
+               return SUCCESS;
+       }
+
+       ZEND_TRY_BINARY_OBJECT_OPERATION(ZEND_SUB, sub_function);
+
        zval op1_copy, op2_copy;
-       int converted = 0;
-       while (1) {
-               if (Z_ISREF_P(op1)) {
-                       op1 = Z_REFVAL_P(op1);
-               } else if (Z_ISREF_P(op2)) {
-                       op2 = Z_REFVAL_P(op2);
-               } else if (!converted) {
-                       ZEND_TRY_BINARY_OBJECT_OPERATION(ZEND_SUB, sub_function);
-
-                       if (EXPECTED(op1 != op2)) {
-                               op1 = zendi_convert_scalar_to_number_noisy(op1, &op1_copy, result);
-                               op2 = zendi_convert_scalar_to_number_noisy(op2, &op2_copy, result);
-                       } else {
-                               op1 = zendi_convert_scalar_to_number_noisy(op1, &op1_copy, result);
-                               op2 = op1;
-                       }
-                       if (EG(exception)) {
-                               if (result != op1) {
-                                       ZVAL_UNDEF(result);
-                               }
-                               return FAILURE;
-                       }
-                       converted = 1;
-               } else {
-                       if (result != op1) {
-                               ZVAL_UNDEF(result);
-                       }
+       if (UNEXPECTED(zendi_try_convert_scalar_to_number(op1, &op1_copy) == FAILURE)
+                       || UNEXPECTED(zendi_try_convert_scalar_to_number(op2, &op2_copy) == FAILURE)) {
+               if (!EG(exception)) {
                        zend_throw_error(NULL, "Unsupported operand types");
-                       return FAILURE; /* unknown datatype */
                }
-               if (sub_function_fast(result, op1, op2) == SUCCESS) {
-                       return SUCCESS;
+               if (result != op1) {
+                       ZVAL_UNDEF(result);
                }
+               return FAILURE;
        }
 
+       if (result == op1) {
+               zval_ptr_dtor(result);
+       }
+
+       if (sub_function_fast(result, &op1_copy, &op2_copy) == SUCCESS) {
+               return SUCCESS;
+       }
+
+       ZEND_ASSERT(0 && "Operation must succeed");
 }
 /* }}} */
 
@@ -1062,249 +1060,243 @@ ZEND_API int ZEND_FASTCALL sub_function(zval *result, zval *op1, zval *op2) /* {
 }
 /* }}} */
 
-ZEND_API int ZEND_FASTCALL mul_function(zval *result, zval *op1, zval *op2) /* {{{ */
+static zend_always_inline int mul_function_fast(zval *result, zval *op1, zval *op2) /* {{{ */
 {
-       zval op1_copy, op2_copy;
-       int converted = 0;
+       zend_uchar type_pair = TYPE_PAIR(Z_TYPE_P(op1), Z_TYPE_P(op2));
 
-       while (1) {
-               zend_uchar type_pair = TYPE_PAIR(Z_TYPE_P(op1), Z_TYPE_P(op2));
+       if (EXPECTED(type_pair == TYPE_PAIR(IS_LONG, IS_LONG))) {
+               zend_long overflow;
+               ZEND_SIGNED_MULTIPLY_LONG(
+                       Z_LVAL_P(op1), Z_LVAL_P(op2),
+                       Z_LVAL_P(result), Z_DVAL_P(result), overflow);
+               Z_TYPE_INFO_P(result) = overflow ? IS_DOUBLE : IS_LONG;
+               return SUCCESS;
+       } else if (EXPECTED(type_pair == TYPE_PAIR(IS_DOUBLE, IS_DOUBLE))) {
+               ZVAL_DOUBLE(result, Z_DVAL_P(op1) * Z_DVAL_P(op2));
+               return SUCCESS;
+       } else if (EXPECTED(type_pair == TYPE_PAIR(IS_LONG, IS_DOUBLE))) {
+               ZVAL_DOUBLE(result, ((double)Z_LVAL_P(op1)) * Z_DVAL_P(op2));
+               return SUCCESS;
+       } else if (EXPECTED(type_pair == TYPE_PAIR(IS_DOUBLE, IS_LONG))) {
+               ZVAL_DOUBLE(result, Z_DVAL_P(op1) * ((double)Z_LVAL_P(op2)));
+               return SUCCESS;
+       } else {
+               return FAILURE;
+       }
+}
 
-               if (EXPECTED(type_pair == TYPE_PAIR(IS_LONG, IS_LONG))) {
-                       zend_long overflow;
+static zend_never_inline int ZEND_FASTCALL mul_function_slow(zval *result, zval *op1, zval *op2) /* {{{ */
+{
+       ZVAL_DEREF(op1);
+       ZVAL_DEREF(op2);
+       if (mul_function_fast(result, op1, op2) == SUCCESS) {
+               return SUCCESS;
+       }
 
-                       ZEND_SIGNED_MULTIPLY_LONG(Z_LVAL_P(op1),Z_LVAL_P(op2), Z_LVAL_P(result),Z_DVAL_P(result),overflow);
-                       Z_TYPE_INFO_P(result) = overflow ? IS_DOUBLE : IS_LONG;
-                       return SUCCESS;
+       ZEND_TRY_BINARY_OBJECT_OPERATION(ZEND_MUL, mul_function);
 
-               } else if (EXPECTED(type_pair == TYPE_PAIR(IS_DOUBLE, IS_DOUBLE))) {
-                       ZVAL_DOUBLE(result, Z_DVAL_P(op1) * Z_DVAL_P(op2));
-                       return SUCCESS;
+       zval op1_copy, op2_copy;
+       if (UNEXPECTED(zendi_try_convert_scalar_to_number(op1, &op1_copy) == FAILURE)
+                       || UNEXPECTED(zendi_try_convert_scalar_to_number(op2, &op2_copy) == FAILURE)) {
+               if (!EG(exception)) {
+                       zend_throw_error(NULL, "Unsupported operand types");
+               }
+               if (result != op1) {
+                       ZVAL_UNDEF(result);
+               }
+               return FAILURE;
+       }
 
-               } else if (EXPECTED(type_pair == TYPE_PAIR(IS_LONG, IS_DOUBLE))) {
-                       ZVAL_DOUBLE(result, ((double)Z_LVAL_P(op1)) * Z_DVAL_P(op2));
-                       return SUCCESS;
+       if (result == op1) {
+               zval_ptr_dtor(result);
+       }
 
-               } else if (EXPECTED(type_pair == TYPE_PAIR(IS_DOUBLE, IS_LONG))) {
-                       ZVAL_DOUBLE(result, Z_DVAL_P(op1) * ((double)Z_LVAL_P(op2)));
-                       return SUCCESS;
+       if (mul_function_fast(result, &op1_copy, &op2_copy) == SUCCESS) {
+               return SUCCESS;
+       }
 
-               } else {
-                       if (Z_ISREF_P(op1)) {
-                               op1 = Z_REFVAL_P(op1);
-                       } else if (Z_ISREF_P(op2)) {
-                               op2 = Z_REFVAL_P(op2);
-                       } else if (!converted) {
-                               ZEND_TRY_BINARY_OBJECT_OPERATION(ZEND_MUL, mul_function);
+       ZEND_ASSERT(0 && "Operation must succeed");
+}
+/* }}} */
 
-                               if (EXPECTED(op1 != op2)) {
-                                       op1 = zendi_convert_scalar_to_number_noisy(op1, &op1_copy, result);
-                                       op2 = zendi_convert_scalar_to_number_noisy(op2, &op2_copy, result);
-                               } else {
-                                       op1 = zendi_convert_scalar_to_number_noisy(op1, &op1_copy, result);
-                                       op2 = op1;
-                               }
-                               if (EG(exception)) {
-                                       if (result != op1) {
-                                               ZVAL_UNDEF(result);
-                                       }
-                                       return FAILURE;
-                               }
-                               converted = 1;
-                       } else {
-                               if (result != op1) {
-                                       ZVAL_UNDEF(result);
-                               }
-                               zend_throw_error(NULL, "Unsupported operand types");
-                               return FAILURE; /* unknown datatype */
-                       }
-               }
+ZEND_API int ZEND_FASTCALL mul_function(zval *result, zval *op1, zval *op2) /* {{{ */
+{
+       if (mul_function_fast(result, op1, op2) == SUCCESS) {
+               return SUCCESS;
+       } else {
+               return mul_function_slow(result, op1, op2);
        }
 }
 /* }}} */
 
-ZEND_API int ZEND_FASTCALL pow_function(zval *result, zval *op1, zval *op2) /* {{{ */
+static int ZEND_FASTCALL pow_function_base(zval *result, zval *op1, zval *op2) /* {{{ */
 {
-       zval op1_copy, op2_copy;
-       int converted = 0;
-
-       while (1) {
-               zend_uchar type_pair = TYPE_PAIR(Z_TYPE_P(op1), Z_TYPE_P(op2));
-
-               if (EXPECTED(type_pair == TYPE_PAIR(IS_LONG, IS_LONG))) {
-                       if (Z_LVAL_P(op2) >= 0) {
-                               zend_long l1 = 1, l2 = Z_LVAL_P(op1), i = Z_LVAL_P(op2);
-
-                               if (i == 0) {
-                                       ZVAL_LONG(result, 1L);
-                                       return SUCCESS;
-                               } else if (l2 == 0) {
-                                       ZVAL_LONG(result, 0);
-                                       return SUCCESS;
-                               }
+       zend_uchar type_pair = TYPE_PAIR(Z_TYPE_P(op1), Z_TYPE_P(op2));
 
-                               while (i >= 1) {
-                                       zend_long overflow;
-                                       double dval = 0.0;
+       if (EXPECTED(type_pair == TYPE_PAIR(IS_LONG, IS_LONG))) {
+               if (Z_LVAL_P(op2) >= 0) {
+                       zend_long l1 = 1, l2 = Z_LVAL_P(op1), i = Z_LVAL_P(op2);
 
-                                       if (i % 2) {
-                                               --i;
-                                               ZEND_SIGNED_MULTIPLY_LONG(l1, l2, l1, dval, overflow);
-                                               if (overflow) {
-                                                       ZVAL_DOUBLE(result, dval * pow(l2, i));
-                                                       return SUCCESS;
-                                               }
-                                       } else {
-                                               i /= 2;
-                                               ZEND_SIGNED_MULTIPLY_LONG(l2, l2, l2, dval, overflow);
-                                               if (overflow) {
-                                                       ZVAL_DOUBLE(result, (double)l1 * pow(dval, i));
-                                                       return SUCCESS;
-                                               }
-                                       }
-                               }
-                               /* i == 0 */
-                               ZVAL_LONG(result, l1);
-                       } else {
-                               ZVAL_DOUBLE(result, pow((double)Z_LVAL_P(op1), (double)Z_LVAL_P(op2)));
+                       if (i == 0) {
+                               ZVAL_LONG(result, 1L);
+                               return SUCCESS;
+                       } else if (l2 == 0) {
+                               ZVAL_LONG(result, 0);
+                               return SUCCESS;
                        }
-                       return SUCCESS;
 
-               } else if (EXPECTED(type_pair == TYPE_PAIR(IS_DOUBLE, IS_DOUBLE))) {
-                       ZVAL_DOUBLE(result, pow(Z_DVAL_P(op1), Z_DVAL_P(op2)));
-                       return SUCCESS;
+                       while (i >= 1) {
+                               zend_long overflow;
+                               double dval = 0.0;
 
-               } else if (EXPECTED(type_pair == TYPE_PAIR(IS_LONG, IS_DOUBLE))) {
-                       ZVAL_DOUBLE(result, pow((double)Z_LVAL_P(op1), Z_DVAL_P(op2)));
-                       return SUCCESS;
-
-               } else if (EXPECTED(type_pair == TYPE_PAIR(IS_DOUBLE, IS_LONG))) {
-                       ZVAL_DOUBLE(result, pow(Z_DVAL_P(op1), (double)Z_LVAL_P(op2)));
-                       return SUCCESS;
-
-               } else {
-                       if (Z_ISREF_P(op1)) {
-                               op1 = Z_REFVAL_P(op1);
-                       } else if (Z_ISREF_P(op2)) {
-                               op2 = Z_REFVAL_P(op2);
-                       } else if (!converted) {
-                               ZEND_TRY_BINARY_OBJECT_OPERATION(ZEND_POW, pow_function);
-
-                               if (Z_TYPE_P(op1) == IS_ARRAY || Z_TYPE_P(op2) == IS_ARRAY) {
-                                       if (result != op1) {
-                                               ZVAL_UNDEF(result);
+                               if (i % 2) {
+                                       --i;
+                                       ZEND_SIGNED_MULTIPLY_LONG(l1, l2, l1, dval, overflow);
+                                       if (overflow) {
+                                               ZVAL_DOUBLE(result, dval * pow(l2, i));
+                                               return SUCCESS;
                                        }
-                                       zend_throw_error(NULL, "Unsupported operand types");
-                                       return FAILURE;
-                               }
-
-                               if (EXPECTED(op1 != op2)) {
-                                       op1 = zendi_convert_scalar_to_number_noisy(op1, &op1_copy, result);
-                                       op2 = zendi_convert_scalar_to_number_noisy(op2, &op2_copy, result);
                                } else {
-                                       op1 = zendi_convert_scalar_to_number_noisy(op1, &op1_copy, result);
-                                       op2 = op1;
-                               }
-
-                               if (EG(exception)) {
-                                       if (result != op1) {
-                                               ZVAL_UNDEF(result);
+                                       i /= 2;
+                                       ZEND_SIGNED_MULTIPLY_LONG(l2, l2, l2, dval, overflow);
+                                       if (overflow) {
+                                               ZVAL_DOUBLE(result, (double)l1 * pow(dval, i));
+                                               return SUCCESS;
                                        }
-                                       return FAILURE;
                                }
-                               converted = 1;
-                       } else {
-                               if (result != op1) {
-                                       ZVAL_UNDEF(result);
-                               }
-                               zend_throw_error(NULL, "Unsupported operand types");
-                               return FAILURE;
                        }
+                       /* i == 0 */
+                       ZVAL_LONG(result, l1);
+               } else {
+                       ZVAL_DOUBLE(result, pow((double)Z_LVAL_P(op1), (double)Z_LVAL_P(op2)));
                }
+               return SUCCESS;
+       } else if (EXPECTED(type_pair == TYPE_PAIR(IS_DOUBLE, IS_DOUBLE))) {
+               ZVAL_DOUBLE(result, pow(Z_DVAL_P(op1), Z_DVAL_P(op2)));
+               return SUCCESS;
+       } else if (EXPECTED(type_pair == TYPE_PAIR(IS_LONG, IS_DOUBLE))) {
+               ZVAL_DOUBLE(result, pow((double)Z_LVAL_P(op1), Z_DVAL_P(op2)));
+               return SUCCESS;
+       } else if (EXPECTED(type_pair == TYPE_PAIR(IS_DOUBLE, IS_LONG))) {
+               ZVAL_DOUBLE(result, pow(Z_DVAL_P(op1), (double)Z_LVAL_P(op2)));
+               return SUCCESS;
+       } else {
+               return FAILURE;
        }
 }
 /* }}} */
 
-#ifdef __clang__
-__attribute__((no_sanitize("float-divide-by-zero")))
-#endif
-ZEND_API int ZEND_FASTCALL div_function(zval *result, zval *op1, zval *op2) /* {{{ */
+ZEND_API int ZEND_FASTCALL pow_function(zval *result, zval *op1, zval *op2) /* {{{ */
 {
+       ZVAL_DEREF(op1);
+       ZVAL_DEREF(op2);
+       if (pow_function_base(result, op1, op2) == SUCCESS) {
+               return SUCCESS;
+       }
+
+       ZEND_TRY_BINARY_OBJECT_OPERATION(ZEND_POW, pow_function);
+
        zval op1_copy, op2_copy;
-       int converted = 0;
+       if (UNEXPECTED(zendi_try_convert_scalar_to_number(op1, &op1_copy) == FAILURE)
+                       || UNEXPECTED(zendi_try_convert_scalar_to_number(op2, &op2_copy) == FAILURE)) {
+               if (!EG(exception)) {
+                       zend_throw_error(NULL, "Unsupported operand types");
+               }
+               if (result != op1) {
+                       ZVAL_UNDEF(result);
+               }
+               return FAILURE;
+       }
 
-       while (1) {
-               zend_uchar type_pair = TYPE_PAIR(Z_TYPE_P(op1), Z_TYPE_P(op2));
+       if (result == op1) {
+               zval_ptr_dtor(result);
+       }
 
-               if (EXPECTED(type_pair == TYPE_PAIR(IS_LONG, IS_LONG))) {
-                       if (Z_LVAL_P(op2) == 0) {
-                               zend_error(E_WARNING, "Division by zero");
-                               ZVAL_DOUBLE(result, ((double) Z_LVAL_P(op1) / (double) Z_LVAL_P(op2)));
-                               return SUCCESS;
-                       } else if (Z_LVAL_P(op2) == -1 && Z_LVAL_P(op1) == ZEND_LONG_MIN) {
-                               /* Prevent overflow error/crash */
-                               ZVAL_DOUBLE(result, (double) ZEND_LONG_MIN / -1);
-                               return SUCCESS;
-                       }
-                       if (Z_LVAL_P(op1) % Z_LVAL_P(op2) == 0) { /* integer */
-                               ZVAL_LONG(result, Z_LVAL_P(op1) / Z_LVAL_P(op2));
-                       } else {
-                               ZVAL_DOUBLE(result, ((double) Z_LVAL_P(op1)) / Z_LVAL_P(op2));
-                       }
-                       return SUCCESS;
+       if (pow_function_base(result, &op1_copy, &op2_copy) == SUCCESS) {
+               return SUCCESS;
+       }
 
-               } else if (EXPECTED(type_pair == TYPE_PAIR(IS_DOUBLE, IS_DOUBLE))) {
-                       if (Z_DVAL_P(op2) == 0) {
-                               zend_error(E_WARNING, "Division by zero");
-                       }
-                       ZVAL_DOUBLE(result, Z_DVAL_P(op1) / Z_DVAL_P(op2));
-                       return SUCCESS;
+       ZEND_ASSERT(0 && "Operation must succeed");
+}
+/* }}} */
 
-               } else if (EXPECTED(type_pair == TYPE_PAIR(IS_DOUBLE, IS_LONG))) {
-                       if (Z_LVAL_P(op2) == 0) {
-                               zend_error(E_WARNING, "Division by zero");
-                       }
-                       ZVAL_DOUBLE(result, Z_DVAL_P(op1) / (double)Z_LVAL_P(op2));
-                       return SUCCESS;
+#ifdef __clang__
+__attribute__((no_sanitize("float-divide-by-zero")))
+#endif
+static int ZEND_FASTCALL div_function_base(zval *result, zval *op1, zval *op2) /* {{{ */
+{
+       zend_uchar type_pair = TYPE_PAIR(Z_TYPE_P(op1), Z_TYPE_P(op2));
 
-               } else if (EXPECTED(type_pair == TYPE_PAIR(IS_LONG, IS_DOUBLE))) {
-                       if (Z_DVAL_P(op2) == 0) {
-                               zend_error(E_WARNING, "Division by zero");
-                       }
-                       ZVAL_DOUBLE(result, (double)Z_LVAL_P(op1) / Z_DVAL_P(op2));
+       if (EXPECTED(type_pair == TYPE_PAIR(IS_LONG, IS_LONG))) {
+               if (Z_LVAL_P(op2) == 0) {
+                       zend_error(E_WARNING, "Division by zero");
+                       ZVAL_DOUBLE(result, ((double) Z_LVAL_P(op1) / (double) Z_LVAL_P(op2)));
                        return SUCCESS;
-
+               } else if (Z_LVAL_P(op2) == -1 && Z_LVAL_P(op1) == ZEND_LONG_MIN) {
+                       /* Prevent overflow error/crash */
+                       ZVAL_DOUBLE(result, (double) ZEND_LONG_MIN / -1);
+                       return SUCCESS;
+               }
+               if (Z_LVAL_P(op1) % Z_LVAL_P(op2) == 0) { /* integer */
+                       ZVAL_LONG(result, Z_LVAL_P(op1) / Z_LVAL_P(op2));
                } else {
-                       if (Z_ISREF_P(op1)) {
-                               op1 = Z_REFVAL_P(op1);
-                       } else if (Z_ISREF_P(op2)) {
-                               op2 = Z_REFVAL_P(op2);
-                       } else if (!converted) {
-                               ZEND_TRY_BINARY_OBJECT_OPERATION(ZEND_DIV, div_function);
+                       ZVAL_DOUBLE(result, ((double) Z_LVAL_P(op1)) / Z_LVAL_P(op2));
+               }
+               return SUCCESS;
+       } else if (EXPECTED(type_pair == TYPE_PAIR(IS_DOUBLE, IS_DOUBLE))) {
+               if (Z_DVAL_P(op2) == 0) {
+                       zend_error(E_WARNING, "Division by zero");
+               }
+               ZVAL_DOUBLE(result, Z_DVAL_P(op1) / Z_DVAL_P(op2));
+               return SUCCESS;
+       } else if (EXPECTED(type_pair == TYPE_PAIR(IS_DOUBLE, IS_LONG))) {
+               if (Z_LVAL_P(op2) == 0) {
+                       zend_error(E_WARNING, "Division by zero");
+               }
+               ZVAL_DOUBLE(result, Z_DVAL_P(op1) / (double)Z_LVAL_P(op2));
+               return SUCCESS;
+       } else if (EXPECTED(type_pair == TYPE_PAIR(IS_LONG, IS_DOUBLE))) {
+               if (Z_DVAL_P(op2) == 0) {
+                       zend_error(E_WARNING, "Division by zero");
+               }
+               ZVAL_DOUBLE(result, (double)Z_LVAL_P(op1) / Z_DVAL_P(op2));
+               return SUCCESS;
+       } else {
+               return FAILURE;
+       }
+}
 
-                               if (EXPECTED(op1 != op2)) {
-                                       op1 = zendi_convert_scalar_to_number_noisy(op1, &op1_copy, result);
-                                       op2 = zendi_convert_scalar_to_number_noisy(op2, &op2_copy, result);
-                               } else {
-                                       op1 = zendi_convert_scalar_to_number_noisy(op1, &op1_copy, result);
-                                       op2 = op1;
-                               }
-                               if (EG(exception)) {
-                                       if (result != op1) {
-                                               ZVAL_UNDEF(result);
-                                       }
-                                       return FAILURE;
-                               }
-                               converted = 1;
-                       } else {
-                               if (result != op1) {
-                                       ZVAL_UNDEF(result);
-                               }
-                               zend_throw_error(NULL, "Unsupported operand types");
-                               return FAILURE; /* unknown datatype */
-                       }
+ZEND_API int ZEND_FASTCALL div_function(zval *result, zval *op1, zval *op2) /* {{{ */
+{
+       ZVAL_DEREF(op1);
+       ZVAL_DEREF(op2);
+       if (div_function_base(result, op1, op2) == SUCCESS) {
+               return SUCCESS;
+       }
+
+       ZEND_TRY_BINARY_OBJECT_OPERATION(ZEND_DIV, div_function);
+
+       zval op1_copy, op2_copy;
+       if (UNEXPECTED(zendi_try_convert_scalar_to_number(op1, &op1_copy) == FAILURE)
+                       || UNEXPECTED(zendi_try_convert_scalar_to_number(op2, &op2_copy) == FAILURE)) {
+               if (!EG(exception)) {
+                       zend_throw_error(NULL, "Unsupported operand types");
                }
+               if (result != op1) {
+                       ZVAL_UNDEF(result);
+               }
+               return FAILURE;
+       }
+
+       if (result == op1) {
+               zval_ptr_dtor(result);
        }
+
+       if (div_function_base(result, &op1_copy, &op2_copy) == SUCCESS) {
+               return SUCCESS;
+       }
+
+       ZEND_ASSERT(0 && "Operation must succeed");
 }
 /* }}} */