原创 DOS下的SVGA编程

2008-7-17 22:42 4715 5 5 分类: 工业电子

DOS下的SVGA编程
第一章 SVGA显示卡和VBE标准
视频图形阵列适配器(vidio graphics array, VGA)是IBM公司在1987年制定的显示卡标准,它提供的字符和图形两种模式,图形分辨率最大是640 * 480 * 16色或者320 * 200 * 256色,这个标准是显示卡发展的一个丰碑,改变了各厂商混战相互不兼容的局面,而且统一了软件接口标准,为程序开发提供了特别大的方便。VGA显示的调用方法放在了BIOS中,统一使用int 10h功能,主要功能有设置显示模式,文本窗口上卷下卷、光标和字符串,使用调色板、位面结构、读像素和写像素等。


当然,VGA逐渐不能满足需要了。在80年代末至90年代初,市场上出现了以TVGA系列、S3系列、Cirrus Logic系列、ET系列等为首的一批显示卡,它们都提供了比VGA分辨率更高、颜色更丰富的显示模式,又完全兼容VGA显示卡,它们被统称为Super VGA(超级VGA,简写为SVGA)。他们在高分辨率高颜色数的控制方面又需要制定一个新的标准。为此视频电子学标准协会VESA(Vidio Electronics Standards Association)提出了一组扩展的BIOS功能调用接口——VBE(VESA BIOS Extension)标准,在软件接口层次上实现了各种SVGA显示卡之间的兼容性。


时至今日,或许有些显示卡已不兼容VGA标准,但是所有的显示卡厂商都无一例外地支持VBE标准。几乎所有的Super VGA卡都提供了符合VESA SVGA标准的扩展BIOS。通过一组int 10h, AH="4Fh中断调用",使用VESA SVGA的扩展功能而不必了解各种显示卡的硬件细节,基于该标准编写的程序就具有非常广泛的硬件兼容性。


VBE标准到已经发布过3个版本。1991年10月VESA发布了VBE1.2,这个标准规定了应用程序访问高性能显示卡的简易性接口,允许应用程序查询显示卡的特性并设置成合适的模式,包括分辨率,色彩丰富的模式。VBE1.2是现代显示卡广泛采用的标准。


1994年11月发布VBE2.0,它最重要的改进是增加了保护模式支持(VBE功能0Ah),提高了VBE的性能。另外,VESA组织还从VBE V2.0版标准开始增加了扩展VBE功能,包括显示器能源管理PM,显示器数据通道DDC等一些非常有用的功能。1998年9月发布VBE V3.0,主要增加了对显示卡,显示器扫描频率的控制和对平面显示器的控制。每个标准都完全向前兼容,在VESA的中文网站上可以免费下载到VBE3.0,网址http://www.vesa.org.cn


下面章节我们就开始讲SVGA 640*480*256色分辨率怎样调用VBE1.2 BIOS功能编程。


第二章 DOS内存和SVGA的位面结构
虽然不同的SVGA显示卡的体系结构不同,但它们最初都是从标准VGA的结构上扩充而来的,包括五大功能模块,即显示控制器、定序器,属性控制器、图形控制器和显示存储器(VRAM)。SVGA卡640*480*256色的显示模式也和VGA卡的256色模式类似,每个像素点8位,整个VRAM地址空间按扫描行连续存放,超过 64K的地址空间采用位面映射机制分块影射到主机提供的地址上。主机提供的地址空间叫做窗口,SVGA把VRAM分成小块(称为位面),每块64K,分别映射到窗口上。


下面的表格就是是DOS系统中的内存安排,注意从地址A0000h(即640KB)开始是用于显示的视频缓冲区和BIOS的地址空间,这正是我们图形编程中要用到的地址。



在上图中“图形模式视频缓冲区”就是主机提供的窗口,大小是64KB。下面的图表示640*480*256色显示模式的位面映射机制。



有的显示模式窗口大小和位面大小不一定是64KB,有可能内存窗口小,显示存储器(VRAM)中的每一个64K位面没有放满,也就是说VRAM中有效的显示信息不是连续的。如果要确定某个显示模式具体的窗口大小、位面尺寸和个数,颜色信息等等,就要调用VBE BIOS功能4F01h。


