Unverified Commit d750fbf6 authored by Paul Yang's avatar Paul Yang Committed by GitHub

Fix more issues for reference values (#5613)

* Fix more issues for reference values

* Revert change in gdb test

* Add more tests
parent 30851ca2
...@@ -161,6 +161,11 @@ bool native_slot_set(upb_fieldtype_t type, const zend_class_entry* klass, ...@@ -161,6 +161,11 @@ bool native_slot_set(upb_fieldtype_t type, const zend_class_entry* klass,
bool native_slot_set_by_array(upb_fieldtype_t type, bool native_slot_set_by_array(upb_fieldtype_t type,
const zend_class_entry* klass, void* memory, const zend_class_entry* klass, void* memory,
zval* value TSRMLS_DC) { zval* value TSRMLS_DC) {
#if PHP_MAJOR_VERSION >= 7
if (Z_ISREF_P(value)) {
ZVAL_DEREF(value);
}
#endif
switch (type) { switch (type) {
case UPB_TYPE_STRING: case UPB_TYPE_STRING:
case UPB_TYPE_BYTES: { case UPB_TYPE_BYTES: {
...@@ -186,12 +191,6 @@ bool native_slot_set_by_array(upb_fieldtype_t type, ...@@ -186,12 +191,6 @@ bool native_slot_set_by_array(upb_fieldtype_t type,
break; break;
} }
case UPB_TYPE_MESSAGE: { case UPB_TYPE_MESSAGE: {
#if PHP_MAJOR_VERSION >= 7
if (Z_ISREF_P(value)) {
ZVAL_DEREF(value);
}
#endif
if (Z_TYPE_P(value) != IS_OBJECT) { if (Z_TYPE_P(value) != IS_OBJECT) {
zend_error(E_USER_ERROR, "Given value is not message."); zend_error(E_USER_ERROR, "Given value is not message.");
return false; return false;
...@@ -219,6 +218,11 @@ bool native_slot_set_by_array(upb_fieldtype_t type, ...@@ -219,6 +218,11 @@ bool native_slot_set_by_array(upb_fieldtype_t type,
bool native_slot_set_by_map(upb_fieldtype_t type, const zend_class_entry* klass, bool native_slot_set_by_map(upb_fieldtype_t type, const zend_class_entry* klass,
void* memory, zval* value TSRMLS_DC) { void* memory, zval* value TSRMLS_DC) {
#if PHP_MAJOR_VERSION >= 7
if (Z_ISREF_P(value)) {
ZVAL_DEREF(value);
}
#endif
switch (type) { switch (type) {
case UPB_TYPE_STRING: case UPB_TYPE_STRING:
case UPB_TYPE_BYTES: { case UPB_TYPE_BYTES: {
......
...@@ -373,6 +373,11 @@ bool protobuf_convert_to_bool(zval* from, int8_t* to) { ...@@ -373,6 +373,11 @@ bool protobuf_convert_to_bool(zval* from, int8_t* to) {
} }
bool protobuf_convert_to_string(zval* from) { bool protobuf_convert_to_string(zval* from) {
#if PHP_MAJOR_VERSION >= 7
if (Z_ISREF_P(from)) {
ZVAL_DEREF(from);
}
#endif
TSRMLS_FETCH(); TSRMLS_FETCH();
switch (Z_TYPE_P(from)) { switch (Z_TYPE_P(from)) {
case IS_STRING: { case IS_STRING: {
......
...@@ -533,31 +533,39 @@ class RepeatedFieldTest extends \PHPUnit\Framework\TestCase ...@@ -533,31 +533,39 @@ class RepeatedFieldTest extends \PHPUnit\Framework\TestCase
# Test reference in array # Test reference in array
######################################################### #########################################################
public function testArrayElementIsReference() public function testArrayElementIsReferenceInSetters()
{ {
// Bool elements
$values = [true];
array_walk($values, function (&$value) {});
$m = new TestMessage(); $m = new TestMessage();
$subs = [1, 2]; $m->setRepeatedBool($values);
foreach ($subs as &$sub) {
$sub = new Sub(['a' => $sub]);
}
$m->setRepeatedMessage($subs); // Int32 elements
} $values = [1];
array_walk($values, function (&$value) {});
$m = new TestMessage();
$m->setRepeatedInt32($values);
public function testArrayIsReference() // Double elements
{ $values = [1.0];
$keys = [['repeated_message' => [['a' => 1]]]]; array_walk($values, function (&$value) {});
$m = new TestMessage();
$m->setRepeatedDouble($values);
foreach ($keys as &$key) { // String elements
foreach ($key['repeated_message'] as &$element) { $values = ['a'];
$element = new Sub($element); array_walk($values, function (&$value) {});
} $m = new TestMessage();
$key = new TestMessage($key); $m->setRepeatedString($values);
}
// Message elements
$m = new TestMessage(); $m = new TestMessage();
$m->setRepeatedDeep($keys); $subs = [1, 2];
foreach ($subs as &$sub) {
$sub = new Sub(['a' => $sub]);
}
$m->setRepeatedMessage($subs);
} }
######################################################### #########################################################
......
...@@ -1375,6 +1375,78 @@ class GeneratedClassTest extends TestBase ...@@ -1375,6 +1375,78 @@ class GeneratedClassTest extends TestBase
$this->assertTrue(true); $this->assertTrue(true);
} }
public function testReferenceInArrayConstructor()
{
$keys = [[
'optional_bool' => true,
'repeated_bool' => [true],
'map_bool_bool' => [true => true],
'optional_double' => 1.0,
'repeated_double' => [1.0],
'map_int32_double' => [1 => 1.0],
'optional_int32' => 1,
'repeated_int32' => [1],
'map_int32_int32' => [1 => 1],
'optional_string' => 'a',
'repeated_string' => ['a'],
'map_string_string' => ['a' => 'a'],
'optional_message' => ['a' => 1],
'repeated_message' => [['a' => 1]],
'map_int32_message' => [1 => ['a' => 1]],
]];
foreach ($keys as &$key) {
foreach ($key as $id => &$value) {
if ($id === 'repeated_bool') {
foreach ($value as &$element) {
}
}
if ($id === 'map_bool_bool') {
foreach ($value as $mapKey => &$element) {
}
}
if ($id === 'repeated_double') {
foreach ($value as &$element) {
}
}
if ($id === 'map_int32_double') {
foreach ($value as $mapKey => &$element) {
}
}
if ($id === 'repeated_int32') {
foreach ($value as &$element) {
}
}
if ($id === 'map_int32_int32') {
foreach ($value as $mapKey => &$element) {
}
}
if ($id === 'repeated_string') {
foreach ($value as &$element) {
}
}
if ($id === 'map_string_string') {
foreach ($value as $mapKey => &$element) {
}
}
if ($id === 'optional_message') {
$value = new Sub($value);
}
if ($id === 'repeated_message') {
foreach ($value as &$element) {
$element = new Sub($element);
}
}
if ($id === 'map_int32_message') {
foreach ($value as $mapKey => &$element) {
$element = new Sub($element);
}
}
}
$key = new TestMessage($key);
}
}
######################################################### #########################################################
# Test message equals. # Test message equals.
######################################################### #########################################################
...@@ -1387,4 +1459,35 @@ class GeneratedClassTest extends TestBase ...@@ -1387,4 +1459,35 @@ class GeneratedClassTest extends TestBase
TestUtil::setTestMessage($n); TestUtil::setTestMessage($n);
$this->assertEquals($m, $n); $this->assertEquals($m, $n);
} }
#########################################################
# Test reference of value
#########################################################
public function testValueIsReference()
{
// Bool element
$values = [true];
array_walk($values, function (&$value) {});
$m = new TestMessage();
$m->setOptionalBool($values[0]);
// Int32 element
$values = [1];
array_walk($values, function (&$value) {});
$m = new TestMessage();
$m->setOptionalInt32($values[0]);
// Double element
$values = [1.0];
array_walk($values, function (&$value) {});
$m = new TestMessage();
$m->setOptionalDouble($values[0]);
// String element
$values = ['a'];
array_walk($values, function (&$value) {});
$m = new TestMessage();
$m->setOptionalString($values[0]);
}
} }
...@@ -442,6 +442,43 @@ class MapFieldTest extends \PHPUnit\Framework\TestCase { ...@@ -442,6 +442,43 @@ class MapFieldTest extends \PHPUnit\Framework\TestCase {
$this->assertSame(3, $i); $this->assertSame(3, $i);
} }
#########################################################
# Test reference in map
#########################################################
public function testMapElementIsReference()
{
// Bool elements
$values = [true => true];
array_walk($values, function (&$value) {});
$m = new TestMessage();
$m->setMapBoolBool($values);
// Int32 elements
$values = [1 => 1];
array_walk($values, function (&$value) {});
$m = new TestMessage();
$m->setMapInt32Int32($values);
// Double elements
$values = [1 => 1.0];
array_walk($values, function (&$value) {});
$m = new TestMessage();
$m->setMapInt32Double($values);
// String elements
$values = ['a' => 'a'];
array_walk($values, function (&$value) {});
$m = new TestMessage();
$m->setMapStringString($values);
// Message elements
$values = [1 => new Sub()];
array_walk($values, function (&$value) {});
$m = new TestMessage();
$m->setMapInt32Message($values);
}
######################################################### #########################################################
# Test memory leak # Test memory leak
######################################################### #########################################################
......
...@@ -31,7 +31,6 @@ message TestMessage { ...@@ -31,7 +31,6 @@ message TestMessage {
Sub optional_message = 17; Sub optional_message = 17;
bar.TestInclude optional_included_message = 18; bar.TestInclude optional_included_message = 18;
TestMessage recursive = 19; TestMessage recursive = 19;
repeated TestMessage repeated_deep = 20;
// Repeated // Repeated
repeated int32 repeated_int32 = 31; repeated int32 repeated_int32 = 31;
......
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment