Table of Contents

PHP RFC: Allow Object Property Writes on Objects Referenced by Constants

Introduction

This RFC allows writing to object properties (and `isset()` / `unset()` on properties) when the object instance is referenced by a constant.

PHP constants are immutable in the sense that the constant binding cannot be changed. However, objects in PHP are mutable by design: mutating an object property does not rebind the constant, it only updates the state of the referenced object.

Today, PHP rejects `CONST_OBJ->prop = ...` with a fatal error (“Cannot use temporary expression in write context”). This RFC proposes permitting property write operations in this specific scenario, aligning behavior with user expectations while keeping the constant binding immutable.

<?php
 
const OBJ = new stdClass();
 
OBJ->value = 123;
var_dump(OBJ->value); // int(123)
 
unset(OBJ->value);
var_dump(isset(OBJ->value)); // bool(false)
 
?>

Proposal

This RFC proposes to allow property write operations where the base expression is a constant that evaluates to an object.

The change is intentionally narrow:

increment/decrement, string concatenation) on objects referenced by constants.

path when the function is declared after the call site).

Specifically, the following operations become valid:

<?php
const OBJ = new stdClass();
OBJ->prop = "value";
?>
<?php
const OBJ = new stdClass();
OBJ->counter = 0;
OBJ->counter++;
OBJ->counter--;
OBJ->counter += 10;
OBJ->str = 'hello';
OBJ->str .= ' world';
?>
<?php
const OBJ = new stdClass();
OBJ->prop = 1;
var_dump(isset(OBJ->prop)); // true
unset(OBJ->prop);
var_dump(isset(OBJ->prop)); // false
?>
<?php
const OBJ = new stdClass();
OBJ->a = new stdClass();
OBJ->a->b = 7;
var_dump(OBJ->a->b); // int(7)
?>
<?php
const OBJ = new stdClass();
OBJ->val = 10;
 
function modify(&$v) { $v = 42; }
modify(OBJ->val);
var_dump(OBJ->val); // int(42)
?>

Semantics

Note on Array Constants

A related but distinct scenario exists with array constants:

<?php
const ARR = [1, 2, 3];
$ref = ARR;
$ref[0] = 9;
var_dump(ARR); // [1, 2, 3] — original unchanged
?>

Because arrays use copy-on-write semantics, modifying the copy has no effect on the original constant. Similarly:

<?php
const C = [1, 2, 3];
function foo(): array { return C; }
foo()[] = 4;
var_dump(C); // [1, 2, 3] — unmodified, assignment is on a temporary
?>

These dim-write cases involve assignment to temporaries that silently have no effect. Whether such no-effect writes should produce a warning or error is a valid concern but is out of scope for this RFC and may be addressed separately (e.g. warning/throwing on `ASSIGN_DIM` when OP1 is not a pointer — i.e. not an indirect, reference, or object — as the operation can never have an effect). See the Future Scope section.

Non-goals / Explicitly Unchanged

This RFC does NOT change:

<?php
const OBJ = new stdClass();
OBJ = new stdClass(); // still illegal (parse error)
?>
<?php
const ARR = [1,2,3];
ARR[0] = 9; // unchanged (still rejected as before)
 
const OBJ = new stdClass();
OBJ['x'] = 1; // unchanged (still rejected as before)
?>

Implementation Details

The compiler change is in `zend_delayed_compile_prop()` in `Zend/zend_compile.c`. When the object AST node is a `ZEND_AST_CONST` and the access type is `BP_VAR_W`, `BP_VAR_RW`, `BP_VAR_UNSET`, or `BP_VAR_FUNC_ARG`, the compiler now emits a `ZEND_FETCH_CONSTANT` opcode that produces a `VAR` result (via `zend_emit_op`) instead of a `TMP` result (via `zend_emit_op_tmp`). This allows the subsequent property write opcodes to operate on the fetched object.

The `zend_compile_const()` function remains unchanged — constants used in read contexts (`BP_VAR_R`, `BP_VAR_IS`) still produce `TMP` results as before.

Why This Brings Value

This feature addresses a common surprise: PHP already permits objects in constants (since PHP 8.1), and developers naturally expect to mutate the object's state through the constant reference. The current fatal error is not about constant immutability per se, but rather about an overly broad “temporary write context” restriction in the compiler. Narrowly enabling property writes reduces friction, improves ergonomics, and matches the runtime model of objects without increasing language complexity for other write contexts.

Backward Incompatible Changes

None.

Code that previously terminated with a fatal error for `CONST_OBJ->prop = ...` will now execute successfully. This is a compatibility improvement — no previously-working code changes behavior.

Proposed PHP Version(s)

PHP 8.6

RFC Impact

To the Ecosystem

To Existing Extensions

No expected impact. This is a core compilation change around property write access, not an extension API change.

To SAPIs

No expected impact.

Open Issues

None currently.

Future Scope

Voting Choices

Primary vote requiring a 2/3 majority to accept the RFC:

Allow object property writes on objects referenced by constants as described in this RFC?
Real name Yes No
Final result: 0 0
This poll has been closed.

Patches and Tests

Proof of concept implementation:

Tests included in the PR:

Implementation

(To be filled after merge)

  1. Merged into: PHP 8.6
  2. Git commits: ...
  3. Manual entry: ...

References

Rejected Features

Changelog

Key changes from v0.1: - Added by-ref passing examples (confirmed working in tests) - Added implementation details section explaining the compiler change - Added “Note on Array Constants” section addressing Ilija's and Tim's feedback about silent no-op dim writes - Expanded Future Scope with Tim's suggestion about warning on ASSIGN_DIM with non-pointer OP1 - Removed “Abstain” from voting (PHP RFCs typically use Yes/No) - Updated vote close date to a future date - Targeted PHP 8.6 explicitly - Listed all test files in the Patches and Tests section - Removed Open Issues (the by-ref question is answered, isset/unset is confirmed) - Added full URLs to References - Added changelog entry for v0.2