How to Continue Execution After Int3

Returning from Exceptions

No longer updated! You are viewing the a post of the first edition of "Writing an OS in Rust", which is no longer updated. You can find the second edition here.

In this post, we learn how to return from exceptions correctly. In the course of this, we will explore the iretq instruction, the C calling convention, multimedia registers, and the red zone.

As always, the complete source code is on GitHub. Please file issues for any problems, questions, or improvement suggestions. There is also a gitter chat and a comment section at the end of this page.

Note: This post describes how to handle exceptions using naked functions (see "Handling Exceptions with Naked Functions" for an overview). Our new way of handling exceptions can be found in the "Handling Exceptions" post.

🔗Introduction

Most exceptions are fatal and can't be resolved. For example, we can't return from a divide-by-zero exception in a reasonable way. However, there are some exceptions that we can resolve:

Imagine a system that uses memory mapped files: We map a file into the virtual address space without loading it into memory. Whenever we access a part of the file for the first time, a page fault occurs. However, this page fault is not fatal. We can resolve it by loading the corresponding page from disk into memory and setting the present flag in the page table. Then we can return from the page fault handler and restart the failed instruction, which now successfully accesses the file data.

Memory mapped files are completely out of scope for us right now (we have neither a file concept nor a hard disk driver). So we need an exception that we can resolve easily so that we can return from it in a reasonable way. Fortunately, there is an exception that needs no resolution at all: the breakpoint exception.

🔗The Breakpoint Exception

The breakpoint exception is the perfect exception to test our upcoming return-from-exception logic. Its only purpose is to temporary pause a program when the breakpoint instruction int3 is executed.

The breakpoint exception is commonly used in debuggers: When the user sets a breakpoint, the debugger overwrites the corresponding instruction with the int3 instruction so that the CPU throws the breakpoint exception when it reaches that line. When the user wants to continue the program, the debugger replaces the int3 instruction with the original instruction again and continues the program. For more details, see the How debuggers work series.

For our use case, we don't need to overwrite any instructions (it wouldn't even be possible since we set the page table flags to read-only). Instead, we just want to print a message when the breakpoint instruction is executed and then continue the program.

🔗Catching Breakpoints

Let's start by defining a handler function for the breakpoint exception:

                      // in src/interrupts/mod.rs                                                extern                        "C"                        fn                        breakpoint_handler(stack_frame:                        &            ExceptionStackFrame) ->                        !                        {                                                let                          stack_frame =                        unsafe                        {                        &            *stack_frame };                                      println!(            "            \n            EXCEPTION: BREAKPOINT at                        {:#x}            \n            {:#?}            "            ,                                      stack_frame.instruction_pointer, stack_frame);                                                loop                        {}                        }                              

We print an error message and also output the instruction pointer and the rest of the stack frame. Note that this function does not return yet, since our handler! macro still requires a diverging function.

We need to register our new handler function in the interrupt descriptor table (IDT):

                      // in src/interrupts/mod.rs                                                lazy_static! {                                                static ref                        IDT            : idt::Idt = {                                                let mut                          idt = idt::Idt::new();                                                              idt.set_handler(            0            , handler!(divide_by_zero_handler));                                      idt.set_handler(            3            , handler!(breakpoint_handler));                        // new                                      idt.set_handler(            6            , handler!(invalid_opcode_handler));                                      idt.set_handler(            14            , handler_with_error_code!(page_fault_handler));                                                              idt                                      };                        }                              

We set the IDT entry with number 3 since it's the vector number of the breakpoint exception.

🔗Testing it

In order to test it, we insert an int3 instruction in our rust_main:

                      // in src/lib.rs                        ...                        #[macro_use]                        // needed for the `int!` macro                        extern crate                          x86_64;                        ...                                                #[no_mangle]                        pub extern                        "C"                        fn                        rust_main(...) {                                                ...                                      interrupts::init();                                                                        // trigger a breakpoint exception                                                unsafe                        { int!(            3            ) };                                                              println!(            "It did not crash!"            );                                                loop                        {}                        }                              

When we execute make run, we see the following:

QEMU showing EXCEPTION: BREAKPOINT at 0x110970 and a dump of the exception stack frame

It works! Now we "just" need to return from the breakpoint handler somehow so that we see the It did not crash message again.

🔗Returning from Exceptions

So how do we return from exceptions? To make it easier, we look at a normal function return first:

function stack frame

When calling a function, the call instruction pushes the return address on the stack. When the called function is finished, it can return to the parent function through the ret instruction, which pops the return address from the stack and then jumps to it.

The exception stack frame, in contrast, looks a bit different:

exception stack frame

Instead of pushing a return address, the CPU pushes the stack and instruction pointers (with their segment descriptors), the RFLAGS register, and an optional error code. It also aligns the stack pointer to a 16 byte boundary before pushing values.

So we can't use a normal ret instruction, since it expects a different stack frame layout. Instead, there is a special instruction for returning from exceptions: iretq.

🔗The iretq Instruction

