-
Notifications
You must be signed in to change notification settings - Fork 17
/
Copy pathmmu.c
111 lines (93 loc) · 2.94 KB
/
mmu.c
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
#include <stdlib.h>
#include "machine.h"
#define SMR_PAGESIZE 4096
#define SMR_SLOTS 16
/* Small MMU register map */
#define SMR_SLOT 0 /* Which slot is described by registers 8-9 */
#define SMR_BASEL 1 /* The base page for lower 8 slots */
#define SMR_BASEH 2 /* The base page for upper 8 slots */
#define SMR_FAULTA 3 /* The faulting address */
#define SMR_FAULTT 5 /* The fault type */
#define SM_GLOBAL_REGS 6
#define SMR_PAGE 6 /* Which 4KB page is mapped to the current slot */
#define SMR_FLAGS 7 /* What are the page flags for this slot */
/* The 'small' MMU is an I/O device that allows remapping a window of
a single device into a fixed region of the CPU. */
struct small_mmu
{
struct hw_device *realdev;
unsigned int page_size;
U8 global_regs[8];
U8 slot_regs[SMR_SLOTS][2];
};
void small_mmu_update_slot (struct small_mmu *mmu, unsigned int slot)
{
unsigned int page = mmu->slot_regs[slot][SMR_PAGE - SM_GLOBAL_REGS];
unsigned int flags = mmu->slot_regs[slot][SMR_FLAGS - SM_GLOBAL_REGS];
bus_map (slot * SMR_PAGESIZE, mmu->realdev->devid, page, SMR_PAGESIZE,
flags & MAP_READWRITE);
}
void small_mmu_update_current (struct small_mmu *mmu)
{
small_mmu_update_slot (mmu, mmu->global_regs[0]);
}
void small_mmu_update_all (struct small_mmu *mmu)
{
unsigned int slot;
for (slot = 0; slot < SMR_SLOTS; slot++)
small_mmu_update_slot (mmu, slot);
}
U8 small_mmu_read (struct hw_device *dev, unsigned long addr)
{
struct small_mmu *mmu = (struct small_mmu *)dev->priv;
if (addr < SM_GLOBAL_REGS)
return mmu->global_regs[addr];
else
return mmu->slot_regs[mmu->global_regs[0]][addr - SM_GLOBAL_REGS];
}
void small_mmu_write (struct hw_device *dev, unsigned long addr, U8 val)
{
struct small_mmu *mmu = (struct small_mmu *)dev->priv;
if (addr < SM_GLOBAL_REGS)
mmu->global_regs[addr] = val;
else
mmu->slot_regs[mmu->global_regs[0]][addr - SM_GLOBAL_REGS] = val;
switch (addr)
{
case SMR_PAGE:
case SMR_FLAGS:
{
unsigned int slot = mmu->global_regs[0];
unsigned int page = mmu->slot_regs[slot][SMR_PAGE - SM_GLOBAL_REGS];
unsigned int flags = mmu->slot_regs[slot][SMR_FLAGS - SM_GLOBAL_REGS];
bus_map (slot * SMR_PAGESIZE, mmu->realdev->devid, page * SMR_PAGESIZE,
SMR_PAGESIZE, flags & MAP_READWRITE);
break;
}
}
}
void small_mmu_reset (struct hw_device *dev)
{
unsigned int page;
struct small_mmu *mmu = (struct small_mmu *)dev->priv;
for (page = 0; page < SMR_SLOTS; page++)
{
small_mmu_write (dev, SMR_SLOT, page);
small_mmu_write (dev, SMR_PAGE, page);
small_mmu_write (dev, SMR_FLAGS, MAP_READWRITE);
}
}
struct hw_class small_mmu_class =
{
.readonly = 0,
.reset = small_mmu_reset,
.read = small_mmu_read,
.write = small_mmu_write,
};
struct hw_device *small_mmu_create (struct hw_device *realdev)
{
struct small_mmu *mmu = malloc (sizeof (struct small_mmu));
mmu->realdev = realdev;
mmu->page_size = SMR_PAGESIZE;
return device_attach (&small_mmu_class, 16, mmu); /* 16 = sizeof I/O window */
}