Files
react/compiler/packages/babel-plugin-react-compiler/docs/passes/39-validateContextVariableLValues.md
Joseph Savona 870cccd656 [compiler] Summaries of the compiler passes to assist agents in development (#35595)
Autogenerated summaries of each of the compiler passes which allow
agents to get the key ideas of a compiler pass, including key
input/output invariants, without having to reprocess the file each time.
In the subsequent diff this seemed to help.

---
[//]: # (BEGIN SAPLING FOOTER)
Stack created with [Sapling](https://sapling-scm.com). Best reviewed
with [ReviewStack](https://reviewstack.dev/facebook/react/pull/35595).
* #35607
* #35298
* #35596
* #35573
* __->__ #35595
* #35539
2026-01-23 11:26:47 -08:00

6.5 KiB

validateContextVariableLValues

File

src/Validation/ValidateContextVariableLValues.ts

Purpose

This validation pass ensures that all load/store references to a given named identifier are consistent with the "kind" of that variable (normal local variable or context variable). Context variables are variables that are captured by closures and require special handling for correct closure semantics.

The pass prevents mixing context variable operations (DeclareContext, StoreContext, LoadContext) with local variable operations (DeclareLocal, StoreLocal, LoadLocal, Destructure) on the same identifier.

Input Invariants

  • The function has been lowered to HIR
  • All instructions have been categorized by kind
  • Nested function expressions have been lowered

Validation Rules

Rule 1: Consistent Variable Kind

All references to the same identifier must use consistent load/store operations:

  • Context variables must only use DeclareContext, StoreContext, LoadContext
  • Local variables must only use DeclareLocal, StoreLocal, LoadLocal

Error (Invariant violation):

Expected all references to a variable to be consistently local or context references
Identifier [place] is referenced as a [kind] variable, but was previously referenced as a [prev.kind] variable

Rule 2: No Destructuring of Context Variables

Context variables cannot be destructured using the Destructure instruction.

Error (Todo):

Support destructuring of context variables

Rule 3: Unhandled Instruction Variants

If an instruction has lvalues that the pass does not handle, it throws a Todo error.

Error (Todo):

ValidateContextVariableLValues: unhandled instruction variant
Handle '[kind]' lvalues

Algorithm

Phase 1: Initialize Tracking

const identifierKinds: Map<IdentifierId, {place: Place, kind: 'local' | 'context' | 'destructure'}> = new Map();

Phase 2: Visit All Instructions

The pass iterates through all blocks and instructions, categorizing each based on its kind:

for (const [, block] of fn.body.blocks) {
  for (const instr of block.instructions) {
    switch (value.kind) {
      case 'DeclareContext':
      case 'StoreContext':
        visit(identifierKinds, value.lvalue.place, 'context');
        break;
      case 'LoadContext':
        visit(identifierKinds, value.place, 'context');
        break;
      case 'StoreLocal':
      case 'DeclareLocal':
        visit(identifierKinds, value.lvalue.place, 'local');
        break;
      case 'LoadLocal':
        visit(identifierKinds, value.place, 'local');
        break;
      case 'PostfixUpdate':
      case 'PrefixUpdate':
        visit(identifierKinds, value.lvalue, 'local');
        break;
      case 'Destructure':
        for (const lvalue of eachPatternOperand(value.lvalue.pattern)) {
          visit(identifierKinds, lvalue, 'destructure');
        }
        break;
      case 'ObjectMethod':
      case 'FunctionExpression':
        // Recursively validate nested functions
        validateContextVariableLValuesImpl(value.loweredFunc.func, identifierKinds);
        break;
    }
  }
}

Phase 3: Check Consistency

For each place visited, the visit function checks if the identifier was previously seen with a different kind:

function visit(identifiers, place, kind) {
  const prev = identifiers.get(place.identifier.id);
  if (prev !== undefined) {
    const wasContext = prev.kind === 'context';
    const isContext = kind === 'context';
    if (wasContext !== isContext) {
      // Check for destructuring of context variable
      if (prev.kind === 'destructure' || kind === 'destructure') {
        CompilerError.throwTodo({
          reason: `Support destructuring of context variables`,
          ...
        });
      }
      // Invariant violation: inconsistent variable kinds
      CompilerError.invariant(false, {
        reason: 'Expected all references to be consistently local or context references',
        ...
      });
    }
  }
  identifiers.set(place.identifier.id, {place, kind});
}

Edge Cases

Nested Function Expressions

The validation recursively processes nested function expressions and object methods, sharing the same identifierKinds map. This ensures that a variable captured by a nested function is consistently treated as a context variable throughout the entire function hierarchy.

Destructuring Patterns

Each operand in a destructure pattern is visited individually, marked as 'destructure' kind. If the same identifier was previously used as a context variable, a Todo error is thrown since destructuring of context variables is not yet supported.

Update Expressions

Both PostfixUpdate (e.g., x++) and PrefixUpdate (e.g., ++x) are treated as local variable operations.

TODOs

  1. Destructuring of context variables - Currently not supported:

    CompilerError.throwTodo({
      reason: `Support destructuring of context variables`,
      ...
    });
    
  2. Unhandled instruction variants - Some instruction types with lvalues may not be handled:

    CompilerError.throwTodo({
      reason: 'ValidateContextVariableLValues: unhandled instruction variant',
      description: `Handle '${value.kind}' lvalues`,
      ...
    });
    

Example

Fixture: error.todo-for-of-loop-with-context-variable-iterator.js

Input:

import {useHook} from 'shared-runtime';

function Component(props) {
  const data = useHook();
  const items = [];
  // NOTE: `item` is a context variable because it's reassigned and also referenced
  // within a closure, the `onClick` handler of each item
  for (let item of props.data) {
    item = item ?? {}; // reassignment to force a context variable
    items.push(
      <div key={item.id} onClick={() => data.set(item)}>
        {item.id}
      </div>
    );
  }
  return <div>{items}</div>;
}

Error:

Todo: Support non-trivial for..of inits

error.todo-for-of-loop-with-context-variable-iterator.ts:8:2
   6 |   // NOTE: `item` is a context variable because it's reassigned and also referenced
   7 |   // within a closure, the `onClick` handler of each item
>  8 |   for (let item of props.data) {
     |   ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
>  9 |     item = item ?? {}; // reassignment to force a context variable
     | ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
...
> 15 |   }
     | ^^^^ Support non-trivial for..of inits

Note: This particular error comes from an earlier pass (lowering), but demonstrates the kind of context variable scenarios that this validation is designed to catch.