xref: /aosp_15_r20/external/perfetto/src/profiling/deobfuscator.cc (revision 6dbdd20afdafa5e3ca9b8809fa73465d530080dc)
1*6dbdd20aSAndroid Build Coastguard Worker /*
2*6dbdd20aSAndroid Build Coastguard Worker  * Copyright (C) 2019 The Android Open Source Project
3*6dbdd20aSAndroid Build Coastguard Worker  *
4*6dbdd20aSAndroid Build Coastguard Worker  * Licensed under the Apache License, Version 2.0 (the "License");
5*6dbdd20aSAndroid Build Coastguard Worker  * you may not use this file except in compliance with the License.
6*6dbdd20aSAndroid Build Coastguard Worker  * You may obtain a copy of the License at
7*6dbdd20aSAndroid Build Coastguard Worker  *
8*6dbdd20aSAndroid Build Coastguard Worker  *      http://www.apache.org/licenses/LICENSE-2.0
9*6dbdd20aSAndroid Build Coastguard Worker  *
10*6dbdd20aSAndroid Build Coastguard Worker  * Unless required by applicable law or agreed to in writing, software
11*6dbdd20aSAndroid Build Coastguard Worker  * distributed under the License is distributed on an "AS IS" BASIS,
12*6dbdd20aSAndroid Build Coastguard Worker  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13*6dbdd20aSAndroid Build Coastguard Worker  * See the License for the specific language governing permissions and
14*6dbdd20aSAndroid Build Coastguard Worker  * limitations under the License.
15*6dbdd20aSAndroid Build Coastguard Worker  */
16*6dbdd20aSAndroid Build Coastguard Worker 
17*6dbdd20aSAndroid Build Coastguard Worker #include "src/profiling/deobfuscator.h"
18*6dbdd20aSAndroid Build Coastguard Worker 
19*6dbdd20aSAndroid Build Coastguard Worker #include <optional>
20*6dbdd20aSAndroid Build Coastguard Worker #include "perfetto/base/status.h"
21*6dbdd20aSAndroid Build Coastguard Worker #include "perfetto/ext/base/file_utils.h"
22*6dbdd20aSAndroid Build Coastguard Worker #include "perfetto/ext/base/scoped_file.h"
23*6dbdd20aSAndroid Build Coastguard Worker #include "perfetto/ext/base/string_splitter.h"
24*6dbdd20aSAndroid Build Coastguard Worker 
25*6dbdd20aSAndroid Build Coastguard Worker #include "perfetto/protozero/scattered_heap_buffer.h"
26*6dbdd20aSAndroid Build Coastguard Worker #include "protos/perfetto/trace/profiling/deobfuscation.pbzero.h"
27*6dbdd20aSAndroid Build Coastguard Worker #include "protos/perfetto/trace/trace.pbzero.h"
28*6dbdd20aSAndroid Build Coastguard Worker #include "protos/perfetto/trace/trace_packet.pbzero.h"
29*6dbdd20aSAndroid Build Coastguard Worker 
30*6dbdd20aSAndroid Build Coastguard Worker namespace perfetto {
31*6dbdd20aSAndroid Build Coastguard Worker namespace profiling {
32*6dbdd20aSAndroid Build Coastguard Worker namespace {
33*6dbdd20aSAndroid Build Coastguard Worker 
34*6dbdd20aSAndroid Build Coastguard Worker struct ProguardClass {
35*6dbdd20aSAndroid Build Coastguard Worker   std::string obfuscated_name;
36*6dbdd20aSAndroid Build Coastguard Worker   std::string deobfuscated_name;
37*6dbdd20aSAndroid Build Coastguard Worker };
38*6dbdd20aSAndroid Build Coastguard Worker 
ParseClass(std::string line)39*6dbdd20aSAndroid Build Coastguard Worker std::optional<ProguardClass> ParseClass(std::string line) {
40*6dbdd20aSAndroid Build Coastguard Worker   base::StringSplitter ss(std::move(line), ' ');
41*6dbdd20aSAndroid Build Coastguard Worker 
42*6dbdd20aSAndroid Build Coastguard Worker   if (!ss.Next()) {
43*6dbdd20aSAndroid Build Coastguard Worker     PERFETTO_ELOG("Missing deobfuscated name.");
44*6dbdd20aSAndroid Build Coastguard Worker     return std::nullopt;
45*6dbdd20aSAndroid Build Coastguard Worker   }
46*6dbdd20aSAndroid Build Coastguard Worker   std::string deobfuscated_name(ss.cur_token(), ss.cur_token_size());
47*6dbdd20aSAndroid Build Coastguard Worker 
48*6dbdd20aSAndroid Build Coastguard Worker   if (!ss.Next() || ss.cur_token_size() != 2 ||
49*6dbdd20aSAndroid Build Coastguard Worker       strncmp("->", ss.cur_token(), 2) != 0) {
50*6dbdd20aSAndroid Build Coastguard Worker     PERFETTO_ELOG("Missing ->");
51*6dbdd20aSAndroid Build Coastguard Worker     return std::nullopt;
52*6dbdd20aSAndroid Build Coastguard Worker   }
53*6dbdd20aSAndroid Build Coastguard Worker 
54*6dbdd20aSAndroid Build Coastguard Worker   if (!ss.Next()) {
55*6dbdd20aSAndroid Build Coastguard Worker     PERFETTO_ELOG("Missing obfuscated name.");
56*6dbdd20aSAndroid Build Coastguard Worker     return std::nullopt;
57*6dbdd20aSAndroid Build Coastguard Worker   }
58*6dbdd20aSAndroid Build Coastguard Worker   std::string obfuscated_name(ss.cur_token(), ss.cur_token_size());
59*6dbdd20aSAndroid Build Coastguard Worker   if (obfuscated_name.empty()) {
60*6dbdd20aSAndroid Build Coastguard Worker     PERFETTO_ELOG("Empty obfuscated name.");
61*6dbdd20aSAndroid Build Coastguard Worker     return std::nullopt;
62*6dbdd20aSAndroid Build Coastguard Worker   }
63*6dbdd20aSAndroid Build Coastguard Worker   if (obfuscated_name.back() != ':') {
64*6dbdd20aSAndroid Build Coastguard Worker     PERFETTO_ELOG("Expected colon.");
65*6dbdd20aSAndroid Build Coastguard Worker     return std::nullopt;
66*6dbdd20aSAndroid Build Coastguard Worker   }
67*6dbdd20aSAndroid Build Coastguard Worker 
68*6dbdd20aSAndroid Build Coastguard Worker   obfuscated_name.resize(obfuscated_name.size() - 1);
69*6dbdd20aSAndroid Build Coastguard Worker   if (ss.Next()) {
70*6dbdd20aSAndroid Build Coastguard Worker     PERFETTO_ELOG("Unexpected data.");
71*6dbdd20aSAndroid Build Coastguard Worker     return std::nullopt;
72*6dbdd20aSAndroid Build Coastguard Worker   }
73*6dbdd20aSAndroid Build Coastguard Worker   return ProguardClass{std::move(obfuscated_name),
74*6dbdd20aSAndroid Build Coastguard Worker                        std::move(deobfuscated_name)};
75*6dbdd20aSAndroid Build Coastguard Worker }
76*6dbdd20aSAndroid Build Coastguard Worker 
77*6dbdd20aSAndroid Build Coastguard Worker enum class ProguardMemberType {
78*6dbdd20aSAndroid Build Coastguard Worker   kField,
79*6dbdd20aSAndroid Build Coastguard Worker   kMethod,
80*6dbdd20aSAndroid Build Coastguard Worker };
81*6dbdd20aSAndroid Build Coastguard Worker 
82*6dbdd20aSAndroid Build Coastguard Worker struct ProguardMember {
83*6dbdd20aSAndroid Build Coastguard Worker   ProguardMemberType type;
84*6dbdd20aSAndroid Build Coastguard Worker   std::string obfuscated_name;
85*6dbdd20aSAndroid Build Coastguard Worker   std::string deobfuscated_name;
86*6dbdd20aSAndroid Build Coastguard Worker };
87*6dbdd20aSAndroid Build Coastguard Worker 
ParseMember(std::string line)88*6dbdd20aSAndroid Build Coastguard Worker std::optional<ProguardMember> ParseMember(std::string line) {
89*6dbdd20aSAndroid Build Coastguard Worker   base::StringSplitter ss(std::move(line), ' ');
90*6dbdd20aSAndroid Build Coastguard Worker 
91*6dbdd20aSAndroid Build Coastguard Worker   if (!ss.Next()) {
92*6dbdd20aSAndroid Build Coastguard Worker     PERFETTO_ELOG("Missing type name.");
93*6dbdd20aSAndroid Build Coastguard Worker     return std::nullopt;
94*6dbdd20aSAndroid Build Coastguard Worker   }
95*6dbdd20aSAndroid Build Coastguard Worker   std::string type_name(ss.cur_token(), ss.cur_token_size());
96*6dbdd20aSAndroid Build Coastguard Worker 
97*6dbdd20aSAndroid Build Coastguard Worker   if (!ss.Next()) {
98*6dbdd20aSAndroid Build Coastguard Worker     PERFETTO_ELOG("Missing deobfuscated name.");
99*6dbdd20aSAndroid Build Coastguard Worker     return std::nullopt;
100*6dbdd20aSAndroid Build Coastguard Worker   }
101*6dbdd20aSAndroid Build Coastguard Worker   std::string deobfuscated_name(ss.cur_token(), ss.cur_token_size());
102*6dbdd20aSAndroid Build Coastguard Worker 
103*6dbdd20aSAndroid Build Coastguard Worker   if (!ss.Next() || ss.cur_token_size() != 2 ||
104*6dbdd20aSAndroid Build Coastguard Worker       strncmp("->", ss.cur_token(), 2) != 0) {
105*6dbdd20aSAndroid Build Coastguard Worker     PERFETTO_ELOG("Missing ->");
106*6dbdd20aSAndroid Build Coastguard Worker     return std::nullopt;
107*6dbdd20aSAndroid Build Coastguard Worker   }
108*6dbdd20aSAndroid Build Coastguard Worker 
109*6dbdd20aSAndroid Build Coastguard Worker   if (!ss.Next()) {
110*6dbdd20aSAndroid Build Coastguard Worker     PERFETTO_ELOG("Missing obfuscated name.");
111*6dbdd20aSAndroid Build Coastguard Worker     return std::nullopt;
112*6dbdd20aSAndroid Build Coastguard Worker   }
113*6dbdd20aSAndroid Build Coastguard Worker   std::string obfuscated_name(ss.cur_token(), ss.cur_token_size());
114*6dbdd20aSAndroid Build Coastguard Worker 
115*6dbdd20aSAndroid Build Coastguard Worker   if (ss.Next()) {
116*6dbdd20aSAndroid Build Coastguard Worker     PERFETTO_ELOG("Unexpected data.");
117*6dbdd20aSAndroid Build Coastguard Worker     return std::nullopt;
118*6dbdd20aSAndroid Build Coastguard Worker   }
119*6dbdd20aSAndroid Build Coastguard Worker 
120*6dbdd20aSAndroid Build Coastguard Worker   ProguardMemberType member_type;
121*6dbdd20aSAndroid Build Coastguard Worker   auto paren_idx = deobfuscated_name.find('(');
122*6dbdd20aSAndroid Build Coastguard Worker   if (paren_idx != std::string::npos) {
123*6dbdd20aSAndroid Build Coastguard Worker     member_type = ProguardMemberType::kMethod;
124*6dbdd20aSAndroid Build Coastguard Worker     deobfuscated_name = deobfuscated_name.substr(0, paren_idx);
125*6dbdd20aSAndroid Build Coastguard Worker     auto colon_idx = type_name.find(':');
126*6dbdd20aSAndroid Build Coastguard Worker     if (colon_idx != std::string::npos) {
127*6dbdd20aSAndroid Build Coastguard Worker       type_name = type_name.substr(colon_idx + 1);
128*6dbdd20aSAndroid Build Coastguard Worker     }
129*6dbdd20aSAndroid Build Coastguard Worker   } else {
130*6dbdd20aSAndroid Build Coastguard Worker     member_type = ProguardMemberType::kField;
131*6dbdd20aSAndroid Build Coastguard Worker   }
132*6dbdd20aSAndroid Build Coastguard Worker   return ProguardMember{member_type, std::move(obfuscated_name),
133*6dbdd20aSAndroid Build Coastguard Worker                         std::move(deobfuscated_name)};
134*6dbdd20aSAndroid Build Coastguard Worker }
135*6dbdd20aSAndroid Build Coastguard Worker 
FlattenMethods(const std::vector<std::string> & v)136*6dbdd20aSAndroid Build Coastguard Worker std::string FlattenMethods(const std::vector<std::string>& v) {
137*6dbdd20aSAndroid Build Coastguard Worker   if (v.size() == 1) {
138*6dbdd20aSAndroid Build Coastguard Worker     return v[0];
139*6dbdd20aSAndroid Build Coastguard Worker   }
140*6dbdd20aSAndroid Build Coastguard Worker   return "[ambiguous]";
141*6dbdd20aSAndroid Build Coastguard Worker }
142*6dbdd20aSAndroid Build Coastguard Worker 
143*6dbdd20aSAndroid Build Coastguard Worker }  // namespace
144*6dbdd20aSAndroid Build Coastguard Worker 
FlattenClasses(const std::map<std::string,std::vector<std::string>> & m)145*6dbdd20aSAndroid Build Coastguard Worker std::string FlattenClasses(
146*6dbdd20aSAndroid Build Coastguard Worker     const std::map<std::string, std::vector<std::string>>& m) {
147*6dbdd20aSAndroid Build Coastguard Worker   std::string result;
148*6dbdd20aSAndroid Build Coastguard Worker   bool first = true;
149*6dbdd20aSAndroid Build Coastguard Worker   for (const auto& p : m) {
150*6dbdd20aSAndroid Build Coastguard Worker     if (!first) {
151*6dbdd20aSAndroid Build Coastguard Worker       result += " | ";
152*6dbdd20aSAndroid Build Coastguard Worker     }
153*6dbdd20aSAndroid Build Coastguard Worker     result += p.first + "." + FlattenMethods(p.second);
154*6dbdd20aSAndroid Build Coastguard Worker     first = false;
155*6dbdd20aSAndroid Build Coastguard Worker   }
156*6dbdd20aSAndroid Build Coastguard Worker   return result;
157*6dbdd20aSAndroid Build Coastguard Worker }
158*6dbdd20aSAndroid Build Coastguard Worker 
159*6dbdd20aSAndroid Build Coastguard Worker // See https://www.guardsquare.com/en/products/proguard/manual/retrace for the
160*6dbdd20aSAndroid Build Coastguard Worker // file format we are parsing.
AddLine(std::string line)161*6dbdd20aSAndroid Build Coastguard Worker base::Status ProguardParser::AddLine(std::string line) {
162*6dbdd20aSAndroid Build Coastguard Worker   auto first_ch_pos = line.find_first_not_of(" \t");
163*6dbdd20aSAndroid Build Coastguard Worker   if (first_ch_pos == std::string::npos || line[first_ch_pos] == '#')
164*6dbdd20aSAndroid Build Coastguard Worker     return base::Status();
165*6dbdd20aSAndroid Build Coastguard Worker 
166*6dbdd20aSAndroid Build Coastguard Worker   bool is_member = line[0] == ' ';
167*6dbdd20aSAndroid Build Coastguard Worker   if (is_member && !current_class_) {
168*6dbdd20aSAndroid Build Coastguard Worker     return base::Status(
169*6dbdd20aSAndroid Build Coastguard Worker         "Failed to parse proguard map. Saw member before class.");
170*6dbdd20aSAndroid Build Coastguard Worker   }
171*6dbdd20aSAndroid Build Coastguard Worker   if (!is_member) {
172*6dbdd20aSAndroid Build Coastguard Worker     auto opt_cls = ParseClass(std::move(line));
173*6dbdd20aSAndroid Build Coastguard Worker     if (!opt_cls)
174*6dbdd20aSAndroid Build Coastguard Worker       return base::Status("Class not found.");
175*6dbdd20aSAndroid Build Coastguard Worker     auto p = mapping_.emplace(std::move(opt_cls->obfuscated_name),
176*6dbdd20aSAndroid Build Coastguard Worker                               std::move(opt_cls->deobfuscated_name));
177*6dbdd20aSAndroid Build Coastguard Worker     if (!p.second) {
178*6dbdd20aSAndroid Build Coastguard Worker       return base::Status("Duplicate class.");
179*6dbdd20aSAndroid Build Coastguard Worker     }
180*6dbdd20aSAndroid Build Coastguard Worker     current_class_ = &p.first->second;
181*6dbdd20aSAndroid Build Coastguard Worker   } else {
182*6dbdd20aSAndroid Build Coastguard Worker     auto opt_member = ParseMember(std::move(line));
183*6dbdd20aSAndroid Build Coastguard Worker     if (!opt_member)
184*6dbdd20aSAndroid Build Coastguard Worker       return base::Status("Failed to parse member.");
185*6dbdd20aSAndroid Build Coastguard Worker     switch (opt_member->type) {
186*6dbdd20aSAndroid Build Coastguard Worker       case (ProguardMemberType::kField): {
187*6dbdd20aSAndroid Build Coastguard Worker         if (!current_class_->AddField(opt_member->obfuscated_name,
188*6dbdd20aSAndroid Build Coastguard Worker                                       opt_member->deobfuscated_name)) {
189*6dbdd20aSAndroid Build Coastguard Worker           return base::Status(std::string("Member redefinition: ") +
190*6dbdd20aSAndroid Build Coastguard Worker                               current_class_->deobfuscated_name().c_str() +
191*6dbdd20aSAndroid Build Coastguard Worker                               "." + opt_member->deobfuscated_name.c_str() +
192*6dbdd20aSAndroid Build Coastguard Worker                               " Proguard map invalid");
193*6dbdd20aSAndroid Build Coastguard Worker         }
194*6dbdd20aSAndroid Build Coastguard Worker         break;
195*6dbdd20aSAndroid Build Coastguard Worker       }
196*6dbdd20aSAndroid Build Coastguard Worker       case (ProguardMemberType::kMethod): {
197*6dbdd20aSAndroid Build Coastguard Worker         current_class_->AddMethod(opt_member->obfuscated_name,
198*6dbdd20aSAndroid Build Coastguard Worker                                   opt_member->deobfuscated_name);
199*6dbdd20aSAndroid Build Coastguard Worker         break;
200*6dbdd20aSAndroid Build Coastguard Worker       }
201*6dbdd20aSAndroid Build Coastguard Worker     }
202*6dbdd20aSAndroid Build Coastguard Worker   }
203*6dbdd20aSAndroid Build Coastguard Worker   return base::Status();
204*6dbdd20aSAndroid Build Coastguard Worker }
205*6dbdd20aSAndroid Build Coastguard Worker 
AddLines(std::string contents)206*6dbdd20aSAndroid Build Coastguard Worker bool ProguardParser::AddLines(std::string contents) {
207*6dbdd20aSAndroid Build Coastguard Worker   size_t lineno = 1;
208*6dbdd20aSAndroid Build Coastguard Worker   for (base::StringSplitter lines(std::move(contents), '\n'); lines.Next();) {
209*6dbdd20aSAndroid Build Coastguard Worker     auto status = AddLine(lines.cur_token());
210*6dbdd20aSAndroid Build Coastguard Worker     if (!status.ok()) {
211*6dbdd20aSAndroid Build Coastguard Worker       PERFETTO_ELOG("Failed to parse proguard map (line %zu): %s", lineno,
212*6dbdd20aSAndroid Build Coastguard Worker                     status.c_message());
213*6dbdd20aSAndroid Build Coastguard Worker       return false;
214*6dbdd20aSAndroid Build Coastguard Worker     }
215*6dbdd20aSAndroid Build Coastguard Worker     lineno++;
216*6dbdd20aSAndroid Build Coastguard Worker   }
217*6dbdd20aSAndroid Build Coastguard Worker   return true;
218*6dbdd20aSAndroid Build Coastguard Worker }
219*6dbdd20aSAndroid Build Coastguard Worker 
MakeDeobfuscationPackets(const std::string & package_name,const std::map<std::string,profiling::ObfuscatedClass> & mapping,std::function<void (const std::string &)> callback)220*6dbdd20aSAndroid Build Coastguard Worker void MakeDeobfuscationPackets(
221*6dbdd20aSAndroid Build Coastguard Worker     const std::string& package_name,
222*6dbdd20aSAndroid Build Coastguard Worker     const std::map<std::string, profiling::ObfuscatedClass>& mapping,
223*6dbdd20aSAndroid Build Coastguard Worker     std::function<void(const std::string&)> callback) {
224*6dbdd20aSAndroid Build Coastguard Worker   protozero::HeapBuffered<perfetto::protos::pbzero::Trace> trace;
225*6dbdd20aSAndroid Build Coastguard Worker   auto* packet = trace->add_packet();
226*6dbdd20aSAndroid Build Coastguard Worker   // TODO(fmayer): Add handling for package name and version code here so we
227*6dbdd20aSAndroid Build Coastguard Worker   // can support multiple dumps in the same trace.
228*6dbdd20aSAndroid Build Coastguard Worker   auto* proto_mapping = packet->set_deobfuscation_mapping();
229*6dbdd20aSAndroid Build Coastguard Worker   proto_mapping->set_package_name(package_name);
230*6dbdd20aSAndroid Build Coastguard Worker   for (const auto& p : mapping) {
231*6dbdd20aSAndroid Build Coastguard Worker     const std::string& obfuscated_class_name = p.first;
232*6dbdd20aSAndroid Build Coastguard Worker     const profiling::ObfuscatedClass& cls = p.second;
233*6dbdd20aSAndroid Build Coastguard Worker 
234*6dbdd20aSAndroid Build Coastguard Worker     auto* proto_class = proto_mapping->add_obfuscated_classes();
235*6dbdd20aSAndroid Build Coastguard Worker     proto_class->set_obfuscated_name(obfuscated_class_name);
236*6dbdd20aSAndroid Build Coastguard Worker     proto_class->set_deobfuscated_name(cls.deobfuscated_name());
237*6dbdd20aSAndroid Build Coastguard Worker     for (const auto& field_p : cls.deobfuscated_fields()) {
238*6dbdd20aSAndroid Build Coastguard Worker       const std::string& obfuscated_field_name = field_p.first;
239*6dbdd20aSAndroid Build Coastguard Worker       const std::string& deobfuscated_field_name = field_p.second;
240*6dbdd20aSAndroid Build Coastguard Worker       auto* proto_member = proto_class->add_obfuscated_members();
241*6dbdd20aSAndroid Build Coastguard Worker       proto_member->set_obfuscated_name(obfuscated_field_name);
242*6dbdd20aSAndroid Build Coastguard Worker       proto_member->set_deobfuscated_name(deobfuscated_field_name);
243*6dbdd20aSAndroid Build Coastguard Worker     }
244*6dbdd20aSAndroid Build Coastguard Worker     for (const auto& field_p : cls.deobfuscated_methods()) {
245*6dbdd20aSAndroid Build Coastguard Worker       const std::string& obfuscated_method_name = field_p.first;
246*6dbdd20aSAndroid Build Coastguard Worker       const std::string& deobfuscated_method_name = field_p.second;
247*6dbdd20aSAndroid Build Coastguard Worker       auto* proto_member = proto_class->add_obfuscated_methods();
248*6dbdd20aSAndroid Build Coastguard Worker       proto_member->set_obfuscated_name(obfuscated_method_name);
249*6dbdd20aSAndroid Build Coastguard Worker       proto_member->set_deobfuscated_name(deobfuscated_method_name);
250*6dbdd20aSAndroid Build Coastguard Worker     }
251*6dbdd20aSAndroid Build Coastguard Worker   }
252*6dbdd20aSAndroid Build Coastguard Worker   callback(trace.SerializeAsString());
253*6dbdd20aSAndroid Build Coastguard Worker }
254*6dbdd20aSAndroid Build Coastguard Worker 
ReadProguardMapsToDeobfuscationPackets(const std::vector<ProguardMap> & maps,std::function<void (std::string)> fn)255*6dbdd20aSAndroid Build Coastguard Worker bool ReadProguardMapsToDeobfuscationPackets(
256*6dbdd20aSAndroid Build Coastguard Worker     const std::vector<ProguardMap>& maps,
257*6dbdd20aSAndroid Build Coastguard Worker     std::function<void(std::string)> fn) {
258*6dbdd20aSAndroid Build Coastguard Worker   for (const ProguardMap& map : maps) {
259*6dbdd20aSAndroid Build Coastguard Worker     const char* filename = map.filename.c_str();
260*6dbdd20aSAndroid Build Coastguard Worker     base::ScopedFstream f(fopen(filename, "re"));
261*6dbdd20aSAndroid Build Coastguard Worker     if (!f) {
262*6dbdd20aSAndroid Build Coastguard Worker       PERFETTO_ELOG("Failed to open %s", filename);
263*6dbdd20aSAndroid Build Coastguard Worker       return false;
264*6dbdd20aSAndroid Build Coastguard Worker     }
265*6dbdd20aSAndroid Build Coastguard Worker     profiling::ProguardParser parser;
266*6dbdd20aSAndroid Build Coastguard Worker     std::string contents;
267*6dbdd20aSAndroid Build Coastguard Worker     PERFETTO_CHECK(base::ReadFileStream(*f, &contents));
268*6dbdd20aSAndroid Build Coastguard Worker     if (!parser.AddLines(std::move(contents))) {
269*6dbdd20aSAndroid Build Coastguard Worker       PERFETTO_ELOG("Failed to parse %s", filename);
270*6dbdd20aSAndroid Build Coastguard Worker       return false;
271*6dbdd20aSAndroid Build Coastguard Worker     }
272*6dbdd20aSAndroid Build Coastguard Worker     std::map<std::string, profiling::ObfuscatedClass> obfuscation_map =
273*6dbdd20aSAndroid Build Coastguard Worker         parser.ConsumeMapping();
274*6dbdd20aSAndroid Build Coastguard Worker 
275*6dbdd20aSAndroid Build Coastguard Worker     // TODO(fmayer): right now, we don't use the profile we are given. We can
276*6dbdd20aSAndroid Build Coastguard Worker     // filter the output to only contain the classes actually seen in the
277*6dbdd20aSAndroid Build Coastguard Worker     // profile.
278*6dbdd20aSAndroid Build Coastguard Worker     MakeDeobfuscationPackets(map.package, obfuscation_map, fn);
279*6dbdd20aSAndroid Build Coastguard Worker   }
280*6dbdd20aSAndroid Build Coastguard Worker   return true;
281*6dbdd20aSAndroid Build Coastguard Worker }
282*6dbdd20aSAndroid Build Coastguard Worker 
GetPerfettoProguardMapPath()283*6dbdd20aSAndroid Build Coastguard Worker std::vector<ProguardMap> GetPerfettoProguardMapPath() {
284*6dbdd20aSAndroid Build Coastguard Worker   const char* env = getenv("PERFETTO_PROGUARD_MAP");
285*6dbdd20aSAndroid Build Coastguard Worker   if (env == nullptr)
286*6dbdd20aSAndroid Build Coastguard Worker     return {};
287*6dbdd20aSAndroid Build Coastguard Worker   std::vector<ProguardMap> res;
288*6dbdd20aSAndroid Build Coastguard Worker   for (base::StringSplitter sp(std::string(env), ':'); sp.Next();) {
289*6dbdd20aSAndroid Build Coastguard Worker     std::string token(sp.cur_token(), sp.cur_token_size());
290*6dbdd20aSAndroid Build Coastguard Worker     size_t eq = token.find('=');
291*6dbdd20aSAndroid Build Coastguard Worker     if (eq == std::string::npos) {
292*6dbdd20aSAndroid Build Coastguard Worker       PERFETTO_ELOG(
293*6dbdd20aSAndroid Build Coastguard Worker           "Invalid PERFETTO_PROGUARD_MAP. "
294*6dbdd20aSAndroid Build Coastguard Worker           "Expected format packagename=filename[:packagename=filename...], "
295*6dbdd20aSAndroid Build Coastguard Worker           "e.g. com.example.package1=foo.txt:com.example.package2=bar.txt.");
296*6dbdd20aSAndroid Build Coastguard Worker       return {};
297*6dbdd20aSAndroid Build Coastguard Worker     }
298*6dbdd20aSAndroid Build Coastguard Worker     res.emplace_back(ProguardMap{token.substr(0, eq), token.substr(eq + 1)});
299*6dbdd20aSAndroid Build Coastguard Worker   }
300*6dbdd20aSAndroid Build Coastguard Worker   return res;  // for Wreturn-std-move-in-c++11.
301*6dbdd20aSAndroid Build Coastguard Worker }
302*6dbdd20aSAndroid Build Coastguard Worker 
303*6dbdd20aSAndroid Build Coastguard Worker }  // namespace profiling
304*6dbdd20aSAndroid Build Coastguard Worker }  // namespace perfetto
305