本文记录了在JOS(或在任意OS)上实现图形界面的方法与一些图形库的实现。
本文中支持的新特性:

  • 支持基本图形显示
  • 支持中英文显示(中英文点阵字库)

相关:VBE VESA MMIO 点阵字库
Github : https://github.com/He11oLiu/MOS

About VESA

Video Electronics Standards Association(视频电子标准协会,简称“VESA”)是制定计算机和小型工作站视频设备标准的国际组织,1989年由NEC及其他8家显卡制造商赞助成立。创立VESA的原始目的是要制定分辨率为800x600的SVGA视频显示标准。其后,VESA公告一系列的个人电脑视频周边功能的相关标准。

VBE 功能调用

参考博客CSDN博客

VBE功能调用

  • AH必须等于4FH,表明是VBE标准
  • AL等于VBE功能号,0<= AL <= 0BH
  • BL等于子功能号,也可以没有子功能
  • 调用INT 10H
  • 返回值在AX中
    • AL=4FH:支持该功能
    • AL!=4FH:不支持该功能
    • AH=00H:调用成功
    • AH=01H:调用失败
    • AH=02H:当前硬件配置不支持该功能
    • AH=03H:当前的显示模式不支持该功能

具体功能

此部分参考VESA编程——GUI离我们并不遥远,原作者博客已关闭。

功能0x00:返回控制器信息

输入:
AX = 4F00h 返回VBE控制器信息
ES:DI = 指向存放VbeInfoBlock结构体的缓冲区指针
输出:
AX = VBE返回状态

这个函数返回一个VbeInfoBlock结构体,该结构体定义如下:

// Vbe Info Block
typedef struct {
unsigned char vbe_signature;
unsigned short vbe_version;
unsigned long oem_string_ptr;
unsigned char capabilities;
unsigned long video_mode_ptr;
unsigned short total_memory;
unsigned short oem_software_rev;
unsigned long oem_vendor_name_ptr;
unsigned long oem_product_name_ptr;
unsigned long oem_product_rev_ptr;
unsigned char reserved[222];
unsigned char oem_data[256];
} VbeInfoBlock;
  • vbe_signature是VBE标识,应该填充的是”VESA”
  • vbe_version是VBE版本,如果是0300h则表示3.0版本
  • oem_string_ptr是指向oem字符串的指针,该指针是一个16位的selector:offset形式的指针,在实模式下可以直接使用。
  • video_mode_ptr是指向视频模式列表的指针,与oem_string_ptr类型一样
  • total_memory是64kb内存块的个数
  • oem_vendor_name_ptr是指向厂商名字符串的指针
  • oem_product_name_ptr是指向产品名字符串的指针

功能01 返回VBE模式信息

输入:
AX = 0x4F01 返回VBE模式信息
CX = 模式号
ES:DI = 指向VBE特定模式信息块的指针
输出:
AX = VBE返回值

这个函数返回一个ModeInfoBlock结构体,其中重要的部分如下:

  • mode_attributes字段,这个字段描述了图形模式的一些重要属性。其中最重要的是第4位和第7位。第4位为1表示图形模式(Graphics mode),为0表示文本模式(Text mode)。第7位为1表示线性帧缓冲模式(Linear frame buffer mode),为0表示非线性帧缓冲模式。我们主要要检查这两个位。
  • xresolution,表示该视频模式的X分辨率。
  • yresolution,表示该视频模式的Y分辨率。
  • bits_per_pixel,表示该视频模式每个像素所占的位数。
  • phys_base_ptr,这是一个非常重要的字段,它给出了平坦内存帧缓冲区的物理地址,你可以理解为显存的首地址。如果每个像素占32位的话,屏幕左上角第一个点所占的缓冲区就是phys_base_ptr所指的第一个4个字节。按照先行后列的顺序,每个像素点所占缓冲区依次紧密排列。我们要想在屏幕上画出像素点,就得操作以phys_base_ptr为起始的物理内存空间。

功能02 设置VBE模式信息