第三章 调色板
在SVGA 640*480*256显示模式下,一个像素对应显示存储器(VRAM)中用一个字节,每个字节表示一种颜色。但它不是一个真正的颜色,而是颜色的索引号,对应于SVGA调色板上的256个颜色寄存器。实际的颜色码来自于颜色寄存器中,每种颜色18位,红绿蓝各6位,共可以表示256K种不同的颜色,不过同时在显示器上显示的只有256种。


通过I/O端口地址3C8h和3C9h设置调色板,这样写代码:
struct COLOR
{
    BYTE R;
    BYTE G;
    BYTE B;
};
struct COLOR colors[256];


//假设颜色变量已经保存在上面的数组中


for(int i="0"; i < 256; i++)
{
    outportb(0x3C8, i);                //调色板寄存器索引号
    outportb(0x3C9, (colors.R)>>2); //传入红色分量,6位
    outportb(0x3C9, (colors.G)>>2); //传入蓝色分量,6位
    outportb(0x3C9, (colors.B)>>2); //传入绿色分量,6位
}


颜色从哪里来,在本目录中有个颜色表文件ColPal.col,每个颜色4个字节,依次是蓝、绿、红、空,把它读出来放到颜色数组中。它的颜色表就是下面的样子:


 


第三章 VBE BIOS功能调用
这一章写怎样使用VBE BIOS功能编写DOS下的图形界面程序,更详细的编程参考资料也放在了本目录中,一个是VBE3.0标准<VBE Core Functions Standard 3.0(1998).pdf>,还有一个详细说明BIOS int 10h功能调用的文件<int 10h.txt>。VBE功能调用有一个约定,或者说是规定:
1. AH必须等于4Fh,说明示调用VBE功能。
2. AL等于VBE的功能号,其中0≤AL≤0Bh;
3. BL等于子功能号,也可以没有子功能;
4. 调用int 10h;
5. 返回值均在AX中,回想起调用Windows API也是把返回值放到AX中。对VBE功能的调用一般需检查AX中的返回值,常见的返回值有:(1)功能调用成功,返回AX = 004Fh;(2)不支持该功能,一般返回AX = 4F00h;(3)支持该功能但该功能调用失败,返回AX = 014Fh。
6. 返回值的含义如下:
(1) AL = 4Fh:支持该功能;
(2) AL != 4Fh:不支持该功能;
(3) AH = 00h:功能调用成功;
(4) AH = 01h:功能调用失败;
(5) AH = 02h:当前的硬件配置不支持该功能;
(6) AH = 03h:当前的显示模式不支持该功能。


1 进入SVGA彩色模式
__asm
{
    mov AX, 4F02h
    mov BX, 101h   //显示模式 640 * 480 * 256色
    int 10h
}
常用的显示模式如下表所示,VBE标准涉及到的最大分辨率就是1280*1024,更高级的显示模式可以由厂家自己定义。
BX
像素分辨率 * 颜色数
101h
640 * 480 * 256
103h
800 * 600 * 256
105h
1024 * 768 * 256
111h
640 * 480 * 64K
112h
640 * 480 * 16M
114h
800 * 600 * 64K
115h
800 * 600 * 16M
118h
1024 * 768 * 16M


2 返回某个显示模式的信息
有时候不能确定显示模式的窗口和位面大小是不是64KB,就用这个函数确定一下。


int NumberOfPlanes;         //这三个变量保存模式的参数
int WinGran;
int WinSize;


struct ModeInfo mode_info;  //模式信息块
WORD segx = FP_SEG(mode_info);
WORD offx = FP_OFF(mode_info);
BYTE result;


__asm
{
    mov AX, 4F01h   //功能号
    mov CX, 118h    //显示模式
    mov ES, segx
    mov DI, offx    //ES:DI指向模式信息块的指针
    int 10h
    mov result, AH
}
if(result == 0x4F)  //调用成功,显示卡支持该功能
{
    NumberOfPlanes = mode_info.NumberOfPlanes; //位平面的个数
    WinGran = mode_info.WinGran;  //位面大小(窗口粒度),以KB为单位
    WinSize = mode_info.Winsize;  //窗口大小,以KB为单位
}


