



# MARNOS

M Mohammad Senan Ali  
A Aditya Rajesh Bawangade  
R Rishabh Jain  
N Nishant Kumar Singh





# ABOUT MARN OS

We initiated this project using an existing codebase for a Learning Operating System that encompasses essential features, such as basic memory management, text output, Global Descriptor Table (GDT) setup, Peripheral Interface Controller (PIC) configuration, and Hardware Abstraction Layer (HAL), among other functionalities.



# Interrupt in OS



An interrupt is a signal to the processor generated by hardware or software indicating an event that needs immediate attention. It requires the operating system (OS) to stop and figure out what to do next. An interrupt temporarily stops or terminates a service or a current process.

## Types of Interrupts:

Hardware Interrupts: Triggered by external hardware devices.

Software Interrupts: Generated by a program requiring OS intervention.

## Purpose :

Interrupts are crucial for multitasking, allowing the CPU to respond to external events while executing other tasks

# How Interrupts Work in OS



Interrupt Handling: When an interrupt occurs, the CPU stops its current tasks, saves its state, and executes a function called an Interrupt Service Routine (ISR).

Context Switching: This involves the OS saving the state of the current task and loading the state of the next task to be processed.



# Real-World Examples of Interrupts



## Divide By Zero exception:

- Definition: The Divide By Zero exception is a type of software interrupt or exception that occurs when a program attempts to divide a number by zero.
- CPU's Role: When the CPU detects a division operation with a zero divisor, it immediately generates an interrupt signal, halting the normal execution flow of the program.

# Real-World Examples of Interrupts



## User-Defined Interrupt:

Description: Triggered by user programs rather than hardware or system events.

Implementation: Typically implemented through system calls. The program issues a command, which is then interpreted by the OS as an interrupt, prompting a switch from user mode to kernel mode to execute the necessary operations.

# Real-World Examples of Interrupts



## Timer Interrupt:

Description: A Timer Interrupt is triggered by the system's hardware clock at regular intervals, preset by the operating system. Its primary function is to allow the operating system to perform regular tasks, such as updating system time, running scheduled tasks, or managing time-sharing among processes.

System Performance: The frequency of Timer Interrupts can impact overall system performance and responsiveness. A high frequency allows for more responsive task switching but can increase overhead due to frequent context switches.

# Real-World Examples of Interrupts



## Keyboard Interrupt:

Description: A Keyboard Interrupt is generated whenever a key is pressed or released on the keyboard. It signals the processor to stop its current activities and execute an Interrupt Service Routine (ISR) to handle the key press.

Input Handling: The ISR typically reads the scan code of the key pressed and either stores it in a buffer or processes it directly, depending on the OS design.

# Intel x86 Architecture



The Intel x86 architecture is a widely used and versatile computer processor architecture. It features a complex instruction set, supports both 32-bit and 64-bit modes, and provides compatibility with a wide range of software. It includes components like registers, memory addressing modes, and a variety of instructions for data manipulation and control flow.

The underlying codebase for this operating system is based on x86 Architecture.



Intel Core 2 Duo, an example of an x86-compatible, 64-bit multicore processor

# Intel x86 Architecture



1. CISC architecture
2. Registers: EAX, EBX, ECX, etc.
3. Memory Segmentation: Historical feature
4. Protected Mode: Introduced with 80286
5. 32-bit and 64-bit support
6. MMX: Multimedia SIMD instructions
7. SSE: Enhanced multimedia processing
8. Hyper-Threading: Simulates multiple processors
9. VT (Virtualization Technology): Hardware-level virtualization
10. Cache Levels: L1, L2, sometimes L3
11. Pipeline Architecture: Increases instruction throughput
12. Out-of-Order Execution: Improves efficiency
13. Power Management: Reduces energy consumption
14. FPU: Handles floating-point operations
15. Execution Units: Simultaneous instruction execution
16. AVX: Parallel data processing
17. NX Bit: Security feature
18. AES-NI: Accelerates encryption
19. Turbo Boost: Dynamic clock speed increase
20. Integrated Graphics: Some processors have it



**CISC Architecture**

# Creating the MARN OS



nanobyte\_os-Part10

