blob: 532213611cf1406aa640fcb398b5319d25c63654 [file] [log] [blame]
/*
* Copyright (C) 2024 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.
*/
#define LOG_TAG "ADynamicInstrumentationManager"
#include <android/dynamic_instrumentation_manager.h>
#include <android/os/instrumentation/ExecutableMethodFileOffsets.h>
#include <android/os/instrumentation/IDynamicInstrumentationManager.h>
#include <android/os/instrumentation/MethodDescriptor.h>
#include <android/os/instrumentation/TargetProcess.h>
#include <binder/Binder.h>
#include <binder/IServiceManager.h>
#include <utils/Log.h>
#include <mutex>
#include <optional>
#include <string>
#include <vector>
namespace android::dynamicinstrumentationmanager {
// Global instance of IDynamicInstrumentationManager, service is obtained only on first use.
static std::mutex mLock;
static sp<os::instrumentation::IDynamicInstrumentationManager> mService;
sp<os::instrumentation::IDynamicInstrumentationManager> getService() {
std::lock_guard<std::mutex> scoped_lock(mLock);
if (mService == nullptr || !IInterface::asBinder(mService)->isBinderAlive()) {
sp<IBinder> binder =
defaultServiceManager()->waitForService(String16("dynamic_instrumentation"));
mService = interface_cast<os::instrumentation::IDynamicInstrumentationManager>(binder);
}
return mService;
}
} // namespace android::dynamicinstrumentationmanager
using namespace android;
using namespace dynamicinstrumentationmanager;
struct ADynamicInstrumentationManager_TargetProcess {
uid_t uid;
uid_t pid;
std::string processName;
ADynamicInstrumentationManager_TargetProcess(uid_t uid, pid_t pid, const char* processName)
: uid(uid), pid(pid), processName(processName) {}
};
ADynamicInstrumentationManager_TargetProcess* ADynamicInstrumentationManager_TargetProcess_create(
uid_t uid, pid_t pid, const char* processName) {
return new ADynamicInstrumentationManager_TargetProcess(uid, pid, processName);
}
void ADynamicInstrumentationManager_TargetProcess_destroy(
const ADynamicInstrumentationManager_TargetProcess* instance) {
delete instance;
}
struct ADynamicInstrumentationManager_MethodDescriptor {
std::string fqcn;
std::string methodName;
std::vector<std::string> fqParameters;
ADynamicInstrumentationManager_MethodDescriptor(const char* fqcn, const char* methodName,
const char* fullyQualifiedParameters[],
size_t numParameters)
: fqcn(fqcn), methodName(methodName) {
std::vector<std::string> fqParameters;
fqParameters.reserve(numParameters);
std::copy_n(fullyQualifiedParameters, numParameters, std::back_inserter(fqParameters));
this->fqParameters = std::move(fqParameters);
}
};
ADynamicInstrumentationManager_MethodDescriptor*
ADynamicInstrumentationManager_MethodDescriptor_create(const char* fullyQualifiedClassName,
const char* methodName,
const char* fullyQualifiedParameters[],
size_t numParameters) {
return new ADynamicInstrumentationManager_MethodDescriptor(fullyQualifiedClassName, methodName,
fullyQualifiedParameters,
numParameters);
}
void ADynamicInstrumentationManager_MethodDescriptor_destroy(
const ADynamicInstrumentationManager_MethodDescriptor* instance) {
delete instance;
}
struct ADynamicInstrumentationManager_ExecutableMethodFileOffsets {
std::string containerPath;
uint64_t containerOffset;
uint64_t methodOffset;
};
ADynamicInstrumentationManager_ExecutableMethodFileOffsets*
ADynamicInstrumentationManager_ExecutableMethodFileOffsets_create() {
return new ADynamicInstrumentationManager_ExecutableMethodFileOffsets();
}
const char* ADynamicInstrumentationManager_ExecutableMethodFileOffsets_getContainerPath(
const ADynamicInstrumentationManager_ExecutableMethodFileOffsets* instance) {
return instance->containerPath.c_str();
}
uint64_t ADynamicInstrumentationManager_ExecutableMethodFileOffsets_getContainerOffset(
const ADynamicInstrumentationManager_ExecutableMethodFileOffsets* instance) {
return instance->containerOffset;
}
uint64_t ADynamicInstrumentationManager_ExecutableMethodFileOffsets_getMethodOffset(
const ADynamicInstrumentationManager_ExecutableMethodFileOffsets* instance) {
return instance->methodOffset;
}
void ADynamicInstrumentationManager_ExecutableMethodFileOffsets_destroy(
const ADynamicInstrumentationManager_ExecutableMethodFileOffsets* instance) {
delete instance;
}
int32_t ADynamicInstrumentationManager_getExecutableMethodFileOffsets(
const ADynamicInstrumentationManager_TargetProcess* targetProcess,
const ADynamicInstrumentationManager_MethodDescriptor* methodDescriptor,
const ADynamicInstrumentationManager_ExecutableMethodFileOffsets** out) {
android::os::instrumentation::TargetProcess targetProcessParcel;
targetProcessParcel.uid = targetProcess->uid;
targetProcessParcel.pid = targetProcess->pid;
targetProcessParcel.processName = targetProcess->processName;
android::os::instrumentation::MethodDescriptor methodDescriptorParcel;
methodDescriptorParcel.fullyQualifiedClassName = methodDescriptor->fqcn;
methodDescriptorParcel.methodName = methodDescriptor->methodName;
methodDescriptorParcel.fullyQualifiedParameters = methodDescriptor->fqParameters;
sp<os::instrumentation::IDynamicInstrumentationManager> service = getService();
if (service == nullptr) {
return INVALID_OPERATION;
}
std::optional<android::os::instrumentation::ExecutableMethodFileOffsets> offsets;
binder_status_t result =
service->getExecutableMethodFileOffsets(targetProcessParcel, methodDescriptorParcel,
&offsets)
.exceptionCode();
if (result != OK) {
return result;
}
if (offsets != std::nullopt) {
auto* value = new ADynamicInstrumentationManager_ExecutableMethodFileOffsets();
value->containerPath = offsets->containerPath;
value->containerOffset = offsets->containerOffset;
value->methodOffset = offsets->methodOffset;
*out = value;
} else {
*out = nullptr;
}
return result;
}