手机写操作系统之 图形模式

news/发布时间2024/5/11 21:11:10

手机写操作系统之
图形模式

进入图形模式,显示图片、汉字。
首先介绍设置显卡模式中断:int 10h
查看文档找到下面资料:
Int 10/AX=4F02h - VESA SuperVGA BIOS - SET SuperVGA VIDEO MODE
这时是设置为SuperVGA模式
AX = 4F02h
BX = new video mode
我们设置成800*600模式 16位色。

Int 10/AX=4F01h - VESA SuperVGA BIOS - GET SuperVGA MODE INFORMATION
这时是读取SuperVGA模式参数
AX = 4F01h
CX = SuperVGA video mode
ES:DI指向预先保留的256字节空间,这里将保存int 10h传来的参数,其中第40个字节开始存有显卡地址的32位值。
在kernelloader.asm中又将此地址存入指定的0x10050处(值e0000000)
显示原理很简单,向显卡地址填入一个16位值,代表颜色,就可以在800X600的框架里显示一个点。
下面看程序实现:

修改kernelloader.asm,源码如下:

[BITS 16]
jmp main
gdt_entries equ 3 ;共有三个段描述符:null,os code32,os data32
pe equ 1 ;bit PE in CR0
null equ 0h
os_code32_sel equ 8h ;1,gdt,rpl=00
os_data32_sel equ 10h ;2,gdt,rpl=00
VESA: times 256 db 0 ;分配一块区域存放 vesa 返回的信息,大小256,我们只需要其中的一个32位值
pdescr times 6 db 0
gdt_table times (gdt_entries*8) db 0

set_video_mode: ;设置显卡模式
push es

;设置显卡模式
mov ax , 0x4f02
mov bx , 0x4114 ;800X600 ( 5:6:5 ) 16位色彩
int 0x10
;取得该模式下显卡地址
mov bx , 0x1000
mov es , bx
mov di , VESA ;es:di指向256空间,int 10h将在此填写数据
mov ax , 0x4f01
mov cx , 0x114
int 0x10
;第40个字节开始存有显卡地址0xe0000000,将此地址再存入指定的地址0x10050
mov eax , [ es:VESA + 40 ]
;将此地址再存入指定的地址0x10050,
mov [ es:0x50 ] , eax ;eax内容为 0xe0000000

pop es
ret
read_kernel: ;读入 kernel 程序
push es

.rk:
mov ax , 0x8000 ;kernel.bin 所在的段基址
mov es , ax
mov bx , 0 ;写入到内存0x8000:0000 物理地址=0x80000
mov ah , 2
mov dh , 0 ;磁头
mov dl , 0 ;驱动器号
mov ch , 0 ;磁道0
mov cl , 4 ;第4个扇区开始
mov al , 2 ;读入扇区数,每个扇区为 512B
int 0x13
jc .rk

pop es
ret

main:
mov ax,1000h
mov ds,ax
;设置显卡模式
call set_video_mode
;读入 kernel
call read_kernel
;打开 A 20 地址线
mov ax , 0x2401
int 0x15
;[1]built up GDT table
cli
mov eax,gdt_table
;item 0:null descriptor,
mov dword[eax],0
mov dword[eax+4],0
add eax,8
;item 1,OS code32 descriptor,
;Base=00000000h,limit=0ffh,G=1,D=1,type=a,dpl=0
mov word[eax],0ffh
mov word[eax+2],0
mov byte[eax+4],00h
mov byte[eax+5],09ah
mov byte[eax+6],0c0h
mov byte[eax+7],00h
add eax,8
;item 2,OS data32 descriptor
;Base=00000000h,Limit=0fffffh,G=1,D=1,Type=2,DPL=0
mov word[eax],0ffffh
mov word[eax+2],0000h
mov byte[eax+4],00h
mov byte[eax+5],092h
mov byte[eax+6],0cfh ;高四位是G D 0 AVL,此处为1100 = c ,低四位是limit bit 16-19 此处为f
mov byte[eax+7],00h
add eax,8
;[2]built false GDT descriptor
mov word[pdescr+0],(gdt_entries*8)
mov dword[pdescr+2],gdt_table+00010000h
lgdt [pdescr]
;[3]enter into protected mode
;刷新CR0
mov eax,cr0
or eax,pe
mov cr0,eax
jmp flush
flush:
mov ax,os_data32_sel
mov ds,ax
mov es,ax
mov ss,ax
mov fs,ax
mov gs,ax
jmp dword os_code32_sel:0x80000 ;跳转到0x8000:0000保护模式 物理地址0x80000

kernel.asm,源码如下:

[BITS 32]
jmp start
start:
mov eax,0x10050 ;参见kernelloader.asm 显卡地址指定存在这里
mov edx,dword [eax]
add edx,80600 ;加上偏移量
mov cx,50 ;循环50次
mov ax,0x7ff ;蓝色值,
next:
mov word [edx],ax ; 向地址e0013ad8存入值07ff
inc edx
inc edx
dec cx
jnz next ;cx不为0则继续
jmp $

boot.asm
[BITS 16] ;编译成16位的指令
[ORG 0x7C00]
jmp main

read_kernelloader: ;读入 kernelloader 程序
push es