输入:
AX = 4F02h 设置VBE模式
BX = 需要设置的模式
D0 - D8 = 模式号
D9 - D10 = 保留(必须为0)
D11 = 0 使用当前缺省刷新率
= 1 使用用户指定的CRTC值为刷新率
D12 - D13 = 为VBE/AF保留(必须为0)
D14 = 0 使用窗口帧缓冲区模式
= 1 使用线性/平坦帧缓冲区模式
D15 = 0 清除显示内存
= 1 不清除显示内存
ES:DI = 指向CRTCInfoBlock结构体的指针 输出:
AX = VBE返回状态

JOS实现

  • Qemu需要添加-vga std

    QEMUOPTS = -drive file=$(OBJDIR)/kern/kernel.img,index=0,media=disk,format=raw -serial mon:stdio -gdb tcp::$(GDBPORT) -vga std
  • boot中实模式中获取VBE,设置VBE

    sti
    call getvideomode
    call setvideomode
    cli

    先获取VBE模式,填充di,然后切换模式,设置VBE模式。

    根据上面查的VESA资料,调用函数,实现这两个功能。

    getvideomode:
    mov $0x4f01, %ax # get mode
    mov $0x105, %cx # mode 0x105
    mov $0x8000, %di # mode info block address
    int $0x10 # VBE int
    ret setvideomode:
    movw $0x4f02, %ax # set mode
    movw $0x105, %bx
    movw $0x8000, %di
    int $0x10 # VBE int
    movl 40(%di), %eax # get memory address
    movl %eax, info_vram
    movw 18(%di), %ax # get x resolution
    movw %ax, info_scrnx
    movw 20(%di), %ax # get y resolution
    movw %ax, info_scrny
    ret

    这里设计了一个结构体来存放从boot传来的东西。

    struct boot_info
    {
    short scrnx, scrny;
    char *vram;
    };
  • init的时候,设计一个获取boot_info的模块

    static void get_boot_info(void)
    {
    struct boot_info *info = (struct boot_info *)(KADDR(0x0ff0));
    // Init Graph info
    graph.scrnx = info->scrnx;
    graph.scrny = info->scrny;
    graph.vram = info->vram;
    }

    这个地方我选择初始化memory layout之后,开启真正的页表的时候才获取信息。所以这里要用KADDR进行物理地址到KVA的转化。

  • 设计一个全局用于保存图像相关信息的结构体

    struct graph_info
    {
    short scrnx,scrny;
    char *vram;
    }; extern struct graph_info graph;
  • 利用MMIO映射一片显存

    void graph_init()
    {
    int i;
    // Init Graph MMIO
    graph.vram =
    (char *)mmio_map_region((physaddr_t)graph.vram,
    graph.scrnx * graph.scrny);
    cprintf("====Graph mode on====\n");
    cprintf(" scrnx = %d\n",graph.scrnx);
    cprintf(" scrny = %d\n",graph.scrny);
    cprintf("MMIO VRAM = %#x\n",graph.vram);
    cprintf("=====================\n");
    // Draw Screen
    for (i = 0; i < graph.scrnx * graph.scrny; i++)
    *(graph.vram + i) = 0x34;
    }

补充图像库

上面基本已经实现了图像显示基本平台。现在补充一些常用的图像库。

#define PIXEL(x, y) *(graph.vram + x + (y * graph.scrnx))
int draw_screen(uint8_t color)
{
int i;
for (i = 0; i < graph.scrnx * graph.scrny; i++)
*(graph.vram + i) = color;
return 0;
} int draw_pixel(short x, short y, uint8_t color)
{
if ((x >= graph.scrnx) || (y >= graph.scrny))
return -1;
*(graph.vram + x + (y * graph.scrnx)) = color;
return 0;
} int draw_rect(short x, short y, short l, short w, uint8_t color)
{
int i, j;
w = (y + w) > graph.scrny ? graph.scrny : (y + w);
l = (x + l) > graph.scrnx ? graph.scrnx : (x + l);
for (j = y; j < w; j++)
for (i = x; i < l; i++)
*(graph.vram + i + j * graph.scrnx) = color;
return 0;
}

