add vorgabe06
This commit is contained in:
@ -20,3 +20,4 @@ unsigned int total_mem; // RAM total
|
||||
// BumpAllocator allocator;
|
||||
// LinkedListAllocator allocator;
|
||||
TreeAllocator allocator;
|
||||
Scheduler scheduler;
|
||||
|
||||
@ -19,6 +19,7 @@
|
||||
#include "kernel/CPU.h"
|
||||
#include "kernel/interrupts/IntDispatcher.h"
|
||||
#include "kernel/interrupts/PIC.h"
|
||||
#include "kernel/threads/Scheduler.h"
|
||||
|
||||
extern CPU cpu; // CPU-spezifische Funktionen
|
||||
extern PCSPK pcspk; // PC-Lautsprecher
|
||||
@ -30,6 +31,7 @@ extern unsigned int total_mem; // RAM total
|
||||
// extern BumpAllocator allocator;
|
||||
// extern LinkedListAllocator allocator;
|
||||
extern TreeAllocator allocator;
|
||||
extern Scheduler scheduler;
|
||||
|
||||
constexpr bool DEBUG = true;
|
||||
|
||||
|
||||
48
c_os/kernel/threads/Dispatch.cc
Executable file
48
c_os/kernel/threads/Dispatch.cc
Executable file
@ -0,0 +1,48 @@
|
||||
/*****************************************************************************
|
||||
* *
|
||||
* D I S P A T C H E R *
|
||||
* *
|
||||
*---------------------------------------------------------------------------*
|
||||
* Beschreibung: Implementierung des Dispatchers. *
|
||||
* Der Dispatcher verwaltet den 'active'-Pointer, der den *
|
||||
* jeweils aktiven Thread angibt. Mit 'start()' wird der *
|
||||
* 'active' Pointer initialisiert und der erste Thread ge- *
|
||||
* startet, alle weiteren Kontextwechsel werden mit *
|
||||
* 'dispatch()' ausgeloest. 'get_active()' liefert den *
|
||||
* 'active' Pointer zurueck. *
|
||||
* *
|
||||
* Autor: Olaf Spinczyk, TU Dortmund *
|
||||
*****************************************************************************/
|
||||
|
||||
#include "kernel/threads/Dispatch.h"
|
||||
|
||||
|
||||
/*****************************************************************************
|
||||
* Methode: Dispatcher::start *
|
||||
*---------------------------------------------------------------------------*
|
||||
* Beschreibung: Thread starten. *
|
||||
* *
|
||||
* Parameter: *
|
||||
* first Zu startender Thread. *
|
||||
*****************************************************************************/
|
||||
void Dispatcher::start (Thread& first) {
|
||||
if (!active) {
|
||||
active = &first;
|
||||
active->start ();
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
/*****************************************************************************
|
||||
* Methode: Dispatcher::dispatch *
|
||||
*---------------------------------------------------------------------------*
|
||||
* Beschreibung: Auf einen gegebenen Thread wechseln. *
|
||||
* *
|
||||
* Parameter: *
|
||||
* next Thread der die CPU erhalten soll. *
|
||||
*****************************************************************************/
|
||||
void Dispatcher::dispatch (Thread& next) {
|
||||
Thread* current = active;
|
||||
active = &next;
|
||||
current->switchTo (next);
|
||||
}
|
||||
35
c_os/kernel/threads/Dispatch.h
Executable file
35
c_os/kernel/threads/Dispatch.h
Executable file
@ -0,0 +1,35 @@
|
||||
/*****************************************************************************
|
||||
* *
|
||||
* D I S P A T C H E R *
|
||||
* *
|
||||
*---------------------------------------------------------------------------*
|
||||
* Beschreibung: Implementierung des Dispatchers. *
|
||||
* Der Dispatcher verwaltet den 'active'-Pointer, der den *
|
||||
* jeweils aktiven Thread angibt. Mit 'start()' wird der *
|
||||
* 'active' Pointer initialisiert und der erste Thread ge- *
|
||||
* startet, alle weiteren Kontextwechsel werden mit *
|
||||
* 'dispatch()' ausgeloest. 'get_active()' liefert den *
|
||||
* 'active' Pointer zurueck. *
|
||||
* *
|
||||
* Autor: Olaf Spinczyk, TU Dortmund *
|
||||
*****************************************************************************/
|
||||
#ifndef __dispatch_include__
|
||||
#define __dispatch_include__
|
||||
|
||||
#include "kernel/threads/Thread.h"
|
||||
|
||||
class Dispatcher {
|
||||
|
||||
private:
|
||||
Thread* active; // aktiver Thread
|
||||
|
||||
Dispatcher(const Dispatcher ©); // Verhindere Kopieren
|
||||
|
||||
public:
|
||||
Dispatcher () : active (0) {}
|
||||
void start (Thread& first);
|
||||
void dispatch (Thread& next);
|
||||
Thread* get_active () { return active; }
|
||||
};
|
||||
|
||||
#endif
|
||||
35
c_os/kernel/threads/IdleThread.h
Executable file
35
c_os/kernel/threads/IdleThread.h
Executable file
@ -0,0 +1,35 @@
|
||||
/*****************************************************************************
|
||||
* *
|
||||
* I D L E T H R E A D *
|
||||
* *
|
||||
*---------------------------------------------------------------------------*
|
||||
* Beschreibung: Wird nur aktiviert, wenn kein Thread arbeiten moechte. *
|
||||
* *
|
||||
* Autor: Michael, Schoettner, HHU, 13.8.2016 *
|
||||
*****************************************************************************/
|
||||
|
||||
#ifndef __IdleThread_include__
|
||||
#define __IdleThread_include__
|
||||
|
||||
|
||||
#include "kernel/threads/Thread.h"
|
||||
#include "kernel/Globals.h"
|
||||
|
||||
|
||||
class IdleThread : public Thread {
|
||||
|
||||
private:
|
||||
IdleThread(const Thread ©); // Verhindere Kopieren
|
||||
|
||||
public:
|
||||
IdleThread () : Thread () { }
|
||||
|
||||
void run () {
|
||||
while (1) {
|
||||
scheduler.yield ();
|
||||
}
|
||||
}
|
||||
|
||||
};
|
||||
|
||||
#endif
|
||||
95
c_os/kernel/threads/Scheduler.cc
Executable file
95
c_os/kernel/threads/Scheduler.cc
Executable file
@ -0,0 +1,95 @@
|
||||
/*****************************************************************************
|
||||
* *
|
||||
* S C H E D U L E R *
|
||||
* *
|
||||
*---------------------------------------------------------------------------*
|
||||
* Beschreibung: Implementierung eines einfachen Zeitscheiben-Schedulers. *
|
||||
* Rechenbereite Threads werden in 'readQueue' verwaltet. *
|
||||
* *
|
||||
* Autor: Michael, Schoettner, HHU, 23.11.2018 *
|
||||
*****************************************************************************/
|
||||
|
||||
#include "kernel/threads/Scheduler.h"
|
||||
#include "kernel/threads/IdleThread.h"
|
||||
|
||||
|
||||
|
||||
/*****************************************************************************
|
||||
* Methode: Scheduler::schedule *
|
||||
*---------------------------------------------------------------------------*
|
||||
* Beschreibung: Scheduler starten. Wird nur einmalig aus main.cc gerufen.*
|
||||
*****************************************************************************/
|
||||
void Scheduler::schedule () {
|
||||
|
||||
/* hier muss Code eingefuegt werden */
|
||||
|
||||
/* Bevor diese Methode anufgerufen wird, muss zumindest der Idle-Thread
|
||||
* in der Queue eingefuegt worden sein.
|
||||
*/
|
||||
}
|
||||
|
||||
|
||||
/*****************************************************************************
|
||||
* Methode: Scheduler::ready *
|
||||
*---------------------------------------------------------------------------*
|
||||
* Beschreibung: Thread in readyQueue eintragen. *
|
||||
* *
|
||||
* Parameter: *
|
||||
* that Einzutragender Thread *
|
||||
*****************************************************************************/
|
||||
void Scheduler::ready (Thread * that) {
|
||||
|
||||
/* hier muss Code eingefuegt werden */
|
||||
|
||||
}
|
||||
|
||||
|
||||
/*****************************************************************************
|
||||
* Methode: Scheduler::exit *
|
||||
*---------------------------------------------------------------------------*
|
||||
* Beschreibung: Thread ist fertig und terminiert sich selbst. Hier muss *
|
||||
* nur auf den naechsten Thread mithilfe des Dispatchers *
|
||||
* umgeschaltet werden. Der aktuell laufende Thread ist *
|
||||
* nicht in der readyQueue. *
|
||||
*****************************************************************************/
|
||||
void Scheduler::exit () {
|
||||
|
||||
/* hier muss Code eingefuegt werden */
|
||||
|
||||
}
|
||||
|
||||
|
||||
/*****************************************************************************
|
||||
* Methode: Scheduler::kill *
|
||||
*---------------------------------------------------------------------------*
|
||||
* Beschreibung: Thread mit 'Gewalt' terminieren. Er wird aus der *
|
||||
* readyQueue ausgetragen und wird dann nicht mehr aufge- *
|
||||
* rufen. Der Aufrufer dieser Methode muss ein anderer *
|
||||
* Thread sein. *
|
||||
* *
|
||||
* Parameter: *
|
||||
* that Zu terminierender Thread *
|
||||
*****************************************************************************/
|
||||
void Scheduler::kill (Thread * that) {
|
||||
|
||||
/* hier muss Code eingefuegt werden */
|
||||
|
||||
}
|
||||
|
||||
|
||||
/*****************************************************************************
|
||||
* Methode: Scheduler::yield *
|
||||
*---------------------------------------------------------------------------*
|
||||
* Beschreibung: CPU freiwillig abgeben und Auswahl des naechsten Threads.*
|
||||
* Naechsten Thread aus der readyQueue holen, den aktuellen *
|
||||
* in die readyQueue wieder eintragen. Das Umschalten soll *
|
||||
* mithilfe des Dispatchers erfolgen. *
|
||||
* *
|
||||
* Achtung: Falls nur der Idle-Thread läuft, so ist die *
|
||||
* readyQueue leer. *
|
||||
*****************************************************************************/
|
||||
void Scheduler::yield () {
|
||||
|
||||
/* hier muss Code eingefuegt werden */
|
||||
|
||||
}
|
||||
46
c_os/kernel/threads/Scheduler.h
Executable file
46
c_os/kernel/threads/Scheduler.h
Executable file
@ -0,0 +1,46 @@
|
||||
/*****************************************************************************
|
||||
* *
|
||||
* S C H E D U L E R *
|
||||
* *
|
||||
*---------------------------------------------------------------------------*
|
||||
* Beschreibung: Implementierung eines einfachen Zeitscheiben-Schedulers. *
|
||||
* Rechenbereite Threads werden in 'readQueue' verwaltet. *
|
||||
* *
|
||||
* Autor: Michael, Schoettner, HHU, 22.8.2016 *
|
||||
*****************************************************************************/
|
||||
|
||||
#ifndef __Scheduler_include__
|
||||
#define __Scheduler_include__
|
||||
|
||||
#include "kernel/threads/Dispatch.h"
|
||||
#include "kernel/threads/Thread.h"
|
||||
#include "lib/Queue.h"
|
||||
|
||||
class Scheduler : public Dispatcher {
|
||||
|
||||
private:
|
||||
Scheduler (const Scheduler ©); // Verhindere Kopieren
|
||||
|
||||
private:
|
||||
Queue readyQueue; // auf die CPU wartende Threads
|
||||
|
||||
public:
|
||||
Scheduler () {}
|
||||
|
||||
// Scheduler starten
|
||||
void schedule ();
|
||||
|
||||
// Thread in readyQueue eintragen
|
||||
void ready (Thread * that);
|
||||
|
||||
// Thread terminiert sich selbst
|
||||
void exit ();
|
||||
|
||||
// Thread mit 'Gewalt' terminieren
|
||||
void kill (Thread * that);
|
||||
|
||||
// CPU freiwillig abgeben und Auswahl des naechsten Threads
|
||||
void yield ();
|
||||
};
|
||||
|
||||
#endif
|
||||
54
c_os/kernel/threads/Thread.h
Executable file
54
c_os/kernel/threads/Thread.h
Executable file
@ -0,0 +1,54 @@
|
||||
/*****************************************************************************
|
||||
* *
|
||||
* T H R E A D *
|
||||
* *
|
||||
*---------------------------------------------------------------------------*
|
||||
* Beschreibung: Implementierung eines kooperativen Thread-Konzepts. *
|
||||
* Thread-Objekte werden vom Scheduler in einer verketteten *
|
||||
* Liste 'readylist' verwaltet. *
|
||||
* *
|
||||
* Im Konstruktor wird der initialie Kontext des Threads *
|
||||
* eingerichtet. Mit 'start' wird ein Thread aktiviert. *
|
||||
* Die CPU sollte mit 'yield' freiwillig abgegeben werden. *
|
||||
* Um bei einem Threadwechsel den Kontext sichern zu *
|
||||
* koennen, enthaelt jedes Threadobjekt eine Struktur *
|
||||
* ThreadState, in dem die Werte der nicht-fluechtigen *
|
||||
* Register gesichert werden koennen. *
|
||||
* *
|
||||
* Autor: Michael, Schoettner, HHU, 16.12.2016 *
|
||||
*****************************************************************************/
|
||||
|
||||
#ifndef __Thread_include__
|
||||
#define __Thread_include__
|
||||
|
||||
#include "kernel/threads/ThreadState.h"
|
||||
#include "lib/Chain.h"
|
||||
|
||||
class Thread : public Chain {
|
||||
|
||||
private:
|
||||
Thread(const Thread ©); // Verhindere Kopieren
|
||||
|
||||
private:
|
||||
struct ThreadState regs;
|
||||
unsigned int *stack;
|
||||
|
||||
public:
|
||||
unsigned int tid; // Thread-ID (wird im Konstruktor vergeben)
|
||||
|
||||
Thread () { }
|
||||
|
||||
// Thread aktivieren
|
||||
void start ();
|
||||
|
||||
// Umschalten auf Thread 'next'
|
||||
void switchTo (Thread& next);
|
||||
|
||||
// Methode des Threads, muss in Sub-Klasse implementiert werden
|
||||
virtual void run () = 0;
|
||||
|
||||
~Thread ();
|
||||
|
||||
};
|
||||
|
||||
#endif
|
||||
44
c_os/lib/Queue.h
Executable file
44
c_os/lib/Queue.h
Executable file
@ -0,0 +1,44 @@
|
||||
/*****************************************************************************
|
||||
* *
|
||||
* Q U E U E *
|
||||
* *
|
||||
*---------------------------------------------------------------------------*
|
||||
* Beschreibung: Implementierung einer einfach verketteten Liste von *
|
||||
* Chain Objekten. Die Implementierung ist etwas trickreich *
|
||||
* 'tail' verweist naemlich nicht, wie oft ueblich, auf das *
|
||||
* letzte Element der Liste, sondern auf den 'next' Zeiger *
|
||||
* des letzten Elements, bzw., solange die Liste noch leer *
|
||||
* ist, auf den 'head' Zeiger der Liste. Dadurch muss beim *
|
||||
* Anfuegen eines Elements an die Liste nicht ueberprueft *
|
||||
* werden, ob bereits Elemente in ihr enthalten sind. Beim *
|
||||
* Entfernen von Elementen kann auf die Fallunterscheidung *
|
||||
* allerdings nicht verzichtet werden. *
|
||||
* *
|
||||
* Autor: Olaf Spinczyk, TU Dortmund *
|
||||
*****************************************************************************/
|
||||
|
||||
#ifndef __Queue_include__
|
||||
#define __Queue_include__
|
||||
|
||||
#include "lib/Chain.h"
|
||||
|
||||
class Queue {
|
||||
|
||||
private:
|
||||
Queue(const Queue& copy); // Verhindere Kopieren
|
||||
|
||||
protected:
|
||||
Chain* head;
|
||||
Chain** tail;
|
||||
|
||||
public:
|
||||
Queue() {
|
||||
head = 0;
|
||||
tail = &head;
|
||||
}
|
||||
void enqueue(Chain* item);
|
||||
Chain* dequeue();
|
||||
void remove(Chain*);
|
||||
};
|
||||
|
||||
#endif
|
||||
26
c_os/user/CoopThreadDemo.cc
Executable file
26
c_os/user/CoopThreadDemo.cc
Executable file
@ -0,0 +1,26 @@
|
||||
/*****************************************************************************
|
||||
* *
|
||||
* C O O P E R A T I V E T H R E A D D E M O *
|
||||
* *
|
||||
*---------------------------------------------------------------------------*
|
||||
* Beschreibung: Beispiel für kooperative Threads. *
|
||||
* *
|
||||
* Autor: Michael Schoettner, HHU, 21.8.2016 *
|
||||
*****************************************************************************/
|
||||
|
||||
#include "kernel/Globals.h"
|
||||
#include "user/CoopThreadDemo.h"
|
||||
#include "user/LoopThread.h"
|
||||
|
||||
|
||||
/*****************************************************************************
|
||||
* Methode: CoopThreadDemo::run *
|
||||
*---------------------------------------------------------------------------*
|
||||
* Beschreibung: Der Anwendungsthread erzeugt drei Threads die Zaehler *
|
||||
* ausgeben und terminiert sich selbst. *
|
||||
*****************************************************************************/
|
||||
void CoopThreadDemo::run () {
|
||||
|
||||
/* Hier muss Code eingefuegt werden */
|
||||
|
||||
}
|
||||
31
c_os/user/CoopThreadDemo.h
Executable file
31
c_os/user/CoopThreadDemo.h
Executable file
@ -0,0 +1,31 @@
|
||||
/*****************************************************************************
|
||||
* *
|
||||
* C O O P E R A T I V E T H R E A D D E M O *
|
||||
* *
|
||||
*---------------------------------------------------------------------------*
|
||||
* Beschreibung: Beispiel für kooperative Threads. *
|
||||
* *
|
||||
* Autor: Michael Schoettner, HHU, 21.8.2016 *
|
||||
*****************************************************************************/
|
||||
#ifndef __coopthreaddemo_include__
|
||||
#define __coopthreaddemo_include__
|
||||
|
||||
|
||||
#include "kernel/threads/Thread.h"
|
||||
|
||||
|
||||
class CoopThreadDemo : public Thread {
|
||||
|
||||
private:
|
||||
CoopThreadDemo (const CoopThreadDemo ©); // Verhindere Kopieren
|
||||
|
||||
public:
|
||||
// Gib dem Anwendungsthread einen Stack.
|
||||
CoopThreadDemo () : Thread () { }
|
||||
|
||||
// Thread-Startmethode
|
||||
void run ();
|
||||
|
||||
};
|
||||
|
||||
#endif
|
||||
27
c_os/user/HelloWorldThread.cc
Executable file
27
c_os/user/HelloWorldThread.cc
Executable file
@ -0,0 +1,27 @@
|
||||
/*****************************************************************************
|
||||
* *
|
||||
* H E L L O W O R L D T H R E A D *
|
||||
* *
|
||||
*---------------------------------------------------------------------------*
|
||||
* Beschreibung: Ein einfacher Thread. *
|
||||
* *
|
||||
* Autor: Michael Schoettner, HHU, 21.8.2016 *
|
||||
*****************************************************************************/
|
||||
|
||||
#include "kernel/Globals.h"
|
||||
#include "user/HelloWorldThread.h"
|
||||
|
||||
|
||||
|
||||
/*****************************************************************************
|
||||
* Methode: HelloWorldThread::run *
|
||||
*---------------------------------------------------------------------------*
|
||||
* Beschreibung: Einstiegsfunktion in unseren Thread. *
|
||||
*****************************************************************************/
|
||||
void HelloWorldThread::run () {
|
||||
|
||||
kout << "Hallo Welt von einem Thread!" << endl;
|
||||
|
||||
// selbst terminieren
|
||||
scheduler.exit ();
|
||||
}
|
||||
30
c_os/user/HelloWorldThread.h
Executable file
30
c_os/user/HelloWorldThread.h
Executable file
@ -0,0 +1,30 @@
|
||||
/*****************************************************************************
|
||||
* *
|
||||
* H E L L O W O R L D T H R E A D *
|
||||
* *
|
||||
*---------------------------------------------------------------------------*
|
||||
* Beschreibung: Ein einfacher Thread. *
|
||||
* *
|
||||
* Autor: Michael Schoettner, HHU, 16.12.2016 *
|
||||
*****************************************************************************/
|
||||
#ifndef __hello_world_thread_include__
|
||||
#define __hello_world_thread_include__
|
||||
|
||||
|
||||
#include "kernel/threads/Thread.h"
|
||||
|
||||
|
||||
class HelloWorldThread : public Thread {
|
||||
|
||||
private:
|
||||
HelloWorldThread (const HelloWorldThread ©); // Verhindere Kopieren
|
||||
|
||||
public:
|
||||
HelloWorldThread () : Thread () { }
|
||||
|
||||
// Thread-Startmethode
|
||||
void run ();
|
||||
|
||||
};
|
||||
|
||||
#endif
|
||||
22
c_os/user/LoopThread.cc
Executable file
22
c_os/user/LoopThread.cc
Executable file
@ -0,0 +1,22 @@
|
||||
/*****************************************************************************
|
||||
* *
|
||||
* L O O P T H R E A D *
|
||||
* *
|
||||
*---------------------------------------------------------------------------*
|
||||
* Beschreibung: Demo eines Threads. Schleife die Zahlen ausgibt. *
|
||||
*****************************************************************************/
|
||||
|
||||
#include "user/LoopThread.h"
|
||||
#include "kernel/Globals.h"
|
||||
|
||||
|
||||
/*****************************************************************************
|
||||
* Methode: LoopThread::run *
|
||||
*---------------------------------------------------------------------------*
|
||||
* Beschreibung: Code des Threads. *
|
||||
*****************************************************************************/
|
||||
void LoopThread::run () {
|
||||
|
||||
/* Hier muss Code eingefuegt werden */
|
||||
|
||||
}
|
||||
29
c_os/user/LoopThread.h
Executable file
29
c_os/user/LoopThread.h
Executable file
@ -0,0 +1,29 @@
|
||||
/*****************************************************************************
|
||||
* *
|
||||
* L O O P T H R E A D *
|
||||
* *
|
||||
*---------------------------------------------------------------------------*
|
||||
* Beschreibung: Demo eines Threads. Schleife die Zahlen ausgibt. *
|
||||
*****************************************************************************/
|
||||
|
||||
#ifndef __loopthread_include__
|
||||
#define __loopthread_include__
|
||||
|
||||
#include "kernel/threads/Thread.h"
|
||||
|
||||
class LoopThread : public Thread {
|
||||
|
||||
private:
|
||||
int id;
|
||||
|
||||
LoopThread (const LoopThread ©); // Verhindere Kopieren
|
||||
|
||||
public:
|
||||
// Gibt der Loop einen Stack und eine Id.
|
||||
LoopThread (int i) : Thread () { id = i; }
|
||||
|
||||
// Zaehlt einen Zaehler hoch und gibt ihn auf dem Bildschirm aus.
|
||||
void run ();
|
||||
};
|
||||
|
||||
#endif
|
||||
Reference in New Issue
Block a user