1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98
//! Interacting with debugging agent //! //! # Example //! //! This example will show how to terminate the QEMU session. The program //! should be running under QEMU with semihosting enabled //! (use `-semihosting` flag). //! //! Target program: //! //! ``` //! #[macro_use] //! extern crate cortex_m_semihosting; //! use cortex_m_semihosting::debug::{self, EXIT_SUCCESS, EXIT_FAILURE}; //! //! fn main() { //! if 2 == 2 { //! // report success //! debug::exit(EXIT_SUCCESS); //! } else { //! // report failure //! debug::exit(EXIT_FAILURE); //! } //! } //! /// This values are taken from section 5.5.2 of /// ADS Debug Target Guide (DUI0058). // TODO document #[allow(missing_docs)] pub enum Exception { // Hardware reason codes BranchThroughZero = 0x20000, UndefinedInstr = 0x20001, SoftwareInterrupt = 0x20002, PrefetchAbort = 0x20003, DataAbort = 0x20004, AddressException = 0x20005, IRQ = 0x20006, FIQ = 0x20007, // Software reason codes BreakPoint = 0x20020, WatchPoint = 0x20021, StepComplete = 0x20022, RunTimeErrorUnknown = 0x20023, InternalError = 0x20024, UserInterruption = 0x20025, ApplicationExit = 0x20026, StackOverflow = 0x20027, DivisionByZero = 0x20028, OSSpecific = 0x20029, } /// Status enum for `exit` syscall. pub type ExitStatus = Result<(), ()>; /// Successful execution of a program. pub const EXIT_SUCCESS: ExitStatus = Ok(()); /// Unsuccessful execution of a program. pub const EXIT_FAILURE: ExitStatus = Err(()); /// Reports to the debugger that the execution has completed. /// /// This call can be used to terminate QEMU session and report back success /// or failure. If you need to pass more than one type of error, consider /// using `report_exception` syscall instead. /// /// This call should not return. However, it is possible for the debugger /// to request that the application continue. In that case this call /// returns normally. /// pub fn exit(status: ExitStatus) { match status { EXIT_SUCCESS => report_exception(Exception::ApplicationExit), EXIT_FAILURE => report_exception(Exception::RunTimeErrorUnknown), } } /// Report an exception to the debugger directly. /// /// Exception handlers can use this SWI at the end of handler chains /// as the default action, to indicate that the exception has not been handled. /// /// This call should not return. However, it is possible for the debugger /// to request that the application continue. In that case this call /// returns normally. /// /// # Arguments /// /// * `reason` - A reason code reported back to the debugger. /// pub fn report_exception(reason: Exception) { let code = reason as usize; unsafe { syscall1!(REPORT_EXCEPTION, code); } }