EXPLORER    ...    C pic.c    C isr.c    ...    C isr.c    ASM isr.asm    ...    C isr.c    C io.h    ...    C main.c    ...

src > kernel > arch > i686 > C isr.c > [g\_Exc]

src > kernel > arch > i686 > ASM isr.asm

src > kernel > arch > i686 > C io.h > ...

src > kernel > C main.c > ...

NANOBYTE\_OS-PART10

io.asm  
io.c  
io.h  
irq.c  
irq.h  
isr.asm  
isr.c  
isr.h  
isrs\_gen.c  
isrs\_gen.inc  
pic.c  
pic.h  
hal  
hal.c  
hal.h  
util  
linker.ld  
main.c  
Makefile  
memory.c  
memory.h  
stdio.c  
stdio.h  
tools  
.gitignore  
bochs\_config  
debug.sh  
LICENSE  
Makefile  
README.md  
run  
run.sh  
test.txt

isr.c

```
1 [bits 32]
2
3 extern i686_ISR_Handler
4
5 ; cpu pushes to the stack:
6
7 %macro ISR_NOERRORCODE 1
8
9 global i686_ISR%1:
10 i686_ISR%1:
11     push 0
12     push %1
13     jmp isr_common
14
15 %endmacro
16
17 %macro ISR_ERRORCODE 1
18 global i686_ISR%1:
19 i686_ISR%1:
20     push %1
21     jmp isr_common
22
23 %endmacro
24
25
26 %include "arch/i686/isrs_gen.asm"
27
28 isr_common:
29     pusha
30
31     xor eax, eax
32     mov ax, ds
33     push eax
34
35     mov ax, 0x10
36     mov ds, ax
37     mov es, ax
38     mov fs, ax
39     mov gs, ax
40
41     push esp
42     call i686_ISR_Handler
43     add esp, 4
44
45 static const char* const g_ErrorString[] = {
46     "Divide by zero error",
47     "Debug",
48     "Non-maskable Interrupt",
49     "Breakpoint",
50     "Overflow",
51     "Bound Range Exceeded",
52     "Invalid Opcode",
53     "Device Not Available",
54     "Double Fault",
55     "Coprocessor Segment Overflow",
56     "Invalid TSS",
57     "Segment Not Present",
58     "Stack-Segment Fault",
59     "General Protection Fault",
60     "Page Fault",
61     "",
62     "x87 Floating-Point Exception",
63     "Alignment Check",
64     "Machine Check",
65     "SIMD Floating-Point Exception",
66     "Virtualization Exception",
67     "Control Protection Exception",
68     "",
69     "",
70     "%macro ISR_NOERRORCODE 1
71     global i686_ISR%1:
72     i686_ISR%1:
73         push 0
74         push %1
75         jmp isr_common
76
77     %endmacro
78
79     %macro ISR_ERRORCODE 1
80     global i686_ISR%1:
81     i686_ISR%1:
82         push %1
83         jmp isr_common
84
85     %endmacro
86
87     %include "arch/i686/isrs_gen.asm"
88
89 isr_common:
90     pusha
91
92     xor eax, eax
93     mov ax, ds
94     push eax
95
96     mov ax, 0x10
97     mov ds, ax
98     mov es, ax
99     mov fs, ax
100    mov gs, ax
101
102    push esp
103    call i686_ISR_Handler
104    add esp, 4
105
106 static const char* const g_ErrorString[] = {
107     "IRQ0 timer",
108     "IRQ1 Keyboard Interrupt",
109     "",
110     "",
111     "",
112     "",
113     "",
114     "",
115     "",
116     "",
117     "",
118     "",
119     "",
120     "",
121     "",
122     "",
123     "",
124     "",
125     "",
126     "",
127     "",
128     "",
129     "",
130     "",
131     "",
132     "",
133     "",
134     "",
135     "",
136     "",
137     "",
138     "",
139     "",
140     "",
141     "",
142     "",
143     "",
144     "",
145     "",
146     "",
147     "",
148     "",
149     "",
150     "",
151     "",
152     "",
153     "",
154     "",
155     "",
156     "",
157     "",
158     "",
159     "",
160     "",
161     "",
162     "",
163     "",
164     "",
165     "",
166     "",
167     "",
168     "",
169     "",
170     "",
171     "",
172     "",
173     "",
174     "",
175     "",
176     "",
177     "",
178     "",
179     "",
180     "",
181     "",
182     "",
183     "",
184     "",
185     "",
186     "",
187     "",
188     "",
189     "",
190     "",
191     "",
192     "",
193     "",
194     "",
195     "",
196     "",
197     "",
198     "",
199     "",
199     ""}
```