.rk:
mov ax , 0x1000 ;kernelloader.bin 所在的段基址
mov es , ax
mov bx , 0
mov ah , 2
mov dl , 0
mov ch , 0
mov cl , 2
mov al , 2 ;读入扇区数,每个扇区为 512B
int 0x13
jc .rk

pop es
ret

main: ;主程序
mov ax , 0x0 ;boot.bin 程序的段基址
mov ds , ax

call read_kernelloader ;读入 kernelloader 程序

jmp dword 0x1000:0 ;跳转到 kernelloader 处执行
times 510-($-$$) db 0
db 0x55
db 0xAA

makefile

######################

声明要编译的所有组成,这里的ya是本工程名称,可以取任何名字,这里就用ya

######################
ya:out/boot.bin out/kernelloader.bin out/kernel.bin out/create_img.o out/write_in_img.o (注意这里无回车)

开始对各部分编译,注意不是空格是Tab键

out/boot.bin:code/boot.asm
nasm code/boot.asm -o out/boot.bin
out/kernelloader.bin:code/kernelloader.asm
nasm code/kernelloader.asm -o out/kernelloader.bin
out/kernel.bin:code/kernel.asm
nasm code/kernel.asm -o out/kernel.bin

制作内核映象工具

out/create_img.o:code/create_img.c
clang++ code/create_img.cpp -o out/create_img.o

写入文件,argv[1]=目标文件 argv[2]=源文件 argv[3]=写入偏移量

out/write_in_img.o:code/write_in_img.cpp
clang++ code/write_in_img.cpp -o out/write_in_img.o
######################

g.sh:

!/bin/sh

echo "=启动 ya=="
cd /sdcard/A/ya/
make

cd ~
cp -r /sdcard/A/ya/out ./
cd out
chmod 755 ./*

执行命令,当前目录下生成a.img文件

./create_img.o a.img

执行命令,向a.img写入代码,内容是boot.bin

写入磁盘位置从0偏移量起始,占1个扇区512字节

./write_in_img.o a.img boot.bin 0

执行命令,向a.img写入代码,内容是kernelloader.bin

boot.bin已经占用了512字节,写入磁盘位置从512偏移量起始,占2个扇区1024字节

./write_in_img.o a.img kernelloader.bin 512

执行命令,向a.img写入代码,内容是kernel.bin

boot.bin+kernelloader.bin已经占用了512+1024 = 1536字节,写入磁盘位置从1536偏移量起始,占1个扇区512字节

./write_in_img.o a.img kernel.bin 1536

结果复制到手机

cp -f ./a.img /sdcard/A/ya/final

cp -f ./newkernel.bin /sdcard/A/ya/final

cd ~
rm -r out

编译过程同上一节,编译完成后运行模拟器,结果显示:

一条兰色直线:

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.ulsteruni.cn/article/43566404.html

如若内容造成侵权/违法违规/事实不符,请联系编程大学网进行投诉反馈email:xxxxxxxx@qq.com,一经查实,立即删除!

相关文章

华为NPU开发流程点滴

华为NPU开发流程点滴 NPU/CPU集成操作流程图介绍了App使用HUAWEI HiAI DDK的集成流程。IR在线模型构建 IR在线模型构建通过IR单算子根据原始模型中的关系级联,配置权重数据,构建IR模型网络。 离线模型转换 离线模型转换需要将Caffe、TensorFlow、ONNX、MindSpore模型转换为HU…

拉格朗日(Lagrange)中值定理

preamble罗尔中值定理是理解拉格朗日中值定理的基础罗尔中值定理是拉格朗日中值定理的1个特殊情况泰勒中值定理是拉格朗日中值定理的推广cite: 罗尔定理: https://www.cnblogs.com/Preparing/p/18156702definition 若函数\(f(x)\)满足下列条件:\(f(x)\) 在闭区间\([a,b]\)上连…

论文解读(MAML)《Model-Agnostic Meta-Learning for Fast Adaptation of Deep Networks》

Note:[ wechat:Y466551 | 可加勿骚扰,付费咨询 ] 论文信息论文标题:Model-Agnostic Meta-Learning for Fast Adaptation of Deep Networks论文作者:Chelsea Finn、Pieter Abbeel、Sergey Levine论文来源:2017 论文地址:download 论文代码:download视屏讲解:click1-摘要…

Fastbin attackDouble free和Unsortbin leak的综合使用

Fastbin attack&&Double free和Unsortbin leak的综合使用✅ 今天做一个综合题目,包括利用Fastbin attack实现多指针指向一个地址,以及利用Unsortbin leak泄露libc基地址和修改__malloc_hook地址为one_gadget 题目是buuctf上面的一道题目,题目链接 https://buuoj.cn/…

python学习思维导图分享

python 本文包含了我的一些python学习的笔记和思维导图 第一部分:python基础导图下载链接 第二部分:函数及其他文件操作导图下载链接 第三部分:类及网络编程导图下载链接 第四部分:mysql导图下载链接

微机结构

微型计算机结构 总体来说,微型计算机的结构是采用总线结构实现相互之间的信息传递。CPU和存储器通过总线相互连接,I/O设备通过I/O接口连接在总线上。 总线是计算机各部件之间传输数据的通道,有三类总线分别是:数据总线、地址总线和控制总线(反馈)。主要特性有:公共性、分…