mirror of
git://projects.qi-hardware.com/iris.git
synced 2025-01-20 00:31:05 +02:00
working keyboard
This commit is contained in:
parent
b39c710e87
commit
9a7abe2fd5
4
Makefile
4
Makefile
@ -17,11 +17,11 @@
|
|||||||
|
|
||||||
# Define some variables.
|
# Define some variables.
|
||||||
CXXFLAGS = -Wno-unused-parameter -fno-strict-aliasing -fno-builtin -nostdinc $(ARCH_CXXFLAGS)
|
CXXFLAGS = -Wno-unused-parameter -fno-strict-aliasing -fno-builtin -nostdinc $(ARCH_CXXFLAGS)
|
||||||
CPPFLAGS = -O5 $(ARCH_CPPFLAGS)
|
CPPFLAGS = -O5 -fno-inline $(ARCH_CPPFLAGS)
|
||||||
CC = $(CROSS)gcc
|
CC = $(CROSS)gcc
|
||||||
LD = $(CROSS)ld
|
LD = $(CROSS)ld
|
||||||
OBJCOPY = $(CROSS)objcopy
|
OBJCOPY = $(CROSS)objcopy
|
||||||
STRIP = $(CROSS)strip
|
STRIP = : $(CROSS)strip
|
||||||
|
|
||||||
headers = kernel.hh iris.h $(arch_headers)
|
headers = kernel.hh iris.h $(arch_headers)
|
||||||
kernel_sources = panic.cc data.cc alloc.cc invoke.cc schedule.cc $(arch_kernel_sources)
|
kernel_sources = panic.cc data.cc alloc.cc invoke.cc schedule.cc $(arch_kernel_sources)
|
||||||
|
31
alloc.ccp
31
alloc.ccp
@ -20,7 +20,7 @@
|
|||||||
|
|
||||||
#define PREV(x) (((Object_base **)(x))[-2])
|
#define PREV(x) (((Object_base **)(x))[-2])
|
||||||
#define NEXT(x) (((Object_base **)(x))[-1])
|
#define NEXT(x) (((Object_base **)(x))[-1])
|
||||||
#define SIZE (2 * sizeof (unsigned))
|
#define SIZE (2 * sizeof (Object_base *))
|
||||||
|
|
||||||
bool Memory::use ():
|
bool Memory::use ():
|
||||||
// Go up to parents, incrementing used.
|
// Go up to parents, incrementing used.
|
||||||
@ -50,6 +50,8 @@ unsigned raw_zalloc ():
|
|||||||
return (unsigned)ret
|
return (unsigned)ret
|
||||||
|
|
||||||
void raw_pfree (unsigned page):
|
void raw_pfree (unsigned page):
|
||||||
|
if !page:
|
||||||
|
return
|
||||||
FreePage *p = (FreePage *)page
|
FreePage *p = (FreePage *)page
|
||||||
p->next = junk_pages
|
p->next = junk_pages
|
||||||
junk_pages = p
|
junk_pages = p
|
||||||
@ -105,7 +107,7 @@ void *Memory::search_free (unsigned size, void **first):
|
|||||||
PREV (f) = NULL
|
PREV (f) = NULL
|
||||||
s = PAGE_SIZE
|
s = PAGE_SIZE
|
||||||
// We have a free block, possibly too large.
|
// We have a free block, possibly too large.
|
||||||
if s >= size + sizeof (Free) + SIZE:
|
if s >= size + sizeof (Free) + 2 * SIZE:
|
||||||
// Create the new object at the end and keep the Free.
|
// Create the new object at the end and keep the Free.
|
||||||
Free *obj = (Free *)((unsigned)f + s - size - SIZE)
|
Free *obj = (Free *)((unsigned)f + s - size - SIZE)
|
||||||
NEXT (obj) = NEXT (f)
|
NEXT (obj) = NEXT (f)
|
||||||
@ -212,18 +214,30 @@ Capability *Memory::alloc_capability (Receiver *target, Capability *parent, Capa
|
|||||||
if !ret:
|
if !ret:
|
||||||
return NULL
|
return NULL
|
||||||
ret->target = target
|
ret->target = target
|
||||||
|
ret->protected_data = protected_data
|
||||||
ret->parent = parent
|
ret->parent = parent
|
||||||
ret->children = NULL
|
ret->children = NULL
|
||||||
ret->sibling_prev = NULL
|
ret->sibling_prev = NULL
|
||||||
ret->sibling_next = parent_ptr ? *parent_ptr : NULL
|
if parent:
|
||||||
|
ret->sibling_next = parent->children
|
||||||
|
parent->children = ret
|
||||||
|
else:
|
||||||
|
if parent_ptr:
|
||||||
|
ret->sibling_next = *parent_ptr
|
||||||
|
else:
|
||||||
|
ret->sibling_next = NULL
|
||||||
if ret->sibling_next:
|
if ret->sibling_next:
|
||||||
ret->sibling_next->sibling_prev = ret
|
ret->sibling_next->sibling_prev = ret
|
||||||
ret->protected_data = protected_data
|
|
||||||
return ret
|
return ret
|
||||||
|
|
||||||
Capability *Memory::clone_capability (Capability *source, bool copy, Capability *ret):
|
Capability *Memory::clone_capability (Capability *source, bool copy, Capability *ret):
|
||||||
if copy:
|
if copy:
|
||||||
return alloc_capability (source->target, source->parent, source->parent ? &source->parent->children : &source->target->capabilities, source->protected_data, ret)
|
if source->parent:
|
||||||
|
return alloc_capability (source->target, source->parent, &source->parent->children, source->protected_data, ret)
|
||||||
|
else if (unsigned)source->target & ~KERNEL_MASK:
|
||||||
|
return alloc_capability (source->target, source->parent, &source->target->capabilities, source->protected_data, ret)
|
||||||
|
else:
|
||||||
|
return alloc_capability (source->target, source->parent, &((Object_base *)source->protected_data)->refs, source->protected_data, ret)
|
||||||
else:
|
else:
|
||||||
return alloc_capability (source->target, source, &source->children, source->protected_data, ret)
|
return alloc_capability (source->target, source, &source->children, source->protected_data, ret)
|
||||||
|
|
||||||
@ -321,8 +335,10 @@ void Capability::invalidate ():
|
|||||||
return
|
return
|
||||||
if sibling_prev:
|
if sibling_prev:
|
||||||
sibling_prev->sibling_next = sibling_next
|
sibling_prev->sibling_next = sibling_next
|
||||||
else if target:
|
else if (unsigned)target & ~KERNEL_MASK:
|
||||||
target->capabilities = sibling_next
|
target->capabilities = sibling_next
|
||||||
|
else:
|
||||||
|
((Object_base *)protected_data)->refs = sibling_next
|
||||||
if sibling_next:
|
if sibling_next:
|
||||||
sibling_next->sibling_prev = sibling_prev
|
sibling_next->sibling_prev = sibling_prev
|
||||||
parent = NULL
|
parent = NULL
|
||||||
@ -374,9 +390,10 @@ void Page::forget ():
|
|||||||
data.share_prev = NULL
|
data.share_prev = NULL
|
||||||
data.share_next = NULL
|
data.share_next = NULL
|
||||||
else:
|
else:
|
||||||
|
if ~data.flags & PAGE_FLAG_PHYSICAL:
|
||||||
raw_pfree (data.frame)
|
raw_pfree (data.frame)
|
||||||
data.frame = 0
|
data.frame = 0
|
||||||
data.flags &= ~(PAGE_FLAG_FRAME | PAGE_FLAG_SHARED)
|
data.flags &= ~(PAGE_FLAG_FRAME | PAGE_FLAG_SHARED | PAGE_FLAG_PHYSICAL | PAGE_FLAG_UNCACHED)
|
||||||
Page_arch_update_mapping (this)
|
Page_arch_update_mapping (this)
|
||||||
|
|
||||||
void Cappage::forget ():
|
void Cappage::forget ():
|
||||||
|
@ -14,7 +14,7 @@ GPIO control:
|
|||||||
|
|
||||||
// GP ... Registers: one set for each port of 32 pins; total 128 pins means 4 groups. Total size: 4 * 0x30 == 0xc0.
|
// GP ... Registers: one set for each port of 32 pins; total 128 pins means 4 groups. Total size: 4 * 0x30 == 0xc0.
|
||||||
#define GPIO_GPDR(n) (GPIO_BASE + (0x00 + (n)*0x30)) // D: data
|
#define GPIO_GPDR(n) (GPIO_BASE + (0x00 + (n)*0x30)) // D: data
|
||||||
#define GPIO_GPDIR(n) (GPIO_BASE + (0x04 + (n)*0x30)) // DI: data in: 1 is input; 0 is output. Disable interrupts on the pin before touching this.
|
#define GPIO_GPDIR(n) (GPIO_BASE + (0x04 + (n)*0x30)) // DI: data in: 0 is input; 1 is output. Disable interrupts on the pin before touching this.
|
||||||
#define GPIO_GPODR(n) (GPIO_BASE + (0x08 + (n)*0x30)) // OD:
|
#define GPIO_GPODR(n) (GPIO_BASE + (0x08 + (n)*0x30)) // OD:
|
||||||
#define GPIO_GPPUR(n) (GPIO_BASE + (0x0c + (n)*0x30)) // PU: pull-up (1 is enable)
|
#define GPIO_GPPUR(n) (GPIO_BASE + (0x0c + (n)*0x30)) // PU: pull-up (1 is enable)
|
||||||
#define GPIO_GPALR(n) (GPIO_BASE + (0x10 + (n)*0x30)) // AL: alternate lower: per 2 bit; 00 means use as gpio; other values mean use as alternate function
|
#define GPIO_GPALR(n) (GPIO_BASE + (0x10 + (n)*0x30)) // AL: alternate lower: per 2 bit; 00 means use as gpio; other values mean use as alternate function
|
||||||
@ -180,3 +180,134 @@ do { \
|
|||||||
REG_GPIO_GPALR(0) &= 0xFF000000; \
|
REG_GPIO_GPALR(0) &= 0xFF000000; \
|
||||||
REG_GPIO_GPALR(0) |= 0x00555555; \
|
REG_GPIO_GPALR(0) |= 0x00555555; \
|
||||||
} while (0)
|
} while (0)
|
||||||
|
|
||||||
|
|
||||||
|
// Pins on the trendtac:
|
||||||
|
0 keyboard
|
||||||
|
1 keyboard
|
||||||
|
2 keyboard
|
||||||
|
3 keyboard
|
||||||
|
4 keyboard
|
||||||
|
5 keyboard
|
||||||
|
6 keyboard
|
||||||
|
7 keyboard
|
||||||
|
8 keyboard interrupt
|
||||||
|
9
|
||||||
|
10
|
||||||
|
11
|
||||||
|
12
|
||||||
|
13 touchpad right button
|
||||||
|
14
|
||||||
|
15
|
||||||
|
16 touchpad left button
|
||||||
|
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 keyboard
|
||||||
|
97 keyboard
|
||||||
|
98 keyboard
|
||||||
|
99 keyboard
|
||||||
|
100 keyboard
|
||||||
|
101 keyboard
|
||||||
|
102 keyboard
|
||||||
|
103 keyboard
|
||||||
|
104 keyboard
|
||||||
|
105 keyboard
|
||||||
|
106 keyboard
|
||||||
|
107 keyboard
|
||||||
|
108 keyboard
|
||||||
|
109 keyboard
|
||||||
|
110 keyboard
|
||||||
|
111 keyboard
|
||||||
|
112
|
||||||
|
113
|
||||||
|
114
|
||||||
|
115
|
||||||
|
116
|
||||||
|
117
|
||||||
|
118
|
||||||
|
119
|
||||||
|
120
|
||||||
|
121
|
||||||
|
122
|
||||||
|
123
|
||||||
|
124
|
||||||
|
125 keyboard
|
||||||
|
126
|
||||||
|
127
|
||||||
|
@ -16,10 +16,15 @@
|
|||||||
// along with this program. If not, see <http://www.gnu.org/licenses/>.
|
// along with this program. If not, see <http://www.gnu.org/licenses/>.
|
||||||
|
|
||||||
.globl __start
|
.globl __start
|
||||||
|
.globl __my_receiver
|
||||||
|
.globl __my_thread
|
||||||
|
.globl __my_memory
|
||||||
|
.globl __my_call
|
||||||
.set noreorder
|
.set noreorder
|
||||||
|
|
||||||
__start:
|
__start:
|
||||||
bal 1f
|
bal 1f
|
||||||
|
nop
|
||||||
.word _gp
|
.word _gp
|
||||||
1:
|
1:
|
||||||
lw $gp, 0($ra)
|
lw $gp, 0($ra)
|
||||||
|
@ -18,63 +18,101 @@
|
|||||||
|
|
||||||
#include "iris.h"
|
#include "iris.h"
|
||||||
|
|
||||||
// GPIO pins for the keyboard: Rows =
|
// GPIO pins for the keyboard://
|
||||||
// Rows = 96, 97, 98, 99, 100, 101, 102, 103, 104, 105, 106, 107, 108, 109, 110, 111, 125
|
// Rows = 96, 97, 98, 99, 100, 101, 102, 103, 104, 105, 106, 107, 108, 109, 110, 111, 125
|
||||||
// Cols = 0, 1, 2, 3, 4, 5, 6, 7
|
// Cols = 0, 1, 2, 3, 4, 5, 6, 7
|
||||||
|
|
||||||
// Rows: 60...6f; 7d
|
// Rows: 60...6f; 7d
|
||||||
|
|
||||||
// Nicely aligned to a port: use cols as output; rows as input.
|
// Nicely aligned to a port: use rows as output; cols as input.
|
||||||
|
|
||||||
// Map memory from b0010000, which is really 10010000 in kseg1.
|
// Map memory from b0010000, which is really 10010000 in kseg1.
|
||||||
|
|
||||||
#define D(n) (*(volatile unsigned *)(0x00 + 0x30 * n + address))
|
#define D(n) (*(volatile unsigned *)(0x00 + 0x30 * n + address))
|
||||||
#define DI(n) (*(volatile unsigned *)(0x04 + 0x30 * n + address))
|
#define DI(n) (*(volatile unsigned *)(0x04 + 0x30 * n + address))
|
||||||
|
#define PU(n) (*(volatile unsigned *)(0x0c + 0x30 * n + address))
|
||||||
#define AL(n) (*(volatile unsigned *)(0x10 + 0x30 * n + address))
|
#define AL(n) (*(volatile unsigned *)(0x10 + 0x30 * n + address))
|
||||||
#define AU(n) (*(volatile unsigned *)(0x14 + 0x30 * n + address))
|
#define AU(n) (*(volatile unsigned *)(0x14 + 0x30 * n + address))
|
||||||
#define IE(n) (*(volatile unsigned *)(0x20 + 0x30 * n + address))
|
#define IE(n) (*(volatile unsigned *)(0x20 + 0x30 * n + address))
|
||||||
|
|
||||||
void event (bool release, unsigned col, unsigned row):
|
unsigned const address = 0x00010000
|
||||||
debug_set_led ((col & 1) | (row & 2) | (release ? 4 : 0))
|
|
||||||
|
static void event (bool release, unsigned row, unsigned col):
|
||||||
|
debug_set_led (col * 2 + (release ? 1 : 0))
|
||||||
|
|
||||||
|
static void delay ():
|
||||||
|
for unsigned i = 0; i < 100000; ++i:
|
||||||
|
IE (3) &= ~0x2000ffff
|
||||||
|
|
||||||
int main ():
|
int main ():
|
||||||
// map memory
|
// map memory
|
||||||
Capability page = memory_create_page (__my_memory)
|
Capability page = memory_create_page (__my_memory)
|
||||||
alloc_physical (page, 0x10010000, 0)
|
alloc_physical (page, 0x10010000, 0)
|
||||||
unsigned const address = 0x00010000
|
memory_map (__my_memory, page, address, 1)
|
||||||
memory_map (__my_memory, page, address)
|
|
||||||
|
|
||||||
|
#if 0
|
||||||
|
// Keyboard stuff doesn't seem to work. Try the simpler part: touchpad buttons.
|
||||||
|
IE (0) &= ~0x00012000
|
||||||
|
AL (0) &= ~0x0c000000
|
||||||
|
AU (0) &= ~0x00000003
|
||||||
|
DI (0) &= ~0x00012000
|
||||||
|
PU (0) &= ~0x00012000
|
||||||
|
|
||||||
|
unsigned old = 0
|
||||||
|
while true:
|
||||||
|
unsigned data = D (0) & 0x00012000
|
||||||
|
if data == old:
|
||||||
|
continue
|
||||||
|
if data & ~old & 0x00010000:
|
||||||
|
event (true, 0, 0)
|
||||||
|
else if ~data & old & 0x00010000:
|
||||||
|
event (false, 0, 0)
|
||||||
|
if data & ~old & 0x00002000:
|
||||||
|
event (true, 1, 1)
|
||||||
|
else if ~data & old & 0x00002000:
|
||||||
|
event (false, 1, 1)
|
||||||
|
old = data
|
||||||
|
#else
|
||||||
// Disable all interrupts.
|
// Disable all interrupts.
|
||||||
IE (3) &= ~0x2000ffff
|
IE (3) &= ~0x2000ffff
|
||||||
IE (0) &= ~0x000000ff
|
IE (0) &= ~0x000001ff
|
||||||
|
|
||||||
// Set all to GPIO
|
// Set all to GPIO.
|
||||||
AL (3) = 0
|
AL (3) = 0
|
||||||
AU (3) &= ~0x0c000000
|
AU (3) &= ~0x0c000000
|
||||||
AL (0) &= ~0x0000ffff
|
AL (0) &= ~0x0003ffff
|
||||||
|
|
||||||
// Set all to input
|
// Set all rows to input and enable the pull-ups.
|
||||||
|
DI (0) &= ~0x000000ff
|
||||||
|
PU (0) |= 0x000000ff
|
||||||
|
|
||||||
|
// Set all columns to output, 0.
|
||||||
DI (3) |= 0x2000ffff
|
DI (3) |= 0x2000ffff
|
||||||
DI (0) |= 0x000000ff
|
D (3) &= ~0x2000ffff
|
||||||
|
|
||||||
unsigned keys[2][8]
|
#define NUM_COLS 17
|
||||||
for unsigned i = 0; i < 8; ++i:
|
unsigned keys[NUM_COLS]
|
||||||
keys[1][i] = 0x2000ffff
|
for unsigned i = 0; i < NUM_COLS; ++i:
|
||||||
|
keys[i] = 0
|
||||||
|
|
||||||
|
// Pin numbers for the cols, relative to the start of the port (so minus 0x60).
|
||||||
|
int const cols[NUM_COLS] = { 0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15, 29 }
|
||||||
while true:
|
while true:
|
||||||
// read keyboard
|
// read keyboard
|
||||||
for unsigned col = 0; col < 8; ++col:
|
unsigned data[NUM_COLS]
|
||||||
// set col to output, all others to input
|
for unsigned col = 0; col < NUM_COLS; ++col:
|
||||||
DI (0) = (DI (0) & ~0x000000ff) | (1 << col)
|
D (3) &= ~0x2000ffff
|
||||||
// read input
|
delay ()
|
||||||
keys[0][col] = D (3) & ~0x2000ffff
|
unsigned zero = ~D (0) & 0x000000ff
|
||||||
// Generate events
|
D (3) = (D (3) & ~0x2000ffff) | (1 << cols[col])
|
||||||
if keys[0][col] == keys[1][col]:
|
delay ()
|
||||||
continue
|
data[col] = D (0) & zero
|
||||||
unsigned bit, b
|
|
||||||
for bit = 1, b = 0; bit < 0x10000; bit <<= 1, ++b:
|
// Generate events.
|
||||||
if (keys[0][col] ^ keys[1][col]) & bit:
|
for unsigned col = 0; col < NUM_COLS; ++col:
|
||||||
event (keys[0][col] & bit, col, b)
|
for unsigned row = 0; row < 8; ++row:
|
||||||
if (keys[0][col] ^ keys[1][col]) & 0x20000000:
|
if (data[col] ^ keys[col]) & (1 << row):
|
||||||
// Not really bit 16, but it's easier to handle.
|
event (data[col] & (1 << row), row, col)
|
||||||
event (keys[0][col] & 0x20000000, col, 16)
|
keys[col] = data[col]
|
||||||
schedule ()
|
schedule ()
|
||||||
|
#endif
|
||||||
|
@ -20,4 +20,4 @@
|
|||||||
|
|
||||||
int main ():
|
int main ():
|
||||||
while true:
|
while true:
|
||||||
__asm__ volatile ("move $v0, $zero; li $a0, 1 ; move $a1, $zero ; move $a2, $a0 ; syscall")
|
schedule ()
|
||||||
|
315
invoke.ccp
315
invoke.ccp
@ -30,6 +30,7 @@ Capability *Memory::find_capability (unsigned code, bool *copy):
|
|||||||
if p->data.frame == (unsigned)page:
|
if p->data.frame == (unsigned)page:
|
||||||
return &page[num]
|
return &page[num]
|
||||||
else:
|
else:
|
||||||
|
code &= ~3
|
||||||
// Normal capability
|
// Normal capability
|
||||||
for Capability *c = capabilities; c; c = c->next:
|
for Capability *c = capabilities; c; c = c->next:
|
||||||
if c == (Capability *)code:
|
if c == (Capability *)code:
|
||||||
@ -46,36 +47,37 @@ bool Receiver::try_deliver ():
|
|||||||
break
|
break
|
||||||
if !m:
|
if !m:
|
||||||
return false
|
return false
|
||||||
Capability *c[4]
|
Capability::Context c
|
||||||
for unsigned i = 0; i < 4; ++i:
|
for unsigned i = 0; i < 4; ++i:
|
||||||
|
c.data[i] = m->data[i]
|
||||||
if !m->capabilities[i]:
|
if !m->capabilities[i]:
|
||||||
c[i] = NULL
|
c.cap[i] = NULL
|
||||||
else:
|
else:
|
||||||
c[i] = owner->address_space->clone_capability (m->capabilities[i], true)
|
c.cap[i] = owner->address_space->clone_capability (m->capabilities[i], true)
|
||||||
if !c[i]:
|
if !c.cap[i]:
|
||||||
for unsigned j = 0; j < i; ++j:
|
for unsigned j = 0; j < i; ++j:
|
||||||
owner->address_space->free_capability (c[i])
|
owner->address_space->free_capability (c.cap[i])
|
||||||
return false
|
return false
|
||||||
Thread_arch_receive (owner, m->data, c)
|
Thread_arch_receive (owner, &c)
|
||||||
owner->unwait ()
|
owner->unwait ()
|
||||||
return true
|
return true
|
||||||
|
|
||||||
bool Receiver::send_message (unsigned protected_data, unsigned data[4], Capability *cap[4], bool copy[4]):
|
bool Receiver::send_message (unsigned protected_data, Capability::Context *c):
|
||||||
bool tried_direct = false
|
bool tried_direct = false
|
||||||
if owner && owner->is_waiting () && (protected_data == reply_protected_data || !protected_only):
|
if owner && owner->is_waiting () && (protected_data == reply_protected_data || !protected_only):
|
||||||
Capability *c[4]
|
Capability::Context n
|
||||||
for unsigned i = 0; i < 4; ++i:
|
for unsigned i = 0; i < 4; ++i:
|
||||||
if !cap[i]:
|
if !c->cap[i]:
|
||||||
c[i] = NULL
|
n.cap[i] = NULL
|
||||||
else:
|
else:
|
||||||
c[i] = owner->address_space->clone_capability (cap[i], copy[i])
|
n.cap[i] = owner->address_space->clone_capability (c->cap[i], c->copy[i])
|
||||||
if !c[i]:
|
if !n.cap[i]:
|
||||||
for unsigned j = 0; j < i; ++j:
|
for unsigned j = 0; j < i; ++j:
|
||||||
owner->address_space->free_capability (c[i])
|
owner->address_space->free_capability (n.cap[i])
|
||||||
tried_direct = true
|
tried_direct = true
|
||||||
break
|
break
|
||||||
if !tried_direct:
|
if !tried_direct:
|
||||||
Thread_arch_receive (owner, data, c)
|
Thread_arch_receive (owner, &n)
|
||||||
owner->unwait ()
|
owner->unwait ()
|
||||||
return true
|
return true
|
||||||
// The owner was not waiting, or it was not possible to deliver the message. Put it in the queue.
|
// The owner was not waiting, or it was not possible to deliver the message. Put it in the queue.
|
||||||
@ -83,11 +85,11 @@ bool Receiver::send_message (unsigned protected_data, unsigned data[4], Capabili
|
|||||||
if !msg:
|
if !msg:
|
||||||
return false
|
return false
|
||||||
for unsigned i = 0; i < 4; ++i:
|
for unsigned i = 0; i < 4; ++i:
|
||||||
msg->data[i] = data[i]
|
msg->data[i] = c->data[i]
|
||||||
if !cap[i]:
|
if !c->cap[i]:
|
||||||
msg->capabilities[i] = NULL
|
msg->capabilities[i] = NULL
|
||||||
else:
|
else:
|
||||||
msg->capabilities[i] = address_space->clone_capability (cap[i], copy[i])
|
msg->capabilities[i] = address_space->clone_capability (c->cap[i], c->copy[i])
|
||||||
if !msg->capabilities[i]:
|
if !msg->capabilities[i]:
|
||||||
for unsigned j = 0; j < i; ++j:
|
for unsigned j = 0; j < i; ++j:
|
||||||
address_space->free_capability (msg->capabilities[j])
|
address_space->free_capability (msg->capabilities[j])
|
||||||
@ -113,106 +115,125 @@ static void fill_cap (Capability *r, unsigned target, unsigned protected_data):
|
|||||||
static void reply_cap (unsigned target, unsigned protected_data):
|
static void reply_cap (unsigned target, unsigned protected_data):
|
||||||
Capability r
|
Capability r
|
||||||
fill_cap (&r, target, protected_data)
|
fill_cap (&r, target, protected_data)
|
||||||
unsigned d[4] = { 0, 0, 0, 0 }
|
Capability::Context c
|
||||||
Capability *caps[4] = { &r, NULL, NULL, NULL }
|
for unsigned i = 0; i < 4; ++i:
|
||||||
bool cops[4] = { true, false, false, false }
|
c.data[i] = 0
|
||||||
|
c.cap[0] = &r
|
||||||
|
c.copy[0] = true
|
||||||
|
for unsigned i = 1; i < 4; ++i:
|
||||||
|
c.cap[i] = NULL
|
||||||
|
c.copy[i] = false
|
||||||
if reply:
|
if reply:
|
||||||
reply->invoke (d, caps, cops)
|
reply->invoke (&c)
|
||||||
else:
|
else if reply_receiver:
|
||||||
reply_receiver->send_message (reply_receiver->reply_protected_data, d, caps, cops)
|
reply_receiver->send_message (reply_receiver->reply_protected_data, &c)
|
||||||
r.invalidate ()
|
r.invalidate ()
|
||||||
|
|
||||||
static void reply_cap (Capability *cap, bool copy):
|
static void reply_cap (Capability *cap, bool copy):
|
||||||
unsigned d[4] = { 0, 0, 0, 0 }
|
Capability::Context c
|
||||||
Capability *caps[4] = { cap, NULL, NULL, NULL }
|
for unsigned i = 0; i < 4; ++i:
|
||||||
bool cops[4] = { copy, false, false, false }
|
c.data[i] = 0
|
||||||
|
c.cap[0] = cap
|
||||||
|
c.copy[0] = copy
|
||||||
|
for unsigned i = 1; i < 4; ++i:
|
||||||
|
c.cap[i] = NULL
|
||||||
|
c.copy[i] = false
|
||||||
if reply:
|
if reply:
|
||||||
reply->invoke (d, caps, cops)
|
reply->invoke (&c)
|
||||||
else:
|
else if reply_receiver:
|
||||||
reply_receiver->send_message (reply_receiver->reply_protected_data, d, caps, cops)
|
reply_receiver->send_message (reply_receiver->reply_protected_data, &c)
|
||||||
|
|
||||||
static void reply_num (unsigned num):
|
static void reply_num (unsigned num):
|
||||||
unsigned d[4] = { num, 0, 0, 0 }
|
Capability::Context c
|
||||||
Capability *caps[4] = { NULL, NULL, NULL, NULL }
|
c.data[0] = num
|
||||||
bool cops[4] = { false, false, false, false }
|
for unsigned i = 1; i < 4; ++i:
|
||||||
|
c.data[i] = 0
|
||||||
|
for unsigned i = 0; i < 4; ++i:
|
||||||
|
c.cap[i] = NULL
|
||||||
|
c.copy[i] = false
|
||||||
if reply:
|
if reply:
|
||||||
reply->invoke (d, caps, cops)
|
reply->invoke (&c)
|
||||||
else:
|
else if reply_receiver:
|
||||||
reply_receiver->send_message (reply_receiver->reply_protected_data, d, caps, cops)
|
reply_receiver->send_message (reply_receiver->reply_protected_data, &c)
|
||||||
|
|
||||||
static void reply_nums (unsigned num1, unsigned num2):
|
static void reply_nums (unsigned num1, unsigned num2):
|
||||||
unsigned d[4] = { num1, num2, 0, 0 }
|
Capability::Context c
|
||||||
Capability *caps[4] = { NULL, NULL, NULL, NULL }
|
c.data[0] = num1
|
||||||
bool cops[4] = { false, false, false, false }
|
c.data[1] = num2
|
||||||
|
c.data[2] = 0
|
||||||
|
c.data[3] = 0
|
||||||
|
for unsigned i = 0; i < 4; ++i:
|
||||||
|
c.cap[i] = NULL
|
||||||
|
c.copy[i] = false
|
||||||
if reply:
|
if reply:
|
||||||
reply->invoke (d, caps, cops)
|
reply->invoke (&c)
|
||||||
else:
|
else if reply_receiver:
|
||||||
reply_receiver->send_message (reply_receiver->reply_protected_data, d, caps, cops)
|
reply_receiver->send_message (reply_receiver->reply_protected_data, &c)
|
||||||
|
|
||||||
static void receiver_invoke (unsigned target, unsigned protected_data, Capability *cap, unsigned data[4]):
|
static void receiver_invoke (unsigned target, unsigned protected_data, Capability::Context *c):
|
||||||
Receiver *receiver = (Receiver *)protected_data
|
Receiver *receiver = (Receiver *)protected_data
|
||||||
switch data[0]:
|
switch c->data[0]:
|
||||||
case CAP_RECEIVER_SET_OWNER:
|
case CAP_RECEIVER_SET_OWNER:
|
||||||
if ((unsigned)cap->target & (CAPTYPE_MASK | ~KERNEL_MASK)) != CAPTYPE_THREAD:
|
if !c->cap[0] || ((unsigned)c->cap[0]->target & (CAPTYPE_MASK | ~KERNEL_MASK)) != CAPTYPE_THREAD:
|
||||||
// FIXME: This makes it impossible to use a fake Thread capability.
|
// FIXME: This makes it impossible to use a fake Thread capability.
|
||||||
return
|
return
|
||||||
receiver->own ((Thread *)cap->protected_data)
|
receiver->own ((Thread *)c->cap[0]->protected_data)
|
||||||
break
|
break
|
||||||
case CAP_RECEIVER_CREATE_CAPABILITY:
|
case CAP_RECEIVER_CREATE_CAPABILITY:
|
||||||
reply_cap ((unsigned)receiver, data[1])
|
reply_cap ((unsigned)receiver, c->data[1])
|
||||||
break
|
break
|
||||||
case CAP_RECEIVER_CREATE_CALL_CAPABILITY:
|
case CAP_RECEIVER_CREATE_CALL_CAPABILITY:
|
||||||
reply_cap (CAPTYPE_RECEIVER | CAP_RECEIVER_CALL | (data[1] ? CAP_RECEIVER_CALL_ASYNC : 0), protected_data)
|
reply_cap (CAPTYPE_RECEIVER | (1 << CAP_RECEIVER_CALL) | (c->data[1] ? 1 << CAP_RECEIVER_CALL_ASYNC : 0), protected_data)
|
||||||
break
|
break
|
||||||
case CAP_RECEIVER_GET_REPLY_PROTECTED_DATA:
|
case CAP_RECEIVER_GET_REPLY_PROTECTED_DATA:
|
||||||
reply_nums (receiver->reply_protected_data, receiver->protected_only ? 1 : 0)
|
reply_nums (receiver->reply_protected_data, receiver->protected_only ? 1 : 0)
|
||||||
break
|
break
|
||||||
case CAP_RECEIVER_SET_REPLY_PROTECTED_DATA:
|
case CAP_RECEIVER_SET_REPLY_PROTECTED_DATA:
|
||||||
receiver->reply_protected_data = data[1]
|
receiver->reply_protected_data = c->data[1]
|
||||||
receiver->protected_only = data[2]
|
receiver->protected_only = c->data[2]
|
||||||
break
|
break
|
||||||
default:
|
default:
|
||||||
break
|
break
|
||||||
|
|
||||||
static void memory_invoke (unsigned target, unsigned protected_data, Capability *cap, unsigned request, unsigned data):
|
static void memory_invoke (unsigned target, unsigned protected_data, Capability::Context *c):
|
||||||
Memory *mem = (Memory *)protected_data
|
Memory *mem = (Memory *)protected_data
|
||||||
switch request:
|
switch c->data[0]:
|
||||||
case CAP_MEMORY_CREATE:
|
case CAP_MEMORY_CREATE:
|
||||||
unsigned rights = data & REQUEST_MASK
|
unsigned rights = c->data[1] & REQUEST_MASK
|
||||||
data &= CAPTYPE_MASK
|
unsigned type = c->data[1] & CAPTYPE_MASK
|
||||||
switch data:
|
switch type:
|
||||||
case CAPTYPE_RECEIVER:
|
case CAPTYPE_RECEIVER:
|
||||||
Receiver *ret = mem->alloc_receiver ()
|
Receiver *ret = mem->alloc_receiver ()
|
||||||
if ret:
|
if ret:
|
||||||
reply_cap (data | (rights & CAP_RECEIVER_ALL_RIGHTS), (unsigned)ret)
|
reply_cap (CAPTYPE_RECEIVER | (rights & CAP_RECEIVER_ALL_RIGHTS), (unsigned)ret)
|
||||||
else:
|
else:
|
||||||
reply_num (0)
|
reply_num (0)
|
||||||
break
|
break
|
||||||
case CAPTYPE_MEMORY:
|
case CAPTYPE_MEMORY:
|
||||||
Memory *ret = mem->alloc_memory ()
|
Memory *ret = mem->alloc_memory ()
|
||||||
if ret:
|
if ret:
|
||||||
reply_cap (data | (rights & CAP_MEMORY_ALL_RIGHTS), (unsigned)ret)
|
reply_cap (CAPTYPE_MEMORY | (rights & CAP_MEMORY_ALL_RIGHTS), (unsigned)ret)
|
||||||
else:
|
else:
|
||||||
reply_num (0)
|
reply_num (0)
|
||||||
break
|
break
|
||||||
case CAPTYPE_THREAD:
|
case CAPTYPE_THREAD:
|
||||||
Thread *ret = mem->alloc_thread ()
|
Thread *ret = mem->alloc_thread ()
|
||||||
if ret:
|
if ret:
|
||||||
reply_cap (data | (rights & CAP_THREAD_ALL_RIGHTS), (unsigned)ret)
|
reply_cap (CAPTYPE_THREAD | (rights & CAP_THREAD_ALL_RIGHTS), (unsigned)ret)
|
||||||
else:
|
else:
|
||||||
reply_num (0)
|
reply_num (0)
|
||||||
break
|
break
|
||||||
case CAPTYPE_PAGE:
|
case CAPTYPE_PAGE:
|
||||||
Page *ret = mem->alloc_page ()
|
Page *ret = mem->alloc_page ()
|
||||||
if ret:
|
if ret:
|
||||||
reply_cap (data | (rights & CAP_PAGE_ALL_RIGHTS), (unsigned)ret)
|
reply_cap (CAPTYPE_PAGE | (rights & CAP_PAGE_ALL_RIGHTS), (unsigned)ret)
|
||||||
else:
|
else:
|
||||||
reply_num (0)
|
reply_num (0)
|
||||||
break
|
break
|
||||||
case CAPTYPE_CAPPAGE:
|
case CAPTYPE_CAPPAGE:
|
||||||
Cappage *ret = mem->alloc_cappage ()
|
Cappage *ret = mem->alloc_cappage ()
|
||||||
if ret:
|
if ret:
|
||||||
reply_cap (data | (rights & CAP_CAPPAGE_ALL_RIGHTS), (unsigned)ret)
|
reply_cap (CAPTYPE_CAPPAGE | (rights & CAP_CAPPAGE_ALL_RIGHTS), (unsigned)ret)
|
||||||
else:
|
else:
|
||||||
reply_num (0)
|
reply_num (0)
|
||||||
break
|
break
|
||||||
@ -220,26 +241,26 @@ static void memory_invoke (unsigned target, unsigned protected_data, Capability
|
|||||||
return
|
return
|
||||||
break
|
break
|
||||||
case CAP_MEMORY_DESTROY:
|
case CAP_MEMORY_DESTROY:
|
||||||
if !cap || cap->address_space != mem || (unsigned)cap->target & ~KERNEL_MASK:
|
if !c->cap[0] || c->cap[0]->address_space != mem || (unsigned)c->cap[0]->target & ~KERNEL_MASK:
|
||||||
return
|
return
|
||||||
switch (unsigned)cap->target & CAPTYPE_MASK:
|
switch (unsigned)c->cap[0]->target & CAPTYPE_MASK:
|
||||||
case CAPTYPE_RECEIVER:
|
case CAPTYPE_RECEIVER:
|
||||||
mem->free_receiver ((Receiver *)cap->protected_data)
|
mem->free_receiver ((Receiver *)c->cap[0]->protected_data)
|
||||||
return
|
return
|
||||||
case CAPTYPE_MEMORY:
|
case CAPTYPE_MEMORY:
|
||||||
mem->free_memory ((Memory *)cap->protected_data)
|
mem->free_memory ((Memory *)c->cap[0]->protected_data)
|
||||||
return
|
return
|
||||||
case CAPTYPE_THREAD:
|
case CAPTYPE_THREAD:
|
||||||
mem->free_thread ((Thread *)cap->protected_data)
|
mem->free_thread ((Thread *)c->cap[0]->protected_data)
|
||||||
return
|
return
|
||||||
case CAPTYPE_PAGE:
|
case CAPTYPE_PAGE:
|
||||||
mem->free_page ((Page *)cap->protected_data)
|
mem->free_page ((Page *)c->cap[0]->protected_data)
|
||||||
return
|
return
|
||||||
case CAPTYPE_CAPABILITY:
|
case CAPTYPE_CAPABILITY:
|
||||||
mem->free_capability ((Capability *)cap->protected_data)
|
mem->free_capability ((Capability *)c->cap[0]->protected_data)
|
||||||
return
|
return
|
||||||
case CAPTYPE_CAPPAGE:
|
case CAPTYPE_CAPPAGE:
|
||||||
mem->free_cappage ((Cappage *)cap->protected_data)
|
mem->free_cappage ((Cappage *)c->cap[0]->protected_data)
|
||||||
return
|
return
|
||||||
default:
|
default:
|
||||||
panic (0x55228930, "invalid case")
|
panic (0x55228930, "invalid case")
|
||||||
@ -249,41 +270,42 @@ static void memory_invoke (unsigned target, unsigned protected_data, Capability
|
|||||||
break
|
break
|
||||||
case CAP_MEMORY_MAP:
|
case CAP_MEMORY_MAP:
|
||||||
// FIXME: this should work for fake pages as well.
|
// FIXME: this should work for fake pages as well.
|
||||||
if (unsigned)cap->target & ~KERNEL_MASK || ((unsigned)cap->target & CAPTYPE_MASK) != CAPTYPE_PAGE:
|
if !c->cap[0] || (unsigned)c->cap[0]->target & ~KERNEL_MASK || ((unsigned)c->cap[0]->target & CAPTYPE_MASK) != CAPTYPE_PAGE:
|
||||||
break
|
break
|
||||||
Page *page = (Page *)cap->protected_data
|
Page *page = (Page *)c->cap[0]->protected_data
|
||||||
if page->address_space != mem:
|
if page->address_space != mem:
|
||||||
break
|
break
|
||||||
mem->map (page, data & PAGE_MASK, data & (unsigned)cap->target & (1 << CAP_PAGE_WRITE))
|
bool writable = c->data[1] & (unsigned)c->cap[0]->target & (1 << CAP_PAGE_WRITE)
|
||||||
|
mem->map (page, c->data[1] & PAGE_MASK, writable)
|
||||||
break
|
break
|
||||||
case CAP_MEMORY_MAPPING:
|
case CAP_MEMORY_MAPPING:
|
||||||
bool write
|
bool write
|
||||||
Page *page = mem->get_mapping (data, &write)
|
Page *page = mem->get_mapping (c->data[1], &write)
|
||||||
unsigned t = CAPTYPE_PAGE | REQUEST_MASK
|
unsigned t = CAPTYPE_PAGE | REQUEST_MASK
|
||||||
if !write:
|
if !write:
|
||||||
t &= ~CAP_PAGE_WRITE
|
t &= ~CAP_PAGE_WRITE
|
||||||
reply_cap (t, (unsigned)page)
|
reply_cap (t, (unsigned)page)
|
||||||
break
|
break
|
||||||
case CAP_MEMORY_SET_LIMIT:
|
case CAP_MEMORY_SET_LIMIT:
|
||||||
mem->limit = data
|
mem->limit = c->data[1]
|
||||||
break
|
break
|
||||||
case CAP_MEMORY_GET_LIMIT:
|
case CAP_MEMORY_GET_LIMIT:
|
||||||
reply_num (mem->limit)
|
reply_num (mem->limit)
|
||||||
break
|
break
|
||||||
case CAP_MEMORY_DROP:
|
case CAP_MEMORY_DROP:
|
||||||
if cap->address_space != mem:
|
if !c->cap[0] || c->cap[0]->address_space != mem:
|
||||||
break
|
break
|
||||||
mem->free_capability (cap)
|
mem->free_capability (c->cap[0])
|
||||||
break
|
break
|
||||||
default:
|
default:
|
||||||
break
|
break
|
||||||
|
|
||||||
static void thread_invoke (unsigned target, unsigned protected_data, Capability *cap, unsigned data[4]):
|
static void thread_invoke (unsigned target, unsigned protected_data, Capability::Context *c):
|
||||||
Thread *thread = (Thread *)protected_data
|
Thread *thread = (Thread *)protected_data
|
||||||
switch data[0]:
|
switch c->data[0]:
|
||||||
case CAP_THREAD_INFO:
|
case CAP_THREAD_INFO:
|
||||||
unsigned *value
|
unsigned *value
|
||||||
switch data[1]:
|
switch c->data[1]:
|
||||||
case CAP_THREAD_INFO_PC:
|
case CAP_THREAD_INFO_PC:
|
||||||
value = &thread->pc
|
value = &thread->pc
|
||||||
break
|
break
|
||||||
@ -292,10 +314,10 @@ static void thread_invoke (unsigned target, unsigned protected_data, Capability
|
|||||||
break
|
break
|
||||||
case CAP_THREAD_INFO_FLAGS:
|
case CAP_THREAD_INFO_FLAGS:
|
||||||
// It is not possible to set the PRIV flag, but it can be reset.
|
// It is not possible to set the PRIV flag, but it can be reset.
|
||||||
data[2] &= ~THREAD_FLAG_PRIV
|
c->data[2] &= ~THREAD_FLAG_PRIV
|
||||||
value = &thread->flags
|
value = &thread->flags
|
||||||
if data[3] & ~THREAD_FLAG_USER:
|
if c->data[3] & ~THREAD_FLAG_USER:
|
||||||
unsigned v = (*value & data[3]) | (data[2] & data[3])
|
unsigned v = (*value & c->data[3]) | (c->data[2] & c->data[3])
|
||||||
if (v & THREAD_FLAG_WAITING) != (*value & THREAD_FLAG_WAITING):
|
if (v & THREAD_FLAG_WAITING) != (*value & THREAD_FLAG_WAITING):
|
||||||
if v & THREAD_FLAG_WAITING:
|
if v & THREAD_FLAG_WAITING:
|
||||||
thread->wait ()
|
thread->wait ()
|
||||||
@ -308,11 +330,11 @@ static void thread_invoke (unsigned target, unsigned protected_data, Capability
|
|||||||
thread->unrun ()
|
thread->unrun ()
|
||||||
break
|
break
|
||||||
default:
|
default:
|
||||||
value = Thread_arch_info (thread, data[1])
|
value = Thread_arch_info (thread, c->data[1])
|
||||||
break
|
break
|
||||||
if value:
|
if value:
|
||||||
*value &= ~data[3]
|
*value &= ~c->data[3]
|
||||||
*value |= data[2] & data[3]
|
*value |= c->data[2] & c->data[3]
|
||||||
reply_num (*value)
|
reply_num (*value)
|
||||||
else:
|
else:
|
||||||
reply_num (0)
|
reply_num (0)
|
||||||
@ -321,24 +343,39 @@ static void thread_invoke (unsigned target, unsigned protected_data, Capability
|
|||||||
schedule ()
|
schedule ()
|
||||||
break
|
break
|
||||||
case CAP_THREAD_DEBUG:
|
case CAP_THREAD_DEBUG:
|
||||||
dbg_leds (data[1] & 1, data[1] & 2, data[1] & 4)
|
for unsigned i = 0; i < 2; ++i:
|
||||||
|
dbg_led (true, true, true)
|
||||||
|
dbg_sleep (100)
|
||||||
|
dbg_led (false, false, false)
|
||||||
|
dbg_sleep (100)
|
||||||
|
//dbg_send (c->data[1], 5)
|
||||||
break
|
break
|
||||||
case CAP_THREAD_REGISTER_INTERRUPT:
|
case CAP_THREAD_REGISTER_INTERRUPT:
|
||||||
// Threads with access to this call are trusted, so no sanity checking is done.
|
// Threads with access to this call are trusted, so no sanity checking is done.
|
||||||
arch_register_interrupt (data[1], cap ? (Receiver *)cap->protected_data : NULL)
|
arch_register_interrupt (c->data[1], c->cap[0] ? (Receiver *)c->cap[0]->protected_data : NULL)
|
||||||
break
|
break
|
||||||
case CAP_THREAD_GET_TOP_MEMORY:
|
case CAP_THREAD_GET_TOP_MEMORY:
|
||||||
// Threads with access to this call are trusted, so no sanity checking is done.
|
// Threads with access to this call are trusted, so no sanity checking is done.
|
||||||
reply_cap (CAPTYPE_MEMORY | (data[1] & CAP_MEMORY_ALL_RIGHTS), (unsigned)&top_memory)
|
reply_cap (CAPTYPE_MEMORY | (c->data[1] & CAP_MEMORY_ALL_RIGHTS), (unsigned)&top_memory)
|
||||||
break
|
break
|
||||||
case CAP_THREAD_MAKE_PRIV:
|
case CAP_THREAD_MAKE_PRIV:
|
||||||
// Threads with access to this call are trusted, so no sanity checking is done.
|
// Threads with access to this call are trusted, so no sanity checking is done.
|
||||||
if data[1] & THREAD_FLAG_PRIV:
|
if c->data[1] & THREAD_FLAG_PRIV:
|
||||||
((Thread *)cap->protected_data)->flags |= THREAD_FLAG_PRIV
|
((Thread *)c->cap[0]->protected_data)->flags |= THREAD_FLAG_PRIV
|
||||||
reply_cap (CAPTYPE_THREAD | (data[1] & CAP_THREAD_ALL_PRIV_RIGHTS), cap->protected_data)
|
reply_cap (CAPTYPE_THREAD | (c->data[1] & CAP_THREAD_ALL_PRIV_RIGHTS), c->cap[0]->protected_data)
|
||||||
break
|
break
|
||||||
case CAP_THREAD_ALLOC_PHYSICAL:
|
case CAP_THREAD_ALLOC_PHYSICAL:
|
||||||
// TODO
|
// Threads with access to this call are trusted, so no sanity checking is done.
|
||||||
|
Page *page = (Page *)c->cap[0]->protected_data
|
||||||
|
page->forget ()
|
||||||
|
if page->data.flags & PAGE_FLAG_PAYING:
|
||||||
|
page->data.flags &= ~PAGE_FLAG_PAYING
|
||||||
|
page->address_space->unuse ()
|
||||||
|
page->data.frame = c->data[1] & PAGE_MASK
|
||||||
|
if c->data[1] & 1:
|
||||||
|
page->data.flags |= PAGE_FLAG_FRAME | PAGE_FLAG_PHYSICAL
|
||||||
|
else:
|
||||||
|
page->data.flags |= PAGE_FLAG_FRAME | PAGE_FLAG_PHYSICAL | PAGE_FLAG_UNCACHED
|
||||||
break
|
break
|
||||||
default:
|
default:
|
||||||
break
|
break
|
||||||
@ -393,7 +430,7 @@ static bool cappage_check_payment (Cappage *cappage):
|
|||||||
p->data.flags &= ~(PAGE_FLAG_SHARED | PAGE_FLAG_FRAME)
|
p->data.flags &= ~(PAGE_FLAG_SHARED | PAGE_FLAG_FRAME)
|
||||||
return false
|
return false
|
||||||
|
|
||||||
static void page_invoke (unsigned target, unsigned protected_data, Capability *cap, bool copy, unsigned data[4]):
|
static void page_invoke (unsigned target, unsigned protected_data, Capability::Context *c):
|
||||||
Page *page
|
Page *page
|
||||||
Cappage *cappage
|
Cappage *cappage
|
||||||
ShareData *share_data
|
ShareData *share_data
|
||||||
@ -405,22 +442,22 @@ static void page_invoke (unsigned target, unsigned protected_data, Capability *c
|
|||||||
page = NULL
|
page = NULL
|
||||||
cappage = (Cappage *)protected_data
|
cappage = (Cappage *)protected_data
|
||||||
share_data = &cappage->data
|
share_data = &cappage->data
|
||||||
switch data[0]:
|
switch c->data[0]:
|
||||||
case CAP_PAGE_SHARE:
|
case CAP_PAGE_SHARE:
|
||||||
if ((unsigned)cap->target & CAPTYPE_MASK) != (target & CAPTYPE_MASK):
|
if !c->cap[0] || ((unsigned)c->cap[0]->target & CAPTYPE_MASK) != (target & CAPTYPE_MASK):
|
||||||
// FIXME: This makes it impossible to use a fake Page capability.
|
// FIXME: This makes it impossible to use a fake Page capability.
|
||||||
break
|
break
|
||||||
if page:
|
if page:
|
||||||
Page *t = (Page *)cap->protected_data
|
Page *t = (Page *)c->cap[0]->protected_data
|
||||||
t->forget ()
|
t->forget ()
|
||||||
if data[1] & PAGE_SHARE_READONLY:
|
if c->data[1] & PAGE_SHARE_READONLY:
|
||||||
t->data.flags &= ~PAGE_FLAG_WRITABLE
|
t->data.flags &= ~PAGE_FLAG_WRITABLE
|
||||||
if !page->data.flags & PAGE_FLAG_FRAME:
|
if !page->data.flags & PAGE_FLAG_FRAME:
|
||||||
break
|
break
|
||||||
if data[1] & PAGE_SHARE_COPY:
|
if c->data[1] & PAGE_SHARE_COPY:
|
||||||
if ~t->data.flags & PAGE_FLAG_PAYING:
|
if ~t->data.flags & PAGE_FLAG_PAYING:
|
||||||
break
|
break
|
||||||
if ~data[1] & PAGE_SHARE_FORGET || page->data.flags & PAGE_FLAG_SHARED:
|
if ~c->data[1] & PAGE_SHARE_FORGET || page->data.flags & PAGE_FLAG_SHARED:
|
||||||
unsigned *d = (unsigned *)page->data.frame
|
unsigned *d = (unsigned *)page->data.frame
|
||||||
if t == page:
|
if t == page:
|
||||||
Page *other = page->data.share_next ? (Page *)page->data.share_next : (Page *)page->data.share_prev
|
Page *other = page->data.share_next ? (Page *)page->data.share_next : (Page *)page->data.share_prev
|
||||||
@ -437,7 +474,7 @@ static void page_invoke (unsigned target, unsigned protected_data, Capability *c
|
|||||||
else:
|
else:
|
||||||
t->data.flags |= PAGE_FLAG_FRAME
|
t->data.flags |= PAGE_FLAG_FRAME
|
||||||
t->data.frame = raw_zalloc ()
|
t->data.frame = raw_zalloc ()
|
||||||
for unsigned i = 0; i <= (data[1] & ~PAGE_MASK); i += 4:
|
for unsigned i = 0; i <= (c->data[1] & ~PAGE_MASK); i += 4:
|
||||||
((unsigned *)t->data.frame)[i >> 2] = d[i >> 2]
|
((unsigned *)t->data.frame)[i >> 2] = d[i >> 2]
|
||||||
else:
|
else:
|
||||||
if t != page:
|
if t != page:
|
||||||
@ -450,7 +487,7 @@ static void page_invoke (unsigned target, unsigned protected_data, Capability *c
|
|||||||
else:
|
else:
|
||||||
if t == page:
|
if t == page:
|
||||||
break
|
break
|
||||||
if data[1] & PAGE_SHARE_FORGET:
|
if c->data[1] & PAGE_SHARE_FORGET:
|
||||||
if ~page->data.flags & PAGE_FLAG_SHARED:
|
if ~page->data.flags & PAGE_FLAG_SHARED:
|
||||||
if t->data.flags & PAGE_FLAG_PAYING:
|
if t->data.flags & PAGE_FLAG_PAYING:
|
||||||
t->data.frame = page->data.frame
|
t->data.frame = page->data.frame
|
||||||
@ -477,16 +514,16 @@ static void page_invoke (unsigned target, unsigned protected_data, Capability *c
|
|||||||
((Page *)t->data.share_prev)->data.share_next = t
|
((Page *)t->data.share_prev)->data.share_next = t
|
||||||
Page_arch_update_mapping (t)
|
Page_arch_update_mapping (t)
|
||||||
else:
|
else:
|
||||||
Cappage *t = (Cappage *)cap->protected_data
|
Cappage *t = (Cappage *)c->cap[0]->protected_data
|
||||||
t->forget ()
|
t->forget ()
|
||||||
if data[1] & PAGE_SHARE_READONLY:
|
if c->data[1] & PAGE_SHARE_READONLY:
|
||||||
t->data.flags &= ~PAGE_FLAG_WRITABLE
|
t->data.flags &= ~PAGE_FLAG_WRITABLE
|
||||||
if !cappage->data.flags & PAGE_FLAG_FRAME:
|
if !cappage->data.flags & PAGE_FLAG_FRAME:
|
||||||
break
|
break
|
||||||
if data[1] & PAGE_SHARE_COPY:
|
if c->data[1] & PAGE_SHARE_COPY:
|
||||||
if ~t->data.flags & PAGE_FLAG_PAYING:
|
if ~t->data.flags & PAGE_FLAG_PAYING:
|
||||||
break
|
break
|
||||||
if ~data[1] & PAGE_SHARE_FORGET || cappage->data.flags & PAGE_FLAG_SHARED:
|
if ~c->data[1] & PAGE_SHARE_FORGET || cappage->data.flags & PAGE_FLAG_SHARED:
|
||||||
unsigned *d = (unsigned *)cappage->data.frame
|
unsigned *d = (unsigned *)cappage->data.frame
|
||||||
if t == cappage:
|
if t == cappage:
|
||||||
Cappage *other = cappage->data.share_next ? (Cappage *)cappage->data.share_next : (Cappage *)cappage->data.share_prev
|
Cappage *other = cappage->data.share_next ? (Cappage *)cappage->data.share_next : (Cappage *)cappage->data.share_prev
|
||||||
@ -502,7 +539,7 @@ static void page_invoke (unsigned target, unsigned protected_data, Capability *c
|
|||||||
else:
|
else:
|
||||||
t->data.flags |= PAGE_FLAG_FRAME
|
t->data.flags |= PAGE_FLAG_FRAME
|
||||||
t->data.frame = raw_zalloc ()
|
t->data.frame = raw_zalloc ()
|
||||||
for unsigned i = 0; i < ((data[1] & ~PAGE_MASK) + 1) * sizeof (Capability); i += 4:
|
for unsigned i = 0; i < ((c->data[1] & ~PAGE_MASK) + 1) * sizeof (Capability); i += 4:
|
||||||
((unsigned *)t->data.frame)[i >> 2] = d[i >> 2]
|
((unsigned *)t->data.frame)[i >> 2] = d[i >> 2]
|
||||||
else:
|
else:
|
||||||
if t != cappage:
|
if t != cappage:
|
||||||
@ -513,7 +550,7 @@ static void page_invoke (unsigned target, unsigned protected_data, Capability *c
|
|||||||
else:
|
else:
|
||||||
if t == cappage:
|
if t == cappage:
|
||||||
break
|
break
|
||||||
if data[1] & PAGE_SHARE_FORGET:
|
if c->data[1] & PAGE_SHARE_FORGET:
|
||||||
if ~cappage->data.flags & PAGE_FLAG_SHARED:
|
if ~cappage->data.flags & PAGE_FLAG_SHARED:
|
||||||
if t->data.flags & PAGE_FLAG_PAYING:
|
if t->data.flags & PAGE_FLAG_PAYING:
|
||||||
t->data.frame = cappage->data.frame
|
t->data.frame = cappage->data.frame
|
||||||
@ -538,10 +575,12 @@ static void page_invoke (unsigned target, unsigned protected_data, Capability *c
|
|||||||
if t->data.share_prev:
|
if t->data.share_prev:
|
||||||
((Cappage *)t->data.share_prev)->data.share_next = t
|
((Cappage *)t->data.share_prev)->data.share_next = t
|
||||||
case CAP_PAGE_FLAGS:
|
case CAP_PAGE_FLAGS:
|
||||||
|
// Always refuse to set reserved flags.
|
||||||
|
c->data[2] &= ~(PAGE_FLAG_PHYSICAL | PAGE_FLAG_UNCACHED)
|
||||||
// Remember the old flags.
|
// Remember the old flags.
|
||||||
unsigned old = share_data->flags
|
unsigned old = share_data->flags
|
||||||
// Compute the new flags.
|
// Compute the new flags.
|
||||||
unsigned new_flags = (share_data->flags & ~data[2]) | (data[1] & data[2])
|
unsigned new_flags = (share_data->flags & ~c->data[2]) | (c->data[1] & c->data[2])
|
||||||
|
|
||||||
// If we stop paying, see if the frame is still paid for. If not, free it.
|
// If we stop paying, see if the frame is still paid for. If not, free it.
|
||||||
if ~new_flags & old & PAGE_FLAG_PAYING:
|
if ~new_flags & old & PAGE_FLAG_PAYING:
|
||||||
@ -626,54 +665,54 @@ static void page_invoke (unsigned target, unsigned protected_data, Capability *c
|
|||||||
reply_num (share_data->flags)
|
reply_num (share_data->flags)
|
||||||
break
|
break
|
||||||
case CAP_CAPPAGE_SET:
|
case CAP_CAPPAGE_SET:
|
||||||
if !cappage || data[1] >= CAPPAGE_SIZE || !(target & CAP_PAGE_WRITE):
|
if !cappage || c->data[1] >= CAPPAGE_SIZE || !(target & CAP_PAGE_WRITE):
|
||||||
return
|
return
|
||||||
Capability *c = &((Capability *)cappage->data.frame)[data[1]]
|
Capability *cap = &((Capability *)cappage->data.frame)[c->data[1]]
|
||||||
c->invalidate ()
|
if cap:
|
||||||
|
cap->invalidate ()
|
||||||
// clone_capability needs a Memory, but doesn't use it when storage is provided.
|
// clone_capability needs a Memory, but doesn't use it when storage is provided.
|
||||||
top_memory.clone_capability (cap, copy, c)
|
top_memory.clone_capability (c->cap[0], c->copy[0], cap)
|
||||||
break
|
break
|
||||||
default:
|
default:
|
||||||
break
|
break
|
||||||
|
|
||||||
static void capability_invoke (unsigned target, unsigned protected_data, Capability *cap, unsigned request, unsigned data):
|
static void capability_invoke (unsigned target, unsigned protected_data, Capability::Context *c):
|
||||||
Capability *capability = (Capability *)protected_data
|
Capability *capability = (Capability *)protected_data
|
||||||
switch request:
|
switch c->data[0]:
|
||||||
case CAP_CAPABILITY_GET:
|
case CAP_CAPABILITY_GET:
|
||||||
reply_cap (capability, true)
|
reply_cap (capability, true)
|
||||||
break
|
break
|
||||||
default:
|
default:
|
||||||
break
|
break
|
||||||
|
|
||||||
static bool kernel_invoke (unsigned target, unsigned protected_data, unsigned d[4], Capability *c[4], bool copy[4], Capability *self):
|
static bool kernel_invoke (unsigned target, unsigned protected_data, Capability::Context *c, Capability *self):
|
||||||
// Kernel calling convention:
|
// Kernel calling convention:
|
||||||
// data[0] is the request.
|
// data[0] is the request.
|
||||||
// cap[0] is the reply capability
|
// cap[0] is the reply capability
|
||||||
// other parameters' meanings depend on the operation.
|
// other parameters' meanings depend on the operation.
|
||||||
if !((1 << d[0]) & target & ~REQUEST_MASK):
|
|
||||||
// You are not allowed to perform this operation.
|
|
||||||
return true
|
|
||||||
if (target & (CAPTYPE_MASK | (1 << CAP_RECEIVER_CALL))) == (CAPTYPE_RECEIVER | (1 << CAP_RECEIVER_CALL)):
|
if (target & (CAPTYPE_MASK | (1 << CAP_RECEIVER_CALL))) == (CAPTYPE_RECEIVER | (1 << CAP_RECEIVER_CALL)):
|
||||||
// This is a call capability.
|
// This is a call capability.
|
||||||
reply_receiver = (Receiver *)protected_data
|
reply_receiver = (Receiver *)protected_data
|
||||||
reply_receiver->protected_only = !(target & (1 << CAP_RECEIVER_CALL_ASYNC))
|
reply_receiver->protected_only = !(target & (1 << CAP_RECEIVER_CALL_ASYNC))
|
||||||
Capability r
|
Capability r
|
||||||
Capability *c0 = c[0]
|
Capability *c0 = c->cap[0]
|
||||||
if ~(unsigned)c0->target & ~KERNEL_MASK:
|
if (unsigned)c0->target & ~KERNEL_MASK:
|
||||||
|
// The call is not to a kernel capability.
|
||||||
fill_cap (&r, protected_data, reply_receiver->reply_protected_data)
|
fill_cap (&r, protected_data, reply_receiver->reply_protected_data)
|
||||||
c[0] = &r
|
c->cap[0] = &r
|
||||||
copy[0] = true
|
c->copy[0] = true
|
||||||
bool ret = kernel_invoke ((unsigned)c0->target, c0->protected_data, d, c, copy, c0)
|
bool ret = c0->target->send_message (c0->protected_data, c)
|
||||||
r.invalidate ()
|
r.invalidate ()
|
||||||
return ret
|
return ret
|
||||||
else:
|
else:
|
||||||
// Kernel call: don't create actual capablities.
|
// Kernel call: don't create actual capablities.
|
||||||
reply = NULL
|
reply = NULL
|
||||||
return kernel_invoke ((unsigned)c0->target, c0->protected_data, d, c, copy, c0)
|
kernel_invoke ((unsigned)c0->target, c0->protected_data, c, c0)
|
||||||
|
return true
|
||||||
if (target & (CAPTYPE_MASK | (1 << CAP_RECEIVER_REPLY))) == (CAPTYPE_RECEIVER | (1 << CAP_RECEIVER_REPLY)):
|
if (target & (CAPTYPE_MASK | (1 << CAP_RECEIVER_REPLY))) == (CAPTYPE_RECEIVER | (1 << CAP_RECEIVER_REPLY)):
|
||||||
// This is a reply capability.
|
// This is a reply capability.
|
||||||
Receiver *r = (Receiver *)protected_data
|
Receiver *r = (Receiver *)protected_data
|
||||||
r->send_message (r->reply_protected_data, d, c, copy)
|
r->send_message (r->reply_protected_data, c)
|
||||||
while self->parent:
|
while self->parent:
|
||||||
self = self->parent
|
self = self->parent
|
||||||
while self->sibling_prev:
|
while self->sibling_prev:
|
||||||
@ -682,36 +721,42 @@ static bool kernel_invoke (unsigned target, unsigned protected_data, unsigned d[
|
|||||||
self->sibling_next->invalidate ()
|
self->sibling_next->invalidate ()
|
||||||
self->invalidate ()
|
self->invalidate ()
|
||||||
return true
|
return true
|
||||||
reply = c[0]
|
// It's a normal kernel capability; check permission.
|
||||||
if d[0] == CAP_DEGRADE:
|
if !((1 << c->data[0]) & target & REQUEST_MASK):
|
||||||
reply_cap (target & d[1], protected_data)
|
// You are not allowed to perform this operation.
|
||||||
|
dbg_send (c->data[0], 5)
|
||||||
|
schedule ()
|
||||||
|
return true
|
||||||
|
if c->data[0] == CAP_DEGRADE:
|
||||||
|
reply_cap (target & (CAPTYPE_MASK | (c->data[1] & REQUEST_MASK)), protected_data)
|
||||||
return true
|
return true
|
||||||
switch target & CAPTYPE_MASK:
|
switch target & CAPTYPE_MASK:
|
||||||
case CAPTYPE_RECEIVER:
|
case CAPTYPE_RECEIVER:
|
||||||
receiver_invoke (target, protected_data, c[1], d)
|
receiver_invoke (target, protected_data, c)
|
||||||
break
|
break
|
||||||
case CAPTYPE_MEMORY:
|
case CAPTYPE_MEMORY:
|
||||||
memory_invoke (target, protected_data, c[1], d[0], d[1])
|
memory_invoke (target, protected_data, c)
|
||||||
break
|
break
|
||||||
case CAPTYPE_THREAD:
|
case CAPTYPE_THREAD:
|
||||||
thread_invoke (target, protected_data, c[1], d)
|
thread_invoke (target, protected_data, c)
|
||||||
break
|
break
|
||||||
case CAPTYPE_PAGE:
|
case CAPTYPE_PAGE:
|
||||||
page_invoke (target, protected_data, c[1], copy[1], d)
|
page_invoke (target, protected_data, c)
|
||||||
break
|
break
|
||||||
case CAPTYPE_CAPABILITY:
|
case CAPTYPE_CAPABILITY:
|
||||||
capability_invoke (target, protected_data, c[1], d[0], d[1])
|
capability_invoke (target, protected_data, c)
|
||||||
break
|
break
|
||||||
case CAPTYPE_CAPPAGE:
|
case CAPTYPE_CAPPAGE:
|
||||||
page_invoke (target, protected_data, c[1], copy[1], d)
|
|
||||||
break
|
break
|
||||||
default:
|
default:
|
||||||
panic (0x99337744, "invalid capability type invoked")
|
panic (0x99337744, "invalid capability type invoked")
|
||||||
return true
|
return true
|
||||||
|
|
||||||
bool Capability::invoke (unsigned data[4], Capability *cap[4], bool copy[4]):
|
bool Capability::invoke (Context *c):
|
||||||
if (unsigned)target & ~KERNEL_MASK:
|
if (unsigned)target & ~KERNEL_MASK:
|
||||||
// This is not a kernel capability: send a message to the receiver.
|
// This is not a kernel capability: send a message to the receiver.
|
||||||
return target->send_message (protected_data, data, cap, copy)
|
return target->send_message (protected_data, c)
|
||||||
// This is a kernel capability. Use a function to allow optimized call capabilities.
|
// This is a kernel capability. Use a function to allow optimized call capabilities.
|
||||||
return kernel_invoke ((unsigned)target, protected_data, data, cap, copy, this)
|
reply = c->cap[0]
|
||||||
|
reply_receiver = NULL
|
||||||
|
return kernel_invoke ((unsigned)target, protected_data, c, this)
|
||||||
|
116
iris.h
116
iris.h
@ -22,6 +22,10 @@
|
|||||||
extern "C" {
|
extern "C" {
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
#define PAGE_BITS (12)
|
||||||
|
#define PAGE_SIZE (1 << PAGE_BITS)
|
||||||
|
#define PAGE_MASK (~(PAGE_SIZE - 1))
|
||||||
|
|
||||||
#define KERNEL_MASK 0xfff
|
#define KERNEL_MASK 0xfff
|
||||||
#define CAPTYPE_MASK 0xe00
|
#define CAPTYPE_MASK 0xe00
|
||||||
#define REQUEST_MASK (KERNEL_MASK & ~CAPTYPE_MASK)
|
#define REQUEST_MASK (KERNEL_MASK & ~CAPTYPE_MASK)
|
||||||
@ -98,12 +102,16 @@ extern "C" {
|
|||||||
/* Flag values for Page and Cappage objects. */
|
/* Flag values for Page and Cappage objects. */
|
||||||
/* A writable page can be written to. This flag can not be set while the frame is shared. */
|
/* A writable page can be written to. This flag can not be set while the frame is shared. */
|
||||||
#define PAGE_FLAG_WRITABLE 1
|
#define PAGE_FLAG_WRITABLE 1
|
||||||
/* When paying, the memory's use is incremented if the page holds a frame. It cannot be lost. Frames are lost when the last payer forgets them. */
|
/* When paying, the memory's use is incremented. If a frame is held, it cannot be lost. Frames are lost when the last payer forgets them. */
|
||||||
#define PAGE_FLAG_PAYING 2
|
#define PAGE_FLAG_PAYING 2
|
||||||
/* Set if this page has a frame associated with it. This flag is automatically reset if the frame is lost because of payment problems. */
|
/* Set if this page has a frame associated with it. This flag is automatically reset if the frame is lost because of payment problems. */
|
||||||
#define PAGE_FLAG_FRAME 4
|
#define PAGE_FLAG_FRAME 4
|
||||||
/* This is a read-only flag, which is set if the Page is shared. */
|
/* This is a read-only flag, which is set if the Page is shared. */
|
||||||
#define PAGE_FLAG_SHARED 8
|
#define PAGE_FLAG_SHARED 8
|
||||||
|
/* This is a read-only flag, saying if this is physical memory, which mustn't be freed. */
|
||||||
|
#define PAGE_FLAG_PHYSICAL 0x10
|
||||||
|
/* This is a read-only flag, saying if this is uncachable memory. */
|
||||||
|
#define PAGE_FLAG_UNCACHED 0x20
|
||||||
|
|
||||||
#define CAP_CAPABILITY_GET 1
|
#define CAP_CAPABILITY_GET 1
|
||||||
#define CAP_CAPABILITY_ALL_RIGHTS 0x1ff
|
#define CAP_CAPABILITY_ALL_RIGHTS 0x1ff
|
||||||
@ -134,42 +142,70 @@ typedef struct Message
|
|||||||
|
|
||||||
static int invoke (Capability target, Message *msg)
|
static int invoke (Capability target, Message *msg)
|
||||||
{
|
{
|
||||||
register int ret __asm__ ("v0");
|
unsigned ret;
|
||||||
register unsigned v0 __asm__ ("v0") = target;
|
__asm__ volatile ("lw $v0, %1\n"
|
||||||
register unsigned a0 __asm__ ("a0") = msg->cap[0];
|
"\tlw $a3, %2\n"
|
||||||
register unsigned a1 __asm__ ("a1") = msg->cap[1];
|
"\tlw $t0, 0($a3)\n"
|
||||||
register unsigned a2 __asm__ ("a2") = msg->cap[2];
|
"\tlw $t1, 4($a3)\n"
|
||||||
register unsigned a3 __asm__ ("a3") = msg->cap[3];
|
"\tlw $t2, 8($a3)\n"
|
||||||
register unsigned t0 __asm__ ("t0") = msg->data[0];
|
"\tlw $t3, 12($a3)\n"
|
||||||
register unsigned t1 __asm__ ("t1") = msg->data[1];
|
"\tlw $a0, 16($a3)\n"
|
||||||
register unsigned t2 __asm__ ("t2") = msg->data[2];
|
"\tlw $a1, 20($a3)\n"
|
||||||
register unsigned t3 __asm__ ("t3") = msg->data[3];
|
"\tlw $a2, 24($a3)\n"
|
||||||
__asm__ volatile ("syscall" : "+r" (v0), "=r" (a0), "=r" (a1), "=r" (a2), "=r" (a3), "=r" (t0), "=r" (t1), "=r" (t2), "=r" (t3));
|
"\tlw $a3, 28($a3)\n"
|
||||||
|
"\tsyscall\n"
|
||||||
|
"\tmove %0, $v0"
|
||||||
|
: "=r"(ret)
|
||||||
|
: "m"(target), "m"(msg)
|
||||||
|
: "v0", "t0", "t1", "t2", "t3", "a0", "a1", "a2", "a3");
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static void wait (Message *msg)
|
||||||
|
{
|
||||||
|
__asm__ volatile ("li $v0, 2\n"
|
||||||
|
"\tsyscall\n"
|
||||||
|
"\tlw $v1, %0\n"
|
||||||
|
"\tsw $t0, 0($v1)\n"
|
||||||
|
"\tsw $t1, 4($v1)\n"
|
||||||
|
"\tsw $t2, 8($v1)\n"
|
||||||
|
"\tsw $t3, 12($v1)\n"
|
||||||
|
"\tsw $a0, 16($v1)\n"
|
||||||
|
"\tsw $a1, 20($v1)\n"
|
||||||
|
"\tsw $a2, 24($v1)\n"
|
||||||
|
"\tsw $a3, 28($v1)"
|
||||||
|
:
|
||||||
|
: "m"(msg)
|
||||||
|
: "memory", "v0", "v1", "t0", "t1", "t2", "t3", "a0", "a1", "a2", "a3");
|
||||||
|
}
|
||||||
|
|
||||||
static int call (Capability target, Message *msg)
|
static int call (Capability target, Message *msg)
|
||||||
{
|
{
|
||||||
register int ret __asm__ ("v0");
|
unsigned ret;
|
||||||
register unsigned v0 __asm__ ("v0") = target;
|
Capability t = cap_copy (target);
|
||||||
register unsigned a0 __asm__ ("a0") = msg->cap[0];
|
__asm__ volatile ("lw $v0, %1\n"
|
||||||
register unsigned a1 __asm__ ("a1") = msg->cap[1];
|
"\tlw $v1, %2\n"
|
||||||
register unsigned a2 __asm__ ("a2") = msg->cap[2];
|
"\tlw $t0, 0($v1)\n"
|
||||||
register unsigned a3 __asm__ ("a3") = msg->cap[3];
|
"\tlw $t1, 4($v1)\n"
|
||||||
register unsigned t0 __asm__ ("t0") = msg->data[0];
|
"\tlw $t2, 8($v1)\n"
|
||||||
register unsigned t1 __asm__ ("t1") = msg->data[1];
|
"\tlw $t3, 12($v1)\n"
|
||||||
register unsigned t2 __asm__ ("t2") = msg->data[2];
|
"\tlw $a0, 16($v1)\n"
|
||||||
register unsigned t3 __asm__ ("t3") = msg->data[3];
|
"\tlw $a1, 20($v1)\n"
|
||||||
__asm__ volatile ("syscall" : "+r" (v0), "+r" (a0), "+r" (a1), "+r" (a2), "+r" (a3), "+r" (t0), "+r" (t1), "+r" (t2), "+r" (t3));
|
"\tlw $a2, 24($v1)\n"
|
||||||
msg->cap[0] = a0;
|
"\tlw $a3, 28($v1)\n"
|
||||||
msg->cap[1] = a1;
|
"\tsyscall\n"
|
||||||
msg->cap[2] = a2;
|
"\tmove %0, $v0\n"
|
||||||
msg->cap[3] = a3;
|
"\tsw $t0, 0($v1)\n"
|
||||||
msg->data[0] = t0;
|
"\tsw $t1, 4($v1)\n"
|
||||||
msg->data[1] = t1;
|
"\tsw $t2, 8($v1)\n"
|
||||||
msg->data[2] = t2;
|
"\tsw $t3, 12($v1)\n"
|
||||||
msg->data[3] = t3;
|
"\tsw $a0, 16($v1)\n"
|
||||||
return ret;
|
"\tsw $a1, 20($v1)\n"
|
||||||
|
"\tsw $a2, 24($v1)\n"
|
||||||
|
"\tsw $a3, 28($v1)"
|
||||||
|
: "=r"(ret)
|
||||||
|
: "m"(t), "m"(msg)
|
||||||
|
: "memory", "v0", "v1", "t0", "t1", "t2", "t3", "a0", "a1", "a2", "a3");
|
||||||
}
|
}
|
||||||
|
|
||||||
static int invoke_01 (Capability t, unsigned d)
|
static int invoke_01 (Capability t, unsigned d)
|
||||||
@ -330,7 +366,7 @@ static void unregister_interrupt (unsigned num)
|
|||||||
|
|
||||||
static void alloc_physical (Capability page, unsigned address, int cachable)
|
static void alloc_physical (Capability page, unsigned address, int cachable)
|
||||||
{
|
{
|
||||||
invoke_12 (__my_thread, page, CAP_THREAD_ALLOC_PHYSICAL, address | (cachable ? 1 : 0));
|
invoke_12 (__my_thread, page, CAP_THREAD_ALLOC_PHYSICAL, (address & PAGE_MASK) | (cachable ? 1 : 0));
|
||||||
}
|
}
|
||||||
|
|
||||||
static int receiver_set_owner (Capability receiver, Capability owner)
|
static int receiver_set_owner (Capability receiver, Capability owner)
|
||||||
@ -370,27 +406,27 @@ static Capability memory_create (Capability memory, unsigned type)
|
|||||||
|
|
||||||
static Capability memory_create_page (Capability memory)
|
static Capability memory_create_page (Capability memory)
|
||||||
{
|
{
|
||||||
return memory_create (memory, CAPTYPE_PAGE);
|
return memory_create (memory, CAPTYPE_PAGE | REQUEST_MASK);
|
||||||
}
|
}
|
||||||
|
|
||||||
static Capability memory_create_thread (Capability memory)
|
static Capability memory_create_thread (Capability memory)
|
||||||
{
|
{
|
||||||
return memory_create (memory, CAPTYPE_THREAD);
|
return memory_create (memory, CAPTYPE_THREAD | REQUEST_MASK);
|
||||||
}
|
}
|
||||||
|
|
||||||
static Capability memory_create_receiver (Capability memory)
|
static Capability memory_create_receiver (Capability memory)
|
||||||
{
|
{
|
||||||
return memory_create (memory, CAPTYPE_RECEIVER);
|
return memory_create (memory, CAPTYPE_RECEIVER | REQUEST_MASK);
|
||||||
}
|
}
|
||||||
|
|
||||||
static Capability memory_create_memory (Capability memory)
|
static Capability memory_create_memory (Capability memory)
|
||||||
{
|
{
|
||||||
return memory_create (memory, CAPTYPE_MEMORY);
|
return memory_create (memory, CAPTYPE_MEMORY | REQUEST_MASK);
|
||||||
}
|
}
|
||||||
|
|
||||||
static Capability memory_create_cappage (Capability memory)
|
static Capability memory_create_cappage (Capability memory)
|
||||||
{
|
{
|
||||||
return memory_create (memory, CAPTYPE_CAPPAGE);
|
return memory_create (memory, CAPTYPE_CAPPAGE | REQUEST_MASK);
|
||||||
}
|
}
|
||||||
|
|
||||||
static int memory_destroy (Capability memory, Capability target)
|
static int memory_destroy (Capability memory, Capability target)
|
||||||
@ -400,8 +436,10 @@ static int memory_destroy (Capability memory, Capability target)
|
|||||||
|
|
||||||
/* TODO: #define CAP_MEMORY_LIST 3 */
|
/* TODO: #define CAP_MEMORY_LIST 3 */
|
||||||
|
|
||||||
static int memory_map (Capability memory, Capability page, unsigned address)
|
static int memory_map (Capability memory, Capability page, unsigned address, int writable)
|
||||||
{
|
{
|
||||||
|
if (writable)
|
||||||
|
address |= 1 << CAP_PAGE_WRITE;
|
||||||
return invoke_12 (memory, page, CAP_MEMORY_MAP, address);
|
return invoke_12 (memory, page, CAP_MEMORY_MAP, address);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
26
kernel.hhp
26
kernel.hhp
@ -83,6 +83,19 @@ struct Message : public Object <Message>:
|
|||||||
unsigned data[4]
|
unsigned data[4]
|
||||||
unsigned protected_data
|
unsigned protected_data
|
||||||
|
|
||||||
|
struct Capability : public Object <Capability>:
|
||||||
|
struct Context:
|
||||||
|
unsigned data[4]
|
||||||
|
Capability *cap[4]
|
||||||
|
bool copy[4]
|
||||||
|
Receiver *target
|
||||||
|
Capability *parent
|
||||||
|
Capability *children
|
||||||
|
Capability *sibling_prev, *sibling_next
|
||||||
|
unsigned protected_data
|
||||||
|
bool invoke (Context *c)
|
||||||
|
void invalidate ()
|
||||||
|
|
||||||
struct Receiver : public Object <Receiver>:
|
struct Receiver : public Object <Receiver>:
|
||||||
Thread *owner
|
Thread *owner
|
||||||
Receiver *prev_owned, *next_owned
|
Receiver *prev_owned, *next_owned
|
||||||
@ -93,16 +106,7 @@ struct Receiver : public Object <Receiver>:
|
|||||||
void own (Thread *o)
|
void own (Thread *o)
|
||||||
void orphan ()
|
void orphan ()
|
||||||
bool try_deliver ()
|
bool try_deliver ()
|
||||||
bool send_message (unsigned protected_data, unsigned data[4], Capability *cap[4], bool copy[4])
|
bool send_message (unsigned protected_data, Capability::Context *c)
|
||||||
|
|
||||||
struct Capability : public Object <Capability>:
|
|
||||||
Receiver *target
|
|
||||||
Capability *parent
|
|
||||||
Capability *children
|
|
||||||
Capability *sibling_prev, *sibling_next
|
|
||||||
unsigned protected_data
|
|
||||||
bool invoke (unsigned data[4], Capability *cap[4], bool copy[4])
|
|
||||||
void invalidate ()
|
|
||||||
|
|
||||||
struct ShareData :
|
struct ShareData :
|
||||||
unsigned frame
|
unsigned frame
|
||||||
@ -194,7 +198,7 @@ void raw_pfree (unsigned page)
|
|||||||
|
|
||||||
// Defined by architecture-specific files.
|
// Defined by architecture-specific files.
|
||||||
void Thread_arch_init (Thread *thread)
|
void Thread_arch_init (Thread *thread)
|
||||||
void Thread_arch_receive (Thread *thread, unsigned d[4], Capability *c[4])
|
void Thread_arch_receive (Thread *thread, Capability::Context *c)
|
||||||
void Thread_arch_receive_fail (Thread *thread)
|
void Thread_arch_receive_fail (Thread *thread)
|
||||||
unsigned *Thread_arch_info (Thread *thread, unsigned num)
|
unsigned *Thread_arch_info (Thread *thread, unsigned num)
|
||||||
void Memory_arch_init (Memory *mem)
|
void Memory_arch_init (Memory *mem)
|
||||||
|
@ -17,7 +17,7 @@
|
|||||||
|
|
||||||
load = 0x80000000
|
load = 0x80000000
|
||||||
|
|
||||||
ARCH_CXXFLAGS = -DNUM_THREADS=2
|
ARCH_CXXFLAGS = -DNUM_THREADS=1
|
||||||
ARCH_CPPFLAGS = -Imips -Wa,-mips32
|
ARCH_CPPFLAGS = -Imips -Wa,-mips32
|
||||||
CROSS = mipsel-linux-gnu-
|
CROSS = mipsel-linux-gnu-
|
||||||
OBJDUMP = $(CROSS)objdump
|
OBJDUMP = $(CROSS)objdump
|
||||||
@ -40,9 +40,6 @@ $(boot_threads): TARGET_FLAGS = -I.
|
|||||||
uimage: kernel.raw.gz Makefile mips/Makefile.arch
|
uimage: kernel.raw.gz Makefile mips/Makefile.arch
|
||||||
mkimage -A MIPS -O Linux -C gzip -a $(load) -e 0x$(shell /bin/sh -c '$(OBJDUMP) -t kernel | grep __start$$ | cut -b-8') -n "Shevek's kernel" -d $< $@ | sed -e 's/:/;/g'
|
mkimage -A MIPS -O Linux -C gzip -a $(load) -e 0x$(shell /bin/sh -c '$(OBJDUMP) -t kernel | grep __start$$ | cut -b-8') -n "Shevek's kernel" -d $< $@ | sed -e 's/:/;/g'
|
||||||
|
|
||||||
elf.h: /usr/include/elf.h
|
|
||||||
ln -s $< $@
|
|
||||||
|
|
||||||
%.o:%.S Makefile mips/Makefile.arch mips/arch.hh
|
%.o:%.S Makefile mips/Makefile.arch mips/arch.hh
|
||||||
$(CC) $(CPPFLAGS) $(TARGET_FLAGS) -DKERNEL_STACK_SIZE=0x2000 -c $< -o $@
|
$(CC) $(CPPFLAGS) $(TARGET_FLAGS) -DKERNEL_STACK_SIZE=0x2000 -c $< -o $@
|
||||||
|
|
||||||
@ -56,4 +53,4 @@ kernel: mips/entry.o $(subst .cc,.o,$(kernel_sources)) mips/boot.o $(subst .cc,.
|
|||||||
%.gz: %
|
%.gz: %
|
||||||
gzip < $< > $@
|
gzip < $< > $@
|
||||||
|
|
||||||
ARCH_CLEAN_FILES = uimage kernel kernel.raw kernel.raw.gz elf.h $(boot_threads) mips/*.o
|
ARCH_CLEAN_FILES = uimage kernel kernel.raw kernel.raw.gz $(boot_threads) mips/*.o
|
||||||
|
@ -45,15 +45,15 @@ void Thread_arch_init (Thread *thread):
|
|||||||
thread->arch.k0 = 0
|
thread->arch.k0 = 0
|
||||||
thread->arch.k1 = 0
|
thread->arch.k1 = 0
|
||||||
|
|
||||||
void Thread_arch_receive (Thread *thread, unsigned d[4], Capability *c[4]):
|
void Thread_arch_receive (Thread *thread, Capability::Context *c):
|
||||||
thread->arch.a0 = (unsigned)c[0]
|
thread->arch.a0 = (unsigned)c->cap[0]
|
||||||
thread->arch.a1 = (unsigned)c[1]
|
thread->arch.a1 = (unsigned)c->cap[1]
|
||||||
thread->arch.a2 = (unsigned)c[2]
|
thread->arch.a2 = (unsigned)c->cap[2]
|
||||||
thread->arch.a3 = (unsigned)c[3]
|
thread->arch.a3 = (unsigned)c->cap[3]
|
||||||
thread->arch.t0 = d[0]
|
thread->arch.t0 = c->data[0]
|
||||||
thread->arch.t1 = d[1]
|
thread->arch.t1 = c->data[1]
|
||||||
thread->arch.t2 = d[2]
|
thread->arch.t2 = c->data[2]
|
||||||
thread->arch.t3 = d[3]
|
thread->arch.t3 = c->data[3]
|
||||||
thread->arch.v0 = 1
|
thread->arch.v0 = 1
|
||||||
|
|
||||||
void Thread_arch_receive_fail (Thread *thread):
|
void Thread_arch_receive_fail (Thread *thread):
|
||||||
@ -222,10 +222,12 @@ static unsigned make_entry_lo (Page *page, bool write):
|
|||||||
if !page->data.frame:
|
if !page->data.frame:
|
||||||
return 0
|
return 0
|
||||||
unsigned flags
|
unsigned flags
|
||||||
if write:
|
if page->data.flags & PAGE_FLAG_UNCACHED:
|
||||||
flags = 0x18 | 0x4 | 0x2
|
flags = 0x10 | 0x2
|
||||||
else
|
else:
|
||||||
flags = 0x18 | 0x2
|
flags = 0x18 | 0x2
|
||||||
|
if write:
|
||||||
|
flags |= 0x4
|
||||||
return ((page->data.frame & ~0x80000000) >> 6) | flags
|
return ((page->data.frame & ~0x80000000) >> 6) | flags
|
||||||
|
|
||||||
bool Memory_arch_map (Memory *mem, Page *page, unsigned address, bool write):
|
bool Memory_arch_map (Memory *mem, Page *page, unsigned address, bool write):
|
||||||
@ -320,30 +322,36 @@ void Page_arch_update_mapping (Page *page):
|
|||||||
tlb_reset (p->mapping & ~1, as->arch.asid, t)
|
tlb_reset (p->mapping & ~1, as->arch.asid, t)
|
||||||
|
|
||||||
void arch_invoke ():
|
void arch_invoke ():
|
||||||
Capability *target, *c[4]
|
Capability *target
|
||||||
bool wait, copy[4]
|
bool wait
|
||||||
Thread *caller = current
|
Thread *caller = current
|
||||||
target = caller->address_space->find_capability (current->arch.v0, &wait)
|
target = caller->address_space->find_capability (current->arch.v0, &wait)
|
||||||
if !target:
|
if !target:
|
||||||
// TODO: there must be no action here. This is just because the rest doesn't work yet.
|
// TODO: there must be no action here. This is just because the rest doesn't work yet.
|
||||||
dbg_led (caller->arch.a0, caller->arch.a1, caller->arch.a2)
|
dbg_send (3, 2)
|
||||||
dbg_sleep (1000)
|
|
||||||
schedule ()
|
schedule ()
|
||||||
// Calling an invalid capability always fails.
|
// Calling an invalid capability always fails.
|
||||||
caller->arch.v0 = 0
|
caller->arch.v0 = 0
|
||||||
else:
|
else:
|
||||||
if wait:
|
if wait:
|
||||||
caller->wait ()
|
caller->wait ()
|
||||||
c[0] = caller->address_space->find_capability (caller->arch.a0, ©[0])
|
Capability::Context c
|
||||||
c[1] = caller->address_space->find_capability (caller->arch.a1, ©[1])
|
c.cap[0] = caller->address_space->find_capability (caller->arch.a0, &c.copy[0])
|
||||||
c[2] = caller->address_space->find_capability (caller->arch.a2, ©[2])
|
c.cap[1] = caller->address_space->find_capability (caller->arch.a1, &c.copy[1])
|
||||||
c[3] = caller->address_space->find_capability (caller->arch.a3, ©[3])
|
c.cap[2] = caller->address_space->find_capability (caller->arch.a2, &c.copy[2])
|
||||||
unsigned d[4]
|
c.cap[3] = caller->address_space->find_capability (caller->arch.a3, &c.copy[3])
|
||||||
d[0] = caller->arch.t0
|
c.data[0] = caller->arch.t0
|
||||||
d[1] = caller->arch.t1
|
c.data[1] = caller->arch.t1
|
||||||
d[2] = caller->arch.t2
|
c.data[2] = caller->arch.t2
|
||||||
d[3] = caller->arch.t3
|
c.data[3] = caller->arch.t3
|
||||||
caller->arch.v0 = target->invoke (d, c, copy) ? 1 : 0
|
caller->arch.v0 = target->invoke (&c) ? 1 : 0
|
||||||
|
if !current:
|
||||||
|
if caller != &idle && (caller->flags & (THREAD_FLAG_RUNNING | THREAD_FLAG_WAITING)) == THREAD_FLAG_RUNNING:
|
||||||
|
current = caller
|
||||||
|
else:
|
||||||
|
schedule ()
|
||||||
|
if !current:
|
||||||
|
current = &idle
|
||||||
if caller != current:
|
if caller != current:
|
||||||
if (Memory *)asids[current->address_space->arch.asid] != current->address_space:
|
if (Memory *)asids[current->address_space->arch.asid] != current->address_space:
|
||||||
if asids[0]:
|
if asids[0]:
|
||||||
|
@ -61,10 +61,6 @@
|
|||||||
#define CP0_DESAVE 31
|
#define CP0_DESAVE 31
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
#define PAGE_BITS (12)
|
|
||||||
#define PAGE_SIZE (1 << PAGE_BITS)
|
|
||||||
#define PAGE_MASK (~(PAGE_SIZE - 1))
|
|
||||||
|
|
||||||
// register save positions in Thread
|
// register save positions in Thread
|
||||||
#define SAVE_PC (5 * 4)
|
#define SAVE_PC (5 * 4)
|
||||||
#define SAVE_SP (SAVE_PC + 4)
|
#define SAVE_SP (SAVE_PC + 4)
|
||||||
|
@ -80,7 +80,7 @@ addr_100:
|
|||||||
addr_180:
|
addr_180:
|
||||||
// General exception
|
// General exception
|
||||||
// Allow new exceptions to update EPC and friends.
|
// Allow new exceptions to update EPC and friends.
|
||||||
//mtc0 $zero, $CP0_STATUS
|
mtc0 $zero, $CP0_STATUS
|
||||||
sw $ra, -0xd88($zero)
|
sw $ra, -0xd88($zero)
|
||||||
bal save_regs
|
bal save_regs
|
||||||
nop
|
nop
|
||||||
|
@ -178,10 +178,12 @@ static void init_threads ():
|
|||||||
if !stackpage || !mem->map (stackpage, 0x7ffff000, true):
|
if !stackpage || !mem->map (stackpage, 0x7ffff000, true):
|
||||||
panic (0x13151719, "unable to map initial stack page")
|
panic (0x13151719, "unable to map initial stack page")
|
||||||
Receiver *recv = mem->alloc_receiver ()
|
Receiver *recv = mem->alloc_receiver ()
|
||||||
|
recv->owner = thread
|
||||||
|
thread->receivers = recv
|
||||||
thread->arch.a0 = mkcap (mem, CAPTYPE_RECEIVER | CAP_RECEIVER_ALL_RIGHTS, recv)
|
thread->arch.a0 = mkcap (mem, CAPTYPE_RECEIVER | CAP_RECEIVER_ALL_RIGHTS, recv)
|
||||||
thread->arch.a1 = mkcap (mem, CAPTYPE_THREAD | CAP_THREAD_ALL_PRIV_RIGHTS, thread)
|
thread->arch.a1 = mkcap (mem, CAPTYPE_THREAD | CAP_THREAD_ALL_PRIV_RIGHTS, thread)
|
||||||
thread->arch.a2 = mkcap (mem, CAPTYPE_MEMORY | CAP_MEMORY_ALL_RIGHTS, mem)
|
thread->arch.a2 = mkcap (mem, CAPTYPE_MEMORY | CAP_MEMORY_ALL_RIGHTS, mem)
|
||||||
thread->arch.a3 = mkcap (mem, CAPTYPE_RECEIVER | CAP_RECEIVER_CALL, recv)
|
thread->arch.a3 = mkcap (mem, CAPTYPE_RECEIVER | (1 << CAP_RECEIVER_CALL), recv)
|
||||||
mem->pfree ((unsigned)pages)
|
mem->pfree ((unsigned)pages)
|
||||||
thread->flags = THREAD_FLAG_RUNNING | THREAD_FLAG_PRIV
|
thread->flags = THREAD_FLAG_RUNNING | THREAD_FLAG_PRIV
|
||||||
thread->schedule_next = NULL
|
thread->schedule_next = NULL
|
||||||
|
@ -30,6 +30,11 @@ Thread *tlb_refill ():
|
|||||||
cp0_get (CP0_ENTRY_HI, EntryHi)
|
cp0_get (CP0_ENTRY_HI, EntryHi)
|
||||||
unsigned *t = directory[EntryHi >> 21]
|
unsigned *t = directory[EntryHi >> 21]
|
||||||
if !t:
|
if !t:
|
||||||
|
unsigned a
|
||||||
|
cp0_get (CP0_EPC, a)
|
||||||
|
dbg_send (a)
|
||||||
|
cp0_get (CP0_BAD_V_ADDR, a)
|
||||||
|
dbg_send (a)
|
||||||
panic (0x99992222, "No page table")
|
panic (0x99992222, "No page table")
|
||||||
// - 2 instead of - 1 means reset bit 0
|
// - 2 instead of - 1 means reset bit 0
|
||||||
unsigned idx = (EntryHi >> 12) & ((1 << 9) - 2)
|
unsigned idx = (EntryHi >> 12) & ((1 << 9) - 2)
|
||||||
@ -50,17 +55,18 @@ Thread *interrupt ():
|
|||||||
status &= ~(1 << (i + 8))
|
status &= ~(1 << (i + 8))
|
||||||
// Send message to interrupt handler.
|
// Send message to interrupt handler.
|
||||||
if arch_interrupt_receiver[i]:
|
if arch_interrupt_receiver[i]:
|
||||||
unsigned data[4] = {0, 0, 0, 0}
|
Capability::Context c
|
||||||
Capability *cap[4] = {NULL, NULL, NULL, NULL}
|
for unsigned j = 0; j < 4; ++j:
|
||||||
bool copy[4] = {false, false, false, false}
|
c.data[j] = 0
|
||||||
arch_interrupt_receiver[i]->send_message (i, data, cap, copy)
|
c.cap[j] = NULL
|
||||||
|
c.copy[j] = false
|
||||||
|
arch_interrupt_receiver[i]->send_message (i, &c)
|
||||||
return current
|
return current
|
||||||
|
|
||||||
/// A general exception has occurred.
|
/// A general exception has occurred.
|
||||||
Thread *exception ():
|
Thread *exception ():
|
||||||
unsigned cause
|
unsigned cause
|
||||||
cp0_get (CP0_CAUSE, cause)
|
cp0_get (CP0_CAUSE, cause)
|
||||||
//dbg_send (cause >> 2, 5)
|
|
||||||
switch (cause >> 2) & 0x1f:
|
switch (cause >> 2) & 0x1f:
|
||||||
case 0:
|
case 0:
|
||||||
// Interrupt. This shouldn't happen, since CAUSE[IV] == 1.
|
// Interrupt. This shouldn't happen, since CAUSE[IV] == 1.
|
||||||
@ -70,6 +76,9 @@ Thread *exception ():
|
|||||||
panic (0x21223344, "TLB modification.")
|
panic (0x21223344, "TLB modification.")
|
||||||
case 2:
|
case 2:
|
||||||
// TLB load or instruction fetch.
|
// TLB load or instruction fetch.
|
||||||
|
unsigned a
|
||||||
|
cp0_get (CP0_EPC, a)
|
||||||
|
dbg_send (a)
|
||||||
panic (0x31223344, "TLB load or instruction fetch.")
|
panic (0x31223344, "TLB load or instruction fetch.")
|
||||||
case 3:
|
case 3:
|
||||||
// TLB store.
|
// TLB store.
|
||||||
@ -79,6 +88,9 @@ Thread *exception ():
|
|||||||
panic (0x51223344, "Address error load or instruction fetch.")
|
panic (0x51223344, "Address error load or instruction fetch.")
|
||||||
case 5:
|
case 5:
|
||||||
// Address error store.
|
// Address error store.
|
||||||
|
unsigned a
|
||||||
|
cp0_get (CP0_EPC, a)
|
||||||
|
dbg_send (a, 16)
|
||||||
panic (0x61223344, "Address error store.")
|
panic (0x61223344, "Address error store.")
|
||||||
case 6:
|
case 6:
|
||||||
// Bus error instruction fetch.
|
// Bus error instruction fetch.
|
||||||
@ -88,6 +100,7 @@ Thread *exception ():
|
|||||||
panic (0x81223344, "Bus error load or store.")
|
panic (0x81223344, "Bus error load or store.")
|
||||||
case 8:
|
case 8:
|
||||||
// Syscall.
|
// Syscall.
|
||||||
|
current->pc += 4
|
||||||
arch_invoke ()
|
arch_invoke ()
|
||||||
break
|
break
|
||||||
case 9:
|
case 9:
|
||||||
|
40
schedule.ccp
40
schedule.ccp
@ -18,36 +18,43 @@
|
|||||||
|
|
||||||
#include "kernel.hh"
|
#include "kernel.hh"
|
||||||
|
|
||||||
|
static void run_thread (Thread *thread):
|
||||||
|
thread->schedule_next = first_scheduled
|
||||||
|
if thread->schedule_next:
|
||||||
|
thread->schedule_next->schedule_prev = thread
|
||||||
|
first_scheduled = thread
|
||||||
|
|
||||||
|
static void unrun_thread (Thread *thread):
|
||||||
|
if current == thread:
|
||||||
|
current = thread->schedule_next
|
||||||
|
if thread->schedule_prev:
|
||||||
|
thread->schedule_prev->schedule_next = thread->schedule_next
|
||||||
|
else:
|
||||||
|
first_scheduled = thread->schedule_next
|
||||||
|
if thread->schedule_next:
|
||||||
|
thread->schedule_next->schedule_prev = thread->schedule_prev
|
||||||
|
|
||||||
void Thread::run ():
|
void Thread::run ():
|
||||||
if flags & THREAD_FLAG_RUNNING:
|
if flags & THREAD_FLAG_RUNNING:
|
||||||
return
|
return
|
||||||
flags |= THREAD_FLAG_RUNNING
|
flags |= THREAD_FLAG_RUNNING
|
||||||
if flags & THREAD_FLAG_WAITING:
|
if flags & THREAD_FLAG_WAITING:
|
||||||
return
|
return
|
||||||
schedule_next = first_scheduled
|
run_thread (this)
|
||||||
if schedule_next:
|
|
||||||
schedule_next->schedule_prev = this
|
|
||||||
first_scheduled = this
|
|
||||||
|
|
||||||
void Thread::unrun ():
|
void Thread::unrun ():
|
||||||
if !(flags & THREAD_FLAG_RUNNING):
|
if !(flags & THREAD_FLAG_RUNNING):
|
||||||
return
|
return
|
||||||
flags &= ~THREAD_FLAG_RUNNING
|
flags &= ~THREAD_FLAG_RUNNING
|
||||||
if !(flags & THREAD_FLAG_WAITING):
|
if flags & THREAD_FLAG_WAITING:
|
||||||
if current == this:
|
return
|
||||||
current = schedule_next
|
unrun_thread (this)
|
||||||
if schedule_prev:
|
|
||||||
schedule_prev->schedule_next = schedule_next
|
|
||||||
else:
|
|
||||||
first_scheduled = schedule_next
|
|
||||||
if schedule_next:
|
|
||||||
schedule_next->schedule_prev = schedule_prev
|
|
||||||
|
|
||||||
void Thread::wait ():
|
void Thread::wait ():
|
||||||
if flags & THREAD_FLAG_WAITING:
|
if flags & THREAD_FLAG_WAITING:
|
||||||
return
|
return
|
||||||
if flags & THREAD_FLAG_RUNNING:
|
if flags & THREAD_FLAG_RUNNING:
|
||||||
unrun ()
|
unrun_thread (this)
|
||||||
flags |= THREAD_FLAG_WAITING
|
flags |= THREAD_FLAG_WAITING
|
||||||
// Try to receive a message from a Receiver immediately.
|
// Try to receive a message from a Receiver immediately.
|
||||||
for Receiver *r = receivers; r; r = r->next_owned:
|
for Receiver *r = receivers; r; r = r->next_owned:
|
||||||
@ -59,8 +66,7 @@ void Thread::unwait ():
|
|||||||
return
|
return
|
||||||
flags &= ~THREAD_FLAG_WAITING
|
flags &= ~THREAD_FLAG_WAITING
|
||||||
if flags & THREAD_FLAG_RUNNING:
|
if flags & THREAD_FLAG_RUNNING:
|
||||||
flags &= ~THREAD_FLAG_RUNNING
|
run_thread (this)
|
||||||
run ()
|
|
||||||
|
|
||||||
void schedule ():
|
void schedule ():
|
||||||
Thread *old = current
|
Thread *old = current
|
||||||
@ -68,5 +74,3 @@ void schedule ():
|
|||||||
current = current->schedule_next
|
current = current->schedule_next
|
||||||
if !current:
|
if !current:
|
||||||
current = first_scheduled
|
current = first_scheduled
|
||||||
if !current:
|
|
||||||
current = &idle
|
|
||||||
|
Loading…
x
Reference in New Issue
Block a user