The iretq instruction is the one and only way to return from exceptions and is specifically designed for this purpose. The AMD64 instruction manual (PDF) even demands that iretq "must be used to terminate the exception or interrupt handler associated with the exception".

IRETQ restores rip, cs, rflags, rsp, and ss from the values saved on the stack and thus continues the interrupted program. The instruction does not handle the optional error code, so it must be popped from the stack before.

We see that iretq treats the stored instruction pointer as return address. For most exceptions, the stored rip points to the instruction that caused the fault. So by executing iretq, we restart the failing instruction. This makes sense because we should have resolved the exception when returning from it, so the instruction should no longer fail (e.g. the accessed part of the memory mapped file is now present in memory).

The situation is a bit different for the breakpoint exception, since it needs no resolution. Restarting the int3 instruction wouldn't make sense, since it would cause a new breakpoint exception and we would enter an endless loop. For this reason the hardware designers decided that the stored rip should point to the next instruction after the int3 instruction.

Let's check this for our breakpoint handler. Remember, the handler printed the following message (see the image above):

                      EXCEPTION: BREAKPOINT at 0x110970                              

So let's disassemble the instruction at 0x110970 and its predecessor:

                      > objdump -d build/kernel-x86_64.bin                        |                        grep -B1                        "110970:"                        11096f:	cc                   	int3                        110970:	48 c7 01 2a 00 00 00 	movq   $0x2a,(%rcx)                              

We see that 0x110970 indeed points to the next instruction after int3. So we can simply jump to the stored instruction pointer when we want to return from the breakpoint exception.

🔗Implementation

Let's update our handler! macro to support non-diverging exception handlers:

                      // in src/interrupts/mod.rs                                                macro_rules! handler {                                      ($name: ident)                        =>                        {{                                      #[naked]                                                extern                        "C"                        fn                        wrapper() ->                        !                        {                                                unsafe                        {                                      asm!(            "mov rdi, rsp                                      sub rsp, 8 // align the stack pointer                                      call $0"                                      ::                        "i"            ($name                        as extern                        "C"                        fn            (                                                &            ExceptionStackFrame))                        // no longer diverging                                      :                        "rdi"                        :                        "intel"            ,                        "volatile"            );                                                                        // new                                      asm!(            "add rsp, 8 // undo stack pointer alignment                                      iretq"                                      ::::                        "intel"            ,                        "volatile"            );                                      ::core::intrinsics::unreachable();                                      }                                      }                                      wrapper                                      }}                        }                              

When an exception handler returns from the call instruction, we use the iretq instruction to continue the interrupted program. Note that we need to undo the stack pointer alignment before, so that rsp points to the end of the exception stack frame again.

We've changed the handler function type, so we need to adjust our existing exception handlers:

                      // in src/interrupts/mod.rs                                                extern "C" fn divide_by_zero_handler(                        -   stack_frame: &ExceptionStackFrame) -> ! {...}                        +   stack_frame: &ExceptionStackFrame) {...}                                                extern "C" fn invalid_opcode_handler(                        -   stack_frame: &ExceptionStackFrame) -> ! {...}                        +   stack_frame: &ExceptionStackFrame) {...}                                                extern "C" fn breakpoint_handler(                        -   stack_frame: &ExceptionStackFrame) -> ! {                        +   stack_frame: &ExceptionStackFrame) {                                      println!(...);                        -   loop {}                        }                              

Note that we also removed the loop {} at the end of our breakpoint_handler so that it no longer diverges. The divide_by_zero_handler and the invalid_opcode_handler still diverge (albeit the new function type would allow a return).

🔗Testing

Let's try our new iretq logic:

QEMU output with EXCEPTION BREAKPOINT and EXCEPTION PAGE FAULT but no It did not crash

Instead of the expected "It did not crash" message after the breakpoint exception, we get a page fault. The strange thing is that our kernel tried to access address 0x1, which should never happen. So it seems like we messed up something important.

🔗Debugging

Let's debug it using GDB. For that we execute make debug in one terminal (which starts QEMU with the -s -S flags) and then make gdb (which starts and connects GDB) in a second terminal. For more information about GDB debugging, check out our Set Up GDB guide.

First we want to check if our iretq was successful. Therefore we set a breakpoint on the println!("It did not crash line!") statement in src/lib.rs. Let's assume that it's on line 61:

                      (gdb) break blog_os/src/lib.rs:61                        Breakpoint 1 at 0x110a95: file /home/.../blog_os/src/lib.rs, line 61.                              

This line is after the int3 instruction, so we know that the iretq succeeded when the breakpoint is hit. To test this, we continue the execution:

                      (gdb) continue                        Continuing.                                                Breakpoint 1, blog_os::rust_main (multiboot_information_address=1539136)                                      at /home/.../blog_os/src/lib.rs:61                        61	    println!("It did not crash!");                                                      

It worked! So our kernel successfully returned from the int3 instruction, which means that the iretq itself works.

