diff --git a/compiler/packages/babel-plugin-react-compiler/src/HIR/Environment.ts b/compiler/packages/babel-plugin-react-compiler/src/HIR/Environment.ts
index 1d2e155848802..855bc039abf37 100644
--- a/compiler/packages/babel-plugin-react-compiler/src/HIR/Environment.ts
+++ b/compiler/packages/babel-plugin-react-compiler/src/HIR/Environment.ts
@@ -231,6 +231,8 @@ const EnvironmentConfigSchema = z.object({
*/
enableUseTypeAnnotations: z.boolean().default(false),
+ enableFunctionDependencyRewrite: z.boolean().default(true),
+
/**
* Enables inlining ReactElement object literals in place of JSX
* An alternative to the standard JSX transform which replaces JSX with React's jsxProd() runtime
diff --git a/compiler/packages/babel-plugin-react-compiler/src/HIR/PropagateScopeDependenciesHIR.ts b/compiler/packages/babel-plugin-react-compiler/src/HIR/PropagateScopeDependenciesHIR.ts
index 7fd44c29dccf6..8aed17f8ee847 100644
--- a/compiler/packages/babel-plugin-react-compiler/src/HIR/PropagateScopeDependenciesHIR.ts
+++ b/compiler/packages/babel-plugin-react-compiler/src/HIR/PropagateScopeDependenciesHIR.ts
@@ -663,35 +663,54 @@ function collectDependencies(
const scopeTraversal = new ScopeBlockTraversal();
- for (const [blockId, block] of fn.body.blocks) {
- scopeTraversal.recordScopes(block);
- const scopeBlockInfo = scopeTraversal.blockInfos.get(blockId);
- if (scopeBlockInfo?.kind === 'begin') {
- context.enterScope(scopeBlockInfo.scope);
- } else if (scopeBlockInfo?.kind === 'end') {
- context.exitScope(scopeBlockInfo.scope, scopeBlockInfo?.pruned);
- }
-
- // Record referenced optional chains in phis
- for (const phi of block.phis) {
- for (const operand of phi.operands) {
- const maybeOptionalChain = temporaries.get(operand[1].identifier.id);
- if (maybeOptionalChain) {
- context.visitDependency(maybeOptionalChain);
+ const handleFunction = (fn: HIRFunction): void => {
+ for (const [blockId, block] of fn.body.blocks) {
+ scopeTraversal.recordScopes(block);
+ const scopeBlockInfo = scopeTraversal.blockInfos.get(blockId);
+ if (scopeBlockInfo?.kind === 'begin') {
+ context.enterScope(scopeBlockInfo.scope);
+ } else if (scopeBlockInfo?.kind === 'end') {
+ context.exitScope(scopeBlockInfo.scope, scopeBlockInfo.pruned);
+ }
+ // Record referenced optional chains in phis
+ for (const phi of block.phis) {
+ for (const operand of phi.operands) {
+ const maybeOptionalChain = temporaries.get(operand[1].identifier.id);
+ if (maybeOptionalChain) {
+ context.visitDependency(maybeOptionalChain);
+ }
}
}
- }
- for (const instr of block.instructions) {
- if (!processedInstrsInOptional.has(instr)) {
- handleInstruction(instr, context);
+ for (const instr of block.instructions) {
+ if (
+ fn.env.config.enableFunctionDependencyRewrite &&
+ (instr.value.kind === 'FunctionExpression' ||
+ instr.value.kind === 'ObjectMethod')
+ ) {
+ context.declare(instr.lvalue.identifier, {
+ id: instr.id,
+ scope: context.currentScope,
+ });
+ /**
+ * Recursively visit the inner function to extract dependencies there
+ */
+ const wasInInnerFn = context.inInnerFn;
+ context.inInnerFn = true;
+ handleFunction(instr.value.loweredFunc.func);
+ context.inInnerFn = wasInInnerFn;
+ } else if (!processedInstrsInOptional.has(instr)) {
+ handleInstruction(instr, context);
+ }
}
- }
- if (!processedInstrsInOptional.has(block.terminal)) {
- for (const place of eachTerminalOperand(block.terminal)) {
- context.visitOperand(place);
+ if (!processedInstrsInOptional.has(block.terminal)) {
+ for (const place of eachTerminalOperand(block.terminal)) {
+ context.visitOperand(place);
+ }
}
}
- }
+ };
+
+ handleFunction(fn);
return context.deps;
}
diff --git a/compiler/packages/babel-plugin-react-compiler/src/__tests__/fixtures/compiler/capturing-func-mutate-2.expect.md b/compiler/packages/babel-plugin-react-compiler/src/__tests__/fixtures/compiler/capturing-func-mutate-2.expect.md
index b31a16da90e3f..c071d5d20ed99 100644
--- a/compiler/packages/babel-plugin-react-compiler/src/__tests__/fixtures/compiler/capturing-func-mutate-2.expect.md
+++ b/compiler/packages/babel-plugin-react-compiler/src/__tests__/fixtures/compiler/capturing-func-mutate-2.expect.md
@@ -26,29 +26,20 @@ export const FIXTURE_ENTRYPOINT = {
```javascript
import { c as _c } from "react/compiler-runtime";
function component(a, b) {
- const $ = _c(5);
- let t0;
- if ($[0] !== b) {
- t0 = { b };
- $[0] = b;
- $[1] = t0;
- } else {
- t0 = $[1];
- }
- const y = t0;
+ const $ = _c(2);
+ const y = { b };
let z;
- if ($[2] !== a || $[3] !== y) {
+ if ($[0] !== a) {
z = { a };
const x = function () {
z.a = 2;
};
x();
- $[2] = a;
- $[3] = y;
- $[4] = z;
+ $[0] = a;
+ $[1] = z;
} else {
- z = $[4];
+ z = $[1];
}
return z;
}
diff --git a/compiler/packages/babel-plugin-react-compiler/src/__tests__/fixtures/compiler/preserve-memo-validation/error.todo-useCallback-captures-reassigned-context-property.expect.md b/compiler/packages/babel-plugin-react-compiler/src/__tests__/fixtures/compiler/preserve-memo-validation/error.todo-useCallback-captures-reassigned-context-property.expect.md
new file mode 100644
index 0000000000000..ae44f27912293
--- /dev/null
+++ b/compiler/packages/babel-plugin-react-compiler/src/__tests__/fixtures/compiler/preserve-memo-validation/error.todo-useCallback-captures-reassigned-context-property.expect.md
@@ -0,0 +1,53 @@
+
+## Input
+
+```javascript
+// @validatePreserveExistingMemoizationGuarantees
+import {useCallback} from 'react';
+import {Stringify} from 'shared-runtime';
+
+/**
+ * TODO: we're currently bailing out because `contextVar` is a context variable
+ * and not recorded into the PropagateScopeDeps LoadLocal / PropertyLoad
+ * sidemap. Previously, we were able to avoid this as `BuildHIR` hoisted
+ * `LoadContext` and `PropertyLoad` instructions into the outer function, which
+ * we took as eligible dependencies.
+ *
+ * One solution is to simply record `LoadContext` identifiers into the
+ * temporaries sidemap when the instruction occurs *after* the context
+ * variable's mutable range.
+ */
+function Foo(props) {
+ let contextVar;
+ if (props.cond) {
+ contextVar = {val: 2};
+ } else {
+ contextVar = {};
+ }
+
+ const cb = useCallback(() => [contextVar.val], [contextVar.val]);
+
+ return ;
+}
+
+export const FIXTURE_ENTRYPOINT = {
+ fn: Foo,
+ params: [{cond: true}],
+};
+
+```
+
+
+## Error
+
+```
+ 22 | }
+ 23 |
+> 24 | const cb = useCallback(() => [contextVar.val], [contextVar.val]);
+ | ^^^^^^^^^^^^^^^^^^^^^^ CannotPreserveMemoization: React Compiler has skipped optimizing this component because the existing manual memoization could not be preserved. The inferred dependencies did not match the manually specified dependencies, which could cause the value to change more or less frequently than expected (24:24)
+ 25 |
+ 26 | return ;
+ 27 | }
+```
+
+
\ No newline at end of file
diff --git a/compiler/packages/babel-plugin-react-compiler/src/__tests__/fixtures/compiler/preserve-memo-validation/error.todo-useCallback-captures-reassigned-context-property.tsx b/compiler/packages/babel-plugin-react-compiler/src/__tests__/fixtures/compiler/preserve-memo-validation/error.todo-useCallback-captures-reassigned-context-property.tsx
new file mode 100644
index 0000000000000..8447e3960dc51
--- /dev/null
+++ b/compiler/packages/babel-plugin-react-compiler/src/__tests__/fixtures/compiler/preserve-memo-validation/error.todo-useCallback-captures-reassigned-context-property.tsx
@@ -0,0 +1,32 @@
+// @validatePreserveExistingMemoizationGuarantees
+import {useCallback} from 'react';
+import {Stringify} from 'shared-runtime';
+
+/**
+ * TODO: we're currently bailing out because `contextVar` is a context variable
+ * and not recorded into the PropagateScopeDeps LoadLocal / PropertyLoad
+ * sidemap. Previously, we were able to avoid this as `BuildHIR` hoisted
+ * `LoadContext` and `PropertyLoad` instructions into the outer function, which
+ * we took as eligible dependencies.
+ *
+ * One solution is to simply record `LoadContext` identifiers into the
+ * temporaries sidemap when the instruction occurs *after* the context
+ * variable's mutable range.
+ */
+function Foo(props) {
+ let contextVar;
+ if (props.cond) {
+ contextVar = {val: 2};
+ } else {
+ contextVar = {};
+ }
+
+ const cb = useCallback(() => [contextVar.val], [contextVar.val]);
+
+ return ;
+}
+
+export const FIXTURE_ENTRYPOINT = {
+ fn: Foo,
+ params: [{cond: true}],
+};
diff --git a/compiler/packages/babel-plugin-react-compiler/src/__tests__/fixtures/compiler/preserve-memo-validation/error.useCallback-infer-less-specific-conditional-access.expect.md b/compiler/packages/babel-plugin-react-compiler/src/__tests__/fixtures/compiler/preserve-memo-validation/error.useCallback-infer-less-specific-conditional-access.expect.md
index 955d391f912cf..940b3975c160d 100644
--- a/compiler/packages/babel-plugin-react-compiler/src/__tests__/fixtures/compiler/preserve-memo-validation/error.useCallback-infer-less-specific-conditional-access.expect.md
+++ b/compiler/packages/babel-plugin-react-compiler/src/__tests__/fixtures/compiler/preserve-memo-validation/error.useCallback-infer-less-specific-conditional-access.expect.md
@@ -44,8 +44,6 @@ function Component({propA, propB}) {
| ^^^^^^^^^^^^^^^^^
> 14 | }, [propA?.a, propB.x.y]);
| ^^^^ CannotPreserveMemoization: React Compiler has skipped optimizing this component because the existing manual memoization could not be preserved. The inferred dependencies did not match the manually specified dependencies, which could cause the value to change more or less frequently than expected (6:14)
-
-CannotPreserveMemoization: React Compiler has skipped optimizing this component because the existing manual memoization could not be preserved. The inferred dependencies did not match the manually specified dependencies, which could cause the value to change more or less frequently than expected (6:14)
15 | }
16 |
```
diff --git a/compiler/packages/babel-plugin-react-compiler/src/__tests__/fixtures/compiler/preserve-memo-validation/useCallback-captures-reassigned-context-property.expect.md b/compiler/packages/babel-plugin-react-compiler/src/__tests__/fixtures/compiler/preserve-memo-validation/useCallback-captures-reassigned-context-property.expect.md
deleted file mode 100644
index db69bc2821b12..0000000000000
--- a/compiler/packages/babel-plugin-react-compiler/src/__tests__/fixtures/compiler/preserve-memo-validation/useCallback-captures-reassigned-context-property.expect.md
+++ /dev/null
@@ -1,81 +0,0 @@
-
-## Input
-
-```javascript
-// @validatePreserveExistingMemoizationGuarantees
-import {useCallback} from 'react';
-import {Stringify} from 'shared-runtime';
-
-function Foo(props) {
- let contextVar;
- if (props.cond) {
- contextVar = {val: 2};
- } else {
- contextVar = {};
- }
-
- const cb = useCallback(() => [contextVar.val], [contextVar.val]);
-
- return ;
-}
-
-export const FIXTURE_ENTRYPOINT = {
- fn: Foo,
- params: [{cond: true}],
-};
-
-```
-
-## Code
-
-```javascript
-import { c as _c } from "react/compiler-runtime"; // @validatePreserveExistingMemoizationGuarantees
-import { useCallback } from "react";
-import { Stringify } from "shared-runtime";
-
-function Foo(props) {
- const $ = _c(6);
- let contextVar;
- if ($[0] !== props.cond) {
- if (props.cond) {
- contextVar = { val: 2 };
- } else {
- contextVar = {};
- }
- $[0] = props.cond;
- $[1] = contextVar;
- } else {
- contextVar = $[1];
- }
-
- const t0 = contextVar;
- let t1;
- if ($[2] !== t0.val) {
- t1 = () => [contextVar.val];
- $[2] = t0.val;
- $[3] = t1;
- } else {
- t1 = $[3];
- }
- contextVar;
- const cb = t1;
- let t2;
- if ($[4] !== cb) {
- t2 = ;
- $[4] = cb;
- $[5] = t2;
- } else {
- t2 = $[5];
- }
- return t2;
-}
-
-export const FIXTURE_ENTRYPOINT = {
- fn: Foo,
- params: [{ cond: true }],
-};
-
-```
-
-### Eval output
-(kind: ok)
{"cb":{"kind":"Function","result":[2]},"shouldInvokeFns":true}
\ No newline at end of file
diff --git a/compiler/packages/babel-plugin-react-compiler/src/__tests__/fixtures/compiler/preserve-memo-validation/useCallback-captures-reassigned-context-property.tsx b/compiler/packages/babel-plugin-react-compiler/src/__tests__/fixtures/compiler/preserve-memo-validation/useCallback-captures-reassigned-context-property.tsx
deleted file mode 100644
index cb6f65a9f4f52..0000000000000
--- a/compiler/packages/babel-plugin-react-compiler/src/__tests__/fixtures/compiler/preserve-memo-validation/useCallback-captures-reassigned-context-property.tsx
+++ /dev/null
@@ -1,21 +0,0 @@
-// @validatePreserveExistingMemoizationGuarantees
-import {useCallback} from 'react';
-import {Stringify} from 'shared-runtime';
-
-function Foo(props) {
- let contextVar;
- if (props.cond) {
- contextVar = {val: 2};
- } else {
- contextVar = {};
- }
-
- const cb = useCallback(() => [contextVar.val], [contextVar.val]);
-
- return ;
-}
-
-export const FIXTURE_ENTRYPOINT = {
- fn: Foo,
- params: [{cond: true}],
-};
diff --git a/compiler/packages/babel-plugin-react-compiler/src/__tests__/fixtures/compiler/preserve-memo-validation/useCallback-captures-reassigned-context.expect.md b/compiler/packages/babel-plugin-react-compiler/src/__tests__/fixtures/compiler/preserve-memo-validation/useCallback-captures-reassigned-context.expect.md
index b66661fbcaa08..41994e1e56e44 100644
--- a/compiler/packages/babel-plugin-react-compiler/src/__tests__/fixtures/compiler/preserve-memo-validation/useCallback-captures-reassigned-context.expect.md
+++ b/compiler/packages/babel-plugin-react-compiler/src/__tests__/fixtures/compiler/preserve-memo-validation/useCallback-captures-reassigned-context.expect.md
@@ -45,18 +45,16 @@ function Foo(props) {
} else {
x = $[1];
}
-
- const t0 = x;
- let t1;
- if ($[2] !== t0) {
- t1 = () => [x];
- $[2] = t0;
- $[3] = t1;
+ let t0;
+ if ($[2] !== x) {
+ t0 = () => [x];
+ $[2] = x;
+ $[3] = t0;
} else {
- t1 = $[3];
+ t0 = $[3];
}
x;
- const cb = t1;
+ const cb = t0;
return cb;
}
diff --git a/compiler/packages/babel-plugin-react-compiler/src/__tests__/fixtures/compiler/preserve-memo-validation/useCallback-extended-contextvar-scope.expect.md b/compiler/packages/babel-plugin-react-compiler/src/__tests__/fixtures/compiler/preserve-memo-validation/useCallback-extended-contextvar-scope.expect.md
index b141c27614d24..9ce4a62e710d4 100644
--- a/compiler/packages/babel-plugin-react-compiler/src/__tests__/fixtures/compiler/preserve-memo-validation/useCallback-extended-contextvar-scope.expect.md
+++ b/compiler/packages/babel-plugin-react-compiler/src/__tests__/fixtures/compiler/preserve-memo-validation/useCallback-extended-contextvar-scope.expect.md
@@ -70,28 +70,26 @@ function useBar(t0, cond) {
if (cond) {
x = b;
}
-
- const t2 = x;
- let t3;
- if ($[1] !== a || $[2] !== t2) {
- t3 = () => [a, x];
+ let t2;
+ if ($[1] !== a || $[2] !== x) {
+ t2 = () => [a, x];
$[1] = a;
- $[2] = t2;
- $[3] = t3;
+ $[2] = x;
+ $[3] = t2;
} else {
- t3 = $[3];
+ t2 = $[3];
}
x;
- const cb = t3;
- let t4;
+ const cb = t2;
+ let t3;
if ($[4] !== cb) {
- t4 = ;
+ t3 = ;
$[4] = cb;
- $[5] = t4;
+ $[5] = t3;
} else {
- t4 = $[5];
+ t3 = $[5];
}
- return t4;
+ return t3;
}
export const FIXTURE_ENTRYPOINT = {
diff --git a/compiler/packages/babel-plugin-react-compiler/src/__tests__/fixtures/compiler/propagate-scope-deps-hir-fork/reduce-reactive-deps/todo-infer-function-uncond-optionals-hoisted.expect.md b/compiler/packages/babel-plugin-react-compiler/src/__tests__/fixtures/compiler/propagate-scope-deps-hir-fork/reduce-reactive-deps/todo-infer-function-uncond-optionals-hoisted.expect.md
index 02e60eff91d61..ed56ff068113b 100644
--- a/compiler/packages/babel-plugin-react-compiler/src/__tests__/fixtures/compiler/propagate-scope-deps-hir-fork/reduce-reactive-deps/todo-infer-function-uncond-optionals-hoisted.expect.md
+++ b/compiler/packages/babel-plugin-react-compiler/src/__tests__/fixtures/compiler/propagate-scope-deps-hir-fork/reduce-reactive-deps/todo-infer-function-uncond-optionals-hoisted.expect.md
@@ -34,9 +34,9 @@ function useFoo(t0) {
const $ = _c(2);
const { a } = t0;
let t1;
- if ($[0] !== a.b) {
+ if ($[0] !== a.b?.c.d?.e) {
t1 = a.b?.c.d?.e} shouldInvokeFns={true} />;
- $[0] = a.b;
+ $[0] = a.b?.c.d?.e;
$[1] = t1;
} else {
t1 = $[1];
diff --git a/compiler/packages/babel-plugin-react-compiler/src/__tests__/fixtures/compiler/reduce-reactive-deps/todo-infer-function-uncond-optionals-hoisted.expect.md b/compiler/packages/babel-plugin-react-compiler/src/__tests__/fixtures/compiler/reduce-reactive-deps/todo-infer-function-uncond-optionals-hoisted.expect.md
index 157e2de81a1ff..bb99a5d90fe2b 100644
--- a/compiler/packages/babel-plugin-react-compiler/src/__tests__/fixtures/compiler/reduce-reactive-deps/todo-infer-function-uncond-optionals-hoisted.expect.md
+++ b/compiler/packages/babel-plugin-react-compiler/src/__tests__/fixtures/compiler/reduce-reactive-deps/todo-infer-function-uncond-optionals-hoisted.expect.md
@@ -31,9 +31,9 @@ function useFoo(t0) {
const $ = _c(2);
const { a } = t0;
let t1;
- if ($[0] !== a.b) {
+ if ($[0] !== a.b?.c.d?.e) {
t1 = a.b?.c.d?.e} shouldInvokeFns={true} />;
- $[0] = a.b;
+ $[0] = a.b?.c.d?.e;
$[1] = t1;
} else {
t1 = $[1];