main.c

```
#include <stdint.h>
#include "stdio.h"
#include "memory.h"
#include <hal/hal.h>
#include <arch/i686/irq.h>
extern uint8_t __bss_start;
extern uint8_t __end;
void __attribute__((cdecl)) subzero();
void haddPaar();
void printLion() {
    printf("Overflow exception\n");
    printf("Value in Registers:\n");
    printf("regs->eax=0xffffffff;\n");
    printf("Interrupt handler=\n");
    printf("Updated value is\n");
}
void Overflow(Registers* regs) {
    int msb = 0;
    n = n / 2;
    while (n != 0) {
        n = n / 2;
        msb++;
    }
    return (32 - msb)/4;
}
int setBitNumber(int n) {
    if (n == 0)
        return 0;
    int msb = 0;
    n = n / 2;
    while (n != 0) {
        n = n / 2;
        msb++;
    }
    return (32 - msb)/4;
}
void divByZero(Registers* regs) {
    uint32_t num2 = regs->eax;
    uint32_t num1 = regs->eax;
    int k = setBitNumber(num1);
    char arr[256];
}
```

Ln 1, Col 1   Spaces: 4   UTF-8   LF   { } C   Go Live   GCC   Prettier

# Interrupts and MARN OS



We enhanced the Operating System with interrupt handling, using assembly and C, allowing the CPU to save its state and execute specific code when interrupts occurred, then return to normal execution with the "IRET" instruction. Different handlers managed various interrupt types, ensuring proper handling and recovery.

```
void divByZero(Registers* regs)
{
    uint32_t num2 = regs->eax;
    uint32_t num1 = regs->edx;
    int k = setBitNumber(num2);
    char arr[256];

    for(int i=0;i<k;i++){
        arr[i]='0';
    }
    arr[k]='\0';

    // uint32_t numerator = regs->ds;
    printf("Division by zero error: Numerator = 0x%x%s%x\n", num1,arr, num2);

    regs->ebx = 2;
    regs->edx = 0;
    printf("Handled this error.\n ");
    return ;
}
```

# Understanding the Control Flow



The control flow of any interrupt is as follows:





# Interrupt Descriptor Table (IDT)

The Interrupt Descriptor Table (IDT) is a crucial data structure within the x86 architecture, serving as a mapping table responsible for managing interrupts and exceptions. These interruptions can arise from a variety of events in a computer system.





# Interrupt Descriptor Table (IDT)

IDT entries in 64Bit format

|                         |               |     |    |                  |          |    |    |    |    |
|-------------------------|---------------|-----|----|------------------|----------|----|----|----|----|
| 63                      | 48            | 47  | 46 | 45               | 44       | 43 | 40 | 39 | 32 |
| <b>Offset</b>           | P             | DPL | 0  | <b>Gate Type</b> | Reserved |    |    |    |    |
| 31                      | 16            | 1   | 0  | 3                | 0        |    |    |    |    |
| 31                      | 16            | 15  |    |                  |          |    |    |    | 0  |
| <b>Segment Selector</b> | <b>Offset</b> |     |    |                  |          |    |    |    | 0  |
| 15                      | 0             | 15  |    |                  |          |    |    |    | 0  |



# Defining IDT Entry in C

C code:

```
typedef struct
{
    uint16_t BaseLow;
    uint16_t SegmentSelector;
    uint8_t Reserved;
    uint8_t Flags;
    uint16_t BaseHigh;
} __attribute__((packed)) IDTEntry;
```



# Defining IDT Table in C

C code:

```
typedef struct
{
    uint16_t Limit;
    IDTEntry* Ptr;
} __attribute__((packed)) IDTDescriptor;
```



# Defining Macros for Flags in C

## C code

