Big picture for today's lab:
-
We'll show how to setup interrupts / exceptions on the r/pi. Interrupts and exceptions are useful for responding quickly to devices, doing pre-emptive threads, handling protection faults, and other things.
-
We strip interrupts down to a fairly small amount of code. You will go over each line and get a feel for what is going on.
-
You will use timer interrupts to implement a simple but useful statistical profiler similar to
gprof
. As is often the case, because the r/pi system is so simple, so too is the code we need to write, especially as compared to on a modern OS.
The good thing about the lab is that interrupts are often made very complicated or just discussed so abstractly it's hard to understand them. Hopefully by the end of today you'll have a reasonable grasp of at least one concrete implementation. If you start kicking its tires and replacing different pieces with equivalant methods you should get a pretty firm grasp.
As you'll see over the rest of the quarter, managing multiple devices can be difficult. Either you check constantly (poll), which means most checks are fruitless. Or you do not check constantly, which means most actions have significant delay since the device has stuff for you to do much before you check it. Interrupts will allow you do to your normal actions, yet get interrupted as soon as an interesting event happens on a device you care about.
You can use interrupts to mitigate the problem of device input by telling the pi to jump to an interrupt handler (a piece of code with some special rules) when something interesting happens. An interrupt will interrupt your normal code, run, finish, and jump back.
Interrupts also allow you to not trust code to complete promptly, by giving you the ability to run it for a given amount, and then interrupt (pre-empt) it and switch to another thread of control. Operating systems and runtime systems use this ability to make a pre-emptive threads package and, later, user-level processes. The timer interrupt we do for today's lab will give you the basic framework to do this.
Traditionally interrupts are used to refer to transfers of control initiated by "external" events (such as a device or timer-expiration). These are sometimes called asynchronous events. Exceptions are more-or-less the same, except they are synchronous events triggered by the currently running thread (often by the previous or current instruction, such as a access fault, divide by zero, illegal instruction, etc. The framework we use will handle these as well; and, in fact, on the arm at a mechanical level there is little difference.
Like everything good, interrupts also have a cost: this will be our
first exposure to race conditions. If both your regular code and the
interrupt handler read / write the same variables (which for us will be
the common case) you can have race conditions where they both overwrite
each other's values (partial solution: have one reader and one writer)
or miss updates because the compiler doesn't know about interrupt handling
and so eliminates variable reads b/c it doesn't see anyone changing them
(partial solution: mark shared variables as volatile
). We'll implement
different ways to mitigate these problems over the next couple of weeks.
Turn-in:
-
timer-int
: give the shortest time between timer interrupts you can make, and two ways to make the code crash. -
Build a simple system call: show your
1-syscall
works. -
Implement
gprof
(in the2-gprof
subdirectory). You should run it and show that the results are reasonable.
Look through the code in timer-int
, compile it, run it. Make sure
you can answer the questions in the comments. We'll walk through it
in class.
One we can get timer exceptions, we (perhaps surprisingly) have enough infrastructure
to make trivial system calls. Since we are already running in supervisor mode,
these are not that useful as-is, but making them now will show how trivial
they actually are. In particular, look in 1-syscall
and write the needed code
in:
-
interrupts-asm.S
: you should be able to largely rip off the timer interrupt code to forward system call. NOTE: a huge difference is that we are already running at supervisor level, so all registers are live. You need to handle this differently. -
syscall.c
: finish the system call vector code (should just be a few lines). You want to act on system call 1 and reject all other calls with a-1
.
This doesn't take much code, but you will have to think carefully about which registers need to be saved, etc.
The nice thing about doing everything from scratch is that simple things are simple to do. We don't have to fight a big OS that can't get out of its own way.
Today's lab is a good example: implementing a statistical profiler. The basic intuition:
- Setup timer interrupts so that we get them fairly often.
- At each interrupt, record which location in the code we interrupted. (I.e., where the program counter is.)
- Over time, we will interrupt where your code spends most of its time more often.
The implementation will take about 30-40 lines of code in total. You'll build two things:
- A
kmalloc
that will allocate memory. We will not have afree
, sokmalloc
is trivial: have a pointer to where "free memory" starts, and increment a counter based on the requested size. - Use
kmalloc
to allocate an array at least as big as the code. - In the interrupt handler, use the program counter value to index into this array
and increment the associated count. NOTE: its very easy to mess up sizes. Each
instruction is 4 bytes, so you'll divide the
pc
by 4. - Periodically you can print out the non-zero values in this array along with the
pc
value they correspond to. You should be able to look in the disassembled code (gprof.list
) to see which instruction these correspond to.
Congratulations! You've built something that not many people in the Gates building know how to do.
There's lots of other things to do:
-
Mess with the timer period to get it as short as possible.
-
To access the banked registers can use load/store multiple (ldm, stm) with the caret after the register list:
stmia sp, {sp, lr}^
. -
Using a suffix for the
CPSR
andSPSR
(e.g.,SPSR_cxsf
) to specify the fields you are [modifying] (https://www.raspberrypi.org/forums/viewtopic.php?t=139856). -
New instructions
cps
,srs
,rfe
to manipulate privileged state (e.g., Section A4-113 indocs/armv6.pdf
). So you can do things likecpsie if @ Enable irq and fiq interrupts cpsid if @ ...and disable them
There's plenty to read, all put in the docs
directory in this lab:
-
If you get confused, the overview at
valvers
was useful: (http://www.valvers.com/open-software/raspberry-pi/step04-bare-metal-programming-in-c-pt4) -
We annotated the Broadcom discussion of general interrupts and timer interrupts on the pi in
7-interrupts/docs/BCM2835-ARM-timer-int.annot.pdf
. It's actually not too bad. -
We annotated the ARM discussion of registers and interrupts in
7-interrupts/docs/armv6-interrupts.annot.pdf
. -
There is also the RealView developer tool document, which has some useful worked out examples (including how to switch contexts and grab the banked registers):
7-interrupts/docs/DUI0203.pdf
. -
There are two useful lectures on the ARM instruction set. Kind of dry, but easier to get through than the arm documents:
7-interrupts/docs/Arm_EE382N_4.pdf
and7-interrupts/docs/Lecture8.pdf
.
If you find other documents that are more helpful, let us know!