Skip to content

Commit 94872e6

Browse files
authored
fix panic_handler (#84)
* add and update docs * update doc * add syscalls
1 parent a08186e commit 94872e6

File tree

3 files changed

+46
-8
lines changed

3 files changed

+46
-8
lines changed

README.md

+4-2
Original file line numberDiff line numberDiff line change
@@ -55,13 +55,15 @@ This will add `pinocchio` as a dependency to your project.
5555

5656
## Defining the program entrypoint
5757

58-
A Solana program needs to define an entrypoint, which will be called by the runtime to begin the program execution. The `entrypoint!` macro emits the common boilerplate to set up the program entrypoint. The macro will also set up [global allocator](https://doc.rust-lang.org/stable/core/alloc/trait.GlobalAlloc.html) and [panic handler](https://doc.rust-lang.org/nomicon/panic-handler.html) using the [default_allocator!](https://docs.rs/pinocchio/latest/pinocchio/macro.default_allocator.html) and [default_panic_handler!](https://docs.rs/pinocchio/latest/pinocchio/macro.default_panic_handler.html) macros.
58+
A Solana program needs to define an entrypoint, which will be called by the runtime to begin the program execution. The `entrypoint!` macro emits the common boilerplate to set up the program entrypoint. The macro will also set up [global allocator](https://doc.rust-lang.org/stable/core/alloc/trait.GlobalAlloc.html) and [custom panic hook](https://github.com/anza-xyz/rust/blob/2830febbc59d44bdd7ad2c3b81731f1d08b96eba/library/std/src/sys/pal/sbf/mod.rs#L49) using the [default_allocator!](https://docs.rs/pinocchio/latest/pinocchio/macro.default_allocator.html) and [default_panic_handler!](https://docs.rs/pinocchio/latest/pinocchio/macro.default_panic_handler.html) macros.
5959

6060
The [`entrypoint!`](https://docs.rs/pinocchio/latest/pinocchio/macro.entrypoint.html) is a convenience macro that invokes three other macros to set all symbols required for a program execution:
6161

6262
* [`program_entrypoint!`](https://docs.rs/pinocchio/latest/pinocchio/macro.program_entrypoint.html): declares the program entrypoint
6363
* [`default_allocator!`](https://docs.rs/pinocchio/latest/pinocchio/macro.default_allocator.html): declares the default (bump) global allocator
64-
* [`default_panic_hanlder!`](https://docs.rs/pinocchio/latest/pinocchio/macro.default_panic_handler.html): declares the default panic handler
64+
* [`default_panic_handler!`](https://docs.rs/pinocchio/latest/pinocchio/macro.default_panic_handler.html): declares the default panic handler
65+
66+
If all dependencies are `no_std`, you should append [`nostd_panic_handler!`](https://docs.rs/pinocchio/latest/pinocchio/macro.nostd_panic_handler.html) to declare a rust runtime panic handler. There's no need to do this if any dependency is `std` since rust compiler will emit std panic handler.
6567

6668
To use the `entrypoint!` macro, use the following in your entrypoint definition:
6769
```rust

sdk/pinocchio/src/entrypoint/mod.rs

+40-6
Original file line numberDiff line numberDiff line change
@@ -220,10 +220,11 @@ pub unsafe fn deserialize<'a, const MAX_ACCOUNTS: usize>(
220220
(program_id, processed, instruction_data)
221221
}
222222

223-
/// Default panic handler.
223+
/// Default panic hook (std).
224224
///
225-
/// This macro sets up a default panic handler that logs the panic message and the file where the
226-
/// panic occurred.
225+
/// This macro sets up a default panic hook that logs the panic message and the file where the
226+
/// panic occurred. Syscall "abort()" will be called after it returns. It acts as a hook after
227+
/// rust runtime panics.
227228
///
228229
/// Note that this requires the `"std"` feature to be enabled.
229230
#[cfg(feature = "std")]
@@ -240,11 +241,11 @@ macro_rules! default_panic_handler {
240241
};
241242
}
242243

243-
/// Default panic handler.
244+
/// Default panic hook (no std).
244245
///
245-
/// This macro sets up a default panic handler that logs the file where the panic occurred.
246+
/// This macro sets up a default panic hook that logs the file where the panic occurred.
246247
///
247-
/// This is used when the `"std"` feature is disabled.
248+
/// This is used when the `"std"` feature is disabled and program is `std`.
248249
#[cfg(not(feature = "std"))]
249250
#[macro_export]
250251
macro_rules! default_panic_handler {
@@ -262,6 +263,39 @@ macro_rules! default_panic_handler {
262263
};
263264
}
264265

266+
/// A rust panic handler for `no_std`.
267+
///
268+
/// When all crates are `no_std`, we need to define a global `#[panic_handler]`.
269+
/// It takes over the default rust panic handler.
270+
///
271+
/// This macro is used when the `"std"` feature is disabled.
272+
#[cfg(not(feature = "std"))]
273+
#[macro_export]
274+
macro_rules! nostd_panic_handler {
275+
() => {
276+
/// A panic handler for `no_std`.
277+
#[cfg(target_os = "solana")]
278+
#[no_mangle]
279+
#[panic_handler]
280+
fn handler(info: &core::panic::PanicInfo<'_>) -> ! {
281+
if let Some(location) = info.location() {
282+
unsafe {
283+
$crate::syscalls::sol_panic_(
284+
location.file().as_ptr(),
285+
location.file().len() as u64,
286+
location.line() as u64,
287+
location.column() as u64,
288+
)
289+
}
290+
} else {
291+
// Panic reporting.
292+
$crate::log::sol_log("** PANICKED **");
293+
unsafe { $crate::syscalls::abort() }
294+
}
295+
}
296+
};
297+
}
298+
265299
/// Default global allocator.
266300
///
267301
/// This macro sets up a default global allocator that uses a bump allocator to allocate memory.

sdk/pinocchio/src/syscalls.rs

+2
Original file line numberDiff line numberDiff line change
@@ -75,6 +75,8 @@ define_syscall!(fn sol_get_epoch_rewards_sysvar(addr: *mut u8) -> u64);
7575
define_syscall!(fn sol_poseidon(parameters: u64, endianness: u64, vals: *const u8, val_len: u64, hash_result: *mut u8) -> u64);
7676
define_syscall!(fn sol_remaining_compute_units() -> u64);
7777
define_syscall!(fn sol_alt_bn128_compression(op: u64, input: *const u8, input_size: u64, result: *mut u8) -> u64);
78+
define_syscall!(fn abort() -> !);
79+
define_syscall!(fn sol_panic_(filename: *const u8, filename_len: u64, line: u64, column: u64) -> !);
7880

7981
#[cfg(target_feature = "static-syscalls")]
8082
pub const fn sys_hash(name: &str) -> usize {

0 commit comments

Comments
 (0)