### III. Timer Interrupts

#### Hardware timer

- □ can be set to expire after specified delay (time or instructions)
- I when it does, control is passed back to the kernel
- Other interrupts (e.g. I/O completion) also give control to kernel

### Interrupt Management



interrupt controller

#### Maskable interrupts

a can be turned off by the CPU for critical processing

#### Nonmaskable interrupts

indicate serious errors (power out warning, unrecoverable memory error, etc.)

### Interrupt Management



Interrupt controllers implements interrupt priorities:

interrupt

controller

- □ Interrupts include descriptor of interrupting device
- D Priority selector circuit examines all interrupting devices, reports highest level to the CPU
- □ Controller can also buffer interrupts coming from different devices

▶ more on this later...

interrupt

# Types of Interrupts

#### Exceptions

- process missteps (e.q. division by zero)
- attempt to perform a privileged instruction
- □ sometime on purpose! (breakpoints)
- synchronous/non-maskable

#### System calls/traps

⊘ user program requests OS service

synchronous/non-

maskable

#### Interrupts

HW device requires OS service

□ timer, I/O device, interprocessor

#### asynchronous/maskable

### Interrupt Handling

#### Two objectives

- $\square$  handle the interrupt and remove the cause
- **n** restore what was running before the interrupt
  - ▶ state may have been modified on purpose
- Two "actors" in handling the interrupt
  - □ the hardware goes first
  - □ the kernel code takes control by running the interrupt handler

# Handling Interrupts: HW

- On interrupt, hardware:
  - $\square$  sets supervisor mode (if not set already)
  - □ disable (masks) interrupts
  - pushes PC, SP, and PSW
- (partially privileged)
- of user program on interrupt stack
- kernel interrupts Condition codes
- $\square$  sets PC to point to the first instruction of the appropriate interrupt handler Interrupt Vector
  - ▷ depends on interrupt type
  - ▶ interrupt handler specified in interrupt vector loaded at boot time

I/O interrupt handler System Call handler Page fault handler

### A Tale of Two Stack Pointers

- Interrupt is a program: it needs a stack!
  - □ so, each process has two stacks pointers:
    - ▷ one when running in kernel mode
    - ▶ another when running in user mode
- Why not using the user-level stack pointer?
  - user SP may be badly aligned or pointing to non writable memory
  - $\square$  user stack may not be large enough, and may spill to overwrite important data
  - $\square$  security:
    - ▶ kernel could leave sensitive data on stack
    - ▶ pointing SP to kernel address could corrupt kernel

# Handling Interrupts: SW

- The wear of the wear of the second terms of terms of
  - IH first pushes the registers' contents on the interrupt stack (part of the PCB)
    - ▶ need registers to run the IH
    - ▶ only saves necessary registers (that's why done in SW, not HW)

# Typical Interrupt Handler Code

#### HandleInterruptX:

PUSH %Rn ... PUSH %R1

only need to save registers not saved by the handler function

CALL \_handleX

POP %R1 ... POP %Rn

restore the registers saved above

RETURN\_FROM\_INTERRUPT

# Starting a new process: the recipe

1. Allocate & initialize PCB

- 2. Setup initial page table (to initialize a new address space)
- 3. Load program intro address space
- 4. Allocate user-level and kernel-level stacks.

5. Copy arguments (if any) to the base of the user-level stack

6. Simulate an interrupt

a) push initial PC, user SP

b)push PSW (supervisor mode off, interrupts enabled)

7. Clear all other registers

8.RETURN\_FROM\_INTERRUPT

# Returning from an Interrupt

- Hardware pops PC, SP, PSW
- Ø Depending on content of PSW
  - $\square$  switch to user mode
  - 🗆 enable interrupts
