xref: /aosp_15_r20/frameworks/base/native/android/dynamic_instrumentation_manager.cpp (revision d57664e9bc4670b3ecf6748a746a57c557b6bc9e)
1 /*
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"
18 #include <android-base/properties.h>
19 #include <android/dynamic_instrumentation_manager.h>
20 #include <android/os/instrumentation/BnOffsetCallback.h>
21 #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>
28 #include <utils/StrongPointer.h>
29 
30 #include <future>
31 #include <mutex>
32 #include <optional>
33 #include <string>
34 #include <vector>
35 
36 namespace android::dynamicinstrumentationmanager {
37 
38 using android::os::instrumentation::BnOffsetCallback;
39 using android::os::instrumentation::ExecutableMethodFileOffsets;
40 
41 // Global instance of IDynamicInstrumentationManager, service is obtained only on first use.
42 static std::mutex mLock;
43 static sp<os::instrumentation::IDynamicInstrumentationManager> mService;
44 
getService()45 sp<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 
57 using namespace android;
58 using namespace dynamicinstrumentationmanager;
59 
60 struct ADynamicInstrumentationManager_TargetProcess {
61     uid_t uid;
62     uid_t pid;
63     std::string processName;
64 
ADynamicInstrumentationManager_TargetProcessADynamicInstrumentationManager_TargetProcess65     ADynamicInstrumentationManager_TargetProcess(uid_t uid, pid_t pid, const char* processName)
66           : uid(uid), pid(pid), processName(processName) {}
67 };
68 
ADynamicInstrumentationManager_TargetProcess_create(uid_t uid,pid_t pid,const char * processName)69 ADynamicInstrumentationManager_TargetProcess* ADynamicInstrumentationManager_TargetProcess_create(
70         uid_t uid, pid_t pid, const char* processName) {
71     return new ADynamicInstrumentationManager_TargetProcess(uid, pid, processName);
72 }
73 
ADynamicInstrumentationManager_TargetProcess_destroy(const ADynamicInstrumentationManager_TargetProcess * instance)74 void ADynamicInstrumentationManager_TargetProcess_destroy(
75         const ADynamicInstrumentationManager_TargetProcess* instance) {
76     delete instance;
77 }
78 
79 struct ADynamicInstrumentationManager_MethodDescriptor {
80     std::string fqcn;
81     std::string methodName;
82     std::vector<std::string> fqParameters;
83 
ADynamicInstrumentationManager_MethodDescriptorADynamicInstrumentationManager_MethodDescriptor84     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 
95 ADynamicInstrumentationManager_MethodDescriptor*
ADynamicInstrumentationManager_MethodDescriptor_create(const char * fullyQualifiedClassName,const char * methodName,const char * fullyQualifiedParameters[],size_t numParameters)96 ADynamicInstrumentationManager_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 
ADynamicInstrumentationManager_MethodDescriptor_destroy(const ADynamicInstrumentationManager_MethodDescriptor * instance)105 void ADynamicInstrumentationManager_MethodDescriptor_destroy(
106         const ADynamicInstrumentationManager_MethodDescriptor* instance) {
107     delete instance;
108 }
109 
110 struct ADynamicInstrumentationManager_ExecutableMethodFileOffsets {
111     std::string containerPath;
112     uint64_t containerOffset;
113     uint64_t methodOffset;
114 };
115 
116 ADynamicInstrumentationManager_ExecutableMethodFileOffsets*
ADynamicInstrumentationManager_ExecutableMethodFileOffsets_create()117 ADynamicInstrumentationManager_ExecutableMethodFileOffsets_create() {
118     return new ADynamicInstrumentationManager_ExecutableMethodFileOffsets();
119 }
120 
ADynamicInstrumentationManager_ExecutableMethodFileOffsets_getContainerPath(const ADynamicInstrumentationManager_ExecutableMethodFileOffsets * instance)121 const char* ADynamicInstrumentationManager_ExecutableMethodFileOffsets_getContainerPath(
122         const ADynamicInstrumentationManager_ExecutableMethodFileOffsets* instance) {
123     return instance->containerPath.c_str();
124 }
125 
ADynamicInstrumentationManager_ExecutableMethodFileOffsets_getContainerOffset(const ADynamicInstrumentationManager_ExecutableMethodFileOffsets * instance)126 uint64_t ADynamicInstrumentationManager_ExecutableMethodFileOffsets_getContainerOffset(
127         const ADynamicInstrumentationManager_ExecutableMethodFileOffsets* instance) {
128     return instance->containerOffset;
129 }
130 
ADynamicInstrumentationManager_ExecutableMethodFileOffsets_getMethodOffset(const ADynamicInstrumentationManager_ExecutableMethodFileOffsets * instance)131 uint64_t ADynamicInstrumentationManager_ExecutableMethodFileOffsets_getMethodOffset(
132         const ADynamicInstrumentationManager_ExecutableMethodFileOffsets* instance) {
133     return instance->methodOffset;
134 }
135 
ADynamicInstrumentationManager_ExecutableMethodFileOffsets_destroy(const ADynamicInstrumentationManager_ExecutableMethodFileOffsets * instance)136 void ADynamicInstrumentationManager_ExecutableMethodFileOffsets_destroy(
137         const ADynamicInstrumentationManager_ExecutableMethodFileOffsets* instance) {
138     delete instance;
139 }
140 
141 class ResultCallback : public BnOffsetCallback {
142 public:
onResult(const::std::optional<ExecutableMethodFileOffsets> & offsets)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 
waitForResult()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 
161 private:
162     std::promise<std::optional<ExecutableMethodFileOffsets>> promise_;
163 };
164 
ADynamicInstrumentationManager_getExecutableMethodFileOffsets(const ADynamicInstrumentationManager_TargetProcess * targetProcess,const ADynamicInstrumentationManager_MethodDescriptor * methodDescriptor,const ADynamicInstrumentationManager_ExecutableMethodFileOffsets ** out)165 int32_t ADynamicInstrumentationManager_getExecutableMethodFileOffsets(
166         const ADynamicInstrumentationManager_TargetProcess* targetProcess,
167         const ADynamicInstrumentationManager_MethodDescriptor* methodDescriptor,
168         const ADynamicInstrumentationManager_ExecutableMethodFileOffsets** out) {
169     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 
184     android::sp<ResultCallback> resultCallback = android::sp<ResultCallback>::make();
185     binder_status_t result =
186             service->getExecutableMethodFileOffsets(targetProcessParcel, methodDescriptorParcel,
187                                                     resultCallback)
188                     .exceptionCode();
189     if (result != OK) {
190         return result;
191     }
192     std::optional<ExecutableMethodFileOffsets> offsets = resultCallback->waitForResult();
193     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;
204 }
205