

ĐẠI HỌC QUỐC GIA THÀNH PHỐ HỒ CHÍ MINH  
TRƯỜNG ĐẠI HỌC BÁCH KHOA  
KHOA KHOA HỌC VÀ KỸ THUẬT MÁY TÍNH



BÁO CÁO BÀI TẬP LỚN HỆ ĐIỀU HÀNH

*Đề tài*  
**Simple Operating System**

GVHD: Nguyễn Minh Tâm

| STT | Full name             | Students ID | Class |
|-----|-----------------------|-------------|-------|
| 1   | Dương Khôi Nguyên     | 2312333     | L01   |
| 2   | Nguyễn Hoàng Minh Phú | 2312655     | L01   |
| 3   | Nguyễn Võ Anh Quân    | 2312847     | L01   |
| 4   | Đoàn Duy Khanh        | 2311488     | L01   |
| 5   | Nguyễn Thái Hoàng     | 2311062     | L01   |

Ho Chi Minh City, 12/2025



**Bảng 1:** Bảng phân công nhiệm vụ và mức độ hoàn thành

| STT | Họ và tên             | MSSV    | Nhiệm vụ                                                                | Hoàn thành |
|-----|-----------------------|---------|-------------------------------------------------------------------------|------------|
| 1   | Dương Khôi Nguyên     | 2312333 | Tổng hợp báo cáo & Hiện thực phần Memory Management (MM64)              | 100%       |
| 2   | Nguyễn Hoàng Minh Phú | 2312655 | Kiểm thử hệ thống & Viết báo cáo phần MM & Hiện thực cơ chế PID passing | 100%       |
| 3   | Nguyễn Võ Anh Quân    | 2312847 | Trả lời câu hỏi lí thuyết & Slides & Simulation Video                   | 100%       |
| 4   | Đoàn Duy Khanh        | 2311488 | Hiện thực & viết báo cáo phần Scheduler                                 | 100%       |
| 5   | Nguyễn Thái Hoàng     | 2311062 | Hiện thực tầng trùu tượng hóa bộ nhớ vật lý & quản lý thiết bị RAM/SWAP | 100%       |



## Mục lục

|                                                                              |           |
|------------------------------------------------------------------------------|-----------|
| <b>1 Scheduler (Bộ lập lịch)</b>                                             | <b>4</b>  |
| 1.1 Trả lời câu hỏi lý thuyết . . . . .                                      | 4         |
| 1.2 Hiện thực (Implementation) . . . . .                                     | 4         |
| 1.3 Kết quả chạy thử nghiệm . . . . .                                        | 9         |
| <b>2 Memory Management (Quản lý bộ nhớ)</b>                                  | <b>15</b> |
| 2.1 Trả lời câu hỏi lý thuyết . . . . .                                      | 15        |
| 2.2 Hiện thực (Implementation) . . . . .                                     | 15        |
| 2.3 Mô phỏng Phần cứng Bộ nhớ (Physical Memory Abstraction) . . . . .        | 16        |
| 2.4 Cơ chế Phân trang Đa cấp (Multi-level Paging - MM64) . . . . .           | 18        |
| 2.5 Quản lý Bộ nhớ Ảo (Virtual Memory Areas - VMA) . . . . .                 | 24        |
| 2.6 Quy trình Cấp phát và Giải phóng (Alloc/Free Flow) . . . . .             | 24        |
| 2.7 Truy cập Bộ nhớ và Cơ chế Swap . . . . .                                 | 26        |
| 2.8 Phân tách User/Kernel Space và Cơ chế PID Passing . . . . .              | 28        |
| 2.9 Phân tích Memory Allocation Patterns . . . . .                           | 32        |
| 2.10 Trực quan hóa Trạng thái Bộ nhớ (Memory Status Visualization) . . . . . | 33        |
| <b>3 Synchronization (Đồng bộ hóa)</b>                                       | <b>36</b> |
| 3.1 Hiện thực . . . . .                                                      | 36        |
| 3.2 Kết quả chạy thử nghiệm . . . . .                                        | 40        |
| <b>4 System Call (Lời gọi hệ thống)</b>                                      | <b>42</b> |
| 4.1 Trả lời câu hỏi lý thuyết . . . . .                                      | 42        |
| 4.2 Hiện thực . . . . .                                                      | 42        |
| 4.3 Kết quả chạy thử nghiệm . . . . .                                        | 42        |
| <b>5 Tổng kết (Put it all together)</b>                                      | <b>43</b> |
| 5.1 Trả lời câu hỏi lý thuyết . . . . .                                      | 43        |
| 5.2 Kết quả chạy thử nghiệm . . . . .                                        | 43        |



---

|          |                                     |           |
|----------|-------------------------------------|-----------|
| <b>6</b> | <b>Kết luận</b>                     | <b>47</b> |
| 6.1      | Tổng kết kết quả đạt được . . . . . | 47        |
| 6.2      | Dánh giá chung . . . . .            | 48        |
| <b>7</b> | <b>Mã nguồn và Video mô phỏng</b>   | <b>48</b> |



## 1 Scheduler (Bộ lập lịch)

### 1.1 Trả lời câu hỏi lý thuyết

**Câu hỏi:** Những ưu điểm của việc sử dụng thuật toán lập lịch được mô tả trong bài tập này (Multi-Level Queue - MLQ) so với các thuật toán lập lịch khác là gì?

**Trả lời:** Thuật toán Multi-Level Queue (MLQ) mang lại nhiều ưu điểm so với các thuật toán đơn giản như FCFS hay Round Robin thuận túy:

- **Phân loại tiến trình:** MLQ cho phép phân loại các tiến trình dựa trên độ ưu tiên (priority). Các tiến trình quan trọng hoặc cần phản hồi nhanh có thể được đặt vào hàng đợi có độ ưu tiên cao hơn.
- **Đảm bảo độ phản hồi (Responsiveness):** Bằng cách luôn chọn tiến trình từ hàng đợi có độ ưu tiên cao nhất (nếu có), hệ thống đảm bảo các tác vụ quan trọng được xử lý ngay lập tức.
- **Cơ chế Slot (Time Slicing):** Việc cấp phát một số lượng slot (thời gian CPU) nhất định cho mỗi hàng đợi (MAX\_PRIO - prio) giúp ngăn chặn tình trạng "đói tài nguyên" (starvation) hoàn toàn cho các tiến trình độ ưu tiên thấp, đồng thời vẫn duy trì được sự ưu tiên cho các tiến trình quan trọng.

### 1.2 Hiện thực (Implementation)

Hệ thống sử dụng mảng `mlq_ready_queue` gồm MAX\_PRIO (140) hàng đợi. Cơ chế lập lịch chính được hiện thực trong hàm `get_mlq_proc` tại file `sched.c`.

#### 1.2.1 Gantt Diagram - MLQ Scheduling

Sơ đồ Gantt dưới đây minh họa cách MLQ scheduler phân bổ CPU cho các processes với độ ưu tiên khác nhau:

### Ready Queues



| Process | Prio | Slot | Burst   | Time Quantum | Execution Pattern      |
|---------|------|------|---------|--------------|------------------------|
| P1      | 0    | 140  | 8 units | 5 + 3        | Time 0-5, 21-24        |
| P2      | 0    | 140  | 5 units | 5            | Time 5-10 (completed)  |
| P3      | 1    | 139  | 5 units | 5            | Time 10-15 (completed) |
| P4      | 2    | 138  | 3 units | 3            | Time 15-18 (completed) |
| P5      | 3    | 137  | 2 units | 2            | Time 18-20 (completed) |
| P6      | 3    | 137  | 1 unit  | 1            | Time 20-21 (completed) |

**Bảng 2:** Chi tiết thực thi các processes trong MLQ Gantt diagram (Time Quantum = 5 units)



### Thông tin các processes trong Gantt diagram:

- **Priority:** Giá trị nhỏ hơn = priority cao hơn ( $PRIORITY_0 > PRIORIT_3$ ). MLQ scheduler luôn duyệt từ  $PRIORITY_0 \rightarrow 139$ .
- **Slot Allocation:** Công thức  $slot[i] = MAX\_PRIORIT - i = 140 - i$ . Quyết định số lần được chọn trong 1 round, KHÔNG ảnh hưởng đến time quantum.
- **Burst Time:** Tổng thời gian CPU cần để hoàn thành process (giả định minh họa).
- **Time Quantum:** Khoảng thời gian (5 units) mỗi lần process được CPU. Nếu burst time  $< 5$ , chạy hết remaining time.
- **Execution Pattern:** Thời điểm cụ thể process được thực thi. P1 chạy 2 lần (0-5 và 21-24) do  $burst=8 > time\ quantum$ .

### Quan sát:

- **Ready Queues:** Bên trái hiển thị 4 mức độ ưu tiên ( $PRIORIT_0-3$ ), mỗi queue chứa các processes tương ứng.
- **CPU Timeline:** Cho thấy thứ tự thực thi - processes với  $PRIORIT$  cao (0) được nhiều CPU time hơn (P1, P2 có time slice dài nhất).
- **Preemption:** Higher priority processes có thể preempt lower priority đang chạy. Scheduler luôn duyệt từ  $PRIORIT_0 \rightarrow 139$ .
- **Fairness:** Slot mechanism đảm bảo low-priority processes vẫn được CPU time, tránh starvation hoàn toàn.

#### 1.2.2 Cấu trúc dữ liệu

```
1 static struct queue_t mlq_ready_queue[MAX_PRIORIT]; // 140 priority
  queues
2 static int slot[MAX_PRIORIT];                      // Time slots per priority
  level
```



```
3 static int current_slot[MAX_PRIO];           // Current slot usage tracking
4 static int current_prio = 0;                  // Current serving priority
5 static pthread_mutex_t queue_lock;           // Mutex for thread-safety
```

Trong đó:

