{Kooperatives Multithreading unter Turbo Pascal und DOS}
{(c) haftmann#software Chemnitz 1996}
{$A+,B-,D+,E-,F-,G-,I-,L+,N-,O-,P-,Q-,R-,S-,T-,V-,X+}
unit threads;
interface
const
RTE_Thread=220; {Laufzeitfehler-Code dieser Unit}
type
PThread=^TThread;
TThread=record {Thread-"Deskriptor" („hnlich wie TSS)}
next: PThread; {N„chster der Einfach Verketteten (Ring-)Liste (EVL)}
stack: Pointer; {Stackbereich von GetMem f�r FreeMem}
case integer of
1: (_sp,_ss: Word); {Speicher f�r SS und SP - mehr ist gar nicht n”tig!}
2: (sssp: Pointer);
end;
var
FirstThread: TThread; {Hauptprogramm; das Feld "stack" ist ung�ltig}
CurThread: PThread; {Zeiger auf "Deskriptor" des lfd. Threads}
function CreateThread(Entry:Pointer; Param:LongInt; Stacksize:Word):PThread;
{Einen neuen Thread erzeugen. Entry zeigt auf eine globale, FAR-codierte
Prozedur mit h”chstens einem Argument mit max. 4 Bytes Gr”áe, z.B.
kein Arg, Char, Integer, LongInt, Pointer, PChar. Der šbergabeparameter
ist Param, der ggf. zu auf LongInt zu typecasten ist. Auf diese Weise
kann ein und dieselbe Thread-Prozedur mehrfach gestartet werden. Der
Parameter Stacksize gibt die GrӇe des Stacks in Bytes an. Er wird auf
dem Heap angelegt. Achtung: Vermutlich funktioniert mit dieser Unit
die Stackpr�fung ($S) innerhalb der Threads nicht mehr!}
procedure Yield; far;
{Zum n„chsten Thread schalten. Muá aufgrund der Kooperativit„t gelegentlich -
m”glichst h„ufig - aufgerufen bzw. eingebaut werden. Zerst”rt in �blicher
Pascal-Manier alle Register auáer CS:IP,SS:SP,DS und BP. DS muá wie immer
in Pascal aufs Datensegment zeigen}
procedure _Yield; far;
{Zum n„chsten Thread schalten, ohne BP zu retten. F�r die Erweiterung zum
pr„emptiven Multitasking oder zur Rettung weiterer Register vorgesehene
Funktion. Nicht geeignet f�r Aufruf in gew”hnlichen Pascal-Programmen}
procedure KillThread(Thread:PThread);
{L”scht den angegebenen Thread. Darf nicht der aktuelle Thread und nicht
FirstThread (das Hauptprogramm) sein, sonst Laufzeitfehler!}
procedure KillCurThread;
{L”scht den momentan laufenden Thread. Darf nicht im Hauptprogramm gerufen
werden, sonst Laufzeitfehler! "Voreilige" Alternative zum Beenden des Threads
mit dem Beenden der Threadprozedur selbst}
implementation
procedure _Yield; assembler; {Zum n„chsten Thread schalten}
{Alle Register auáer DS,CS,IP,SS,SP werden zerst”rt}
{DS muá - wie immer in Pascal - ins Standard-Datensegment zeigen}
asm
les di,[CurThread]
mov es:TThread [di]._ss,ss {Stackzeiger retten}
mov es:TThread [di]._sp,sp
les di,es:TThread [di].next {N„chster Thread}
mov word [CurThread],di {Neuer aktueller Thread}
mov word [CurThread+2],es
mov ss,es:TThread [di]._ss {Stackzeiger setzen}
mov sp,es:TThread [di]._sp
end; {springt zum Thread oder zu "pop bp"}
procedure Yield; assembler; {Zum n„chsten Thread schalten}
{BP wird zus„tzlich gerettet - Pascal-freundlich}
asm
push bp
call _Yield
pop bp
end;
function CreateThread(Entry:Pointer; Param:LongInt; Stacksize:Word):PThread;
var
P: PThread;
SP: Pointer; {Stack-Ersatzzeiger}
begin
{Ende der verketteten Liste suchen}
P:=@FirstThread;
while P^.next<>@FirstThread do P:=P^.next;
New(P^.next); {Neuen EVL-Knoten erzeugen}
P:=P^.next; {Diesen adressieren}
P^.next:=@FirstThread;{N„chster Knoten ist der Anfang}
GetMem(SP,StackSize); {Stack-Speicher anfordern}
P^.stack:=SP;
Word(SP^):=StackSize; {Stack-GrӇe zur Freigabe vorn vermerken}
Inc(PChar(SP),Stacksize); {ans Stack-Speicher-Ende}
Dec(PChar(SP),sizeof(LongInt));
LongInt(SP^):=Param; {Simulate_Push Argument}
Dec(PChar(SP),sizeof(Pointer));
Pointer(SP^):=@KillCurThread; {Simulate_Push Return-CS:IP}
Dec(PChar(SP),sizeof(Pointer));
Pointer(SP^):=Entry; {Simulate_Push CS:IP}
P^.sssp:=SP; {Thread-Stackspitze abspeichern}
CreateThread:=P;
end;
procedure ThreadError;
begin
RunError(RTE_Thread);
end;
function GetPrevThread(Thread:PThread):PThread;
var
P:PThread;
begin
P:=@FirstThread;
repeat
if P^.next=nil then ThreadError; {Thread-Zeigerfehler}
if P^.next=@FirstThread then ThreadError; {Ende der Liste}
if P^.next=Thread then break;
P:=P^.next;
until false;
GetPrevThread:=P;
end;
procedure KillThread(Thread:PThread);
begin
if Thread=CurThread then ThreadError; {Nicht f�r laufenden Thread!}
if Thread=@FirstThread then ThreadError; {Nicht f�r Hauptprogramm}
GetPrevThread(Thread)^.next:=Thread^.next; {Aktuellen Thread aush„ngen}
FreeMem(Thread^.stack,Word(Thread^.stack^)); {Speicher freigeben}
Dispose(Thread); {EVL-Knoten freigeben}
end;
procedure KillCurThread; assembler;
asm
les bx,[CurThread] {Retten des Stackpointers kann entfallen}
mov cx,es {Zeiger retten (nicht auf dem Stack!)}
les di,es:TThread [bx].next {N„chster Thread}
mov word [CurThread],di {Neuer aktueller Thread}
mov word [CurThread+2],es
mov ss,es:TThread [di]._ss {Stackzeiger setzen}
mov sp,es:TThread [di]._sp
push cx
push bx {Thread, nun nicht mehr CurThread!}
call KillThread
end;
{Initialisierungsteil}
begin
FirstThread.next:=@FirstThread;{das Hauptprogramm allein}
CurThread:=@FirstThread; {Momentan l„uft nur dieses}
end.
Detected encoding: ANSI (CP1252) | 4
|
|