上一节我们写了字符串打印的函数和软盘读取函数,在软盘读取函数中,我们是直接给定了要读的逻辑扇区号,这一节我们来实现一个查找文件的功能,也就是根据给定的文件名,在根目录区中查找指定的文件是否存在,涉及到文件名的查找,就会涉及到内存的比较,因此,我们需要实现两个函数,分别为内存比较函数和根目录区查找函数。

  整体的流程如下:

 

  首先将根目录区加载到内存中的指定位置上,这里面包含了一定数目的根目录项,而根目录项的前11个字节为文件名,因此,查找文件的工作就可以进行了,我们只需要将目标文件名和每一个根目录项的前11个字节进行比较即可,如果文件存在,必定会在某一个目录项中找到,只需将目录项的偏移位置通过寄存器返回即可。

  内存比较的示意图如下:

 

  需要用到的两个指令如下:

  比较指令:cmp cx, 0    ;  比较cx的值是否为零

  跳转:jz equal  ; 如果比较的结果为真,则跳转到equal处

  用于比较的汇编程序片段如下:

 

  

  1 org 0x7c00
  2 
  3 jmp short start
  4 nop
  5 
  6 define:
  7     BaseOfStack      equ 0x7c00
  8     RootEntryOffset  equ 19
  9     RootEntryLength  equ 14
 10 
 11 header:
 12     BS_OEMName     db "D.T.Soft"
 13     BPB_BytsPerSec dw 512
 14     BPB_SecPerClus db 1
 15     BPB_RsvdSecCnt dw 1
 16     BPB_NumFATs    db 2
 17     BPB_RootEntCnt dw 224
 18     BPB_TotSec16   dw 2880
 19     BPB_Media      db 0xF0
 20     BPB_FATSz16    dw 9
 21     BPB_SecPerTrk  dw 18
 22     BPB_NumHeads   dw 2
 23     BPB_HiddSec    dd 0
 24     BPB_TotSec32   dd 0
 25     BS_DrvNum      db 0
 26     BS_Reserved1   db 0
 27     BS_BootSig     db 0x29
 28     BS_VolID       dd 0
 29     BS_VolLab      db "D.T.OS-0.01"
 30     BS_FileSysType db "FAT12   "
 31 
 32 start:
 33     mov ax, cs
 34     mov ss, ax
 35     mov ds, ax
 36     mov es, ax
 37     mov sp, BaseOfStack
 38     
 39     mov si, MsgStr
 40     mov di, DEST
 41     mov cx, MsgLen
 42     
 43     call MemCmp
 44     
 45     cmp cx, 0
 46     jz label
 47     jmp last
 48     
 49     
 50 label:
 51     mov bp, MsgStr
 52     mov cx, MsgLen
 53     
 54     call Print
 55     
 56 output:    
 57     mov bp, MsgStr
 58     mov cx, MsgLen
 59     call Print
 60     
 61 last:
 62     hlt
 63     jmp last    
 64 
 65 
 66 ; ds:si --> source
 67 ; es:di --> destination
 68 ; cx    --> length
 69 ;
 70 ; return:
 71 ;        (cx == 0) ? equal : noequal
 72 MemCmp:
 73     push si
 74     push di
 75     push ax
 76     
 77 compare:
 78     cmp cx, 0
 79     jz equal
 80     mov al, [si]
 81     cmp al, byte [di]
 82     jz goon
 83     jmp noequal
 84 goon:
 85     inc si
 86     inc di
 87     dec cx
 88     jmp compare
 89     
 90 equal:
 91 noequal:   
 92     pop ax
 93     pop di
 94     pop si
 95     
 96     ret
 97 
 98 ; es:bp --> string address
 99 ; cx    --> string length
