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