```
typedef enum
{
    IDT_FLAG_GATE_TASK          = 0x5,
    IDT_FLAG_GATE_16BIT_INT     = 0x6,
    IDT_FLAG_GATE_16BIT_TRAP    = 0x7,
    IDT_FLAG_GATE_32BIT_INT     = 0xE,
    IDT_FLAG_GATE_32BIT_TRAP    = 0xF,

    IDT_FLAG_RING0              = (0 << 5),
    IDT_FLAG_RING1              = (1 << 5),
    IDT_FLAG_RING2              = (2 << 5),
    IDT_FLAG_RING3              = (3 << 5),

    IDT_FLAG_PRESENT            = 0x80,
} IDT_FLAGS;
```



# Initializing IDT Entry in C

## C code

```
void i686_IDT_SetGate(int interrupt, void* base, uint16_t segmentDescriptor, uint8_t
{
    g_IDT[interrupt].BaseLow = ((uint32_t)base) & 0xFFFF;
    g_IDT[interrupt].SegmentSelector = segmentDescriptor;
    g_IDT[interrupt].Reserved = 0;
    g_IDT[interrupt].Flags = flags;
    g_IDT[interrupt].BaseHigh = ((uint32_t)base >> 16) & 0xFFFF;
}
```



# Exceptions Stored in IDT

## C code

```
static const char* const g_Exceptions [] = {
    "Divide by zero error",
    "Debug",
    "Non-maskable Interrupt",
    "Breakpoint",
    "Overflow",
    

---


    "Bound Range Exceeded",
    

---


    "Invalid Opcode",
    "Device Not Available",
    "Double Fault",
    "Coprocessor Segment Overrun",
```



# Creating and Handling Exceptions

## Assembly code for the subzero subroutine

```
global subzero
subzero:
    xor edx, edx      ; Clear the EDX register (for the remainder)
    mov eax, 9        ; Set EAX to the dividend (9)
    mov ebx, 0        ; Set EBX to the divisor (4)
    div ebx          ; Divide EAX by EBX, quotient in EAX, remainder in EDX

    mov ebx, 1
    div ebx

    ret
```



# Unhandled Exceptions

State of the O.S. in a divide by zero Unhandled Exception

```
Welcome to MARN OS!
Unhandled exception 0 Divide by zero error
eax=9 ebx=0 ecx=0 edx=0 esi=7e19 edi=2460
esp=10004c ebp=ffa4 eip=1000c3 eflags=46 cs=8 ds=10 ss=100000
interrupt=0 errorcode=0
KERNEL PANIC!
```



# Handling the Exception

State of the O.S. in a divide by zero Handled Exception

```
Welcome to MARN OS!
Division by zero error: Numerator = 0x000000009
Handled this error.
Division by zero error: Numerator = 0x100000004
Handled this error.
```



# Overflow Interrupt

## Causing the Overflow using Assembly Program

```
global haddPaar
haddPaar:
    xor edx, edx
    mov eax, 0xFFFFFFFF; Load the value into the EAX register
    add eax, 1
    int 4
    ret
```



# Overflow Interrupt

State of the O.S. in an Unhandled Overflow Interrupt

```
Welcome to MARN OS!
Unhandled exception 4 Overflow
eax=0 ebx=2500 ecx=0 edx=0 esi=7e19 edi=2460
esp=10004c ebp=ffa4 eip=1000b6 eflags=57 cs=8 ds=10 ss=100000
interrupt=4 errorcode=0
KERNEL PANIC!
```



# Overflow Interrupt

State of the O.S. In handled Overflow Interrupt

```
Welcome to MARN OS!
Overflow exception
Value in Register eax 0
Interrupt handled as follows:
Updated value in the register eax is now set to 4294967295
```



# User-Defined Interrupt

State of the O.S. in an Unhandled User-Defined Interrupt

```
Welcome to MARN OS!  
Unhandled interrupt 48!  
-
```



# User-Defined Interrupt

State of the O.S. in a handled User-Defined Interrupt

