2020-09-30 16:53:55 +02:00

177 lines
4.1 KiB
NASM

;
;
; Copyright (C) Microsoft Corporation, 1987
;
; This Module contains Proprietary Information of Microsoft
; Corporation and should be treated as Confidential.
;
subttl emthread.asm - Emulator multi-thread support for OS/2
page
thread1static=1 ; set DS = EMULATOR_DATA for thread 1
; dynamically allocate data areas for other threads
_DATA segment word public 'DATA'
_DATA ends
DGROUP group _DATA
_DATA segment word public 'DATA'
extrn __FPDSARRAY:WORD
ifdef i386
extrn __threadid:FWORD ; far pointer to ???? thread id
else
extrn __threadid:DWORD ; far pointer to WORD thread id
endif
_DATA ends
dataoffset equ offset DGROUP:
include os2dll.inc ; defines _SIGNAL_LOCK
; and other lock values;
; must be synchronized with
; \clib\include\86\os2dll.inc
; and \clib\include\os2dll.h
extrn __lockf:FAR
extrn __unlockf:FAR
LOAD_DS_EDI macro
local TIDOk, LoadDone
;
; loads ds:edi with far pointer to thread's DS selector
; (pointer into __FPDSARRAY)
;
; uses eax,edi,es,ds
;
; sets eax = 2 * (thread id)
;
mov ax,DGROUP
mov ds,ax ; set DS = DGROUP temporarily
assume ds:DGROUP
les edi,[__threadid] ; ES:eDI = far pointer to thread id
mov ax,es:[edi] ; AX = thread id (far ptr to WORD)
cmp ax, MAXTHREADID
jbe TIDOk
push ax
call LoadDS_EDI
pop ax
shl ax, 1 ; thread id times 2
jmp short LoadDone
TIDOk:
shl eax,1 ; thread id times 2
mov edi,dataoffset __FPDSARRAY
add edi,eax ; index into __FPDSARRAY
LoadDone:
endm
LOADthreadDS macro
;
; loads thread's DS from __FPDSARRAY indexed by thread id
; preserves all registers except DS and eAX
;
; __FPDSARRAY[0] = MAXTHREAD
; __FPDSARRAY[i] = emulator DS for thread i, 1<=i<=MAXTHREAD
;
push edi ; save eDI
push es ; save ES
LOAD_DS_EDI ; get pointer (ds:edi) to thread's DS
mov ds,ds:[edi] ; set up DS to thread's data area
assume ds:EMULATOR_DATA ; or dynamically allocated copy
pop es
pop edi ; restore DI
endm
ALLOCthreadDS macro
pub allocperthread
LOAD_DS_EDI ; get pointer into __FPDSARRAY
; eAX = 2 * (thread_id)
ifdef thread1static
;
; for thread 1, use EMULATOR_DATA segment
;
cmp eax,2 ; thread 1?
jnz allocds ; no - dynamically allocate DS
mov ax,EMULATOR_DATA ; yes - use static area
mov ds:[edi],ax ; store new DS into __FPDSARRAY
mov ds,ax
assume ds:EMULATOR_DATA ; or dynamically allocated copy
mov es,ax ; ES = DS = EMULATOR_DATA
jmp allocdone
else
jmp allocds
endif ;thread1static
pub allocerror
mov ax,-3 ; return allocation error
stc
ret
;
ifdef thread1static
; for threads other than thread 1, allocate new DS from the system
else
; for all threads, allocate new DS from the system
endif
;
pub allocds
assume ds:DGROUP
push offset __fptaskdata ; size of per-thread data area
push ds ; ds:di = addr of thread's DS
push edi
push 0 ; non-shared segment
os2call DOSALLOCSEG
or ax,ax ; allocation error?
jnz allocerror ; yes - cause thread init to fail
mov di,ds:[edi] ; set ES = DS = thread's data selector
mov ds,di
mov es,di
assume ds:EMULATOR_DATA ; or dynamically allocated copy
pub allocdone
;
; ES = DS = selector for emulator data area
;
mov edx,offset __fptaskdata ; dx = size of emulator data area
sub edx,offset EMULATOR_DATA; jwm
xor ax,ax ; prepare to zero out data segment
xor edi,edi ; start at offset zero
mov edi,offset EMULATOR_DATA; jwm : begin at the beginning
mov ecx,edx ; cx = size of segment (even)
shr ecx,1 ; halve it
rep stosw ; zero it!
endm
FREEthreadDS macro
pub freeperthread
assume ds:EMULATOR_DATA ; or dynamically allocated copy
ifdef thread1static
mov ax,ds
cmp ax,EMULATOR_DATA ; don't free thread 1's area
je nofreeseg
endif ;thread1static
push ds
os2call DOSFREESEG ; free per-thread emulator data area
nofreeseg:
LOAD_DS_EDI ; get pointer into __FPDSARRAY
mov word ptr ds:[edi],0 ; zero out __FPDSARRAY element
; for the current thread
assume ds:EMULATOR_DATA
endm