8
|
1 \chapter{CbC インターフェース}
|
|
2
|
|
3 構造図書く(今のcbcxv6と同じか確認してから)
|
|
4
|
|
5
|
|
6 Gears OS では Meta Code Gear で Context から値を取り出し、ノーマルレベルの Code Gear に値を渡す。
|
|
7 しかし、Code Gaer がどの Data Gear の番号に対応するかを指定する必要があったり、 % ぱるすさんコード必要?
|
|
8 ノーマルレベルとメタレベルで見え方が異なる Data Gear を Meta Code Gear によって 調整する必要があったりと、 % みつきさん
|
|
9 メタレベルからノーマルレベルの継続の記述が煩雑になるため、Interface 化をしている。
|
|
10 Interface は Data Gear に対しての操作を行う Code Gear であり、実装は別で定義する。
|
|
11 % Interface で定義した Code Gear に
|
|
12
|
|
13
|
|
14 % Xv6 の書き換えは Interface を用いてモジュール化する。
|
|
15 そうすることで Gears OS の機能を置き換えることできるようになる。
|
|
16
|
|
17
|
|
18
|
|
19 \section{インターフェースの定義}
|
|
20 インターフェースはある Data Gear の定義と、
|
|
21 それに対する操作を行う Code Gear の集合を表現する Meta Data Gear である。
|
|
22 Context では全ての Code Gaer と Data Gear の集合を表現していることに対し、
|
|
23 インターフェースは一部の Code Gear と一部の Data Gear の集合を表現する。
|
13
|
24
|
|
25
|
8
|
26 インターフェースを記述することによってノーマルレベルとメタレベルの分離が可能となる。
|
|
27
|
12
|
28 Paging のインターフェースを記述したコードを ソースコード \ref{interface} に示す。
|
8
|
29
|
13
|
30 \lstinputlisting[label=interface, caption={\footnotesize vm のインターフェースの定義(vm.h)}]{./src/vm.h}
|
8
|
31
|
12
|
32 1行目ので実装名を定義している。
|
|
33 typedef struct の直後に実装名(vm)を書く。
|
13
|
34
|
|
35
|
8
|
36 2行目から19行目で引数の Data Gear 郡を定義している。
|
12
|
37 初期化された Data Gear がそれぞれの Code Gear の引数として扱われる。
|
8
|
38 例として、2行目で定義された vm が21行目から32行目までの引数と対応している。
|
13
|
39
|
|
40
|
8
|
41 % インターフェースの Code Gear の goto による継続先は基本的に不定となっており、継続元から渡される。
|
12
|
42 Code Gear は\_\_code CodeGearName () で記述する。
|
|
43 第一引数である Impl* vm が Code Gear の型になる。
|
13
|
44
|
|
45
|
8
|
46 \_\_code next(...) の引数 ... は複数の Input Data Gear を持つという意味である。
|
|
47 後述する実装によって条件分岐によって複数の継続先が設定されることがある。
|
13
|
48
|
|
49
|
8
|
50 Code Gaer は 20行目から33行目のように "\_\_code [Code Gear名]([引数])"で定義する。
|
|
51 この引数が input Data Gear になる。
|
|
52
|
|
53
|
|
54 % 実装側に書く
|
|
55 % 引数の Data Gear はその Code Gear の Input Data Gear になり、引数の Code Gear の中の引 数が Output Data Gear になる。Code Gear の第一引数には Interface を実装した Data Gear を渡す。これは、Code Gear の操作の対象となる Data Gear を設定し ており、後述する継続構文では引数として記述を行わない。
|
|
56
|
|
57
|
|
58 \section{インターフェースの実装}
|
|
59 インターフェースは Data Gear に対しての Code Gear とその Code Gear で扱われている Data Gear の集合を抽象化した Meta Data Gear で、vm.c に対応する実装は別で定義する。
|
13
|
60
|
|
61
|
12
|
62 インターフェースの実装についてソースコード \ref{impl_vm} で示す。
|
8
|
63
|
|
64
|
13
|
65 \lstinputlisting[frame=lrbt,label=impl_vm,caption={\footnotesize vm インターフェースの実装}]{./src/vm_impl.cbc}
|
8
|
66
|
13
|
67 2行目のようにインターフェースのヘッダーファイルは \#interface で呼び出す。
|
8
|
68
|
12
|
69
|
|
70 create\_impl の関数内で、インターフェースを vm で定義し、23行目の vm-$>$void\_ret のようにそれぞれのインターフェースに対応させていく。
|
13
|
71
|
|
72
|
12
|
73 CbCは1つ1つの関数の信頼性を保障させるために細かくする必要があるので、for文やif文がある場合はさらに実装を分ける。vm と同じように vm\_impl を定義し、遷移する関数名に対応させていく。分けた実装はさらに別で実装する(vm\_impl\_private.cbc)。
|
|
74
|
|
75
|
|
76 % インターフェースを呼び出す場合は、struct vm* vm = createvm_impl(cbc_context);
|
13
|
77
|
|
78
|
12
|
79
|
|
80
|
|
81
|
|
82
|
|
83
|
|
84
|
8
|
85
|
|
86
|
|
87 \section{インターフェース内の private メソッド}
|
|
88 インターフェースで定義した Code Gear 以外の Code Gaer も記述することができる。
|
|
89 この Code Gear は基本的にインターフェースで指定された Code Gear 内からのみ継続されるため、
|
|
90 Java の private メソッドのように扱われる。
|
13
|
91
|
|
92
|
12
|
93 インターフェースと同じようにヘッダーファイルをソースコード \ref{impl_vm_privateh} で定義する。
|
8
|
94
|
|
95
|
13
|
96 \lstinputlisting[frame=lrbt,label=impl_vm_privateh,caption={\footnotesize vm private のヘッダーファイル}]{./src/vm_impl_private.h}
|
12
|
97
|
11
|
98
|
12
|
99 private での CbC の記述を vm.c と比べて説明する。
|
|
100 全体の記述量が多いため、if文とfor文のある loaduvm という関数で説明を行う。
|
|
101
|
|
102
|
|
103 \begin{lstlisting}[frame=lrbt,label=vm_loaduvm,caption={\footnotesize vm.c のloaduvm}]
|
13
|
104 // Return the address of the PTE in page directory that corresponds to
|
|
105 // virtual address va. If alloc!=0, create any required page table pages.
|
|
106 static pte_t* walkpgdir (pde_t *pgdir, const void *va, int alloc)
|
|
107 {
|
|
108 pde_t *pde;
|
|
109 pte_t *pgtab;
|
8
|
110
|
13
|
111 // pgdir points to the page directory, get the page direcotry entry (pde)
|
|
112 pde = &pgdir[PDE_IDX(va)];
|
|
113
|
|
114 if (*pde & PE_TYPES) {
|
|
115 pgtab = (pte_t*) p2v(PT_ADDR(*pde));
|
|
116
|
|
117 } else {
|
|
118 if (!alloc || (pgtab = (pte_t*) kpt_alloc()) == 0) {
|
|
119 return 0;
|
|
120 }
|
|
121
|
|
122 // Make sure all those PTE_P bits are zero.
|
|
123 memset(pgtab, 0, PT_SZ);
|
|
124
|
|
125 // The permissions here are overly generous, but they can
|
|
126 // be further restricted by the permissions in the page table
|
|
127 // entries, if necessary.
|
|
128 *pde = v2p(pgtab) | UPDE_TYPE;
|
|
129 }
|
|
130
|
|
131 return &pgtab[PTE_IDX(va)];
|
|
132 }
|
|
133
|
|
134 // Load a program segment into pgdir. addr must be page-aligned
|
|
135 // and the pages from addr to addr+sz must already be mapped.
|
12
|
136 int loaduvm (pde_t *pgdir, char *addr, struct inode *ip, uint offset, uint sz)
|
|
137 {
|
|
138 uint i, pa, n;
|
|
139 pte_t *pte;
|
8
|
140
|
12
|
141 if ((uint) addr % PTE_SZ != 0) {
|
|
142 panic("loaduvm: addr must be page aligned");
|
|
143 }
|
8
|
144
|
12
|
145 for (i = 0; i < sz; i += PTE_SZ) {
|
|
146 if ((pte = walkpgdir(pgdir, addr + i, 0)) == 0) {
|
|
147 panic("loaduvm: address should exist");
|
|
148 }
|
8
|
149
|
12
|
150 pa = PTE_ADDR(*pte);
|
8
|
151
|
12
|
152 if (sz - i < PTE_SZ) {
|
|
153 n = sz - i;
|
|
154 } else {
|
|
155 n = PTE_SZ;
|
|
156 }
|
8
|
157
|
12
|
158 if (readi(ip, p2v(pa), offset + i, n) != n) {
|
|
159 return -1;
|
|
160 }
|
|
161 }
|
8
|
162
|
12
|
163 return 0;
|
8
|
164 }
|
|
165
|
|
166 \end{lstlisting}
|
|
167
|
13
|
168 vm\_impl.cbc の Code Gear であるloaduvmvm\_impl から goto で loaduvm\_ptesize\_checkvm\_impl に遷移する。
|
|
169 vm.c での最初の if 文までの処理を 1つの Code Gear として loaduvm\_ptesize\_checkvm\_impl に記述する。(3行目~11行目)
|
|
170
|
|
171
|
|
172
|
|
173
|
8
|
174
|
12
|
175
|
|
176
|
|
177 \begin{lstlisting}[frame=lrbt,label=impl_vm_loaduvm,caption={\footnotesize privateでの loaduvm の実装}]
|
|
178 #interface "vm_impl.h"
|
|
179
|
|
180 __code loaduvm_ptesize_checkvm_impl(struct vm_impl* vm_impl, __code next(int ret, ...)) {
|
|
181 char* addr = vm_impl->addr;
|
|
182
|
|
183 if ((uint) addr %PTE_SZ != 0) {
|
|
184 // goto panic
|
|
185 }
|
|
186
|
|
187 goto loaduvm_loopvm_impl(vm_impl, next(ret, ...));
|
|
188 }
|
|
189
|
|
190 __code loaduvm_loopvm_impl(struct vm_impl* vm_impl, __code next(int ret, ...)) {
|
|
191 uint i = vm_impl->i;
|
|
192 uint sz = vm_impl->sz;
|
|
193
|
|
194 if (i < sz) {
|
|
195 goto loaduvm_check_pgdir(vm_impl, next(ret, ...));
|
|
196 }
|
|
197
|
|
198 goto loaduvm_exit(vm_impl, next(ret, ...));
|
|
199 }
|
|
200
|
|
201
|
|
202 static pte_t* walkpgdir (pde_t *pgdir, const void *va, int alloc)
|
|
203 {
|
|
204 pde_t *pde;
|
|
205 pte_t *pgtab;
|
|
206
|
|
207 // pgdir points to the page directory, get the page direcotry entry (pde)
|
|
208 pde = &pgdir[PDE_IDX(va)];
|
|
209
|
|
210 if (*pde & PE_TYPES) {
|
|
211 pgtab = (pte_t*) p2v(PT_ADDR(*pde));
|
|
212
|
|
213 } else {
|
|
214 if (!alloc || (pgtab = (pte_t*) kpt_alloc()) == 0) {
|
|
215 return 0;
|
|
216 }
|
|
217
|
|
218 // Make sure all those PTE_P bits are zero.
|
|
219 memset(pgtab, 0, PT_SZ);
|
|
220
|
|
221 // The permissions here are overly generous, but they can
|
|
222 // be further restricted by the permissions in the page table
|
|
223 // entries, if necessary.
|
|
224 *pde = v2p(pgtab) | UPDE_TYPE;
|
|
225 }
|
|
226
|
|
227 return &pgtab[PTE_IDX(va)];
|
|
228 }
|
|
229
|
|
230
|
|
231 __code loaduvm_check_pgdir(struct vm_impl* vm_impl, __code next(int ret, ...)) {
|
|
232 pte_t* pte = vm_impl->pte;
|
|
233 pde_t* pgdir = vm_impl->pgdir;
|
|
234 uint i = vm_impl->i;
|
|
235 char* addr = vm_impl->addr;
|
|
236 uint pa = vm_impl->pa;
|
|
237
|
|
238 if ((pte = walkpgdir(pgdir, addr + i, 0)) == 0) {
|
|
239 // goto panic
|
|
240 }
|
|
241 pa = PTE_ADDR(*pte);
|
|
242
|
|
243 vm_impl->pte = pte;
|
|
244 vm_impl->pgdir = pgdir;
|
|
245 vm_impl->addr = addr;
|
|
246 vm_impl->pa = pa;
|
|
247
|
|
248 goto loaduvm_check_PTE_SZ(vm_impl, next(ret, ...));
|
|
249 }
|
|
250
|
|
251 __code loaduvm_check_PTE_SZ(struct vm_impl* vm_impl, __code next(int ret, ...)) {
|
|
252 uint sz = vm_impl->sz;
|
|
253 uint i = vm_impl->i;
|
|
254 uint n = vm_impl->n;
|
|
255 struct inode* ip = vm_impl->ip;
|
|
256 uint pa = vm_impl->pa;
|
|
257 uint offset = vm_impl->offset;
|
|
258
|
|
259 if (sz - i < PTE_SZ) {
|
|
260 n = sz - i;
|
|
261 } else {
|
|
262 n = PTE_SZ;
|
|
263 }
|
|
264
|
|
265 if (readi(ip, p2v(pa), offset + i, n) != n) {
|
|
266 ret = -1;
|
|
267 goto next(ret, ...);
|
|
268 }
|
|
269
|
|
270 vm_impl->n = n;
|
|
271
|
|
272 goto loaduvm_loopvm_impl(vm_impl, next(ret, ...));
|
|
273 }
|
|
274
|
|
275 __code loaduvm_exit(struct vm_impl* vm_impl, __code next(int ret, ...)) {
|
|
276 ret = 0;
|
|
277 goto next(ret, ...);
|
|
278 }
|
|
279
|
|
280 \end{lstlisting}
|
|
281
|
|
282
|
|
283
|
|
284
|
|
285
|
10
|
286 \section{インターフェースの呼び出し}
|
12
|
287 定義したインターフェースの呼び出し方について説明する。
|
10
|
288 CbC の場合 goto による遷移を行うので、関数呼び出しのように goto 以降のコードを実行できない。
|
12
|
289 例として、ソースコード \ref{cbc_goto} の16行目のように goto によってインターフェースで定義した命令を行うと、戻ってこれないため17行目以降が実行されなくなる。
|
10
|
290
|
|
291
|
13
|
292 \lstinputlisting[frame=lrbt,label=cbc_goto,caption={\footnotesize cbc インターフェースのgoto}]{./src/failure_example_userinit}
|
10
|
293
|
13
|
294 \lstinputlisting[frame=lrbt,label=dummy,caption={\footnotesize dummy を使った呼び出し}]{./src/dummy}
|
12
|
295 ソースコードの説明
|
|
296
|
|
297
|
|
298
|
|
299
|
|
300
|
|
301
|
|
302
|
|
303
|