However, when we continue the execution again, we get the page fault. So the exception occurs somewhere in the println logic. This means that it occurs in code generated by the compiler (and not e.g. in inline assembly). But the compiler should never access 0x1, so how is this happening?

The answer is that we've used the wrong calling convention for our exception handlers. Thus, we violate some compiler invariants so that the code that works fine without intermediate exceptions starts to violate memory safety when it's executed after a breakpoint exception.

🔗Calling Conventions

Exceptions are quite similar to function calls: The CPU jumps to the first instruction of the (handler) function and executes the function. Afterwards, if the function is not diverging, the CPU jumps to the return address and continues the execution of the parent function.

However, there is a major difference between exceptions and function calls: A function call is invoked voluntary by a compiler inserted call instruction, while an exception might occur at any instruction. In order to understand the consequences of this difference, we need to examine function calls in more detail.

Calling conventions specify the details of a function call. For example, they specify where function parameters are placed (e.g. in registers or on the stack) and how results are returned. On x86_64 Linux, the following rules apply for C functions (specified in the System V ABI):

  • the first six integer arguments are passed in registers rdi, rsi, rdx, rcx, r8, r9
  • additional arguments are passed on the stack
  • results are returned in rax and rdx

Note that Rust does not follow the C ABI (in fact, there isn't even a Rust ABI yet). So these rules apply only to functions declared as extern "C" fn.

🔗Preserved and Scratch Registers

The calling convention divides the registers in two parts: preserved and scratch registers.

The values of the preserved register must remain unchanged across function calls. So a called function (the "callee") is only allowed to overwrite these registers if it restores their original values before returning. Therefore these registers are called "callee-saved". A common pattern is to save these registers to the stack at the function's beginning and restore them just before returning.

In contrast, a called function is allowed to overwrite scratch registers without restrictions. If the caller wants to preserve the value of a scratch register across a function call, it needs to backup and restore it (e.g. by pushing it to the stack before the function call). So the scratch registers are caller-saved.

On x86_64, the C calling convention specifies the following preserved and scratch registers:

preserved registers scratch registers
rbp, rbx, rsp, r12, r13, r14, r15 rax, rcx, rdx, rsi, rdi, r8, r9, r10, r11
callee-saved caller-saved

The compiler knows these rules, so it generates the code accordingly. For example, most functions begin with a push rbp, which backups rbp on the stack (because it's a callee-saved register).

🔗The Exception Calling Convention

In contrast to function calls, exceptions can occur on any instruction. In most cases we don't even know at compile time if the generated code will cause an exception. For example, the compiler can't know if an instruction causes a stack overflow or an other page fault.

Since we don't know when an exception occurs, we can't backup any registers before. This means that we can't use a calling convention that relies on caller-saved registers for our exception handlers. But we do so at the moment: Our exception handlers are declared as extern "C" fn and thus use the C calling convention.

So here is what happens:

  • rust_main is executing; it writes some memory address into rax.
  • The int3 instruction causes a breakpoint exception.
  • Our breakpoint_handler prints to the screen and assumes that it can overwrite rax freely (since it's a scratch register). Somehow the value 0 ends up in rax.
  • We return from the breakpoint exception using iretq.
  • rust_main continues and accesses the memory address in rax.
  • The CPU tries to access address 0x1, which causes a page fault.

So our exception handler erroneously assumes that the scratch registers were saved by the caller. But the caller (rust_main) couldn't save any registers since it didn't know that an exception occurs. So nobody saves rax and the other scratch registers, which leads to the page fault.

The problem is that we use a calling convention with caller-saved registers for our exception handlers. Instead, we need a calling convention means that preserves all registers. In other words, all registers must be callee-saved:

                      extern                        "all-registers-callee-saved"                        fn                        exception_handler() {            ...            }                              

Unfortunately, Rust does not support such a calling convention. It was proposed once, but did not get accepted for various reasons. The primary reason was that such calling conventions can be simulated by writing a naked wrapper function.

(Remember: Naked functions are functions without prologue and can contain only inline assembly. They were discussed in the previous post.)

🔗A naked wrapper function

Such a naked wrapper function might look like this:

                      #[naked]                        extern                        "C"                        fn                        calling_convention_wrapper() {                                                unsafe                        {                                      asm!(            "                                      push rax                                      push rcx                                      push rdx                                      push rsi                                      push rdi                                      push r8                                      push r9                                      push r10                                      push r11                                      // TODO: call exception handler with C calling convention                                      pop r11                                      pop r10                                      pop r9                                      pop r8                                      pop rdi                                      pop rsi                                      pop rdx                                      pop rcx                                      pop rax                                      "                        ::::                        "intel"            ,                        "volatile"            );                                      }                        }                              

This wrapper function saves all scratch registers to the stack before calling the exception handler and restores them afterwards. Note that we pop the registers in reverse order.

We don't need to backup preserved registers since they are callee-saved in the C calling convention. Thus, the compiler already takes care of preserving their values.

🔗Fixing our Handler Macro

Let's update our handler macro to fix the calling convention problem. Therefore we need to backup and restore all scratch registers. For that we create two new macros:

                      // in src/interrupts/mod.rs                                                macro_rules! save_scratch_registers {                                      ()                        =>                        {                                      asm!(            "push rax                                      push rcx                                      push rdx                                      push rsi                                      push rdi                                      push r8                                      push r9                                      push r10                                      push r11                                      "                        ::::                        "intel"            ,                        "volatile"            );                                      }                        }                                                macro_rules! restore_scratch_registers {                                      ()                        =>                        {                                      asm!(            "pop r11                                      pop r10                                      pop r9                                      pop r8                                      pop rdi                                      pop rsi                                      pop rdx                                      pop rcx                                      pop rax                                      "                        ::::                        "intel"            ,                        "volatile"            );                                      }                        }                              

We need to declare these macros above our handler macro, since macros are only available after their declaration.

Now we can use these macros to fix our handler! macro:

                      // in src/interrupts/mod.rs                                                macro_rules! handler {                                      ($name: ident)                        =>                        {{                                      #[naked]                                                extern                        "C"                        fn                        wrapper() ->                        !                        {                                                unsafe                        {                                      save_scratch_registers!();                                      asm!(            "mov rdi, rsp                                      add rdi, 9*8 // calculate exception stack frame pointer                                      // sub rsp, 8 (stack is aligned already)                                      call $0"                                      ::                        "i"            ($name                        as                                                extern                        "C"                        fn            (            &            ExceptionStackFrame))                                      :                        "rdi"                        :                        "intel"            ,                        "volatile"            );                                                              restore_scratch_registers!();                                      asm!(            "                                      // add rsp, 8 (undo stack alignment; not needed anymore)                                      iretq"                                      ::::                        "intel"            ,                        "volatile"            );                                      ::core::intrinsics::unreachable();                                      }                                      }                                      wrapper                                      }}                        }                              

It's important that we save the registers first, before we modify any of them. After the call instruction (but before iretq) we restore the registers again. Because we're now changing rsp (by pushing the register values) before we load it into rdi, we would get a wrong exception stack frame pointer. Therefore we need to adjust it by adding the number of bytes we push. We push 9 registers that are 8 bytes each, so 9 * 8 bytes in total.

Note that we no longer need to manually align the stack pointer, because we're pushing an uneven number of registers in save_scratch_registers. Thus the stack pointer already has the required 16-byte alignment.

🔗Testing it again

Let's test it again with our corrected handler! macro:

QEMU output with EXCEPTION BREAKPOINT and It did not crash

The page fault is gone and we see the "It did not crash" message again!

So the page fault occurred because our exception handler didn't preserve the scratch register rax. Our new handler! macro fixes this problem by saving all scratch registers (including rax) before calling exception handlers. Thus, rax still contains the valid memory address when rust-main continues execution.

When we discussed calling conventions above, we assumed that a x86_64 CPU only has the following 16 registers: rax, rbx, rcx, rdx, rsi, rdi, rsp, rbp, r8, r9, r10, r11.r12, r13, r14, and r15. These registers are called general purpose registers since each of them can be used for arithmetic and load/store instructions.

However, modern CPUs also have a set of special purpose registers, which can be used to improve performance in several use cases. On x86_64, the most important set of special purpose registers are the multimedia registers. These registers are larger than the general purpose registers and can be used to speed up audio/video processing or matrix calculations. For example, we could use them to add two 4-dimensional vectors in a single CPU instruction:

(1,2,3,4) + (5,6,7,8) = (6,8,10,12)

Such multimedia instructions are called Single Instruction Multiple Data (SIMD) instructions, because they simultaneously perform an operation (e.g. addition) on multiple data words. Good compilers are able to transform normal loops into such SIMD code automatically. This process is called auto-vectorization and can lead to huge performance improvements.

However, auto-vectorization causes a problem for us: Most of the multimedia registers are caller-saved. According to our discussion of calling conventions above, this means that our exception handlers erroneously assume that they are allowed to overwrite them without preserving their values.

We don't use any multimedia registers explicitly, but the Rust compiler might auto-vectorize our code (including the exception handlers). Thus we could silently clobber the multimedia registers, which leads to the same problems as above:

example: program uses mm0, mm1, and mm2. Then the exception handler clobbers mm1.

This example shows a program that is using the first three multimedia registers (mm0 to mm2). At some point, an exception occurs and control is transferred to the exception handler. The exception handler uses mm1 for its own data and thus overwrites the previous value. When the exception is resolved, the CPU continues the interrupted program again. However, the program is now corrupt since it relies on the original mm1 value.

🔗Saving and Restoring Multimedia Registers

In order to fix this problem, we need to backup all caller-saved multimedia registers before we call the exception handler. The problem is that the set of multimedia registers varies between CPUs. There are different standards:

  • MMX: The MMX instruction set was introduced in 1997 and defines eight 64 bit registers called mm0 through mm7. These registers are just aliases for the registers of the x87 floating point unit.
  • SSE: The Streaming SIMD Extensions instruction set was introduced in 1999. Instead of re-using the floating point registers, it adds a completely new register set. The sixteen new registers are called xmm0 through xmm15 and are 128 bits each.
  • AVX: The Advanced Vector Extensions are extensions that further increase the size of the multimedia registers. The new registers are called ymm0 through ymm15 and are 256 bits each. They extend the xmm registers, so e.g. xmm0 is the lower (or upper?) half of ymm0.

The Rust compiler (and LLVM) assume that the x86_64-unknown-linux-gnu target supports only MMX and SSE, so we don't need to save the ymm0 through ymm15. But we need to save xmm0 through xmm15 and also mm0 through mm7. There is a special instruction to do this: fxsave. This instruction saves the floating point and multimedia state to a given address. It needs 512 bytes to store that state.

In order to save/restore the multimedia registers, we could add new macros:

                      macro_rules! save_multimedia_registers {                                      ()                        =>                        {                                      asm!(            "sub rsp, 512                                      fxsave [rsp]                                      "                        ::::                        "intel"            ,                        "volatile"            );                                      }                        }                                                macro_rules! restore_multimedia_registers {                                      ()                        =>                        {                                      asm!(            "fxrstor [rsp]                                      add rsp, 512                                      "                        ::::                        "intel"            ,                        "volatile"            );                                      }                        }                              

First, we reserve the 512 bytes on the stack and then we use fxsave to backup the multimedia registers. In order to restore them later, we use the fxrstor instruction. Note that fxsave and fxrstor require a 16 byte aligned memory address.

However, we won't do it that way. The problem is the large amount of memory required. We will reuse the same code when we handle hardware interrupts in a future post. So for each mouse click, pressed key, or arrived network package we need to write 512 bytes to memory. This would be a huge performance problem.

Fortunately, there exists an alternative solution.

🔗Disabling Multimedia Extensions

We just disable MMX, SSE, and all the other fancy multimedia extensions in our kernel1. This way, our exception handlers won't clobber the multimedia registers because they won't use them at all.

This solution has its own disadvantages, of course. For example, it leads to slower kernel code because the compiler can't perform any auto-vectorization optimizations. But it's still the faster solution (since we save many memory accesses) and most kernels do it this way (including Linux).

So how do we disable MMX and SSE? Well, we just tell the compiler that our target system doesn't support it. Since the very beginning, we're compiling our kernel for the x86_64-unknown-linux-gnu target. This worked fine so far, but now we want a different target without support for multimedia extensions. We can do so by creating a target configuration file.

🔗Target Specifications

In order to disable the multimedia extensions for our kernel, we need to compile for a custom target. We want a target that is equal to x86_64-unknown-linux-gnu, but without MMX and SSE support. Rust allows us to specify such a target using a JSON configuration file.

A minimal target specification that describes the x86_64-unknown-linux-gnu target looks like this:

                      {                                                "llvm-target"            :                        "x86_64-unknown-linux-gnu"            ,                                                "data-layout"            :                        "e-m:e-i64:64-f80:128-n8:16:32:64-S128"            ,                                                "target-endian"            :                        "little"            ,                                                "target-pointer-width"            :                        "64"            ,                                                "target-c-int-width"            :                        "32"            ,                                                "arch"            :                        "x86_64"            ,                                                "os"            :                        "none"                        }                              

The llvm-target field specifies the target triple that is passed to LLVM. We want to derive a 64-bit Linux target, so we choose x86_64-unknown-linux-gnu. The data-layout field is also passed to LLVM and specifies how data should be laid out in memory. It consists of various specifications separated by a - character. For example, the e means little endian and S128 specifies that the stack should be 128 bits (= 16 byte) aligned. The format is described in detail in the LLVM documentation but there shouldn't be a reason to change this string.

The other fields are used for conditional compilation. This allows crate authors to use cfg variables to write special code for depending on the OS or the architecture. There isn't any up-to-date documentation about these fields but the corresponding source code is quite readable.

🔗Disabling MMX and SSE

In order to disable the multimedia extensions, we create a new target named x86_64-blog_os. To describe this target, we create a file named x86_64-blog_os.json in the project root with the following content:

                      {                                                "llvm-target"            :                        "x86_64-unknown-linux-gnu"            ,                                                "data-layout"            :                        "e-m:e-i64:64-f80:128-n8:16:32:64-S128"            ,                                                "target-endian"            :                        "little"            ,                                                "target-pointer-width"            :                        "64"            ,                                                "target-c-int-width"            :                        "32"            ,                                                "arch"            :                        "x86_64"            ,                                                "os"            :                        "none"            ,                                                "features"            :                        "-mmx,-sse"                        }                              

It's equal to x86_64-unknown-linux-gnu target but has one additional option: "features": "-mmx,-sse". So we added two target features: -mmx and -sse. The minus prefix defines that our target does not support this feature. So by specifying -mmx and -sse, we disable the default mmx and sse features.

In order to compile for the new target, we need to adjust our Makefile:

                      # in `Makefile`                                                              arch ?= x86_64                        -target ?= $(arch)-unknown-linux-gnu                        +target ?= $(arch)-blog_os                        ...                              

The new target name (x86_64-blog_os) is the file name of the JSON configuration file without the .json extension.

🔗Cross compilation

Let's try if our kernel still works with the new target:

                      > make run                        Compiling raw-cpuid v2.0.1                        Compiling rlibc v0.1.5                        Compiling x86 v0.7.1                        Compiling spin v0.3.5                        error[E0463]: can't find crate for `core`                                                error: aborting due to previous error                                                Build failed, waiting for other jobs to finish...                        ...                        Makefile:52: recipe for target 'cargo' failed                        make: *** [cargo] Error 101                              

It doesn't compile anymore. The error tells us that the Rust compiler no longer finds the core library.

The core library is implicitly linked to all no_std crates and contains things such as Result, Option, and iterators. We've used that library without problems since the very beginning, so why is it no longer available?

The problem is that the core library is distributed together with the Rust compiler as a precompiled library. So it is only valid for the host triple, which is x86_64-unknown-linux-gnu in our case. If we want to compile code for other targets, we need to recompile core for these targets first.

🔗Xargo

That's where xargo comes in. It is a wrapper for cargo that eases cross compilation. We can install it by executing:

                      cargo install xargo                              

Xargo depends on the rust source code, which we can install with rustup component add rust-src.

Xargo is "a drop-in replacement for cargo", so every cargo command also works with xargo. You can do e.g. xargo --help, xargo clean, or xargo doc. However, the build command gains additional functionality: xargo build will automatically cross compile the core library when compiling for custom targets.

That's exactly what we want, so we change one letter in our Makefile:

                      # in `Makefile`                        ...                                                cargo:                        -	@cargo build --target $(target)                        +	@xargo build --target $(target)                        ...                              

Now the build goes through xargo, which should fix the compilation error. Let's try it out:

                      > make run                        Compiling core v0.0.0 (file:///home/…/.rustup/toolchains/nightly-x86_64-unknown-linux-gnu/lib/rustlib/src/rust/src/libcore)                        LLVM ERROR: SSE register return with SSE disabled                        error: Could not compile `core`.                              

Well, we get a different error now, so it seems like we're making progress :). It seems like there is a "SSE register return" although SSE is disabled. But what's an "SSE register return"?

🔗SSE Register Return

Remember when we discussed calling conventions above? The calling convention defines which registers are used for return values. Well, the System V ABI defines that xmm0 should be used for returning floating point values. So somewhere in the core library a function returns a float and LLVM doesn't know what to do. The ABI says "use xmm0" but the target specification says "don't use xmm registers".

In order to fix this problem, we need to change our float ABI. The idea is to avoid normal hardware-supported floats and use a pure software implementation instead. We can do so by enabling the soft-float feature for our target. For that, we edit x86_64-blog_os.json:

                      {                                                "llvm-target"            :                        "x86_64-unknown-linux-gnu"            ,                                                ...                                                "features"            :                        "-mmx,-sse,+soft-float"                        }                              

The plus prefix tells LLVM to enable the soft-float feature.

Let's try make run again:

                      > make run                                      Compiling core v0.0.0 (file:///…/.rustup/toolchains/nightly-x86_64-unknown-linux-gnu/lib/rustlib/src/rust/src/libcore)                                      Finished release [optimized] target(s) in 21.95 secs                                      Compiling spin v0.4.5                                      Compiling once v0.3.2                                      Compiling x86 v0.8.0                                      Compiling bitflags v0.9.1                                      Compiling raw-cpuid v2.0.1                                      Compiling rlibc v0.1.5                                      Compiling linked_list_allocator v0.2.3                                      Compiling volatile v0.1.0                                      Compiling bitflags v0.4.0                                      Compiling bit_field v0.5.0                                      Compiling spin v0.3.5                                      Compiling multiboot2 v0.1.0                                      Compiling lazy_static v0.2.2                                      Compiling hole_list_allocator v0.1.0 (file:///…/libs/hole_list_allocator)                                      Compiling blog_os v0.1.0 (file:///…)                        error[E0463]: can't find crate for `alloc`                                      --> src/lib.rs:33:1                                      |                        33 | extern crate alloc;                                      | ^^^^^^^^^^^^^^^^^^^ can't find crate                                                error: aborting due to previous error                              

We see that xargo now compiles the core crate in release mode. Then it starts the normal cargo build. Cargo then recompiles all dependencies, since it needs to generate different code for the new target.

However, the build still fails. The reason is that xargo only installs core by default, but we also need the alloc crate. We can enable it by creating a file named Xargo.toml with the following contents:

                      # Xargo.toml                                                [            target.x86_64-blog_os.dependencies            ]                        alloc                        = {}                              

Now xargo compiles alloc, too:

                      > make run                                      Compiling core v0.0.0 (file:///…/.rustup/toolchains/nightly-x86_64-unknown-linux-gnu/lib/rustlib/src/rust/src/libcore)                                      Compiling std_unicode v0.0.0 (file:///…/.rustup/toolchains/nightly-x86_64-unknown-linux-gnu/lib/rustlib/src/rust/src/libstd_unicode)                                      Compiling alloc v0.0.0 (file:///…/.rustup/toolchains/nightly-x86_64-unknown-linux-gnu/lib/rustlib/src/rust/src/liballoc)                                      Finished release [optimized] target(s) in 28.84 secs                                      Compiling blog_os v0.1.0 (file:///…/Documents/blog_os/master)                        warning: unused variable: `allocator` […]                        warning: unused variable: `frame` […]                                                              Finished debug [unoptimized + debuginfo] target(s) in 1.75 secs                              

It worked! Now we have a kernel that never touches the multimedia registers! We can verify this by executing:

                      > objdump -d build/kernel-x86_64.bin | grep "mm[0-9]"                              

If the command produces no output, our kernel uses neither MMX (mm0mm7) nor SSE (xmm0xmm15) registers.

So now our return-from-exception logic works without problems in most cases. However, there is still a pitfall hidden in the C calling convention, which might cause hideous bugs in some rare cases.

🔗The Red Zone

The red zone is an optimization of the System V ABI that allows functions to temporary use the 128 bytes below its stack frame without adjusting the stack pointer:

stack frame with red zone

The image shows the stack frame of a function with n local variables. On function entry, the stack pointer is adjusted to make room on the stack for the local variables.

The red zone is defined as the 128 bytes below the adjusted stack pointer. The function can use this area for temporary data that's not needed across function calls. Thus, the two instructions for adjusting the stack pointer can be avoided in some cases (e.g. in small leaf functions).

However, this optimization leads to huge problems with exceptions. Let's assume that an exception occurs while a function uses the red zone:

red zone overwritten by exception handler

The CPU and the exception handler overwrite the data in red zone. But this data is still needed by the interrupted function. So the function won't work correctly anymore when we return from the exception handler. It might fail or cause another exception, but it could also lead to strange bugs that take weeks to debug.

🔗Adjusting our Exception Handler?

The problem is that the System V ABI demands that the red zone "shall not be modified by signal or interrupt handlers." Our current exception handlers do not respect this. We could try to fix it by subtracting 128 from the stack pointer before pushing anything:

                      sub                        rsp,                        128                        save_scratch_registers()                        ...                        call                        ...                        ...                        restore_scratch_registers()                        add                        rsp,                        128                                                iretq                              

This will not work. The problem is that the CPU pushes the exception stack frame before even calling our handler function. So the CPU itself will clobber the red zone and there is nothing we can do about that. So our only chance is to disable the red zone.

🔗Disabling the Red Zone

The red zone is a property of our target, so in order to disable it we edit our x86_64-blog_os.json a last time:

                      {                                                "llvm-target"            :                        "x86_64-unknown-linux-gnu"            ,                                                ...                                                "features"            :                        "-mmx,-sse,+soft-float"            ,                                                "disable-redzone"            :                        true                        }                              

We add one additional option at the end: "disable-redzone": true. As you might guess, this option disables the red zone optimization.

Now we have a red zone free kernel!

🔗Exceptions with Error Codes

We're now able to correctly return from exceptions without error codes. However, we still can't return from exceptions that push an error code (e.g. page faults). Let's fix that by updating our handler_with_error_code macro:

                      // in src/interrupts/mod.rs                                                macro_rules! handler_with_error_code {                                      ($name: ident)                        =>                        {{                                      #[naked]                                                extern                        "C"                        fn                        wrapper() ->                        !                        {                                                unsafe                        {                                      asm!(            "pop rsi // pop error code into rsi                                      mov rdi, rsp                                      sub rsp, 8 // align the stack pointer                                      call $0"                                      ::                        "i"            ($name                        as extern                        "C"                        fn            (                                                &            ExceptionStackFrame,                        u64            ))                                      :                        "rdi"            ,            "rsi"                        :                        "intel"            );                                      asm!(            "iretq"                        ::::                        "intel"            ,                        "volatile"            );                                      ::core::intrinsics::unreachable();                                      }                                      }                                      wrapper                                      }}                        }                              

First, we change the type of the handler function: no more -> !, so it no longer needs to diverge. We also add an iretq instruction at the end.

Now we can make our page_fault_handler non-diverging:

                      // in src/interrupts/mod.rs                                                              extern "C" fn page_fault_handler(stack_frame: &ExceptionStackFrame,                        -   error_code: u64) -> ! { ... }                        +   error_code: u64) { ... }                              

However, now we have the same problem as above: The handler function will overwrite the scratch registers and cause bugs when returning. Let's fix this by invoking save_scratch_registers at the beginning:

                      // in src/interrupts/mod.rs                                                macro_rules! handler_with_error_code {                                      ($name: ident)                        =>                        {{                                      #[naked]                                                extern                        "C"                        fn                        wrapper() ->                        !                        {                                                unsafe                        {                                      save_scratch_registers!();                                      asm!(            "pop rsi // pop error code into rsi                                      mov rdi, rsp                                      add rdi, 10*8 // calculate exception stack frame pointer                                      sub rsp, 8 // align the stack pointer                                      call $0                                      add rsp, 8 // undo stack pointer alignment                                      "                        ::                        "i"            ($name                        as extern                        "C"                        fn            (                                                &            ExceptionStackFrame,                        u64            ))                                      :                        "rdi"            ,            "rsi"                        :                        "intel"            );                                      restore_scratch_registers!();                                      asm!(            "iretq"                        ::::                        "intel"            ,                        "volatile"            );                                      ::core::intrinsics::unreachable();                                      }                                      }                                      wrapper                                      }}                        }                              

Now we backup the scratch registers to the stack right at the beginning and restore them just before the iretq. Like in the handler macro, we now need to add 10*8 to rdi in order to get the correct exception stack frame pointer (save_scratch_registers pushes nine 8 byte registers, plus the error code). We also need to undo the stack pointer alignment after the call 2.

Now we have one last bug: We pop the error code into rsi, but the error code is no longer at the top of the stack (since save_scratch_registers pushed 9 registers on top of it). So we need to do it differently:

                      // in src/interrupts/mod.rs                                                macro_rules! handler_with_error_code {                                      ($name: ident)                        =>                        {{                                      #[naked]                                                extern                        "C"                        fn                        wrapper() ->                        !                        {                                                unsafe                        {                                      save_scratch_registers!();                                      asm!(            "mov rsi, [rsp + 9*8] // load error code into rsi                                      mov rdi, rsp                                      add rdi, 10*8 // calculate exception stack frame pointer                                      sub rsp, 8 // align the stack pointer                                      call $0                                      add rsp, 8 // undo stack pointer alignment                                      "                        ::                        "i"            ($name                        as extern                        "C"                        fn            (                                                &            ExceptionStackFrame,                        u64            ))                                      :                        "rdi"            ,            "rsi"                        :                        "intel"            );                                      restore_scratch_registers!();                                      asm!(            "add rsp, 8 // pop error code                                      iretq"                        ::::                        "intel"            ,                        "volatile"            );                                      ::core::intrinsics::unreachable();                                      }                                      }                                      wrapper                                      }}                        }                              

Instead of using pop, we're calculating the error code address manually (save_scratch_registers pushes nine 8 byte registers) and load it into rsi using a mov. So now the error code stays on the stack. But iretq doesn't handle the error code, so we need to pop it before invoking iretq.

Phew! That was a lot of fiddling with assembly. Let's test if it still works.

🔗Testing

First, we test if the exception stack frame pointer and the error code are still correct:

                      // in rust_main in src/lib.rs                                                ...                        unsafe                        { int!(            3            ) };                                                // provoke a page fault                        unsafe                        { *(            0xdeadbeaf                        as *mut u64            ) =                        42            ; }                                                println!(            "It did not crash!"            );                        loop                        {}                              

This should cause the following error message:

                      EXCEPTION: PAGE FAULT while accessing 0xdeadbeaf                        error code: CAUSED_BY_WRITE                        ExceptionStackFrame {                                      instruction_pointer: 1114753,                                      code_segment: 8,                                      cpu_flags: 2097158,                                      stack_pointer: 1171104,                                      stack_segment: 16                        }                              

The error code should still be CAUSED_BY_WRITE and the exception stack frame values should also be correct (e.g. code_segment should be 8 and stack_segment should be 16).

🔗Returning from Page Faults

Let's see what happens if we comment out the trailing loop in our page fault handler:

QEMU printing the same page fault message again and again

We see that the same error message is printed over and over again. Here is what happens:

  • The CPU executes rust_main and tries to access 0xdeadbeaf. This causes a page fault.
  • The page fault handler prints an error message and returns without fixing the cause of the exception (0xdeadbeaf is still unaccessible).
  • The CPU restarts the instruction that caused the page fault and thus tries to access 0xdeadbeaf again. Of course, this causes a page fault again.
  • The page fault handler prints the error message and returns.

… and so on. Thus, our code indefinitely jumps between the page fault handler and the instruction that accesses 0xdeadbeaf.

This is a good thing! It means that our iretq logic is working correctly, since it returns to the correct instruction every time. So our handler_with_error_code macro seems to be correct.

🔗What's next?

We are now able to catch exceptions and to return from them. However, there are still exceptions that completely crash our kernel by causing a triple fault. In the next post, we will fix this issue by handling a special type of exception: the double fault. Thus, we will be able to avoid random reboots in our kernel.



Comments (Archived)

jarretttride1971.blogspot.com

Source: https://os.phil-opp.com/returning-from-exceptions/

0 Response to "How to Continue Execution After Int3"

Post a Comment

Iklan Atas Artikel

Iklan Tengah Artikel 1

Iklan Tengah Artikel 2

Iklan Bawah Artikel