100 Print:
101     mov ax, 0x1301
102     mov bx, 0x0007
103     int 0x10
104     ret
105 
106 ; no parameter
107 ResetFloppy:
108     push ax
109     push dx
110     
111     mov ah, 0x00
112     mov dl, [BS_DrvNum]
113     int 0x13
114     
115     pop dx
116     pop ax
117     
118     ret
119 
120 ; ax    --> logic sector number
121 ; cx    --> number of sector
122 ; es:bx --> target address
123 ReadSector:
124     push bx
125     push cx
126     push dx
127     push ax
128     
129     call ResetFloppy
130     
131     push bx
132     push cx
133     
134     mov bl, [BPB_SecPerTrk]
135     div bl
136     mov cl, ah
137     add cl, 1
138     mov ch, al
139     shr ch, 1
140     mov dh, al
141     and dh, 1
142     mov dl, [BS_DrvNum]
143     
144     pop ax
145     pop bx
146     
147     mov ah, 0x02
148 
149 read:    
150     int 0x13
151     jc read
152     
153     pop ax
154     pop dx
155     pop cx
156     pop bx
157     
158     ret
159 
160 MsgStr db  "Hello World"    
161 MsgLen equ ($-MsgStr)
162 DEST db  "Hello World" 
163 
164 Buf:
165     times 510-($-$$) db 0x00
166     db 0x55, 0xaa

 

  运行结果如下:

 

  上图的程序中,我们直接比较的字符串“Hello World”,如果比较成功,就将字符串打印出来。

  下面我们进行文件的查找,具体流程如下图:

  首先加载根目录区,将根目录区加载到指定的缓冲区中,程序片段如下:

 

   访问栈顶数据时,不能通过sp直接访问,而要经过其他寄存器间接访问,如下:

 

  有了以上的思路后,我们直接给出查找文件的汇编程序:

  1 org 0x7c00
  2 
  3 jmp short start
  4 nop
  5 
  6 define:
  7     BaseOfStack      equ 0x7c00
  8     RootEntryOffset  equ 19
  9     RootEntryLength  equ 14
 10 
 11 header:
 12     BS_OEMName     db "D.T.Soft"
 13     BPB_BytsPerSec dw 512
 14     BPB_SecPerClus db 1
 15     BPB_RsvdSecCnt dw 1
 16     BPB_NumFATs    db 2
 17     BPB_RootEntCnt dw 224
 18     BPB_TotSec16   dw 2880
 19     BPB_Media      db 0xF0
 20     BPB_FATSz16    dw 9
 21     BPB_SecPerTrk  dw 18
 22     BPB_NumHeads   dw 2
 23     BPB_HiddSec    dd 0
 24     BPB_TotSec32   dd 0
 25     BS_DrvNum      db 0
 26     BS_Reserved1   db 0
 27     BS_BootSig     db 0x29
 28     BS_VolID       dd 0
 29     BS_VolLab      db "D.T.OS-0.01"
 30     BS_FileSysType db "FAT12   "
 31 
 32 start:
 33     mov ax, cs
 34     mov ss, ax
 35     mov ds, ax
 36     mov es, ax
 37     mov sp, BaseOfStack
 38     
 39     mov ax, RootEntryOffset
 40     mov cx, RootEntryLength
 41     mov bx, Buf
 42     
 43     call ReadSector
 44     
 45     mov si, Target
 46     mov cx, TarLen
 47     mov dx, 0
 48     
 49     call FindEntry
 50     
 51     cmp dx, 0
 52     jz output
 53     jmp last
 54     
 55 output:    
 56     mov bp, MsgStr
 57     mov cx, MsgLen
 58     call Print
 59     
 60 last:
 61     hlt
 62     jmp last    
 63 
 64 ; es:bx --> root entry offset address
 65 ; ds:si --> target string
 66 ; cx    --> target length
 67 ;
 68 ; return:
 69 ;     (dx != 0) ? exist : noexist
 70 ;        exist --> bx is the target entry
 71 FindEntry:
 72     push di
 73     push bp
 74     push cx
 75     
 76     mov dx, [BPB_RootEntCnt]
 77     mov bp, sp
 78     
 79 find:
 80     cmp dx, 0
 81     jz noexist
 82     mov di, bx
 83     mov cx, [bp]
 84     call MemCmp
 85     cmp cx, 0
 86     jz exist
 87     add bx, 32
 88     dec dx
 89     jmp find
 90 
 91 exist:
 92 noexist:
 93     pop cx
 94     pop bp
 95     pop di
 96        
 97     ret
 98 
 99 ; ds:si --> source
100 ; es:di --> destination
101 ; cx    --> length
102 ;
103 ; return:
104 ;        (cx == 0) ? equal : noequal
105 MemCmp:
106     push si
107     push di
108     push ax
109     
110 compare:
111     cmp cx, 0
112     jz equal
113     mov al, [si]
114     cmp al, byte [di]
115     jz goon
116     jmp noequal
117 goon:
118     inc si
119     inc di
120     dec cx
121     jmp compare
122     
123 equal:
124 noequal:   
125     pop ax
126     pop di
127     pop si
128     
129     ret
130 
131 ; es:bp --> string address
132 ; cx    --> string length
133 Print:
134     mov ax, 0x1301
135     mov bx, 0x0007
136     int 0x10
137     ret
138 
139 ; no parameter
140 ResetFloppy:
141     push ax
142     push dx
143     
144     mov ah, 0x00
145     mov dl, [BS_DrvNum]
146     int 0x13
147     
148     pop dx
149     pop ax
150     
151     ret
152 
153 ; ax    --> logic sector number
154 ; cx    --> number of sector
155 ; es:bx --> target address
156 ReadSector:
157     push bx
158     push cx
159     push dx
160     push ax
161     
162     call ResetFloppy
163     
164     push bx
165     push cx
166     
167     mov bl, [BPB_SecPerTrk]
168     div bl
169     mov cl, ah
170     add cl, 1
171     mov ch, al
172     shr ch, 1
173     mov dh, al
174     and dh, 1
175     mov dl, [BS_DrvNum]
176     
177     pop ax
178     pop bx
179     
180     mov ah, 0x02
181 
182 read:    
183     int 0x13
184     jc read
185     
186     pop ax
187     pop dx
188     pop cx
189     pop bx
190     
191     ret
192 
193 MsgStr db  "No START ..."    
194 MsgLen equ ($-MsgStr)
195 Target db  "START      "
196 TarLen equ ($-Target)
197 Buf:
198     times 510-($-$$) db 0x00
199     db 0x55, 0xaa

  如果找不到文件会打印出“No START”,如果能找到,则不打印任何字符。不打印字符的话却无法证明我们的程序是正确的,我们下面进行断点调试,看一下寄存器内容,首先使用ndisasm -o 0x7c00 boot.bin > boot.txt进行反编译,找到适合的断点。

  我们应该在第51行cmp dx, 0处对应的地方打断点,根据boot.txt文件中的内容,我们来确定断点地址,如下:

 

  可以看到,断点地址应该为0x7c61,接下来,启动bochs,进行调试,结果如下:

 

  根据上图可以看出,dx寄存器的值为219,dx为输出寄存器,也就是文件所在的目录项的位置,如果dx为0,说明文件不存在,dx不为0,则说明文件找到了,到此为止,我们查找文件的实验成功了。

 

参考狄泰软件学院操作系统教程

 

posted on 2018-06-27 19:42  周伯通789  阅读(314)  评论(0编辑  收藏  举报