Skip to content

Route enableCppPropsIteratorSetter through a copy ctor + RawProps::forEachItem#57328

Open
javache wants to merge 2 commits into
react:mainfrom
javache:export-D109568749
Open

Route enableCppPropsIteratorSetter through a copy ctor + RawProps::forEachItem#57328
javache wants to merge 2 commits into
react:mainfrom
javache:export-D109568749

Conversation

@javache

@javache javache commented Jun 24, 2026

Copy link
Copy Markdown
Contributor

Summary:
Today the iterator-setter path in ConcreteComponentDescriptor::cloneProps runs three sequential walks over the input — RawProps::parse(parser) (builds keyIndexToValueIndex_ for convertRawProp), static_cast<folly::dynamic>(rawProps) (materializes a folly::dynamic via jsi::dynamicFromValue in JSI mode), and then dynamic.items() to dispatch setProp. Only the third is actually used: convertRawProp is never called on the iterator-setter branch, and the folly::dynamic materialization exists only as iteration scaffolding.

Restructure so the runtime flag picks one of two construction paths up front:

  • Iterator-setter — copy-construct from sourceProps via the (re-enabled) Props copy ctor, then walk rawProps in-place via the new RawProps::forEachItem helper and route each entry through setProp. parse() is skipped entirely; the folly::dynamic materialization is skipped in Mode::JSI.
  • Classic — unchanged: parse() + 3-arg convertRawProp-driven ctor.

forEachItem switches on RawProps::Mode:

  • Mode::JSI — walks value_.asObject(*runtime_).getPropertyNames(...) and constructs RawValue from each jsi::Value directly, no folly::dynamic in between.
  • Mode::Dynamic — iterates dynamic_.items() (same as today).
  • Mode::Empty — no-op.

A new HasIteratorSetterCtor<T> concept (std::copy_constructible<T>) documents the contract and feeds a static_assert in cloneProps, so a future Props type that deletes its copy ctor fails at compile time rather than silently diverging at runtime between the two flag states.

The RN_SERIALIZABLE_STATE Props 2.0 accumulation branch keeps its existing dynamic-iteration shape — when fallbackToDynamicRawPropsAccumulation is true, initializeDynamicProps has already merged the source's rawProps with the input onto shadowNodeProps->rawProps, so we iterate that merged dynamic rather than the raw input.

The per-field flag ? sourceProps.X : convertRawProp(...) ternaries across every Props .cpp file become dead in the flag-on path (the copy ctor handles those fields) but are still functional in the flag-off path. They get removed in a follow-up cleanup; this diff is structurally non-breaking on either flag state.

Changelog:
[Internal]

Differential Revision: D109568749

NickGerleman and others added 2 commits June 24, 2026 10:19
Summary:
RawPropsKey previously stored three `const char*` fields
(prefix, name, suffix) that were concatenated at runtime to form
property names.

This is pretty niche, used to make a few patterns simpler, but also can lead to confusing conflicts when the same property name can be represented in different ways (e.g. T174300106). Iterator style props parsing also completely avoids it.

Lets change the API to a flat name instead.

This change is breaking, but could only find a single user (Nitro module) effected, searching through `react-native-libraries`.

Changelog:
[General][Breaking]  - Remove RawPropsKey prefix and suffix

Differential Revision: D94367880

