翻转线订单组件32位

时间:2018-05-06 18:31:28

标签: assembly x86 masm masm32

我需要翻转文件的行顺序并将它们写入另一个文件,但是我有一些问题。由于某种原因,我无法在file2中写入...任何建议和提示都会有用,它&# 39;这是我这个类型的第一个问题。我老师的一个暗示是使用fseek而我使用了它但是卡住了。

示例:

输入file1:

line1
line 2
line 3

所需的输出文件2:

line 3
line2
line 1

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

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

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

    ;sectiunile programului, date, respectiv cod
    .data
    ;aici declaram date
    s db 99 dup(0)
    read db "r",0
    write db "w",0
    nume db "fisier1.txt",0
    nume2 db "fisier2.txt",0
    seek_end dd 2
    format db "%s",0
    .code

    start:
        ;open first file to read
        push offset read
        push offset nume
        call fopen
        add esp,8
        mov esi,eax;save pointer of file

        ;open second file to write
        push offset write
        push offset nume2
        call fopen
        add esp,8
        mov edi,eax;save pointer of file

        ;find the end of file
        push seek_end
        push -1
        push esi
        call fseek
        add esp,12

        ;save in ecx current position
        push esi
        call ftell
        add esp,4
        mov ecx,eax

        et:
            push esi
            call getc
            add esp,4
            cmp eax,0ah;verify if isn't new line
            jne previous
            previous:
                ;move to the previous line
                push 1
                push -1
                push esi
                call fseek
                add esp,12
                jmp cont
            read_write:
                ;read the line in string s 
                push offset s
                push offset format
                push esi
                call fscanf
                add esp,12

                ;print string s in second file
                push offset s
                push offset format
                push edi
                call fprintf
                add esp,12

                jmp previous

        cont:
            dec ecx
            ;verify if isn't the beginning of file  
            cmp ecx,0
            jne et

        push 0
        call exit
    end start

1 个答案:

答案 0 :(得分:1)

这绝对是我不会用汇编语言编写的东西......但是,为了完成这项工作,我首先要用高级语言编写算法。如果您可以使逻辑在更高级别的语言中正常工作,那么您可以使组装工作。

in = fopen("source.txt", "r");
fseek(in, 0, SEEK_END);
size = ftell(in);
fseek(in, 0, SEEK_SET);
out = fopen("destination.txt", "w");
ftruncate(out, size);
fseek(out, 0, SEEK_END);
while(fgets(buf, sizeof(buf), in))
{
    len = strlen(buf);
    fseek(out, -len, SEEK_CUR);
    fwrite(out, 1, len, buf);
    fseek(out, -len, SEEK_CUR);
}

此函数对由sizeof(buf)确定的行的大小有限制,并且该行中存在错误:

  

如果fgets()可以读取sizeof(buf)个字节,则buf[sizeof(buf) - 1] = '\0'; ... fgets(buf, sizeof(buf) - 1, in) 不会返回以空字符结尾的字符串。

不幸的是,这就是C库的糟糕程度。这个bug的一个简单修复:

'\0'

即。您将fopen("destination.txt"...)放在缓冲区的末尾,并且永远不会覆盖它。所以至少你永远不会超出你的缓冲区。

现在您可以在程序集中转换该C代码。

在进行实际的实际编码之前,首先要了解你的算法。

注意:我也没有测试任何错误代码。错误处理是个好主意。 (即如果{{1}}失败了怎么办?)