-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathgdt.cpp
122 lines (92 loc) · 3.71 KB
/
gdt.cpp
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
/***************************************************************************
* gdt.cpp
*
* Copyright 2008 Matthias v.d. Vlies
* matthias@mserver.nl
****************************************************************************/
/*
* This program is free software; you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation; either version 2 of the License, or
* (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program; if not, write to the Free Software
* Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
*/
/*! \file gdt.cpp
* \brief GDT Manager
*
* This file defines the GDT class. It handles the i386 Global Descriptor Table
*
*/
#include <I386/gdt.h>
#include <errors.h>
// set instance pointer to a null pointer
I386::GDT* I386::GDT::_instance = 0;
I386::GDT* I386::GDT::getInstance() {
// check for exsisting instance
if(_instance == 0) {
// none found, create new instance
_instance = new GDT();
// check if we got a valid address
if(_instance == reinterpret_cast<GDT*>(E_ALLOC_NOMEM)) {
_instance = 0;
// no, major oops here!
return E_FAILURE;
}
}
// auto register
Core::ResourceManager::getInstance()->registerResource(_instance);
// return the instance
return _instance;
}
unsigned long I386::GDT::startResource() {
// setup the GDT pointer
this->_gdtPointer->limit = (sizeof(struct I386::GDTEntry) * GDT_SIZE) - 1;
this->_gdtPointer->base = reinterpret_cast<unsigned int>(this->_gdtEntries);
// null descriptor
this->setGate(0, 0, 0, 0, 0);
// code segment for ring0
this->setGate(KERNEL_CS, 0, 0xffffffff, 0x9a, 0xcf);
// data segment for ring1
this->setGate(KERNEL_DS, 0, 0xffffffff, 0x92, 0xcf);
// load GDT pointer
asm volatile ("lgdt %0" : "=m" (*this->_gdtPointer));
// jump to new segment
asm volatile("ljmp $(0x08), $reload_segments");
// we need to set al non-code segments to KERNEL_DS
asm volatile("reload_segments:");
asm volatile("movl $0x10, %eax");
asm volatile("movl %eax, %ds");
asm volatile("movl %eax, %es");
asm volatile("movl %eax, %fs");
asm volatile("movl %eax, %gs");
asm volatile("movl %eax, %ss");
return E_SUCCESS;
}
const char* I386::GDT::getResourceName() {
return "Global Descriptor Table";
}
I386::GDT::GDT() {
this->_gdtPointer = new struct I386::GDTPointer();
this->_gdtEntries = reinterpret_cast<struct I386::GDTEntry*>(new struct I386::GDTEntry[GDT_SIZE]);
}
void I386::GDT::setGate(int segment, unsigned long base, unsigned long limit, unsigned char access, unsigned char granularity) {
int index = segment / 8;
// setup base address
this->_gdtEntries[index].base_low = (base & 0xffff);
this->_gdtEntries[index].base_middle = (base >> 16) & 0xff;
this->_gdtEntries[index].base_high = (base >> 24) & 0xff;
// setup limits
this->_gdtEntries[index].limit_low = (limit & 0xffff);
this->_gdtEntries[index].granularity = ((limit >> 16) & 0x0f);
// setup the granularity and access flags
this->_gdtEntries[index].granularity |= (granularity & 0xf0);
this->_gdtEntries[index].access = access;
}