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:

Specifically, the following operations become valid:

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:

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 Abstain
Final result: 0 0 0
This poll has been closed.

Patches and Tests

Proof of concept implementation:

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