字库实现

这部分也是老生常谈了,板子上各种系统都实现过点阵字库。

int draw_ascii(short x, short y, char *str, uint8_t color)
{
char *font;
int i, j, k = 0;
for (k = 0; str[k] != 0; k++)
{
font = (char *)(ascii_8_16 + (str[k] - 0x20) * 16);
for (i = 0; i < 16; i++)
for (j = 0; j < 8; j++)
if ((font[i] << j) & 0x80)
PIXEL((x + j), (y + i)) = color;
x += 8;
}
return k;
} int draw_cn(short x, short y, char *str, uint8_t color)
{
uint16_t font;
int i, j, k;
int offset;
for (k = 0; str[k] != 0; k += 2)
{
offset = ((char)(str[k] - 0xa0 - 1) * 94 +
((char)(str[k + 1] - 0xa0) - 1)) *
32;
for (i = 0; i < 16; i++)
{
font = cn_lib[offset + i * 2] << 8 |
cn_lib[offset + i * 2 + 1];
for (j = 0; j < 16; j++)
if ((font << j) & 0x8000)
PIXEL((x + j), (y + i)) = color;
}
x += 16;
}
return 0;
}

直接把之前单片机的点阵字库拿过来,不过单片机当时开发环境是win,找的字库寻址模式是GB2312的。这里把这个文件的编码改为GB2312来正确编码中文即可。实现效果见文章头。

frambuffer

实际上,直接对显存写是很不负责任的行为。很早之前在写java的界面的时候,就接触了双缓冲技术,其实与显示有关的思想都是差不多的,我们应该提供一个framebuffer。当完成一个frame后,再将这个frame update到显存中。

uint8_t *framebuffer;
void init_framebuffer(){
if((framebuffer = (uint8_t *) kmalloc((size_t)(graph.scrnx*graph.scrny)))== NULL)
panic("Not enough memory for framebuffer!");
} void update_screen(){
memcpy(graph.vram,framebuffer,graph.scrnx*graph.scrny);
}

经过实现kmallockfree,已经可以分配这个缓冲区,并直接向缓冲区写入,最后再进行update

#define PIXEL(x, y) *(framebuffer + x + (y * graph.scrnx))
int draw_xx()
{
xxx;
update_screen();
}

总结

至此,基本的GUI底层接口已基本实现,后面的就是各种数据结构的设计,窗口树设计之类。这里暂不打算继续深究,转而研究其余内核的东西。

[自制操作系统] 图形界面&VBE工具&MMIO显存&图形库/字库的更多相关文章

  1. 10款最好用的MySQL数据库客户端图形界面管理工具

    MySQL Workbench 该工具由MySQL开发,是一个跨平台的可视化数据库设计工具.它是DBDesigner4项目备受期待的替代者,它是一个本地图形化工具,支持的操作系统包括Windows.L ...

  2. Windows Azure Storage图形界面管理工具

    上一篇我们介绍了用PowerShell将Windows Azure的存储服务当网盘来使用.如果感觉还不够简单,那么这次我们来看看还有哪些使用起来更方便的图形界面管理工具吧.当然,这些工具必要支持中国版 ...

  3. redis的图形界面管理工具:phpredisadmin

    大部分人都知道redis是一款用在缓存服务器上的软件,它与memcache类似,都可以存储海量的数据,用在大访问量的web网站.聊天记录存放等方面,但是又与memcache不同: 1.缓存数据可以持久 ...

  4. Cordova - Windows版本图形界面管理工具,告别命令行输入方式!

    Cordova本身提供的是命令行管理工具,并没有提供图形界面管理工具,虽然命令行管理工具可以完成所有Cordova管理,但是对于我这种懒蛋,可真不希望每次都输入命令,而且我更担心一旦输错一个字符,命令 ...

  5. 使用图形界面管理工具Navicat for MySQL连接Mysql数据库时提示错误:Can't connect to MySQL server (10060)

    版权声明:本文为 testcs_dn(微wx笑) 原创文章,非商用自由转载-保持署名-注明出处,谢谢. https://blog.csdn.net/testcs_dn/article/details/ ...

  6. redis的图形界面管理工具

    大部分人都知道redis是一款用在缓存服务器上的软件,它与memcache类似,都可以存储海量的数据,用在大访问量的web网站.聊天记录存放等方面,但是又与memcache不同: 1.缓存数据可以持久 ...

  7. 搭建KVM环境——07 带GUI的Linux上安装KVM图形界面管理工具

    清空yum源缓存,并查看yun源 [root@CentOS2 ~]# yum clean all Loaded plugins: fastestmirror, langpacks Cleaning r ...

  8. Java GUI图形界面开发工具

    Applet 应用程序     一种可以在 Web 浏览器中执行的小程序,扩展了浏览器中的网页功能. 缺: 1.需要下载 Applet 及其相关文件 2.Applet 的功能是受限制的 优: 3.无需 ...

  9. kafka集群图形界面管理工具kafka-manager

    应用说明: 图形web相对于命令行很多时候显得更直观,kafka-manager是yahoo开源出来的项目,web界面还挺好用,安装更是很便捷. 安装环境: 具体安装: 1. 下载已经编译好的zip包 ...

