日韩性视频-久久久蜜桃-www中文字幕-在线中文字幕av-亚洲欧美一区二区三区四区-撸久久-香蕉视频一区-久久无码精品丰满人妻-国产高潮av-激情福利社-日韩av网址大全-国产精品久久999-日本五十路在线-性欧美在线-久久99精品波多结衣一区-男女午夜免费视频-黑人极品ⅴideos精品欧美棵-人人妻人人澡人人爽精品欧美一区-日韩一区在线看-欧美a级在线免费观看

歡迎訪問 生活随笔!

生活随笔

當前位置: 首頁 > 编程资源 > 编程问答 >内容正文

编程问答

第13章 程序的动态加载和执行(一,引导)

發布時間:2023/12/10 编程问答 27 豆豆
生活随笔 收集整理的這篇文章主要介紹了 第13章 程序的动态加载和执行(一,引导) 小編覺得挺不錯的,現在分享給大家,幫大家做個參考.

這個引導扇區的代碼沒什么困難,比較容易讀懂。較上一章不同的是,引導扇區除了給本身代碼準備GDT之外,還要給“簡易操作系統”準備GDT。本章中的引導扇區代碼,內核代碼,用戶代碼一定的要弄的清楚明白熟悉后才能進行后面的概念,因為后面的“任務切換”等概念都是建立在這三段代碼之上的,本篇只討論引導扇區代碼。

代碼如下:

;代碼清單13-1;文件名:c13_mbr.asm;文件說明:硬盤主引導扇區代碼 ;創建日期:2011-10-28 22:35 ;設置堆棧段和棧指針 core_base_address equ 0x00040000 ;常數,內核加載的起始內存地址 core_start_sector equ 0x00000001 ;常數,內核的起始邏輯扇區號 mov ax,cs mov ss,axmov sp,0x7c00;計算GDT所在的邏輯段地址mov eax,[cs:pgdt+0x7c00+0x02] ;GDT的32位物理地址 xor edx,edxmov ebx,16div ebx ;分解成16位邏輯地址 mov ds,eax ;令DS指向該段以進行操作mov ebx,edx ;段內起始偏移地址 ;跳過0#號描述符的槽位 ;創建1#描述符,這是一個數據段,對應0~4GB的線性地址空間mov dword [ebx+0x08],0x0000ffff ;基地址為0,段界限為0xFFFFFmov dword [ebx+0x0c],0x00cf9200 ;粒度為4KB,存儲器段描述符 ;創建保護模式下初始代碼段描述符mov dword [ebx+0x10],0x7c0001ff ;基地址為0x00007c00,界限0x1FF mov dword [ebx+0x14],0x00409800 ;粒度為1個字節,代碼段描述符 ;建立保護模式下的堆棧段描述符 ;基地址為0x00007C00,界限0xFFFFE mov dword [ebx+0x18],0x7c00fffe ;粒度為4KB mov dword [ebx+0x1c],0x00cf9600;建立保護模式下的顯示緩沖區描述符 mov dword [ebx+0x20],0x80007fff ;基地址為0x000B8000,界限0x07FFF mov dword [ebx+0x24],0x0040920b ;粒度為字節;初始化描述符表寄存器GDTRmov word [cs: pgdt+0x7c00],39 ;描述符表的界限 lgdt [cs: pgdt+0x7c00]in al,0x92 ;南橋芯片內的端口 or al,0000_0010Bout 0x92,al ;打開A20cli ;中斷機制尚未工作mov eax,cr0or eax,1mov cr0,eax ;設置PE位;以下進入保護模式... ...jmp dword 0x0010:flush ;16位的描述符選擇子:32位偏移;清流水線并串行化處理器[bits 32] flush: mov eax,0x0008 ;加載數據段(0..4GB)選擇子mov ds,eaxmov eax,0x0018 ;加載堆棧段選擇子 mov ss,eaxxor esp,esp ;堆棧指針 <- 0 ;以下加載系統核心程序 mov edi,core_base_address mov eax,core_start_sectormov ebx,edi ;起始地址 call read_hard_disk_0 ;以下讀取程序的起始部分(一個扇區) ;以下判斷整個程序有多大mov eax,[edi] ;核心程序尺寸xor edx,edx mov ecx,512 ;512字節每扇區div ecxor edx,edxjnz @1 ;未除盡,因此結果比實際扇區數少1 dec eax ;已經讀了一個扇區,扇區總數減1 @1:or eax,eax ;考慮實際長度≤512個字節的情況 jz setup ;EAX=0 ?;讀取剩余的扇區mov ecx,eax ;32位模式下的LOOP使用ECXmov eax,core_start_sectorinc eax ;從下一個邏輯扇區接著讀@2:call read_hard_disk_0inc eaxloop @2 ;循環讀,直到讀完整個內核 setup:mov esi,[0x7c00+pgdt+0x02] ;不可以在代碼段內尋址pgdt,但可以;通過4GB的段來訪問;建立公用例程段描述符mov eax,[edi+0x04] ;公用例程代碼段起始匯編地址mov ebx,[edi+0x08] ;核心數據段匯編地址sub ebx,eaxdec ebx ;公用例程段界限 add eax,edi ;公用例程段基地址mov ecx,0x00409800 ;字節粒度的代碼段描述符call make_gdt_descriptormov [esi+0x28],eaxmov [esi+0x2c],edx;建立核心數據段描述符mov eax,[edi+0x08] ;核心數據段起始匯編地址mov ebx,[edi+0x0c] ;核心代碼段匯編地址 sub ebx,eaxdec ebx ;核心數據段界限add eax,edi ;核心數據段基地址mov ecx,0x00409200 ;字節粒度的數據段描述符 call make_gdt_descriptormov [esi+0x30],eaxmov [esi+0x34],edx ;建立核心代碼段描述符mov eax,[edi+0x0c] ;核心代碼段起始匯編地址mov ebx,[edi+0x00] ;程序總長度sub ebx,eaxdec ebx ;核心代碼段界限add eax,edi ;核心代碼段基地址mov ecx,0x00409800 ;字節粒度的代碼段描述符call make_gdt_descriptormov [esi+0x38],eaxmov [esi+0x3c],edxmov word [0x7c00+pgdt],63 ;描述符表的界限lgdt [0x7c00+pgdt] jmp far [edi+0x10] ;------------------------------------------------------------------------------- read_hard_disk_0: ;從硬盤讀取一個邏輯扇區;EAX=邏輯扇區號;DS:EBX=目標緩沖區地址;返回:EBX=EBX+512 push eax push ecxpush edxpush eaxmov dx,0x1f2mov al,1out dx,al ;讀取的扇區數inc dx ;0x1f3pop eaxout dx,al ;LBA地址7~0inc dx ;0x1f4mov cl,8shr eax,clout dx,al ;LBA地址15~8inc dx ;0x1f5shr eax,clout dx,al ;LBA地址23~16inc dx ;0x1f6shr eax,clor al,0xe0 ;第一硬盤 LBA地址27~24out dx,alinc dx ;0x1f7mov al,0x20 ;讀命令out dx,al.waits:in al,dxand al,0x88cmp al,0x08jnz .waits ;不忙,且硬盤已準備好數據傳輸 mov ecx,256 ;總共要讀取的字數mov dx,0x1f0.readw:in ax,dxmov [ebx],axadd ebx,2loop .readwpop edxpop ecxpop eaxret;------------------------------------------------------------------------------- make_gdt_descriptor: ;構造描述符;輸入:EAX=線性基地址; EBX=段界限; ECX=屬性(各屬性位都在原始; 位置,其它沒用到的位置0) ;返回:EDX:EAX=完整的描述符mov edx,eaxshl eax,16 or ax,bx ;描述符前32位(EAX)構造完畢and edx,0xffff0000 ;清除基地址中無關的位rol edx,8bswap edx ;裝配基址的31~24和23~16 (80486+)xor bx,bxor edx,ebx ;裝配段界限的高4位or edx,ecx ;裝配屬性 ret;-------------------------------------------------------------------------------pgdt dw 0dd 0x00007e00 ;GDT的物理地址 ;------------------------------------------------------------------------------- times 510-($-$$) db 0db 0x55,0xaa
體會:這段代碼很有意思,閱讀思考的過程中就感覺自己站在CPU旁邊觀察CPU運行一樣,讓人著迷,可以看出本書作者的編程功底相當深厚的,這些匯編寫的都很優雅。










創作挑戰賽新人創作獎勵來咯,堅持創作打卡瓜分現金大獎

總結

以上是生活随笔為你收集整理的第13章 程序的动态加载和执行(一,引导)的全部內容,希望文章能夠幫你解決所遇到的問題。

如果覺得生活随笔網站內容還不錯,歡迎將生活随笔推薦給好友。