Команда seek_cur устанавливает курсор в неизвестном месте в файле - PullRequest
0 голосов
/ 10 апреля 2019

У меня есть файл, который содержит одно слово в каждой строке (количество слов и их длина неизвестны), и мне нужно переписать эти слова в другой файл, начиная с последнего слова и заканчивая первым.Когда я печатаю последнее слово в файле, я пытаюсь установить курсор (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 Ответ

2 голосов
/ 10 апреля 2019

string db 0 резервирует место для 1 байта (инициализируется нулем).

Затем вы вызываете fgets(fp, string, back), который перезапишет более поздние элементы в вашем разделе данных, если он прочитает более 1 байта (включая завершающий 0).

Используйте больший буфер в BSS, например пару МБ или что-то в этом роде.


Использование отладчика для отслеживания вызовов функций / системных вызовов. В Linux вы можете использовать ltrace для отслеживания функций libc stdio или strace для отслеживания используемых ими системных вызовов. На Windows IDK. Вы всегда можете пройтись и посмотреть на аргументы, которые вы помещали в стек перед каждым вызовом, чтобы убедиться, что они нормальны, но часто легче увидеть стиль листинга в лог-файле, когда ищите неправильные аргументы.

...