- From exception and system call, increment PC on return (we don't want to execute again the same instruction)
  - on exception, handler changes PC at the base of the stack
  - on system call, increment is done by hw when saving user level state

# Interrupt Handling



# Interrupt Handling on x86



# Interrupt Handling on x86



#### Interrupt Handling on x86 User-level Registers Kernel Process Code Code SSIESP foo() { handler() { CS:EIP while(...) { pusha EFLAGS x = x+1;y = y-2Other SS:ESP CS:EIP **Registers:** EFLAGS EAX. EBX. Stack Interrupt Stack Hardware performs these steps 1. Change mode bit 2. Disable interrupts 3. Save key registers to temporary location 4. Switch onto the kernel interrupt stack 5. Push key registers onto new stack

#### on x86 User-level Registers Kernel Process Code Code SS:ESP foo() { handler() { CS:EIP while(...) { pusha EFLAGS x = x+1;y = y-2Other **Registers:** EAX. EBX. Stack Interrupt Stack SS:ESP Hardware performs these steps CS:EIP EFLAGS 1. Change mode bit 2. Disable interrupts 3. Save key registers to temporary location 4. Switch onto the kernel interrupt stack 5. Push key registers onto new stack 6. Save error code (optional)

Interrupt Handling

# Interrupt Handling on x86



# Interrupt Handling on x86



#### Interrupt Handling on x86 User-level Registers Kernel Process Code Code SS:ESP foo() { → handler() { CS:EIP while(...) { pusha EFLAGS x = x+1;y = y-2Other **Registers:** EAX, EBX, Interrupt Stack Stack Hardware performs these steps 1. Change mode bit EFLAGS 2. Disable interrupts 3. Save key registers to temporary location 4. Switch onto the kernel interrupt stack 5. Push key registers onto new stack Software (handler) performs this step 6. Save error code (optional) 8. Handler Rushes all repisters an stack

| In                                                    | terrupt Ho                                                                                                                                                                                                                                                                                                                  |                         |
|-------------------------------------------------------|-----------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|-------------------------|
|                                                       | on x86                                                                                                                                                                                                                                                                                                                      | ,<br>)                  |
| er-level<br>rocess                                    | Registers                                                                                                                                                                                                                                                                                                                   | Kernel                  |
| nde<br>o() {<br>while() {<br>x = x+1;<br>y = y-2<br>} | SS:ESP<br>CS:EIP<br>EFLAGS<br>Other<br>Registers:<br>EAX, EBX,                                                                                                                                                                                                                                                              | Code<br>handler() {<br> |
|                                                       | Hardware performs these steps<br>1. Change mode bit<br>2. Disable interrupts<br>3. Save key registers to temporary location<br>4. Switch onto the kernel interrupt stack<br>5. Push key registers onto new stack<br>6. Save error code (optional)<br>7. Transfer control to interrupt handler<br>Software (handler) perform | Interrupt Stack         |
|                                                       | 8. Handler pushes all register                                                                                                                                                                                                                                                                                              | rs on stack             |

Use

PI

Co

Sta

### Interrupt Safety

- Kernel should disable device interrupts as little as possible
  - □ interrupts are best serviced quickly
- Thus, device interrupts are often disabled selectively
  - e.g., clock interrupts enabled during disk interrupt handling
- This leads to potential "race conditions"
  - system's behavior depends on timing of uncontrollable events

# Making code interrupt-safe

- Make sure interrupts are disabled while accessing mutable data!
- ø But don't we have locks?
  - □ Consider void function ()

{ lock(mtx); /\* code \*/ unlock(mtx);

Is function thread-safe? Operates correctly when accessed simultaneously by multiple threads To make it so, grab a lock Is function interrupt-safe? Operates correctly when called again (re-entered) before it completes To make it so, disable interrupts

### Interrupt Race Example

- Disk interrupt handler enqueues a task to be executed after a particular time
  - 🛛 while clock interrupts are enabled
- Clock interrupt handler checks queue for tasks to be executed
  - □ may remove tasks from the queue
- O Clock interrupt may happen during enqueue
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O
   O

Concurrent access to a shared data structure (the queue!)

# Example of Interrupt-Safe Code

void enqueue(struct task \*task) {
 int level = interrupt\_disable();
 /\* update queue \*/
 interrupt\_restore(level);

Why not simply re-enable interrupts?

- Say we did. What if then we call enqueue from code that expects interrupts to be disabled?
   > Oops...
- Instead, remember interrupt level at time of call; when done, restore that level

# Many Standard C Functions are not Interrupt-Safe

- Pure system calls are interrupt-safe
  - 🗆 e.g., read(), write(), etc.
- Functions that don't use global data are interrupt-safe
  - □ e.g., strlen(), strcpy(), etc.
- malloc(), free (), and printf() are not interrupt-safe
  - must disable interrupts before using it in an interrupt handler
  - □ and you may not want to anyway (printf() is huge!)

But they

are all

thread-safe!



# System calls

- Programming interface to the services the OS provides:
  - 🗆 read input/write to screen
  - □ create/read/write/delete files
  - 🗆 create new processes
  - □ send/receive network packets
  - 🛛 get the time / set alarms
  - □ terminate current process
  - Ω...

# Executing a System Call

#### Ø Process:

- 🛛 Calls system call function in library
- D Places arguments in registers and/or pushes them onto user stack
- D Places syscall type in a dedicated register
- **Executes syscall** machine instruction
- 🛛 Kernel
  - Executes syscall interrupt handler
  - $\ensuremath{\square}$  Places result in dedicated register
  - Executes RETURN\_FROM\_INTERRUPT
- Ø Process:
  - Executes RETURN\_FROM\_FUNCTION

# Executing read System Call



note: interrupt stack is empty while process running

# Executing read System Call



# Executing read System Call



- USP: user stack pointer
- KSP: kernel stack pointer
  - note: interrupt stack is empty while process running

# Executing read System Call





# Executing read System Call



# Executing read System Call



# Executing read System Call



# Executing read System Call



# What if read needs to block?

- read may need to block if
  - 🗆 It reads from a terminal
  - $\hfill\square$  It reads from disk, and block is not in cache
  - 🛛 It reads from a remote file server

We should run another process!