Tag Archives: ASM
[ASM] 是男人就下100层,交出序列号来
用汇编实现符串操作函数
不管是在系统开发还是在平时的编程当中,字符串操作都是很重要的一部分。在C中,已经有库提供了strcpy、strcmp、strcat等函数, 而在开发用汇编开发自己的系统时,并没有现在的库可用,这就要求我们自己来实现字符串操作了。以下如果没有特别说明,字符串均以0为结束标志。
strcpy 字 符串复制
在字符串复制当中,为了简便,可以像在C中一样,不考虑边界问题,把这个问题交调用者,不过这样就有可能产生缓冲区溢出 了:)字符串复制还是比较容易实现的,只要在复制每一个字节之前判断是不是0,如果是就结束,不是则继续复制下一个字节。我给出一个简单的例子,当然,你 可以把它优化以产生更好的性能。
strcpy:
; in si 源字符串起始地址
; di 目标地 址
; out 无
push si
push di ; 保护寄存器
next:
lodsb ; 载 入一个字节
or al,al ; 是0吗?
je end ; 是则结束
stosb ; 不 是则放入目标中
jmp next ; 继续下一个字节
end:
mov [di],byte 0 ; 结 束标志
pop di ; 恢复寄存器
pop si
ret
strlen 取 字符串长度
应该说,这一个比上一个容易,因为这个只需要考虑什么结束,而不需要去复制字节。
strlen:
; in si 源 字符串
; out ax 字符串长度
push si
push cx ; 保护寄存器
xor cx,cx ; 计 数器清零
next:
lodsb ; 载入一个字节
or al,al ; 是0?
je end ; 是 则结束
inc cx ; 字符串长度+1
jmp next ; 继续下一个字节
end:
mov ax,cx ; 将 字符串长度放到AX中作为返回值
pop bx ; 恢复寄存器
pop si
ret
strcat 字 符串连接
就个人来说,字符串连接用得并不是很多,但既然在C中有这个函数,就必然有它存在的理由,所以我们还是有必要来实现它的。 同样的,在字符串连接时不考虑目标缓冲区是否足够的问题,把这个交给调用者。
要将一个字符串连接到另一个字符之后,就需要先找出目 标字符串的结尾,即0的地址,然后就可以把这个地址做为目标,把需要连接的字符串的首地址做为源,调用strcpy即可完成。而找出0的地址,原理和取字 符串长度一样,载入一个字节然后判断是否为0。这个留大家自己去实现。
上面的例子使用的寄存器都是16位的,所以只能用在16位的 程序当中,当然,如果是你自己写的,那么自己懂得它实现的原理,移植到32位就是件很容易的事。
今天这篇就到这里喽~~难得写教 程之类的文章,今天还算顺手^-^~~
实模式进保护模式
在写自己的操作系统时,不能总在实模式下,进入保护模式才是”正道”,不过怎么进入保护模式又是一个问题,我把
XuOS里进入保护模式的代码分析一下.
; 装入GDT
mov eax,ds ;设置GDT在物理内存中的正确位置
shl eax,4
add [gdt_addr+2],eax
cli ; 关中断
lgdt [gdt_addr] ;载入GDT
; 下面打开A20地址线,这段代码可以在OSzone上找到相关解释.
call Empty_8042
mov al,0xd1
out 0×64,al
call Empty_8042
mov al,0xdf
out 0×60,al
call Empty_8042
; 进入保护模式
mov eax,cr0 ;置PE位
or eax,1
mov cr0,eax
jmp oscodesel:code_32 ; 跳到32位代码处执行
Empty_8042:
in al,0×64
test al,0×2
jnz Empty_8042
ret
; GDT的内容
gdt:
gdt_null:
dd 0×0000
dd 0×0000
gdt_system_code:
oscodesel equ $-gdt ; 段选择子
dd 0x0000ffff,0x00cf9a00
gdt_system_data:
osdatasel equ $-gdt
dd 0x1000ffff,0x00cf9200
videosel equ $-gdt
dd 0x0000ffff,0x00cf920a
gdt_addr:
dw gdt_addr-gdt-1
dd gdt ; ;GDT表的位置
当然,由于XuOS目前还比较简单,还没有设置IDT,GDT中段也比较少,况且我的理解也有可能有些偏差或者表述
有些不清楚,这里只提供一个思路,更多更强的功能还是要靠自己才行.
写你自己的操作系统
这是转载的,原文可以在http://www.xemean.net的文档中心里找到.
因为原文中的代码编译后运行有错误,这里我把改过后能正确运行的代码讲一下
org 0x07c00 ; 起始地址是0000:7c00
jmp begin_boot ; 跳过其它的数据,跳转到引导程序的开始处
OEM_ID db ”OSeg ” ;软盘信息,具体请参考”FAT格式”
BytesPerSector dw 0×0200
SectorsPerCluster db 0×01
ReservedSectors dw 0×0001
TotalFATs db 0×02
MaxRootEntries dw 0x00E0
TotalSectorsSmall dw 0x0B40
MediaDescriptor db 0xF0
SectorsPerFAT dw 0×0009
SectorsPerTrack dw 0×0012
NumHeads dw 0×0002
HiddenSectors dd 0×00000000
TotalSectorsLarge dd 0×00000000
DriveNumber db 0×00
Flags db 0×00
Signature db 0×29
VolumeID dd 0xFFFFFFFF
VolumeLabel db ”OSexample ”
SystemID db ”FAT12 ”
print_mesg: ;打印信息调用
mov ah,0×13 ; 使用中断10h的功能13,在屏幕上写一个字符串
mov al,0×00 ; 决定调用函数后光标所处的位置
mov bx,0×0007 ; 设置显示属性
mov cx,0×20 ; 在此字符串长度为32
mov dx,0×0000 ; 光标的起始行和列
int 0×10 ; 调用BIOS的中断10h
ret ; 返回调用程序
get_key: ;等待按键
mov ah,0×00
int 0×16 ; Get_key使用中断16h的功能0,读取下一个字符
ret
clrscr: ;清屏
mov ax,0×0600 ; 使用中断10h的功能6,实现卷屏,如果al=0则清屏
mov cx,0×0000 ; 清屏
mov dx,0x174f ; 卷屏至23,79
mov bh,0 ; 使用颜色0来填充
int 0×10 ; 调用10h中断
ret ;返回
begin_boot: ;;引导程序开始
mov ax,cs ;设置段寄存器
mov ds,ax
mov es,ax
call clrscr ; 先清屏
mov bp,bootmesg ; 提供串地址
call print_mesg ; 输出信息
call get_key ; 等待用户按下任一键
bits 16 ;以16位方式编译
call clrscr ; 清屏
mov ax,0xb800 ; 使gs指向显示内存
mov gs,ax ; 在实模式下显示一个棕色的A
mov word [gs:0],0×641 ; 显示
call get_key ; 调用Get_key等待用户按下任一键
mov bp,pm_mesg ; 设置串指针
call print_mesg ; 调用print_mesg子程序
call get_key ; 等待按键
call clrscr ; 清屏
cli ; 关中断
mov eax,ds ;设置GDT物理地址
shl eax,4
add [gdtr+2],eax
lgdt [gdtr] ; 加载GDT
mov eax,cr0 ;进入保护模式
or al,0×01 ; 设置保护模式位
mov cr0,eax ; 将更改后的字送至控制寄存器中
jmp codesel:go_pm ;跳至32位代码
bits 32 ;以32位方式编译
go_pm:
mov ax,datasel ;设置段寄存器
mov ds,ax ; 初始化ds和es,使其指向数据段
mov es,ax
mov ax,videosel ;使gs指向显存
mov gs,ax
mov word [gs:0],0×0400+’B' ; 在保护模式下显示一个红色的字符B
jmp $ ; 无限循环
bits 16
gdtr:
dw gdt_end-gdt-1 ; gdt的长度
dd gdt ; gdt的物理地址
gdt: ;具体GDT信息请参考GDT格式
nullsel equ $-gdt ; $指向当前位置,所以nullsel = 0h
dd 0
dd 0 ; 所有的段描述符都是64位的
codesel equ $-gdt ; 这是8h也就是gdt的第二个描述符
code_gdt:
dw 0x0ffff ; 段描述符的界限是4Gb
dw 0×0000
db 0×00
db 0x09a
db 0x0cf
db 0×00
datasel equ $-gdt
data_gdt:
dw 0x0ffff
dw 0×0000
db 0×00
db 0×092
db 0x0cf
db 0×00
videosel equ $-gdt
dw 3999
dw 0×8000 ; 基址是0xb8000
db 0x0b
db 0×92
db 0xcf
db 0×00
gdt_end:
bootmesg db ”Our OS boot sector loading… ” ;信息数据
pm_mesg db ”Switching to protected mode… ”
times 510-($-$$) db 0 ;填满512个字节
dw 0x0AA55 ;引导扇区标志
Recent Comments