-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathos.go
245 lines (197 loc) · 6.12 KB
/
os.go
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
package sham
import (
log "github.com/sirupsen/logrus"
"sync"
"time"
)
// OS 是模拟的「操作系统」,一个持有并管理 CPU,内存、IO 设备的东西。
// 单核,支持多道程序。
type OS struct {
CPU CPU
Mem Memory
Devs map[string]Device
ProcsMutex sync.RWMutex
RunningProc *Process
ReadyProcs []*Process
BlockedProcs []*Process
Scheduler Scheduler
Interrupts []Interrupt
}
// NewOS 构建一个「操作系统」。
// 新的操作系统有自己控制的 CPU、内存、IO 设备,
// 包含一个 Noop 的进程表以及默认的 NoScheduler 调度器。
func NewOS() *OS {
return &OS{
CPU: CPU{},
Mem: Memory{},
Devs: map[string]Device{
"stdout": NewStdOut(),
"stdin": NewStdIn(),
},
ReadyProcs: []*Process{&Noop},
BlockedProcs: []*Process{},
Scheduler: NoScheduler{},
Interrupts: []Interrupt{},
}
}
// Boot 启动操作系统。即启动操作系统的调度器。
// 调度器退出标志着操作系统的退出,也就是关机。
func (os *OS) Boot() {
field := "[OS] "
log.Info(field, "OS Boot: start scheduler")
os.Scheduler.schedule(os)
log.Info(field, "No process to run. Showdown OS.")
}
// HandleInterrupts 处理中断队列中的中断
func (os *OS) HandleInterrupts() {
var i Interrupt
for len(os.Interrupts) > 0 {
i, os.Interrupts = os.Interrupts[0], os.Interrupts[1:]
log.WithFields(log.Fields{
"type": i.Typ,
"data": i.Data,
}).Info("[OS] Handle Interrupt")
i.Handler(os, i.Data)
os.clockTick()
}
}
/********* 👇 SYSTEM CALLS 👇 ***************/
// OSInterface 是操作系统暴露出来的「系统调用」接口
type OSInterface interface {
CreateProcess(pid string, precedence uint, timeCost uint, runnable Runnable)
InterruptRequest(thread *Thread, typ string, channel chan interface{})
FindProcess(pid string) *Process
// 这个只是模拟的内部需要,不是真正意义上的系统调用。
clockTick()
}
// CreateProcess 创建一个进程,放到进程表里
func (os *OS) CreateProcess(pid string, precedence uint, timeCost uint, runnable Runnable) {
// process
p := Process{
Id: pid,
Precedence: precedence,
Devices: map[string]Device{},
}
// init mem
// give new process a var table
os.Mem = append(os.Mem, Object{
Pid: pid,
Content: nil,
})
p.Memory = os.Mem[len(os.Mem)-1:]
// thread
p.Thread = &Thread{
runnable: runnable,
contextual: &Contextual{
Process: &p,
OS: os,
},
remainingTime: timeCost,
}
// append to ReadyProcs
os.ReadyProcs = append(os.ReadyProcs, &p)
}
/// InterruptRequest 发出中断请求,阻塞当前进程
func (os *OS) InterruptRequest(thread *Thread, typ string, channel chan interface{}) {
log.WithFields(log.Fields{
"thread": thread,
"type": typ,
"channel": channel,
}).Info("[OS] InterruptRequest")
i := GetInterrupt(thread.contextual.Process.Id, typ, channel)
os.Interrupts = append(os.Interrupts, i)
os.CPU.Cancel(StatusBlocked)
}
// FindProcess 通过 pid 获取一个进程
func (os *OS) FindProcess(pid string) *Process {
os.ProcsMutex.Lock()
defer os.ProcsMutex.Unlock()
if pid == os.RunningProc.Id {
return os.RunningProc
}
for _, p := range append(os.ReadyProcs, os.BlockedProcs...) {
if p.Id == pid {
return p
}
}
return nil
}
// clockTick 时钟增长
// 这里模拟需要,所以是软的实现,而不是真的"硬件"时钟。
func (os *OS) clockTick() {
os.CPU.Clock += 1
time.Sleep(time.Second)
if os.CPU.Clock%10 == 0 && os.RunningProc.Status == StatusRunning { // 时钟中断
ch := make(chan interface{}, 1) // buffer 很重要!
os.InterruptRequest(os.RunningProc.Thread, ClockInterrupt, ch)
ch <- os.RunningProc
os.CPU.Clock = 0
}
}
/********* 👆 SYSTEM CALLS 👆 ***************/
/********* 👇 进程状态转换 👇 ***************/
// RunningToBlocked 阻塞当前运行的进程
func (os *OS) RunningToBlocked() {
os.ProcsMutex.Lock()
defer os.ProcsMutex.Unlock()
log.WithField("process", os.RunningProc).Info("[OS] RunningToBlocked")
os.RunningProc.Status = StatusBlocked
os.BlockedProcs = append(os.BlockedProcs, os.RunningProc)
os.CPU.Unlock()
}
// RunningToReady 把当前运行的进程变成就绪,并释放 CPU
func (os *OS) RunningToReady() {
os.ProcsMutex.Lock()
defer os.ProcsMutex.Unlock()
log.WithField("process", os.RunningProc).Info("[OS] RunningToReady")
os.RunningProc.Status = StatusReady
os.ReadyProcs = append(os.ReadyProcs, os.RunningProc)
os.CPU.Unlock()
}
// RunningToDone 把当前运行的进程标示成完成,并释放 CPU
func (os *OS) RunningToDone() {
os.ProcsMutex.Lock()
defer os.ProcsMutex.Unlock()
log.WithField("process", os.RunningProc).Info("[OS] RunningToDone")
os.RunningProc.Status = StatusDone
os.CPU.Unlock()
}
// ReadyToRunning 把就绪队列中的 pid 进程变成运行状态呀
// 这个方法会引导 CPU 切换运行进程,并锁上 CPU
func (os *OS) ReadyToRunning(pid string) {
os.ProcsMutex.Lock()
defer os.ProcsMutex.Unlock()
key := -1
for i, p := range os.ReadyProcs {
if p.Id == pid {
key = i
}
}
log.WithField("process", os.ReadyProcs[key]).Info("[OS] ReadyToRunning")
os.ReadyProcs[key].Status = StatusRunning
os.RunningProc = os.ReadyProcs[key]
os.ReadyProcs = append(os.ReadyProcs[:key], os.ReadyProcs[key+1:]...) // 从就绪队列里删除
os.CPU.Lock()
os.CPU.Clock = 0 // 重置时钟计数
os.CPU.Switch(os.RunningProc.Thread)
}
// BlockedToReady 把阻塞中的 pid 进程变为就绪状态
func (os *OS) BlockedToReady(pid string) {
os.ProcsMutex.Lock()
defer os.ProcsMutex.Unlock()
key := -1
for i, p := range os.BlockedProcs {
if p.Id == pid {
key = i
}
}
if key == -1 {
log.WithField("pid", pid).Warn("[OS] BlockedToReady Failed: No such Blocked Process")
return
}
log.WithField("process", os.BlockedProcs[key]).Info("[OS] BlockedToReady")
os.BlockedProcs[key].Status = StatusReady
os.ReadyProcs = append(os.ReadyProcs, os.BlockedProcs[key]) // append BlockedProcs[key] into ReadyProcs
os.BlockedProcs = append(os.BlockedProcs[:key], os.BlockedProcs[key+1:]...) // Delete BlockedProcs[key]
}
/********* 👆 进程状态转换 👆 ***************/