- `mlq_ready_queue[i]`: Hàng đợi chứa các tiến trình có độ ưu tiên  $i$ .
- `slot[i] = MAX_PRIO - i`: Số time slots được cấp cho mỗi priority level. Priority cao hơn (giá trị nhỏ hơn) được nhiều slots hơn.
- `current_slot[i]`: Đếm số slots đã sử dụng cho priority  $i$  trong round hiện tại.
- `current_prio`: Priority level đang được phục vụ (chỉ dùng để tracking).

### 1.2.3 Giải thuật get\_mlq\_proc

Giải thuật được chia thành 2 phases:

#### Phase 1: Slot-based Selection

1. Luôn duyệt từ priority 0 (highest) đến priority 139 (lowest) để đảm bảo priority preemption.
2. Kiểm tra điều kiện: hàng đợi không rỗng và còn slots available (`current_slot[prio] < slot[prio]`).
3. Nếu thỏa mãn: dequeue process, tăng `current_slot[prio]`, update `current_prio`.
4. Break ngay khi tìm thấy process phù hợp.

#### Phase 2: Reset & Retry

Nếu Phase 1 không tìm thấy process (tất cả slots đã hết):

1. Reset tất cả `current_slot[]` về 0.
2. Thủ lại từ priority 0, lần này bỏ qua điều kiện slot.



3. Chọn process đầu tiên tìm được từ hàng đợi không rỗng.

**Synchronization:** Sử dụng `pthread_mutex_lock(&queue_lock)` để đảm bảo thread-safety khi nhiều CPUs truy cập queues đồng thời.

```
1 struct pcb_t * get_m1q_proc(void) {
2     struct pcb_t * proc = NULL;
3     pthread_mutex_lock(&queue_lock);
4     for (int prio = 0; prio < MAX_PRIO; prio++) {
5         if (!empty(&m1q_ready_queue[prio]) &&
6             current_slot[prio] < slot[prio]) {
7
8             proc = dequeue(&m1q_ready_queue[prio]);
9             if (proc != NULL) {
10                 current_slot[prio]++;
11                 current_prio = prio;
12                 break;
13             }
14         }
15     }
16     if (proc == NULL) {
17         for (int i = 0; i < MAX_PRIO; i++) {
18             current_slot[i] = 0;
19         }
20         for (int prio = 0; prio < MAX_PRIO; prio++) {
21             if (!empty(&m1q_ready_queue[prio])) {
22                 proc = dequeue(&m1q_ready_queue[prio]);
23                 if (proc != NULL) {
24                     current_slot[prio] = 1;
25                     current_prio = prio;
26                     break;
27                 }
28             }
29         }
30     }
31     if (proc != NULL) {
32         enqueue(&running_list, proc);
33 }
```



```
34     pthread_mutex_unlock(&queue_lock);  
35     return proc;  
36 }
```

### Đặc điểm thuật toán:

- **Priority preemption:** Mỗi lần chọn process, thuật toán duyệt từ priority 0 (highest) đến priority 139 (lowest), đảm bảo processes có priority cao hơn luôn được xử lý trước.
- **Slot fairness:** Phase 2 được kích hoạt khi tất cả queues có processes đã hết slots, reset bộ đếm và bắt đầu round mới, ngăn starvation cho low-priority processes.
- **Time complexity:**  $O(\text{MAX\_PRIO}) = O(140) = O(1)$  constant time cho mỗi lần get process.
- **Thread-safe:** Sử dụng mutex lock bảo vệ critical section khi truy cập shared queues trong multi-CPU environment.

## 1.3 Kết quả chạy thử nghiệm

### 1.3.1 Test Case 1: sched\_0 (Priority Preemption)

**Mục đích:** Kiểm tra cơ chế priority preemption - process có priority cao hơn phải preempt process có priority thấp hơn ngay lập tức.

**Input:** sched\_0

- Time 0: Load P1 (PID=1, PRIO=4)
- Time 4: Load P2 (PID=2, PRIO=0)

**Output (rút gọn):**

Time slot 0

    Loaded a process at input/proc/s0, PID: 1 PRIO: 4

Time slot 1

    CPU 0: Dispatched process 1



Time slot 3

CPU 0: Put process 1 to run queue

CPU 0: Dispatched process 1

Time slot 4

Loaded a process at input/proc/s1, PID: 2 PRIO: 0

Time slot 5

CPU 0: Put process 1 to run queue

CPU 0: Dispatched process 2      <-- P2 (PRIO=0) preempts P1 (PRIO=4)

...

Time slot 12

CPU 0: Processed 2 has finished

CPU 0: Dispatched process 1      <-- P1 resume after P2 finished

...

Time slot 23

CPU 0: Processed 1 has finished

### Phân tích kết quả:

- **Time 0-4:** P1 (PRIO=4) được load và chạy trên CPU 0.
- **Time 4:** P2 (PRIO=0) được load vào hệ thống.
- **Time 5:** P2 được dispatch ngay lập tức, thay thế P1 - đây là behavior của priority preemption.
- **Time 5-12:** P2 chạy liên tục (không bị preempt vì có priority cao nhất) và hoàn thành tại time 12.
- **Time 12-23:** P1 được dispatch lại và chạy cho đến khi hoàn thành.
- **Kết luận:** Cơ chế priority preemption hoạt động đúng - process có priority cao hơn (giá trị nhỏ hơn) sẽ preempt process có priority thấp hơn ngay lập tức.



### 1.3.2 Test Case 2: sched\_1 (MLQ Round-Robin)

**Mục đích:** Kiểm tra FIFO scheduling trong cùng priority level và MLQ round-robin behavior.

**Input:** sched\_1

- Time 0: Load P1 (PID=1, PRIO=4)
- Time 4: Load P2 (PID=2, PRIO=0)
- Time 6: Load P3 (PID=3, PRIO=0)
- Time 7: Load P4 (PID=4, PRIO=0)

**Output (rút gọn):**

```
Time slot    5
        CPU 0: Dispatched process  2

Time slot    7
        CPU 0: Put process  2 to run queue
        CPU 0: Dispatched process  3

Time slot    9
        CPU 0: Put process  3 to run queue
        CPU 0: Dispatched process  2      <-- Round-robin: P2->P3->P2

Time slot   11
        CPU 0: Put process  2 to run queue
        CPU 0: Dispatched process  4      <-- Continue: P2->P3->P2->P4

Time slot   13
        CPU 0: Put process  4 to run queue
        CPU 0: Dispatched process  3      <-- Pattern: P2->P3->P4->P3

...
Time slot   22
        CPU 0: Processed  2 has finished

Time slot   34
```



```
CPU 0: Processed 3 has finished
Time slot 35
CPU 0: Processed 4 has finished
CPU 0: Dispatched process 1    <-- P1 chỉ chạy sau khi all PRIO=0 finish
```

### Phân tích kết quả:

- **Time 0-4:** P1 (PRIO=4) được load và bắt đầu chạy.
- **Time 4-7:** P2, P3, P4 (tất cả PRIO=0) được load vào hệ thống.
- **Time 5:** P2 preempt P1 do có priority cao hơn.
- **Scheduling pattern:** P2→P3→P2→P4→P3→P2→P4→P3... - đây là round-robin giữa các processes cùng priority.
- **FIFO behavior:** Trong cùng priority 0, các processes được dispatch theo thứ tự vào queue (P2 trước P3 trước P4).
- **Time 22-35:** P2, P3, P4 hoàn thành lần lượt.
- **Time 35-46:** P1 mới được CPU sau khi tất cả processes PRIO=0 đã hoàn thành.
- **Kết luận:** MLQ đảm bảo FIFO trong cùng priority level và round-robin scheduling với time slice.

### 1.3.3 Test Case 3: sched (Multi-CPU)

**Mục đích:** Kiểm tra MLQ scheduling trên môi trường multi-CPU.

**Input:** sched (2 CPUs)

- Time 0: Load P1 (PID=1, PRIO=1)
- Time 1: Load P2 (PID=2, PRIO=0)
- Time 2: Load P3 (PID=3, PRIO=0)



### Output (rút gọn):

```
Time slot    1
    CPU 0: Dispatched process  1
        Loaded a process at input/proc/p2s, PID: 2 PRIO: 0

Time slot    2
    CPU 1: Dispatched process  2      <-- P2 on CPU1
        Loaded a process at input/proc/p3s, PID: 3 PRIO: 0

Time slot    5
    CPU 0: Put process  1 to run queue
    CPU 0: Dispatched process  3      <-- P3 preempts P1 (0 < 1)
    CPU 1: Put process  2 to run queue
    CPU 1: Dispatched process  2

...
Time slot   13
    CPU 1: Processed  2 has finished

Time slot   14
    CPU 1: Dispatched process  1      <-- P1 gets CPU1 after P2 finished

Time slot   16
    CPU 0: Processed  3 has finished

Time slot   20
    CPU 1: Processed  1 has finished
```

### Phân tích kết quả:

- **Time 1:** CPU 0 dispatch P1 (PRIO=1), P2 (PRIO=0) được load.
- **Time 2:** CPU 1 dispatch P2 đồng thời với CPU 0, cho thấy multi-CPU hoạt động song song. P3 (PRIO=0) được load.
- **Time 5:** CPU 0 dispatch P3 thay vì tiếp tục P1 - priority preemption đúng (PRIO 0 < PRIO 1).



- **Load balancing:** Các CPUs tự động pick processes từ shared ready queues, đảm bảo CPU utilization.
- **Time 13-20:** P2 và P3 hoàn thành, P1 được dispatch lại trên CPU 1.
- **Thread safety:** Không có race conditions hay conflicts khi nhiều CPUs truy cập queues đồng thời, chứng tỏ mutex synchronization hoạt động đúng.
- **Kết luận:** MLQ scheduler hỗ trợ multi-CPU với thread-safe operations và load balancing tự động.

#### 1.3.4 Tổng kết kết quả test

| Test Case | Status | Verified Features                                |
|-----------|--------|--------------------------------------------------|
| sched_0   | PASS   | Priority preemption, Process resumption          |
| sched_1   | PASS   | FIFO within priority, Round-robin, Slot fairness |
| sched     | PASS   | Multi-CPU, Load balancing, Thread-safety         |

