summaryrefslogtreecommitdiff
path: root/Zend/zend_operators.c
diff options
context:
space:
mode:
authorDmitry Stogov <dmitry@zend.com>2016-12-02 13:58:44 +0300
committerDmitry Stogov <dmitry@zend.com>2016-12-02 13:58:44 +0300
commitdbf39cddd93ad06ce0353695bd02d448a5d6dac9 (patch)
tree319c0fe27f185055fa1e6a740aa074a7259a7b6a /Zend/zend_operators.c
parent67d90a4870499f53faefc3314f50181d4150d7d0 (diff)
downloadphp-git-dbf39cddd93ad06ce0353695bd02d448a5d6dac9.tar.gz
Fixed behavior of failing compound assignments (they shouldn't change the source value).
Diffstat (limited to 'Zend/zend_operators.c')
-rw-r--r--Zend/zend_operators.c60
1 files changed, 40 insertions, 20 deletions
diff --git a/Zend/zend_operators.c b/Zend/zend_operators.c
index 2d7fd96279..0d3c8e46e7 100644
--- a/Zend/zend_operators.c
+++ b/Zend/zend_operators.c
@@ -1187,10 +1187,6 @@ ZEND_API int ZEND_FASTCALL mod_function(zval *result, zval *op1, zval *op2) /* {
convert_op1_op2_long(op1, op1_lval, op2, op2_lval, ZEND_MOD, mod_function);
- if (op1 == result) {
- zval_dtor(result);
- }
-
if (op2_lval == 0) {
/* modulus by zero */
if (EG(current_execute_data) && !CG(in_compilation)) {
@@ -1198,10 +1194,16 @@ ZEND_API int ZEND_FASTCALL mod_function(zval *result, zval *op1, zval *op2) /* {
} else {
zend_error_noreturn(E_ERROR, "Modulo by zero");
}
- ZVAL_UNDEF(result);
+ if (op1 != result) {
+ ZVAL_UNDEF(result);
+ }
return FAILURE;
}
+ if (op1 == result) {
+ zval_dtor(result);
+ }
+
if (op2_lval == -1) {
/* Prevent overflow error/crash if op1==LONG_MIN */
ZVAL_LONG(result, 0);
@@ -1545,13 +1547,12 @@ ZEND_API int ZEND_FASTCALL shift_left_function(zval *result, zval *op1, zval *op
convert_op1_op2_long(op1, op1_lval, op2, op2_lval, ZEND_SL, shift_left_function);
- if (op1 == result) {
- zval_dtor(result);
- }
-
/* prevent wrapping quirkiness on some processors where << 64 + x == << x */
if (UNEXPECTED((zend_ulong)op2_lval >= SIZEOF_ZEND_LONG * 8)) {
if (EXPECTED(op2_lval > 0)) {
+ if (op1 == result) {
+ zval_dtor(result);
+ }
ZVAL_LONG(result, 0);
return SUCCESS;
} else {
@@ -1560,11 +1561,17 @@ ZEND_API int ZEND_FASTCALL shift_left_function(zval *result, zval *op1, zval *op
} else {
zend_error_noreturn(E_ERROR, "Bit shift by negative number");
}
- ZVAL_UNDEF(result);
+ if (op1 != result) {
+ ZVAL_UNDEF(result);
+ }
return FAILURE;
}
}
+ if (op1 == result) {
+ zval_dtor(result);
+ }
+
ZVAL_LONG(result, op1_lval << op2_lval);
return SUCCESS;
}
@@ -1576,14 +1583,13 @@ ZEND_API int ZEND_FASTCALL shift_right_function(zval *result, zval *op1, zval *o
convert_op1_op2_long(op1, op1_lval, op2, op2_lval, ZEND_SR, shift_right_function);
- if (op1 == result) {
- zval_dtor(result);
- }
-
/* prevent wrapping quirkiness on some processors where >> 64 + x == >> x */
if (UNEXPECTED((zend_ulong)op2_lval >= SIZEOF_ZEND_LONG * 8)) {
if (EXPECTED(op2_lval > 0)) {
ZVAL_LONG(result, (op1_lval < 0) ? -1 : 0);
+ if (op1 == result) {
+ zval_dtor(result);
+ }
return SUCCESS;
} else {
if (EG(current_execute_data) && !CG(in_compilation)) {
@@ -1591,11 +1597,17 @@ ZEND_API int ZEND_FASTCALL shift_right_function(zval *result, zval *op1, zval *o
} else {
zend_error_noreturn(E_ERROR, "Bit shift by negative number");
}
- ZVAL_UNDEF(result);
+ if (op1 != result) {
+ ZVAL_UNDEF(result);
+ }
return FAILURE;
}
}
+ if (op1 == result) {
+ zval_dtor(result);
+ }
+
ZVAL_LONG(result, op1_lval >> op2_lval);
return SUCCESS;
}
@@ -1603,6 +1615,7 @@ ZEND_API int ZEND_FASTCALL shift_right_function(zval *result, zval *op1, zval *o
ZEND_API int ZEND_FASTCALL concat_function(zval *result, zval *op1, zval *op2) /* {{{ */
{
+ zval *orig_op1 = op1;
zval op1_copy, op2_copy;
int use_copy1 = 0, use_copy2 = 0;
@@ -1615,11 +1628,7 @@ ZEND_API int ZEND_FASTCALL concat_function(zval *result, zval *op1, zval *op2) /
ZEND_TRY_BINARY_OBJECT_OPERATION(ZEND_CONCAT, concat_function);
use_copy1 = zend_make_printable_zval(op1, &op1_copy);
if (use_copy1) {
- /* We have created a converted copy of op1. Therefore, op1 won't become the result so
- * we have to free it.
- */
if (result == op1) {
- zval_dtor(op1);
if (UNEXPECTED(op1 == op2)) {
op2 = &op1_copy;
}
@@ -1650,7 +1659,15 @@ ZEND_API int ZEND_FASTCALL concat_function(zval *result, zval *op1, zval *op2) /
if (UNEXPECTED(op1_len > SIZE_MAX - op2_len)) {
zend_throw_error(NULL, "String size overflow");
- ZVAL_FALSE(result);
+ if (UNEXPECTED(use_copy1)) {
+ zval_dtor(op1);
+ }
+ if (UNEXPECTED(use_copy2)) {
+ zval_dtor(op2);
+ }
+ if (orig_op1 != result) {
+ ZVAL_UNDEF(result);
+ }
return FAILURE;
}
@@ -1660,6 +1677,9 @@ ZEND_API int ZEND_FASTCALL concat_function(zval *result, zval *op1, zval *op2) /
} else {
result_str = zend_string_alloc(result_len, 0);
memcpy(ZSTR_VAL(result_str), Z_STRVAL_P(op1), op1_len);
+ if (result == orig_op1) {
+ zval_dtor(orig_op1);
+ }
}
/* This has to happen first to account for the cases where result == op1 == op2 and