Reviewed By: javache
…:forEachItem`

Summary:
Today the iterator-setter path in `ConcreteComponentDescriptor::cloneProps` runs three sequential walks over the input — `RawProps::parse(parser)` (builds `keyIndexToValueIndex_` for `convertRawProp`), `static_cast<folly::dynamic>(rawProps)` (materializes a `folly::dynamic` via `jsi::dynamicFromValue` in JSI mode), and then `dynamic.items()` to dispatch `setProp`. Only the third is actually used: `convertRawProp` is never called on the iterator-setter branch, and the `folly::dynamic` materialization exists only as iteration scaffolding.

Restructure so the runtime flag picks one of two construction paths up front:

- **Iterator-setter** — copy-construct from `sourceProps` via the (re-enabled) `Props` copy ctor, then walk `rawProps` in-place via the new `RawProps::forEachItem` helper and route each entry through `setProp`. `parse()` is skipped entirely; the `folly::dynamic` materialization is skipped in `Mode::JSI`.
- **Classic** — unchanged: `parse()` + 3-arg `convertRawProp`-driven ctor.

`forEachItem` switches on `RawProps::Mode`:
- `Mode::JSI` — walks `value_.asObject(*runtime_).getPropertyNames(...)` and constructs `RawValue` from each `jsi::Value` directly, no `folly::dynamic` in between.
- `Mode::Dynamic` — iterates `dynamic_.items()` (same as today).
- `Mode::Empty` — no-op.

A new `HasIteratorSetterCtor<T>` concept (`std::copy_constructible<T>`) documents the contract and feeds a `static_assert` in `cloneProps`, so a future Props type that deletes its copy ctor fails at compile time rather than silently diverging at runtime between the two flag states.

The `RN_SERIALIZABLE_STATE` Props 2.0 accumulation branch keeps its existing dynamic-iteration shape — when `fallbackToDynamicRawPropsAccumulation` is true, `initializeDynamicProps` has already merged the source's rawProps with the input onto `shadowNodeProps->rawProps`, so we iterate that merged dynamic rather than the raw input.

The per-field `flag ? sourceProps.X : convertRawProp(...)` ternaries across every Props .cpp file become dead in the flag-on path (the copy ctor handles those fields) but are still functional in the flag-off path. They get removed in a follow-up cleanup; this diff is structurally non-breaking on either flag state.

Changelog:
[Internal]

Differential Revision: D109568749
@meta-cla meta-cla Bot added the CLA Signed This label is managed by the Facebook bot. Authors need to sign the CLA before a PR can be reviewed. label Jun 24, 2026
@meta-codesync

meta-codesync Bot commented Jun 24, 2026

Copy link
Copy Markdown

@javache has exported this pull request. If you are a Meta employee, you can view the originating Diff in D109568749.

javache added a commit to javache/react-native that referenced this pull request Jun 25, 2026
…:forEachItem` (react#57328)

Summary:

Today the iterator-setter path in `ConcreteComponentDescriptor::cloneProps` runs three sequential walks over the input — `RawProps::parse(parser)` (builds `keyIndexToValueIndex_` for `convertRawProp`), `static_cast<folly::dynamic>(rawProps)` (materializes a `folly::dynamic` via `jsi::dynamicFromValue` in JSI mode), and then `dynamic.items()` to dispatch `setProp`. Only the third is actually used: `convertRawProp` is never called on the iterator-setter branch, and the `folly::dynamic` materialization exists only as iteration scaffolding.

Restructure so the runtime flag picks one of two construction paths up front:

- **Iterator-setter** — copy-construct from `sourceProps` via the (re-enabled) `Props` copy ctor, then walk `rawProps` in-place via the new `RawProps::forEachItem` helper and route each entry through `setProp`. `parse()` is skipped entirely; the `folly::dynamic` materialization is skipped in `Mode::JSI`.
- **Classic** — unchanged: `parse()` + 3-arg `convertRawProp`-driven ctor.

`forEachItem` switches on `RawProps::Mode`:
- `Mode::JSI` — walks `value_.asObject(*runtime_).getPropertyNames(...)` and constructs `RawValue` from each `jsi::Value` directly, no `folly::dynamic` in between.
- `Mode::Dynamic` — iterates `dynamic_.items()` (same as today).
- `Mode::Empty` — no-op.

A new `HasIteratorSetterCtor<T>` concept (`std::copy_constructible<T>`) documents the contract and feeds a `static_assert` in `cloneProps`, so a future Props type that deletes its copy ctor fails at compile time rather than silently diverging at runtime between the two flag states.

The `RN_SERIALIZABLE_STATE` Props 2.0 accumulation branch keeps its existing dynamic-iteration shape — when `fallbackToDynamicRawPropsAccumulation` is true, `initializeDynamicProps` has already merged the source's rawProps with the input onto `shadowNodeProps->rawProps`, so we iterate that merged dynamic rather than the raw input.

The per-field `flag ? sourceProps.X : convertRawProp(...)` ternaries across every Props .cpp file become dead in the flag-on path (the copy ctor handles those fields) but are still functional in the flag-off path. They get removed in a follow-up cleanup; this diff is structurally non-breaking on either flag state.

Changelog:
[Internal]

Differential Revision: D109568749
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment

Labels

CLA Signed This label is managed by the Facebook bot. Authors need to sign the CLA before a PR can be reviewed. meta-exported p: Facebook Partner: Facebook Partner

Projects

None yet

Development

Successfully merging this pull request may close these issues.

2 participants