@@ -23,13 +23,13 @@ float MyFuncRetZero();
23
23
24
24
#define TIMESTRING1 "Process %d has run for"
25
25
#define TIMESTRING2 " %.3f s\n"
26
- #define TIMESTRING3 "Process %d 's priority is %d \n"
26
+ #define TIMESTRING3 "Process %d 's priority is %d\n"
27
27
28
28
#define TOTAL_QUANTA_MAX 10
29
29
#define PROCESS_QUANTA_MAX 4
30
30
31
- #define FALSE 0
32
31
#define TRUE 1
32
+ #define FALSE 0
33
33
34
34
// Pointer to the current PCB. This is used by the assembly language
35
35
// routines for context switches.
@@ -63,9 +63,8 @@ static processQuantum = DLX_PROCESS_QUANTUM;
63
63
// String listing debugging options to print out.
64
64
char debugstr [200 ];
65
65
66
- // BEGIN BRIAN CODE
67
66
static uint32 totalQuanta = 0 ;
68
- // END BRIAN CODE
67
+ static uint32 startTime = 0 ;
69
68
70
69
uint32 my_timer_get () {
71
70
return totalQuanta * 100 + totalQuanta ;
@@ -231,18 +230,34 @@ ProcessSchedule ()
231
230
}*/
232
231
dbprintf ('p' , "Entering ProcessSchedule [context switch] with current PCB: %p\n" ,currentPCB );
233
232
234
- currentPCB -> p_quanta ++ ; // do this here or later?
235
- currentPCB -> estcpu ++ ; // do this here or later?
233
+ currentPCB -> p_quanta ++ ;
236
234
237
235
totalQuanta ++ ;
238
236
239
237
pcb = ProcessHighestPriority ();
240
238
239
+ currentPCB -> runtime += my_timer_get () - startTime ;
240
+
241
+ if (currentPCB -> p_info == 1 ) {
242
+ printf (TIMESTRING1 , currentPCB - pcbs );
243
+ printf (TIMESTRING2 , currentPCB -> runtime / (float )1000 );
244
+ printf (TIMESTRING3 , currentPCB - pcbs , currentPCB -> prio );
245
+ }
246
+
247
+ startTime = my_timer_get ();
248
+
249
+ if (!pcb ) {
250
+ printf ("No runnable processes - exiting!\n" );
251
+ exitsim ();
252
+ }
253
+
241
254
dbprintf ('p' , "PCB (%p) currentPCB (%p)\n" ,pcb ,currentPCB );
242
255
243
256
244
257
// If last run process is still the highest priority (ie. not asleep/a zombie)
245
258
if (pcb == currentPCB ) {
259
+ currentPCB -> estcpu ++ ;
260
+
246
261
QueueRemove (& pcb -> l );
247
262
QueueInsertLast (& runQueue [pcb -> runQueueNum ], & pcb -> l );
248
263
@@ -270,10 +285,10 @@ ProcessSchedule ()
270
285
dbprintf ('p' , "Last links registered\n" );
271
286
272
287
for (i = 0 ; i < 32 ; i ++ ) {
273
- // if(QueueEmpty(&runQueue[i])) atEndOfQueue = TRUE;
288
+ // if(QueueEmpty(&runQueue[i])) atEndOfQueue = TRUE;
274
289
n = (& runQueue [i ])-> nitems ;
275
290
276
- // while(!atEndOfQueue) {
291
+ // while(!atEndOfQueue) {
277
292
for (j = 0 ; j < n ; j ++ ) {
278
293
pcb = (PCB * )((QueueFirst (& runQueue [i ]))-> object );
279
294
pcb -> estcpu = (int )((((float )2 * pcb -> load )/((float )2 * pcb -> load + 1 )) * pcb -> estcpu ) + pcb -> p_nice ; // Decay the estimated CPU time of all processes
@@ -294,39 +309,45 @@ ProcessSchedule ()
294
309
QueueRemove (& currentPCB -> l );
295
310
QueueInsertLast (& runQueue [currentPCB -> runQueueNum ], & currentPCB -> l );
296
311
}
297
- }
298
- //}
299
-
300
- currentPCB = ProcessHighestPriority ();
301
-
302
- // Move the front of the queue to the end, if it is the running process.
303
- /*
304
- pcb = (PCB *)((QueueFirst (&runQueue))->object);
305
- if (pcb == currentPCB)
306
- {
307
- QueueRemove (&pcb->l);
308
- QueueInsertLast (&runQueue, &pcb->l);
309
- }
310
-
311
- // Now, run the one at the head of the queue.
312
- pcb = (PCB *)((QueueFirst (&runQueue))->object);
313
- currentPCB = pcb;
314
- dbprintf ('p',"About to switch to PCB 0x%x,flags=0x%x @ 0x%x\n",
315
- pcb, pcb->flags,
316
- pcb->sysStackPtr[PROCESS_STACK_IAR]);
317
- */
318
-
319
- // Clean up zombie processes here. This is done at interrupt time
320
- // because it can't be done while the process might still be running
321
- while (!QueueEmpty (& zombieQueue )) {
322
- pcb = (PCB * )(QueueFirst (& zombieQueue )-> object );
323
- dbprintf ('p' , "Freeing zombie PCB 0x%x.\n" , pcb );
324
- QueueRemove (& pcb -> l );
325
- ProcessFreeResources (pcb );
326
- }
327
- // Set the timer so this process gets at most a fixed quantum of time.
328
- TimerSet (processQuantum );
329
- dbprintf ('p' , "Leaving ProcessSchedule (cur=0x%x)\n" , currentPCB );
312
+ }
313
+ //}
314
+
315
+ pcb = ProcessHighestPriority ();
316
+
317
+ // }
318
+
319
+ currentPCB = pcb ;
320
+
321
+ // currentPCB = ProcessHighestPriority();
322
+
323
+ // Move the front of the queue to the end, if it is the running process.
324
+ /*
325
+ pcb = (PCB *)((QueueFirst (&runQueue))->object);
326
+ if (pcb == currentPCB)
327
+ {
328
+ QueueRemove (&pcb->l);
329
+ QueueInsertLast (&runQueue, &pcb->l);
330
+ }
331
+
332
+ // Now, run the one at the head of the queue.
333
+ pcb = (PCB *)((QueueFirst (&runQueue))->object);
334
+ currentPCB = pcb;
335
+ dbprintf ('p',"About to switch to PCB 0x%x,flags=0x%x @ 0x%x\n",
336
+ pcb, pcb->flags,
337
+ pcb->sysStackPtr[PROCESS_STACK_IAR]);
338
+ */
339
+
340
+ // Clean up zombie processes here. This is done at interrupt time
341
+ // because it can't be done while the process might still be running
342
+ while (!QueueEmpty (& zombieQueue )) {
343
+ pcb = (PCB * )(QueueFirst (& zombieQueue )-> object );
344
+ dbprintf ('p' , "Freeing zombie PCB 0x%x.\n" , pcb );
345
+ QueueRemove (& pcb -> l );
346
+ ProcessFreeResources (pcb );
347
+ }
348
+ // Set the timer so this process gets at most a fixed quantum of time.
349
+ TimerSet (processQuantum );
350
+ dbprintf ('p' , "Leaving ProcessSchedule (cur=0x%x)\n" , currentPCB );
330
351
}
331
352
332
353
//----------------------------------------------------------------------
@@ -370,13 +391,34 @@ ProcessSuspend (PCB *suspend)
370
391
void
371
392
ProcessWakeup (PCB * wakeup )
372
393
{
394
+ int i , sleeptime_inseconds ;
395
+ float temp_estcpu , power ;
396
+
373
397
dbprintf ('p' ,"Waking up PCB 0x%x.\n" , wakeup );
374
398
// Make sure it's not yet a runnable process.
375
399
ASSERT (wakeup -> flags & PROCESS_STATUS_WAITING ,
376
400
"Trying to wake up a non-sleeping process!\n" );
401
+
402
+ sleeptime_inseconds = (my_timer_get () - wakeup -> sleeptime )/1000 ;
403
+
404
+ // Adjust the estimated CPU time
405
+ if (sleeptime_inseconds >= 1 ) {
406
+ temp_estcpu = ((((float )2 * wakeup -> load )/((float )2 * wakeup -> load + 1 )) * wakeup -> estcpu ); // Base
407
+ power = temp_estcpu ;
408
+ for (i = 1 ; i < sleeptime_inseconds ; i ++ ) {
409
+ temp_estcpu *= power ; // To the power of sleeptime_inseconds
410
+ }
411
+ wakeup -> estcpu = (int )temp_estcpu ;
412
+ }
413
+
414
+ // Recalculate priority
415
+ wakeup -> prio = PUSER + (wakeup -> estcpu /4 ) + (2 * wakeup -> p_nice );
416
+ wakeup -> runQueueNum = wakeup -> prio /4 ;
417
+
418
+ // Put the process on a run queue based on Priority
377
419
ProcessSetStatus (wakeup , PROCESS_STATUS_RUNNABLE );
378
420
QueueRemove (& wakeup -> l );
379
- QueueInsertLast (& runQueue , & wakeup -> l );
421
+ QueueInsertLast (& runQueue [ wakeup -> runQueueNum ] , & wakeup -> l );
380
422
381
423
}
382
424
@@ -529,20 +571,19 @@ ProcessFork (VoidFunc func, uint32 param, int p_nice, int p_info,char *name, int
529
571
530
572
//---------------------------------------
531
573
// Lab3: initialized pcb member for your scheduling algorithm here
532
- // BEGIN BRIAN CODE
533
574
if ((isUser && p_nice < 0 ) || p_nice > 19 ) { // p_nice should never be greater than 19
534
575
pcb -> p_nice = 0 ;
535
576
}
536
- else pcb -> p_nice = p_nice ;
537
- pcb -> estcpu = 0 ;
538
- pcb -> runtime = 0 ;
539
- pcb -> sleeptime = 0 ;
540
- pcb -> prio = PUSER ;
541
- pcb -> runQueueNum = ( pcb -> prio )/ 4 ;
542
- pcb -> load = 1 ;
543
- pcb -> p_info = p_info ;
544
- pcb -> p_quanta = 0 ;
545
- // END BRIAN CODE
577
+ else pcb -> p_nice = p_nice ;
578
+ pcb -> estcpu = 0 ;
579
+ pcb -> runtime = 0 ;
580
+ // pcb->runtime_lastrun = -1; // -1 to show hasn't been run before
581
+ pcb -> sleeptime = 0 ;
582
+ pcb -> prio = PUSER ;
583
+ pcb -> runQueueNum = ( pcb -> prio )/ 4 ;
584
+ pcb -> load = 1 ;
585
+ pcb -> p_info = p_info ;
586
+ pcb -> p_quanta = 0 ;
546
587
//--------------------------------------
547
588
548
589
0 commit comments