```
UMMM .AMMY' ,AMMMMMMMMMMMMMMMMMMD
`UMM, AMMU' ,AMMMMMMMMMMMMMMMMMMM,
UMMMmMMU' ,AMY~~' 'MMMMMMMMMMMM' '~~
`YMMMM' AMM' 'UMMMMMMMMP'
AMMM' VMMA. YUmmmmMMMMMMMMML MmmmY
,AMMA ___,HMMMMMdMMMMMMMMMMMMMMML `UMU'
AMMMA _'MMMMMMMMMMMMMMMMMMMMMMMA '
,AMMMMMMMMMMMMMMMMMMMMMMMMMMA
AMMMMMMMMM' ``YMMMMMMMMMMMMMMMMMA ,AMU
UMU MMMMMU 'YMMMMMMMMMMMMMMMMY 'UMMY' adMMMM
`U MMMM' 'YMMMMMMMU.~~~~~,aado,'U' MMMMM
aMMMMmu 'YMMMMMMM, ,/AMMMMA, YMMMM
UMMM, ,v YMMMMMMMMMo oMMMMMMMM' a, YMMMM
`YMMMMY' 'YMMMMMMMY' 'YMMMMMMY MMmMM
AMMM , ~~~~,aooooa,~~~~~ MMM
YMMMB,d' dMMMMMMMMMMMD, a,, AMMM
YMMMM, A YMMMMMMMMMMY ,MMMM
AMMMMMMMMM A~~~~~ A
`YMMMM' ,A, ,AMMM
,AMMMMMMMMMMA, ,aAMMM
,AMMMMMMMMMMA, AMMM
,AMMMMMMMMMMMMA AMMM
AMMMMMMMMMMMMAaAMMM
Handling MARN Interrupt
```

# Programmable Interrupt Controller



The x86 architecture traditionally used an 8259 PIC (Intel 8259A Programmable Interrupt Controller) or a more advanced APIC (Advanced Programmable Interrupt Controller) for multiprocessor systems.



# Programmable Interrupt Controller



Programmable Interrupt  
Controller (PIC)



# Programmable Interrupt Controller



## Programmable Interrupt Controller (PIC)





# Programmable Interrupt Controller

Programming the PIC in C

```
#define PIC1_COMMAND_PORT      0x20
#define PIC1_DATA_PORT          0x21
#define PIC2_COMMAND_PORT      0xA0
#define PIC2_DATA_PORT          0xA1
```

```
enum {
    PIC_ICW1_ICW4           = 0x01,
    PIC_ICW1_SINGLE          = 0x02,
    PIC_ICW1_INTERVAL4       = 0x04,
    PIC_ICW1_LEVEL            = 0x08,
    PIC_ICW1_INITIALIZE       = 0x10
} PIC_ICW1;
```

# Programmable Interrupt Controller



ICW1 From the Intel Manual for x86 Architecture



# Programmable Interrupt Controller



ICW2 From the Intel Manual for x86 Architecture



# Programmable Interrupt Controller



ICW3 From the Intel Manual for x86 Architecture



# Programmable Interrupt Controller



ICW4 From the Intel Manual for x86 Architecture



# Programmable Interrupt Controller

## Configuring PIC



```
void i686_PIC_Configure(uint8_t offsetPic1, uint8_t offsetPic2)
{
    // initialization control word 1
    i686_outb(PIC1_COMMAND_PORT, PIC_ICW1_ICW4 | PIC_ICW1_INITIALIZE);
    i686_iowait();
    i686_outb(PIC2_COMMAND_PORT, PIC_ICW1_ICW4 | PIC_ICW1_INITIALIZE);
    i686_iowait();

    // initialization control word 2 - the offsets
    i686_outb(PIC1_DATA_PORT, offsetPic1);
    i686_iowait();
    i686_outb(PIC2_DATA_PORT, offsetPic2);
    i686_iowait();

    // initialization control word 3
    i686_outb(PIC1_DATA_PORT, 0x4);           // tell PIC1 that it has a
                                              // slave at IRQ2 (0000 0100)
    i686_iowait();
    i686_outb(PIC2_DATA_PORT, 0x2);           // tell PIC2 its cascade
                                              // identity (0000 0010)
    i686_iowait();

    // initialization control word 4
    i686_outb(PIC1_DATA_PORT, PIC_ICW4_8086);
    i686_iowait();

    i686_outb(PIC2_DATA_PORT, PIC_ICW4_8086);
    i686_iowait();

    // clear data registers
    i686_outb(PIC1_DATA_PORT, 0);
    i686_iowait();
    i686_outb(PIC2_DATA_PORT, 0);
    i686_iowait();
}
```

