版權說明:本文檔由用戶提供并上傳,收益歸屬內(nèi)容提供方,若內(nèi)容存在侵權,請進行舉報或認領
文檔簡介
1、<p> 操作系統(tǒng)課程設計報告</p><p><b> 目錄</b></p><p><b> 目錄2</b></p><p><b> 一、 前期工作3</b></p><p><b> 1.1平臺搭建3</b></p&
2、gt;<p><b> 二、 代碼分工3</b></p><p><b> 三、設計及實現(xiàn)4</b></p><p> 3.1 Task1.1 KThread.join()4</p><p> 3.1.1 要求分析4</p><p> 3.1.2 設計方案4<
3、/p><p> 3.1.3 實現(xiàn)代碼4</p><p> 3.1.4 測試代碼及結果5</p><p> 3.2 Task1.2 condition2類7</p><p> 3.2.1 要求分析7</p><p> 3.2.2 設計方案8</p><p> 3.2.3 實現(xiàn)代碼
4、8</p><p> 3.2.4 測試代碼及結果8</p><p> 3.3 Task1.3 alram類11</p><p> 3.3.1 要求分析11</p><p> 3.3.2 設計方案11</p><p> 3.3.3 實現(xiàn)代碼12</p><p> 3.3.4
5、 測試代碼及結果12</p><p> 3.4 Task1.4 communicator類14</p><p> 3.4.1要求分析14</p><p> 3.4.2 設計方案14</p><p> 3.4.3 實現(xiàn)代碼14</p><p> 3.4.4 測試代碼及結果15</p>
6、<p> 3.5 Task1.5 priority scheduler類17</p><p> 3.5.1 要求分析17</p><p> 3.5.2 設計方案17</p><p> 3.5.3 實現(xiàn)代碼18</p><p> 3.5.4 測試代碼及結果18</p><p> 3.6 T
7、ask1.6 boat類23</p><p> 3.6.1 要求分析23</p><p> 3.6.2 設計方案23</p><p> 3.6.3 實現(xiàn)代碼23</p><p> 3.6.4 測試代碼及結果30</p><p> 3.7 Task2.1 系統(tǒng)調(diào)用creat, open, read,
8、write, close, unlink30</p><p> 3.7.1 要求分析30</p><p> 3.7.2 設計方案31</p><p> 3.7.3 實現(xiàn)代碼32</p><p> 3.7.4 測試代碼及結果34</p><p> 3.8 Task2.2 多進程內(nèi)存分配及訪問35&l
9、t;/p><p> 3.8.1 要求分析35</p><p> 3.8.2 設計方案35</p><p> 3.8.3 實現(xiàn)代碼36</p><p> 3.8.4 測試代碼及結果39</p><p> 3.9 Task2.3 系統(tǒng)調(diào)用exec, join, exit41</p><p
10、> 3.9.1 要求分析41</p><p> 3.9.2 設計方案41</p><p> 3.9.3 實現(xiàn)代碼42</p><p> 3.9.4 測試代碼及結果43</p><p> 3.10 Task2.4 Lottery Schedule類44</p><p> 3.10.1 要求分析
11、44</p><p> 3.10.2 設計方案45</p><p> 3.10.3 實現(xiàn)代碼45</p><p> 3.10.4 測試代碼及結果45</p><p><b> 總結48</b></p><p><b> 前期工作</b></p>
12、<p><b> 1.1平臺搭建</b></p><p> Nachos For Java</p><p> phrase1部分:IDE環(huán)境可采用Eclipse。</p><p> Phase 2及以后階段:需要把C程序編譯成MIPS二進制文件COFF,需要MIPS的C編譯器 mips-x86.linux-xgcc(ub
13、untu12.04平臺下)。</p><p><b> 代碼分工</b></p><p><b> 三、設計及實現(xiàn)</b></p><p> 3.1 Task1.1 KThread.join()</p><p> 3.1.1 要求分析</p><p> Join()
14、方法的含義:當前線程a在運行,執(zhí)行b.join(),則a阻塞,直到線程b結束,a繼續(xù)執(zhí)行。</p><p> Join函數(shù)的作用即為等待某線程運行完畢。當前線程 (唯一一個正在運行的線程) A調(diào)用另一個線程 (處于就緒狀態(tài)) B的join函數(shù)時 (A 和 B 在Nachos中均為KThread類型對象),A被掛起,直到B運行結束后, join函數(shù)返回,A才能繼續(xù)運行。注意在一個KThread對象上只能調(diào)用一次j
15、oin,且當前線程不能對自身調(diào)用join。</p><p> 3.1.2 設計方案</p><p> 為每個線程創(chuàng)建一個線程隊列joinqueue,joinqueue由對本線程調(diào)用join方法的其他線程對象構成,可“捐贈”優(yōu)先級 ,以及布爾形變量joined,判斷本線程是否被其他線程調(diào)用過join方法,以及一個判斷線程狀態(tài)方法IsAlive()。</p><p>
16、; 假設當前線程A調(diào)用就緒線程B的join函數(shù). 在join函數(shù)中線程A被添加到線程B的線程隊列joinqueue中,將線程A“休眠”, 線程B得到執(zhí)行,在線程B結束時(此時線程B成為當前線程, 發(fā)生在KThead.finish函數(shù)中) 選擇B的joinqueue的nextThread()執(zhí)行, A喚醒并繼續(xù). </p><p> 3.1.3 實現(xiàn)代碼</p><p> 在KThre
17、ad.java 中: </p><p> private ThreadQueue joinQueue = null; </p><p> private boolean Joined = false;</p><p> public boolean IsAlive(){</p><p> if(this.status==sta
18、tusNew||status==statusReady||</p><p> status==statusRunning||status==statusBlocked)</p><p> return true;</p><p> else return false;</p><p><b> }</b><
19、/p><p> public void join() {</p><p><b> //</b></p><p> if (!this.IsAlive()) return;</p><p> while (IsAlive()) {</p><p><b> //</b>
20、;</p><p> this.joinQueue.acquire(this);</p><p><b> //</b></p><p><b> }</b></p><p> this.joinQueue.waitForAccess(KThread.currentThread());<
21、;/p><p> KThread.sleep();</p><p><b> //</b></p><p><b> }</b></p><p><b> }</b></p><p> public static void finish() {&l
22、t;/p><p><b> ///</b></p><p> if (currentThread.Joined) {</p><p> currentThread.joinQueue.nextThread().ready();</p><p><b> }</b></p><
23、;p><b> ///</b></p><p><b> }</b></p><p> 3.1.4 測試代碼及結果</p><p> 創(chuàng)建AThread和Bthread兩個線程,AThread循環(huán)打印“AThread循環(huán)第..次”語句,Bthread開始打印“B_thread 就緒”語句,中間執(zhí)行AThrea
24、d。Join()方法,最后打印“B_thread執(zhí)行結束”語句。</p><p><b> 測試代碼:</b></p><p> package nachos.threads;</p><p> import nachos.machine.*;</p><p> public class KThreadTest
25、{</p><p> public KThreadTest() {</p><p><b> }</b></p><p> public static void simpleJoinTest() {</p><p> KThread A_thread = new KThread(new KThreadTest.A
26、_thread(5));</p><p> KThread B_thread = new KThread(new KThreadTest.B_thread(A_thread));</p><p> B_thread.fork();</p><p> B_thread.join();</p><p><b> }</b&
27、gt;</p><p> public static class B_thread implements Runnable {</p><p> B_thread(KThread joinee) {</p><p> this.joinee = joinee;</p><p><b> }</b></p&g
28、t;<p> public void run() {</p><p> System.out.println("B_thread 就緒");</p><p> System.out.println("Forking and joining A_thread...");</p><p> this.join
29、ee.fork();</p><p> this.joinee.join();</p><p> System.out.println("B_thread 執(zhí)行結束");</p><p><b> }</b></p><p> private KThread joinee;</p>
30、<p><b> }</b></p><p> public static class A_thread implements Runnable {</p><p> A_thread(int num) {</p><p> this.num = num;</p><p><b> }&
31、lt;/b></p><p> public void run() {</p><p> System.out.println("A_thread 就緒");</p><p> System.out.println("A_thread開始執(zhí)行");</p><p> // This sho
32、uld just kill some cycles</p><p> for (int i = 0; i < this.num; ++i) {</p><p> System.out.println("A_thread 循環(huán) 第" + i + " 次");</p><p> KThread.currentThrea
33、d().yield();</p><p><b> }</b></p><p> System.out.println("A_thread 執(zhí)行結束");</p><p><b> }</b></p><p> private int num;</p>&l
34、t;p><b> }</b></p><p><b> }</b></p><p><b> 測試結果:</b></p><p> 3.2 Task1.2 condition2類</p><p> 3.2.1 要求分析</p><p>
35、 threads.Lock類提供了鎖以保證互斥. 在臨界代碼區(qū)的兩端執(zhí)行Lock.acquire() 和Lock.release() 即可保證同時只有一個線程訪問臨界代碼區(qū). 條件變量建立在鎖之上, 由threads.Condition實現(xiàn), 它是用來保證同步的工具. 每一個條件變量擁有一個鎖變量 (該鎖變量亦可被執(zhí)行acquire和release操作, 多個條件變量可共享同一個鎖變量). 當處于臨界區(qū)內(nèi)的擁有某鎖Lock的當前線程對與
36、鎖Lock聯(lián)系的條件變量執(zhí)行sleep操作時, 該線程失去鎖L并被掛起. 下一個等待鎖L的線程獲得鎖L (這個過程由調(diào)度程序完成) 并進入臨界區(qū). 當擁有鎖L的臨界區(qū)內(nèi)的當前線程對與鎖L聯(lián)系的條件變量執(zhí)行wake操作時 (通常調(diào)用wake之后緊接著就是Lock.release), 等待在該條件變量上的至多一個被掛起的線程 (由調(diào)用sleep引起) 被重新喚醒并設置為就緒狀態(tài). 若執(zhí)行wakeall操作, 則等待在該條件變量上的所有被掛起
37、的線程都被喚醒. threads.condition已經(jīng)實現(xiàn)了一個條件變量 (采用信號量實現(xiàn)), 題目要求用屏蔽/禁止中斷</p><p> 3.2.2 設計方案</p><p> 為每個條件變量添加一個鎖conditionLock和一個Kthread對象組成的等待隊列waitqueue,condition.sleep 將調(diào)用sleep()的線程加入到等待隊列,釋放鎖,并阻塞,以便其他
38、線程喚醒它,一旦’喚醒’立即要求鎖,并在sleep函數(shù)開始/結尾處屏蔽/允許中斷以保證原子性;condition.wake中從等待隊列中取出線程進行ready()實現(xiàn)喚醒, </p><p> (同樣要在wake函數(shù)開始/結尾處屏蔽/允許中斷), wakeall函數(shù)的實現(xiàn)依賴于wake. 只需不斷地wake 直到隊列為空為止. </p><p> 3.2.3 實現(xiàn)代碼</p&g
39、t;<p> 見threads/condition2.java </p><p> private Lock conditionLock;</p><p> private LinkedList<KThread> waitQueue;</p><p> public void sleep() {</p><p&g
40、t;<b> //</b></p><p> waitQueue.add(KThread.currentThread());</p><p> conditionLock.release();</p><p> KThread.sleep();</p><p> conditionLock.acquire();
41、</p><p><b> //</b></p><p><b> }</b></p><p> public void wake() {</p><p><b> //</b></p><p> waitQueue.remove().read
42、y();</p><p><b> //</b></p><p><b> }</b></p><p> 3.2.4 測試代碼及結果</p><p> 創(chuàng)建共有條件變量c2test,一個臨界資源count和三個線程,thread1,thread2,thread3,初始化三個線程后thread
43、1,thread2調(diào)用c2test.sleep()后開始“睡眠”,thread3調(diào)用c2test.wakeAll()喚醒所有線程,thread1,thead2開始交替訪問臨界資源count并更改其“余額”數(shù)值。</p><p><b> 測試代碼:</b></p><p> package nachos.threads;</p><p>
44、 import nachos.machine.*;</p><p> public class Condition2Test { </p><p> Lock condlock = new Lock();</p><p> Condition2 c2test = new Condition2(condlock);</p><p> p
45、ublic Condition2Test(){ </p><p><b> }</b></p><p> public void simpleCondition2Test(){ </p><p> System.out.println("\n ****Condition2Test is now executing.***
46、*");</p><p> System.out.println("初始化賬戶余額為10000");</p><p> final MyCount myCount = new MyCount("0000001", 10000); //創(chuàng)建并發(fā)訪問的賬戶</p><p> KThread thread1 = n
47、ew KThread(new Runnable() {</p><p> public void run() {</p><p> new SaveThread("張三", myCount, 2000);</p><p> System.out.println("張三 goes to sleep");</p>
48、<p> condlock.acquire();</p><p> c2test.sleep();</p><p> System.out.println("張三 reacquires lock when woken.");</p><p> condlock.release();</p><p>
49、 System.out.println("張三 is awake!!!");</p><p> myCount.save(2000," 張三");</p><p><b> }</b></p><p><b> });</b></p><p> KTh
50、read thread2 = new KThread(new Runnable() {</p><p> public void run() {</p><p> new SaveThread("李四", myCount, 2000);</p><p> System.out.println("李四 goes to sleep&
51、quot;);</p><p> condlock.acquire();</p><p> c2test.sleep();</p><p> System.out.println("李四 reacquires lock when woken.");</p><p> condlock.release();</
52、p><p> System.out.println("李四 is awake!!!");</p><p> myCount.save(2000," 李四");</p><p><b> }</b></p><p><b> });</b></p&g
53、t;<p> KThread thread3 = new KThread(new Runnable() {</p><p> public void run() {</p><p> System.out.println("thread 3 Waking up the thread");</p><p> condlock.
54、acquire();</p><p> c2test.wakeAll();</p><p> condlock.release();</p><p> System.out.println("張三 and 李四 woke up by wakeAll");</p><p><b> }</b>&
55、lt;/p><p><b> });</b></p><p> thread1.fork();</p><p> thread2.fork();</p><p> thread3.fork();</p><p> thread1.join();</p><p> t
56、hread2.join();</p><p> thread3.join();</p><p> System.out.println("****Condition2Test finished.****\n");</p><p> } </p><p><b> } </b&
57、gt;</p><p> class SaveThread extends KThread { </p><p> private String name; //操作人 </p><p> private MyCount myCount; //賬戶 </p><p> private int
58、 x; //存款金額 </p><p> SaveThread(String name, MyCount myCount, int x) { </p><p> this.name = name; </p><p> this.myCount = myCount; </p><p>
59、 this.x = x; </p><p><b> } </b></p><p> public void run() { </p><p> myCount.save(x, name); </p><p><b> } </b></p><p><b>
60、; } </b></p><p> class MyCount { </p><p> private String id; //賬號 </p><p> private int cash; //賬戶余額 </p>
61、;<p> Lock lock = new Lock();//賬戶鎖 </p><p> Condition2 c2test = new Condition2(lock);</p><p> MyCount(String id, int cash) { </p><p> this.id = id; </p><p
62、> this.cash = cash; </p><p><b> } </b></p><p> public void save(int x, String name) { </p><p> lock.acquire(); //獲取鎖 </p><p> if
63、 (x > 0) { </p><p> cash += x; //存款 </p><p> System.out.println(name + "存款" + x + ",當前余額為" + cash); </p><p><b> } </b></p
64、><p> lock.release(); //釋放鎖 </p><p><b> } </b></p><p><b> }</b></p><p><b> 測試結果:</b></p><p> 3.3 Task1.3 a
65、lram類</p><p> 3.3.1 要求分析</p><p> 一個線程調(diào)用waitUntil(x) 后, 它即被掛起. 在當前時鐘走過x個滴答后, 該線程被重新喚醒. 線程被喚醒后并不一定立即進入運行態(tài), 只要將其放入就緒隊列中即可. 另外, 多個線程可以分別調(diào)用waitUntil函數(shù)。</p><p> 3.3.2 設計方案</p>&
66、lt;p> 于Alarm類有關的是machine.Timer類. 它在大約每500個時鐘滴答使調(diào)用回調(diào)函數(shù) (由Timer.setInterruptHandler函數(shù)設置). 因此, Alarm類的構造函數(shù)中首先要設置該回調(diào)函數(shù)Alarm.timerInterrupt(). </p><p> 為了實現(xiàn)waitUntil, 需要在Alarm類中實現(xiàn)一個隊列 Alarm.alarmQueue. 隊列中的每
67、個項目是 (線程, 喚醒時間, 相應條件變量) 的三元組. 在調(diào)用waitUntil(x) 函數(shù)時, 首先得到關于</p><p> 以上三元組的信息: (線程: 當前線程, 喚醒時間: 當前時間+x, 新分配的條件變量), 然后將該元組放入隊列中, 并對條件變量sleep操作使當前線程掛起. 在時鐘回調(diào)函數(shù)中 (大約每500個時鐘間隔調(diào)用一次) 則依次檢查隊列中的每個三元組. 如果喚醒時間大于當前時間, 則將
68、元組移出隊列并對元組中的條件變量執(zhí)行wake操作將相應線程喚醒. </p><p> 3.3.3 實現(xiàn)代碼</p><p> public void timerInterrupt() {</p><p><b> //</b></p><p> while(!alarmqueue.isEmpty() &&
69、amp; (alarmqueue.peek().wakeTime <= Machine.timer().getTime())){</p><p> AlarmThread thread = alarmqueue.remove();</p><p> thread.waitThread.ready();</p><p><b> }</b&
70、gt;</p><p><b> // </b></p><p><b> }</b></p><p> public void waitUntil(long x) {</p><p><b> //</b></p><p> Alarm
71、Thread thread = new AlarmThread(KThread.currentThread(), wakeTime,conditionlock);</p><p> alarmqueue.add(thread);</p><p> KThread.sleep();</p><p><b> //</b></p>
72、<p><b> }</b></p><p> private KThread waitThread;</p><p> private Condition timecondition;</p><p> private long wakeTime;</p><p> 3.3.4 測試代碼及結果&
73、lt;/p><p> 創(chuàng)建5個線程,依次“睡眠”20000ticks,線程醒來后各自打印出睡眠開始時間,要求睡眠時間,醒來時間,以及實際睡眠時間。</p><p><b> 測試代碼: </b></p><p> public static void selfTest(int numOfTest){</p><p&g
74、t; Runnable a = new Runnable() {</p><p> public void run() {</p><p> AlarmTestThread();</p><p><b> }</b></p><p><b> };</b></p><p
75、> for(int i=0; i<numOfTest; i++){</p><p> int numOfThreads = 5;</p><p> print("Creating " + numOfThreads + " num of threads");</p><p> for(int j=0; j&l
76、t;numOfThreads; j++){</p><p> KThread thread = new KThread(a);</p><p> thread.setName("thread");</p><p> thread.fork(); }</p><p> ThreadedKernel.alarm.wai
77、tUntil(30000000); </p><p><b> } </b></p><p><b> }</b></p><p> static void AlarmTestThread(){</p><p> long sleepTime =20000;</p><p
78、> long timeBeforeSleep = Machine.timer().getTime();</p><p> ThreadedKernel.alarm.waitUntil(sleepTime);</p><p> long timeAfterSleep = Machine.timer().getTime();</p><p> long a
79、ctualSleepTime = timeAfterSleep - timeBeforeSleep;</p><p> print(KThread.currentThread().toString() + " 被要求在: " + timeBeforeSleep + " 睡眠 " + sleepTime + " ticks");</p>&
80、lt;p> print(KThread.currentThread().toString() + " 被喚醒時間: " + Machine.timer().getTime() + "要求睡眠時間:: " + sleepTime + " ticks, 實際睡眠時間: " + actualSleepTime);</p><p><b>
81、}</b></p><p> public static void print(String Message) {</p><p> System.out.println(Message);</p><p><b> }</b></p><p><b> 測試結果:</b><
82、;/p><p> 3.4 Task1.4 communicator類</p><p><b> 3.4.1要求分析</b></p><p> 用條件變量實現(xiàn)Communicator類中的speaker()和listen()函數(shù)一個鎖lock</p><p> 兩個條件變量speaker(lock),listener(
83、lockw)用來用來控制聽說動作。</p><p> 3.4.2 設計方案</p><p> 每個Communicator有一個鎖 lock(保證操作的原子性) 和與該鎖聯(lián)系的兩個條件變量用于保證speaker和listener間的同步. 在speak函數(shù)中, 首先檢查若已經(jīng)有一個speaker在等待(message變量) 或無listener等待, 則掛起. 否則設置message變
84、量, 準備數(shù)據(jù)并喚醒一個listener. 在listen函數(shù)中, 增加一個listener后, 若speaker數(shù)目非零且listener數(shù)目為1(及恰好可以配對)首先喚醒speaker, 然后將自己掛起以等待 speaker準備好數(shù)據(jù)再將自己喚醒. 這個問題其實是一個緩沖區(qū)長度為0的生產(chǎn)者/消費者問題。</p><p> 3.4.3 實現(xiàn)代碼</p><p> Communicat
85、or.java</p><p> int speakersize,listenersize;</p><p> int Message; </p><p> Condition speaker,listener;</p><p> public void speak(int word) {</p><p>&l
86、t;b> //</b></p><p> while(speakersize!=0)</p><p> {speaker.sleep();}</p><p> this.Message=word;</p><p> speakersize++;</p><p> while(listene
87、rsize==0)</p><p> {speaker.sleep();}</p><p> listener.wake();</p><p> speakersize--;</p><p> speaker.wake();</p><p><b> //</b></p>
88、<p><b> }</b></p><p> public int listen() {</p><p><b> //</b></p><p> listenersize++;</p><p> if(listenersize==1&&speakersize!
89、=0)</p><p> speaker.wake();</p><p> listener.sleep();</p><p> int myMessage = this.Message;</p><p> listenersize--;</p><p><b> //</b></
90、p><p> return myMessage;</p><p><b> }</b></p><p> 3.4.4 測試代碼及結果</p><p><b> 測試代碼:</b></p><p> package nachos.threads;</p>&
91、lt;p> import nachos.machine.*;</p><p> import java.util.ArrayList;</p><p> import java.util.Random;</p><p> public class CommunicatorTest {</p><p> public Commu
92、nicatorTest(){ </p><p> Message = 1;</p><p> numOfSpeakers = 5;</p><p> numOfListeners = 5;</p><p> communicator = new Communicator();</p><p><b>
93、 }</b></p><p> public void commTest(int num){</p><p> System.out.println("\nCommunicatorTest開始"); </p><p> for (int i=0; i<num; i++){ </p><p> c
94、reateSpeakers(numOfSpeakers);</p><p> createListeners(numOfListeners);</p><p> print("\n 演講者: " + numOfSpeakers);</p><p> print(" 收聽者: " + numOfListeners + &q
95、uot;\n");</p><p> sleep(numOfSpeakers+numOfListeners);</p><p> print("\n 演講者與收聽者創(chuàng)建完畢");</p><p><b> }</b></p><p> print(" Communic
96、ator Test結束 ");</p><p><b> }</b></p><p> public void sleep(int numThreadsCreated){</p><p> ThreadedKernel.alarm.waitUntil(numThreadsCreated*100);</p><
97、p><b> }</b></p><p> public class Listener implements Runnable{</p><p> public void run(){</p><p> int messageToRecieve = communicator.listen();</p><p&g
98、t; System.out.print(Message);</p><p> print(" " + KThread.currentThread().getName() + " 收到信息 " + messageToRecieve);</p><p><b> }</b></p><p><
99、b> }</b></p><p> public class Speaker implements Runnable{</p><p> public void run(){</p><p> communicator.speak(Message++);</p><p> System.out.print(Mess
100、age);</p><p> print(" " + KThread.currentThread().getName() + " 發(fā)出信息 " + Message); </p><p><b> }</b></p><p><b> }</b&g
101、t;</p><p> public void createSpeakers(int speakers){</p><p><b> int j;</b></p><p> for(j=1; j<=speakers; j++){ </p><p> KThread speaker
102、Thread = new KThread(new Speaker());</p><p> speakerThread.setName("Speaker" + j);</p><p> speakerThread.fork();</p><p><b> };</b></p><p><b
103、> }</b></p><p> public void createListeners(int listeners){</p><p><b> int k;</b></p><p> for(k=1; k<=listeners; k++){ </p><p> KThr
104、ead listenerThread = new KThread(new Listener());</p><p> listenerThread.setName("Listener" + k);</p><p> listenerThread.fork(); </p><p><b> }</b&
105、gt;</p><p><b> }</b></p><p> public static void print(String Message) {</p><p> System.out.println(Message);</p><p><b> }</b></p><
106、;p> public static int MAX_THREADS = 245;</p><p> private int Message; </p><p> private Communicator communicator;</p><p> private int numOfSpeakers;</p><p> pri
107、vate int numOfListeners;</p><p><b> }</b></p><p><b> 測試結果:</b></p><p> 3.5 Task1.5 priority scheduler類</p><p> 3.5.1 要求分析</p><p&g
108、t; 實現(xiàn)優(yōu)先級調(diào)度。優(yōu)先級調(diào)度的傳統(tǒng)算法如下: 每個線程擁有一個優(yōu)先級 (在Nachos中, 優(yōu)先級是一個0到7之間的整數(shù), 默認為1, 在線程調(diào)度時, 調(diào)度程序選擇一個擁有最高優(yōu)先級的處于就緒狀態(tài)的線程運行. 這種算法的問題是可能出現(xiàn) “饑餓” 現(xiàn)象: 設想有一個低優(yōu)先級的線程處于臨界區(qū)中運行而高優(yōu)先級的線程在臨界區(qū)外等待. 由于前者優(yōu)先級較低, 它可能不會被調(diào)度器選中從而高優(yōu)先級的線程也不得不浪費時間等待. 需要實現(xiàn)一種 “讓
109、出” 優(yōu)先級的機制 (Priority Donation) : 提高擁有鎖的低優(yōu)先級線程的優(yōu)先級以使它迅速完成臨界區(qū), 不使其它較高優(yōu)先級的線程等待太久. 重新計算后的優(yōu)先級稱為有效優(yōu)先級, 它可以不斷變化. 以有效優(yōu)先級為評判標準。</p><p> 3.5.2 設計方案</p><p> 要完善的的PriorityScheduler繼承自Scheduler。</p>
110、<p> 每個KThread類的線程都有一個ThreadState類型的 “線程狀態(tài)” 對象. 它保存了線程的優(yōu)先級priority, 有效優(yōu)先級effectivepriority以及正在等待該線程的線程隊列IHave,該線程正在等待的線程隊列Iwant。</p><p> 不同的調(diào)度器有不同類型的線程隊列 (均繼承自ThreadQueue), 線程隊列由threadstate對象組成,是否可提高擁
111、有鎖線程優(yōu)先級取決于創(chuàng)建隊列時transferPriority的值,它有兩個重要方法: waitForAccess (KThread thread) 和 acquire (KThread thread). 前者將一個線程加入等待隊列(實際上加入的是該線程的線程狀態(tài)); 后者將一個線程從等待隊列中取出, 表示該線程當前處于運行狀態(tài). 這兩個方法中, 首先找出相應線程的線程狀態(tài), 再對線程狀態(tài)執(zhí)行相應的waitForAccess和acqui
112、re操作 (對隊列的真正操作是在ThreadState.waitForAccess和ThreadState.acquire中完成的). </p><p> 設置, 返回, 增加, 減少優(yōu)先級的函數(shù)在Scheduler中. ThreadQueue.nextThread 返回當前隊列中有效優(yōu)先級最高的線程以運行之. 而對某線程有效優(yōu)先級的計算在ThreadState.getEffectivePriority中. 當
113、該線程沒有正在等待該線程的線程隊列時,其有效優(yōu)先級等于其原始優(yōu)先級,不然其有效優(yōu)先級增加到不小于其等待隊列IHave中各線程的有效優(yōu)先級的最大值。它也有兩個重要方法: waitForAccess (PriorityQueue waitQueue) 和 acquire (PriorityQueue waitQueue). 前者將waitqueue從IHave中移出(如果有的話),加入到Iwant中;后者將waitqueue從Iwant中移
114、出(如果有的話),加入到IHave中。</p><p> 3.5.3 實現(xiàn)代碼</p><p> 實現(xiàn)代碼見PriorityScheduler.java</p><p> 3.5.4 測試代碼及結果</p><p> 測試方法:創(chuàng)建兩個可捐贈優(yōu)先級q1,q2及一個不可捐贈優(yōu)先級隊列q3,三個優(yōu)先級分別高中低線程l,m,h。第一次將l,
115、m加到q1隊列,l,m加到q2隊列,測試q1的nextthread是m。分別測試優(yōu)先級是否“捐贈”,低優(yōu)先級線程提告優(yōu)先級后是否被選中執(zhí)行。</p><p><b> 測試代碼:</b></p><p> package nachos.threads;</p><p> import nachos.machine.*;</p>
116、<p> import nachos.threads.*;</p><p> import java.util.*;</p><p> public final class PrioritySchedulerTest {</p><p><b> /**</b></p><p> * A very
117、 simple test of priority donation.</p><p><b> */</b></p><p> public static void simplePrioritySchedulerTest() {</p><p> final boolean intStatus = Machine.interrupt().
118、disable();</p><p> final Scheduler s = new PriorityScheduler();</p><p> // Enable priority donation.</p><p> final ThreadQueue q1 = s.newThreadQueue(false);</p><p>
119、 final ThreadQueue q2 = s.newThreadQueue(false);</p><p> final ThreadQueue q3 = s.newThreadQueue(true);</p><p> final Runnable dummyRunnable = new Runnable() {</p><p> public voi
120、d run() {</p><p> // do nothing</p><p><b> }</b></p><p><b> };</b></p><p> // Create dummy Threads</p><p> final KThread lowPr
121、iorityThread = new KThread(dummyRunnable);</p><p> final KThread mediumPriorityThread = new KThread(dummyRunnable);</p><p> final KThread highPriorityThread = new KThread(dummyRunnable);</p
122、><p> // Link dummy Threads with dummy ThreadState</p><p> s.setPriority(lowPriorityThread, PriorityScheduler.priorityMinimum);</p><p> s.setPriority(mediumPriorityThread, PriorityS
123、cheduler.priorityDefault);</p><p> s.setPriority(highPriorityThread, PriorityScheduler.priorityMaximum);</p><p> // Test that effectivePriorities start as the initial priorities</p><
124、;p> Lib.assertTrue(s.getEffectivePriority(lowPriorityThread) == PriorityScheduler.priorityMinimum);</p><p> Lib.assertTrue(s.getEffectivePriority(mediumPriorityThread) == PriorityScheduler.priorityDefau
125、lt);</p><p> Lib.assertTrue(s.getEffectivePriority(highPriorityThread) == PriorityScheduler.priorityMaximum);</p><p> // Low priority and Medium priority thread are on the ready queue</p>
126、;<p> q1.waitForAccess(lowPriorityThread);</p><p> q1.waitForAccess(mediumPriorityThread);</p><p> q2.waitForAccess(lowPriorityThread);</p><p> q2.waitForAccess(mediumPri
溫馨提示
- 1. 本站所有資源如無特殊說明,都需要本地電腦安裝OFFICE2007和PDF閱讀器。圖紙軟件為CAD,CAXA,PROE,UG,SolidWorks等.壓縮文件請下載最新的WinRAR軟件解壓。
- 2. 本站的文檔不包含任何第三方提供的附件圖紙等,如果需要附件,請聯(lián)系上傳者。文件的所有權益歸上傳用戶所有。
- 3. 本站RAR壓縮包中若帶圖紙,網(wǎng)頁內(nèi)容里面會有圖紙預覽,若沒有圖紙預覽就沒有圖紙。
- 4. 未經(jīng)權益所有人同意不得將文件中的內(nèi)容挪作商業(yè)或盈利用途。
- 5. 眾賞文庫僅提供信息存儲空間,僅對用戶上傳內(nèi)容的表現(xiàn)方式做保護處理,對用戶上傳分享的文檔內(nèi)容本身不做任何修改或編輯,并不能對任何下載內(nèi)容負責。
- 6. 下載文件中如有侵權或不適當內(nèi)容,請與我們聯(lián)系,我們立即糾正。
- 7. 本站不保證下載資源的準確性、安全性和完整性, 同時也不承擔用戶因使用這些下載資源對自己和他人造成任何形式的傷害或損失。
最新文檔
- 操作系統(tǒng)程序設計課程設計報告-操作系統(tǒng)模擬實現(xiàn)
- 操作系統(tǒng)課程設計——操作系統(tǒng)課程設計模擬操作系統(tǒng)
- 操作系統(tǒng)課程設計報告
- 課程設計報告--操作系統(tǒng)
- 操作系統(tǒng)課程設計報告
- 《操作系統(tǒng)》課程設計報告
- 操作系統(tǒng)課程設計報告
- 操作系統(tǒng)課程設計報告
- 操作系統(tǒng)課程設計--模擬操作系統(tǒng)的實現(xiàn)
- 操作系統(tǒng)課程設計-- 操作系統(tǒng)
- 操作系統(tǒng)課程設計子用戶界面及托盤實現(xiàn)
- 操作系統(tǒng)課程設計報告2014217151
- 《操作系統(tǒng)原理》課程設計報告
- 操作系統(tǒng)課程設計報告2
- 操作系統(tǒng)課程設計-子用戶界面及托盤實現(xiàn)
- 操作系統(tǒng)課程設計
- 操作系統(tǒng)課程設計
- 操作系統(tǒng)課程設計
- 操作系統(tǒng)課程設計
- 操作系統(tǒng)課程設計
評論
0/150
提交評論