-
Notifications
You must be signed in to change notification settings - Fork 48.1k
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
[Compiler Bug]: enableFunctionOutlining
breaks react-native-reanimated
API callbacks
#32580
Comments
Thanks for posting! I'm just curious, why can't you infer that |
It's because Reanimated's multithreading APIs aren't symmetrical and because it depends on the Metro bundler in React Native. Let's say we have two files: // file.js
import {runOnJS} from 'react-native-reanimated';
export function foo(callback) {
'worklet';
console.log('scheduling back to JS');
runOnJS(callback);
return null;
} // App.jsx
import {useDerivedValue} from 'react-native-reanimated';
import {myRunOnJS} from './file';
export default function App() {
const dv = useDerivedValue(() => foo(bar));
return null;
}
function bar() {
console.log('received on JS');
} In here To infer if The design decision about asymmetry, |
This comment has been minimized.
This comment has been minimized.
This comment has been minimized.
This comment has been minimized.
I initially thought this issue could be resolved using the Reanimated's babel plugin, but I it's a bit counterintuitive for the compiler to extract functions for Reanimated only to move them back inside the scope. A potential improvement would be to allow the Alternatively, |
This comment has been minimized.
This comment has been minimized.
This comment has been minimized.
This comment has been minimized.
1 similar comment
This comment has been minimized.
This comment has been minimized.
Reanimated Babel plugin runs only after the Compiler, since the Compiler works on the
The community wanted less boilerplate so we added recognizing worklets implicitly so the |
Hmm, sorry but I’m not quite following. The restrictions you mentioned, @tjzel, seem related to multi-file ordering. But the outlining transform we’re talking about here is within file. It would seem like you could traverse the file, find useDeferredValue calls, traverse within those to find the call to temp2(), follow that declaration, and then see the call to temp(), follow that declaration, and then add all of those to be treated as worklets. If at any point an invalid API like runOnJs() is called you bail out. What strikes me in particular is that it’s perfectly reasonable for a developer to write code like this directly, so it seems like it would be nice for reanimated to support this pattern if it’s a helper function in the same file. Is there a concrete problem with that approach? |
The problem is that I can't make any assumptions about the functions used inside the In the example |
Sure, but that actually doesn't have anything to do with whether the callback is outlined or not:
Unless I'm missing something, whether this is safe to turn into a worklet or not is a property of It might help to have a more specific example to look at, maybe something was lost in trying to distill down the problem for a repro? Because in general the outlining doesn't seem to be the problem re safety of workletizing, if outlining is local to the file. |
// A
useDeferredValue(() => {
obj.method(() => { ... });
}) In here, if |
Ahhhh ok. So Reanimated uses the distinction between inline callbacks vs refs to function declarations to decide what to auto-workletize. In that case I wonder if we can just add some property onto functions that we outline so that Reanimated's plugin can detect that. So in the original example, we could add an annotation onto the FunctionDeclaration node for temp and temp2 to say eg isOutlinedByReactCompiler:true. Would that work? |
I think it could work. I think the annotation should only be applied to functions outlined from the worklet context - I'm afraid that we'll encountered something that has been outlined and falsely assume that it should be a worklet. I haven't came up for an edge-case where it would manifest but I'm not crossing it out as a possibility. |
Another alternative would be to add some customization to the Compiler. You could expose APIs that other plugins can hook into during the transpilation process. This way Reanimated plugin could mark all worklets with the |
It seems like you could avoid this case by design. Presumably you start from known worker functions, ie callbacks to useDeferredValue. You then traverse into subexpressions, following inline function expressions but excluding reference to named variables. You could slightly amend this to check: is the variable reference to a local function declaration that was marked as outlined by React Compiler? If so, include that function as a root and keep going (maybe even inline it back again!). Ie if you’re starting from known safe-to-workletize functions, you’ll only reach an outlined function in the case that it was originally inlined. Does that sound like it would work? Re allowing pre-passes: extensive configuration is part of what make Babel exceedingly frustrating to use, and our design principle is to avoid such configuration as much as possible. Per above it seems like there is a good alternative. |
Yes, that's what we do right now, I just don't know the full extent of the changes the Compiler introduces so I'm trying to play it safe. I couldn't break it with the current version of the online playground so maybe I'm just too careful. I think we can try the approach you are suggesting and circle back on it in the future if there's any trouble.
I always thought it was the part of the awful documentation 🙈 |
Ok cool, we can add a property to outlined function declarations to make this clear to subsequent passes. |
What kind of issue is this?
Link to repro
https://playground.react.dev/#N4Igzg9grgTgxgUxALhHCA7MAXABAFQRwGEIBbAB0wQzwF5cAKYXDKMgIwRlwF8BKXHQB8uYAB0MuXOix4A1ggCe+CAEkMAEwQAPIbihgEAEW4BLAG4JNANQCGAGygJGjQSMnTpAbQCMAGlwAJkCAZgBdADoyOwpXd1E2Bwd+T1x+AG5JNJgEbFgpJIcsjF4SkF4gA
Repro steps
Current version the Compiler started to transform
react-native-reanimated
API callbacks too much, breaking the code. I compared it with an older version[email protected]
and the problem didn't happen there. It seems to be due to theenableFunctionOutlining
feature.Given the code:
Compiler changes it to
Note that
_temp
and_temp2
are extracted outside. It's ok to extract_temp2
, the callback ofuseDerivedValue
, it's handled on our side as a part of the support for the Compiler. It's not ok to extract_temp
.The problem lies in the fact that we can (and we do) deduce that
_temp2
must be a worklet, but we cannot in general infer if_temp
should be a worklet. This leads to_temp
not being workletized and causing a runtime error. We also can't workletize it regardless, in some flows it actually shouldn't be a worklet.To fix it we need to keep all functions defined in a worklet to stay in this worklet.
How often does this bug happen?
Every time
What version of React are you using?
19
What version of React Compiler are you using?
19.0.0-beta-bafa41b-20250307
The text was updated successfully, but these errors were encountered: