xref: /aosp_15_r20/system/chre/host/common/metrics_reporter.cc (revision 84e339476a462649f82315436d70fd732297a399)
1*84e33947SAndroid Build Coastguard Worker /*
2*84e33947SAndroid Build Coastguard Worker  * Copyright (C) 2023 The Android Open Source Project
3*84e33947SAndroid Build Coastguard Worker  *
4*84e33947SAndroid Build Coastguard Worker  * Licensed under the Apache License, Version 2.0 (the "License");
5*84e33947SAndroid Build Coastguard Worker  * you may not use this file except in compliance with the License.
6*84e33947SAndroid Build Coastguard Worker  * You may obtain a copy of the License at
7*84e33947SAndroid Build Coastguard Worker  *
8*84e33947SAndroid Build Coastguard Worker  *      http://www.apache.org/licenses/LICENSE-2.0
9*84e33947SAndroid Build Coastguard Worker  *
10*84e33947SAndroid Build Coastguard Worker  * Unless required by applicable law or agreed to in writing, software
11*84e33947SAndroid Build Coastguard Worker  * distributed under the License is distributed on an "AS IS" BASIS,
12*84e33947SAndroid Build Coastguard Worker  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13*84e33947SAndroid Build Coastguard Worker  * See the License for the specific language governing permissions and
14*84e33947SAndroid Build Coastguard Worker  * limitations under the License.
15*84e33947SAndroid Build Coastguard Worker  */
16*84e33947SAndroid Build Coastguard Worker 
17*84e33947SAndroid Build Coastguard Worker #include "chre_host/metrics_reporter.h"
18*84e33947SAndroid Build Coastguard Worker #include <chre_atoms_log.h>
19*84e33947SAndroid Build Coastguard Worker #include "chre_host/log.h"
20*84e33947SAndroid Build Coastguard Worker 
21*84e33947SAndroid Build Coastguard Worker #include <cinttypes>
22*84e33947SAndroid Build Coastguard Worker #include <limits>
23*84e33947SAndroid Build Coastguard Worker #include <mutex>
24*84e33947SAndroid Build Coastguard Worker 
25*84e33947SAndroid Build Coastguard Worker #include <android/binder_manager.h>
26*84e33947SAndroid Build Coastguard Worker 
27*84e33947SAndroid Build Coastguard Worker namespace android::chre {
28*84e33947SAndroid Build Coastguard Worker 
29*84e33947SAndroid Build Coastguard Worker using ::aidl::android::frameworks::stats::IStats;
30*84e33947SAndroid Build Coastguard Worker using ::aidl::android::frameworks::stats::VendorAtom;
31*84e33947SAndroid Build Coastguard Worker using ::aidl::android::frameworks::stats::VendorAtomValue;
32*84e33947SAndroid Build Coastguard Worker using ::android::chre::Atoms::CHRE_AP_WAKE_UP_OCCURRED;
33*84e33947SAndroid Build Coastguard Worker using ::android::chre::Atoms::CHRE_EVENT_QUEUE_SNAPSHOT_REPORTED;
34*84e33947SAndroid Build Coastguard Worker using ::android::chre::Atoms::CHRE_HAL_NANOAPP_LOAD_FAILED;
35*84e33947SAndroid Build Coastguard Worker using ::android::chre::Atoms::CHRE_PAL_OPEN_FAILED;
36*84e33947SAndroid Build Coastguard Worker using ::android::chre::Atoms::ChreHalNanoappLoadFailed;
37*84e33947SAndroid Build Coastguard Worker using ::android::chre::Atoms::ChrePalOpenFailed;
38*84e33947SAndroid Build Coastguard Worker 
getStatsService()39*84e33947SAndroid Build Coastguard Worker std::shared_ptr<IStats> MetricsReporter::getStatsService() {
40*84e33947SAndroid Build Coastguard Worker   const std::string statsServiceName =
41*84e33947SAndroid Build Coastguard Worker       std::string(IStats::descriptor).append("/default");
42*84e33947SAndroid Build Coastguard Worker   if (!AServiceManager_isDeclared(statsServiceName.c_str())) {
43*84e33947SAndroid Build Coastguard Worker     LOGE("Stats service is not declared.");
44*84e33947SAndroid Build Coastguard Worker     return nullptr;
45*84e33947SAndroid Build Coastguard Worker   }
46*84e33947SAndroid Build Coastguard Worker 
47*84e33947SAndroid Build Coastguard Worker   ndk::SpAIBinder statsServiceBinder =
48*84e33947SAndroid Build Coastguard Worker       ndk::SpAIBinder(AServiceManager_waitForService(statsServiceName.c_str()));
49*84e33947SAndroid Build Coastguard Worker   if (statsServiceBinder.get() == nullptr) {
50*84e33947SAndroid Build Coastguard Worker     LOGE("Failed to get the IStats service binder");
51*84e33947SAndroid Build Coastguard Worker     return nullptr;
52*84e33947SAndroid Build Coastguard Worker   }
53*84e33947SAndroid Build Coastguard Worker 
54*84e33947SAndroid Build Coastguard Worker   binder_status_t status = AIBinder_linkToDeath(
55*84e33947SAndroid Build Coastguard Worker       statsServiceBinder.get(), AIBinder_DeathRecipient_new([](void *cookie) {
56*84e33947SAndroid Build Coastguard Worker         auto *metricsReporter = static_cast<MetricsReporter *>(cookie);
57*84e33947SAndroid Build Coastguard Worker         metricsReporter->onBinderDied();
58*84e33947SAndroid Build Coastguard Worker       }),
59*84e33947SAndroid Build Coastguard Worker       this);
60*84e33947SAndroid Build Coastguard Worker   if (status != STATUS_OK) {
61*84e33947SAndroid Build Coastguard Worker     LOGE("Failed to link to death the stats service binder");
62*84e33947SAndroid Build Coastguard Worker     return nullptr;
63*84e33947SAndroid Build Coastguard Worker   }
64*84e33947SAndroid Build Coastguard Worker 
65*84e33947SAndroid Build Coastguard Worker   std::shared_ptr<IStats> statsService = IStats::fromBinder(statsServiceBinder);
66*84e33947SAndroid Build Coastguard Worker   if (statsService == nullptr) {
67*84e33947SAndroid Build Coastguard Worker     LOGE("Failed to get IStats service");
68*84e33947SAndroid Build Coastguard Worker     return nullptr;
69*84e33947SAndroid Build Coastguard Worker   }
70*84e33947SAndroid Build Coastguard Worker   return statsService;
71*84e33947SAndroid Build Coastguard Worker }
72*84e33947SAndroid Build Coastguard Worker 
reportMetric(const VendorAtom & atom)73*84e33947SAndroid Build Coastguard Worker bool MetricsReporter::reportMetric(const VendorAtom &atom) {
74*84e33947SAndroid Build Coastguard Worker   ndk::ScopedAStatus ret;
75*84e33947SAndroid Build Coastguard Worker   {
76*84e33947SAndroid Build Coastguard Worker     std::lock_guard<std::mutex> lock(mStatsServiceMutex);
77*84e33947SAndroid Build Coastguard Worker     if (mStatsService == nullptr) {
78*84e33947SAndroid Build Coastguard Worker       mStatsService = getStatsService();
79*84e33947SAndroid Build Coastguard Worker       if (mStatsService == nullptr) {
80*84e33947SAndroid Build Coastguard Worker         return false;
81*84e33947SAndroid Build Coastguard Worker       }
82*84e33947SAndroid Build Coastguard Worker     }
83*84e33947SAndroid Build Coastguard Worker 
84*84e33947SAndroid Build Coastguard Worker     ret = mStatsService->reportVendorAtom(atom);
85*84e33947SAndroid Build Coastguard Worker   }
86*84e33947SAndroid Build Coastguard Worker 
87*84e33947SAndroid Build Coastguard Worker   if (!ret.isOk()) {
88*84e33947SAndroid Build Coastguard Worker     LOGE("Failed to report vendor atom with ID %" PRIi32, atom.atomId);
89*84e33947SAndroid Build Coastguard Worker   }
90*84e33947SAndroid Build Coastguard Worker   return ret.isOk();
91*84e33947SAndroid Build Coastguard Worker }
92*84e33947SAndroid Build Coastguard Worker 
logApWakeupOccurred(uint64_t nanoappId)93*84e33947SAndroid Build Coastguard Worker bool MetricsReporter::logApWakeupOccurred(uint64_t nanoappId) {
94*84e33947SAndroid Build Coastguard Worker   std::vector<VendorAtomValue> values(1);
95*84e33947SAndroid Build Coastguard Worker   values[0].set<VendorAtomValue::longValue>(nanoappId);
96*84e33947SAndroid Build Coastguard Worker 
97*84e33947SAndroid Build Coastguard Worker   const VendorAtom atom{
98*84e33947SAndroid Build Coastguard Worker       .atomId = CHRE_AP_WAKE_UP_OCCURRED,
99*84e33947SAndroid Build Coastguard Worker       .values{std::move(values)},
100*84e33947SAndroid Build Coastguard Worker   };
101*84e33947SAndroid Build Coastguard Worker 
102*84e33947SAndroid Build Coastguard Worker   return reportMetric(atom);
103*84e33947SAndroid Build Coastguard Worker }
104*84e33947SAndroid Build Coastguard Worker 
logNanoappLoadFailed(uint64_t nanoappId,ChreHalNanoappLoadFailed::Type type,ChreHalNanoappLoadFailed::Reason reason)105*84e33947SAndroid Build Coastguard Worker bool MetricsReporter::logNanoappLoadFailed(
106*84e33947SAndroid Build Coastguard Worker     uint64_t nanoappId, ChreHalNanoappLoadFailed::Type type,
107*84e33947SAndroid Build Coastguard Worker     ChreHalNanoappLoadFailed::Reason reason) {
108*84e33947SAndroid Build Coastguard Worker   std::vector<VendorAtomValue> values(3);
109*84e33947SAndroid Build Coastguard Worker   values[0].set<VendorAtomValue::longValue>(nanoappId);
110*84e33947SAndroid Build Coastguard Worker   values[1].set<VendorAtomValue::intValue>(type);
111*84e33947SAndroid Build Coastguard Worker   values[2].set<VendorAtomValue::intValue>(reason);
112*84e33947SAndroid Build Coastguard Worker 
113*84e33947SAndroid Build Coastguard Worker   const VendorAtom atom{
114*84e33947SAndroid Build Coastguard Worker       .atomId = CHRE_HAL_NANOAPP_LOAD_FAILED,
115*84e33947SAndroid Build Coastguard Worker       .values{std::move(values)},
116*84e33947SAndroid Build Coastguard Worker   };
117*84e33947SAndroid Build Coastguard Worker 
118*84e33947SAndroid Build Coastguard Worker   return reportMetric(atom);
119*84e33947SAndroid Build Coastguard Worker }
120*84e33947SAndroid Build Coastguard Worker 
logPalOpenFailed(ChrePalOpenFailed::ChrePalType pal,ChrePalOpenFailed::Type type)121*84e33947SAndroid Build Coastguard Worker bool MetricsReporter::logPalOpenFailed(ChrePalOpenFailed::ChrePalType pal,
122*84e33947SAndroid Build Coastguard Worker                                        ChrePalOpenFailed::Type type) {
123*84e33947SAndroid Build Coastguard Worker   std::vector<VendorAtomValue> values(2);
124*84e33947SAndroid Build Coastguard Worker   values[0].set<VendorAtomValue::intValue>(pal);
125*84e33947SAndroid Build Coastguard Worker   values[1].set<VendorAtomValue::intValue>(type);
126*84e33947SAndroid Build Coastguard Worker 
127*84e33947SAndroid Build Coastguard Worker   const VendorAtom atom{
128*84e33947SAndroid Build Coastguard Worker       .atomId = CHRE_PAL_OPEN_FAILED,
129*84e33947SAndroid Build Coastguard Worker       .values{std::move(values)},
130*84e33947SAndroid Build Coastguard Worker   };
131*84e33947SAndroid Build Coastguard Worker 
132*84e33947SAndroid Build Coastguard Worker   return reportMetric(atom);
133*84e33947SAndroid Build Coastguard Worker }
134*84e33947SAndroid Build Coastguard Worker 
logEventQueueSnapshotReported(int32_t snapshotChreGetTimeMs,int32_t maxEventQueueSize,int32_t meanEventQueueSize,int32_t numDroppedEvents)135*84e33947SAndroid Build Coastguard Worker bool MetricsReporter::logEventQueueSnapshotReported(
136*84e33947SAndroid Build Coastguard Worker     int32_t snapshotChreGetTimeMs, int32_t maxEventQueueSize,
137*84e33947SAndroid Build Coastguard Worker     int32_t meanEventQueueSize, int32_t numDroppedEvents) {
138*84e33947SAndroid Build Coastguard Worker   std::vector<VendorAtomValue> values(6);
139*84e33947SAndroid Build Coastguard Worker   values[0].set<VendorAtomValue::intValue>(snapshotChreGetTimeMs);
140*84e33947SAndroid Build Coastguard Worker   values[1].set<VendorAtomValue::intValue>(maxEventQueueSize);
141*84e33947SAndroid Build Coastguard Worker   values[2].set<VendorAtomValue::intValue>(meanEventQueueSize);
142*84e33947SAndroid Build Coastguard Worker   values[3].set<VendorAtomValue::intValue>(numDroppedEvents);
143*84e33947SAndroid Build Coastguard Worker 
144*84e33947SAndroid Build Coastguard Worker   // TODO(b/298459533): Implement these two values
145*84e33947SAndroid Build Coastguard Worker   // Last two values are not currently populated and will be implemented
146*84e33947SAndroid Build Coastguard Worker   // later. To avoid confusion of the interpretation, we use UINT32_MAX
147*84e33947SAndroid Build Coastguard Worker   // as a placeholder value.
148*84e33947SAndroid Build Coastguard Worker   values[4].set<VendorAtomValue::longValue>(
149*84e33947SAndroid Build Coastguard Worker       std::numeric_limits<int64_t>::max());
150*84e33947SAndroid Build Coastguard Worker   values[5].set<VendorAtomValue::longValue>(
151*84e33947SAndroid Build Coastguard Worker       std::numeric_limits<int64_t>::max());
152*84e33947SAndroid Build Coastguard Worker 
153*84e33947SAndroid Build Coastguard Worker   const VendorAtom atom{
154*84e33947SAndroid Build Coastguard Worker       .atomId = CHRE_EVENT_QUEUE_SNAPSHOT_REPORTED,
155*84e33947SAndroid Build Coastguard Worker       .values{std::move(values)},
156*84e33947SAndroid Build Coastguard Worker   };
157*84e33947SAndroid Build Coastguard Worker 
158*84e33947SAndroid Build Coastguard Worker   return reportMetric(atom);
159*84e33947SAndroid Build Coastguard Worker }
160*84e33947SAndroid Build Coastguard Worker 
onBinderDied()161*84e33947SAndroid Build Coastguard Worker void MetricsReporter::onBinderDied() {
162*84e33947SAndroid Build Coastguard Worker   LOGI("MetricsReporter: stats service died - reconnecting");
163*84e33947SAndroid Build Coastguard Worker 
164*84e33947SAndroid Build Coastguard Worker   std::lock_guard<std::mutex> lock(mStatsServiceMutex);
165*84e33947SAndroid Build Coastguard Worker   mStatsService.reset();
166*84e33947SAndroid Build Coastguard Worker   mStatsService = getStatsService();
167*84e33947SAndroid Build Coastguard Worker }
168*84e33947SAndroid Build Coastguard Worker 
169*84e33947SAndroid Build Coastguard Worker }  // namespace android::chre
170