| // Copyright 2022, 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. |
| |
| //! Exception handlers. |
| |
| use vmbase::{ |
| arch::aarch64::exceptions::{ |
| handle_permission_fault, handle_translation_fault, ArmException, Esr, HandleExceptionError, |
| }, |
| logger, read_sysreg, |
| }; |
| |
| fn handle_exception(exception: &ArmException) -> Result<(), HandleExceptionError> { |
| // Handle all translation faults on both read and write, and MMIO guard map |
| // flagged invalid pages or blocks that caused the exception. |
| // Handle permission faults for DBM flagged entries, and flag them as dirty on write. |
| match exception.esr { |
| Esr::DataAbortTranslationFault => handle_translation_fault(exception.far), |
| Esr::DataAbortPermissionFault => handle_permission_fault(exception.far), |
| _ => Err(HandleExceptionError::UnknownException), |
| } |
| } |
| |
| #[no_mangle] |
| extern "C" fn sync_exception_current(elr: u64, _spsr: u64) { |
| // Disable logging in exception handler to prevent unsafe writes to UART. |
| let _guard = logger::suppress(); |
| |
| let exception = ArmException::from_el1_regs(); |
| if let Err(e) = handle_exception(&exception) { |
| exception.print_and_reboot("sync_exception_current", e, elr); |
| } |
| } |
| |
| #[no_mangle] |
| extern "C" fn irq_current(_elr: u64, _spsr: u64) { |
| panic!("irq_current"); |
| } |
| |
| #[no_mangle] |
| extern "C" fn fiq_current(_elr: u64, _spsr: u64) { |
| panic!("fiq_current"); |
| } |
| |
| #[no_mangle] |
| extern "C" fn serr_current(_elr: u64, _spsr: u64) { |
| let esr = read_sysreg!("esr_el1"); |
| panic!("serr_current, esr={esr:#08x}"); |
| } |
| |
| #[no_mangle] |
| extern "C" fn sync_lower(_elr: u64, _spsr: u64) { |
| let esr = read_sysreg!("esr_el1"); |
| panic!("sync_lower, esr={esr:#08x}"); |
| } |
| |
| #[no_mangle] |
| extern "C" fn irq_lower(_elr: u64, _spsr: u64) { |
| panic!("irq_lower"); |
| } |
| |
| #[no_mangle] |
| extern "C" fn fiq_lower(_elr: u64, _spsr: u64) { |
| panic!("fiq_lower"); |
| } |
| |
| #[no_mangle] |
| extern "C" fn serr_lower(_elr: u64, _spsr: u64) { |
| let esr = read_sysreg!("esr_el1"); |
| panic!("serr_lower, esr={esr:#08x}"); |
| } |