selectors, descriptors, more interrupt init
[3sys] / sys1 / kern / src / desc.c
1 #include "desc.h"
2
3 #define SEG_TYPE_SYSTEM 0
4 #define SEG_TYPE_CODE   0x00001800
5 #define SEG_TYPE_DATA   0x00001000
6 #define SEG_TYPE_TSS    0x00000100
7 #define SEG_TYPE_INTR   0x00000e00
8 #define SEG_TYPE_TRAP   0x00000f00
9
10 #define SEG_ACCESSED    0x00000100
11 #define SEG_PRESENT             0x00008000
12 #define SEG_AVL                 0x00100000
13 #define SEG_GRAN                0x00800000
14
15 /* data segment type flags */
16 #define SEG_WR                  0x00000200
17 #define SEG_EXPDN               0x00000400
18 #define SEG_BIG                 0x00400000
19 /* code segment type flags */
20 #define SEG_RD                  0x00000200
21 #define SEG_CONFORM             0x00000400
22 #define SEG_DEF                 0x00400000
23
24
25 #define GATE_TYPE_INTR  0x0e00
26 #define GATE_TYPE_TRAP  0x0f00
27 #define GATE_PRESENT    0x8000
28
29
30 void desc_seg(struct desc *desc, int type, uint32_t base, uint32_t limit, int dpl)
31 {
32         if(type == SEG_NULL) {
33                 desc->d[0] = desc->d[1] = 0;
34                 return;
35         }
36
37         desc->d[0] = (limit & 0xffff) | (base << 16);
38         desc->d[1] = ((base >> 16) & 0xff) | (type << 8) | SEG_PRESENT | (dpl << 13);
39         desc->d[1] |= (base & 0xff000000) | (limit & 0xf0000) | SEG_AVL | SEG_GRAN;
40
41         switch(type) {
42         case SEG_CODE:
43                 desc->d[1] |= (3 << 11) | SEG_DEF | SEG_RD;
44                 break;
45
46         case SEG_DATA:
47                 desc->d[1] |= (2 << 11) | SEG_WR;
48                 break;
49         }
50 }
51
52
53
54 void desc_intr(struct desc *desc, int type, uint16_t sel, uint32_t offs, int dpl)
55 {
56         uint32_t gate_type[] = { GATE_TYPE_INTR, GATE_TYPE_TRAP };
57
58         desc->d[0] = (offs & 0xffff) | ((uint32_t)sel << 16);
59         desc->d[1] = (offs & 0xffff0000) | GATE_PRESENT | (dpl << 13) | gate_type[type];
60 }