| /* |
| * Copyright (C) 2019 The Android Open Source Project |
| * |
| * Licensed under the Apache License, Version 2.0 (the "License"); |
| * you may not use this file except in compliance with the License. |
| * You may obtain a copy of the License at |
| * |
| * http://www.apache.org/licenses/LICENSE-2.0 |
| * |
| * Unless required by applicable law or agreed to in writing, software |
| * distributed under the License is distributed on an "AS IS" BASIS, |
| * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. |
| * See the License for the specific language governing permissions and |
| * limitations under the License. |
| */ |
| |
| #include "src/trace_processor/syscall_tracker.h" |
| |
| #include <type_traits> |
| #include <utility> |
| |
| #include <inttypes.h> |
| |
| #include "src/trace_processor/stats.h" |
| |
| #include "src/trace_processor/syscalls_aarch32.h" |
| #include "src/trace_processor/syscalls_aarch64.h" |
| #include "src/trace_processor/syscalls_armeabi.h" |
| #include "src/trace_processor/syscalls_x86_64.h" |
| |
| namespace perfetto { |
| namespace trace_processor { |
| namespace { |
| |
| template <typename T> |
| constexpr size_t GetSyscalls(const T&) { |
| static_assert(std::extent<T>::value <= kMaxSyscalls, |
| "kMaxSyscalls too small"); |
| return std::extent<T>::value; |
| } |
| |
| } // namespace |
| |
| // TODO(primiano): The current design is broken in case of 32-bit processes |
| // running on 64-bit kernel. At least on ARM, the syscal numbers don't match |
| // and we should use the kSyscalls_Aarch32 table for those processes. But this |
| // means that the architecture is not a global property but is per-process. |
| // Which in turn means that somehow we need to figure out what is the bitness |
| // of each process from the trace. |
| SyscallTracker::SyscallTracker(TraceProcessorContext* context) |
| : context_(context) { |
| SetArchitecture(kUnknown); |
| } |
| |
| SyscallTracker::~SyscallTracker() = default; |
| |
| void SyscallTracker::SetArchitecture(Architecture arch) { |
| const char* kSyscalls_Unknown[] = {nullptr}; |
| size_t num_syscalls = 0; |
| const char* const* syscall_table = nullptr; |
| |
| switch (arch) { |
| case kArmEabi: |
| num_syscalls = GetSyscalls(kSyscalls_ArmEabi); |
| syscall_table = &kSyscalls_ArmEabi[0]; |
| break; |
| case kAarch32: |
| num_syscalls = GetSyscalls(kSyscalls_Aarch32); |
| syscall_table = &kSyscalls_Aarch32[0]; |
| break; |
| case kAarch64: |
| num_syscalls = GetSyscalls(kSyscalls_Aarch64); |
| syscall_table = &kSyscalls_Aarch64[0]; |
| break; |
| case kX86_64: |
| num_syscalls = GetSyscalls(kSyscalls_x86_64); |
| syscall_table = &kSyscalls_x86_64[0]; |
| break; |
| case kUnknown: |
| num_syscalls = 0; |
| syscall_table = &kSyscalls_Unknown[0]; |
| break; |
| } |
| |
| for (size_t i = 0; i < kMaxSyscalls; i++) { |
| StringId id = 0; |
| if (i < num_syscalls && syscall_table[i] && *syscall_table[i]) { |
| const char* name = syscall_table[i]; |
| id = context_->storage->InternString(name); |
| if (!strcmp(name, "sys_write")) |
| sys_write_string_id_ = id; |
| } else { |
| char unknown_str[64]; |
| sprintf(unknown_str, "sys_%zu", i); |
| id = context_->storage->InternString(unknown_str); |
| } |
| arch_syscall_to_string_id_[i] = id; |
| } |
| } |
| |
| } // namespace trace_processor |
| } // namespace perfetto |