VBE把特定模式的信息保存在struct ModeInfo结构中,我们有时间可以了解一下,反正没有坏处。
struct ModeInfo //共256字节
{
    WORD ModeAttr;           //模式的属性
  BYTE WinAAttr, WinBAttr; //窗口A,B的属性   
  /*
    还有其他的位面-窗口映射方法中包含两个窗口,不过现在这种情况极少
    */
    WORD WinGran;             //位面大小(窗口粒度),以KB为单位
    WORD WinSize;             //窗口大小,以KB为单位
    WORD WinASeg, WinBSeg;    //窗口A,B的起始段址
  BYTE far *BankFunc;       //换页调用入口指针
  /*
  换页时可以调用该功能,也可以用VBE功能05h完成,但是直接调用该功能可以加快调用速度,因为int指令需要耗费大量的CPU周期。高性能的程序设计都是以直接调用该功能代替05h功能进行换页。(PS:我们暂时不编高性能的程序,所以使用int 10,AX=5F05h换页)
  */
    WORD BytesPerScanLine;    //每条水平扫描线所占的字节数
    WORD XRes, YRes;          //水平,垂直方向的分辨率
    BYTE XCharSize, YCharSize;//字符的宽度和高度
    BYTE NumberOfplanes;      //位平面的个数
    BYTE BitsPerPixel;        //每像素的位数
    BYTE NumberOfBanks        //CGA逻辑扫描线分组数
    BYTE MemoryModel;         //显示内存模式
    BYTE BankSize;            //CGA每组扫描线的大小
    BYTE NumberOfImagePages;  //可同时载入的最大满屏图像数
    BYTE reserve1;            //为页面功能保留


    //对直接写颜色模式的定义区域
    BYTE RedMaskSize;          //红色所占的位数
    BYTE RedFieldPosition;     //红色的最低有效位位置
    BYTE GreenMaskSize;        //绿色所占位数
    BYTE GreenFieldPosition;   //绿色的最低有效位位置
    BYTE BlueMaskSize;         //蓝色所占位数
    BYTE BlueFieldPosition;    //蓝色最低有效位位置
    BYTE RsvMaskSize;          //保留色所占位数
    BYTE RsvFieldPosition;     //保留色的最低有效位位置
    BYTE DirectColorModeInfo;  //直接颜色模式属性


    //以下为VBE2.0版本以上定义
    BYTE far *PhyBasePtr;      //可使用的大的帧缓存时为指向其首址的32位物理地址
    DWORD OffScreenMenOffset;  //帧缓存首址的32位偏移量
    WORD OffScreenMemSize;     //可用的,连续的显示缓冲区,以KB为单位


    //以下为VBE3.0版以上定义
    WORD LinBytesPerScanLine;   //线形缓冲区中每条扫描线的长度,以字节为单位
    BYTE BnkNumberOfImagePages; //使用窗口功能时的显示页面数
    BYTE LinNumberOfImagePages; //使用大的线性缓冲区时的显示页面数
    BYTE LinRedMaskSize;        //使用大的线性缓冲区时红色所占位数
    BYTE LinRedFieldPosition;   //使用大的线性缓冲区时红色最低有效位位置
    BYTE LinGreenMaskSize;      //使用大的线性缓冲区时绿色所占的位数
    BYTE LinGreenFieldPosition; //使用大的线性缓冲区时绿色最低有效位位置
    BYTE LinBlueMaskSize;       //使用大的线性缓冲区时蓝色所占的位数
    BYTE LinBlueFieldPosition;  //使用大的线性缓冲区时蓝色最低有效位位置
    BYTE LinRsvMaskSize;        //使用大的线性缓冲区时保留色所占位数
    BYTE LinRsvFieldPosition;   //使用大的线性缓冲区时保留色最低有效位位置
    BYTE reserve2[194];         //保留


}


3 保存和恢复视频状态
在实现屏幕保护的时候用到这个功能,如果发现一段时间没有输入,就把显示器关闭,等到输入设备有动作的时候就恢复显示器。在把视频状态保存到缓冲区之前,先确定缓冲区的大小。
int size = 0;
__asm
{
    mov AX, 4F04h  //功能号
    mov DX, 0      //子功能
    mov CX, 1      //CX==1表示保存硬件控制器状态
    int 10h
    mov size, BX   //返回得到缓冲区大小,以64字节为单位
}
假设得到的size大小是128字节(64 * 2),下面接着写保存硬件控制器状态的程序:
BYTE* vga_mode[128];
WORD segx = FP_SEG(vga_mode);
WORD offx = FP_OFF(vga_mode);
__asm
{
    mov AX, 4F04h
    mov DX, 1       //子功能——保存
    mov CX, 1       //保存硬件控制器状态
    mov ES, segx
    mov BX, offx
    int 10h
}
恢复:
__asm
{
    mov AX, 4F04h
    mov DX, 2       //子功能——恢复
    mov CX, 1       //恢复硬件控制器状态
    mov ES, segx
    mov BX, offx
    int 10h
}
dongsuoying发明了发现的关闭显示器的一种方法,挺好玩:
//首先:
BYTE* vga_mode[128];
for(int i="0"; i < 128; i++)
    vga_mode = 0;
