/* * 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 #include #include #include #include #include #include "NativePerformanceObserver.h" namespace facebook::react { struct PerformanceMark { std::string name; double timeStamp; }; struct PerformanceMarkHash { size_t operator()(const PerformanceMark *mark) const { return std::hash()(mark->name); } }; struct PerformanceMarkEqual { bool operator()(const PerformanceMark *lhs, const PerformanceMark *rhs) const { return lhs->name == rhs->name; } }; using PerformanceMarkRegistryType = std:: unordered_set; // Only the MARKS_BUFFER_SIZE amount of the latest marks will be kept in // memory for the sake of the "Performance.measure" mark name lookup constexpr size_t MARKS_BUFFER_SIZE = 1024; enum class PerformanceEntryType { UNDEFINED = 0, MARK = 1, MEASURE = 2, _COUNT = 3, }; class PerformanceEntryReporter { public: PerformanceEntryReporter(PerformanceEntryReporter const &) = delete; void operator=(PerformanceEntryReporter const &) = delete; // NOTE: This class is not thread safe, make sure that the calls are made from // the same thread. // TODO: Consider passing it as a parameter to the corresponding modules at // creation time instead of having the singleton. static PerformanceEntryReporter &getInstance(); void setReportingCallback(std::optional> callback); void startReporting(PerformanceEntryType entryType); void stopReporting(PerformanceEntryType entryType); const std::vector &getPendingEntries() const; std::vector popPendingEntries(); void clearPendingEntries(); void logEntry(const RawPerformanceEntry &entry); bool isReportingType(PerformanceEntryType entryType) const { return reportingType_[static_cast(entryType)]; } void mark(const std::string &name, double startTime, double duration); void clearMarks(const std::optional &markName); void measure( const std::string &name, double startTime, double endTime, const std::optional &duration, const std::optional &startMark, const std::optional &endMark); void clearMeasures(const std::optional &measureName); private: PerformanceEntryReporter() {} double getMarkTime(const std::string &markName) const; void clearEntries(std::function predicate); std::optional> callback_; std::vector entries_; std::array reportingType_{false}; // Mark registry for "measure" lookup PerformanceMarkRegistryType marks_registry_; std::array marks_buffer_; size_t marks_buffer_position_{0}; }; } // namespace facebook::react