随机推荐

  1. 用u盘装系统,进入bios后没有usb启动项怎么办

    开机按DEL进入BIOS(现在还这么说吧,不同的主板进入方法不太一样),找到BOOT选项. 选择Boot mood:legacy support(引导模式,逻辑支持) boot priorty:leg ...

  2. 算法竞赛入门经典 习题2-10 排列(permutation)

    习题2-10 排列(permutation) 用1,2,3,-,9组成3个三位数 abc, def, 和ghi,每个数字恰好使用一次,要求 abc:def:ghi = 1:2:3.输出所有解.提示:不 ...

  3. Unity 继承MonoBehaviour脚本 执行顺序 详解

    先看结果 Awake ->OnEnable-> Start ->-> FixedUpdate-> Update  -> LateUpdate ->OnGUI ...

  4. MQ选型对比文档

    几种MQ产品说明:     ZeroMQ :  扩展性好,开发比较灵活,采用C语言实现,实际上他只是一个socket库的重新封装,如果我们做为消息队列使用,需要开发大量的代码    RabbitMQ  ...

  5. 浅尝Java(二、代码折叠插件的使用)

    主题:eclipse代码折叠插件的使用. 工作中在使用eclipse开发Java项目时,我们会写很多if,for循环啊什么的,这使得我们的项目代码会有很多很多行.写完后要想检查或者查看,就要从头一行一 ...

  6. python学习===判断两个日期的间距天数

    import datetime   d1 = datetime.date(2015,10,7) d2 = datetime.date(2015,8,15) print((d1-d2).days)

  7. 2017年最重要的HTML5开发手册,传播正能量

    今天给大家推荐这个HTML5开发手册,希望能帮助正在学习web前端的人,鄙人也是刚学习前端没多久,借助于一点资讯平台能够结识更多前端大牛,这是我的HTML5进阶学习一点资讯群:250777811,里面 ...

  8. HDU 3682 To Be an Dream Architect:查重【三维坐标系中点在实数上的映射】

    题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=3682 题意: 有一个n*n*n的立方体,左下角坐标为(1,1,1),接下来进行m次操作. 每个操作形如 ...

  9. 解决Visual Studio For Mac Restore失败的问题

    之前就了解到微软出了mac版的VS,没太多的关注,自己也就是使用 DotNet Core SDK + VS Code 做一些小demo. 前两天发布了DotNet Core 2.0 ,Visual S ...

  10. Struts2.5的的环境搭建及跑通流程

    Struts2.5 struts是开源框架.使用Struts的目的是为了帮助我们减少在运用MVC设计模型来开发Web应用的时间.如果我们想混合使用Servlets和JSP的优点来建立可扩展的应用,st ...