//然后:
WORD segx = FP_SEG(vga_mode);
WORD offx = FP_OFF(vga_mode);
__asm
{
    mov AX, 4F04h
    mov DX, 2       //子功能--恢复
    mov CX, 1       //恢复硬件控制器状态
    mov ES, segx
    mov BX, offx
    int 10h
}


4 选择位面
从前面的位面-窗口映射图上可以看到,因为内存窗口太小,不可能包含整个显示区域,所以一定要选择位面,在作图过程中会不停地选择位面。
void SelectPlane(int page)
{
    __asm
    {
        mov AX, 4F05h
        mov BX, 0      //表示当前窗口
        mov DX, page   //在显示存储器中的位面号
        int 10h
    }
}
选择位面的意思就是说,在内存的视频缓冲区中作图时,它会在显示器的哪一块区域显示出来。


5 写像素
这是作图的最基本的元素,我们就理解成在内存中640K地址的的一块区域描点,然后显示卡通过某种机制,把这些点送到在显示器上。下面就是描点函数,参数x, y是屏幕上的坐标,color是颜色索引值, 在640*480*256色显示模式中:
void SetPixel(int x, int y, BYTE color)
{
    //先求线性坐标
    int pos = y * 640 + x;
    //然后求对应显示存储器中(VRAM)的的位面,一个位面大小是64KB
    int page = pos / (64 * 1024);
    //选择位面
    SelectPlane(page);
    //图形模式视频缓冲区地址
    BYTE far * VramBase = (BYTE far *)0xA0000000L;
    //在"窗口"画点
 *(VramBase + pos) = color;
}


可能有人问内存中的"窗口"才64K,(VramBase + pos)越界了怎么办?以前我也这么想,但是碰到高人指点了一下就醒悟了。原来VramBase是个<段地址:偏移地址>形式的远指针,VramBase再加上pos不会改变段地址,如果pos超过了64K,最终地址就会在段内折返回来。*(VramBase + pos) = color;语句的功能和*(VramBase + (pos % (64 * 1024))) = color;语句的功能是一样的。读像素和写像素的方法相同。


6 构造自己的图形函数库
Borland C++没有提供SVGA显示模式的图形函数库,据说有可以使用的商业图形库,但它们编译后的占用的空间太大了,有点划不来。所以必须根据需要,自己动手写一些常用的图形函数,比如画线,圆,长方形,填充等等。下面就要画一条简单的水平线。
void DrawLineH(int y, int x1, int x2, BYTE color)
{
    int x;
    if(x1 > x2) //为了方便循环,交换一下
    {
        x = x1;
        x1 = x2;
        x2 = x;
    }
    for(x = x1; x <= x2; x++)
    {
        SetPixel(x, y, color); //写像素
    }
}
不管任何作图函数,都要写像素,所以要把写像素的函数优化。比如,把SetPixel()做成内联函数,在选择位面之前先判断一下是否有必要,如果位面没有改变就不用去选择了。其它的作图函数也是要用写像素的方法实现,不过更要麻烦一些。dongsuoying在过电压程序中创造了很多函数,请大家编程时参考。


7 显示汉字
假如一个16*16的点阵汉字,它的字模存放在BYTE Pattern[32]数组中,数组每两个字节表示一行,每一位表示在屏幕上显示的一个点。
void ShowHanZiSample(int x, int y, BYTE color, BYTE* Pattern)
{
    //掩码
  static BYTE mask[8]={0x80, 0x40, 0x20, 0x10, 0x08, 0x04, 0x02, 0x01};
  
    for(int i="0"; i < 16; i++)
    {
        for(int j="0"; j < 16; j++)
        {
            BYTE b1 = Pattern[i * 2 + j / 8];  //读字模
            b1 = b1 & mask[j % 8];             //这个位置是0还是1
            if(b1)                             //是不是要显示这个点
                SetPixel(x + j, y + i, color); //描点         
        }   
    }
}
显示ASCII字符也一样,就是把字模拷贝到屏幕上。实际中应用是先根据汉字或字符的编码,在字库中查找字模,然后根据字模在屏幕上画出来。总之,点阵字符很简单。还有一些图形拷贝运算函数图形加速技术,介于篇幅就不易以介绍了。


