• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 /*
2  * Copyright (C) 2020 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 #include "MapRecordReader.h"
18 
19 #include <stdint.h>
20 #include <sys/mman.h>
21 #include <sys/stat.h>
22 
23 #include <vector>
24 
25 #include <android-base/strings.h>
26 
27 #include "environment.h"
28 
29 namespace simpleperf {
30 
ReadKernelMaps()31 bool MapRecordReader::ReadKernelMaps() {
32   KernelMmap kernel_mmap;
33   std::vector<KernelMmap> module_mmaps;
34   GetKernelAndModuleMmaps(&kernel_mmap, &module_mmaps);
35 
36   MmapRecord mmap_record(attr_, true, UINT_MAX, 0, kernel_mmap.start_addr, kernel_mmap.len, 0,
37                          kernel_mmap.filepath, event_id_);
38   if (!callback_(&mmap_record)) {
39     return false;
40   }
41   for (const auto& module_mmap : module_mmaps) {
42     MmapRecord mmap_record(attr_, true, UINT_MAX, 0, module_mmap.start_addr, module_mmap.len, 0,
43                            module_mmap.filepath, event_id_);
44     if (!callback_(&mmap_record)) {
45       return false;
46     }
47   }
48   return true;
49 }
50 
ReadProcessMaps(pid_t pid,uint64_t timestamp)51 bool MapRecordReader::ReadProcessMaps(pid_t pid, uint64_t timestamp) {
52   std::vector<pid_t> tids = GetThreadsInProcess(pid);
53   return ReadProcessMaps(pid, std::unordered_set<pid_t>(tids.begin(), tids.end()), timestamp);
54 }
55 
ReadProcessMaps(pid_t pid,const std::unordered_set<pid_t> & tids,uint64_t timestamp)56 bool MapRecordReader::ReadProcessMaps(pid_t pid, const std::unordered_set<pid_t>& tids,
57                                       uint64_t timestamp) {
58   // Dump mmap records.
59   std::vector<ThreadMmap> thread_mmaps;
60   if (!GetThreadMmapsInProcess(pid, &thread_mmaps)) {
61     // The process may exit before we get its info.
62     return true;
63   }
64   for (const auto& map : thread_mmaps) {
65     if (!(map.prot & PROT_EXEC) && !keep_non_executable_maps_) {
66       continue;
67     }
68     Mmap2Record record(attr_, false, pid, pid, map.start_addr, map.len, map.pgoff, map.prot,
69                        map.name, event_id_, timestamp);
70     if (!callback_(&record)) {
71       return false;
72     }
73   }
74   // Dump process name.
75   std::string process_name = GetCompleteProcessName(pid);
76   if (!process_name.empty()) {
77     CommRecord record(attr_, pid, pid, process_name, event_id_, timestamp);
78     if (!callback_(&record)) {
79       return false;
80     }
81   }
82   // Dump thread info.
83   for (const auto& tid : tids) {
84     std::string name;
85     if (tid != pid && GetThreadName(tid, &name)) {
86       // If a thread name matches the suffix of its process name, probably the thread name
87       // is stripped by TASK_COMM_LEN.
88       if (android::base::EndsWith(process_name, name)) {
89         name = process_name;
90       }
91       CommRecord comm_record(attr_, pid, tid, name, event_id_, timestamp);
92       if (!callback_(&comm_record)) {
93         return false;
94       }
95     }
96   }
97   return true;
98 }
99 
MapRecordThread(const MapRecordReader & map_record_reader)100 MapRecordThread::MapRecordThread(const MapRecordReader& map_record_reader)
101     : map_record_reader_(map_record_reader), fp_(nullptr, fclose) {
102   map_record_reader_.SetCallback([this](Record* r) { return WriteRecordToFile(r); });
103   tmpfile_ = ScopedTempFiles::CreateTempFile();
104   fp_.reset(fdopen(tmpfile_->release(), "r+"));
105   thread_ = std::thread([this]() { thread_result_ = RunThread(); });
106 }
107 
~MapRecordThread()108 MapRecordThread::~MapRecordThread() {
109   if (thread_.joinable()) {
110     early_stop_ = true;
111     thread_.join();
112   }
113 }
114 
RunThread()115 bool MapRecordThread::RunThread() {
116   if (!fp_) {
117     return false;
118   }
119   if (!map_record_reader_.ReadKernelMaps()) {
120     return false;
121   }
122   for (auto pid : GetAllProcesses()) {
123     if (early_stop_) {
124       return false;
125     }
126     if (!map_record_reader_.ReadProcessMaps(pid, 0)) {
127       return false;
128     }
129   }
130   return true;
131 }
132 
WriteRecordToFile(Record * record)133 bool MapRecordThread::WriteRecordToFile(Record* record) {
134   if (fwrite(record->Binary(), record->size(), 1, fp_.get()) != 1) {
135     PLOG(ERROR) << "failed to write map records to file";
136     return false;
137   }
138   return true;
139 }
140 
Join()141 bool MapRecordThread::Join() {
142   thread_.join();
143   if (!thread_result_) {
144     LOG(ERROR) << "map record thread failed";
145   }
146   return thread_result_;
147 }
148 
ReadMapRecords(const std::function<bool (Record *)> & callback)149 bool MapRecordThread::ReadMapRecords(const std::function<bool(Record*)>& callback) {
150   off_t offset = ftello(fp_.get());
151   if (offset == -1) {
152     PLOG(ERROR) << "ftello() failed";
153     return false;
154   }
155   uint64_t file_size = static_cast<uint64_t>(offset);
156   if (fseek(fp_.get(), 0, SEEK_SET) != 0) {
157     PLOG(ERROR) << "fseek() failed";
158     return false;
159   }
160   uint64_t nread = 0;
161   std::vector<char> buffer(1024);
162   while (nread < file_size) {
163     if (fread(buffer.data(), Record::header_size(), 1, fp_.get()) != 1) {
164       PLOG(ERROR) << "fread() failed";
165       return false;
166     }
167     RecordHeader header;
168     if (!header.Parse(buffer.data())) {
169       return false;
170     }
171     if (buffer.size() < header.size) {
172       buffer.resize(header.size);
173     }
174     if (fread(buffer.data() + Record::header_size(), header.size - Record::header_size(), 1,
175               fp_.get()) != 1) {
176       PLOG(ERROR) << "fread() failed";
177       return false;
178     }
179     auto r = ReadRecordFromBuffer(map_record_reader_.Attr(), header.type, buffer.data(),
180                                   buffer.data() + header.size);
181     CHECK(r);
182     if (!callback(r.get())) {
183       return false;
184     }
185     nread += header.size;
186   }
187   return true;
188 }
189 
190 }  // namespace simpleperf
191