/* * Copyright (C) 2020 The Android Open Source Project * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ #include #include #include "LongArrayMultiStateCounter.h" #include "core_jni_helpers.h" namespace android { static constexpr uint64_t NSEC_PER_MSEC = 1000000; static jlongArray copyVecsToArray(JNIEnv *env, std::vector> &vec) { jsize s = 0; for (const auto &subVec : vec) s += subVec.size(); jlongArray ar = env->NewLongArray(s); jsize start = 0; for (auto &subVec : vec) { for (uint32_t i = 0; i < subVec.size(); ++i) subVec[i] /= NSEC_PER_MSEC; env->SetLongArrayRegion(ar, start, subVec.size(), reinterpret_cast(subVec.data())); start += subVec.size(); } return ar; } static jlongArray getUidCpuFreqTimeMs(JNIEnv *env, jclass, jint uid) { auto out = android::bpf::getUidCpuFreqTimes(uid); if (!out) return env->NewLongArray(0); return copyVecsToArray(env, out.value()); } /** * Computes delta of CPU time-in-freq from the previously supplied counts and adds the delta * to the supplied multi-state counter in accordance with the counter's state. */ static jboolean addCpuTimeInFreqDelta( jint uid, jlong counterNativePtr, jlong timestampMs, std::optional>> timeInFreqDataNanos, jlong deltaOutContainerNativePtr) { if (!timeInFreqDataNanos) { return false; } battery::LongArrayMultiStateCounter *counter = reinterpret_cast(counterNativePtr); size_t s = 0; for (const auto &cluster : *timeInFreqDataNanos) s += cluster.size(); std::vector flattened; flattened.reserve(s); auto offset = flattened.begin(); for (const auto &cluster : *timeInFreqDataNanos) { flattened.insert(offset, cluster.begin(), cluster.end()); offset += cluster.size(); } for (size_t i = 0; i < s; ++i) { flattened[i] /= NSEC_PER_MSEC; } if (s != counter->getCount(0).size()) { // Counter has at least one state ALOGE("Mismatch between eBPF data size (%d) and the counter size (%d)", (int)s, (int)counter->getCount(0).size()); return false; } const std::vector &delta = counter->updateValue(flattened, timestampMs); if (deltaOutContainerNativePtr) { std::vector *vector = reinterpret_cast *>(deltaOutContainerNativePtr); *vector = delta; } return true; } static jboolean addDeltaFromBpf(jint uid, jlong counterNativePtr, jlong timestampMs, jlong deltaOutContainerNativePtr) { return addCpuTimeInFreqDelta(uid, counterNativePtr, timestampMs, android::bpf::getUidCpuFreqTimes(uid), deltaOutContainerNativePtr); } static jboolean addDeltaForTest(JNIEnv *env, jclass, jint uid, jlong counterNativePtr, jlong timestampMs, jobjectArray timeInFreqDataNanos, jlong deltaOutContainerNativePtr) { if (!timeInFreqDataNanos) { return addCpuTimeInFreqDelta(uid, counterNativePtr, timestampMs, std::optional>>(), deltaOutContainerNativePtr); } std::vector> timeInFreqData; jsize len = env->GetArrayLength(timeInFreqDataNanos); for (jsize i = 0; i < len; i++) { std::vector cluster; ScopedLongArrayRO row(env, (jlongArray)env->GetObjectArrayElement(timeInFreqDataNanos, i)); cluster.reserve(row.size()); for (size_t j = 0; j < row.size(); j++) { cluster.push_back(row[j]); } timeInFreqData.push_back(cluster); } return addCpuTimeInFreqDelta(uid, counterNativePtr, timestampMs, std::optional(timeInFreqData), deltaOutContainerNativePtr); } static const JNINativeMethod g_single_methods[] = { {"readBpfData", "(I)[J", (void *)getUidCpuFreqTimeMs}, // @CriticalNative {"addDeltaFromBpf", "(IJJJ)Z", (void *)addDeltaFromBpf}, // Used for testing {"addDeltaForTest", "(IJJ[[JJ)Z", (void *)addDeltaForTest}, }; int register_com_android_internal_os_KernelSingleUidTimeReader(JNIEnv *env) { return RegisterMethodsOrDie(env, "com/android/internal/os/KernelSingleUidTimeReader$Injector", g_single_methods, NELEM(g_single_methods)); } }