8 退出SVGA图形模式
回到默认字符模式:
__asm
{
    mov AX, 03h
    int 10h
}
VGA BIOS功能调用,AH=0子功能是设置模式,AL=3是将要设置的模式编号,对于VGA来说AL=3代表了字符模式,80*25,16色。


第四章 鼠标
鼠标和VBE好像不相干,但是图形界面中必须的工具。DOS7.10中包含鼠标驱动程序CTMOUSE.EXE,在config.sys文件中增加一行
DEVICE=C:\DOS71\CTMOUSE.EXE
就可以在程序中应用鼠标功能了,但是要注意,一个只有5K的驱动程序不会把鼠标的样子也在屏幕上画出来的,所以我们如果想在程序运行的时候看到鼠标,必须自己动手画出来。据说也有的驱动程序能自己画鼠标,不过我们写过的程序还暂时用不到,所以不去关心它了。


学过SVGA之后,我想即使驱动程序自己可以画出鼠标的样子来,这个程序也是相当的复杂。因为在显卡功能标准还不统一的古代,它要判断显示卡是哪一种,显示模式是哪一种,然后才能决定怎样作图。即使大多数情况下能画出来,我想它也未必认得“先进的”SVGA 800 * 600 * 256色模式。所以要是真的想看到鼠标的话,还是不要偷懒,自己动手画出来。


鼠标功能通过BIOS的int 33h调用。
1 检测是否安装了鼠标
int check;
__asm
{
    mov AX, 0
    int 33h
    mov check, AX          //返回AX=0是未安装,AX=1是安装
}
if(check)
    printf("Installed");   //安装了
else
    printf("not install"); //未安装
2 设置鼠标移动范围
如果我们的屏幕分辨率是640 * 480,那么就这样设置鼠标的范围:
__asm
{
    mov AX, 7
    mov CX, 0
    mov DX, 639 //设置鼠标水平范围0-639
    int 33h
    mov AX, 8
    mov CX, 0
    mov DX, 479 //设置鼠标垂直范围0-479
    int 33h
}


3 显示和隐藏鼠标
会显示一个默认的鼠标形状,这个功能有时候能用,很多时候用不了。在DosBox模拟器中可以显示,调试的时候可能用得着。
__asm
{
    mov AX, 1 //显示
    int 33h
}
__asm
{
    mov AX, 2 //隐藏
    int 33h
}
4 得到鼠标位置和按钮状态
int mouseX, mouseY //保存鼠标的坐标
int mouseBtn
__asm
{
    mov AX, 3           //BIOS鼠标功能3
    int 33h
    mov mouseX, CX      //在屏幕上横坐标
    mov mouseY, DX      //在屏幕上纵坐标
    mov mouseBtn, BX    //鼠标按键状态
}
if(mouseBtn == 1)       //鼠标左键按下了
{
    printf("mouse left key down,X=%d Y=%d !\n", mouseX, mouseY);
}


在嵌入汇编语句时的注意事项
在C语言中嵌入汇编有100个好处,但是要注意:不要去改变DS,SS, BP,SP寄存器的值。比较可靠的方法是,除了AX, BX,CX, DX四个通用寄存器和SI, DI两个变址寄存器外,其他的寄存器在使用后一定要恢复过来。由于C语言中的寄存器变量实际上使用SI和DI,所以在函数中有寄存器变量的时候也不要改变SI和DI的值。


总结
VESA / SVGA和图形学编程的内容完全可以写几大套专著出来,但是我们在实际用到的功能不太复杂,而且局限在DOS实模式编程,所以本篇就先写这么多。感谢dongsuoying提供的程序,终于写完了!

PARTNER CONTENT

文章评论0条评论)

登录后参与讨论
EE直播间
更多
我要评论
0
5
关闭 站长推荐上一条 /3 下一条