seek_cur命令将光标设置在文件中的未知位置

时间:2019-04-10 19:59:02

标签: windows assembly x86 masm

我有一个文件,每行包含一个单词(单词的数量及其长度未知),我需要将这些单词重写到另一个文件中,从最后一个单词开始,直到第一个单词。当我打印文件中的最后一个单词时,我尝试设置光标(seek_cur)来寻找下一个单词,但是它将其设置在一个未知的位置。尝试打印当前光标以查看会发生什么,并显示“ @A”之类的字符。

第二个jmp get_out在写完最后一个单词后停止程序,如果删除了该单词,它将进入jmp搜索标记,然后无限打印相同的最后一个单词。

.386
.model flat, stdcall
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;

;includem biblioteci, si declaram ce functii vrem sa importam
includelib msvcrt.lib
extern exit: proc
extern fopen: proc
extern fclose: proc
extern fscanf: proc
extern fprintf: proc
extern fseek: proc
extern fgets:proc

;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;

;declaram simbolul start ca public - de acolo incepe executia
public start
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;

;sectiunile programului, date, respectiv cod
.data
file_read db "r",0
file_write db "w",0

file_name_read db "read.txt",0
file_name_write db "write.txt",0

seek_end dd 2
seek_cur dd 1
seek_set dd 0

caracter_format db "%c",0
string_format db "%s",0
decimal_format db "%d",0

string db 0
caracter db 0

back dd 0
first_word db 0

.code
start:
    push offset file_read
    push offset file_name_read
    call fopen
    add esp,8
    mov esi,eax

    push offset file_write
    push offset file_name_write
    call fopen
    add esp,8
    mov edi,eax

    ;in first_word se pune cuvantul de pe prima linie ca sa fie posibila comparatia mai incolo si sa se iese din bucla cand se ajunge la primul cuvant
    repeat_search_first_word:
        push offset caracter
        push offset caracter_format
        push esi
        call fscanf
        add esp,12

        inc back

        cmp caracter,0Ah
        je out_of_search_first_word 

    jmp repeat_search_first_word

        out_of_search_first_word:

        inc back
        neg back
        push seek_cur
        push back
        push esi
        call fseek
        add esp,12  

        neg back
        push esi
        push back
        push offset first_word
        call fgets
        add esp,12

        mov back,0


    ;incepe cautarea cuvintelor de la capat
    push seek_end
    push -1
    push esi
    call fseek
    add esp,12  

    search:
        push offset caracter
        push offset caracter_format
        push esi
        call fscanf
        add esp,12

        inc back

        cmp caracter,0Ah
        jne is_caracter

            push esi
            push back
            push offset string
            call fgets
            add esp,12

            push offset string
            push offset string_format
            push edi
            call fprintf
            add esp,12

            ;testam daca cuvantul coincide cu primul (first_word)
            mov ebx,0
            mov bl,string
            cmp bl,first_word
            je get_out


            add back,2
            neg back

            ;!!!!!!!!!!!!!!
            ;problema pentru rularea infinita ii aici fiindca seek_cur muta cursorul intr-o zona necunoscuta din fisier 
            push seek_cur
            push back
            push esi
            call fseek
            add esp,12


            mov back,0
            jmp get_out ;linia 152 lasata fara comentariu permite afisare ultimului cuvant fara sa intre in rularea infinita a buclei
            jmp search

        is_caracter:

        push seek_cur
        push -2
        push esi
        call fseek
        add esp,12  

    jmp search  
    get_out:

    push edi
    call fclose 
    add esp,4

    push esi
    call fclose 
    add esp,4



    push 0
    call exit
end start

read.txt包含:

abc                          
defg                         
hijklm                      

write.txt应该是:

hijklm
defg
abc

1 个答案:

答案 0 :(得分:2)

string db 0保留1个字节的空间(初始化为零)。

然后调用fgets(fp, string, back),如果它读取的数据超过1个字节(包括结尾的0),它将覆盖数据部分中的后续内容。

在BSS中使用更大的缓冲区,例如几个MB或类似的东西。


使用调试器跟踪功能调用/系统调用。在Linux上,您可以使用ltrace来跟踪libc stdio函数,或者使用strace来跟踪它们使用的系统调用。在Windows IDK上。在每次调用之前,您始终可以逐步检查并检查已推送到堆栈中的参数,以确保它们是理智的,但是在查找带有错误参数的列表时,通常更容易看到列表的日志文件样式。