# Programmable Interrupt Controller



OCW1 From the Intel Manual for x86 Architecture



# Programmable Interrupt Controller



OCW2 From the Intel Manual for x86 Architecture



# Programmable Interrupt Controller

OCW3 From the Intel Manual for x86 Architecture



# Programmable Interrupt Controller

## OCW implementation



```
1 void i686_PIC_SendEndOfInterrupt(int irq)
2 {
3     if (irq >= 8)
4         i686_outb(PIC2_COMMAND_PORT, PIC_CMD_END_OF_INTERRUPT);
5     i686_outb(PIC1_COMMAND_PORT, PIC_CMD_END_OF_INTERRUPT);
6 }
7
8 void i686_PIC_Disable()
9 {
10    i686_outb(PIC1_DATA_PORT, 0xFF);           // mask all
11    i686_iowait();
12    i686_outb(PIC2_DATA_PORT, 0xFF);           // mask all
13    i686_iowait();
14 }
15
16 void i686_PIC_Mask(int irq)
17 {
18     uint8_t port;
19
20     if (irq < 8)
21     {
22         port = PIC1_DATA_PORT;
23     }
24     else
25     {
26         irq -= 8;
27         port = PIC2_DATA_PORT;
28     }
29
30     uint8_t mask = i686_inb(PIC1_DATA_PORT);
31     i686_outb(PIC1_DATA_PORT, mask | (1 << irq));
32 }
33 }
```

```
34
35 void i686_PIC_Unmask(int irq)
36 {
37     uint8_t port;
38
39     if (irq < 8)
40     {
41         port = PIC1_DATA_PORT;
42     }
43     else
44     {
45         irq -= 8;
46         port = PIC2_DATA_PORT;
47     }
48
49     uint8_t mask = i686_inb(PIC1_DATA_PORT);
50     i686_outb(PIC1_DATA_PORT, mask & ~(1 << irq));
51 }
52
53 uint16_t i686_PIC_ReadIrqRequestRegister()
54 {
55     i686_outb(PIC1_COMMAND_PORT, PIC_CMD_READ_IRR);
56     i686_outb(PIC2_COMMAND_PORT, PIC_CMD_READ_IRR);
57     return ((uint16_t)i686_inb(PIC2_COMMAND_PORT)) | (((uint16_t)i686_inb(PIC2_COMMAND_PORT)) << 8);
58 }
59
60 uint16_t i686_PIC_ReadInServiceRegister()
61 {
62     i686_outb(PIC1_COMMAND_PORT, PIC_CMD_READ_ISR);
63     i686_outb(PIC2_COMMAND_PORT, PIC_CMD_READ_ISR);
64     return ((uint16_t)i686_inb(PIC2_COMMAND_PORT)) | (((uint16_t)i686_inb(PIC2_COMMAND_PORT)) << 8);
65 }
```



# Hardware Interrupts

Hardware interrupts are signals from hardware devices. They are managed using Interrupt Requests (IRQs) numbered from 0 to 15. In the mentioned program, Timer (IRQ0) is used for timekeeping, and Keyboard Interrupt (IRQ1) captures user input like keypresses. IRQs help the operating system handle hardware events.



# Timer : IRQ0

## Handled Timer Interrupt:





# Keyboard Interrupt : IRQ1

Handled Keyboard Interrupt:





# IRQ0 & IRQ1 Implementation

```
void timer(Registers* regs){  
    printf(".");  
    return;  
}  
  
void keyboard(Registers* regs) {  
    printf("MARN -- Keyboard Interrupt");  
    printf(".");  
}
```

# Conclusions



- Implemented advanced interrupt management features, enhancing the system's capability to both generate and handle interrupts.
- Crafted user-defined interrupts, adding a layer of customization and unhandled interrupts were now handled.
- Resolved previously unhandled interrupt scenarios, increasing system reliability



# References:

- [MIT Resources](#)
- [x86 Architecture resources](#)
- [x86 Architecture resources2](#)
- [x86 Architecture resources3](#)
- [Codebase](#)





# Thank You