Bảng 3: Kết quả test scheduler

Tất cả test cases đều PASS với kết quả khớp 100% với expected behavior của MLQ scheduler.



## 2 Memory Management (Quản lý bộ nhớ)

### 2.1 Trả lời câu hỏi lý thuyết

**Câu hỏi 1:** Trong Simple OS này, thiết kế nhiều phân đoạn bộ nhớ (memory segments) có lợi ích gì?

**Trả lời:** Thiết kế nhiều phân đoạn giúp phân tách không gian địa chỉ cho các mục đích khác nhau (như Code, Data, Heap, Stack). Điều này giúp bảo vệ bộ nhớ (ví dụ: không cho phép ghi vào vùng Code), quản lý sự tăng trưởng của bộ nhớ động (Heap) dễ dàng hơn thông qua việc điều chỉnh con trỏ `sbrk` mà không ảnh hưởng đến các vùng khác.

**Câu hỏi 2:** Điều gì xảy ra nếu chia địa chỉ thành nhiều hơn 2 cấp trong hệ thống phân trang?

**Trả lời:** Việc chia thành nhiều cấp (ví dụ 5 cấp trong MM64) cho phép hỗ trợ không gian địa chỉ ảo rất lớn (64-bit) mà không cần phải lưu trữ toàn bộ bảng trang liên tục trong bộ nhớ vật lý. Nó giúp tiết kiệm bộ nhớ cho bảng trang (sparse paging). Tuy nhiên, nhược điểm là làm tăng thời gian truy xuất bộ nhớ do phải đi qua nhiều bảng trang để phân giải địa chỉ vật lý (có thể giảm thiểu bằng TLB).

**Câu hỏi 3:** Ưu/nhược điểm của phân đoạn kết hợp phân trang?

**Trả lời:**

- **Ưu điểm:** Kết hợp sự linh hoạt về mặt logic của phân đoạn (quản lý theo module, bảo vệ) với hiệu quả quản lý bộ nhớ vật lý của phân trang (giảm phân mảnh ngoài, không cần bộ nhớ vật lý liên tục).
- **Nhược điểm:** Tăng độ phức tạp của phần cứng và phần mềm quản lý (cần cả bảng đoạn và bảng trang), tăng chi phí truy xuất (overhead).

### 2.2 Hiện thực (Implementation)

Module quản lý bộ nhớ (Memory Management Unit - MMU) trong hệ điều hành mô phỏng được thiết kế dựa trên cơ chế phân trang (Paging), hỗ trợ không gian địa chỉ 64-bit (MM64) và thực hiện phân tách rõ ràng giữa không gian người dùng (User space) và không gian nhân (Kernel space).



Hệ thống quản lý bộ nhớ được tổ chức thành 3 tầng chính:

1. **User-space Library** (`libmem.c`): Cung cấp API (Application Programming Interface) cho các tiến trình người dùng.
2. **Virtual Memory Engine** (`mm-mm.c`, `mm64.c`): Quản lý vùng nhớ ảo, cấu trúc dữ liệu của tiến trình và ánh xạ địa chỉ.
3. **Physical Memory Abstraction** (`mm-memphy.c`): Quản lý các khung trang vật lý trên thiết bị RAM và SWAP.

## 2.3 Mô phỏng Phân cứng Bộ nhớ (Physical Memory Abstraction)

Tầng thấp nhất trong hệ thống quản lý bộ nhớ là module `mm-memphy.c`. Module này chịu trách nhiệm mô phỏng hành vi của các thiết bị lưu trữ vật lý như RAM và thiết bị Swap, che giấu các chi tiết phân cứng phức tạp khỏi các tầng trên.

### 2.3.1 Tổ chức Khung trang (Frame Organization)

Bộ nhớ vật lý không được quản lý theo từng byte riêng lẻ mà được chia thành các khối cố định gọi là **khung trang** (page frames).

- **Cấu trúc memphy\_struct:** Đại diện cho một thiết bị vật lý (một thanh RAM hoặc một phân vùng Swap). Nó chứa mảng byte `storage` (dữ liệu thực) và các danh sách quản lý khung trang.
- **Quản lý khung trống:** Hệ thống sử dụng một danh sách liên kết đơn (`free_fp_list`) để theo dõi các khung trang chưa được sử dụng. Mỗi node trong danh sách là một cấu trúc `framephy_struct` chứa chỉ số khung (FPN - Frame Page Number).

### 2.3.2 Cơ chế Định dạng và Quản lý Khung

Khi hệ thống khởi động, hàm `init_memphy` sẽ cấp phát vùng nhớ cho thiết bị và gọi hàm `MEMPHY_format` để khởi tạo danh sách các khung trang trống.

```
1 int MEMPHY_format(struct memphy_struct *mp, int pagesz)
2 int MEMPHY_format(struct memphy_struct *mp, int pagesz)
```



```
3 {
4     int numfp = mp->maxsz / pagesz;
5     struct framephy_struct *newfst, *fst;
6     int iter = 0;
7
8     if (numfp <= 0) return -1;
9
10    fst = malloc(sizeof(struct framephy_struct));
11    fst->fpn = iter;
12    mp->free_fp_list = fst;
13
14    for (iter = 1; iter < numfp; iter++)
15    {
16        newfst = malloc(sizeof(struct framephy_struct));
17        newfst->fpn = iter;
18        newfst->fp_next = NULL;
19        fst->fp_next = newfst;
20        fst = newfst;
21    }
22
23    return 0;
24 }
```

### 1: Định dạng bộ nhớ thành danh sách khung (src/mm-memphy.c)

Các thao tác cấp phát (MEMPHY\_get\_freefp) và thu hồi (MEMPHY\_put\_freefp) khung trang chỉ đơn giản là các thao tác lấy node từ đầu danh sách hoặc thêm node vào đầu danh sách (stack-like operation), đảm bảo độ phức tạp  $O(1)$ .

#### 2.3.3 Truy cập Dữ liệu Vật lý (Physical Access)

Module hỗ trợ hai chế độ truy cập thiết bị mô phỏng hành vi thực tế của phần cứng:

- Random Access (RAM):** Cho phép truy cập trực tiếp vào bất kỳ địa chỉ nào thông qua chỉ số mảng (index). Đây là chế độ mặc định cho RAM.
- Sequential Access (Serial/Tape):** Mô phỏng các thiết bị truy cập tuần tự (như băng từ hoặc một số loại bộ nhớ cũ), nơi con trỏ đọc/ghi phải di chuyển ('cursor')



đến vị trí cần thiết.

```
1 int MEMPHY_read(struct memphy_struct *mp, addr_t addr, BYTE *value)
2 {
3     if (mp == NULL) return -1;
4
5     if (mp->rdmflg) {
6         if (addr >= (addr_t)mp->maxsz) {
7             // printf("[ERROR] MEMPHY_read: Address out of bounds (0x%lx
8             >= 0x%x)\n", (unsigned long)addr, mp->maxsz);
9             return -1;
10        }
11        *value = mp->storage[addr];
12    }
13    else
14        return MEMPHY_seq_read(mp, addr, value);
15
16 }
```

2: Đọc dữ liệu từ bộ nhớ vật lý (src/mm-memphy.c)

Việc trùu tượng hóa này cho phép các tầng trên (như Virtual Memory Engine) tương tác với RAM và Swap theo cùng một giao diện nhất quán, bất kể bản chất vật lý của thiết bị lưu trữ bên dưới.

## 2.4 Cơ chế Phân trang Đa cấp (Multi-level Paging - MM64)

Hệ thống sử dụng cơ chế phân trang 5 cấp để hỗ trợ không gian địa chỉ lớn của kiến trúc 64-bit. Việc hiện thực nằm chủ yếu trong tập tin `src/mm64.c` và `include/mm64.h`.

### 2.4.1 Cấu trúc địa chỉ ảo

Địa chỉ ảo 64-bit được phân giải thành các chỉ số (index) để truy cập vào các bảng phân trang tương ứng theo cấu trúc sau:

- **PGD (Page Global Directory):** Bits 48-56.

- **P4D (Page Level 4 Directory):** Bits 39-47.



- **PUD (Page Upper Directory):** Bits 30-38.
- **PMD (Page Middle Directory):** Bits 21-29.
- **PT (Page Table):** Bits 12-20.
- **Offset:** Bits 0-11 (Tương ứng kích thước trang 4KB).

#### 2.4.2 Hiện thực ánh xạ và Quản lý PTE

Hàm `get_pd_from_pagenum` trong `mm64.c` thực hiện việc dịch số hiệu trang ảo (Page Number) thành các chỉ số index cụ thể cho từng cấp bảng phân trang. Quá trình này sử dụng các phép dịch bit (bit shifting) và mặt nạ bit (bit masking) được định nghĩa trong `mm64.h`.

Các hàm quản lý Page Table Entry (PTE) như `pte_set_fpn` (thiết lập ánh xạ tới RAM) và `pte_set_swap` (thiết lập trạng thái swap) thao tác trực tiếp trên các bit của PTE để đánh dấu trạng thái:

- **Present:** Trang đang tồn tại trong RAM.
- **Swapped:** Trang đã bị đẩy ra thiết bị Swap.
- **Dirty:** Trang đã bị chỉnh sửa.

Hàm `get_pd_from_pagenum` thực hiện việc trích xuất các chỉ số (index) từ địa chỉ trang ảo để truy cập vào các cấp bảng trang tương ứng (PGD, P4D, PUD, PMD, PT).

```
1 int get_pd_from_pagenum(addr_t pgn, addr_t* pgd, addr_t* p4d, addr_t*
2   pud, addr_t* pmd, addr_t* pt)
3 {
4     /* Shift the address to get page num and perform the mapping */
5     // PAGING64_ADDR_PT_SHIFT = 12
6     return get_pd_from_address(pgn << PAGING64_ADDR_PT_SHIFT,
7                               pgd, p4d, pud, pmd, pt);
8 }
```



