/* * Copyright (C) 2023 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 "berberis/runtime_primitives/recovery_code.h" #include #include #include #include "berberis/base/checks.h" #include "berberis/base/forever_map.h" #include "berberis/base/tracing.h" #include "berberis/guest_state/guest_state_opaque.h" #include "berberis/runtime_primitives/code_pool.h" namespace berberis { namespace { ForeverMap g_recovery_map; bool g_recovery_map_initialized_ = false; uintptr_t FindExtraRecoveryCodeUnsafe(uintptr_t fault_addr) { CHECK(g_recovery_map_initialized_); auto it = g_recovery_map.find(fault_addr); if (it != g_recovery_map.end()) { return it->second; } return 0; } } // namespace void InitExtraRecoveryCodeUnsafe( std::initializer_list> fault_recovery_pairs) { CHECK(!g_recovery_map_initialized_); for (auto pair : fault_recovery_pairs) { g_recovery_map[pair.first] = pair.second; } g_recovery_map_initialized_ = true; } uintptr_t FindRecoveryCode(uintptr_t fault_addr, ThreadState* state) { uintptr_t recovery_addr; CHECK(state); // Only look up in CodePool if we are inside generated code (interrupted by a // signal). If a signal interrupts CodePool::Add then calling FindRecoveryCode // in this state can cause deadlock. if (GetResidence(*state) == kInsideGeneratedCode) { // TODO(b/228188293): we might need to traverse all code pool instances. recovery_addr = GetDefaultCodePoolInstance()->FindRecoveryCode(fault_addr); if (recovery_addr) { return recovery_addr; } } // Extra recovery code is in read-only mode after the init, so we don't need mutexes. // Note, that we cannot simply add extra recovery code to CodePool, since these // fault addresses may be outside of generated code (e.g. interpreter). recovery_addr = FindExtraRecoveryCodeUnsafe(fault_addr); if (recovery_addr) { TRACE("found recovery address outside of code pool"); } return recovery_addr; } } // namespace berberis