xref: /aosp_15_r20/art/dex2oat/sdk_checker.cc (revision 795d594fd825385562da6b089ea9b2033f3abf5a)
1*795d594fSAndroid Build Coastguard Worker /*
2*795d594fSAndroid Build Coastguard Worker  * Copyright (C) 2020 The Android Open Source Project
3*795d594fSAndroid Build Coastguard Worker  *
4*795d594fSAndroid Build Coastguard Worker  * Licensed under the Apache License, Version 2.0 (the "License");
5*795d594fSAndroid Build Coastguard Worker  * you may not use this file except in compliance with the License.
6*795d594fSAndroid Build Coastguard Worker  * You may obtain a copy of the License at
7*795d594fSAndroid Build Coastguard Worker  *
8*795d594fSAndroid Build Coastguard Worker  *      http://www.apache.org/licenses/LICENSE-2.0
9*795d594fSAndroid Build Coastguard Worker  *
10*795d594fSAndroid Build Coastguard Worker  * Unless required by applicable law or agreed to in writing, software
11*795d594fSAndroid Build Coastguard Worker  * distributed under the License is distributed on an "AS IS" BASIS,
12*795d594fSAndroid Build Coastguard Worker  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13*795d594fSAndroid Build Coastguard Worker  * See the License for the specific language governing permissions and
14*795d594fSAndroid Build Coastguard Worker  * limitations under the License.
15*795d594fSAndroid Build Coastguard Worker  */
16*795d594fSAndroid Build Coastguard Worker 
17*795d594fSAndroid Build Coastguard Worker #include "sdk_checker.h"
18*795d594fSAndroid Build Coastguard Worker 
19*795d594fSAndroid Build Coastguard Worker #include "art_method-inl.h"
20*795d594fSAndroid Build Coastguard Worker #include "base/utils.h"
21*795d594fSAndroid Build Coastguard Worker #include "dex/art_dex_file_loader.h"
22*795d594fSAndroid Build Coastguard Worker #include "mirror/class-inl.h"
23*795d594fSAndroid Build Coastguard Worker 
24*795d594fSAndroid Build Coastguard Worker namespace art HIDDEN {
25*795d594fSAndroid Build Coastguard Worker 
SdkChecker()26*795d594fSAndroid Build Coastguard Worker SdkChecker::SdkChecker() : enabled_(true) {}
27*795d594fSAndroid Build Coastguard Worker 
Create(const std::string & public_sdk,std::string * error_msg)28*795d594fSAndroid Build Coastguard Worker SdkChecker* SdkChecker::Create(const std::string& public_sdk, std::string* error_msg) {
29*795d594fSAndroid Build Coastguard Worker   std::vector<std::string> dex_file_paths;
30*795d594fSAndroid Build Coastguard Worker   Split(public_sdk, ':', &dex_file_paths);
31*795d594fSAndroid Build Coastguard Worker 
32*795d594fSAndroid Build Coastguard Worker   std::unique_ptr<SdkChecker> sdk_checker(new SdkChecker());
33*795d594fSAndroid Build Coastguard Worker   for (const std::string& path : dex_file_paths) {
34*795d594fSAndroid Build Coastguard Worker     DexFileLoader dex_file_loader(path);
35*795d594fSAndroid Build Coastguard Worker     if (!dex_file_loader.Open(/*verify=*/true,
36*795d594fSAndroid Build Coastguard Worker                               /*verify_checksum*/ false,
37*795d594fSAndroid Build Coastguard Worker                               error_msg,
38*795d594fSAndroid Build Coastguard Worker                               &sdk_checker->sdk_dex_files_)) {
39*795d594fSAndroid Build Coastguard Worker       return nullptr;
40*795d594fSAndroid Build Coastguard Worker     }
41*795d594fSAndroid Build Coastguard Worker   }
42*795d594fSAndroid Build Coastguard Worker   return sdk_checker.release();
43*795d594fSAndroid Build Coastguard Worker }
44*795d594fSAndroid Build Coastguard Worker 
ShouldDenyAccess(ArtMethod * art_method) const45*795d594fSAndroid Build Coastguard Worker bool SdkChecker::ShouldDenyAccess(ArtMethod* art_method) const {
46*795d594fSAndroid Build Coastguard Worker   if (!enabled_) {
47*795d594fSAndroid Build Coastguard Worker     return false;
48*795d594fSAndroid Build Coastguard Worker   }
49*795d594fSAndroid Build Coastguard Worker 
50*795d594fSAndroid Build Coastguard Worker   std::string_view declaring_class_descriptor = art_method->GetDeclaringClassDescriptorView();
51*795d594fSAndroid Build Coastguard Worker   const char* name = art_method->GetName();
52*795d594fSAndroid Build Coastguard Worker 
53*795d594fSAndroid Build Coastguard Worker   bool found = false;
54*795d594fSAndroid Build Coastguard Worker   for (const std::unique_ptr<const DexFile>& dex_file : sdk_dex_files_) {
55*795d594fSAndroid Build Coastguard Worker     const dex::TypeId* declaring_type_id = dex_file->FindTypeId(declaring_class_descriptor);
56*795d594fSAndroid Build Coastguard Worker     if (declaring_type_id == nullptr) {
57*795d594fSAndroid Build Coastguard Worker       continue;
58*795d594fSAndroid Build Coastguard Worker     }
59*795d594fSAndroid Build Coastguard Worker     const dex::StringId* name_id = dex_file->FindStringId(name);
60*795d594fSAndroid Build Coastguard Worker     if (name_id == nullptr) {
61*795d594fSAndroid Build Coastguard Worker       continue;
62*795d594fSAndroid Build Coastguard Worker     }
63*795d594fSAndroid Build Coastguard Worker 
64*795d594fSAndroid Build Coastguard Worker     dex::TypeIndex return_type_idx;
65*795d594fSAndroid Build Coastguard Worker     std::vector<dex::TypeIndex> param_type_idxs;
66*795d594fSAndroid Build Coastguard Worker     if (!dex_file->CreateTypeList(
67*795d594fSAndroid Build Coastguard Worker             art_method->GetSignature().ToString(), &return_type_idx, &param_type_idxs)) {
68*795d594fSAndroid Build Coastguard Worker       continue;
69*795d594fSAndroid Build Coastguard Worker     }
70*795d594fSAndroid Build Coastguard Worker     const dex::ProtoId* proto_id = dex_file->FindProtoId(return_type_idx, param_type_idxs);
71*795d594fSAndroid Build Coastguard Worker     if (proto_id == nullptr) {
72*795d594fSAndroid Build Coastguard Worker       continue;
73*795d594fSAndroid Build Coastguard Worker     }
74*795d594fSAndroid Build Coastguard Worker 
75*795d594fSAndroid Build Coastguard Worker     const dex::MethodId* method_id =
76*795d594fSAndroid Build Coastguard Worker         dex_file->FindMethodId(*declaring_type_id, *name_id, *proto_id);
77*795d594fSAndroid Build Coastguard Worker     if (method_id != nullptr) {
78*795d594fSAndroid Build Coastguard Worker       found = true;
79*795d594fSAndroid Build Coastguard Worker       break;
80*795d594fSAndroid Build Coastguard Worker     }
81*795d594fSAndroid Build Coastguard Worker   }
82*795d594fSAndroid Build Coastguard Worker 
83*795d594fSAndroid Build Coastguard Worker   if (!found) {
84*795d594fSAndroid Build Coastguard Worker     VLOG(verifier) << "Deny for " << art_method->PrettyMethod(true);
85*795d594fSAndroid Build Coastguard Worker   }
86*795d594fSAndroid Build Coastguard Worker 
87*795d594fSAndroid Build Coastguard Worker   // Deny access if we didn't find the descriptor in the public api dex files.
88*795d594fSAndroid Build Coastguard Worker   return !found;
89*795d594fSAndroid Build Coastguard Worker }
90*795d594fSAndroid Build Coastguard Worker 
ShouldDenyAccess(ArtField * art_field) const91*795d594fSAndroid Build Coastguard Worker bool SdkChecker::ShouldDenyAccess(ArtField* art_field) const {
92*795d594fSAndroid Build Coastguard Worker   if (!enabled_) {
93*795d594fSAndroid Build Coastguard Worker     return false;
94*795d594fSAndroid Build Coastguard Worker   }
95*795d594fSAndroid Build Coastguard Worker 
96*795d594fSAndroid Build Coastguard Worker   std::string_view declaring_class_descriptor = art_field->GetDeclaringClassDescriptorView();
97*795d594fSAndroid Build Coastguard Worker   const char* name = art_field->GetName();
98*795d594fSAndroid Build Coastguard Worker   std::string_view type_descriptor = art_field->GetTypeDescriptorView();
99*795d594fSAndroid Build Coastguard Worker 
100*795d594fSAndroid Build Coastguard Worker   bool found = false;
101*795d594fSAndroid Build Coastguard Worker   for (const std::unique_ptr<const DexFile>& dex_file : sdk_dex_files_) {
102*795d594fSAndroid Build Coastguard Worker     const dex::TypeId* declaring_type_id = dex_file->FindTypeId(declaring_class_descriptor);
103*795d594fSAndroid Build Coastguard Worker     if (declaring_type_id == nullptr) {
104*795d594fSAndroid Build Coastguard Worker       continue;
105*795d594fSAndroid Build Coastguard Worker     }
106*795d594fSAndroid Build Coastguard Worker     const dex::StringId* name_id = dex_file->FindStringId(name);
107*795d594fSAndroid Build Coastguard Worker     if (name_id == nullptr) {
108*795d594fSAndroid Build Coastguard Worker       continue;
109*795d594fSAndroid Build Coastguard Worker     }
110*795d594fSAndroid Build Coastguard Worker     const dex::TypeId* type_id = dex_file->FindTypeId(type_descriptor);
111*795d594fSAndroid Build Coastguard Worker     if (type_id == nullptr) {
112*795d594fSAndroid Build Coastguard Worker       continue;
113*795d594fSAndroid Build Coastguard Worker     }
114*795d594fSAndroid Build Coastguard Worker 
115*795d594fSAndroid Build Coastguard Worker     const dex::FieldId* field_id = dex_file->FindFieldId(*declaring_type_id, *name_id, *type_id);
116*795d594fSAndroid Build Coastguard Worker     if (field_id != nullptr) {
117*795d594fSAndroid Build Coastguard Worker       found = true;
118*795d594fSAndroid Build Coastguard Worker       break;
119*795d594fSAndroid Build Coastguard Worker     }
120*795d594fSAndroid Build Coastguard Worker   }
121*795d594fSAndroid Build Coastguard Worker 
122*795d594fSAndroid Build Coastguard Worker   if (!found) {
123*795d594fSAndroid Build Coastguard Worker     VLOG(verifier) << "Deny for " << ArtField::PrettyField(art_field, true);
124*795d594fSAndroid Build Coastguard Worker   }
125*795d594fSAndroid Build Coastguard Worker 
126*795d594fSAndroid Build Coastguard Worker   // Deny access if we didn't find the descriptor in the public api dex files.
127*795d594fSAndroid Build Coastguard Worker   return !found;
128*795d594fSAndroid Build Coastguard Worker }
129*795d594fSAndroid Build Coastguard Worker 
ShouldDenyAccess(std::string_view descriptor) const130*795d594fSAndroid Build Coastguard Worker bool SdkChecker::ShouldDenyAccess(std::string_view descriptor) const {
131*795d594fSAndroid Build Coastguard Worker   if (!enabled_) {
132*795d594fSAndroid Build Coastguard Worker     return false;
133*795d594fSAndroid Build Coastguard Worker   }
134*795d594fSAndroid Build Coastguard Worker 
135*795d594fSAndroid Build Coastguard Worker   bool found = false;
136*795d594fSAndroid Build Coastguard Worker   for (const std::unique_ptr<const DexFile>& dex_file : sdk_dex_files_) {
137*795d594fSAndroid Build Coastguard Worker     const dex::TypeId* type_id = dex_file->FindTypeId(descriptor);
138*795d594fSAndroid Build Coastguard Worker     if (type_id != nullptr) {
139*795d594fSAndroid Build Coastguard Worker       dex::TypeIndex type_idx = dex_file->GetIndexForTypeId(*type_id);
140*795d594fSAndroid Build Coastguard Worker       if (dex_file->FindClassDef(type_idx) != nullptr) {
141*795d594fSAndroid Build Coastguard Worker         found = true;
142*795d594fSAndroid Build Coastguard Worker         break;
143*795d594fSAndroid Build Coastguard Worker       }
144*795d594fSAndroid Build Coastguard Worker     }
145*795d594fSAndroid Build Coastguard Worker   }
146*795d594fSAndroid Build Coastguard Worker 
147*795d594fSAndroid Build Coastguard Worker   if (!found) {
148*795d594fSAndroid Build Coastguard Worker     VLOG(verifier) << "Deny for " << descriptor;
149*795d594fSAndroid Build Coastguard Worker   }
150*795d594fSAndroid Build Coastguard Worker 
151*795d594fSAndroid Build Coastguard Worker   // Deny access if we didn't find the descriptor in the public api dex files.
152*795d594fSAndroid Build Coastguard Worker   return !found;
153*795d594fSAndroid Build Coastguard Worker }
154*795d594fSAndroid Build Coastguard Worker 
155*795d594fSAndroid Build Coastguard Worker }  // namespace art
156