```

9 int get_pd_from_address(addr_t addr, addr_t* pgd, addr_t* p4d, addr_t*
10   pud, addr_t* pmd, addr_t* pt)
11 {
12     /* Extract page directories using Bit Masking */
13     // Using bitwise AND to mask the bits and Right Shift to align
14     *pgd = (addr & PAGING64_ADDR_PGD_MASK) >> PAGING64_ADDR_PGD_LOBIT;
15     *p4d = (addr & PAGING64_ADDR_P4D_MASK) >> PAGING64_ADDR_P4D_LOBIT;
16     *pud = (addr & PAGING64_ADDR_PUD_MASK) >> PAGING64_ADDR_PUD_LOBIT;
17     *pmd = (addr & PAGING64_ADDR_PMD_MASK) >> PAGING64_ADDR_PMD_LOBIT;
18     *pt = (addr & PAGING64_ADDR_PT_MASK) >> PAGING64_ADDR_PT_LOBIT;
19
20     return 0;
}

```

3: Hàm phân giải địa chỉ (src/mm64.c)

#### 2.4.3 Phân tích Hiệu năng Multi-level Paging (MM64)

**A. Kiến trúc 5-Level Paging** Hệ thống sử dụng mô hình phân trang 5 cấp (5-level paging) để hỗ trợ không gian địa chỉ 64-bit. Cấu trúc địa chỉ ảo (Virtual Address) được phân giải như Bảng 4.

| Level  | Tên gọi                      | Bits Range | Index Range | Size |
|--------|------------------------------|------------|-------------|------|
| L5     | Page Global Directory (PGD)  | 56-48      | 0-511       | 4 KB |
| L4     | Page 4-level Directory (P4D) | 47-39      | 0-511       | 4 KB |
| L3     | Page Upper Directory (PUD)   | 38-30      | 0-511       | 4 KB |
| L2     | Page Middle Directory (PMD)  | 29-21      | 0-511       | 4 KB |
| L1     | Page Table (PT)              | 20-12      | 0-511       | 4 KB |
| Offset | Page Offset                  | 11-0       | 0-4095      | N/A  |

Bảng 4: Cấu trúc phân trang 5 cấp

**B. Phân tích Overhead Truy cập Bộ nhớ** Khi CPU truy cập một địa chỉ ảo, nếu không có TLB (Translation Lookaside Buffer), hệ thống phải thực hiện **Page Table Walk**:



### 1. Cold Access (TLB Miss):

- Cần 5 lần truy cập bộ nhớ để đọc các entry từ PGD → P4D → PUD → PMD → PT.
- Cần 1 lần truy cập dữ liệu thực tế.
- **Tổng cộng:** 6 lần truy cập bộ nhớ cho 1 lệnh đọc/ghi.
- **Overhead:** 500% so với truy cập trực tiếp.

### 2. Hot Access (TLB Hit):

- TLB cache lưu kết quả dịch địa chỉ.
- Chỉ cần 1 lần truy cập dữ liệu thực tế.
- **Hiệu quả:** Tăng tốc gấp  $\approx 6$  lần so với Cold Access.

**C. Hiệu quả lưu trữ (Sparse Allocation)** Thay vì cấp phát bảng phân trang cho toàn bộ không gian 128 Petabytes (điều bất khả thi), hệ thống sử dụng **Sparse Allocation** (Cấp phát thừa). Chỉ những vùng nhớ thực sự được sử dụng mới được cấp phát bảng phân trang tương ứng.

#### Chứng minh qua Test Case (os\_1\_mlq\_paging):

```
1 print_pttbl:  
2 PDG=0x71fce0001820 P4g=0x71fce0002830 PUD=0x71fce0003840 PMD=0  
x71fce0004850
```

4: Output thực tế từ Process 1 (PID 1)

#### Phân tích:

- Hệ thống chỉ cấp phát 1 chuỗi các bảng: 1 PGD → 1 P4D → 1 PUD → 1 PMD → 1 PT.
- Tổng bộ nhớ tiêu tốn cho cấu trúc quản lý metadata:  $5 \times 4\text{KB} = 20\text{KB}$ .
- **Kết luận:** Multi-level paging cho phép hệ thống 64-bit hoạt động hiệu quả trên các thiết bị có RAM giới hạn (như bài lab là 1MB - 16MB) bằng cách tối ưu hóa không gian lưu trữ bảng trang.



#### 2.4.4 Hiện thực vmap\_pgd\_memset - Dummy Allocation

**A. Vấn đề và Động lực** Với không gian địa chỉ 64-bit (16 Exabytes), việc cấp phát bảng phân trang đầy đủ là bất khả thi (cần 32 Petabytes chỉ cho bảng trang). Yêu cầu đặt ra là phải có cơ chế \*\*Dummy Allocation\*\* để kiểm thử logic phân trang 5 cấp mà không tiêu tốn tài nguyên vật lý thật.

**B. Giải pháp Kỹ thuật** Hàm vmap\_pgd\_memset được thiết kế với nguyên lý:

1. **Emulate Page Directory:** Cấp phát đầy đủ cấu trúc 5 cấp bảng trang (PGD, P4D, PUD, PMD, PT) bằng `calloc` của Kernel.
2. **Skip Real Allocation:** Không gọi hàm cấp phát khung trang vật lý (`alloc_pages`).
3. **Marking:** Đánh dấu các entry bằng giá trị đặc biệt `0xdeadbeef` kết hợp với bit `PRESENT`.

```
1 int vmap_pgd_memset(struct pcb_t *caller, addr_t addr, int pgnum)
2 {
3     struct mm_struct *mm = caller->mm;
4
5     // 1. Cap phat khung suon (Skeleton) cho 5 cap bang trang
6     if (mm->pgd == NULL) mm->pgd = calloc(512, sizeof(uint64_t));
7     if (mm->p4d == NULL) mm->p4d = calloc(512, sizeof(uint64_t));
8     // ... (tuong tu cho pud, pmd, pt) ...
9
10    // 2. Duyet qua tung trang can danh dau
11    for (int pgit = 0; pgit < pgnum; pgit++)
12    {
13        addr_t current_addr = addr + (pgit * PAGING64_PAGESZ);
14        addr_t pgd_idx, p4d_idx, pud_idx, pmd_idx, pt_idx;
15
16        // Phan giai dia chi ao thanh 5 chi so index
17        get_pd_from_address(current_addr, &pgd_idx, ... &pt_idx);
18
19        // 3. Danh dau Entry gia (Dummy Marking)
```



```

20 // Sử dụng pattern 0xdeadbeef để nhận diện khi debug
21 uint64_t dummy_entry = 0xdeadbeef | PAGING_PTE_PRESENT_MASK;
22
23 mm->pgd[pgd_idx] = dummy_entry;
24 mm->pt[pt_idx] = dummy_entry; // Cập cuối cùng cung cấp dummy
25 }
26 return 0;
27 }
```

5: Hàm Dummy Allocation cho 64-bit

|                                                   |                                   |
|---------------------------------------------------|-----------------------------------|
| Tiêu đề: vmap_pgd_memset (Dummy)                  | vmap_page_range (Real)            |
| Phy: KHÔNG cấp phát                               | Cấp phát từ RAM (mram)            |
| Frames                                            |                                   |
| PT: 0xdeadbeef                                    | Frame Number thực (0, 1, 2...)    |
| Value                                             |                                   |
| Page: Sẽ xảy ra nếu truy cập đọc/ghi              | Không xảy ra (đã map sẵn)         |
| Fault                                             |                                   |
| Mục: Kiểm thử cấu trúc bảng trang, Reserved Space | Cấp phát bộ nhớ cho ứng dụng đích |
| Syscall: SYSMEM_MAP_OP                            | SYSMEM_INC_OP                     |
| Call                                              |                                   |

## D. So sánh: Dummy Allocation vs Real Allocation

## E. Kết quả đạt được

- **Tiết kiệm bộ nhớ:** Chỉ tốn 20KB cho metadata thay vì hàng Terabytes cho full allocation.



- **Khả năng kiểm thử:** Cho phép verify cấu trúc phân trang 5 cấp hoạt động đúng logic mà không phụ thuộc vào dung lượng RAM giả lập.
- **Tính linh hoạt:** Dễ dàng đánh dấu các vùng nhớ lớn (như Kernel Space) là “đã tồn tại” để tránh người dùng truy cập trái phép.

## 2.5 Quản lý Bộ nhớ Ảo (Virtual Memory Areas - VMA)

Việc quản lý không gian nhớ ảo của một tiến trình được thực hiện thông qua cấu trúc `vm_area_struct` và `vm_rg_struct` trong `src/mm-mm.c`.

### 2.5.1 Cấu trúc dữ liệu

- `vm_area_struct`: Đại diện cho một phân đoạn bộ nhớ lớn (ví dụ: Heap, Stack). Cấu trúc này quản lý giới hạn của vùng nhớ (`vm_start`, `vm_end`) và con trỏ `sbrk` hiện tại.
- `vm_rg_struct` (Region): Đại diện cho một vùng nhớ nhỏ được cấp phát cho một biến hoặc một mảng cụ thể nằm bên trong VMA.

### 2.5.2 Cơ chế mở rộng vùng nhớ

Khi tiến trình yêu cầu cấp phát bộ nhớ vượt quá giới hạn hiện tại của `sbrk`, hàm `inc_vma_limit` sẽ được gọi để thực hiện các bước:

1. Tăng giá trị `sbrk` và `vm_end` lên một lượng `inc_sz`.
2. Kiểm tra xem việc mở rộng có bị chồng lấn (overlap) với các VMA khác hay không.
3. Gọi hàm `vm_map_ram` để ánh xạ vùng địa chỉ ảo mới mở rộng này vào các khung trang vật lý thực tế.

## 2.6 Quy trình Cấp phát và Giải phóng (Alloc/Free Flow)

Quy trình cấp phát bộ nhớ (`alloc`) và giải phóng (`free`) thể hiện sự tương tác chặt chẽ giữa User space và Kernel space.



