blob: 074973188c6659dc36a500b66c51d1da064984cb [file] [log] [blame]
Matt Gilbrideae83ac22024-09-24 22:48:38 +00001/*
2 * Copyright (C) 2024 The Android Open Source Project
3 *
4 * Licensed under the Apache License, Version 2.0 (the "License");
5 * you may not use this file except in compliance with the License.
6 * You may obtain a copy of the License at
7 *
8 * http://www.apache.org/licenses/LICENSE-2.0
9 *
10 * Unless required by applicable law or agreed to in writing, software
11 * distributed under the License is distributed on an "AS IS" BASIS,
12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13 * See the License for the specific language governing permissions and
14 * limitations under the License.
15 */
16
17#define LOG_TAG "ADynamicInstrumentationManager"
Yu-Ting Tsengd857b662024-11-08 12:42:45 -080018#include <android-base/properties.h>
Matt Gilbrideae83ac22024-09-24 22:48:38 +000019#include <android/dynamic_instrumentation_manager.h>
Yu-Ting Tsengd857b662024-11-08 12:42:45 -080020#include <android/os/instrumentation/BnOffsetCallback.h>
Matt Gilbrideae83ac22024-09-24 22:48:38 +000021#include <android/os/instrumentation/ExecutableMethodFileOffsets.h>
22#include <android/os/instrumentation/IDynamicInstrumentationManager.h>
23#include <android/os/instrumentation/MethodDescriptor.h>
24#include <android/os/instrumentation/TargetProcess.h>
25#include <binder/Binder.h>
26#include <binder/IServiceManager.h>
27#include <utils/Log.h>
Yu-Ting Tsengd857b662024-11-08 12:42:45 -080028#include <utils/StrongPointer.h>
Matt Gilbrideae83ac22024-09-24 22:48:38 +000029
Yu-Ting Tsengd857b662024-11-08 12:42:45 -080030#include <future>
Matt Gilbrideae83ac22024-09-24 22:48:38 +000031#include <mutex>
32#include <optional>
33#include <string>
34#include <vector>
35
36namespace android::dynamicinstrumentationmanager {
37
Yu-Ting Tsengd857b662024-11-08 12:42:45 -080038using android::os::instrumentation::BnOffsetCallback;
39using android::os::instrumentation::ExecutableMethodFileOffsets;
40
Matt Gilbrideae83ac22024-09-24 22:48:38 +000041// Global instance of IDynamicInstrumentationManager, service is obtained only on first use.
42static std::mutex mLock;
43static sp<os::instrumentation::IDynamicInstrumentationManager> mService;
44
45sp<os::instrumentation::IDynamicInstrumentationManager> getService() {
46 std::lock_guard<std::mutex> scoped_lock(mLock);
47 if (mService == nullptr || !IInterface::asBinder(mService)->isBinderAlive()) {
48 sp<IBinder> binder =
49 defaultServiceManager()->waitForService(String16("dynamic_instrumentation"));
50 mService = interface_cast<os::instrumentation::IDynamicInstrumentationManager>(binder);
51 }
52 return mService;
53}
54
55} // namespace android::dynamicinstrumentationmanager
56
57using namespace android;
58using namespace dynamicinstrumentationmanager;
59
60struct ADynamicInstrumentationManager_TargetProcess {
61 uid_t uid;
62 uid_t pid;
63 std::string processName;
64
65 ADynamicInstrumentationManager_TargetProcess(uid_t uid, pid_t pid, const char* processName)
66 : uid(uid), pid(pid), processName(processName) {}
67};
68
69ADynamicInstrumentationManager_TargetProcess* ADynamicInstrumentationManager_TargetProcess_create(
70 uid_t uid, pid_t pid, const char* processName) {
71 return new ADynamicInstrumentationManager_TargetProcess(uid, pid, processName);
72}
73
74void ADynamicInstrumentationManager_TargetProcess_destroy(
Matt Gilbride49caa7d2024-11-18 17:40:37 +000075 const ADynamicInstrumentationManager_TargetProcess* instance) {
Matt Gilbrideae83ac22024-09-24 22:48:38 +000076 delete instance;
77}
78
79struct ADynamicInstrumentationManager_MethodDescriptor {
80 std::string fqcn;
81 std::string methodName;
82 std::vector<std::string> fqParameters;
83
84 ADynamicInstrumentationManager_MethodDescriptor(const char* fqcn, const char* methodName,
85 const char* fullyQualifiedParameters[],
86 size_t numParameters)
87 : fqcn(fqcn), methodName(methodName) {
88 std::vector<std::string> fqParameters;
89 fqParameters.reserve(numParameters);
90 std::copy_n(fullyQualifiedParameters, numParameters, std::back_inserter(fqParameters));
91 this->fqParameters = std::move(fqParameters);
92 }
93};
94
95ADynamicInstrumentationManager_MethodDescriptor*
96ADynamicInstrumentationManager_MethodDescriptor_create(const char* fullyQualifiedClassName,
97 const char* methodName,
98 const char* fullyQualifiedParameters[],
99 size_t numParameters) {
100 return new ADynamicInstrumentationManager_MethodDescriptor(fullyQualifiedClassName, methodName,
101 fullyQualifiedParameters,
102 numParameters);
103}
104
105void ADynamicInstrumentationManager_MethodDescriptor_destroy(
Matt Gilbride49caa7d2024-11-18 17:40:37 +0000106 const ADynamicInstrumentationManager_MethodDescriptor* instance) {
Matt Gilbrideae83ac22024-09-24 22:48:38 +0000107 delete instance;
108}
109
110struct ADynamicInstrumentationManager_ExecutableMethodFileOffsets {
111 std::string containerPath;
112 uint64_t containerOffset;
113 uint64_t methodOffset;
114};
115
116ADynamicInstrumentationManager_ExecutableMethodFileOffsets*
117ADynamicInstrumentationManager_ExecutableMethodFileOffsets_create() {
118 return new ADynamicInstrumentationManager_ExecutableMethodFileOffsets();
119}
120
121const char* ADynamicInstrumentationManager_ExecutableMethodFileOffsets_getContainerPath(
Matt Gilbride49caa7d2024-11-18 17:40:37 +0000122 const ADynamicInstrumentationManager_ExecutableMethodFileOffsets* instance) {
Matt Gilbrideae83ac22024-09-24 22:48:38 +0000123 return instance->containerPath.c_str();
124}
125
126uint64_t ADynamicInstrumentationManager_ExecutableMethodFileOffsets_getContainerOffset(
Matt Gilbride49caa7d2024-11-18 17:40:37 +0000127 const ADynamicInstrumentationManager_ExecutableMethodFileOffsets* instance) {
Matt Gilbrideae83ac22024-09-24 22:48:38 +0000128 return instance->containerOffset;
129}
130
131uint64_t ADynamicInstrumentationManager_ExecutableMethodFileOffsets_getMethodOffset(
Matt Gilbride49caa7d2024-11-18 17:40:37 +0000132 const ADynamicInstrumentationManager_ExecutableMethodFileOffsets* instance) {
Matt Gilbrideae83ac22024-09-24 22:48:38 +0000133 return instance->methodOffset;
134}
135
136void ADynamicInstrumentationManager_ExecutableMethodFileOffsets_destroy(
Matt Gilbride49caa7d2024-11-18 17:40:37 +0000137 const ADynamicInstrumentationManager_ExecutableMethodFileOffsets* instance) {
Matt Gilbrideae83ac22024-09-24 22:48:38 +0000138 delete instance;
139}
140
Yu-Ting Tsengd857b662024-11-08 12:42:45 -0800141class ResultCallback : public BnOffsetCallback {
142public:
143 ::android::binder::Status onResult(
144 const ::std::optional<ExecutableMethodFileOffsets>& offsets) override {
145 promise_.set_value(offsets);
146 return android::binder::Status::ok();
147 }
148
149 std::optional<ExecutableMethodFileOffsets> waitForResult() {
150 std::future<std::optional<ExecutableMethodFileOffsets>> futureResult =
151 promise_.get_future();
152 auto futureStatus = futureResult.wait_for(
153 std::chrono::seconds(1 * android::base::HwTimeoutMultiplier()));
154 if (futureStatus == std::future_status::ready) {
155 return futureResult.get();
156 } else {
157 return std::nullopt;
158 }
159 }
160
161private:
162 std::promise<std::optional<ExecutableMethodFileOffsets>> promise_;
163};
164
Matt Gilbrideae83ac22024-09-24 22:48:38 +0000165int32_t ADynamicInstrumentationManager_getExecutableMethodFileOffsets(
166 const ADynamicInstrumentationManager_TargetProcess* targetProcess,
167 const ADynamicInstrumentationManager_MethodDescriptor* methodDescriptor,
Matt Gilbride49caa7d2024-11-18 17:40:37 +0000168 const ADynamicInstrumentationManager_ExecutableMethodFileOffsets** out) {
Matt Gilbrideae83ac22024-09-24 22:48:38 +0000169 android::os::instrumentation::TargetProcess targetProcessParcel;
170 targetProcessParcel.uid = targetProcess->uid;
171 targetProcessParcel.pid = targetProcess->pid;
172 targetProcessParcel.processName = targetProcess->processName;
173
174 android::os::instrumentation::MethodDescriptor methodDescriptorParcel;
175 methodDescriptorParcel.fullyQualifiedClassName = methodDescriptor->fqcn;
176 methodDescriptorParcel.methodName = methodDescriptor->methodName;
177 methodDescriptorParcel.fullyQualifiedParameters = methodDescriptor->fqParameters;
178
179 sp<os::instrumentation::IDynamicInstrumentationManager> service = getService();
180 if (service == nullptr) {
181 return INVALID_OPERATION;
182 }
183
Yu-Ting Tsengd857b662024-11-08 12:42:45 -0800184 android::sp<ResultCallback> resultCallback = android::sp<ResultCallback>::make();
Matt Gilbrideae83ac22024-09-24 22:48:38 +0000185 binder_status_t result =
186 service->getExecutableMethodFileOffsets(targetProcessParcel, methodDescriptorParcel,
Yu-Ting Tsengd857b662024-11-08 12:42:45 -0800187 resultCallback)
Matt Gilbrideae83ac22024-09-24 22:48:38 +0000188 .exceptionCode();
189 if (result != OK) {
190 return result;
191 }
Yu-Ting Tsengd857b662024-11-08 12:42:45 -0800192 std::optional<ExecutableMethodFileOffsets> offsets = resultCallback->waitForResult();
Matt Gilbrideae83ac22024-09-24 22:48:38 +0000193 if (offsets != std::nullopt) {
194 auto* value = new ADynamicInstrumentationManager_ExecutableMethodFileOffsets();
195 value->containerPath = offsets->containerPath;
196 value->containerOffset = offsets->containerOffset;
197 value->methodOffset = offsets->methodOffset;
198 *out = value;
199 } else {
200 *out = nullptr;
201 }
202
203 return result;
Yu-Ting Tsengd857b662024-11-08 12:42:45 -0800204}