Skip to content
项目
群组
代码片段
帮助
当前项目
正在载入...
登录 / 注册
切换导航面板
X
xv6-public
项目
项目
详情
活动
周期分析
仓库
仓库
文件
提交
分支
标签
贡献者
分支图
比较
统计图
问题
0
问题
0
列表
看板
标记
里程碑
合并请求
0
合并请求
0
CI / CD
CI / CD
流水线
作业
日程
统计图
Wiki
Wiki
代码片段
代码片段
成员
成员
折叠边栏
关闭边栏
活动
图像
聊天
创建新问题
作业
提交
问题看板
Open sidebar
银宸时代
OS Lab Group
奖励实验
xv6-public
提交
372d4c19
提交
372d4c19
5月 16, 2011
创建
作者:
Frans Kaashoek
浏览文件
操作
浏览文件
下载
电子邮件补丁
差异文件
Start on on-demand paging
上级
5c24b9a8
显示空白字符变更
内嵌
并排
正在显示
4 个修改的文件
包含
81 行增加
和
19 行删除
+81
-19
defs.h
defs.h
+2
-1
exec.c
exec.c
+12
-0
proc.h
proc.h
+5
-0
vm.c
vm.c
+62
-18
没有找到文件。
defs.h
浏览文件 @
372d4c19
...
@@ -167,7 +167,8 @@ void kvmalloc(void);
...
@@ -167,7 +167,8 @@ void kvmalloc(void);
void
vmenable
(
void
);
void
vmenable
(
void
);
pde_t
*
setupkvm
(
void
);
pde_t
*
setupkvm
(
void
);
char
*
uva2ka
(
pde_t
*
,
char
*
);
char
*
uva2ka
(
pde_t
*
,
char
*
);
struct
vmnode
*
vmn_allocpg
(
uint
);
struct
vmnode
*
vmn_alloc
(
uint
,
uint
);
struct
vmnode
*
vmn_allocpg
(
uint
);
void
vmn_free
(
struct
vmnode
*
);
void
vmn_free
(
struct
vmnode
*
);
int
vmn_load
(
struct
vmnode
*
,
struct
inode
*
,
uint
,
uint
);
int
vmn_load
(
struct
vmnode
*
,
struct
inode
*
,
uint
,
uint
);
struct
vmap
*
vmap_alloc
(
void
);
struct
vmap
*
vmap_alloc
(
void
);
...
...
exec.c
浏览文件 @
372d4c19
...
@@ -21,6 +21,9 @@ exec(char *path, char **argv)
...
@@ -21,6 +21,9 @@ exec(char *path, char **argv)
pde_t
*
pgdir
=
0
,
*
oldpgdir
;
pde_t
*
pgdir
=
0
,
*
oldpgdir
;
struct
vmap
*
vmap
=
0
,
*
oldvmap
;
struct
vmap
*
vmap
=
0
,
*
oldvmap
;
struct
vmnode
*
vmn
=
0
;
struct
vmnode
*
vmn
=
0
;
int
odp
=
0
;
cprintf
(
"exec: %s
\n
"
,
path
);
if
((
ip
=
namei
(
path
))
==
0
)
if
((
ip
=
namei
(
path
))
==
0
)
return
-
1
;
return
-
1
;
...
@@ -58,16 +61,25 @@ exec(char *path, char **argv)
...
@@ -58,16 +61,25 @@ exec(char *path, char **argv)
brk
=
va_end
;
brk
=
va_end
;
int
npg
=
(
va_end
-
va_start
)
/
PGSIZE
;
int
npg
=
(
va_end
-
va_start
)
/
PGSIZE
;
if
(
odp
)
{
if
((
vmn
=
vmn_alloc
(
npg
,
ONDEMAND
))
==
0
)
goto
bad
;
}
else
{
if
((
vmn
=
vmn_allocpg
(
npg
))
==
0
)
if
((
vmn
=
vmn_allocpg
(
npg
))
==
0
)
goto
bad
;
goto
bad
;
}
if
(
vmn_load
(
vmn
,
ip
,
ph
.
offset
,
ph
.
filesz
)
<
0
)
if
(
vmn_load
(
vmn
,
ip
,
ph
.
offset
,
ph
.
filesz
)
<
0
)
goto
bad
;
goto
bad
;
if
(
vmap_insert
(
vmap
,
vmn
,
ph
.
va
)
<
0
)
if
(
vmap_insert
(
vmap
,
vmn
,
ph
.
va
)
<
0
)
goto
bad
;
goto
bad
;
vmn
=
0
;
vmn
=
0
;
}
}
if
(
odp
)
iunlock
(
ip
);
else
{
iunlockput
(
ip
);
iunlockput
(
ip
);
ip
=
0
;
ip
=
0
;
}
// Allocate a vmnode for the heap.
// Allocate a vmnode for the heap.
// XXX pre-allocate 32 pages..
// XXX pre-allocate 32 pages..
...
...
proc.h
浏览文件 @
372d4c19
...
@@ -31,6 +31,7 @@ struct context {
...
@@ -31,6 +31,7 @@ struct context {
enum
procstate
{
UNUSED
,
EMBRYO
,
SLEEPING
,
RUNNABLE
,
RUNNING
,
ZOMBIE
};
enum
procstate
{
UNUSED
,
EMBRYO
,
SLEEPING
,
RUNNABLE
,
RUNNING
,
ZOMBIE
};
enum
vmatype
{
PRIVATE
,
COW
};
enum
vmatype
{
PRIVATE
,
COW
};
enum
vmntype
{
EAGER
,
ONDEMAND
};
// Virtual memory
// Virtual memory
struct
vmnode
{
struct
vmnode
{
...
@@ -38,6 +39,10 @@ struct vmnode {
...
@@ -38,6 +39,10 @@ struct vmnode {
char
*
page
[
32
];
char
*
page
[
32
];
uint
ref
;
uint
ref
;
uint
alloc
;
// in use?
uint
alloc
;
// in use?
enum
vmntype
type
;
struct
inode
*
ip
;
uint
offset
;
uint
sz
;
};
};
struct
vma
{
struct
vma
{
...
...
vm.c
浏览文件 @
372d4c19
...
@@ -238,35 +238,43 @@ struct {
...
@@ -238,35 +238,43 @@ struct {
}
vmaps
;
}
vmaps
;
struct
vmnode
*
struct
vmnode
*
vmn_alloc
(
void
)
vmn_alloc
(
uint
npg
,
uint
type
)
{
{
for
(
uint
i
=
0
;
i
<
sizeof
(
vmnodes
.
n
)
/
sizeof
(
vmnodes
.
n
[
0
]);
i
++
)
{
for
(
uint
i
=
0
;
i
<
sizeof
(
vmnodes
.
n
)
/
sizeof
(
vmnodes
.
n
[
0
]);
i
++
)
{
struct
vmnode
*
n
=
&
vmnodes
.
n
[
i
];
struct
vmnode
*
n
=
&
vmnodes
.
n
[
i
];
if
(
n
->
alloc
==
0
&&
__sync_bool_compare_and_swap
(
&
n
->
alloc
,
0
,
1
))
{
if
(
n
->
alloc
==
0
&&
__sync_bool_compare_and_swap
(
&
n
->
alloc
,
0
,
1
))
{
n
->
npages
=
0
;
if
(
npg
>
sizeof
(
n
->
page
)
/
sizeof
(
n
->
page
[
0
]))
{
panic
(
"vmnode too big
\n
"
);
}
n
->
npages
=
npg
;
n
->
ref
=
0
;
n
->
ref
=
0
;
n
->
ip
=
0
;
n
->
type
=
type
;
return
n
;
return
n
;
}
}
}
}
panic
(
"out of vmnodes"
);
panic
(
"out of vmnodes"
);
}
}
st
ruct
vmnode
*
st
atic
int
vmn_
allocpg
(
uint
npg
)
vmn_
doallocpg
(
struct
vmnode
*
n
)
{
{
struct
vmnode
*
n
=
vmn_alloc
();
for
(
uint
i
=
0
;
i
<
n
->
npages
;
i
++
)
{
if
(
npg
>
sizeof
(
n
->
page
)
/
sizeof
(
n
->
page
[
0
]))
{
cprintf
(
"vmnode too big: %d
\n
"
,
npg
);
return
0
;
}
for
(
uint
i
=
0
;
i
<
npg
;
i
++
)
{
if
((
n
->
page
[
i
]
=
kalloc
())
==
0
)
{
if
((
n
->
page
[
i
]
=
kalloc
())
==
0
)
{
vmn_free
(
n
);
vmn_free
(
n
);
return
0
;
return
-
1
;
}
}
memset
((
char
*
)
n
->
page
[
i
],
0
,
PGSIZE
);
memset
((
char
*
)
n
->
page
[
i
],
0
,
PGSIZE
);
n
->
npages
++
;
}
}
return
0
;
}
struct
vmnode
*
vmn_allocpg
(
uint
npg
)
{
struct
vmnode
*
n
=
vmn_alloc
(
npg
,
EAGER
);
if
(
n
==
0
)
return
0
;
if
(
vmn_doallocpg
(
n
)
<
0
)
return
0
;
return
n
;
return
n
;
}
}
...
@@ -274,9 +282,13 @@ void
...
@@ -274,9 +282,13 @@ void
vmn_free
(
struct
vmnode
*
n
)
vmn_free
(
struct
vmnode
*
n
)
{
{
for
(
uint
i
=
0
;
i
<
n
->
npages
;
i
++
)
{
for
(
uint
i
=
0
;
i
<
n
->
npages
;
i
++
)
{
if
(
n
->
page
[
i
])
{
kfree
((
char
*
)
n
->
page
[
i
]);
kfree
((
char
*
)
n
->
page
[
i
]);
n
->
page
[
i
]
=
0
;
n
->
page
[
i
]
=
0
;
}
}
}
if
(
n
->
ip
)
panic
(
"vmn_free: drop inode ref"
);
n
->
alloc
=
0
;
n
->
alloc
=
0
;
}
}
...
@@ -307,7 +319,7 @@ vmap_alloc(void)
...
@@ -307,7 +319,7 @@ vmap_alloc(void)
if
(
m
->
alloc
==
0
&&
__sync_bool_compare_and_swap
(
&
m
->
alloc
,
0
,
1
))
{
if
(
m
->
alloc
==
0
&&
__sync_bool_compare_and_swap
(
&
m
->
alloc
,
0
,
1
))
{
for
(
uint
j
=
0
;
j
<
sizeof
(
m
->
e
)
/
sizeof
(
m
->
e
[
0
]);
j
++
){
for
(
uint
j
=
0
;
j
<
sizeof
(
m
->
e
)
/
sizeof
(
m
->
e
[
0
]);
j
++
){
m
->
e
[
j
].
n
=
0
;
m
->
e
[
j
].
n
=
0
;
m
->
e
[
i
].
va_type
=
PRIVATE
;
m
->
e
[
j
].
va_type
=
PRIVATE
;
m
->
e
[
j
].
lock
.
name
=
"vma"
;
m
->
e
[
j
].
lock
.
name
=
"vma"
;
}
}
m
->
lock
.
name
=
"vmap"
;
m
->
lock
.
name
=
"vmap"
;
...
@@ -415,9 +427,8 @@ vmap_copy(struct vmap *m, pde_t* pgdir, int share)
...
@@ -415,9 +427,8 @@ vmap_copy(struct vmap *m, pde_t* pgdir, int share)
return
c
;
return
c
;
}
}
// Load a program segment into a vmnode.
static
int
int
vmn_doload
(
struct
vmnode
*
vmn
,
struct
inode
*
ip
,
uint
offset
,
uint
sz
)
vmn_load
(
struct
vmnode
*
vmn
,
struct
inode
*
ip
,
uint
offset
,
uint
sz
)
{
{
for
(
uint
i
=
0
;
i
<
sz
;
i
+=
PGSIZE
){
for
(
uint
i
=
0
;
i
<
sz
;
i
+=
PGSIZE
){
uint
n
;
uint
n
;
...
@@ -432,6 +443,20 @@ vmn_load(struct vmnode *vmn, struct inode *ip, uint offset, uint sz)
...
@@ -432,6 +443,20 @@ vmn_load(struct vmnode *vmn, struct inode *ip, uint offset, uint sz)
return
0
;
return
0
;
}
}
// Load a program segment into a vmnode.
int
vmn_load
(
struct
vmnode
*
vmn
,
struct
inode
*
ip
,
uint
offset
,
uint
sz
)
{
if
(
vmn
->
type
==
ONDEMAND
)
{
vmn
->
ip
=
ip
;
vmn
->
offset
=
offset
;
vmn
->
sz
=
sz
;
return
0
;
}
else
{
return
vmn_doload
(
vmn
,
ip
,
offset
,
sz
);
}
}
// Free a page table and all the physical memory pages
// Free a page table and all the physical memory pages
// in the user part.
// in the user part.
void
void
...
@@ -529,12 +554,29 @@ pagefault(pde_t *pgdir, struct vmap *vmap, uint va, uint err)
...
@@ -529,12 +554,29 @@ pagefault(pde_t *pgdir, struct vmap *vmap, uint va, uint err)
if
(
m
==
0
)
if
(
m
==
0
)
return
-
1
;
return
-
1
;
//
cprintf("%d: pf addr=0x%x err 0x%x check = %d\n", proc->pid, va, err, check
);
//
cprintf("%d: pf addr=0x%x err 0x%x\n", proc->pid, va, err
);
// cprintf("%d: pf vma type = %d refcnt %d
pte=0x%x\n", proc->pid, m->va_type, m->n->ref
, *pte);
// cprintf("%d: pf vma type = %d refcnt %d
vmn type %d pte=0x%x\n", proc->pid, m->va_type, m->n->ref, m->n->type
, *pte);
uint
npg
=
(
PGROUNDDOWN
(
va
)
-
m
->
va_start
)
/
PGSIZE
;
uint
npg
=
(
PGROUNDDOWN
(
va
)
-
m
->
va_start
)
/
PGSIZE
;
if
(
m
->
n
&&
m
->
n
->
ip
&&
*
pte
==
0x0
)
{
//cprintf("ODP\n");
if
(
vmn_doallocpg
(
m
->
n
)
<
0
)
{
panic
(
"pagefault: couldn't allocate pages"
);
}
release
(
&
m
->
lock
);
if
(
vmn_doload
(
m
->
n
,
m
->
n
->
ip
,
m
->
n
->
offset
,
m
->
n
->
sz
)
<
0
)
{
panic
(
"pagefault: couldn't load"
);
}
acquire
(
&
m
->
lock
);
pte
=
walkpgdir
(
pgdir
,
(
const
void
*
)
va
,
0
);
if
(
pte
==
0x0
)
panic
(
"pagefault: not paged in???"
);
cprintf
(
"ODP done
\n
"
);
}
if
(
m
->
va_type
==
COW
&&
(
err
&
FEC_WR
))
{
if
(
m
->
va_type
==
COW
&&
(
err
&
FEC_WR
))
{
// Write to a COW page
// Write to a COW page
// cprintf("write to cow\n");
if
(
m
->
n
->
ref
==
1
)
{
// if vma isn't shared any more, make it private
if
(
m
->
n
->
ref
==
1
)
{
// if vma isn't shared any more, make it private
m
->
va_type
=
PRIVATE
;
m
->
va_type
=
PRIVATE
;
*
pte
=
PADDR
(
m
->
n
->
page
[
npg
])
|
PTE_P
|
PTE_U
|
PTE_W
;
*
pte
=
PADDR
(
m
->
n
->
page
[
npg
])
|
PTE_P
|
PTE_U
|
PTE_W
;
...
@@ -552,8 +594,10 @@ pagefault(pde_t *pgdir, struct vmap *vmap, uint va, uint err)
...
@@ -552,8 +594,10 @@ pagefault(pde_t *pgdir, struct vmap *vmap, uint va, uint err)
*
pte
=
PADDR
(
m
->
n
->
page
[
npg
])
|
PTE_P
|
PTE_U
|
PTE_W
;
*
pte
=
PADDR
(
m
->
n
->
page
[
npg
])
|
PTE_P
|
PTE_U
|
PTE_W
;
}
}
}
else
if
(
m
->
va_type
==
COW
)
{
}
else
if
(
m
->
va_type
==
COW
)
{
// cprintf("cow\n");
*
pte
=
PADDR
(
m
->
n
->
page
[
npg
])
|
PTE_P
|
PTE_U
|
PTE_COW
;
*
pte
=
PADDR
(
m
->
n
->
page
[
npg
])
|
PTE_P
|
PTE_U
|
PTE_COW
;
}
else
{
}
else
{
// cprintf("fill in pte\n");
if
(
m
->
n
->
ref
>
1
)
if
(
m
->
n
->
ref
>
1
)
panic
(
"pagefault"
);
panic
(
"pagefault"
);
*
pte
=
PADDR
(
m
->
n
->
page
[
npg
])
|
PTE_P
|
PTE_U
|
PTE_W
;
*
pte
=
PADDR
(
m
->
n
->
page
[
npg
])
|
PTE_P
|
PTE_U
|
PTE_W
;
...
...
编写
预览
您添加了
0
人
到此讨论。请谨慎行事。
请先完成此评论的编辑!
取消
请
注册
或者
登录
后发表评论