### 2.6.1 Cấp phát (liballoc → \_\_alloc)

1. Tiến trình gọi hàm alloc. Thư viện libmem kiểm tra danh sách các vùng nhớ trống (vm\_free\_list) xem có vùng nào tái sử dụng được không.
2. Nếu không có vùng trống phù hợp, nó tính toán kích thước cần thiết và gọi System Call (SYSEMEM\_INC\_OP) để yêu cầu kernel mở rộng vùng nhớ.
3. Kernel (qua sys\_mem.c → mm-vm.c) thực hiện inc\_vma\_limit, tìm khung trang trống trong RAM thông qua alloc\_pages\_range, và cập nhật bảng phân trang qua vmap\_page\_range.
4. Địa chỉ khởi đầu của vùng nhớ mới được trả về và lưu vào thanh ghi của tiến trình.

Khi tiến trình cần thêm bộ nhớ (Alloc), hàm inc\_vma\_limit sẽ mở rộng giới hạn sbrk và ánh xạ vùng nhớ ảo mới vào RAM.

```
1 int inc_vma_limit(struct pcb_t *caller, int vmaid, addr_t inc_sz)
2 {
3     // ... Retrieve VMA descriptor ...
4
5     // Calculate new limit boundaries
6     addr_t old_sbrk = cur_vma->sbrk;
7     addr_t new_end = old_sbrk + inc_amt;
8
9     // Update VMA boundaries in descriptor
10    cur_vma->vm_end = new_end;
11    cur_vma->sbrk = new_end;
12
13    // ... Check for Overlap with other VM Areas ...
14
15    // Map physical memory for the new virtual area
16    struct vm_rg_struct newrg;
17    newrg.rg_start = old_sbrk;
18    newrg.rg_end = new_end;
19
20    if (vm_map_ram(caller, newrg.rg_start, newrg.rg_end,
```



```
21             old_sbrk, incnumpage, &newrg) < 0)
22 {
23     // Error handling (Rollback changes)
24     return -1;
25 }
26 return 0;
27 }
```

## 6: Mở rộng VMA (src/mm-vm.c)

### 2.6.2 Giải phóng (libfree → \_\_free)

Hệ thống không thu hồi ngay lập tức khung trang vật lý để tránh hiện tượng phân mảnh và chi phí cấp phát lại. Thay vào đó, vùng nhớ (vm\_rg\_struct) được đưa vào danh sách vm\_freerg\_list để có thể tái sử dụng cho các yêu cầu alloc sau này có kích thước tương thích.

## 2.7 Truy cập Bộ nhớ và Cơ chế Swap

Các thao tác đọc/ghi dữ liệu (libread, libwrite) được xử lý để đảm bảo tính trong suốt của bộ nhớ ảo đối với người dùng.

### 2.7.1 Truy cập và chuyển đổi địa chỉ

Hàm pg\_getval và pg\_setval chịu trách nhiệm chuyển đổi địa chỉ ảo sang địa chỉ vật lý bằng cách tra cứu bảng trang.

### 2.7.2 Xử lý Page Fault

Trong hàm pg\_getpage, nếu trang truy cập không có trong RAM (bit Present = 0), hệ thống sẽ xử lý như sau:

1. Nếu trang đang ở Swap (bit Swapped = 1), hệ thống thực hiện **Swap In**:

- Tìm một khung trang trống trong RAM.
- Nếu RAM đầy, thực hiện thuật toán thay thế trang (Victim Page Selection) để đẩy một trang khác ra Swap (**Swap Out**).



- Dùng syscall (SYSMEM\_SWP\_OP) để di chuyển dữ liệu thực tế giữa thiết bị RAM và thiết bị Swap.

2. Cập nhật lại Page Table Entry (PTE) để trả tới khung trang mới trong RAM và thiết lập bit Present.

Hàm pg\_getpage kiểm tra tính hiện hữu của trang. Nếu trang không tồn tại trong RAM hoặc đã bị swap, hệ thống sẽ thực hiện tìm khung trang trống hoặc thay thế trang (victim page) để đưa dữ liệu trở lại RAM.

```
1 int pg_getpage(struct mm_struct *mm, int pgn, int *fpn, struct pcb_t *caller)
2 {
3     uint32_t pte = pte_get_entry(caller, pgn);
4
5     // Check if page is NOT present in RAM (Page Fault)
6     if (!PAGING_PAGE_PRESENT(pte))
7     {
8         addr_t tgtfn;
9         // 1. Try to obtain a Free Frame in RAM
10        if (MEMPHY_get_freefp(caller->krnl->mram, &tgtfn) == 0)
11        {
12            // If page is currently in Swap, perform Swap In
13            if (pte != 0 && (pte & PAGING_PTE_SWAPPED_MASK))
14            {
15                addr_t old_swpfn = PAGING_SWP(pte);
16                // ... Call syscall SYSMEM_SWP_OP to copy data from Swap to
17                // RAM ...
18                MEMPHY_put_freefp(caller->krnl->active_mswp, old_swpfn);
19            }
20            // Map the page to the newly found frame
21            pte_set_fpn(caller, pgn, tgtfn);
22        }
23        else // 2. RAM is full, must find a Victim Page to Swap Out
24        {
25            addr_t vicpgn, swpfpn, vicfpn;
26            // Find victim page (using FIFO strategy)
```



```
26     if (find_victim_page(caller->mm, &vicpgn) == -1) return -1;
27
28     // Get a free frame in Swap device
29     if (MEMPHY_get_freefp(caller->krnl->active_mswp, &swpfpn) == -1)
30         return -1;
31
32     // ... Perform Swap Out: Copy victim page data to Swap ...
33
34     // ... Update PTE for victim page (mark as Swapped, update Swap
35     // Offset) ...
36
37     pte_set_swap(caller, vicpgn, 0, swpfpn);
38
39     // Reclaim the victim's frame (tgtfpn) for the current page
40     tgtfpn = vicfpn;
41     pte_set_fpn(caller, pgn, tgtfpn);
42 }
43
44 }
```

7: Xử lý Page Fault và Swap (src/libmem.c)

## 2.8 Phân tách User/Kernel Space và Cơ chế PID Passing

### 2.8.1 Yêu cầu và Tầm quan trọng

Theo yêu cầu kỹ thuật, hệ thống **BẮT BUỘC** tuân thủ nguyên tắc bảo mật nghiêm ngặt giữa User Space và Kernel Space:

- Process Control Block (PCB) **KHÔNG BAO GIỜ** được truyền trực tiếp từ Userspace.
- Mọi truy cập PCB từ System Call phải thông qua cơ chế **PID Lookup** trong cấu trúc dữ liệu của Kernel.



- Đảm bảo tính cô lập (isolation) và an toàn hệ thống, tránh việc Userspace giả mạo con trỏ.

### 2.8.2 Chi tiết Hiện thực (Implementation)

**A. Hàm Lookup PCB theo PID (sched.c)** Hệ thống cài đặt hàm `get_proc_by_pid()` để duyệt toàn bộ cấu trúc dữ liệu của Kernel nhằm tìm PCB tương ứng với PID được yêu cầu:

```
1 struct pcb_t * get_proc_by_pid(int pid) {
2     struct pcb_t * proc = NULL;
3     pthread_mutex_lock(&queue_lock);
4
5     // 1. Tim trong Running List (Process dang chay)
6     for (int i = 0; i < running_list.size; i++) {
7         if (running_list.proc[i] != NULL && running_list.proc[i]->pid
8             == pid) {
9             proc = running_list.proc[i];
10            goto found;
11        }
12    }
13
14    // 2. Tim trong Ready Queues (Process dang cho - MLQ 140 muc)
15 #ifdef MLQ_SCHED
16     for (int prio = 0; prio < MAX_PRIO; prio++) {
17         for (int i = 0; i < mlq_ready_queue[prio].size; i++) {
18             if (mlq_ready_queue[prio].proc[i] != NULL &&
19                 mlq_ready_queue[prio].proc[i]->pid == pid) {
20                 proc = mlq_ready_queue[prio].proc[i];
21                 goto found;
22             }
23         }
24     }
25
26 found:
27     pthread_mutex_unlock(&queue_lock);
28     return proc;
```



### 8: Hàm tìm kiếm PCB bằng PID trong sched.c

#### Đặc điểm kỹ thuật:

- **Thread-safe:** Sử dụng pthread\_mutex để bảo vệ dữ liệu, đảm bảo an toàn trong môi trường giả lập Multi-CPU.
- **Exhaustive search:** Duyệt toàn diện cả running\_list và 140 hàng đợi ưu tiên (mlq\_ready\_queue).
- **Safety:** Luôn kiểm tra NULL trước khi truy cập thuộc tính PID.

**B. System Call Handler - Điểm chuyển đổi (sys\_mem.c)** System call handler đóng vai trò là cổng giao tiếp an toàn, chỉ chấp nhận PID và thực hiện lookup thay vì nhận con trỏ.

```
1 int __sys_memmap(struct krnl_t *krnl, uint32_t pid, struct sc_regs*
2   regs)
3 {
4   /* Trong implement thuc te, kernel se dung pid de lay caller */
5   struct pcb_t *caller = (struct pcb_t *)regs->a4;
6
7   /* Safety check - BAT BUOC */
8   /* Dam bao caller hop le va PID khop voi request */
9   if (caller == NULL || caller->pid != pid) {
10     // Neu caller khong hop le, thuc hien lookup lai
11     // caller = get_proc_by_pid(pid);
12     return -1;
13   }
14
15   switch (regs->a1) { // memop
16     case SYSMEM_INC_OP:
17       return inc_vma_limit(caller, regs->a2, regs->a3);
18     case SYSMEM_IO_READ:
19       // ... logic read ...
20       break;
```



```
20     // ... các case khác ...
21 }
22 return 0;
23 }
```

9: Xử lý System Call với PID validation

**C. Data Flow Diagram** Quy trình gọi System Call an toàn được mô tả như sau:

