123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120 |
- { example for :
- interrupt redirection
- software vs. hardware interrupts
- set_pm_interrupt()
- get_pm_interrupt()
- get_cs()
- tseginfo
- trealregs
- processor access
- get_cs(), get_ds(), get_ss()
- }
- { This example shows the difference between protected and real mode
- interrupts; it redirects the protected mode handler to an own handler which
- returns an impossible function result and calls it afterwards. Then the real
- mode handler is called directly, to show the difference between the two.
- Used Interrupt:
- get DOS version Int 21h / function 30h
- Input: AH = $30
- AL = $1
- Return: AL = major version number
- AH = minor version number
- }
- uses crt, { used for clreol(), gotoxy() }
- go32;
- {$ASMMODE DIRECT}
- var r : trealregs;
- axreg : Word; { temporary variable used for the protected mode int call }
- oldint21h : tseginfo;
- newint21h : tseginfo;
- { this is our int 21h protected mode interupt handler. It catches the function
- call to get the DOS version, all other int 21h calls are redirected to the
- old handler; it is written in assembly because the old handler can't be
- called with pascal }
- procedure int21h_handler; assembler;
- asm
- cmpw $0x3001, %ax
- jne CallOld
- movw $0x3112, %ax
- iret
- CallOld:
- ljmp %cs:OLDHANDLER
- OLDHANDLER: .long 0
- .word 0
- end;
- { a small helper procedure, which waits for a keypress }
- procedure resume;
- begin
- Writeln;
- Write('-- press any key to resume --'); readkey;
- gotoxy(1, wherey); clreol;
- end;
- begin
- { see the text messages for further detail }
- clrscr;
- Writeln('Executing real mode interrupt');
- resume;
- r.ah := $30; r.al := $01; realintr($21, r);
- Writeln('DOS v', r.al,'.',r.ah, ' detected');
- resume;
- Writeln('Executing protected mode interrupt without our own handler');
- Writeln;
- asm
- movb $0x30, %ah
- movb $0x01, %al
- int $0x21
- movw %ax, _AXREG
- end;
- Writeln('DOS v', r.al,'.',r.ah, ' detected');
- resume;
- Writeln('As you can see the DPMI hosts default protected mode handler');
- Writeln('simply redirects it to the real mode handler');
- resume;
- Writeln('Now exchanging the protected mode interrupt with our own handler');
- resume;
- newint21h.offset := @int21h_handler;
- newint21h.segment := get_cs;
- get_pm_interrupt($21, oldint21h);
- { storing old handler address in interrupt handler }
- asm
- movl _OLDINT21H, %eax
- movl %eax, OLDHANDLER
- movw 4+_OLDINT21H, %ax
- movw %ax, 4+OLDHANDLER
- end;
- set_pm_interrupt($21, newint21h);
- Writeln('Executing real mode interrupt again');
- resume;
- r.ah := $30; r.al := $01; realintr($21, r);
- Writeln('DOS v', r.al,'.',r.ah, ' detected');
- Writeln;
- Writeln('See, it didn''t change in any way.');
- resume;
- Writeln('Now calling protected mode interrupt');
- resume;
- asm
- movb $0x30, %ah
- movb $0x01, %al
- int $0x21
- movw %ax, _AXREG
- end;
- Writeln('DOS v', lo(axreg),'.',hi(axreg), ' detected');
- Writeln;
- Writeln('Now you can see that there''s a distinction between the two ways of ');
- Writeln('calling interrupts...');
- set_pm_interrupt($21, oldint21h);
- end.
|