blob: b11fceec1c47e64c558d5fb737c60cd0d82bbda0 [file] [log] [blame]
/*
* Copyright (C) 2021 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 "libstatspull_lazy.h"
#include <mutex>
#include <dlfcn.h>
#include <stdatomic.h>
#include "log/log.h"
#include "stats_pull_atom_callback.h"
// This file provides a lazy interface to libstatspull.so to address early boot dependencies.
// Specifically bootanimation, surfaceflinger, and lmkd run before the statsd APEX is loaded and
// libstatspull.so is in the statsd APEX.
// Method pointers to libstatspull methods are held in an array which simplifies checking
// all pointers are initialized.
enum MethodIndex {
// PullAtomMetadata APIs in stats_pull_atom_callback.h.
k_AStatsManager_PullAtomMetadata_obtain,
k_AStatsManager_PullAtomMetadata_release,
k_AStatsManager_PullAtomMetadata_setCoolDownMillis,
k_AStatsManager_PullAtomMetadata_getCoolDownMillis,
k_AStatsManager_PullAtomMetadata_setTimeoutMillis,
k_AStatsManager_PullAtomMetadata_getTimeoutMillis,
k_AStatsManager_PullAtomMetadata_setAdditiveFields,
k_AStatsManager_PullAtomMetadata_getNumAdditiveFields,
k_AStatsManager_PullAtomMetadata_getAdditiveFields,
// AStatsEventList APIs in stats_pull_atom_callback.h
k_AStatsEventList_addStatsEvent,
// PullAtomCallback APIs in stats_pull_atom_callback.h
k_AStatsManager_setPullAtomCallback,
k_AStatsManager_clearPullAtomCallback,
// Marker for count of methods
k_MethodCount
};
// Table of methods pointers in libstatspull APIs.
static void* g_Methods[k_MethodCount];
//
// Libstatspull lazy loading.
//
static atomic_bool gPreventLibstatspullLoading = false; // Allows tests to block loading.
void PreventLibstatspullLazyLoadingForTests() {
gPreventLibstatspullLoading.store(true);
}
static void* LoadLibstatspull(int dlopen_flags) {
if (gPreventLibstatspullLoading.load()) {
return nullptr;
}
return dlopen("libstatspull.so", dlopen_flags);
}
//
// Initialization and symbol binding.
static void BindSymbol(void* handle, const char* name, enum MethodIndex index) {
void* symbol = dlsym(handle, name);
LOG_ALWAYS_FATAL_IF(symbol == nullptr, "Failed to find symbol '%s' in libstatspull.so: %s",
name, dlerror());
g_Methods[index] = symbol;
}
static void InitializeOnce() {
void* handle = LoadLibstatspull(RTLD_NOW);
LOG_ALWAYS_FATAL_IF(handle == nullptr, "Failed to load libstatspull.so: %s", dlerror());
#undef BIND_SYMBOL
#define BIND_SYMBOL(name) BindSymbol(handle, #name, k_##name);
// PullAtomMetadata APIs in stats_pull_atom_callback.h.
BIND_SYMBOL(AStatsManager_PullAtomMetadata_obtain);
BIND_SYMBOL(AStatsManager_PullAtomMetadata_release);
BIND_SYMBOL(AStatsManager_PullAtomMetadata_setCoolDownMillis);
BIND_SYMBOL(AStatsManager_PullAtomMetadata_getCoolDownMillis);
BIND_SYMBOL(AStatsManager_PullAtomMetadata_setTimeoutMillis);
BIND_SYMBOL(AStatsManager_PullAtomMetadata_getTimeoutMillis);
BIND_SYMBOL(AStatsManager_PullAtomMetadata_setAdditiveFields);
BIND_SYMBOL(AStatsManager_PullAtomMetadata_getNumAdditiveFields);
BIND_SYMBOL(AStatsManager_PullAtomMetadata_getAdditiveFields);
// AStatsEventList APIs in stats_pull_atom_callback.h
BIND_SYMBOL(AStatsEventList_addStatsEvent);
// PullAtomCallback APIs in stats_pull_atom_callback.h
BIND_SYMBOL(AStatsManager_setPullAtomCallback);
BIND_SYMBOL(AStatsManager_clearPullAtomCallback);
#undef BIND_SYMBOL
// Check every symbol is bound.
for (int i = 0; i < k_MethodCount; ++i) {
LOG_ALWAYS_FATAL_IF(g_Methods[i] == nullptr,
"Uninitialized method in libstatspull_lazy at index: %d", i);
}
}
static void EnsureInitialized() {
static std::once_flag initialize_flag;
std::call_once(initialize_flag, InitializeOnce);
}
#define INVOKE_METHOD(name, args...) \
do { \
EnsureInitialized(); \
void* method = g_Methods[k_##name]; \
return reinterpret_cast<decltype(&name)>(method)(args); \
} while (0)
//
// Forwarding for methods in stats_pull_atom_callback.h.
//
AStatsManager_PullAtomMetadata* AStatsManager_PullAtomMetadata_obtain() {
INVOKE_METHOD(AStatsManager_PullAtomMetadata_obtain);
}
void AStatsManager_PullAtomMetadata_release(AStatsManager_PullAtomMetadata* metadata) {
INVOKE_METHOD(AStatsManager_PullAtomMetadata_release, metadata);
}
void AStatsManager_PullAtomMetadata_setCoolDownMillis(AStatsManager_PullAtomMetadata* metadata,
int64_t cool_down_millis) {
INVOKE_METHOD(AStatsManager_PullAtomMetadata_setCoolDownMillis, metadata, cool_down_millis);
}
int64_t AStatsManager_PullAtomMetadata_getCoolDownMillis(AStatsManager_PullAtomMetadata* metadata) {
INVOKE_METHOD(AStatsManager_PullAtomMetadata_getCoolDownMillis, metadata);
}
void AStatsManager_PullAtomMetadata_setTimeoutMillis(AStatsManager_PullAtomMetadata* metadata,
int64_t timeout_millis) {
INVOKE_METHOD(AStatsManager_PullAtomMetadata_setTimeoutMillis, metadata, timeout_millis);
}
int64_t AStatsManager_PullAtomMetadata_getTimeoutMillis(AStatsManager_PullAtomMetadata* metadata) {
INVOKE_METHOD(AStatsManager_PullAtomMetadata_getTimeoutMillis, metadata);
}
void AStatsManager_PullAtomMetadata_setAdditiveFields(AStatsManager_PullAtomMetadata* metadata,
int32_t* additive_fields,
int32_t num_fields) {
INVOKE_METHOD(AStatsManager_PullAtomMetadata_setAdditiveFields, metadata, additive_fields,
num_fields);
}
int32_t AStatsManager_PullAtomMetadata_getNumAdditiveFields(
AStatsManager_PullAtomMetadata* metadata) {
INVOKE_METHOD(AStatsManager_PullAtomMetadata_getNumAdditiveFields, metadata);
}
void AStatsManager_PullAtomMetadata_getAdditiveFields(AStatsManager_PullAtomMetadata* metadata,
int32_t* fields) {
INVOKE_METHOD(AStatsManager_PullAtomMetadata_getAdditiveFields, metadata, fields);
}
AStatsEvent* AStatsEventList_addStatsEvent(AStatsEventList* pull_data) {
INVOKE_METHOD(AStatsEventList_addStatsEvent, pull_data);
}
void AStatsManager_setPullAtomCallback(int32_t atom_tag, AStatsManager_PullAtomMetadata* metadata,
AStatsManager_PullAtomCallback callback, void* cookie) {
INVOKE_METHOD(AStatsManager_setPullAtomCallback, atom_tag, metadata, callback, cookie);
}
void AStatsManager_clearPullAtomCallback(int32_t atom_tag) {
INVOKE_METHOD(AStatsManager_clearPullAtomCallback, atom_tag);
}