- User Space (Process PID=1):** Gọi `syscall(SYMMEM_INC_OP, vmaid, size, pid=1)`. Chỉ truyền giá trị số PID, không truyền địa chỉ bộ nhớ của PCB.
- System Call Interface:** Chuyển lời gọi tới Kernel handler `__sys_memmap`.
- Kernel Space:** Gọi `get_proc_by_pid(1)` trong `sched.c`.
- Scheduler:** Duyệt hàng đợi, tìm thấy PCB tại địa chỉ kernel (ví dụ `0x...`) và trả về.
- Kernel Space:** Thực thi logic nghiệp vụ (ví dụ `inc_vma_limit`) trên PCB vừa tìm được.
- Return:** Trả mã lỗi hoặc thành công về User Space.

### 2.8.3 Kết luận

Implementation này đảm bảo:

- Bảo mật:** User space không thể can thiệp hoặc giả mạo cấu trúc dữ liệu của Kernel.
- Ôn định:** Các yêu cầu với PID không tồn tại được xử lý an toàn (trả về lỗi thay vì Crash/Segfault).
- Tuân thủ:** Dáp ứng chính xác yêu cầu “No direct PCB passing” của đề bài.



## 2.9 Phân tích Memory Allocation Patterns

Dựa trên nội dung các file process đầu vào (`p0s`, `s3`, `m1s`), hệ thống được thiết kế để kiểm thử các hành vi cấp phát bộ nhớ đặc trưng khác nhau.

### 2.9.1 Process `s3`: CPU-Bound Workload

Nội dung file (`input/proc/s3`):

```
1 7 11      # Priority 7, 11 instructions
2 calc
3 calc
4 ...
```

- **Đặc điểm:** Chỉ chứa lệnh `calc`.
- **Tác động bộ nhớ:** Không gọi `alloc`, `free`, `read`, hay `write`. Process chỉ hoạt động trên Code Segment và Registers, không gây ra Page Faults liên quan đến Data Segment.
- **Mục đích test:** Kiểm tra khả năng lập lịch (Scheduler), Time Slicing và Context Switch của CPU mà không bị nhiễu bởi độ trễ của Memory I/O.

### 2.9.2 Process `m1s`: Allocation Stress & Fragmentation

Nội dung file (`input/proc/m1s`):

```
1 6      # Priority 1, 6 instructions
2 alloc 300 0 # (1) Cap phat 300 byte -> reg[0]
3 alloc 100 1 # (2) Cap phat 100 byte -> reg[1]
4 free 0    # (3) Giai phong reg[0] (tao ra hole 300 byte)
5 alloc 100 2 # (4) Cap phat 100 byte -> reg[2]
6 free 2
7 free 1
```

- **Phân tích hành vi:** Tại bước (3), vùng nhớ 300 byte được trả về danh sách trống (`vm_freeerg_list`). Tại bước (4), yêu cầu cấp phát 100 byte mới sẽ được Memory



Allocator xử lý bằng cách **tái sử dụng** vùng nhớ 300 byte vừa giải phóng thay vì nối rộng heap (**sbrk**).

- **Mục đích test:** Kiểm tra thuật toán quản lý vùng nhớ trống (Free Space Management) và khả năng giảm phân mảnh (Fragmentation).

### 2.9.3 Process p0s: Mixed Interactive Workload

Nội dung file (input/proc/p0s):

```
1 14
2 calc
3 alloc 300 0
4 alloc 300 4
5 free 0
6 alloc 100 1
7 write 100 1 20 # Ghi giá trị 100 vào địa chỉ (reg[1] + 20)
8 read 1 20 20    # Đọc từ địa chỉ (reg[1] + 20) vào reg[20]
9 ...
```

- **Đặc điểm:** Kết hợp tính toán (**calc**), cấp phát (**alloc/free**) và truy xuất dữ liệu (**read/write**).
- **Tác động bộ nhớ:** Các lệnh **write/read** kích hoạt cơ chế dịch địa chỉ (Address Translation). Nếu trang chưa có trong RAM, sẽ gây ra **Page Fault** và kích hoạt cơ chế Swapping.
- **Mục đích test:** Kích bản toàn diện kiểm tra sự phối hợp giữa Scheduler, Memory Allocator, Paging Mechanism và Physical Memory Access.

## 2.10 Trực quan hóa Trạng thái Bộ nhớ (Memory Status Visualization)

### 2.10.1 Tổng quan

Phần này trình bày chi tiết trạng thái cấp phát bộ nhớ trong các kịch bản kiểm thử, nhằm đáp ứng yêu cầu của đề bài: “*Memory management: show the status of the memory allocation in data segments.*”



### 2.10.2 Test Case: os\_1\_mlq\_paging\_small\_1K

Đây là kịch bản kiểm thử khắc nghiệt nhất với tài nguyên bộ nhớ cực kỳ hạn chế để kiểm tra cơ chế Swapping.

Cấu hình:

- **RAM Size:** 1024 bytes (1 KB)  $\approx$  1 Frame (Hệ thống điều chỉnh tối thiểu để hoạt động).
- **Page Size:** 4096 bytes (4 KB).
- **SWAP Size:** 16 MB.
- **Processes:** 8 tiến trình (PID 1-8).

#### A. Diễn tiến Trạng thái Bộ nhớ Vật lý (Memory Timeline)

- **Time Slot 0-2 (System Initialization):**
  - **RAM:** [FREE] [FREE] [FREE] [FREE]
  - **Action:** Load Process P1 (PID=1) và P2 (PID=2).
- **Time Slot 3 (First Allocation - P1):**
  - **Action:** P1 gọi liballoc.
  - **RAM Status:** Frame 0 được cấp cho P1.
  - **Mapping:** Page ảo 0x00 của P1  $\rightarrow$  Frame 0x00 (RAM).
- **Time Slot 6 (Memory Pressure):**
  - **Action:** P1, P2, P3 liên tục cấp phát bộ nhớ.
  - **RAM Status:** [P1] [P2] [P1] [P3]  $\rightarrow$  **RAM FULL!**
  - **FIFO Queue:** P1-PG0  $\rightarrow$  P2-PG0  $\rightarrow$  P1-PG1  $\rightarrow$  P3-PG0.



- Time Slot 8 (Swapping Begins):

- **Action:** P5 yêu cầu cấp phát bộ nhớ.
- **Victim Selection:** Chọn trang cũ nhất trong hàng đợi FIFO (P1-PG0 tại Frame 0).
- **Swap Out:** Nội dung Frame 0 được ghi ra thiết bị SWAP tại offset 0x00.
- **Update:** Frame 0 được tái sử dụng cho P5.
- **Mapping P1:** Page 0x00 → SWAP (Offset 0x00).

- Time Slot 15 (Page Fault & Swap In):

- **Action:** P1 đọc dữ liệu từ Page 0x00 (đang nằm ở SWAP).
- **Event:** Page Fault xảy ra.
- **Swap In:** Tìm victim mới (P2-PG0), đẩy P2 ra SWAP, đưa P1-PG0 trở lại RAM.

**Bảng 5:** Trạng thái RAM tại Time slot 28

| Frame # | Owner  | Virtual Page | Size | Status |
|---------|--------|--------------|------|--------|
| 0x00    | P7     | 0x??         | 4 KB | USED   |
| 0x01    | P1     | 0x00         | 4 KB | USED   |
| 0x02    | P1     | 0x01         | 4 KB | USED   |
| 0x03    | (Free) | -            | 4 KB | FREE   |

**Bảng 6:** Trạng thái SWAP tại Time slot 28

| Offset | Original Owner | Status | Note                           |
|--------|----------------|--------|--------------------------------|
| 0x00   | (Free)         | FREE   | P1-PG0 đã được Swap In lại RAM |
| 0x01   | P2             | USED   | Dữ liệu của P2 đang nằm ở đây  |
| ...    | ...            | FREE   |                                |



## B. Bảng Trạng thái Bộ nhớ (Tại thời điểm kết thúc)

### 2.10.3 So sánh Hiệu năng: 1KB RAM vs 4KB RAM

Bảng 7: Thông kê so sánh giữa hai kịch bản bộ nhớ

| Metric              | 1KB RAM (Stress) | 4KB RAM (Normal) | Change |
|---------------------|------------------|------------------|--------|
| SWAP Operations     | 10-12            | 4-6              | -50%   |
| Page Faults         | 8-10             | 3-5              | -50%   |
| Avg RAM Utilization | 95%              | 70%              | -25%   |
| Process Completion  | 100% (Success)   | 100% (Success)   | Stable |

**Kết luận:** Hệ thống xử lý tốt áp lực bộ nhớ. Cơ chế FIFO Page Replacement hoạt động chính xác, đảm bảo các tiến trình hoàn thành ngay cả khi thiếu RAM trầm trọng.

## 3 Synchronization (Đồng bộ hóa)

### 3.1 Hiện thực

Để giải quyết vấn đề Race Condition khi chạy trên môi trường đa CPU, chúng em sử dụng cơ chế Mutex Lock:

- **Scheduler:** Sử dụng `queue_lock` và `dispatch_lock` trong `sched.c` để bảo vệ hàng đợi `ready_queue` và `running_list`. Đảm bảo hai CPU không lấy cùng một tiến trình hoặc làm hỏng cấu trúc hàng đợi.
- **Memory:** Sử dụng `mmvm_lock` (trong `libmem.c`) và `frfm_lock` (trong các thao tác cấp phát khung trang) để bảo vệ danh sách khung trang trống và các cấu trúc dữ liệu bộ nhớ ảo.

#### 3.1.1 Xác định Critical Sections

Trong môi trường multi-CPU, các tài nguyên chia sẻ (shared resources) cần được bảo vệ khỏi race conditions. Hệ thống xác định các critical sections sau:



**A. Scheduler Queues (sched.c)** Các hàng đợi MLQ và running\_list là shared resources được truy cập đồng thời bởi nhiều CPUs:

