diff --git a/flow-typed/npm/tinybench_v3.1.x.js b/flow-typed/npm/tinybench_v3.1.x.js new file mode 100644 index 00000000000000..ce884b5b94ddfe --- /dev/null +++ b/flow-typed/npm/tinybench_v3.1.x.js @@ -0,0 +1,121 @@ +/** + * Copyright (c) Meta Platforms, Inc. and affiliates. + * + * This source code is licensed under the MIT license found in the + * LICENSE file in the root directory of this source tree. + * + * @flow strict + * @format + * @oncall react_native + */ + +declare module 'tinybench' { + declare export class Task extends EventTarget { + name: string; + result: void | $ReadOnly; + runs: number; + + reset(): void; + run(): Promise; + runSync(): Task; + warmup(): Promise; + } + + export type Hook = ( + task: Task, + mode: 'run' | 'warmup', + ) => Promise | void; + + export type BenchOptions = { + iterations?: number, + name?: string, + now?: () => number, + setup?: Hook, + signal?: AbortSignal, + teardown?: Hook, + throws?: boolean, + time?: number, + warmup?: boolean, + warmupIterations?: number, + warmupTime?: number, + }; + + export interface Statistics { + aad: void | number; + critical: number; + df: number; + mad: void | number; + max: number; + mean: number; + min: number; + moe: number; + p50: void | number; + p75: void | number; + p99: void | number; + p995: void | number; + p999: void | number; + rme: number; + samples: number[]; + sd: number; + sem: number; + variance: number; + } + + export interface TaskResult { + critical: number; + df: number; + error?: Error; + hz: number; + latency: Statistics; + max: number; + mean: number; + min: number; + moe: number; + p75: number; + p99: number; + p995: number; + p999: number; + period: number; + rme: number; + samples: number[]; + sd: number; + sem: number; + throughput: Statistics; + totalTime: number; + variance: number; + } + + export type FnOptions = { + afterAll?: (this: Task) => void | Promise, + afterEach?: (this: Task) => void | Promise, + beforeAll?: (this: Task) => void | Promise, + beforeEach?: (this: Task) => void | Promise, + }; + + export type Fn = () => Promise | mixed; + + declare export class Bench extends EventTarget { + concurrency: null | 'task' | 'bench'; + name?: string; + opts: $ReadOnly; + threshold: number; + + constructor(options?: BenchOptions): this; + + // $FlowExpectedError[unsafe-getters-setters] + get results(): Array<$ReadOnly>; + + // $FlowExpectedError[unsafe-getters-setters] + get tasks(): Array; + + add(name: string, fn: Fn, fnOpts?: FnOptions): this; + getTask(name: string): void | Task; + remove(name: string): this; + reset(): void; + run(): Promise>; + runSync(): Array; + table( + convert?: (task: Task) => Record, + ): void | Array>; + } +} diff --git a/package.json b/package.json index 45b397ff3c0c5b..9f96dc46e28281 100644 --- a/package.json +++ b/package.json @@ -102,6 +102,7 @@ "shelljs": "^0.8.5", "signedsource": "^1.0.0", "supports-color": "^7.1.0", + "tinybench": "^3.1.0", "typescript": "5.0.4", "ws": "^6.2.3" }, diff --git a/packages/react-native-fantom/runner/entrypoint-template.js b/packages/react-native-fantom/runner/entrypoint-template.js index c52c66209e0e8a..dc99d9f0551d84 100644 --- a/packages/react-native-fantom/runner/entrypoint-template.js +++ b/packages/react-native-fantom/runner/entrypoint-template.js @@ -18,12 +18,14 @@ module.exports = function entrypointTemplate({ featureFlagsModulePath, featureFlags, snapshotConfig, + isRunningFromCI, }: { testPath: string, setupModulePath: string, featureFlagsModulePath: string, featureFlags: FantomTestConfigJsOnlyFeatureFlags, snapshotConfig: SnapshotConfig, + isRunningFromCI: boolean, }): string { return `/** * Copyright (c) Meta Platforms, Inc. and affiliates. @@ -38,6 +40,7 @@ module.exports = function entrypointTemplate({ */ import {registerTest} from '${setupModulePath}'; +import {setConstants} from '@react-native/fantom'; ${ Object.keys(featureFlags).length > 0 ? `import * as ReactNativeFeatureFlags from '${featureFlagsModulePath}'; @@ -50,6 +53,10 @@ ${Object.entries(featureFlags) : '' } +setConstants({ + isRunningFromCI: ${String(isRunningFromCI)}, +}); + registerTest(() => require('${testPath}'), ${JSON.stringify(snapshotConfig)}); `; }; diff --git a/packages/react-native-fantom/runner/getFantomTestConfig.js b/packages/react-native-fantom/runner/getFantomTestConfig.js index 3351ea9838db8a..995671b45e6427 100644 --- a/packages/react-native-fantom/runner/getFantomTestConfig.js +++ b/packages/react-native-fantom/runner/getFantomTestConfig.js @@ -46,6 +46,8 @@ const DEFAULT_MODE: FantomTestConfigMode = const FANTOM_FLAG_FORMAT = /^(\w+):(\w+)$/; +const FANTOM_BENCHMARK_SUITE_RE = /\nbenchmark(\s*)\.suite\(/g; + /** * Extracts the Fantom configuration from the test file, specified as part of * the docblock comment. E.g.: @@ -69,7 +71,9 @@ const FANTOM_FLAG_FORMAT = /^(\w+):(\w+)$/; export default function getFantomTestConfig( testPath: string, ): FantomTestConfig { - const docblock = extract(fs.readFileSync(testPath, 'utf8')); + const testContents = fs.readFileSync(testPath, 'utf8'); + + const docblock = extract(testContents); const pragmas = parse(docblock) as DocblockPragmas; const config: FantomTestConfig = { @@ -102,6 +106,10 @@ export default function getFantomTestConfig( default: throw new Error(`Invalid Fantom mode: ${mode}`); } + } else { + if (FANTOM_BENCHMARK_SUITE_RE.test(testContents)) { + config.mode = FantomTestConfigMode.Optimized; + } } const maybeRawFlagConfig = pragmas.fantom_flags; diff --git a/packages/react-native-fantom/runner/runner.js b/packages/react-native-fantom/runner/runner.js index 3c270350443021..b6580745ea394a 100644 --- a/packages/react-native-fantom/runner/runner.js +++ b/packages/react-native-fantom/runner/runner.js @@ -24,6 +24,7 @@ import { getBuckModesForPlatform, getDebugInfoFromCommandResult, getShortHash, + isRunningFromCI, printConsoleLog, runBuck2, runBuck2Sync, @@ -198,6 +199,7 @@ module.exports = async function runTest( updateSnapshot: snapshotState._updateSnapshot, data: getInitialSnapshotData(snapshotState), }, + isRunningFromCI: isRunningFromCI(), }); const entrypointPath = path.join( diff --git a/packages/react-native-fantom/runner/utils.js b/packages/react-native-fantom/runner/utils.js index 93873737d3f0d6..4ea73e21c51ac7 100644 --- a/packages/react-native-fantom/runner/utils.js +++ b/packages/react-native-fantom/runner/utils.js @@ -67,6 +67,16 @@ export type SyncCommandResult = { stderr: string, }; +function isEmpty(value: ?string): boolean { + return value == null || value === ''; +} + +export function isRunningFromCI(): boolean { + return ( + !isEmpty(process.env.SANDCASTLE) || !isEmpty(process.env.GITHUB_ACTIONS) + ); +} + function maybeLogCommand(command: string, args: Array): void { if (EnvironmentOptions.logCommands) { console.log(`RUNNING \`${command} ${args.join(' ')}\``); diff --git a/packages/react-native-fantom/src/Benchmark.js b/packages/react-native-fantom/src/Benchmark.js new file mode 100644 index 00000000000000..d8ff1fafeb348b --- /dev/null +++ b/packages/react-native-fantom/src/Benchmark.js @@ -0,0 +1,129 @@ +/** + * Copyright (c) Meta Platforms, Inc. and affiliates. + * + * This source code is licensed under the MIT license found in the + * LICENSE file in the root directory of this source tree. + * + * @flow strict-local + * @format + */ + +import {getConstants} from './index'; +import nullthrows from 'nullthrows'; +import NativeCPUTime from 'react-native/src/private/specs/modules/NativeCPUTime'; +import { + Bench, + type BenchOptions, + type FnOptions, + type TaskResult, +} from 'tinybench'; + +type SuiteOptions = $ReadOnly<{ + ...Pick< + BenchOptions, + 'iterations' | 'time' | 'warmup' | 'warmupIterations' | 'warmupTime', + >, + disableOptimizedBuildCheck?: boolean, +}>; + +type SuiteResults = Array<$ReadOnly>; + +interface SuiteAPI { + add(name: string, fn: () => void, options?: FnOptions): SuiteAPI; + verify(fn: (results: SuiteResults) => void): SuiteAPI; +} + +export function suite( + suiteName: string, + suiteOptions?: ?SuiteOptions, +): SuiteAPI { + const tasks: Array<{ + name: string, + fn: () => void, + options: FnOptions | void, + }> = []; + const verifyFns = []; + + global.it(suiteName, () => { + if (tasks.length === 0) { + throw new Error('No benchmark tests defined'); + } + + const {isRunningFromCI} = getConstants(); + + // If we're running from CI and there's no verification function, there's + // no point in running the benchmark. + // We still run a single iteration of each test just to make sure that the + // logic in the benchmark doesn't break. + const isTestOnly = isRunningFromCI && verifyFns.length === 0; + + const overriddenOptions: BenchOptions = isTestOnly + ? { + warmupIterations: 1, + warmupTime: 0, + iterations: 1, + time: 0, + } + : {}; + + const {disableOptimizedBuildCheck, ...benchOptions} = suiteOptions ?? {}; + + const bench = new Bench({ + ...benchOptions, + ...overriddenOptions, + name: suiteName, + throws: true, + now: () => NativeCPUTime.getCPUTimeNanos() / 1000000, + }); + + for (const task of tasks) { + bench.add(task.name, task.fn, task.options); + } + + bench.runSync(); + + if (!isTestOnly) { + printBenchmarkResults(bench); + } + + for (const verify of verifyFns) { + verify(bench.results); + } + + if (!isTestOnly && !NativeCPUTime.hasAccurateCPUTimeNanosForBenchmarks()) { + throw new Error( + '`NativeCPUTime` module does not provide accurate CPU time information in this environment. Please run the benchmarks in an environment where it does.', + ); + } + + if (__DEV__ && disableOptimizedBuildCheck !== true) { + throw new Error('Benchmarks should not be run in development mode'); + } + }); + + const suiteAPI = { + add(name: string, fn: () => void, options?: FnOptions): SuiteAPI { + tasks.push({name, fn, options}); + return suiteAPI; + }, + verify(fn: (results: SuiteResults) => void): SuiteAPI { + verifyFns.push(fn); + return suiteAPI; + }, + }; + + return suiteAPI; +} + +function printBenchmarkResults(bench: Bench) { + const longestTaskNameLength = bench.tasks.reduce( + (maxLength, task) => Math.max(maxLength, task.name.length), + 0, + ); + const separatorWidth = 121 + longestTaskNameLength - 'Task name'.length; + + console.log('-'.repeat(separatorWidth)); + console.log(bench.name); + console.table(nullthrows(bench.table())); + console.log('-'.repeat(separatorWidth) + '\n'); +} diff --git a/packages/react-native-fantom/src/index.js b/packages/react-native-fantom/src/index.js index cf78e5d9fda77e..3124225ff92634 100644 --- a/packages/react-native-fantom/src/index.js +++ b/packages/react-native-fantom/src/index.js @@ -14,6 +14,7 @@ import type { } from './getFantomRenderedOutput'; import type {MixedElement} from 'react'; +import * as Benchmark from './Benchmark'; import getFantomRenderedOutput from './getFantomRenderedOutput'; import ReactFabric from 'react-native/Libraries/Renderer/shims/ReactFabric'; import NativeFantom from 'react-native/src/private/specs/modules/NativeFantom'; @@ -63,7 +64,7 @@ class Root { this.#hasRendered = true; } - ReactFabric.render(element, this.#surfaceId, () => {}, true); + ReactFabric.render(element, this.#surfaceId, null, true); } getMountingLogs(): Array { @@ -137,3 +138,85 @@ export function runWorkLoop(): void { export function createRoot(rootConfig?: RootConfig): Root { return new Root(rootConfig); } + +export const benchmark = Benchmark; + +type FantomConstants = $ReadOnly<{ + isRunningFromCI: boolean, +}>; + +let constants: FantomConstants = { + isRunningFromCI: false, +}; + +export function getConstants(): FantomConstants { + return constants; +} + +export function setConstants(newConstants: FantomConstants): void { + constants = newConstants; +} + +/** + * Quick and dirty polyfills required by tinybench. + */ + +if (typeof global.Event === 'undefined') { + global.Event = class Event { + constructor() {} + }; +} else { + console.warn( + 'The global Event class is already defined. If this API is already defined by React Native, you might want to remove this logic.', + ); +} + +if (typeof global.EventTarget === 'undefined') { + global.EventTarget = class EventTarget { + listeners: $FlowFixMe; + + constructor() { + this.listeners = {}; + } + + addEventListener(type: string, cb: () => void) { + if (!(type in this.listeners)) { + this.listeners[type] = []; + } + this.listeners[type].push(cb); + } + + removeEventListener(type: string, cb: () => void): void { + if (!(type in this.listeners)) { + return; + } + let handlers = this.listeners[type]; + for (let i in handlers) { + if (cb === handlers[i]) { + handlers.splice(i, 1); + return; + } + } + } + + dispatchEvent(type: string, event: Event) { + if (!(type in this.listeners)) { + return; + } + let handlers = this.listeners[type]; + for (let i in handlers) { + handlers[i].call(this, event); + } + } + + clearEventListeners() { + for (let i in this.listeners) { + delete this.listeners[i]; + } + } + }; +} else { + console.warn( + 'The global Event class is already defined. If this API is already defined by React Native, you might want to remove this logic.', + ); +} diff --git a/packages/react-native/ReactCommon/react/nativemodule/cputime/CPUTime.h b/packages/react-native/ReactCommon/react/nativemodule/cputime/CPUTime.h new file mode 100644 index 00000000000000..fe260ff03f2141 --- /dev/null +++ b/packages/react-native/ReactCommon/react/nativemodule/cputime/CPUTime.h @@ -0,0 +1,55 @@ +/* + * Copyright (c) Meta Platforms, Inc. and affiliates. + * + * This source code is licensed under the MIT license found in the + * LICENSE file in the root directory of this source tree. + */ + +#pragma once + +#ifdef USE_POSIX_TIME +#include +#else +#include +#endif + +#ifdef USE_POSIX_TIME + +namespace { +const double NANOSECONDS_IN_A_SECOND = 1000000000; +} // namespace + +#endif + +namespace facebook::react { + +#ifdef USE_POSIX_TIME + +inline double getCPUTimeNanos() { + struct timespec time {}; + clock_gettime(CLOCK_THREAD_CPUTIME_ID, &time); + return static_cast(time.tv_sec) * NANOSECONDS_IN_A_SECOND + + static_cast(time.tv_nsec); +} + +inline bool hasAccurateCPUTimeNanosForBenchmarks() { + return true; +} + +#else + +inline double getCPUTimeNanos() { + auto now = std::chrono::steady_clock::now(); + return static_cast( + std::chrono::duration_cast( + now.time_since_epoch()) + .count()); +} + +inline bool hasAccurateCPUTimeNanosForBenchmarks() { + return false; +} + +#endif + +} // namespace facebook::react diff --git a/packages/react-native/ReactCommon/react/nativemodule/cputime/NativeCPUTime.cpp b/packages/react-native/ReactCommon/react/nativemodule/cputime/NativeCPUTime.cpp new file mode 100644 index 00000000000000..498a21d09889dc --- /dev/null +++ b/packages/react-native/ReactCommon/react/nativemodule/cputime/NativeCPUTime.cpp @@ -0,0 +1,35 @@ +/* + * Copyright (c) Meta Platforms, Inc. and affiliates. + * + * This source code is licensed under the MIT license found in the + * LICENSE file in the root directory of this source tree. + */ + +#include "NativeCPUTime.h" + +#include "CPUTime.h" + +#ifdef RN_DISABLE_OSS_PLUGIN_HEADER +#include "Plugins.h" +#endif + +std::shared_ptr NativeCPUTimeModuleProvider( + std::shared_ptr jsInvoker) { + return std::make_shared(std::move(jsInvoker)); +} + +namespace facebook::react { + +NativeCPUTime::NativeCPUTime(std::shared_ptr jsInvoker) + : NativeCPUTimeCxxSpec(std::move(jsInvoker)) {} + +double NativeCPUTime::getCPUTimeNanos(jsi::Runtime& /*runtime*/) { + return facebook::react::getCPUTimeNanos(); +} + +bool NativeCPUTime::hasAccurateCPUTimeNanosForBenchmarks( + jsi::Runtime& /*runtime*/) { + return facebook::react::hasAccurateCPUTimeNanosForBenchmarks(); +} + +} // namespace facebook::react diff --git a/packages/react-native/ReactCommon/react/nativemodule/cputime/NativeCPUTime.h b/packages/react-native/ReactCommon/react/nativemodule/cputime/NativeCPUTime.h new file mode 100644 index 00000000000000..c6e60b4f66623b --- /dev/null +++ b/packages/react-native/ReactCommon/react/nativemodule/cputime/NativeCPUTime.h @@ -0,0 +1,28 @@ +/* + * Copyright (c) Meta Platforms, Inc. and affiliates. + * + * This source code is licensed under the MIT license found in the + * LICENSE file in the root directory of this source tree. + */ + +#pragma once + +#if __has_include("rncoreJSI.h") // Cmake headers on Android +#include "rncoreJSI.h" +#elif __has_include("FBReactNativeSpecJSI.h") // CocoaPod headers on Apple +#include "FBReactNativeSpecJSI.h" +#else +#include +#endif + +namespace facebook::react { + +class NativeCPUTime : public NativeCPUTimeCxxSpec { + public: + explicit NativeCPUTime(std::shared_ptr jsInvoker); + + double getCPUTimeNanos(jsi::Runtime& runtime); + bool hasAccurateCPUTimeNanosForBenchmarks(jsi::Runtime& runtime); +}; + +} // namespace facebook::react diff --git a/packages/react-native/src/private/specs/modules/NativeCPUTime.js b/packages/react-native/src/private/specs/modules/NativeCPUTime.js new file mode 100644 index 00000000000000..a2736dc86a083e --- /dev/null +++ b/packages/react-native/src/private/specs/modules/NativeCPUTime.js @@ -0,0 +1,24 @@ +/** + * Copyright (c) Meta Platforms, Inc. and affiliates. + * + * This source code is licensed under the MIT license found in the + * LICENSE file in the root directory of this source tree. + * + * @flow strict + * @format + */ + +import type {TurboModule} from '../../../../Libraries/TurboModule/RCTExport'; + +import * as TurboModuleRegistry from '../../../../Libraries/TurboModule/TurboModuleRegistry'; + +/** + * This is an internal native module meant to be used for performance + * measurements and benchmarks. It is not meant to be used in production. + */ +export interface Spec extends TurboModule { + +getCPUTimeNanos: () => number; + +hasAccurateCPUTimeNanosForBenchmarks: () => boolean; +} + +export default (TurboModuleRegistry.getEnforcing('CPUTimeCxx'): Spec); diff --git a/yarn.lock b/yarn.lock index 0fc7c24ecfa840..307ba0037c4576 100644 --- a/yarn.lock +++ b/yarn.lock @@ -8348,6 +8348,11 @@ tiny-invariant@^1.3.3: resolved "https://registry.yarnpkg.com/tiny-invariant/-/tiny-invariant-1.3.3.tgz#46680b7a873a0d5d10005995eb90a70d74d60127" integrity sha512-+FbBPE1o9QAYvviau/qC5SE3caw21q3xkvWKBtja5vgqOWIHHJ3ioaq1VPfn/Szqctz2bU/oYeKd9/z5BL+PVg== +tinybench@^3.1.0: + version "3.1.0" + resolved "https://registry.yarnpkg.com/tinybench/-/tinybench-3.1.0.tgz#ec68451ff05233cf3de12c46f39f06011897109a" + integrity sha512-Km+oMh2xqNCxuyoUsqbRmHgFSd8sATh7v7xreP+kHN6x67w28Pawr83WmBxcaORvxkc0Ex6zgqK951yBnTFaaQ== + tmp@^0.0.33: version "0.0.33" resolved "https://registry.yarnpkg.com/tmp/-/tmp-0.0.33.tgz#6d34335889768d21b2bcda0aa277ced3b1bfadf9"