14514f5e3Sopenharmony_ci/* 24514f5e3Sopenharmony_ci * Copyright (c) 2024 Huawei Device Co., Ltd. 34514f5e3Sopenharmony_ci * Licensed under the Apache License, Version 2.0 (the "License"); 44514f5e3Sopenharmony_ci * you may not use this file except in compliance with the License. 54514f5e3Sopenharmony_ci * You may obtain a copy of the License at 64514f5e3Sopenharmony_ci * 74514f5e3Sopenharmony_ci * http://www.apache.org/licenses/LICENSE-2.0 84514f5e3Sopenharmony_ci * 94514f5e3Sopenharmony_ci * Unless required by applicable law or agreed to in writing, software 104514f5e3Sopenharmony_ci * distributed under the License is distributed on an "AS IS" BASIS, 114514f5e3Sopenharmony_ci * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 124514f5e3Sopenharmony_ci * See the License for the specific language governing permissions and 134514f5e3Sopenharmony_ci * limitations under the License. 144514f5e3Sopenharmony_ci */ 154514f5e3Sopenharmony_ci 164514f5e3Sopenharmony_ci#ifndef ECMASCRIPT_MEM_GC_KEY_STATS_H 174514f5e3Sopenharmony_ci#define ECMASCRIPT_MEM_GC_KEY_STATS_H 184514f5e3Sopenharmony_ci 194514f5e3Sopenharmony_ci#include <ctime> 204514f5e3Sopenharmony_ci#include <chrono> 214514f5e3Sopenharmony_ci 224514f5e3Sopenharmony_ci#include "libpandabase/macros.h" 234514f5e3Sopenharmony_ci 244514f5e3Sopenharmony_ci#include "ecmascript/mem/mem_common.h" 254514f5e3Sopenharmony_ci#include "ecmascript/log_wrapper.h" 264514f5e3Sopenharmony_ci 274514f5e3Sopenharmony_cinamespace panda::ecmascript { 284514f5e3Sopenharmony_ciusing Clock = std::chrono::high_resolution_clock; 294514f5e3Sopenharmony_ciclass Heap; 304514f5e3Sopenharmony_ciclass GCStats; 314514f5e3Sopenharmony_ci 324514f5e3Sopenharmony_cienum class RecordKeyData : uint8_t { 334514f5e3Sopenharmony_ci GC_TOTAL_MEM_USED = 0, 344514f5e3Sopenharmony_ci GC_TOTAL_MEM_COMMITTED, 354514f5e3Sopenharmony_ci GC_ACTIVE_MEM_USED, 364514f5e3Sopenharmony_ci GC_ACTIVE_MEM_COMMITTED, 374514f5e3Sopenharmony_ci GC_OLD_MEM_USED, 384514f5e3Sopenharmony_ci GC_OLD_MEM_COMMITTED, 394514f5e3Sopenharmony_ci GC_HUGE_MEM_USED, 404514f5e3Sopenharmony_ci GC_HUGE_MEM_COMMITTED, 414514f5e3Sopenharmony_ci NUM_OF_KEY_DATA, 424514f5e3Sopenharmony_ci FIRST_DATA = GC_TOTAL_MEM_USED, 434514f5e3Sopenharmony_ci}; 444514f5e3Sopenharmony_ci 454514f5e3Sopenharmony_cienum class RecordKeyDuration : uint8_t { 464514f5e3Sopenharmony_ci GC_TOTAL_TIME = 0, 474514f5e3Sopenharmony_ci GC_MARK_TIME, 484514f5e3Sopenharmony_ci GC_EVACUATE_TIME, 494514f5e3Sopenharmony_ci NUM_OF_KEY_DURATION, 504514f5e3Sopenharmony_ci FIRST_DATA = GC_TOTAL_TIME, 514514f5e3Sopenharmony_ci}; 524514f5e3Sopenharmony_ci 534514f5e3Sopenharmony_ciclass GCKeyStats { 544514f5e3Sopenharmony_cistatic constexpr size_t KEY_PAUSE_TIME = 20; // ms 554514f5e3Sopenharmony_cistatic constexpr int NEED_SEND_TIMES = 200; 564514f5e3Sopenharmony_cistatic constexpr int MIN_SEND_INTERVAL = 60; // seconds 574514f5e3Sopenharmony_cipublic: 584514f5e3Sopenharmony_ci explicit GCKeyStats(const Heap *heap) : heap_(heap) {} 594514f5e3Sopenharmony_ci GCKeyStats(const Heap *heap, GCStats *gcStats) : heap_(heap), 604514f5e3Sopenharmony_ci gcStats_(gcStats) {} 614514f5e3Sopenharmony_ci ~GCKeyStats() = default; 624514f5e3Sopenharmony_ci 634514f5e3Sopenharmony_ci void AddGCStatsToKey(); 644514f5e3Sopenharmony_ci bool CheckIfMainThread() const; 654514f5e3Sopenharmony_ci bool CheckIfKeyPauseTime() const; 664514f5e3Sopenharmony_ci void SendSysEventBeforeDump(std::string type, size_t limitSize, size_t activeMemory) const; 674514f5e3Sopenharmony_ci 684514f5e3Sopenharmony_ci void IncGCCount() 694514f5e3Sopenharmony_ci { 704514f5e3Sopenharmony_ci gcCount_++; 714514f5e3Sopenharmony_ci } 724514f5e3Sopenharmony_ci 734514f5e3Sopenharmony_ci void AddRecordDataStats(RecordKeyData idx, int value) 744514f5e3Sopenharmony_ci { 754514f5e3Sopenharmony_ci recordDataStats_[GetRecordDataStatsIndex(idx)] += value; 764514f5e3Sopenharmony_ci } 774514f5e3Sopenharmony_ci 784514f5e3Sopenharmony_ci int GetRecordDataStats(RecordKeyData idx) const 794514f5e3Sopenharmony_ci { 804514f5e3Sopenharmony_ci return recordDataStats_[GetRecordDataStatsIndex(idx)]; 814514f5e3Sopenharmony_ci } 824514f5e3Sopenharmony_ci 834514f5e3Sopenharmony_ci void AddRecordKeyDuration(RecordKeyDuration idx, float value) 844514f5e3Sopenharmony_ci { 854514f5e3Sopenharmony_ci recordDurationStats_[GetRecordKeyDurationIndex(idx)] += value; 864514f5e3Sopenharmony_ci } 874514f5e3Sopenharmony_ci 884514f5e3Sopenharmony_ci float GetRecordKeyDuration(RecordKeyDuration idx) const 894514f5e3Sopenharmony_ci { 904514f5e3Sopenharmony_ci return recordDurationStats_[GetRecordKeyDurationIndex(idx)]; 914514f5e3Sopenharmony_ci } 924514f5e3Sopenharmony_ci 934514f5e3Sopenharmony_ciprivate: 944514f5e3Sopenharmony_ci void SendSysEvent() const; 954514f5e3Sopenharmony_ci void InitializeRecordList(); 964514f5e3Sopenharmony_ci void PrintKeyStatisticResult() const; 974514f5e3Sopenharmony_ci 984514f5e3Sopenharmony_ci bool CheckLastSendTimeIfSend() const 994514f5e3Sopenharmony_ci { 1004514f5e3Sopenharmony_ci return lastSendTimestamp_ == Clock::time_point::min() || 1014514f5e3Sopenharmony_ci std::chrono::duration_cast<std::chrono::seconds>(Clock::now() - lastSendTimestamp_).count() >= 1024514f5e3Sopenharmony_ci MIN_SEND_INTERVAL; 1034514f5e3Sopenharmony_ci } 1044514f5e3Sopenharmony_ci 1054514f5e3Sopenharmony_ci bool CheckCountIfSend() const 1064514f5e3Sopenharmony_ci { 1074514f5e3Sopenharmony_ci return recordCount_ >= NEED_SEND_TIMES; 1084514f5e3Sopenharmony_ci } 1094514f5e3Sopenharmony_ci 1104514f5e3Sopenharmony_ci int GetRecordDataStatsIndex(RecordKeyData dataIdx) const 1114514f5e3Sopenharmony_ci { 1124514f5e3Sopenharmony_ci return (int)dataIdx - (int)RecordKeyData::FIRST_DATA; 1134514f5e3Sopenharmony_ci } 1144514f5e3Sopenharmony_ci 1154514f5e3Sopenharmony_ci int GetRecordKeyDurationIndex(RecordKeyDuration dataIdx) const 1164514f5e3Sopenharmony_ci { 1174514f5e3Sopenharmony_ci return (int)dataIdx - (int)RecordKeyDuration::FIRST_DATA; 1184514f5e3Sopenharmony_ci } 1194514f5e3Sopenharmony_ci 1204514f5e3Sopenharmony_ci int SizeToIntKB(size_t size) const 1214514f5e3Sopenharmony_ci { 1224514f5e3Sopenharmony_ci return static_cast<int>(size / 1_KB); 1234514f5e3Sopenharmony_ci } 1244514f5e3Sopenharmony_ci 1254514f5e3Sopenharmony_ci const Heap *heap_; 1264514f5e3Sopenharmony_ci GCStats *gcStats_; 1274514f5e3Sopenharmony_ci 1284514f5e3Sopenharmony_ci int gcCount_{0}; 1294514f5e3Sopenharmony_ci int recordCount_{0}; 1304514f5e3Sopenharmony_ci Clock::time_point lastSendTimestamp_; 1314514f5e3Sopenharmony_ci float recordDurationStats_[(uint8_t)RecordKeyDuration::NUM_OF_KEY_DURATION]{0.0f}; 1324514f5e3Sopenharmony_ci int recordDataStats_[(uint8_t)RecordKeyData::NUM_OF_KEY_DATA]{0}; 1334514f5e3Sopenharmony_ci}; 1344514f5e3Sopenharmony_ci 1354514f5e3Sopenharmony_ci} // namespace panda::ecmascript 1364514f5e3Sopenharmony_ci#endif // ECMASCRIPT_MEM_GC_KEY_STATS_H