;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; RDOS operating system ; Copyright (C) 1988-2000, Leif Ekblad ; ; This program is free software; you can redistribute it and/or modify ; it under the terms of the GNU General Public License as published by ; the Free Software Foundation; either version 2 of the License, or ; (at your option) any later version. The only exception to this rule ; is for commercial usage in embedded systems. For information on ; usage in commercial embedded systems, contact embedded@rdos.net ; ; This program is distributed in the hope that it will be useful, ; but WITHOUT ANY WARRANTY; without even the implied warranty of ; MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the ; GNU General Public License for more details. ; ; You should have received a copy of the GNU General Public License ; along with this program; if not, write to the Free Software ; Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA ; ; The author of this program may be contacted at leif@rdos.net ; ; TEXTMODE.ASM ; PC based text-mode support ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; NAME textmode GateSize = 16 INCLUDE ..\os\system.def INCLUDE ..\os\protseg.def INCLUDE ..\os\driver.def INCLUDE ..\os\user.def INCLUDE ..\os\virt.def INCLUDE ..\os\os.def INCLUDE ..\os\system.inc INCLUDE ..\os\user.inc INCLUDE ..\os\virt.inc INCLUDE ..\os\os.inc INCLUDE ..\os\video.inc INCLUDE pcvideo.inc video_object STRUC v_base video_api_struc <> v_buf_sel DW ? v_buf_base DD ? v_mem_base DD ? v_has_focus DB ? v_row DW ? v_col DW ? video_object ENDS .386p code SEGMENT byte public use16 'CODE' assume cs:code PAGE ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: SetCursorPhysical ; ; DESCRIPTION: Set cursor position ; ; PARAMETERS: DS Object ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; SetCursorPhysical Proc near push ax push bx push edx ; mov ax,80 mul ds:v_row add ax,ds:v_col mov bx,ax ; mov edx,ds:v_mem_base cmp edx,0B0000h mov dx,3B4h je set_curs_phys_do mov dx,3D4h set_curs_phys_do: mov al,0Eh out dx,al ; inc dx mov al,bh out dx,al ; dec dx mov al,0Fh out dx,al ; inc dx mov al,bl out dx,al ; pop edx pop bx pop ax ret SetCursorPhysical Endp PAGE ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: switch_to ; ; DESCRIPTION: Enter focus ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; switch_to Proc far push es pushad ; mov ax,pc_video_data_sel mov es,ax mov ax,es:v_curr_object or ax,ax jnz switch_check_mode ; mov ax,ds:v_mode cmp ax,3 je switch_mode_done jmp switch_set_mode switch_check_mode: mov es,ax mov ax,es:v_mode cmp ax,ds:v_mode je switch_mode_done switch_set_mode: push ds mov ax,ds:v_mode xor bx,bx mov ds,bx mov es,bx push 10h V86BiosInt pop ds switch_mode_done: mov ax,pc_video_data_sel mov es,ax mov es:v_curr_object,ds call SetCursorPhysical ; mov edx,ds:v_mem_base GetPhysicalPage push eax mov eax,edx or al,7 SetPhysicalPage ; EnterSection ds:v_section push ds GetFocusThread mov bx,ax mov edx,ds:v_mem_base mov eax,edx or al,7 SetThreadPhysicalPage ; mov eax,cr3 mov cr3,eax ; mov ds:v_has_focus,1 mov ds,ds:v_buf_sel xor si,si mov cx,dosB800 mov es,cx xor di,di mov cx,400h rep movsd ; pop ds LeaveSection ds:v_section ; pop eax push ax GetFocusThread mov es,ax mov edx,es:p_cr3 GetThread mov es,ax pop ax cmp edx,es:p_cr3 je switch_to_done ; mov edx,ds:v_mem_base SetPhysicalPage mov eax,cr3 mov cr3,eax switch_to_done: ; popad pop es ret switch_to Endp PAGE ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: SwitchFrom ; ; DESCRIPTION: Leave focus ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; switch_from Proc far push es pushad ; mov ax,pc_video_data_sel mov es,ax mov ax,es:v_curr_object or ax,ax jz switch_from_done ; mov edx,ds:v_mem_base GetPhysicalPage push eax mov eax,edx or al,7 SetPhysicalPage ; EnterSection ds:v_section push ds mov eax,cr3 mov cr3,eax mov ds:v_has_focus,0 mov es,ds:v_buf_sel xor di,di mov cx,dosB800 mov ds,cx xor si,si mov cx,400h rep movsd pop ds ; GetFocusThread mov bx,ax mov edx,ds:v_buf_base GetPhysicalPage mov edx,ds:v_mem_base SetThreadPhysicalPage LeaveSection ds:v_section ; pop eax push ax mov es,bx mov edx,es:p_cr3 GetThread mov es,ax pop ax cmp edx,es:p_cr3 je switch_from_done ; mov edx,ds:v_mem_base SetPhysicalPage mov eax,cr3 mov cr3,eax switch_from_done: mov ebx,cr3 mov cr3,ebx ; popad pop es ret switch_from Endp PAGE ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: SetCursorPos ; ; DESCRIPTION: Set cursor position ; ; PARAMETERS: DX ROW ; CX COL ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; set_cursor_pos PROC far push ax push bx push dx ; EnterSection ds:v_section mov ds:v_row,dx mov ds:v_col,cx mov al,ds:v_has_focus or al,al jz set_cursor_done ; call SetCursorPhysical set_cursor_done: LeaveSection ds:v_section ; pop dx pop bx pop ax ret set_cursor_pos ENDP PAGE ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: WriteChar ; ; DESCRIPTION: Write a char ; ; PARAMETERS: AL Char ; BL Fore color ; BH Back color ; CX Column ; DX Row ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; write_char Proc far push ax push edi ; EnterSection ds:v_section push ds mov ah,bh shl ah,4 or ah,bl push ax push dx ; mov ds:v_col,cx mov ds:v_row,dx mov ax,80 mul dx add ax,cx add ax,ax mov di,ax mov ax,dosB800 mov ds,ax pop dx pop ax mov [di],ax pop ds LeaveSection ds:v_section ; pop edi pop ax ret write_char Endp PAGE ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: ReadChar ; ; DESCRIPTION: Read a char ; ; PARAMETERS: CX Column ; DX Row ; ; RETURNS: AL Char ; BL Fore color ; BH Back color ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; read_char Proc far push ds push edi ; mov ax,80 mul dx add ax,cx add ax,ax mov di,ax mov ax,dosB800 mov ds,ax mov ax,[di] mov bh,ah mov bl,ah shr bh,4 and bl,0Fh ; pop edi pop ds ret read_char Endp PAGE ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: clear ; ; DESCRIPTION: Clear display region ; ; PARAMETERS: BL Blank fore color ; BH Blank back color ; CX Upper Column ; DX Upper Row ; SI Lower Column ; DI Lower Row ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; clear Proc far push ax push dx ; EnterSection ds:v_section push ds push es mov ax,dosB800 mov ds,ax mov es,ax ; mov ah,bh shl ah,4 or ah,bl mov al,' ' clear_row_loop: push di push ax push dx mov ax,80 mul dx mov di,ax add di,cx add di,di pop dx pop ax push cx sub cx,si neg cx stosw rep stosw pop cx pop di ; inc dx cmp di,dx jae clear_row_loop clear_done: pop es pop ds LeaveSection ds:v_section ; pop dx pop ax ret clear Endp PAGE ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: scroll_up ; ; DESCRIPTION: Scroll display up ; ; PARAMETERS: AX Number of rows ; BL Blank fore color ; BH Blank back color ; CX Upper Column ; DX Upper Row ; SI Lower Column ; DI Lower Row ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; suLowerRow EQU 0 suLowerCol EQU 2 suRows EQU 4 scroll_up Proc far push bp sub sp,6 mov bp,sp ; push ax push dx push si push di EnterSection ds:v_section push ds push es ; mov [bp].suLowerRow,di mov [bp].suLowerCol,si mov [bp].suRows,ax ; mov ax,dosB800 mov ds,ax mov es,ax scroll_up_row_loop: push dx mov ax,80 mul dx mov di,ax add di,cx ; mov ax,80 mul word ptr [bp].suRows mov si,ax add si,di add si,si add di,di pop dx ; mov ax,[bp].suRows add ax,dx cmp ax,[bp].suLowerRow ja scroll_up_clear ; push cx sub cx,[bp].suLowerCol neg cx movsw rep movsw pop cx ; cmp dx,[bp].suLowerRow jae scroll_up_done ; inc dx jmp scroll_up_row_loop scroll_up_clear: mov ah,bh shl ah,4 or ah,bl mov al,' ' scroll_up_clear_row_loop: push ax push dx mov ax,80 mul dx mov di,ax add di,cx add di,di pop dx pop ax ; push cx sub cx,[bp].suLowerCol neg cx stosw rep stosw pop cx ; cmp dx,[bp].suLowerRow jae scroll_up_done ; inc dx jmp scroll_up_clear_row_loop scroll_up_done: pop es pop ds LeaveSection ds:v_section ; pop di pop si pop dx pop ax add sp,6 pop bp ret scroll_up Endp PAGE ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: scroll_down ; ; DESCRIPTION: Scroll display down ; ; PARAMETERS: BL Blank fore color ; BH Blank back color ; CX Upper Column ; DX Upper Row ; SI Lower Column ; DI Lower Row ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; sdUpperRow EQU 0 sdLowerCol EQU 2 sdRows EQU 4 scroll_down Proc far push bp sub sp,6 mov bp,sp ; push ax push dx push si push di EnterSection ds:v_section push ds push es ; mov [bp].sdUpperRow,dx mov [bp].sdLowerCol,si mov [bp].sdRows,ax mov dx,di ; mov ax,dosB800 mov ds,ax mov es,ax scroll_down_row_loop: push dx mov ax,80 mul dx mov di,ax add di,cx ; mov ax,80 mul word ptr [bp].sdRows mov si,di sub si,ax add si,si add di,di pop dx ; mov ax,dx sub ax,[bp].sdRows jc scroll_down_clear cmp ax,[bp].sdUpperRow jb scroll_down_clear ; push cx sub cx,[bp].sdLowerCol neg cx movsw rep movsw pop cx ; cmp dx,[bp].sdUpperRow jbe scroll_down_done ; dec dx jmp scroll_down_row_loop scroll_down_clear: mov ah,bh shl ah,4 or ah,bl mov al,' ' scroll_down_clear_row_loop: push ax push dx mov ax,80 mul dx mov di,ax add di,cx add di,di pop dx pop ax ; push cx sub cx,[bp].sdLowerCol neg cx stosw rep stosw pop cx ; cmp dx,[bp].sdUpperRow jbe scroll_down_done ; dec dx jmp scroll_down_clear_row_loop scroll_down_done: pop es pop ds LeaveSection ds:v_section ; pop di pop si pop dx pop ax add sp,6 pop bp ret scroll_down Endp PAGE ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: init_mode3 ; ; DESCRIPTION: Init video-mode 3 ; ; RETURNS: AX Handle ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; init_mode3 Proc far push ds push es push cx push si push di ; mov eax,SIZE video_object AllocateSmallGlobalMem mov cx,18 mov ax,cs mov ds,ax mov si,OFFSET ModeTab xor di,di rep movsd ; push es mov eax,1000h AllocateBigLinear AllocateGdt mov ecx,eax CreateDataSelector16 mov es,bx mov ax,0720h xor di,di mov cx,800h rep stosw pop ax ; mov ds,ax mov ds:v_mode,3 mov ds:v_buf_base,edx mov ds:v_buf_sel,es mov ds:v_has_focus,0 mov ds:v_row,0 mov ds:v_col,0 InitSection ds:v_section ; mov bx,gdt_sel mov es,bx mov bx,DosB800 mov edx,es:[bx+2] rol edx,8 mov dl,es:[bx+7] ror edx,8 mov ds:v_mem_base,edx ; push ax mov edx,0B0000h xor eax,eax init_mono_loop: SetPhysicalPage add edx,1000h cmp edx,0C0000h jne init_mono_loop ; mov edx,ds:v_buf_base GetPhysicalPage or al,7 mov edx,ds:v_mem_base SetPhysicalPage ; mov eax,cr3 mov cr3,eax pop ax clc ; pop di pop si pop cx pop es pop ds ret init_mode3 Endp PAGE ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: delete_mode3 ; ; DESCRIPTION: Delete video-mode 3 ; ; RETURNS: DS Handle ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; delete_mode3 Proc far push ax push bx push ecx push edx ; mov bx,ds:v_buf_sel FreeGdt mov edx,ds:v_buf_base mov ecx,1000h FreeLinear mov ax,ds mov es,ax xor ax,ax mov ds,ax FreeMem ; pop edx pop ecx pop bx pop ax ret delete_mode3 Endp PAGE ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: ModeTab ; ; DESCRIPTION: Dispatch table ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; error Proc far stc ret error Endp ModeTab: mt00 DW OFFSET delete_mode3, pc_video_code_sel mt01 DW OFFSET switch_to, pc_video_code_sel mt02 DW OFFSET switch_from, pc_video_code_sel mt03 DW OFFSET clear, pc_video_code_sel mt04 DW OFFSET set_cursor_pos, pc_video_code_sel mt05 DW OFFSET write_char, pc_video_code_sel mt06 DW OFFSET read_char, pc_video_code_sel mt07 DW OFFSET scroll_up, pc_video_code_sel mt08 DW OFFSET scroll_down, pc_video_code_sel mt09 DW OFFSET error, pc_video_code_sel mt0A DW OFFSET error, pc_video_code_sel mt0B DW OFFSET error, pc_video_code_sel mt0C DW OFFSET error, pc_video_code_sel mt0D DW OFFSET error, pc_video_code_sel mt0E DW OFFSET error, pc_video_code_sel mt0F DW OFFSET error, pc_video_code_sel mt10 DW OFFSET error, pc_video_code_sel mt11 DW OFFSET error, pc_video_code_sel mt12 DW OFFSET error, pc_video_code_sel PAGE ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: init_text_mode ; ; DESCRIPTION: Init text-mode support ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; public init_text_mode init_text_mode Proc near mov ax,cs mov es,ax mov ax,3 mov di,OFFSET init_mode3 RegisterVideoMode ret init_text_mode Endp code ENDS END