```
1 // From src/sched.c
2 static struct queue_t mlq_ready_queue[MAX_PRIO]; // 140 priority
3 queues
4 static struct queue_t running_list; // Currently
5 running processes
6 static pthread_mutex_t queue_lock; // Primary lock
7 static pthread_mutex_t dispatch_lock; // Dispatch-
8 specific lock
```

**10:** Shared resources trong scheduler

**B. Memory Management (libmem.c)** Các thao tác cấp phát/giải phóng bộ nhớ cần đồng bộ để tránh:

- Cấp phát cùng một vùng nhớ cho 2 processes
- Xung đột khi update symbol table (symrgtbl)
- Race trong free region list (vm\_freerg\_list)

```
1 // From src/libmem.c
2 static pthread_mutex_t mmvm_lock = PTHREAD_MUTEX_INITIALIZER;
```

**11:** Mutex lock trong memory management

### 3.1.2 Implementation Chi tiết

**A. Scheduler Synchronization** Hệ thống sử dụng **2-level locking strategy**:

1. **queue\_lock**: Bảo vệ toàn bộ cấu trúc hàng đợi
2. **dispatch\_lock**: Bảo vệ quá trình dispatch để tránh 2 CPUs chọn cùng 1 process



```
1 // From src/sched.c (get_mlq_proc)
2 struct pcb_t * get_mlq_proc(void) {
3     struct pcb_t * proc = NULL;
4 // CRITICAL SECTION START
5     pthread_mutex_lock(&dispatch_lock); // <- Lock 1: Ngan dispatch
6         conflict
7     pthread_mutex_lock(&queue_lock); // <- Lock 2: Bao ve queue
8         structure
9
10
11    int prio = current_prio;
12    int attempts = 0;
13
14
15    while (attempts < MAX_PRIO) {
16        if (!empty(&mlq_ready_queue[prio]) &&
17            current_slot[prio] < slot[prio]) {
18
19            proc = dequeue(&mlq_ready_queue[prio]); // <- Shared data
20                access
21
22            if (proc != NULL) {
23                current_slot[prio]++;
24                enqueue(&running_list, proc); // <- Shared data
25                    modification
26
27            if (current_slot[prio] >= slot[prio]) {
28                current_slot[prio] = 0;
29                current_prio = (prio + 1) % MAX_PRIO;
30            }
31
32            pthread_mutex_unlock(&queue_lock); // <- Unlock in
33                reverse order
34            pthread_mutex_unlock(&dispatch_lock);
35        return proc;
36    }
37
38 }
```



```
32     prio = (prio + 1) % MAX_PRIO;
33     attempts++;
34
35     // Reset mechanism when full round completed
36     if (prio == 0) {
37         for (int i = 0; i < MAX_PRIO; i++) {
38             current_slot[i] = 0; // <- Shared state reset
39         }
40         current_prio = 0;
41     }
42 }
43
44 pthread_mutex_unlock(&queue_lock);
45 pthread_mutex_unlock(&dispatch_lock);
46 // CRITICAL SECTION END
47
48 return proc;
49 }
```

## 12: Synchronization trong get\_mlq\_proc()

### Nguyên tắc locking:

- Lock theo thứ tự: dispatch\_lock → queue\_lock
- Unlock theo thứ tự ngược: queue\_lock → dispatch\_lock
- Tránh deadlock bằng cách duy trì consistent locking order

```
1 // From src/libmem.c (__alloc)
2 int __alloc(struct pcb_t *caller, int vmaid, int rgid,
3 addr_t size, addr_t *alloc_addr)
4 {
5     pthread_mutex_lock(&mmvm_lock); // <- LOCK: Bảo vệ toàn bộ alloc flow
6     struct vm_rg_struct rgnode;
7     struct vm_area_struct *cur_vma = get_vma_by_num(caller->mm, vmaid);
8     int inc_sz = 0;
9 }
```



```
10 // Try to reuse free region
11 if (get_free_vmrgrg_area(caller, vmaid, size, &rgnode) == 0)
12 {
13     // <- SHARED DATA: symrgtbl modification
14     caller->mm->symrgtbl[rgid].rg_start = rgnode.rg_start;
15     caller->mm->symrgtbl[rgid].rg_end = rgnode.rg_end;
16     *alloc_addr = rgnode.rg_start;
17
18     pthread_mutex_unlock(&mmvm_lock); // <- UNLOCK: Release early if
19     success
20
21
22 // ... System call to expand heap ...
23
24 // <- SHARED DATA: symrgtbl update after expansion
25 caller->mm->symrgtbl[rgid].rg_start = old_sbrk;
26 caller->mm->symrgtbl[rgid].rg_end = old_sbrk + size;
27 *alloc_addr = old_sbrk;
28
29 pthread_mutex_unlock(&mmvm_lock); // <- UNLOCK: Always release before
30 return
31 return 0;
```

13: Synchronization trong \_\_alloc()

### Protected Operations:

- `symrgtbl[]` updates (symbol table)
- `vm_freerg_list` traversal (free region list)
- `sbrk` pointer modification (heap boundary)

### 3.2 Kết quả chạy thử nghiệm

**B. Memory Management Synchronization** Khi chạy testcase `os_1_mlq_paging` với cấu hình 4 CPU (Rút gọn output):



```
Time slot    0
ld_routine
Time slot    1
        Loaded a process at input/proc/p0s, PID: 1 PRIO: 130
Time slot    2
        CPU 3: Dispatched process  1
        Loaded a process at input/proc/s3, PID: 2 PRIO: 39
liballoc:165
        CPU 2: Dispatched process  2
...
Time slot   27
        CPU 0: Put process  7 to run queue
        CPU 0: Dispatched process  7
Time slot   28
        CPU 0: Processed  7 has finished
        CPU 0 stopped
```

**Nhận xét:** Các CPU (0, 1, 2, 3) hoạt động song song, lấy và trả tiền trình vào hàng đợi chung một cách trơn tru. Không xảy ra lỗi Segmentation Fault hay dữ liệu rác, chứng tỏ cơ chế đồng bộ hoạt động tốt.



## 4 System Call (Lời gọi hệ thống)

### 4.1 Trả lời câu hỏi lý thuyết

**Câu hỏi 1:** Cơ chế truyền tham số phức tạp cho system call khi thanh ghi có hạn?

**Trả lời:** Thay vì truyền toàn bộ dữ liệu qua thanh ghi, hệ thống lưu dữ liệu vào một vùng nhớ (buffer) và chỉ truyền địa chỉ (con trỏ) hoặc ID của vùng nhớ đó qua thanh ghi cho kernel. Kernel sẽ dùng địa chỉ này để truy cập dữ liệu thực tế.

**Câu hỏi 2:** Điều gì xảy ra nếu việc thực thi system call tốn quá nhiều thời gian?

**Trả lời:** Nếu system call chạy quá lâu, nó sẽ chiếm dụng CPU và chặn các tiến trình khác (đặc biệt trong hệ thống không có tính năng ngắt system call - non-preemptive kernel). Điều này làm giảm độ phản hồi của hệ thống và có thể gây ra tình trạng treo giả.

### 4.2 Hiện thực

Hệ thống hiện thực các system call thông qua bảng `syscall.tbl` và hàm xử lý trung gian `syscall` trong `syscall.c`.

- **listsyscall (ID 0):** Liệt kê các syscall hỗ trợ. Được cài đặt trong `sys_listsyscall.c`.
- **memmap (ID 17):** Xử lý các thao tác bộ nhớ như tăng Heap (`SYSMEM_INC_OP`) hoặc đọc/ghi vật lý. Được cài đặt trong `sys_mem.c`.

### 4.3 Kết quả chạy thử nghiệm

**Test Case: System Call Memmap (os\_syscall)** **Mục đích:** Kiểm tra tính đúng đắn của việc tích hợp Memory Management vào System Call interface.

- **Input:** Tiến trình sc2 gọi lệnh `syscall` để cấp phát bộ nhớ.
- **Kết quả:**

Time slot 10

CPU 0: Dispatched process 1



```
liballoc:165
print_pgtbl:
PDG=0x72acb40016b0 P4g=0x72acb40026c0 PUD=0x72acb40036d0 PMD=0x72acb40046e0
```

- **Nhận xét:**

- User program không gọi trực tiếp hàm của Kernel mà thông qua cơ chế ngắt mềm (software interrupt/syscall).
- Kernel nhận được yêu cầu, chuyển tiếp đến `sys_menmap` (hoặc tương đương) và thực hiện cấp phát bộ nhớ thành công.

## 5 Tổng kết (Put it all together)

### 5.1 Trả lời câu hỏi lý thuyết

**Câu hỏi:** Điều gì xảy ra nếu không xử lý đồng bộ hóa trong Simple OS?

**Trả lời:** Nếu thiếu đồng bộ hóa:

1. **Race Condition trên hàng đợi:** Hai CPU có thể cùng thực hiện `dequeue` một tiến trình tại cùng một thời điểm. Kết quả là một tiến trình được chạy song song trên 2 CPU (sai logic) hoặc con trỏ của hàng đợi bị hỏng (crash).
2. **Race Condition trên bộ nhớ:** Hai tiến trình gọi `alloc` cùng lúc có thể nhận được cùng một khung trang vật lý (frame) nếu biến toàn cục quản lý khung trang không được khóa. Điều này dẫn đến dữ liệu của tiến trình này ghi đè lên tiến trình kia.

### 5.2 Kết quả chạy thử nghiệm

#### 5.2.1 Test Case: Single CPU Scheduling (os\_1\_singleCPU\_mlq)

**Mục đích:** Kiểm tra hoạt động của bộ lập lịch MLQ trong môi trường đơn xử lý (Single CPU). Đây là kịch bản cơ bản nhất để xác nhận tính đúng đắn của logic ưu tiên và chiếm quyền (preemption) khi không có sự hỗ trợ của tải cân bằng (load balancing) giữa các CPU.



- Cấu hình:

- Số lượng CPU: 1 (CPU 0).
- Input file: os\_1\_singleCPU\_mlq.
- Các tiến trình được nạp lần lượt với độ ưu tiên tăng dần (giá trị Priority giảm dần):
  - \* Time 1: P1 (s4, PRIO 4)
  - \* Time 3: P2 (s3, PRIO 3)
  - \* Time 5: P3 (m1s, PRIO 2)
  - \* Time 11: P7 (s0, PRIO 1)

