Program Segment Prefix


The Program Segment Prefix is a data structure used in DOS systems to store the state of a program. It resembles the Zero Page in the CP/M operating system. The PSP has the following structure:
OffsetimagesSizeContents
00h-01h2 bytes CP/M-80-like exit
02h-03hword Segment of the first byte beyond the memory allocated to the program
04hbyteReserved
05h-09h5 bytes CP/M-80-like far call entry into DOS, and program segment size
0Ah-0Dhdword Terminate address of previous program
0Eh-11hdwordBreak address of previous program
12h-15hdwordCritical error address of previous program
16h-17hwordParent's PSP segment
18h-2Bh20 bytesJob File Table
2Ch-2DhwordEnvironment segment
2Eh-31hdwordSS:SP on entry to last INT 21h call
32h-33hwordJFT size
34h-37hdwordPointer to JFT
38h-3BhdwordPointer to previous PSP
3Ch-3Fh4 bytesReserved
40h-41hwordDOS version to return
42h-4Fh14 bytesReserved
50h-52h3 bytes Unix-like far call entry into DOS
53h-54h2 bytesReserved
55h-5Bh7 bytesReserved
5Ch-6Bh16 bytesUnopened Standard FCB 1
6Ch-7Fh20 bytesUnopened Standard FCB 2
80h1 byteNumber of bytes on command-line
81h-FFh127 bytesCommand-line tail

The PSP is most often used to get the command line arguments of a DOS program; for example, the command "FOO.EXE /A /F" executes FOO.EXE with the arguments '/A' and '/F'.
If the PSP entry for the command line length is non-zero and the pointer to the environment segment is neither 0000h nor FFFFh, programs should first try to retrieve the command line from the environment variable %CMDLINE% before extracting it from the PSP. This way, it is possible to pass command lines longer than 126 characters to applications.
The segment address of the PSP is passed in the DS register when the program is executed. It can also be determined later by using Int 21h function 51h or Int 21h function 62h. Either function will return the PSP address in register BX.
Alternatively, in.COM programs loaded at offset 100h, one can address the PSP directly just by using the offsets listed above. Offset 000h points to the beginning of the PSP, 0FFh points to the end, etc.
For example, the following code displays the command line arguments:

org 100h ;.COM - not using ds
; INT 21h subfunction 9 requires '$' to terminate string
xor bx,bx
mov bl,
cmp bl,7Eh
ja exit ; preventing overflow
mov byte ,'$'
; print the string
mov ah,9
mov dx,81h
int 21h
exit:
mov ax,4C00h ; subfunction 4C
int 21h

In DOS 1.x, it was necessary for the CS register to contain the same segment as the PSP at program termination, thus standard programming practice involved saving the DS register to the stack at program start and terminating the program with a RETF instruction, which would pop the saved segment value off the stack and jump to address 0 of the PSP, which contained an INT 20h instruction.

; save
push ds
xor ax,ax
push ax
; move to the default data group
mov ax,@data
mov ds,ax
; print message in mess1
mov dx,mess1
mov ah,9
int 21h
retf

If the executable was a.COM file, this procedure was unnecessary and the program could be terminated merely with a direct INT 20h instruction or else calling INT 21h Function 0. However, the programmer still had to ensure that the CS register contained the segment address of the PSP at program termination. Thus,

jmp start
mess1 db 'Hello world!$'
start:
mov dx,mess1
mov ah,9
int 21h
int 20h

In DOS 2.x and higher, program termination was accomplished instead with INT 21h Function 4Ch which did not require the CS register to contain the segment value of the PSP.