# المقاطعات Interrupts

المقاطعات هي طريقة لإيقاف المعالج بشكل مؤقت من تنفيذ عملية ما (Current Process) والبدء بتنفيذ أوامر أخرى . وكمثال على ذلك هو عند الضغط على أي حرف في لوحة المفاتيح فان هذا يولد مقاطعة (Interrupt) تأتي كإشارة الى المعالج بأن يوقف ما يعمل عليه حاليا ويحفظ كل القيم التي يحتاجها لكي يستطيع مواصلة ما تم قطعه ، وفي حالة وجود دالة للتعامل مع هذه المقاطعة (مقاطعة لوحة المفاتيح) وتسمى دالة معالجة المقاطعة (Interrupt Handler) أو دالة خدمة المقاطعة (مثلا يتم قراءة الحرف الذي تم ادخاله من متحكم لوحة يتنقل اليها تلقائيا ، و يتم فيها معالجة هذه المقاطعة (مثلا يتم قراءة الحرف الذي تم ادخاله من متحكم لوحة المفاتيح ومن ثم ارساله الى متغير في الذاكرة) وعندما تنتهي دالة معالجة المقاطعة من عملها فان المعالج يعود ليكمّل تنفيذ العملية التي كان يعمل عليها. والمقاطعات إما تكون مقاطعات عتادية (Hardware Interrupt) وتصدر من عتاد الحاسب أو تكون برمجية (Software Interrupt) وتصدر من خلال البرامج عن طريق تعليمة وتصدر من عتاد الحاسب أو تكون برمجية (Page Fault نفسه عند حدوث خطأ ما (مثلا عن القسمة على العدد صفر أو عند حدوث الأخطاء (Page Fault) وتسمى هذه المقاطعات بأخطاء المعالج أو استثنائات المعالج العالجة هذه الأخطاء (Error Handler) لألها توقف عمل النظام في حالة لم تتوفر دالة لمعالجةها.

# ۱.۱ المقاطعات البرمجية Software Interrupts

المقاطعات البرمجية هي مقاطعات يتم اطلاقها من داخل البرنامج (عن طريق الأمر int n) لِنقل التنفيذ الى دالة أخرى تعالج هذه المقاطعة (Interrupt handler)، وغالبا ما تستخدم هذه المقاطعات في برامج المستخدم (Ring3 user mode) للاستفادة من خدمات النظام (مثلا للقراءة والكتابة في أجهزة الإدخال والإخراج حيث لا توجد طريقة اخرى لذلك في نمط المستخدم).

# ١.١.١ المقاطعات في النمط الحقيقي

في النمط الحقيقي عندما يتم تنفيذ أمر المقاطعة (وهو ما يسمى بطلب تنفيذ المقاطعة (Interrupt Request) فإن المعالج يأخذ رقم المقاطعة المطلوب تنفيذها ويذهب بها الى جدول المقاطعات (IRQ وتختصر بـــ (IRQ) فإن المعالج يأخذ رقم المقاطعة المطلوب تنفيذها ويذهب بها الى جدول المقاطعات (Vector Table ويحوي كل سجل (Vector Table )، هذا الجدول يبدأ من العنوان الحقيقي  $0 \times 0$  وينتهي عند العنوان المخدول يبدأ من العنوان هو أربع فيه على عنوان دالة معالجة المقاطعة (IR) والتي يجب تنفيذها لتخديم المقاطعة المطلوبة. حجم العنوان هو أربع بايت وتكون كالتالي:

جدو ل ۱.۱: Interrupt Vector Table

| miterrupt vector ruble : 1: 1 6 3-5:     |                  |               |  |  |  |
|------------------------------------------|------------------|---------------|--|--|--|
| Description                              | Interrupt Number | Base Address  |  |  |  |
| Divide by 0                              | 0                | 0x000         |  |  |  |
| Single step (Debugger)                   | 1                | 0x004         |  |  |  |
| Non Maskable Interrupt (NMI) Pin         | 2                | 0x008         |  |  |  |
| Breakpoint (Debugger)                    | 3                | 0x00C         |  |  |  |
| Overflow                                 | 4                | 0x010         |  |  |  |
| Bounds check                             | 5                | 0x014         |  |  |  |
| Undefined Operation Code                 | 6                | 0x018         |  |  |  |
| No coprocessor                           | 7                | 0x01C         |  |  |  |
| Double Fault                             | 8                | 0x020         |  |  |  |
| Coprocessor Segment Overrun              | 9                | 0x024         |  |  |  |
| nvalid Task State Segment (TSS) ${ m I}$ | 10               | 0x028         |  |  |  |
| Segment Not Present                      | 11               | 0x02C         |  |  |  |
| Stack Segment Overrun                    | 12               | 0x030         |  |  |  |
| General Protection Fault (GPF)           | 13               | 0x034         |  |  |  |
| tPage Faul                               | 14               | 0x038         |  |  |  |
| Unassigned                               | 15               | 0x03C         |  |  |  |
| Coprocessor error                        | 16               | 0x040         |  |  |  |
| Alignment Check (486+ Only)              | 17               | 0x044         |  |  |  |
| Machine Check (Pentium/586+ Only)        | 18               | 0x048         |  |  |  |
| Reserved exceptions                      | 19-31            | 0x05C         |  |  |  |
| Interrupts free for software use         | 32-255           | 0x068 - 0x3FF |  |  |  |

- Byte 0: Low offset address of IR.
- Byte 1: High offset address of IR.
- Byte 2: Low Segment address of IR.
- Byte 3: High Segment Address of IR.

ويتكون الجدول من 256 مقاطعة (وبحسبة بسيطة يكون حجم الجدول هو 1024 بايت وهي ناتجة من ضَّربٌ عدد الْمُقاطعات في حجم كلُ سجل )، بعض منَّها محجوِّز والبَّعض الاخر يستخدمه المعَّالج والبقية متروكة لمبرمج نظام التشغيل لدعم المزيد من المقطاعات. وبسبب أن الجدول يتكون فقط من عناوين لدوال معالجة المقاطعات فأن هذا يمكننا من وضع الدالة في أي مكّان على الذاكرة ومن ثم وضع عنوالها داخل هذا السجل (يتم هذا عن طريق مقاطعات البايوس)، والجدول ١٠١ يوضح ١٧٢ والمقاطعات الموجودة فيه.

# ٢.١.١ المقاطعات في النمط المحمي

في النمط المحمي يستخدم المعالج حدولاً خاصاً يسمى بجدول واصفات المقاطعات (Interrupt Descriptor) ويختصر ب IDT ، هذا الجدول يشابه حدول IVT حيث يتكون من 256 واصفة كل واصفة مخصصة لمقاطعة ما (اذاً الجدول يحوي 256 مقاطعة) ، حجم كل واصفة هو 8 بايت تحوي عنوان دالة معالجة المقاطعة (IR) و نوع الناحب (selector type: code or data) في حدول GDT الذي تعمل عليه دالة معالجة المقاطعة ، بالاضافة الى مستوى الحماية المطلوب والعديد من الخصائص توضحها التركيبة التالية.

- Bits 0-15:
  - Interrupt / Trap Gate: Offset address Bits 0-15 of IR
  - Task Gate: Not used.
- Bits 16-31:
  - Interrupt / Trap Gate: Segment Selector (Useually 0x10)
  - Task Gate: TSS Selector
- Bits 31-35: Not used
- Bits 36-38:
  - Interrupt / Trap Gate: Reserved. Must be 0.
  - Task Gate: Not used.
- Bits 39-41:
  - Interrupt Gate: Of the format 0D110, where D determins size
    - \* 01110 32 bit descriptor
    - \* 00110 16 bit descriptor
  - Task Gate: Must be 00101
  - Trap Gate: Of the format 0D111, where D determins size
    - \* 01111 32 bit descriptor
    - \* 00111 16 bit descriptor
- Bits 42-44: Descriptor Privedlge Level (DPL)
  - 00: Ring 0
  - 01: Ring 1
  - 10: Ring 2
  - 11: Ring 3

- Bit 45: Segment is present (1: Present, 0:Not present)
- Bits 46-62:

Interrupt / Trap Gate: Bits 16-31 of IR address

- Task Gate: Not used

والمثال التالي يوضح انشاء واصفة واحدة بلغة التجميع حتى يسهل تتبع القيم ، وسيتم كتابة مثال كامل لاحقا بلغة السي.

## Example \.\: Example of interrupt descriptor

```
v idt_descriptor:
                       0x0
    baseLow
                  dw
    selector
                    dw
                          0x8
    reserved
                          0x0
                    ďb
    flags
                  db
                       0x8e
                                       ; 010001110
    baseHi
                  dw
                       0x0
```

المتغير الأول baseLow هو أول 16 بت من عنوان دالة معالجة المقاطعة IR ويكمل الجزء الاخر من العنوان المتغير baseHi وفي هذا المثال العنوان هو  $0 \times 0$ . معنى أن دالة تخديم المقاطعة ستكون في العنوان  $0 \times 0$ . وبما أن دالة معالجة (تخديم) المقاطعة تحوي شفرة برمجية للتنفيذ وليست بيانات (Data) فان قيمة المتغير GDT) فان تكون  $0 \times 0$  للإشارة الى ناخب الشفرة (Code Selector) في جدول الواصفات العام (GDT). أما المتغير flags فان قيمته هي  $0 \times 0$  مستوى الحماية هي  $0 \times 0$  الحماية صفر (Ring0).

وبعد أن يتم انشاء أغلب الواصفات بشكل متسلسل (في أي مكان على الذاكرة) ، يجب أن ننشئ حدول IDT وهذا يتم عن طريق حفظ عنوان أول واصفة في متغير وليكن idt\_start وعنوان نهاية الواصفات في المتغير وليكن idt\_end ومن ثم انشاء مؤشراً يسمى idt\_ptr والذي يجب أن يكون في صورة معينة بحيث يحفظ عنوان بداية الجدول ونهايته :

#### Example \. \ : Value to put in IDTR

```
v idt_ptr:
v limit dw idt_end - idt_start ; bits 0-15 is size of idt
v base dd idt_start ; base of idt
```

هذا المؤشر يجب أن يتم تحميله الى المسجل IDTR (وهو مسجل داخل المعالج) عن طريق تنفيذ الامر lidt \lidt بالشكل التالي [idt\_ptr].

'بعد تنفيذ هذا الأمر فان حدول المقاطعات سيتم استبداله بالجدول الجديد والذي نجد عنوانه بداخل المسجل idtr ، وهذا الأمر لا يُنفّذ إلاَّ اذا كانت قيمة العلم (CPL flag) هي صفر. وعند حدوث أي مقاطعة فان المعالج ينهي الأمر الذي يعمل عليه و يأخذ رقم المقاطعة ويذهب به الى جدول IDT (عنوان هذا الجدول يتواجد بداخل المسجل IDT) ، وبعد ذلك يقوم بحساب مكان الواصفة بالمعادلة IDT (عنوان هذا الجدول يتواجد بداخل المسجل IDT) ، وبعد ذلك يقوم بحساب مكان الواصفة بالمعادلة IDT) و IDT (IDT) و IDT) الى دالة معالجة المقاطعة فانه يجب أن يقوم بعملية حفظ الأعلام IDT) ومسجل مقطع الشفرة IDT) ومسجل عنوان عندما تعود دالة معالجة المقاطعة . ويتم حفظ الأعلام IDT) و IDT و IDT و IDT و IDT في المكدس أيضا. و شفرة الخطأ هي بطول IDT و IDT و IDT و IDT المكدس أيضا. و شفرة الخطأ هي بطول IDT و IDT و IDT المكدس أيضا.

- Bit 0: External event
  - 0: Internal or software event triggered the error.
  - − 1: External or hardware event triggered the error.
- Bit 1: Description location
  - 0: Index portion of error code refers to descriptor in GDT or current LDT.
  - 1: Index portion of error code refers to gate descriptor in IDT.
- Bit 2: GDT/LDT. Only use if the descriptor location is 0.
  - 0: This indicates the index portion of the error code refers to a descriptor in the current GDT.
  - 1: This indicates the index portion of the error code refers to a segment or gate descriptor in the LDT.
- Bits 3-15: Segment selector index. This is an index into the IDT, GDT, or current LDT to the segment or gate selector bring refrenced by the error code.
- Bits 16-31: Reserved.

وعندما تنتهي دالة معالجة المقاطعة من عملها فانه يجب أن تنفذ الأمر iret أو iret حتى يتم ارجاع القيم التي تم دفعها الى المكدس (قيم الأعلام FLAGS). وبالتالي يُكْمِل المعالج عمله.

## ٣.١.١ أخطاء المعالج

خلال تنفيذ المعالج للأوامر فانه ربما يحدث خطأ ما مما يجعل المعالج يقوم بتوليد استثناء يعرف باستثناء المعالج، ويوجد له عدة أنواع:

- الخطأ Fault: عندما تعمل دالة معالجة هذا النوع من الاستثناء فربما يتم اصلاح هذا الخطأ ، وعنوان العودة الذي يتم دفعه الى المكدس هو عنوان الأمر الذي تسبب في هذا الخطأ.
  - الخطأ Trap: عنوان العودة هو عنوان التعليمة التي تلي الأمر الذي تسبب في الخطأ.

Description Class Interrupt Number Divide by 0 **Fault** Trap/Fault Single step 1 Unclassed 2 Non Maskable Interrupt (NMI) Pin Breakpoint Trap 3 Overflow Trap 4 Bounds check Fault 5 **Unvalid OPCode** Fault 6 7 Device not available Fault **Double Fault** Abort 8 9 Coprocessor Segment Overrun Abort **Invalid Task State Segment** Fault 10 Segment Not Present Fault 11 Stack Fault Exception Fault 12 **General Protection Fault** Fault 13 14 Page Fault Fault 15 Unassigned x87 FPU Error Fault 16 Alignment Check 17 Fault Machine Check Abort 18 SIMD FPU Exception 19 Fault Reserved 20-31 Avilable for software use 32-255

جدو ل X86 Processor Exceptions Table : ۲.۱ جدو

• الخطأ Abort: لا يوجد عنوان للعودة ، ولن يكمل البرنامج عمله بعد انتهاء دالة معالجة الخطأ.

والجدول ٢.١ يوضح أخطاء المعالج والمقاطعات التي يقوم بتوليدها.

ويجدر بنا الوقوف على ملاحظة كنّا قد ذكرناها في الفصول السابقة وهي إلغاء المقاطعات (بواسطة الأمر cli) عند الانتقال الى النمط المحمى حتى لا يتسبب في حدوث خطأ General Protection Fault وبالتالي توقف النظام عن العمل وسبب ذلك هو أن عدم تنفيذ الأمر cli يعني أن المقاطعات العتادية مفعلة وبالتالي أي عتاد يمكنه أن يرسل مقاطعة الى المعالج لكي ينقل التنفيذ الى دالة تخديمها . وعند بداية الانتقال الى النمط المحمى فان جدول المقاطعات IDT لم يتم انشائه وأي محاولة لاستخدامه سيؤدي الى هذا الخطأ. أحد المتحكمات التي ترسل مقاطعات الى المعالج بشكل ثابت هو متحكم Prpgrammable Interval Timer وتختصر بمتحكم PIT وهي تمثل ساعة النظام System Timer بحيث ترسل مقاطعة بشكل دائم الى المعالج والذي بدوره ينقل التنفيذ الى دالة تخديم هذه المقاطعة . وبسبب أن جدول المقطاعات غير متواجد في بداية المرحلة الثانية من محمل النظام وكذلك لا توجد دالة لتخديم هذه المقاطعة فان هذا يؤدي الى توقف النظام ، لذلك يجب ايقاف المقاطعات العتادية لحين انشاء حدول المقطاعات وكتابة دوال معالجة المقاطعات. كذلك توجد مشكلة أحرى لبعض المقاطعات العتادية حيث الها تستخدم نفس أرقام المقاطعات التي يستخدمها المعالج للإستثناءات وحلها هو بإعادة برمجة الشريحة المسؤولة عن استقبال الاشارات من العتاد وتحويلها الى مقاطعات وارسالها الى المعالج ، هذه الشريحة تسمى Programmable Interrupt Controller وتختصر ب PIC ويجب إعادة برمجتها وتغيير ارقام المقاطعات للأجهزة التي تستخدم أرقاماً متشابحة.

وفيما يلي سيتم إنشاء حدول المقاطعات (IDT) باستخدام لغة السي وتوفير ال 256 دالة لمعالجة المقطاعات وحاليا سيقتصر عمل الدوال على طباعة رسالة ، وقبل ذلك سنقوم بانشاء حدول الواصفات العام (GDT) محددا (أي سيتم الغاء الجدول الذي قمنا بانشائه في مرحلة الاقلاع) وبعد ذلك سنبدأ في برمجة متحكم PIC وعدد ترقيم مقاطعات الأجهزة وكذلك برمجة ساعة النظام لارسال مقاطعة بوقت محدد.

## ٤.١.١ إنشاء جدول الواصفات العام GDT

الهدف الرئيسي في نواة نظام التشغيل هي المحمولية على صعيد المنصات ، وهذا ما أدى الى اعتماد فكرة طبقة HAL والتي يقبع تحتها كل ما يتعلق بعتاد الحاسب وادارته وكل ما يجعل النظام معتمداً على معمارية معينة أيضا نجده تحت طبقة HAL ، وحدول الواصفات العام - كما ذكرنا في الفصول السابقة - يحدد ويقسم لنا الذاكرة الرئيسية كأجزاء قابلة للتنفيذ وأجزاء تحوي بيانات وغيرها ، ونظراً لأن إنشاء هذا الجدول يعتمد على معمارية المعالج والأوامر المدعومة فيه فانه يجب ان يقع تحت طبقة HAL وهذا يعني أن نقل النظام الى معمارية حاسوب آخر يتطلب فقط إعادة برمجة طبقة HAL . بداية سنبدأ بتصميم الواجهة العامة لطبقة HAL ويجب أن نراعي أن تكون الواجهة مفصولة تماما عن التطبيق بداية سنبدأ بتصميم الواجهة العامة لطبقة HAL ويجب أن نراعي أن تكون الواجهة مفصولة تماما عن التطبيق

بداية سنبدأ بتصميم الواجهة العامة لطبقة HAL ويجب أن نراعي أن تكون الواجهة مفصولة تماما عن التطبيق حتى يتمكن أي مطور من إعادة تطبيقها لاحقاً على معمارية حاسوب آخر.

#### Example \.\\\: include/hal.h:Hardware Abstraction Layer Interface

```
" #ifndef HAL_H
" #define HAL_H
"
" #ifndef i386
" #error "HAL is not implemented in this platform"
" #endif
" #include <stdint.h>
" #ifdef _MSC_VER
" #define interrupt __declspec(naked)
" #else
```

من منظور آخر هذه الجداول (GDT,LDT and IDT) هي جداول للمعالج لذلك يجب أن تكون في طبقة HAL.

```
15 #define interrupt
√∘ #endif
w #define far
w #define near
     Interface */
rr extern int _cdecl hal_init();
rs extern int _cdecl hal_close();
ro extern void _cdecl gen_interrupt(int);
TA #endif // HAL_H
```

وحالياً واجهة طبقة HAL مكونة من ثلاث دوال تم الإعلان عنها بأنها extern وهذا يعني أن أي تطبيق (Implementation) لهذه الواجهة يجب أن يُعرِّف هذه الدوال. الدالة الاولى هي () hal\_init والتي تقوم بتهيئة العتاد وجداول المعالج بينما الدالة الثانية () hal\_close تقوم بعملية الحذف والتحرير وأخيرا الدالة gen\_interrupt والتي تم وضعها لغرض تحربة إرسال مقاطعة برمجية والتأكد من أن دالة معالجة المقاطعة

نعود بالحديثُ الى حدول الواصفات العام (GDT) ٣ حيث سيتم انشائه بلغة السي وهذا ما سيسمح لنا باستخدام تراكيب عالية للتعبير عن الجدول و المؤشر مما يعطى وضوح ومقروئية أكثر في الشفرة.وسوف نحتاج الى تعريف ثلاث دوال ؛:

- الدالة i386\_gdt\_init: تقوم بتهيئة واصفة خالية وواصفة للشفرة وللبيانات وكذلك انشاء مؤشر
- الدالة i386\_gdt\_set\_desc: دالة هيئة الواصفة حيث تستقبل القيم وتعينها الى الواصفة المطلوبة.
- الدالة gdt\_install: تقوم بتحميل المؤشر الذي يحوي حجم الجدول وعنوان بدايته الى المسجل .GDTR

و الشفرة التالية توضح كيفية انشاء الجدول°.

## Example \. \: hal/gdt.cpp:Install GDT

الغرض التنظيم والتقسيم لا أكثر ولا أقل.

°راجع شفرة النظام لقراءة ملف الرأس hal/gdt.h.

```
r #include <string.h>
r #include "gdt.h"
• static struct gdt_desc _gdt[MAX_GDT_DESC];
¬ static struct gdtr _gdtr;
4 static void gdt_install();
nr static void gdt_install() {
\r #ifdef _MSC_VER
15 _asm lgdt [_gdtr];
v∘ #endif
17 }
vw extern void i386_gdt_set_desc(uint32_t index,uint64_t base,uint64_t
     limit, uint8_t access, uint8_t grand) {
۱۹
    if ( index > MAX_GDT_DESC )
     return;
۲١
77
7 ٣
    // clear the desc.
    memset((void*) &_gdt[index], 0, sizeof(struct gdt_desc));
۲ ٤
۲0
    // set limit and base.
    _gdt[index].low_base = uint16_t(base & 0xffff);
۲٧
    _gdt[index].mid_base = uint8_t((base >> 16) & 0xff);
۲۸
    _gdt[index].high_base = uint8_t((base >> 24) & 0xff);
۲9
    _gdt[index].limit = uint16_t(limit & 0xffff);
۳.
   // set flags and grandularity bytes
٣٢
    _gdt[index].flags = access;
٣٣
    _gdt[index].grand = uint8_t((limit >> 16) & 0x0f);
٣٤
    _gdt[index].grand = _gdt[index].grand | grand & 0xf0;
40
m1 }
rx extern gdt_desc* i386_get_gdt_desc(uint32_t index) {
   if ( index >= MAX_GDT_DESC )
     return 0;
```

```
else
      return &_gdt[index];
٤٢
٤٣ }
٤٤
so extern int i386_gdt_init() {
٤٦
    // init _gdtr
٤٧
    _gdtr.limit = sizeof(struct gdt_desc) * MAX_GDT_DESC - 1;
٤٨
    _gdtr.base = (uint32_t)&_gdt[0];
    // set null desc.
    i386_gdt_set_desc(0,0,0,0,0);
    // set code desc.
٥٤
    i386_gdt_set_desc(1,0,0xffffffff,
      i386_gdt_code_desc|i386_gdt_data_desc|i386_gdt_readwrite|
          I386_GDT_MEMORY, // 10011010
      1386_GDT_LIMIT_HI | 1386_GDT_32BIT | 1386_GDT_4K
                                                                     //
٥٧
         11001111
    );
٦.
    // set data desc.
٦١
    i386_gdt_set_desc(2,0,0xffffffff,
٦٢
     i386_gdt_data_desc|i386_gdt_readwrite|i386_gdt_memory, //
٦٣
     I386_GDT_LIMIT_HI | I386_GDT_32BIT | I386_GDT_4K // 11001111
    );
٦٥
٦٦
    // install gdtr
٦٧
   gdt_install();
٦٨
    return 0;
٧١ }
```

# ۲.۱ متحكم المقاطعات القابل للبرمجة Programmable Interrupt Controller

السبب الرئيسي في تعطيل المقاطعات العتادية عند الإنتقال الى النمط المحمي (PMode) هو بسبب عدم توفر دوال لمعالجة المقاطعات في تلك اللحظة ، وحتى لو قمنا بتوفير ال ٢٥٦ دالة لمعالجة المقاطعات فان هنالك مشكلة استخدام نفس رقم المقاطعة لأكثر من غرض ، فمثلا مؤقتة النظام PIT التي ترسل مقاطعات بشكل دائم تستخدم المقاطعة رقم ٨ والتي هي أيضا أحد استثناءات المعالج ، لذلك في كلتا الحالات سيتم استدعاء دالة تخديم واحدة وهو شيء مرفوض تماماً. لذلك الحل الوحيد هو بإعادة برمجة المتحكم المسؤول عن استقبال الإشارات من متحكمات العتاد وتعيين أرقام مختلفة بخلاف تالك الأرقام التي يستخدمها المعالج للأخطاء والاستثناءات ، هذا المتحكم (انظر الشكل ١٠١) وظيفته هي استقبال إشارات من متحكمات العتاد ومن ثم يقوم بتقل التنفيذ اليها ، ويعرف هذا المتحكم .عتحكم Programmable Interrupt Controller ويعرف أيضا بالإسم ويعرف هذا البحث سنستخدم المسمى متحكم .PIC





## ۱.۲.۱ المقاطعات العتادية ۱.۲.۱

قبل أن نبدأ في الدخول في تفاصيل متحكم PIC يجب إعطاء نبذة عن المقاطعات العتادية حيث ذكرنا ألها مقاطعات تختلف عن المقاطعات البرمجية من ناحية أن مصدرها يكون من العتاد وليس من برنامج ما ، وهذا ما أدى الى ظهور لقب مسير للأحداث (Interrupt Driven) على أجهزة الحاسب. حيث قديمًا لم يكن هناك طريقة للتعامل مع العتاد إلا باستخدام حلقة برمجية (loop) على مسجل ما في متحكم العتاد حتى تتغير قيمته دلالة على أن هناك قيمة أو نتيجة قد حاءت من العتاد ، هذه الطريقة في التخاطب مع العتاد تسمى Polling وهي تضيع وقت المعالج في انتظار قيمة لا يُعرف هل ستظهر أم لا وقد تم إلغائها في التخاطب مع العتاد حيث الان أصبح أي متحكم عتاد يدعم إرسال الإشارات (وبالتالي المقاطعات) الى المعالج والذي قد يعمل على عملية أخرى ، وهكذا تم الإستفادة من وقت المعالج وأصبح التخاطب هو غير متزامن (Asynchronous). وعندما يبدأ الحاسب في الإقلاع فان نظام البايوس يقوم بترقيم عتاد الحاسب وإعطاء رقم مقاطعة لكل متحكم وبسبب تكرار هذه الأرقام فانه يجب تغييرها لأرقام أحرى وهذا

آوتسمي أيضا ب Busy Waiting.

| 700 ÷ 1                  |              |                    |  |  |  |
|--------------------------|--------------|--------------------|--|--|--|
| الوصف                    | رقم المقاطعة | رقم المشبك(الدبوس) |  |  |  |
| المؤقتة Timer            | 0x08         | IRQ0               |  |  |  |
| لوحة المفاتيح            | 0x09         | IRQ1               |  |  |  |
| يُربط مع متحكم PIC ثانوي | 0x0a         | IRQ2               |  |  |  |
| المنفذ التسلسلي ٢        | 0x0b         | IRQ3               |  |  |  |
| المنفذ التسلسلي ١        | 0х0с         | IRQ4               |  |  |  |
| منفذ التوازي ۲           | 0x0d         | IRQ5               |  |  |  |
| متحكم القرص المرن        | 0x0e         | IRQ6               |  |  |  |
| منفذ التوازي ١           | 0x0f         | IRQ7               |  |  |  |
| ساعة ال CMOS             | 0x70         | IRQ8/IRQ0          |  |  |  |
| CGA vertical retrace     | 0x71         | IRQ9/IRQ1          |  |  |  |
| محجوزة                   | 0x72         | IRQ10/IRQ2         |  |  |  |
| محجوزة                   | 0x73         | IRQ11/IRQ3         |  |  |  |
| محجوزة                   | 0x74         | IRQ12/IRQ4         |  |  |  |
| وحدة FPU                 | 0x75         | IRQ13/IRQ5         |  |  |  |
| متحكم القرص الصلب        | 0x76         | IRQ14/IRQ6         |  |  |  |
| محجوزة                   | 0x77         | IRQ15/IRQ7         |  |  |  |

جدول ٣.١: مقاطعات العتاد لحواسيب x86

يتم بسهولة في النمط الحقيقي وذلك باستخدام مقاطعات البايوس أما في النمط المحمى فيجب أن نقوم بالتخاطب المباشر مع المتحكم الذي لديه أرقام المقاطعات ومن ثم تغييرها . والجدول ٣.١ يوضح أرقام المقاطعات لمتحكمات الحاسب.

## ۲.۲.۱ برمجة متحكم PIC

متحكم PIC يستقبل إشارات (Signals) من متحكمات العتاد والتي تكون موصولة به ومن ثم يقوم بتحويلها الى أرقام مقاطعات لكي يقوم المعالج بنقل التنفيذ الى دالة تخديمها ، ويراعي متحكم PIC أولية متحكمات العتاد ، فمثلا لو تم إرسال إشارتين في نفس الوقت الى متحكم PIC فان المتحكم سوف يراعي الأولية ويقوم بارسال رقم مقاطعة العتاد ذو الأولية أولا وبعد أن تنتهي دالة تخديم المقاطعة يقوم المتحكم بارسال الرقم الآخر . ونظراً لتعقيدات بناء المتحكم فانه يتعامل فقط مع ٨ أجهزة مختلفة (أي ٨ مقاطعات IRQ) وهذا ما أدى مصنعي الحاسب الى توفير متحكم PIC آخر يعرفُ بالمتحكم الثانوي (Secondary/Slave PIC) . المتحكم الرئيسي (Primary PIC) يوجد داخل المعالج ويرتبط مع المتحكم الثانوي والذي يتواجد في الجسر الجنوبي (SouthBridge) .

# مشابك المتحكم PIC's Pins

تعتبر مشابك المتحكم هي طريقة ارسال البيانات من المتحكم الى المعالج (أو الى متحكم رئيسي) ، ونظراً لان كل مشبك لديه وظيفة محددة فانه يجب دراسة هذه المشابك ولكن لن نفصّل كثيراً حيث أن الموضوع متشعب ويخص دراسي المنطق الرقمي (Digital Logic). ويوضح الشكل ٢.١ هذه المشابك.



شكل ٢.١: مشابك متحكم PIC

حيث أن المشابك DO-D7 هي لإرسال البيانات الى متحكم المتحكم الما المشابك DO-D7 المتخدم للتخاطب بين متحكمات PIC الرئيسية والثانوية ، والمشبك INT يرتبط مع مشبك للمعالج وهو INTR كذلك المشبك INTA يرتبط مع مشبك المعالج INTA وهذه المشابك لها العديد من الفوائد حيث عندما يقوم المعالج بتنفيذ أي مقاطعة فانه يقوم بتعطيل قيم العلمين IF and TF وهذا ما يجعل مشبك المعالج INTR يغلق مباشرة وبالتالي لا يمكن لمتحكم PIC إرسال أي مقاطعة عبر مشبك INT حيث أن الجهة المقابلة لها تم غلقها وبالتالي لا يمكن لمشبكه INT حجرها في مسجل لمقاطعة أخرى وإنما يتم حجرها في مسجل لمقاطعة أخرى وإنما يتم حجرها في مسجل

داخل PIC الى أن ينتهي المعالج من تنفيذ المقاطعة والعودة بإشارة (تسمى إشارة نهاية المقاطعة PIC المجابعة المجابع

# مسجلات متحكم PIC

يحوي متحكم PIC على عدة مسجلات داخلية وهي:

- مسجل الأوامر (Command Reigster): ويستخدم لإرسال الأوامر الى المتحكم ، وهناك عدد من الأوامر مثل أمر القراءة من مسجل ما أو أمر ارسال اشارة EOI.
  - مسجل الحالة (Status Register): وهو مسجل للقراءة فقط حيث تظهر عليه حالة المتحكم.
- مسجل طلبات المقاطعات (Interrupt Request Register): يحفظ هذا المسجل الأجهزة التي طلبت تنفيذ مقاطعتها وهي بانتظار وصول إشعار (Acnowledges) من المعالج ، والجدول ٤.١ يوضح بتات هذا المسجل.

جدول ٤.١: مسجل IRR/ISR/IMR

| IRQ Number (Slave controller) | IRQ Number (Primary controller) | Bit Number |
|-------------------------------|---------------------------------|------------|
| IRQ8                          | IRQ0                            | 0          |
| IRQ9                          | IRQ1                            | 1          |
| IRQ10                         | IRQ2                            | 2          |
| IRQ11                         | IRQ3                            | 3          |
| IRQ12                         | IRQ4                            | 4          |
| IRQ13                         | IRQ5                            | 5          |
| IRQ14                         | IRQ6                            | 6          |
| IRQ15                         | IRQ7                            | 7          |

جدول ۱.٥: عناوين المنافذ لمتحكم PIC

| الوصف                                                           | رقم المنفذ |
|-----------------------------------------------------------------|------------|
| Primary PIC Command and Status Register                         | 0x20       |
| Primary PIC Interrupt Mask Register and Data Register           | 0x21       |
| Secondary (Slave) PIC Command and Status Register               | 0xA0       |
| Secondary (Slave) PIC Interrupt Mask Register and Data Register | 0xA1       |

وفي حالة كانت قيمة أي بت هي ١ فهذا يعني أن متحكم العتاد بانتظار الإشعار من المعالج.

- مسجل الخدمة (In Service Register (ISR)): يدل على المسجل على أن طلب المقاطعة قد نجح وأن الإشعار قد وصل لكن لم تنتهي دالة تخديم المقاطعة من عملها.
- مسجل (Interrupt Mask Register (IMR)): يحدد هذا المسجل ما هي المقاطعات التي يجب تجاهلها وعدم ارسال إشعار لها وذلك حتى يتم التركيز على المقاطعات الأهم.

والجدول ٥.١ يوضح عناوين منافذ المسجلات في حواسيب x86.

# بر مجة متحكم PIC

لبرمجة متحكم PIC وإعادة ترقيم المقاطعات فإن ذلك يتطلب إرسال بعض الأوامر الى المتحكم بحيث تأخذ هذه الأوامر نمط معين تُحَدَّد بها عمل المتحكم. وتوجد أريع أوامر تميئة يجب إرسالها لتهيئة المتحكم تعرف ب Initialization Control Words وتختصر بأوامر تميئة ICW ، وكذلك توجد ثلاث أوامر تحكم في عمل متحكم PIC تعرف ب Operation Control Words وتختصر ب PIC . وفي حالة توفر أكثر من متحكم PIC على النظام فيجب أن تُرسل أو امر التهيئة الى المتحكم الآخر كذلك. الأمر الأول ICW1 وهو أمر التهيئة

جدول ٦.١: الأمر الأول ICW1

|                               | • )    |          |
|-------------------------------|--------|----------|
| الوصف                         | القيمة | رقم البت |
| إرسال الأمر ICW4              | IC4    | 0        |
| هل يوجد متحكم PIC واحد        | SNGL   | 1        |
| تأخذ القيمة صفر في حواسيب x86 | ADI    | 2        |
| نمط عمل المقاطعة              | LTIM   | 3        |
| بت التهيئة                    | 1      | 4        |
| تأخذ القيمة صفر في حواسيب x86 | 0      | 5        |
| تأخذ القيمة صفر في حواسيب x86 | 0      | 6        |
| تأخذ القيمة صفر في حواسيب x86 | 0      | 7        |

الرئيسي والذي يجب إرساله أولا الى المتحكم الرئيسي والثانوي ويأخذ ٧ بتات ويوضح الجدول ٦.١ هذه البتات ووظيفة كل بت.

حيث أن البت الأول يحدد ما اذا كان يجب إرسال أمر التحكم ICW4 أم لا وفي حالة كان قيمة البت هي ا فإنه يجب إرسال الأمر ICW4 أما البت الثاني فغالباً يأخذ القيمة صفر دلالة على أن هناك أكثر من متحكم PIC في النظام ، والبت الثالث غير مستخدم أما الرابع فيحدد نمط عمل المقاطعة هل هي Level Triggered أم Mode أم Edge Triggered Mode ، أما البت الخامس فيحب أن يأخذ القيمة ا دلالة على أننا سنقوم بتهيئة متحكم PIC وبقية البتات غير مستخدمة في حواسيب X86. والشفرة ١٠٥ توضح إرسال الأمر الأول الى متحكم PIC الرئيسي والثانوي.

#### Example \.o: Initialization Control Words 1

```
; Setup to initialize the primary PIC. Send ICW 1
mov al, 0x11 ; 00010001
out 0x20, al
;
; Send ICW 1 to second PIC command register
out 0xA0, al
```

الأمر الثاني ICW2 يستخدم لإعادة تغيير عنواين جدول IVT الرئيسية للطلبات المقاطعات IRQ وبالتالي عن طريق هذا الأمر يمكن أن نغير أرقام المقاطعات لل IRQ الى أرقام أخرى . ويجب أن يرسل هذا الأمر مباشرة بعد الأمر الأول كذلك يجب أن يتم اختيار أرقاما غير مستخدمة من قبل المعالج حتى لا نقع في نفس المشكلة السابقة ( وهي أكثر من IRQ يستخدم نفس رقم المقاطعة وبالتالي لديهم دالة تخديم واحدة). والمثال ١٠٦ يوضح كيفية تغيير أرقام IRQ لمتحكم PIC الرئيسي والثانوي بحيث يتم استخدام أرقام المقاطعات ٣٩-٣٩ للمتحكم الأول والأرقام من ٤٠-٤٧ للمتحكم الثانوي وهي أرقاماً خالية لا يستخدمها المعالج وتقع مباشرة بعد آخر مقاطعة للمعالج الذي يستخدم ٢٣ مقاطعة بدءاً من الصفر وانتهاءاً بالمقاطعة ٣١.

| الوصف                                 | القيمة | رقم البت |
|---------------------------------------|--------|----------|
| رقم IRQ التي يتصل بها المتحكم الثانوي | S0-S7  | 0-7      |

جدول ٨.١: الأمر الثالث للمتحكم الثانوي ICW3 for Slave PIC

| 1                                        |        | -        |
|------------------------------------------|--------|----------|
| الوصف                                    | القيمة | رقم البت |
| رقم IRQ التي يتصل بها مع المتحكم الرئيسي | ID0    | 0-2      |
| محجوزة                                   | 3-7    | 3-7      |

## Example 1.7: Initialization Control Words 2

```
; send ICW 2 to primary PIC

nov al, 0x20

cout 0x21, al

; Primary PIC handled IRQ 0..7. IRQ 0 is now mapped to interrupt
    number 0x20

;

v ; send ICW 2 to secondary PIC

nov al, 0x28

out 0xA1, al

; Secondary PIC handles IRQ's 8..15. IRQ 8 is now mapped to use
    interrupt 0x28
```

الأمر الثالث ICW3 يستخدم في حالة كان هناك أكثر من متحكم PIC حيث يجب أن نحدد رقم طلب المقاطعة IRQ التي يستخدمها المتحكم الثانوي للتخاطب مع المتحكم الرئيسي. وفي حواسيب x86 غالباً ما يستخدم IRQ2 لذا يجب إرسال هذا الأمر الى المتحكم، لكن كل متحكم يتوقع الأمر بصيغة معينة يوضحها الجدولان ٧٠١ و ٨٠١.

ويجب إرسال الأمر بحسب الصيغة التي يقبلها مسجل البيانات للمتحكم ، فمتحكم PIC الرئيسي يستقبل رقم IRQ على شكل ٧ بت بحيث يتم تفعيل رقم البت المقابل لرقم IRQ وفي مثالثا يرتبط المتحكم الرئيسي مع الثانوي عبر IRQ2 لذلك يجب تفعيل قيمة البت ٢ (أي يجب إرسال القيمة 00001000 وهي تعادل 00x4) بينما المتحكم الثانوي يقبل رقم IRQ عن طريق إرسال قيمته على الشكل الثنائي وهي ٢ (وتعادل بالترميز الثنائي 000) وبقية البتات محجوزة (انظر حدول ٨٠١) ، والمثال ١٠٧ يوضح كيفية إرسال الأمر الثالث الى المتحكمين.

## Example \.Y: Initialization Control Words 3

جدول ٩.١: الأمر الرابع ICW4

| الوصف                                                       | القيمة | رقم البت |
|-------------------------------------------------------------|--------|----------|
| يجب تفعيل هذا البت في حواسيب x86                            | uPM    | 0        |
| جعل المتحكم يقوم بإرسال إشارة EOI                           | AEOI   | 1        |
| If set (1), selects buffer master. Cleared if buffer slave. | M/S    | 2        |
| If set, controller operates in buffered mode                | BUF    | 3        |
| تأخذ القيمة صفر في حواسيب x86                               | SFNM   | 4        |
| تأخذ القيمة صفر في حواسيب x86                               | 0      | 5-7      |

```
; Send ICW 3 to primary PIC

mov al, 0x4 ; 0x04 => 0100, second bit (IR line 2)

out 0x21, al ; write to data register of primary PIC

; Send ICW 3 to secondary PIC

mov al, 0x2 ; 010=> IR line 2

v out 0xA1, al ; write to data register of secondary PIC
```

الأمر الرابع ICW4 هو آخر أمر تحكم يجب إرساله الى المتحكمين ويأخذ التركيبة التي يوضحها حدول ٩.١. وفي الغالب لا يوجد حوجة لتفعيل كل هذه الخصائص ، فقط أول بت يجب تفعيله حيث يستخدم مع حواسيب x86 . والمثال ١.٨ يوضح كيفة إرسال الأمر الرابع الى المتحكم PIC الرئيسي والثانوي.

#### Example \.A: Initialization Control Words 4

```
mov al, 1  ; bit 0 enables 80x86 mode

r  ; send ICW 4 to both primary and secondary PICs
cout 0x21, al
cout 0xA1, al
```

وبعد إرسال هذه الأوامر الأربع تكتمل عملية قميئة متحكم PIC الرئيسي والثانوي ، وفي حالة حدوث أي مقاطعة من متحكم لعتاد ما ، فإن أرقام المقاطعات التي سترسل الى المعالج هي الأرقام التي قمنا بتعيينها في الأمر الثاني (وتبدأ من ٣٢ الى ٤٧) وهي تختلف بالطبع عن الأرقام التي يستخدمها المعالج. وبخصوص أوامر التحكم الثلاث OCW فلن نحتاج اليها جميعاً وسيتم الحديث عن الأمر الثاني OCW2 نظراً لأنه يجب أن يُرسل دائماً بعد أن تنتهي دالة تخديم المقاطعة من عملها وذلك حتى يتم السماح لبقية المقاطعات أن تأخذ دوراً لمعالجتها. والجدول ١٠٠١ يوضح البتات التي يجب إرسالها الى مسجل التحكم . ويهمنا البتات ٥-٧ حيث أن قيمهم تحدد بعض الخصائص التي يوضحها الجدول ١٠١١. والمثال ٩٠١ يوضح كيفية إرسال إشارة لهاية

جدول ١٠.١: أمر التحكم الثاني OCW2

| <b></b> 1                                            |          |          |
|------------------------------------------------------|----------|----------|
| الوصف                                                | القيمة   | رقم البت |
| Interrupt level upon which the controller must react | L0/L1/L2 | 0-2      |
| محجوزة                                               | 0        | 3-4      |
| End of Interrupt (EOI)                               | EOI      | 5        |
| Selection                                            | SL       | 6        |
| Rotation option                                      | R        | 7        |

## جدول ۱۱.۱: أمر OCW2

| Description                          | EOI Bit | SL Bit | R Bit |
|--------------------------------------|---------|--------|-------|
| Rotate in Automatic EOI mode (CLEAR) | 0       | 0      | 0     |
| Non specific EOI command             | 1       | 0      | 0     |
| No operation                         | 0       | 1      | 0     |
| Specific EOI command                 | 1       | 1      | 0     |
| Rotate in Automatic EOI mode (SET)   | 0       | 0      | 1     |
| Rotate on non specific EOI           | 1       | 0      | 1     |
| Set priority command                 | 0       | 1      | 1     |
| Rotate on specific EOI               | 1       | 1      | 1     |

عمل دالة تخديم المقاطعة (EOI) حيث يجب ضبط البتات لإختيار Non specific EOI command.

## Example \. 9: Send EOI

```
; send EOI to primary PIC

mov al, 0x20 ; set bit 4 of OCW 2

out 0x20, al ; write to primary PIC command register
```

## كيف تعمل مقاطعات العتاد

عندما يحتاج متحكم أي عتاد لفت انتباه المعالج الى شيء ما فأول خطوة يقوم بها هي إرسال إشارة الى متحكم PIC (وعلى سبيل المثال سنفرض أن هذا المتحكم هو متحكم المؤقتة PIT والتي ترتبط بالمشبك IRO) هذه الإشارة ترسل عبر مشبك IRO ، حينها يقوم متحكم PIC بتسجيل طلب المتحكم IRO في مسجل يسمى مسجل طلبات المقاطعات (Interrupt Request Register) ويعرف اختصاراً . بمسجل IRR . هذا المسجل بطول ۸ بت كل بت فيه يمثل رقم IRQ ويتم تفعيل أي بت عند طلب مقاطعة من المتحكم ، وفي مثالنا سيتم تفعيل البت 0 بسبب أن المؤقتة ترتبط مع IRO. بعد ذلك يقوم متحكم PIC بفحص مسجل Interrupt

Mask Register ليتأكد من أنه لا توجد هناك مقاطعة ذات أولية أعلى حيث في هذه الحالة على المقاطعة الجديدة أن تننظر حتى يتم تخديم كل المقاطعات ذات الأولوية. وبعد ذلك يُرسل PIC إشارة الى المعالج من المعالم مشبك INTA لأحبار المعالج بأن هناك مقاطعة يجب تنفيذها. وهنا يأتي دور المعالج حيث يقوم بالإنتهاء من تنفيذ الأمر الحالي الذي يعمل عليه ومن ثم يقوم بفحص قيمة العلم العيث في حالة كانت غير مفعلة فان المعالج سوف يتجاهل طلب تنفيذ المقاطعة، أما إذا وجد المعالج قيمة العلم مفعلة فانه يقوم بارسال إشعار (Acnowledges) عبر مشبك INTA الى متحكم PIC الذي بدوره يستقبلها من مشبك INS ويضع رقم المقاطعة ورقم IRQ في المشابك In Service Register الذي بدوره يستقبلها من مشبك In دلالة على أن مقاطعة المؤقتة حاري تنفيذها. وعندما يحصل المعالج على رقم المقاطعة فانه يقوم بوقف العملية التي يعمل عليها ويحفظ قيم مسجل الأعلام ومسجل CS and EIP وإذا كان المعالج يعمل في النمط الحقيقي فإنه يأخذ رقم المقاطعة ويذهب بما كدليل الى حدول المقطاعات IVT حيث يجد عنوان دالة تخديم المقاطعة ومن ثم واصفات المقاطعات حيث يجد دالة تخديم المقاطعة. وعندما تنتهي دالة تخديم المقاطعة من عملها فالها يجب أن ترسل إشارة EOI حتى يتم تفعيل المقاطعات بحدداً.

# ۳. ۱ المؤقتة Programmable Interval Timer

المؤقتة هي شريحة (Counters or Channels) تعمل Dual Inline Package (DIP) تعمل كمؤقتات لإدارة ثلاث أشياء (انظر الشكل ٣٠٠). العداد الأول ويُعرف بمؤقت النظام (System Timer) وظيفته ارسال طلب مقاطعة (IRQ0) الى متحكم PIC وذلك لتنفيذ مقاطعة ما كل فترة محددة ، هذه الفترة يتم تحديدها عند برمجة هذه المؤقتة ويُستفاد من هذه المؤقتة في عملية تزامن العمليات وتوفير بنية تحتية لمفهوم تعدد العمليات والمسالك (Multitask and Multithread) حيث أن الفترة التي تقوم بما مؤقتة النظام لاصدار طلب المقاطعة سيكون هو الوقت المحدد لأي عملية (Process) موجودة في طابور العمليات (Queue والعمليات ألم تنتهي من عملها بعد ويبدأ المعالج في تنفيذ العملية التالية تحت نفس الفترة المحددة. أما العداد الثاني فيُستخدم في عملية تنعيش الذاكرة الرئيسية (RAM refreshing) وأصبحت هذه المؤقتة لا تستخدم في العادة. أما العداد الأحير فيستخدم في عملية إرسال الصوت الى سماعات الحاسب ويجدر بنا ذكر أن هذه المهمة العداد الأحير فيستخدم في عملية إرسال الصوت الى سماعات الحاسب (PC Speaker).

لا يقصد هذه كرت الصوت وإنما يوجد في كل حاسب سماعات داخلية تستخدم في إصدار الصوت والنغمات وأحد استخداماتها
 لإصدار رسائل الخطأ بعد عملية فحص الحاسب (POST) في مرحلة الإقلاع.





## 1.٣.١ برمجة المؤقتة PIT

مؤخراتم نقل المؤقتة من اللوحة الأم (MotherBoard) كشريحة DIP مستلقة الى الجسر الشمالي (SouthBridge). وسوف نركز على برمجة العداد الأول وهو مؤقت النظام حيث أنه يوفر الدعم العتادي اللازم للنظام حتى يدعم تعدد العمليات والمسالك.

## مشابك المؤقتة PIT's Pins

8253 18 CLK 2 17 DOUT 2 16 GATE 2 15 CLK 1 14 GATE 1 13 OUT 1 CLK 0 0 9 OUT 0 0 10 GATE 0 □ 11 GND □ 12

شكل ٤.١: مشابك المؤقتة PIT

تُرسَل الأوامر والبيانات الى المؤقتة وذلك عبر مسار البيانات (Data Bus) حيث يرتبط هذا المسار مع مشابك البيانات في المؤقتة وهي ٨ مشابك DO...D7 وتمثل ٨ بتات. وعند إرسال بيانات الى المؤقتة (عملية كتابة) فان مشبك الكتابة WR يأخذ قيمة منخفضة دلالة على أن هناك عملية إرسال بيانات الى المؤقتة وكذلك في حالة قراءة بيانات من المؤقتة فإن مشبك القراءة RD يأحذ قيمة منخفضة دلالة على أن هناك عملية قراءة من المؤقتة. ويتحكم في مشبك القراءة والكتابة مشبك CS حيث تحدد قيمته

تعطيل أو تَفعيلَ عمل الشبكين السابقين ، ويرتبط مشبك CS مع مسار العناوين (Address Bus) بينما يرتبط مشبك القراءة والكتابة مع مسار التحكم (Control Bus). وتُحدد قيمة المشبكين AO,A1 -واللذان يرتبطان مع مسار العنواين- المسجلات المطلوب الوصول اليها داخل المؤقتة. أما المشابك (CLK, OUT, and GATE) فهِّي لكل عداد بداخل المؤقتة أي بمعنى أنه توجد ثلاث مشابك من كل واحدة منهم ، ويعتبر المشبكين (CLK (Clock Input) and GATE) مشابك إدخال للعداد بينما المشبك (OUT) مشبك إخراج حيث يستخدم لربط العداد مع العتاد فمثلا مشبك الإحراج في العداد الأول (مؤقتة النظام) يرتبط مع متحكم PIC حيث من خلاله تستطيع مؤقتة النظام إرسال طلب المقاطعة (IRQO) الى متحكم PIC والذي يقوم بتحويل الطلب الى المعالج لكي ينفذ دالة التخديم.

## مسجلات المؤقتة PIT

توجد ٤ مسجلات بداخل المؤقتة PIT ، ثلاث منها تستخدم للعدادات (الأول والثاني والثالث) حيث من خلالها يمكن قراءة قيمة العداد أو الكتابة فيه ، وطول مسجل العداد هو ١٦ بت . وبسبب أن مشابك البيانات التي تربط المؤقتة ومسار البيانات هي من الطول ٨ بت فانه لن نتمكن من إرسال البيانات بهذ الشكل . لذلك يجب إستخدام مسجل اخر وهو مسجل التحكم (Control Word) بحيث قبل إرسال بيانات أو قراءة بيانات من أي عداد فانه يجب إرسال الأمر المطلوب الى مسجل التحكم وبعد ذلك يتم إرسال البيانات أو قرائتها. والجدول ١٢.١ يوضح هذا المسجلات وعنوان منافذ الإدخال والإخراج المستخدمة للتعامل معها ، ويجب ملاحظة قيم خط القراءة والكتابة وخط العنوان (AO,A1) حيث تؤثر قيمهم في تحديد نوع العملية المطلوبة (قراءة أم كتابة ورقم العداد). وتوضح التركيبة التالية ماهية البتات المستخدمة في مسجل التحكم

| محدول ۱۱۱. مسجارت الموقفة 6253 |       |       |       |       |            |              |
|--------------------------------|-------|-------|-------|-------|------------|--------------|
| الوظيفة                        | خط A1 | خط A0 | خط WR | خط RD | رقم المنفذ | اسم المسجل   |
| كتابة الى المسجل 0             | 0     | 0     | 0     | 1     | 0x40       | Counter 0    |
| قراءة المسجل 0                 | 0     | 0     | 1     | 0     |            |              |
| كتابة الى المسجل 1             | 1     | 0     | 0     | 1     | 0x41       | Counter 1    |
| قراءة المسجل 1                 | 1     | 0     | 1     | 0     |            |              |
| كتابة الى المسجل 2             | 0     | 1     | 0     | 1     | 0x42       | Counter 2    |
| قراءة المسجل 2                 | 0     | 1     | 1     | 0     |            |              |
| کتابة Control Word             | 1     | 1     | 0     | 1     | 0x43       | Control Word |
| لا توجد عملية                  | 1     | 1     | 1     | 0     |            |              |

جدول ٢.١: مسجلات المؤقتة 8253 PIT

(وهو مسجل بطول ٨ بت) حيث يجب إرسال قيم معينة حتى نتمكن من القراءة أو الكتابة في عداد ما.

• Bit 0: (BCP) Binary Counter

- 0: Binary

1: Binary Coded Decimal (BCD)

• Bit 1-3: (M0, M1, M2) Operating Mode. See above sections for a description of each.

000: Mode 0: Interrupt or Terminal Count

- 001: Mode 1: Programmable one-shot

- 010: Mode 2: Rate Generator

011: Mode 3: Square Wave Generator

100: Mode 4: Software Triggered Strobe

101: Mode 5: Hardware Triggered Strobe

- 110: Undefined; Don't use

- 111: Undefined; Don't use
- Bits 4-5: (RL0, RL1) Read/Load Mode. We are going to read or send data to a counter register
  - 00: Counter value is latched into an internal control register at the time of the I/O write operation.
  - 01: Read or Load Least Significant Byte (LSB) only
  - 10: Read or Load Most Significant Byte (MSB) only
  - 11: Read or Load LSB first then MSB
- Bits 6-7: (SC0-SC1) Select Counter. See above sections for a description of each.
  - 00: Counter 0
  - 01: Counter 1
  - 10: Counter 2
  - 11: Illegal value

والمثال ١٠١٠ يوضح كيفية برمجة عداد مؤقت النظام لإرسال طلب مقاطعة كل 100Hz (كل ١٠ مؤقت النظام لإرسال طلب مقاطعة كل 100Hz) ، وهذا يتم عن طريق إرسال أمر التحكم أولاً ومن ثم إرسال الوقت المطلوب الى العداد المطلوب.

## Example \.\ ·: PIT programming

```
; COUNT = input hz / frequency
   mov dx, 1193180 / 100 ; 100hz, or 10 milliseconds
    ; FIRST send the command word to the PIT. Sets binary counting,
    ; Mode 3, Read or Load LSB first then MSB, Channel 0
   mov al, 110110b
   out 0x43, al
    ; Now we can write to channel O. Because we set the "Load LSB first
١١
        then MSB" bit, that is
    ; the way we send it
۱۲
۱٤
   mov ax, dx
    out 0x40, al
                  ;LSB
   xchg ah, al
١٦
    out 0x40, al ;MSB
```

# 1.3 توسعة طبقة HAL

طبقة HAL تبعد نواة النظام من التعامل المباشر مع العتاد وتعمل كواجهة أو طبقة ما بين النواة والعتاد ، وفيها نجد تعريفات العتاد. وسيتم إضافة أوامر برمجة متحكم PIC التي تقوم بإعادة تعيين أرقام المقاطعات بداخل هذه الطبقة وكذلك سيتم إضافة شفرة برمجة المؤقتة و التي تحدد الوقت اللازم للمؤقتة لكي تقوم بارسال طلب المقاطعة (IRQ0) .

## ۱.٤.۱ دعم PIC

في القسم ٢.٢.١ تم عرض متحكم PIC وكيفية برمجته بالتفصيل ، وفي هذا القسم سيتم تطبيق ما تم عرضه على نواة نظام إقرأ. ويوجد ملفين لمتحكم PIC الأول هو ملف الرأس (hal/pic.h) الذي يحوي الإعلان عن الدوال وكذلك الثوابت والثاني هو ملف التطبيق (hal/pic.cpp) الذي يحوي على تعريف تلك الدوال. والمثال ١٠١١ يعرض ملف الرأس الذي يغلف العديد من الأرقام والعناوين في صورة ثوابت (باستخدام الماكرو) بحيث تزيد من مقروئية ووضوح الشفرة^.

## Example \.\\: hal/pic.h: PIC Interface

```
\ // PIC 1 Devices IRO
r #define I386_PIC_IRQ_TIMER
r #define I386_PIC_IRQ_KEYBOARD
#define I386_PIC_IRQ_SERIAL2
• #define I386_PIC_IRQ_SERIAL1
r #define I386_PIC_IRQ_PARALLEL2
                                     5
v #define I386_PIC_IRQ_DESKETTE
A #define I386_PIC_IRQ_PARALLEL1
· // PIC 2 Devices IRQ
v #define I386_PIC_IRQ_CMOSTIMER
                                     0
ny #define I386_PIC_IRQ_CGARETRACE
\r #define I386_PIC_IRQ_AUXILIRY
15 #define I386_PIC_IRQ_FPU
vo #define I386_PIC_IRQ_HDC
v // Operation Command Word 2 (OCW2)
vx #define I386_PIC_OCW2_MASK_L1
19 #define I386_PIC_OCW2_MASK_L2
r. #define I386_PIC_OCW2_MASK_L3
```

^إرجع الى القسم ٢.٢.١ لمعرفة وظيفة هذه القيم الثابتة.

```
rv #define I386_PIC_OCW2_MASK_EOI
                                      0x20
tr #define I386_PIC_OCW2_MASK_SL
rr #define I386_PIC_OCW2_MASK_ROTATE 0x80
v1 // Operation Command Word 3 (OCW3)
rv #define I386_PIC_OCW3_MASK_RIS
TA #define I386_PIC_OCW3_MASK_RIR
ra #define I386_PIC_OCW3_MASK_MODE
r. #define I386_PIC_OCW3_MASK_SMM
                                     0x20
r\ #define I386_PIC_OCW3_MASK_ESMM
                                     0x40
ry #define I386_PIC_OCW3_MASK_D7
                                   0x80
v∘ // PIC 1 port address
rt #define I386_PIC1_COMMAND_REG
                                   0x20
rv #define I386_PIC1_STATUS_REG
                                   0x20
TA #define I386_PIC1_IMR_REG
                                 0x21
rq #define I386_PIC1_DATA_REG
                                   0x21
ex // PIC 2 port address
#define I386_PIC2_COMMAND_REG
                                   0xa0
#define I386_PIC2_STATUS_REG
                                   0xa0
#define I386_PIC2_IMR_REG
                                 0xa1
#define I386_PIC2_DATA_REG
                                   0xa1
1/2 Initializing Command Word 1 (ICW1) Mask
#define I386_PIC_ICW1_MASK_IC4
• #define I386_PIC_ICW1_MASK_SNGL
                                      0x2
ov #define I386_PIC_ICW1_MASK_ADI
                                     0x4
or #define I386_PIC_ICW1_MASK_LTIM
                                      0x8
or #define I386_PIC_ICW1_MASK_INIT
                                     0x10
0 5
on // Initializing Command Word 4 (ICW4) Mask
ov #define I386_PIC_ICW4_MASK_UPM
                                      0x1
•A #define I386_PIC_ICW4_MASK_AEOI
                                      0x2
•9 #define I386_PIC_ICW4_MASK_MS
                                   0 \times 4
T. #define I386_PIC_ICW4_MASK_BUF
                                     0x8
n #define I386_PIC_ICW4_MASK_SFNM
                                     0x10
```

```
٦٢
18 // Initializing command 1 control bits
no #define I386_PIC_ICW1_IC4_EXPECT
n #define I386_PIC_ICW1_IC4_NO
ny #define I386_PIC_ICW1_SNGL_YES
TA #define I386_PIC_ICW1_SNGL_NO
19 #define I386_PIC_ICW1_ADI_CALLINTERVAL4
v. #define I386_PIC_ICW1_ADI_CALLINTERVAL8
vv #define I386_PIC_ICW1_LTIM_LEVELTRIGGERED 8
vv #define I386_PIC_ICW1_LTIM_EDGETRIGGERED
vr #define I386_PIC_ICW1_INIT_YES
                                         0 \times 10
vs #define I386_PIC_ICW1_INIT_NO
vy // Initializing command 4 control bits
vv #define I386_PIC_ICW4_UPM_86MODE
va #define I386_PIC_ICW4_UPM_MCSMODE
va #define I386_PIC_ICW4_AEOI_AUTOEOI
A. #define I386_PIC_ICW4_AEOI_NOAUTOEOI
#define I386_PIC_ICW4_MS_BUFFERMASTER
AT #define I386_PIC_ICW4_MS_BUFFERSLAVE
Ar #define I386_PIC_ICW4_BUF_MODEYES
At #define I386_PIC_ICW4_BUF_MODENO
ho #define I386_PIC_ICW4_SFNM_NESTEDMODE
                                            0 \times 10
AT #define I386_PIC_ICW4_SFNM_NOTNESTED
A9 extern uint8_t i386_pic_read_data(uint8_t pic_num);
extern void i386_pic_send_data(uint8_t data,uint8_t pic_num);
av extern void i386_pic_send_command(uint8_t cmd,uint8_t pic_num);
extern void i386_pic_init(uint8_t base0, uint8_t base1);
```

وتحوي الواجهة ٤ دوال منها دالتان للقراءة والكتابة من مسجل البيانات ودالة لإرسال الأوامر الى مسجل التحكم والدالة الأخيرة هي لتهئية المتحكم وهي الدالة التي يجب استدعائها. والمثال ١٠١٢ يوضح تعريف هذه الدوال.

#### Example \.\ \: hal/pic.cpp: PIC Implementation

```
v uint8_t i386_pic_read_data(uint8_t pic_num) {
v    if (pic_num > 1)
v    return 0;
```

```
uint8_t req = (pic_num == 1)?I386_PIC2_DATA_REG:I386_PIC1_DATA_REG;
    return inportb(reg);
v }
void i386_pic_send_data(uint8_t data,uint8_t pic_num) {
    if (pic_num > 1)
      return;
۱۱
    uint8_t reg = (pic_num == 1)?I386_PIC2_DATA_REG:I386_PIC1_DATA_REG;
    outportb(reg,data);
١٤
10 }
١٦
vv void i386_pic_send_command(uint8_t cmd, uint8_t pic_num) {
    if (pic_num > 1)
۱۹
      return;
۲.
۲١
    uint8_t reg = (pic_num == 1)?I386_PIC2_COMMAND_REG:
        1386_PIC1_COMMAND_REG;
    outportb(reg,cmd);
11 }
70
۲٦
void i386_pic_init(uint8_t base0, uint8_t base1) {
    uint8_t icw = 0;
۳.
    disable_irq(); /* disable hardware interrupt (cli) */
٣٢
    /* init PIC, send ICW1 */
٣٤
    icw = (icw & ~1386_PIC_ICW1_MASK_INIT) | 1386_PIC_ICW1_INIT_YES;
    icw = (icw & ~I386_PIC_ICW1_MASK_IC4) | I386_PIC_ICW1_IC4_EXPECT;
    /* icw = 0x11 */
٣٨
    i386_pic_send_command(icw,0);
    i386_pic_send_command(icw,1);
٤.
    /* ICW2 : remapping irq */
٤٢
    i386_pic_send_data(base0,0);
```

```
i386_pic_send_data(base1,1);
٤٤
٤٥
٤٦
    /* ICW3 : irq for master/slave pic*/
    i386_pic_send_data(0x4,0);
    i386_pic_send_data(0x2,1);
٤٨
٤٩
    /* ICW4: enable i386 mode. */
٥,
    icw = (icw & ~I386_PIC_ICW4_MASK_UPM) | I386_PIC_ICW4_UPM_86MODE ;
       /* icw = 1 */
    i386_pic_send_data(icw,0);
    i386_pic_send_data(icw,1);
٥٣
٥٤
.. }
```

## ۲.٤.۱ دعم PIT

#### Example \.\\\: hal/pit.h: PIt Interface

```
v #define I386_PIT_COUNTER0_REG
                                                                                                                                                                   0 \times 40
  r #define I386_PIT_COUNTER1_REG
                                                                                                                                                                  0x41
  r #define I386_PIT_COUNTER2_REG
                                                                                                                                                                  0x42
  #define I386_PIT_COMMAND_REG
                                                                                                                                                                  0x43
  r #define I386_PIT_OCW_MASK_BINCOUNT
                                                                                                                                                                                   0x1
  v #define I386_PIT_OCW_MASK_MODE
                                                                                                                                                                            0xe
  A #define I386_PIT_OCW_MASK_RL
                                                                                                                                                                  0x30
  4 #define I386_PIT_OCW_MASK_COUNTER
                                                                                                                                                                          0xc0
\r #define I386_PIT_OCW_BINCOUNT_BINARY
                                                                                                                                                                                   0 \times 0
\r #define I386_PIT_OCW_BINCOUNT_BCD
                                                                                                                                                                           0x1
vo #define I386_PIT_OCW_MODE_TERMINALCOUNT
n #define I386_PIT_OCW_MODE_ONESHOT
vv #define I386_PIT_OCW_MODE_RATEGEN
n/A #define I386_PIT_OCW_MODE_SQUAREWAVEGEN
                                                                                                                                                                                            0x6
19 #define I386_PIT_OCW_MODE_SOFTWARETRIG
                                                                                                                                                                                             0x8

f #define I386_PIT_OCW_MODE_HARDWARETRIG

f #define I386_PIT_OCW_M
                                                                                                                                                                                            0xa
```

```
rr #define I386_PIT_OCW_RL_LATCH
rr #define I386_PIT_OCW_RL_LSBONLY
                                        0x10

** #define I386_PIT_OCW_RL_MSBONLY

                                         0x20
ro #define I386_PIT_OCW_RL_DATA
                                       0x30
ty #define I386_PIT_OCW_COUNTER_0
                                         0x0
TA #define I386_PIT_OCW_COUNTER_1
                                         0x40
rq #define I386_PIT_OCW_COUNTER_2
                                         0x80
ry extern void i386_pit_send_command(uint8_t cmd);
rr extern void i386_pit_send_data(uint16_t data,uint8_t counter);
rr extern uint8_t i386_pit_read_data(uint16_t counter);
r: extern uint32_t i386_pit_set_tick_count(uint32_t i);
ro extern uint32_t i386_pit_get_tick_count();
ra extern void i386_pit_start_counter(uint32_t freq,uint8_t counter,
     uint8_t mode);
rv extern void _cdecl i386_pit_init();
TA extern bool _cdecl i386_pit_is_initialized();
```

#### Example \.\\\ \tau: hal/pit.cpp: PIT Implementation

```
v static volatile uint32_t _pit_ticks = 0;
r static bool _pit_is_init = false;
void _cdecl i386_pit_irg();
void i386_pit_send_command(uint8_t cmd) {
   outportb(I386_PIT_COMMAND_REG,cmd);
٨ }
void i386_pit_send_data(uint16_t data,uint8_t counter) {
    uint8_t port;
    if (counter == I386_PIT_OCW_COUNTER_0)
    port = I386_PIT_COUNTER0_REG;
١٤
    else if ( counter == I386_PIT_OCW_COUNTER_1)
١٥
     port = I386_PIT_COUNTER1_REG;
١٦
١٨
     port = I386_PIT_COUNTER2_REG;
```

```
outportb(port, uint8_t (data));
11 }
rr uint8_t i386_pit_read_data(uint16_t counter) {
    uint8_t port;
۲ ٤
۲0
   if (counter == I386_PIT_OCW_COUNTER_0)
77
     port = I386_PIT_COUNTER0_REG;
    else if ( counter == I386_PIT_OCW_COUNTER_1)
۲۸
      port = I386_PIT_COUNTER1_REG;
    else
۳.
     port = I386_PIT_COUNTER2_REG;
٣٢
  return inportb(port);
٣٣
r: }
30
ru uint32_t i386_pit_set_tick_count(uint32_t i) {
rv uint32_t prev = _pit_ticks;
rA _pit_ticks = i;
   return prev;
٤٠ }
iv uint32_t i386_pit_get_tick_count() {
fr return _pit_ticks;
٤٤ }
void i386_pit_start_counter(uint32_t freq,uint8_t counter,uint8_t
     mode) {
   if (freq == 0)
٤٧
     return;
٤٨
٤٩
    uint16_t divisor = uint16_t(1193181/uint16_t(freq));
٥١
    /* send operation command */
٥٢
   uint8_t ocw = 0;
٥٣
٥٤
   ocw = (ocw & ~I386_PIT_OCW_MASK_MODE) | mode;
    ocw = (ocw & ~1386_PIT_OCW_MASK_RL) | I386_PIT_OCW_RL_DATA;
    ocw = (ocw & ~I386_PIT_OCW_MASK_COUNTER) | counter;
٥٨
i386_pit_send_command(ocw);
```

```
/* set frequency rate */
    i386_pit_send_data(divisor & 0xff,0);
    i386_pit_send_data((divisor >> 8) & 0xff,0);
    /* reset ticks count */
    _pit_ticks = 0;
٦٦
٦٧ }
ra void _cdecl i386_pit_init() {
    set_vector(32,i386_pit_irq);
    _pit_is_init = true;
٧٢ }
vs bool _cdecl i386_pit_is_initialized() {
    return _pit_is_init;
٧٦ }
vn void _cdecl i386_pit_irq() {
    _asm {
      add esp,12
٨١
      pushad
٨٢
٨٣
٨٤
    _pit_ticks++;
    int_done(0);
۸٧
\Lambda\Lambda
    _asm {
٨٩
      popad
٩.
      iretd
98 }
```

# ٣.٤.١ واجهة HAL الجديدة

المثال ١.١٥ يوضح الواجهة العامة لطبقة HAL

Example \.\o: New HAL Interface

```
extern int _cdecl hal_init();
r extern int _cdecl hal_close();
r extern void _cdecl gen_interrupt(int);
extern void _cdecl int_done(unsigned int n);
extern void _cdecl sound(unsigned int f);
r extern unsigned char _cdecl inportb(unsigned short port_num);
v extern void _cdecl outportb(unsigned short port_num,unsigned char value);
A extern void _cdecl enable_irq();
extern void _cdecl disable_irq();
extern void _cdecl set_vector(unsigned int int_num,void (_cdecl far & vect)());
extern void (_cdecl far * _cdecl get_vector(unsigned int int_num))();
extern const char* _cdecl get_cpu_vendor();
r extern int _cdecl get_tick_count();
```

#### Example \.\\:\: New HAL Impelmentation

```
 int _cdecl hal_init() {
i386_cpu_init();
  i386_pic_init(0x20,0x28);
i386_pit_init();
   i386_pit_start_counter(100, I386_PIT_OCW_COUNTER_0,
       1386_PIT_OCW_MODE_SQUAREWAVEGEN);
   /* enable irq */
   enable_irq();
   return 0;
11 }
int _cdecl hal_close() {
  i386_cpu_close();
   return 0;
10
17 }
void _cdecl gen_interrupt(int n) {
19 #ifdef _MSC_VER
r. _asm {
    mov al, byte ptr [n]
```

```
mov byte ptr [address+1], al
      jmp address
۲٤
      address:
        int 0 // will execute int n.
TA #endif
79 }
rr void _cdecl int_done(unsigned int n) {
    if (n > 16)
      return;
٤٣
    if (n > 7)
      /* send EOI to pic2 */
      i386_pic_send_command(I386_PIC_OCW2_MASK_EOI,1);
    /* also send toi the primary pic */
٤.
    i386_pic_send_command(I386_PIC_OCW2_MASK_EOI,0);
٤١
٤٢ }
٤٣
set void _cdecl sound(unsigned int f) {
    outportb(0x61,3 | unsigned char(f << 2));
٤٦ }
sh unsigned char _cdecl inportb(unsigned short port_num) {
!9 #ifdef _MSC_VER
    _asm {
      mov dx,word ptr [port_num]
٥١
      in al,dx
      mov byte ptr [port_num],al
•• #endif
    return unsigned char(port_num);
٠, }
٥٩
void _cdecl outportb(unsigned short port_num,unsigned char value) {
nr #ifdef _MSC_VER
```

```
ιτ _asm {
     mov al,byte ptr[value]
٦٤
     mov dx,word ptr[port_num]
     out dx,al
٦٧ }
TA #endif
٦٩ }
vv void _cdecl enable_irq() {
vy #ifdef _MSC_VER
vr _asm sti
v: #endif
٧٥ }
vv void _cdecl disable_irq() {
vA #ifdef _MSC_VER
va _asm cli
A. #endif
٨١ }
^r void _cdecl set_vector(unsigned int int_num, void (_cdecl far & vect)
     ()) {
   i386_idt_install_ir(int_num, I386_IDT_32BIT | I386_IDT_PRESENT /*
        10001110*/,0x8 /*code desc*/,vect);
٨٠ }
^^v void (_cdecl far * _cdecl get_vector(unsigned int int_num))() {
    idt_desc* desc = i386_get_idt_ir(int_num);
٨٩
    if (desc == 0)
۹.
     return 0;
91
9 ٢
uint32_t address = desc->base_low | (desc->base_high << 16);</pre>
٩ ٤
   I386_IRQ_HANDLER irq = (I386_IRQ_HANDLER) address;
90
    return irq;
97
97 }
n const char* _cdecl get_cpu_vendor() {
return i386_cpu_vendor();
· · · }
```

```
'''
'''
'''

'''

return i386_pit_get_tick_count();
'''

}
```

## Example \.\Y: kernel/main.cpp

```
v int _cdecl main()
۲ {
   hal_init();
    enable_irq();
    set_vector(0, (void (_cdecl &) (void))divide_by_zero_fault);
    set_vector(1, (void (_cdecl &) (void)) single_step_trap);
    set_vector(2, (void (_cdecl &) (void))nmi_trap);
    set_vector(3, (void (_cdecl &) (void))breakpoint_trap);
    set_vector(4, (void (_cdecl &) (void)) overflow_trap);
    set_vector(5, (void (_cdecl &) (void))bounds_check_fault);
    set_vector(6, (void (_cdecl &) (void))invalid_opcode_fault);
    set_vector(7, (void (_cdecl &) (void)) no_device_fault);
۱۳
    set_vector(8, (void (_cdecl &) (void)) double_fault_abort);
١٤
    set_vector(10, (void (_cdecl &) (void))invalid_tss_fault);
    set_vector(11, (void (_cdecl &) (void)) no_segment_fault);
    set_vector(12, (void (_cdecl &) (void)) stack_fault);
    set_vector(13, (void (_cdecl &) (void))general_protection_fault);
۱۸
    set_vector(14, (void (_cdecl &) (void))page_fault);
    set_vector(16, (void (_cdecl &) (void))fpu_fault);
    set_vector(17, (void (_cdecl &) (void))alignment_check_fault);
    set_vector(18, (void (_cdecl &) (void)) machine_check_abort);
    set_vector(19, (void (_cdecl &) (void))simd_fpu_fault);
۲ ٤
۲0
```

## Example \.\A: kernel/exception.h

```
7 /* Single step */
v extern void _cdecl single_step_trap(uint32_t cs,uint32_t eip,uint32_t
      eflags);
4 /* No Maskable interrupt trap */
v· extern void _cdecl nmi_trap(uint32_t cs,uint32_t eip,uint32_t eflags)
\r /* Breakpoint hit */
vr extern void _cdecl breakpoint_trap(uint32_t cs,uint32_t eip,uint32_t
     eflags);

\* Overflow trap */
n extern void _cdecl overflow_trap(uint32_t cs,uint32_t eip,uint32_t
     eflags);
\\ /* Bounds check */
va extern void _cdecl bounds_check_fault(uint32_t cs,uint32_t eip,
     uint32_t eflags);
*/ /* invalid opcode instruction */
rr extern void _cdecl invalid_opcode_fault(uint32_t cs,uint32_t eip,
     uint32_t eflags);
Y1 /* Device not available */
ro extern void _cdecl no_device_fault(uint32_t cs,uint32_t eip,uint32_t
     eflags);
rv /* Double Fault */
rx extern void _cdecl double_fault_abort(uint32_t cs,uint32_t err,
     uint32_t eip,uint32_t eflags);
r. /* Invalid TSS */
rv extern void _cdecl invalid_tss_fault(uint32_t cs,uint32_t err,
     uint32_t eip,uint32_t eflags);
"" /* Segment not present */
r: extern void _cdecl no_segment_fault(uint32_t cs,uint32_t err,uint32_t
      eip,uint32_t eflags);
```

```
ri /* Stack fault */
rv extern void _cdecl stack_fault(uint32_t cs,uint32_t err,uint32_t eip,
     uint32_t eflags);
rq /* General Protection Fault */
€ extern void _cdecl general_protection_fault(uint32_t cs,uint32_t err,
     uint32_t eip,uint32_t eflags);
17 /* Page Fault */
er extern void _cdecl page_fault(uint32_t cs,uint32_t err,uint32_t eip,
     uint32_t eflags);
٤٤
% /* FPU error */
en extern void _cdecl fpu_fault(uint32_t cs,uint32_t eip,uint32_t eflags
     );
* /* Alignment Check */
extern void _cdecl alignment_check_fault(uint32_t cs,uint32_t err,
     uint32_t eip,uint32_t eflags);

   /* Machine Check */
or extern void _cdecl machine_check_abort(uint32_t cs,uint32_t eip,
     uint32_t eflags);
ده /* FPU Single Instruction Multiple Data (SIMD) error */
•• extern void _cdecl simd_fpu_fault(uint32_t cs,uint32_t eip,uint32_t
     eflags);
```

## Example \.\ \: kernel/exception.cpp

```
v /* Divide by zero */
void _cdecl divide_by_zero_fault(uint32_t cs,uint32_t eip,uint32_t
        eflags) {
v kernel_panic("Divide by 0");
for (;;);
}
```

## Example \. \ \ \ : kernel/panic.cpp

```
void _cdecl kernel_panic(const char* msg,...) {
```

```
disable_irq();
   va_list args;
   va_start(args,msg);
    /* missing */
   va_end(args);
    char* panic = "\nSorry, eqraOS has encountered a problem and has
       been shutdown.\n\n";
١١
   kclear(0x1f);
    kgoto_xy(0,0);
    kset_color(0x1f);
١٤
    kputs(panic);
    kprintf("*** STOP: %s",msg);
١٦
    /* hang */
١٨
    for (;;) ;
۱۹
7. }
```

# شكل ٥.١: واجهة النظام بعد توسعة طبقة HAL

# شكل 7.۱: دالة تخديم المقاطعات الإفتراضية المعادد الله عليه المعادد الله عدد الله المعادد المعادد الله المعادد الله المعادد المعادد المعادد المعادد المعادد المعادد المعادد المعادد المعادد