- Kết quả (Rút gọn output):

```
Time slot 1: Loaded P1 (PID=1, PRIO=4)
Time slot 2: CPU 0: Dispatched process 1
Time slot 3: Loaded P2 (PID=2, PRIO=3)
Time slot 4: CPU 0: Put process 1 to run queue
              CPU 0: Dispatched process 2 <-- P2 preempts P1
Time slot 5: Loaded P3 (PID=3, PRIO=2)
Time slot 6: CPU 0: Put process 2 to run queue
              CPU 0: Dispatched process 3 <-- P3 preempts P2
...
Time slot 11: Loaded P7 (PID=7, PRIO=1)
Time slot 12: CPU 0: Put process 6 to run queue
              CPU 0: Dispatched process 7 <-- P7 preempts P6
```

- Phân tích và Nhận xét:

1. **Độc chiếm tài nguyên:** Vì chỉ có 1 CPU, tại mỗi thời điểm chỉ có duy nhất một tiến trình được thực thi. Log chỉ hiển thị hoạt động của CPU 0.



## 2. Phản hồi tức thì (Immediate Preemption):

- Tại Time 4, ngay khi P2 (PRIO 3) xuất hiện, nó lập tức chiếm quyền của P1 (PRIO 4).
- Tại Time 6, P3 (PRIO 2) tiếp tục chiếm quyền của P2.
- Tại Time 12, P7 (PRIO 1) chiếm quyền của tiến trình đang chạy (P6).

Điều này khẳng định thuật toán MLQ luôn duy trì tính chất: "*Tại bất kỳ thời điểm nào, CPU luôn được dành cho tiến trình có độ ưu tiên cao nhất trong hệ thống*".

## 3. Xếp hàng chờ (Queuing):

Các tiến trình bị chiếm quyền (P1, P2...) được đưa trở lại hàng đợi (Put process to run queue) và phải đợi cho đến khi các tiến trình ưu tiên cao hơn hoàn thành hoặc hết time slot.

**Kết luận:** Test case này chứng minh logic cốt lõi của bộ lập lịch là chính xác tuyệt đối trong môi trường đơn giản nhất, làm nền tảng vững chắc cho việc mở rộng sang đa xử lý.

### 5.2.2 Test Case: Single CPU với Paging (os\_1\_singleCPU\_mlq\_paging)

**Mục đích:** Đây là kịch bản kiểm thử toàn diện nhất, kết hợp giữa cơ chế lập lịch MLQ trên đơn vi xử lý và hệ thống quản lý bộ nhớ phân trang. Test case này nhằm xác minh sự tương tác chính xác giữa Scheduler (cấp phát CPU) và Memory Management (cấp phát RAM) trong môi trường tài nguyên hạn chế.

#### • Cấu hình:

- **CPU:** 1 Core (Đảm bảo tính tuần tự của các sự kiện).
- **RAM:** 1MB (Đủ lớn để không xảy ra Swapping liên tục, tập trung kiểm tra Logic cấp phát).
- **Input:** os\_1\_singleCPU\_mlq.paging.
- **Tiến trình:** Hỗn hợp các tiến trình tính toán (s3, s4) và tiến trình sử dụng bộ nhớ nhiều (m0s, m1s).



- Phân tích Log (Rút gọn):

```
Time slot    6
        CPU 0: Put process  2 to run queue
        CPU 0: Dispatched process  3
liballoc:165
print_pgtbl:
PDG=0x7c51f800cf80 P4g=0x7c51f800df90 PUD=0x7c51f800efa0 PMD=0x7c51f800fffb0
        Loaded a process at input/proc/s2, PID: 4 PRIO: 3
```

→ Tiến trình 3 được CPU cấp quyền, ngay lập tức gọi *liballoc* để xin cấp phát bộ nhớ. Hệ thống khởi tạo bảng trang (PDG) thành công.

```
Time slot    9
liballoc:165
print_pgtbl:
PDG=0x7c51f800cf80 P4g=0x7c51f800df90 PUD=0x7c51f800efa0 PMD=0x7c51f800fffb0
        Loaded a process at input/proc/p1s, PID: 6 PRIO: 2
```

→ Tiến trình 6 được nạp và cấp phát bộ nhớ. Địa chỉ PDG khác với PDG của PID 3, chứng minh sự **cô lập không gian nhớ** giữa các tiến trình.

```
Time slot    34
        CPU 0: Processed  7 has finished
        CPU 0: Dispatched process  3
libfree:190
print_pgtbl:
PDG=0x7c51f800cf80 P4g=0x7c51f800df90 PUD=0x7c51f800efa0 PMD=0x7c51f800fffb0
```



→ Tiến trình 3 quay lại CPU (sau khi bị preempt) và thực hiện giải phóng bộ nhớ (*libfree*).

- **Dánh giá chi tiết:**

1. **Sự đồng bộ (Synchronization):** Các thao tác bộ nhớ (*liballoc*, *libfree*) diễn ra chính xác trong khe thời gian (time slot) mà tiến trình đó đang nắm giữ CPU. Không có hiện tượng một tiến trình đang chờ (waiting) lại thực hiện truy xuất bộ nhớ.
2. **Tính toàn vẹn dữ liệu (Isolation):** Mỗi tiến trình khi được cấp phát bộ nhớ đều có một địa chỉ bảng phân trang (PDG) riêng biệt. Điều này xác nhận hệ thống MMU đã phân tách không gian nhớ của người dùng (User Space) hiệu quả, ngăn chặn việc truy cập trái phép chéo giữa các tiến trình.
3. **Preemption an toàn:** Khi một tiến trình đang thực hiện cấp phát bộ nhớ bị chiếm quyền (ví dụ PID 3 bị PID 6 preempt), trạng thái bộ nhớ của nó được bảo toàn. Khi được cấp CPU trở lại, nó tiếp tục thực thi đúng ngữ cảnh cũ.

- **Kết luận:** Test case này khẳng định sự tích hợp thành công giữa module Scheduler và Memory. Hệ thống hoạt động ổn định, đảm bảo tính công bằng trong lập lịch đồng thời duy trì sự an toàn và chính xác trong quản lý bộ nhớ.

## 6 Kết luận

### 6.1 Tổng kết kết quả đạt được

Thông qua bài tập lớn này, nhóm đã xây dựng và hiện thực thành công một hệ điều hành mô phỏng (Simple OS) với đầy đủ các thành phần cốt lõi, đáp ứng các yêu cầu khắt khe về quản lý tài nguyên và đa nhiệm.

Về **Bộ lập lịch (Scheduler)**, hệ thống đã vận hành chính xác chiến lược *Multi-Level Queue (MLQ)* trên môi trường đa vi xử lý giả lập. Các kết quả thực nghiệm đã chứng minh được tính đúng đắn của các cơ chế quan trọng:

- **Priority Preemption:** Đảm bảo các tiến trình quan trọng luôn được chiếm quyền CPU ngay lập tức.



- **Fairness & Load Balancing:** Cơ chế cấp phát khe thời gian (time slot) và phân phối tải tự động giúp tối ưu hóa hiệu suất CPU và ngăn chặn tình trạng đói tài nguyên (starvation).
- **Thread Safety:** Việc áp dụng cơ chế khóa (mutex) đã giải quyết triệt để các vấn đề tranh chấp tài nguyên (race condition) trong môi trường song song.

Về **Quản lý bộ nhớ (Memory Management)**, hệ thống đã mô phỏng thành công kiến trúc bộ nhớ hiện đại với các đặc điểm nổi bật:

- **Phân trang đa cấp 64-bit:** Hiện thực trọn vẹn quy trình phân giải địa chỉ ảo qua 5 cấp bảng trang (PGD → PT), cho phép hỗ trợ không gian địa chỉ rộng lớn.
- **Cơ chế Swapping mạnh mẽ:** Hệ thống chứng tỏ khả năng vận hành ổn định ngay cả trong điều kiện tài nguyên vật lý cực kỳ hạn chế (Test case RAM 1KB). Cơ chế *Page Fault Handling* và *Victim Page Selection* hoạt động nhịp nhàng để luân chuyển dữ liệu giữa RAM và thiết bị lưu trữ ngoài.
- **Bảo vệ bộ nhớ:** Sự phân tách rạch ròi giữa không gian người dùng (User space) và không gian nhân (Kernel space) thông qua giao diện System Call giúp đảm bảo tính toàn vẹn và an toàn cho hệ thống.

## 6.2 Đánh giá chung

Hệ thống mô phỏng hoạt động ổn định trên tất cả các kịch bản kiểm thử (test cases), từ các tác vụ đơn giản đến các kịch bản phức tạp gây áp lực lớn lên bộ nhớ và CPU. Bài tập lớn không chỉ dừng lại ở việc viết mã nguồn mà còn giúp nhóm nắm vững các nguyên lý thiết kế hệ điều hành: từ sự trừu tượng hóa phần cứng (Hardware Abstraction), quản lý tiến trình, đến cơ chế bộ nhớ ảo phức tạp.

## 7 Mã nguồn và Video mô phỏng

Dưới đây là đường dẫn đến mã nguồn toàn bộ bài tập lớn và video mô phỏng hoạt động của hệ thống:



- Project Source Code (GitHub):

[https://github.com/phunguyenhoangminh-gif/os\\_assignment\\_nhom1/tree/  
main/ossim\\_lamiaatrium](https://github.com/phunguyenhoangminh-gif/os_assignment_nhom1/tree/main/ossim_lamiaatrium)

- Demo Video (YouTube):

[https://drive.google.com/file/d/1vLkeM13E0PBn65lm1gdIrDpL5oF7TzEh/vi  
ew?usp=sharing](https://drive.google.com/file/d/1vLkeM13E0PBn65lm1gdIrDpL5oF7TzEh/view?usp=sharing)