Contributor: TRISDARESA SUMARJOSO { TRISDARESA SUMARJOSO > I was wondering if anyone knew how to make a split screen While > making EXEC calls and not losing your Windows? > Anyone got any ideas or routines that do this? I can do it easily > using TTT when I just stay Within the Program, but the problems arise > when I do the SwapVectors and do my Exec call, all hell breaks loose. > Lynn. Here is a Unit that I've created to trap Int 29h. the Function of this Unit is to trap the output that Dos spits through the Int 29h (such as XCopy, PkZip, etc) and redirect it into a predefined Window. Here is the stuff: } Unit I29UnitA; { This Unit will trap Dos output which use Int 29h. Any other method of writing the scren, such as Direct Write which bypasses Int 29h call, will not be trapped. } Interface { Initialize the view that will be use to output the Dos output. Will also draw basic Window frame. } Procedure InitView(XX1, XY1, XX2, XY2 : Byte); { Clear the pre-defined view. } Procedure ClearView; { Procedure to redirect the Turbo Pascal Write and WriteLn Procedure. (standard OutPut only). Do not call this Procedure twice in the row. More than once call to this Procedure will result Pascal's standard output Procedure will not be restored properly. } Procedure TrapWrite; { Restore Pascal's Write and WriteLn Procedure into its original condition that was altered With TRAPWrite. (standard OutPut only). } Procedure UnTrapWrite; Implementation Uses Dos; Type VioCharType = Record Case Boolean Of True : (Ch, Attr : Byte); False : (Content : Word); end; DrvFunc = Function(Var F : TextRec) : Integer; VioBufType = Array [0..24, 0..79] Of VioCharType; Var OldInt29 : Pointer; OldExit : Pointer; OldIOFunc : DrvFunc; OldFlushFunc : DrvFunc; TrapWriteVar : Boolean; X1, Y1, X2, Y2 : Byte; XVio : Byte; YVio : Byte; VioBuffer : ^VioBufType; VioCurLoc : Word Absolute $0040:$0050; {$F+} Procedure NewInt29(Flags, CS, IP, AX, BX, CX, DX, SI, DI, DS, ES, BP: Word); Interrupt; begin VioBuffer^[YVio, XVio].Attr := VioBuffer^[YVio, XVio].Attr And Not 112; if (Lo(AX) = 13) Then begin XVio := X1; AX := 0; end else if (Lo(AX) = 10) Then begin Inc(YVio); AX := 0; end; begin if (XVio > X2) Then begin XVio := X1; Inc(YVio); end; if (YVio > Y2) Then begin Asm Mov AH, 06 Mov AL, YVio Sub AL, Y2 Mov CH, Y1 Mov CL, X1 Mov DH, Y2 Mov DL, X2 Mov BH, 07 Int 10h end; YVio := Y2; end; if (Lo(AX) = 32) Then begin if (Lo(VioCurLoc) < XVio) Then begin XVio := Lo(VioCurLoc); VioBuffer^[YVio, XVio].Ch := Lo(AX); end else begin VioBuffer^[YVio, XVio].Ch := Lo(AX); Inc(XVio); end; end else begin VioBuffer^[YVio, XVio].Ch := Lo(AX); Inc(XVio); end; VioCurLoc := YVio Shl 8 + XVio; end; VioBuffer^[YVio, XVio].Attr := VioBuffer^[YVio, XVio].Attr Or 112; end; {$F-} {$F+} Procedure RestoreInt29; begin ExitProc := OldExit; SetIntVec($29, OldInt29); if TrapWriteVar Then begin TextRec(OutPut).InOutFunc := @OldIOFunc; TextRec(OutPut).FlushFunc := @OldFlushFunc; end; end; {$F-} Procedure HookInt29; begin GetIntVec($29, OldInt29); SetIntVec($29, @NewInt29); OldExit := ExitProc; ExitProc := @RestoreInt29; end; Procedure InitView(XX1, XY1, XX2, XY2: Byte); Var I : Byte; begin X1 := XX1+1; Y1 := XY1+1; X2 := XX2-1; Y2 := XY2-1; XVio := X1; YVio := Y1; For I := XX1 To XX2 Do begin VioBuffer^[XY1, I].Ch := 205; VioBuffer^[XY2, I].Ch := 205; end; For I := XY1+1 To XY2-1 Do begin VioBuffer^[I, XX1].Ch := 179; VioBuffer^[I, XX2].Ch := 179; end; VioBuffer^[XY1, XX1].Ch := 213; VioBuffer^[XY2, XX1].Ch := 212; VioBuffer^[XY1, XX2].Ch := 184; VioBuffer^[XY2, XX2].Ch := 190; VioCurLoc := YVio Shl 8 + XVio; end; Procedure DoWriteStuff(F : TextRec); Var I : Integer; Regs : Registers; begin For I := 0 To F.BufPos-1 Do begin Regs.AL := Byte(F.BufPtr^[I]); Intr($29, Regs); end; end; {$F+} Function NewOutputFunc(Var F : TextRec) : Integer; begin DoWriteStuff(F); F.BufPos := 0; NewOutPutFunc := 0; end; {$F-} {$F+} Function NewFlushFunc(Var F : TextRec) : Integer; begin DoWriteStuff(F); F.BufPos := 0; NewFlushFunc := 0; end; {$F-} Procedure TrapWrite; begin if Not TrapWriteVar Then begin With TextRec(OutPut) Do begin OldIOFunc := DrvFunc(InOutFunc); InOutFunc := @NewOutPutFunc; OldFlushFunc := DrvFUnc(FlushFunc); FlushFunc := @NewFlushFunc; end; TrapWriteVar := True; end; end; Procedure UnTrapWrite; begin if TrapWriteVar Then begin TextRec(OutPut).InOutFunc := @OldIOFunc; TextRec(OutPut).FlushFunc := @OldFlushFunc; TrapWriteVar := False; end; end; Procedure ClearView; begin Asm Mov AH, 06 Mov AL, 0 Mov CH, Y1 Mov CL, X1 Mov DH, Y2 Mov DL, X2 Mov BH, 07 Int 10h end; XVio := X1; YVio := Y1; VioCurLoc := YVio Shl 8 + XVio; end; Procedure CheckMode; Var MyRegs : Registers; begin MyRegs.AH := $F; Intr($10, MyRegs); Case MyRegs.AL Of 0, 1, 2, 3 : VioBuffer := Ptr($B800, $0000); 7 : VioBuffer := Ptr($B000, $0000); end; end; begin X1 := 0; Y1 := 0; X2 := 79; Y2 := 24; XVio := 0; YVio := 0; VioCurLoc := YVio Shl 8 + XVio; HookInt29; TrapWriteVar := False; CheckMode; end. Program Int29Testing; {$A+,B-,D+,E+,F-,G-,I+,L+,N-,O-,P-,Q-,R-,S+,T-,V+,X+} {$M $800,0,0} Uses Dos, Crt, I29UnitA; Var CmdLine : String; I : Byte; { Function to convert a String to upper case. Return the upper-case String. } Function Str2Upr(Str : String) : String; Assembler; Asm Push DS CLD LDS SI, Str LES DI, @Result LodSB Or AL, AL Jz @Done StoSB Xor CH, CH Mov CL, AL @@1: LodSB Cmp AL, 'a' JB @@2 Cmp AL, 'z' JA @@2 Sub AL, 20h @@2: StoSB Loop @@1 @Done: Pop DS end; begin ClrScr; GotoXY(1,1); WriteLn('Output interceptor.'); { Initialize redirector's area. } InitView(0,2,79,24); Repeat { Redirect Turbo's output into the predefined Window. } TrapWrite; Write(#0,' Please enter Dos command (Done to Exit): '); ReadLn(CmdLine); WriteLn; { Restore Turbo's original Output routine. } UnTrapWrite; GotoXY(1,2); WriteLn('Command executed : ', CmdLine); CmdLine := Str2Upr(CmdLine); if (CmdLine <> 'DONE') And (CmdLine <> '') Then begin SwapVectors; Exec('C:\Command.Com', '/C'+CmdLine); SwapVectors; end; GotoXY(1,2); WriteLn('Command execution done. Press anykey to continue...'); Repeat Until ReadKey <> #0; ClearView; GotoXY(1,2); WriteLn(' '); Until (CmdLine = 'DONE'); ClrScr; end. { Both the testing Program and the Unit itself (expecially the Unit), is by no mean perfect. Use With caution. It might not wise to use such redirector (my int 29 Unit) in a Program that swaps itself out of memory. The above Programs were not optimized in anyway (so it might slow your Program a little). And I don't guarantee that this Program will work on your computer (it work Without a problem on mine). if you like this Unit, you can use it anyway you desire. Just remember I can guarantee nothing For this method. }