Terminate and Stay Resident Programming

See below for an example of TSR programming. This one turns off the internal speaker.
See below for an example of TSR programming. This one sends the current DOS screen to the Windows Clipboard.
See below for an example of TSR programming to hook INT 21h.

Terminate and Stay Resident (TSR) programming is one of the most interesting services provided by MS-DOS. (Norton's Programming Bible)

The following code runs a simple routine. It installs the TSR, invokes the TSR service (which removes the installation code form memory but not the actual TSR code). Then when installed, it hooks into the keyboard interrupt and looks to see if the Ctrl-Del sequence was entered. If it wasn't, it sends control to the keyboard interrupt and allows it and DOS to function normally. If the Ctrl-Del sequence has been entered, then it prints an 'A' on the screen and jumps to the previous interrupt.

This DOS service restores the interrupt vectors for 22h, 23h, and 24h to the defaults, so changing their vector interrupt address' will not do you any good.

When using the TSR service, you should release the memory of the environment block that your installation program has.

Also, this DOS service does not work in DOS 1.0


Now for the hard part. DOS is not reentrant. This means that you can not call a DOS service while DOS is currently running a DOS service. If the user presses a key, the hardware sets and runs the code pointed to be the INT 09h vector instantly, whether in DOS or not. If DOS is running, and you call a DOS service, you will most likely crash the machine.

So, how do we write our code so that it doesn't crash the machine? DOS has left us a few things that we can do to make our TSR writing and running a lot easier. The first is the INDOS flag. It is exactly what it sounds like. Are we INDOS or not. DOS will set this flag to non-zero if it is currently running a DOS service. This flag is located in low memory (see code below), and we have to get the address of this flag and save it in the TSR code for later use.

Also, DOS has given us INT 28h, or the IDLE INTERRUPT. This interrupt is called when DOS has nothing to do. So if we hook this interrupt and call our interrupt handler code from this interrupt, we can pretty much assume that DOS is not busy, and we can do our code without crashing the machine.

But what if DOS is busy? How do we wait for DOS to be done and then run our interrupt handler? Well, this is where another interrupt handler comes into play. Interrupt 08h, the timer interrupt. We can place a variable in the timer interrupt to decrement each time this interrupt is called. Now we use interrupt 28h (described above), to see if we are waiting to call our interrupt code. In the interrupt 28h handler, we check to see if our wait variable is above zero. If so, we set the wait variable to zero (so that we don't try to call it again), and call our interrupt vector. If we actually decrement the variable to zero without calling our interrupt vector, this means that DOS is probably hung, and your interrupt vector will not be called. This usually does not happen, if we have a decent delay time. A value of 9 is usually quite enough time for DOS to get done.

So, how do we safely hook and call these interrupts? The safest thing to do is to call the old interrupt handler either before our code (most likely the better case), or after our code. What if we didn't call the previously loaded interrupt handler, and this handler set all the BIOS data for us? The key would not get into the BIOS buffer, the scan code would not be created, and other things like this. So we always call the previously loaded interrupt handler.

Another thing that we have to worry about. What if the user presses a key, which in turn calls our interrupt handler, while we are currently processing our interrupt handler? This could be bad. The simplest thing for us to do is to call the previously loaded interrupt handler, and forget that the keypress was ever made. However, if you needed this keypress badly, you could create a handler to save it on a stack, run the interrupt handler, and then see if a there are any keys on this stack. However, this is not likely with a regular DOS TSR. A system program, now that's a different story.

How about another thing to worry about. When we first load our TSR into memory, how do we know if we have previously loaded our TSR? We don't want it loaded, and do the process twice. This is actually quite simple. Place a known string of characters at a known offset in your interrupt handler. Then when you load the TSR, first check to see if this string of characters is at this location in the current interrupt handler. If it is, exit with an error.

Another item to talk about is the stack. You don't know where or how much stack there is when your interrupt handler gets called. You can pretty much assume that there will be about a paragraph of stack free for your use. However, a paragraph can only hold eight (8) 16-bit values. So we can create our own stack to practically whatever we want to use. However, before we create this stack, we must save in memory all registers used. You cannot modify ANY registers without saving and restoring them.

We also need to wait for the break code of the key press instead of the make code. If we caught the make code, we would call our interrupt handler repeatedly until the key was released. This is not what we want to do. The break code is actually the make code with bit 7 set.

Let us get to the code.
;
;  assembled with NBASM
;
Delay      equ 09     ; delay while INDOS != 0

.model tiny
.code
.186
           org 100h

start:     jmp  Init        ; at first load up, just to initialization code

ourint09:  jmp  DoIt        ; our actual interrupt handler starts here.

OurID      db  'FY'         ; our id, to make sure we don't load twice
OurKey     db  00h          ; this byte saves the key press for our handler
oldint09O  dw  00h          ; this dword points to old interrupt vector
oldint09S  dw  00h          ;
SavedAX    dw  00h          ; temp word to store ax before our stack is setup
SavedDX    dw  00h          ;   ditto for dx
SavedES    dw  00h          ;   ditto for es
SavedSP    dw  00h          ; saved SP
SavedSS    dw  00h          ; saved SS
OurStack   dup 126,0        ; Our Stack (64 word stack)
TopofStack dw  00h          ; top of stack

; this is our interrupt code for hardware int 09h
DoIt:      pushf                        ; save flags
           mov  cs:SavedAX,ax           ; save ax (not on stack)
           in   al,60h                  ; get data from keyboard

           cmp  byte cs:tsraktiv,0      ; are we INDOS
           jne  short NotOurs           ;

           cmp  byte cs:tsrnow,0        ; can we run the tsr now
           jne  short NotOurs           ;

           cmp  al,(53h|80h)            ; 53h (set bit 7 for break code)
           je   short IsOurs
NotOurs:   mov  ax,cs:SavedAX           ; restore ax (not from stack)
           popf                         ; restore flags
           jmp  far cs:oldint09O        ; jump to old interrupt handler

IsOurs:    mov  cs:SavedDX,dx           ; save dx (not on stack)
           mov  cs:SavedES,es           ; save es (not on stack)
           xor  dx,dx
           mov  es,dx
           test byte es:[0417h],00000100b  ; was CTRL key pressed
           mov  es,cs:SavedES           ; restore es (not from stack)
           mov  dx,cs:SavedDX           ; restore dx (not from stack)
           jz   short NotOurs

           mov  cs:OurKey,al            ; we need to save the key that got us here

           call dosaktiv                ; are we INDOS?
           je   short i9_tsr

W4BIOS:    mov  byte cs:tsrnow,Delay    ; delay timer ticks to wait for INDOS
           jmp  short NotOurs

i9_tsr:    mov  byte cs:tsraktiv,1      ; set flag so that we don't reenter our code
           mov  byte cs:tsrnow,0        ; and clear the delay flag
           pushf                        ; simulate an INT call to previous handler
           call far cs:oldint09O        ;  (by pushing the flags and doing a FAR CALL)
           call DoOurInt09              ; then do our code
           popf                         ; restore flags
           iret                         ;


; here is where we do our TSR code because HOTKEY was found
DoOurInt09 proc near

; we need to create our own stack so that we don't modify the parent stack
           cli                          ; temp stop interrupts
           mov  cs:SavedAX,ax           ; save ax
           mov  cs:SavedSS,ss           ; save ss:sp
           mov  cs:SavedSP,sp           ;
           mov  ax,cs                   ; create new stack
           mov  ss,ax                   ;
           mov  sp,offset cs:TopofStack ;
           mov  ax,cs:SavedAX           ; restore ax
           sti                          ; restore interrupts

           pusha                        ; save registers used (on our stack)
           push ds                      ;
           push es                      ;

           push cs                      ; make es = ds = cs
           push cs                      ;
           pop  ds                      ;
           pop  es                      ;

           mov  al,OurKey               ; get key entered



;  this is where you do your code.  I have found that if you are careful,
;   and write good clean code, you can do just about anything you want
;   within reason.



; exit code (done with TSR code, so exit back to parent program)
NoErrExit: pop  es                      ; restore all registers we used
           pop  ds                      ;
           popa                         ;

           cli                          ; temp stop interrupts
           mov  sp,cs:SavedSP           ; restore org stack
           mov  ss,cs:SavedSS           ;
           sti                          ; restore interrupts

           mov  byte cs:tsraktiv,0      ; clear flag so that we can do 
                                        ;   this again, next time
           ret
DoOurInt09 endp


tsrnow     db  00h   ; the delay flag
tsraktiv   db  00h   ; the 'are we in here' flag
oldint08O  dw  00h   ; this dword points to old interrupt vector
oldint08S  dw  00h

OurInt08   proc near

           cmp  byte cs:tsrnow,0        ; don't try to decrement it 
           je   short i8_end            ;   if it is already zero!

           dec  byte cs:tsrnow          ; else do the delay

           call dosaktiv                ; check to see if INDOS
           je   short i8_tsr            ;  if we are not, do our interrupt

i8_end:    jmp  far cs:oldint08O        ; else just do the previous handler

i8_tsr:    mov  byte cs:tsrnow,0        ; clear delay flag
           mov  byte cs:tsraktiv,1      ; set 'in here' flag
           pushf                        ; simulate an INT
           call far cs:oldint08O        ; using INT 08H emulation
           call DoOurInt09              ; then do our handler
           iret
OurInt08   endp

oldint28O  dw  00h   ; this dword points to old interrupt vector
oldint28S  dw  00h

OurInt28   proc near
									
           cmp  byte cs:tsrnow,0        ; Is TSR waiting for activation?
           jne  short i28_tsr           ; No --> Return to caller

i28_end:   jmp  far cs:oldint28O        ; Return to old handler

i28_tsr:   mov  byte cs:tsrnow,00       ; TSR no longer waiting for activation
           mov  byte cs:tsraktiv,1      ; TSR is (already) active

           pushf                        ; Call old interrupt handler
           call far cs:oldint28O        ; using INT 28H emulation
           call DoOurInt09
           iret                         ; Return to caller

OurInt28   endp

InDOSfoff  dw  00h
InDOSfseg  dw  00h

dosaktiv   proc near uses ax bx es      ; this is the 'get INDOS flag' code
           mov  bx,cs:InDOSfoff         ; we loaded these two words 
           mov  ax,cs:InDOSfseg         ;  with the address at startup
           mov  es,ax
           cmp  byte es:[bx],00
           ret
dosaktiv   endp


; this is the initialization procedure.  Once we are done with this
;  and load the TSR, this part is not needed and removed from memory.
;
Init:      push cs                      ; make sure ds=cs
           pop  ds                      ;

           mov  ax,3508h                ; get old interrupt vector (08h)
           int  21h                     ;
           mov  [oldint08O],bx          ; and save in static data area
           mov  [oldint08S],es          ;

           mov  ax,3528h                ; get old interrupt vector (08h)
           int  21h                     ;
           mov  [oldint28O],bx          ; and save in static data area
           mov  [oldint28S],es          ;

           mov  ax,3509h                ; get old interrupt vector (09h)
           int  21h                     ;
           mov  [oldint09O],bx          ; and save in static data area
           mov  [oldint09S],es          ;

           mov  ax,es:[bx+03]           ; Check to see if already loaded
           cmp  ax,'YF'                 ; is it 'FY'?
           jne  short OKtoLoad
           mov  dx,offset AlreadyS
           mov  ah,09
           int  21h
           .exit 01

OKtoLoad:  push cs                      ; set es = cs
           pop  es

           mov  ah,34h                  ; get indos flag address
           int  21h                     ;  returns address in es:bx
           mov  InDOSfoff,bx            ; save for our interrupt handler
           mov  InDOSfseg,es            ;


; you can do other initialization code here.
; maybe check the command line for parameters, etc.



           mov  es,[002Ch]              ; free environment block
           mov  ah,49h                  ;
           int  21h                     ;

           mov  dx,offset OurInt08      ; set new interrupt vector (08h)
           mov  ax,2508h                ;
           int  21h                     ;

           mov  dx,offset ourint09      ; set new interrupt vector (09h)
           mov  ax,2509h                ;
           int  21h                     ;

           mov  dx,offset OurInt28      ; set new interrupt vector (28h)
           mov  ax,2528h                ;
           int  21h                     ;

           mov  dx,offset Init          ; get paragraphs needed
           sub  dx,offset Start         ;
           shr  dx,04                   ; 
           add  dx,17                   ; add 16 paras for PSP + 1 extra
           mov  ax,3100h                ; terminate and stay resident
           int  21h                     ;  (00h ERRORLEVEL)


AlreadyS  db 13,10,'Already loaded',13,10,36

.end start


Again, be careful with this code if you do any modifications, or the such, you could crash your system and have to reboot.

If you have any other questions or see a mistake that I made, please e-mail me and I will do my best to help out.

Another TSR demo that turns off the PC internal speaker (on older machines)

The following code creates a 292 byte .COM file, and when installed will take up only 464 bytes in memory.

When run for the first time, this code "hooks" into two interrupts. Interrupt 16h (the keyboard interrupt) and Interrupt 1Ch (the timer interrupt). The reason why we hook into the keyboard interrupt is that it is an easy way to check for our TSR in memory.

What we do is invoke INT 16h with a sub-service of 66h and a job number (0 or 1) in AL. (Service 66h is not included with the BIOS keyboard interrupt service, it is one we install the first time NOSOUND is run). If a 66h is returned in AL then we have already installed NOSOUND.

If AL does not = 66h, then we must install NOSOUND. We first "hook" into the timer interrupt (1Ch). We then "hook" in to the keyboard interrupt (16h). We then install the TSR. Now once the TSR is installed, the hook to INT 1Ch will check the "job" flag that we have set and either "stop" the speaker if it is zero. When the INT 16h, service 66h (our service) is called, it checks for installation; and if it is installed it returns 66h in AL.

Since we hooked into INT 16h, we can call it every time NOSOUND is run to check for installation. This way we won't install more than one TSR at a time.

This code is for learning purposes only. Please see the above code to make this a correctly written TSR. Typing 'NOSOUND off' turns sound off, 'NOSOUND on' turns sound on.
;
;  assembled with NBASM

.model tiny
.code
           org 100h
start:     jmp install

Copyright  db  13,10,'NOSOUND  Quiets the hardware speaker.'
           db  13,10,'Copyright  1984-2025  Forever Young Software',13,10,36
AllowY     db  13,10,'  Allowing Sound$'
AllowN     db  13,10,'  Not Allowing Any Sound$'
Old1CAdr   dw  00h   ; these remember the original Int 1Ch address
Old1CSeg   dw  00h   ; they must be in the code segment
Old16Adr   dw  00h   ; these remember the original Int 16h address
Old16Seg   dw  00h   ; they must be in the code segment
job        db  00h   ; 0 = sound off, else sound on
                
NewInt1Ch: pushf                 ; save flags
           cmp  byte cs:job,00h  ; if job != 0 then skip ours
           jnz  short SoundOn    ; 
           push dx               ;

;If you notice any delay between hitting <Enter> after your "NOSOUD off"
;command, it's because of the 18th of a second delay (maximum) before
;the speaker is actually turned off.

           mov  dx,61h        ; turn sound off
           in   al,dx         ;  .
           and  al,0FCh       ; clear bits 1 & 0
           out  dx,al         ;  .
           pop  dx            ;
SoundOn:   popf                         ; restore flags
           jmp  far cs:Old1CAdr

NewInt16h: pushf              ; save flags
           cmp  ah,66h        ; if our service number
           jne  short SkipOurs
           mov  cs:job,al     ; then put 'job' in job above
           mov  al,ah         ; send installed flag
           iret               ;
SkipOurs:  popf               ; restore flags
           jmp  far cs:Old16Adr ;

Install:   mov  dx,offset Copyright ; print message
           mov  ah,09h        ;
           int  21h           ;
           xor  al,al         ; assume no sound
           mov  dx,offset AllowN  ; print message
           mov  ah,[0083h]    ; get command line 'n' or 'f'
           cmp  ah,'n'        ; if 'n' then job != 0
           jne  short SoundOff1 ;
           mov  al,0FFh       ;
           mov  dx,offset AllowY  ; print message
SoundOff1: push ax            ; save al
           mov  ah,09h        ;
           int  21h           ;
           pop  ax            ; restore al
           mov  ah,66h        ; call interrupt 16h w/our service #
           int  16h           ; on return:
           cmp  al,66h        ; 
           jne  short NotInstld ; if al = 66h, then is installed
           .exit              ; and exit (no TSR it)

NotInstld: mov  es,[002Ch]    ; free environment block
           mov  ah,49h
           int  21h

           mov  ax,351Ch       ; ask DOS for existing Int 1Ch vector address
           int  21h            ; DOS returns the segment:address in ES:BX
           mov  Old1CAdr,bx    ; save it locally
           mov  Old1CSeg,es

           mov  ax,251Ch       ; point Interrupt 1Ch to our own handler
           mov  dx,offset NewInt1Ch
           int  21h
      
           mov  ax,3516h       ; ask DOS for existing Int 16h vector address
           int  21h            ; DOS returns the segment:address in ES:BX
           mov  Old16Adr,bx    ; save it locally
           mov  Old16Seg,es

           mov  ax,2516h       ; point Interrupt 16h to our own handler
           mov  dx,offset NewInt16h
           int  21h
        
           mov  dx,(install-start+256+15) ; save all TSR code + PSP
           mov  cl,04h         ;   + 15 bytes to make sure we get all of TSR
           shr  dx,cl          ; (paragraphs)
           mov  ax,3100h       ; exit to DOS but stay resident
           int  21h            ;

.end       start

Another TSR demo that sends the current DOS screen contents to the clipboard.

This TSR sends the characters stored in screen mode 3's memory (which are at physical address 0xB8000) to the Windows clipboard.

I created this because there were times that I wanted to send some info from a file I was viewing in DOS's EDIT to the Windows clipboard. (Sure I could have just pressed Alt-PrtScr but then I couldn't have created another TSR demo source for you.)

This TSR works fairly well on most machines. However, it does not like DOSKEY at all.

; This is a TSR that when activated and the CTRL-ALT-C
; key combination is pressed, it will send whatever text is on the
; screen (memory at 0B800h) to the Windows Clipboard. 
;
;  assembled with NBASM

Delay      equ 09     ; delay while INDOS != 0

.model tiny
.186
.code

start:     jmp  install

comment |
  We could put the Copyright Notice in the install part of the code
   so that when we are a TSR, it would make for a smaller memory
   allocation, but this way if someone is disassembling Int 09h,
   they see our copyright notice. |

ourint09:  jmp  short NewInt9S  ; skip data area

ID         db  'FY'      ; ID word to see if we are already here
Copyright  db  13,10,'DOS Screen to Windows Clipboard    v1.00'
           db  13,10,'Copyright  1984-2025  Forever Young Software'
           db  13,10,36
oldint09O  dw  00h          ; these remember the original Int 9 address
oldint09S  dw  00h          ; they must be in the code segment
SavedAX    dw  00h
SavedDX    dw  00h
SavedES    dw  00h
OurKey     db  00h

NewInt9S:  pushf                        ; save flags
           mov  cs:SavedAX,ax           ; save ax (not on stack)
           in   al,60h                  ; get data from keyboard

           cmp  byte cs:tsraktiv,0      ; are we INDOS
           jne  short NotOurs           ;

           cmp  byte cs:tsrnow,0        ; can we run the tsr now
           jne  short NotOurs           ;

           cmp  al,(2Eh|80h)            ; 2Eh (set bit 7 for break code)
           jne  short IsOurs
NotOurs:   mov  ax,cs:SavedAX           ; restore ax (not from stack)
           popf                         ; restore flags
           jmp  far cs:oldint09O        ; jump to old interrupt handler

IsOurs:    mov  cs:SavedDX,dx           ; save dx (not on stack)
           mov  cs:SavedES,es           ; save es (not on stack)
           xor  dx,dx
           mov  es,dx
           test byte es:[0417h],00000100b  ; was CTRL key pressed
           mov  es,cs:SavedES           ; restore es (not from stack)
           mov  dx,cs:SavedDX           ; restore dx (not from stack)
           jz   short NotOurs

           mov  cs:OurKey,al            ; we need to save the key that got us here

           call dosaktiv                ; are we INDOS?
           je   short i9_tsr

W4BIOS:    mov  byte cs:tsrnow,Delay    ; delay timer ticks to wait for INDOS
           jmp  short NotOurs

i9_tsr:    mov  byte cs:tsraktiv,1      ; set flag so we don't reenter our code
           mov  byte cs:tsrnow,0        ; and clear the delay flag
           pushf                        ; simulate an INT call to previous handler
           call far cs:oldint09O        ;  (by pushing flags & doing a FAR CALL)
           call DoOurInt09              ; then do our code
           popf                         ; restore flags
           iret                         ;


; here is where we do our TSR code because HOTKEY was found
DoOurInt09 proc near

           pusha                        ; save the registers we'll be using
           push es

           mov  ax,1700h                ; check clipboard for usage
           int  2Fh                     ;
           cmp  ax,1700h                ; if 1700h then error
           je   short NoErrExit

           mov  ax,1701h                ; open clipboard
           int  2Fh

           push ds
           push cs
           pop  es
           mov  cx,25
           xor  si,si
           mov  di,offset cs:Install
           mov  ax,0B800h
           mov  ds,ax
GetScrL1:  push cx
           mov  cl,80        ; we know that ch = 0 from mov cx,25 above
GetScrL2:  lodsw
           stosb
           loop GetScrL2
           mov  ax,0A0Dh                ; put a CRLF at end of line
           stosw
           pop  cx
           loop GetScrL1
           pop  ds

           mov  ax,1703h                ; put screen 03h into clipboard
           push cs
           pop  es
           mov  bx,offset cs:Install
           mov  dx,01h
           mov  cx,2050                 ; (25x80) + 25 CRLF's
           xor  si,si
           int  2Fh

           mov  ax,1708h                ; close clipboard
           int  2Fh

           ; print smiley face in upper right corner
           mov  ax,0B800h               ;
           mov  es,ax
           mov  bx,es:[009Eh]
           mov  word es:[009Eh],1701h   ; make print smiley white on blue

           ; pause for a few milliseconds
           mov  cx,06
           mov  dx,03DAh
delay_l1:  in   al,dx
           test al,08h
           jne  short delay_l1
delay_l2:  in   al,dx
           test al,08h
           je   short delay_l2
           loop delay_l1

           ; restore char that was originally there
           mov  es:[009Eh],bx

; exit code (done with TSR code, so exit back to parent program)
NoErrExit: pop  es                      ; restore all registers we used
           popa                         ;

           mov  byte cs:tsraktiv,0      ; clear flag so that we can 
                                        ;   do this again, next time
           ret
DoOurInt09 endp

tsrnow     db  00h   ; the delay flag
tsraktiv   db  00h   ; the 'are we in here' flag
oldint08O  dw  00h   ; this dword points to old interrupt vector
oldint08S  dw  00h

OurInt08   proc near

           cmp  byte cs:tsrnow,0        ; don't try to decrement it 
           je   short i8_end            ;  if it is already zero!

           dec  byte cs:tsrnow          ; else do the delay

           call dosaktiv                ; check to see if INDOS
           je   short i8_tsr            ;  if we are not, do our interrupt code

i8_end:    jmp  far cs:oldint08O        ; else just do the previous handler

i8_tsr:    mov  byte cs:tsrnow,0        ; clear delay flag
           mov  byte cs:tsraktiv,1      ; set 'in here' flag
           pushf                        ; simulate an INT
           call far cs:oldint08O        ; using INT 08H emulation
           call DoOurInt09              ; then do our handler
           iret
OurInt08   endp

oldint28O  dw  00h   ; this dword points to old interrupt vector
oldint28S  dw  00h

OurInt28   proc near
									
           cmp  byte cs:tsrnow,0        ; Is TSR waiting for activation?
           jne  short i28_tsr           ; No --> Return to caller

i28_end:   jmp  far cs:oldint28O        ; Return to old handler

i28_tsr:   mov  byte cs:tsrnow,00       ; TSR no longer waiting for activation
           mov  byte cs:tsraktiv,1      ; TSR is (already) active

           pushf                        ; Call old interrupt handler
           call far cs:oldint28O        ; using INT 28H emulation
           call DoOurInt09
           iret                         ; Return to caller

OurInt28   endp

InDOSfoff  dw  00h
InDOSfseg  dw  00h

dosaktiv   proc near uses ax bx es      ; this is the 'get INDOS flag' code
           mov  bx,cs:InDOSfoff         ; we loaded these two words
           mov  ax,cs:InDOSfseg         ;    with the address at startup
           mov  es,ax
           cmp  byte es:[bx],00
           ret
dosaktiv   endp


; This is the install part.  It gets wiped out by our buffer used above.
; After TSR'ed, we no longer need this code.
;
Install:   mov  dx,offset Copyright     ; print our copyright string
           mov  ah,09
           int  21h

           mov  ax,3508h                ; get old interrupt vector (08h)
           int  21h                     ;
           mov  [oldint08O],bx          ; and save in static data area
           mov  [oldint08S],es          ;

           mov  ax,3528h                ; get old interrupt vector (08h)
           int  21h                     ;
           mov  [oldint28O],bx          ; and save in static data area
           mov  [oldint28S],es          ;

           mov  ax,3509h                ; get old interrupt vector (09h)
           int  21h                     ;
           mov  [oldint09O],bx          ; and save in static data area
           mov  [oldint09S],es          ;

           mov  ax,es:[bx+02]           ; Check to see if already loaded
           cmp  ax,'YF'                 ; is it 'FY'?
           jne  short OKtoLoad
           mov  dx,offset AlreadyS
           mov  ah,09
           int  21h
           .exit 01

OKtoLoad:  push cs                      ; set es = cs
           pop  es

           mov  ah,34h                  ; get indos flag address
           int  21h                     ;  returns address in es:bx
           mov  InDOSfoff,bx            ; save for our interrupt handler
           mov  InDOSfseg,es            ;

           mov  es,[002Ch]              ; free environment block
           mov  ah,49h                  ;
           int  21h                     ;

           mov  dx,offset OurInt08      ; set new interrupt vector (08h)
           mov  ax,2508h                ;
           int  21h                     ;

           mov  dx,offset ourint09      ; set new interrupt vector (09h)
           mov  ax,2509h                ;
           int  21h                     ;

           mov  dx,offset OurInt28      ; set new interrupt vector (28h)
           mov  ax,2528h                ;
           int  21h                     ;

           mov  dx,offset install       ; get paragraphs needed
           sub  dx,offset Start         ;
           shr  dx,04                   ; 
           add  dx,17                   ; add 16 paras for PSP + 1 extra
           mov  ax,3100h                ; terminate and stay resident
           int  21h                     ;  (00h ERRORLEVEL)
      

AlreadyS   db  13,10,'Already installed...',13,10,36

.end start

The following code hooks into INT 21h and allows you to do a little bit of code before and a little bit after each time the original INT 21h gets called. Remember, you can only do a little bit of stuff. Nothing major and absolutely nothing that calls INT 21h or you will be in an endless loop (unless you add code to make it recursive). Be careful, this is for example purposes only.

This TSR hooks into INT 21h. When the processor calls INT 21h, this code prints a '1' to row 6, column 1 of the screen, calls the original 21h interrupt, and then prints a '2' to the screen just after the 1.
; Assemble with NBASM

.model tiny
.186
.code
           org  100h
start:     jmp  short Install

olddosint  dw 00h,00h

newDOSInt:
           cli                          ; put TODO stuff here
           push es                      ; .
           push 0B800h                  ; .
           pop  es                      ; .
           mov  byte es:[1920],'1'      ; .   row 6, col 1 : print '1'
           pop  es                      ; .
           sti                          ; end TODO stuff here
        
           pushf                        ; create our own descriptor
           push cs                      ;   (could be a  call far  instead)
           push offset Back             ;
           jmp  far cs:olddosint        ; jmp to old interrupt handler

Back:      pushf                        ; save flags returned by orig int 21h
           cli                          ; put TODO stuff here
           push es                      ; .
           push 0B800h                  ; .
           pop  es                      ; .
           mov  byte es:[1922],'2'      ; .   row 6, col 2 : print '2'
           pop  es                      ; .
           sti                          ; end TODO stuff here
        
           cli                          ; clean up the stack
           push ax                      ; so that it will ret to the
           push bp                      ; correct place
           mov  bp,sp                   ; 
           mov  ax,[bp-6]               ;
           mov  [bp-8],ax               ;
           mov  ax,[bp-4]               ;
           mov  [bp-6],ax               ;
           pop  bp                      ;
           pop  ax                      ;
           popf                         ; restore flags ret'd by orig 21h
           sti                          ;
           retf 2                       ;

Install:   mov  ax,3521h                ; Save old interrupt vector 21h
           int  21h

           mov  [olddosint],bx
           mov  [olddosint+2],es

           push cs
           pop  ds
           mov  dx,offset cs:newdosint  ; set new interrupt vector 21h
           mov  ax,2521h                ; to newdosint
           int  21h

           mov  es,cs:[002Ch]           ; free environment block
           mov  ah,49h
           int  21h

           mov  dx,offset cs:Install     ; get paragraphs needed
           sub  dx,offset cs:Start
           shr  dx,04
           add  dx,17                   ; add 16 paras for PSP + 1 extra
           mov  ax,3100h                ; termination and stay resident
           int  21h

.end       start