比较FASM中的两个字符串

时间:2013-10-10 12:49:37

标签: string assembly x86 compare fasm

对于FASM,我是新手,对ASM来说一般都是新手,但我正在尝试比较存储在“变量”中的两个字符串:user_inputexit_cmd

目前,它组装得很好但是当我在提示中输入任何内容时它会崩溃。是的,我的代码很乱,我想要完成的任务似乎超出了我所知的ASM水平,但我已经用其他语言完成了所以我在ASM中尝试了。

如果字符串匹配,我可以看到我正在使用宏CompareStrings(来源未知;它不是我的)将EAX设置为1,但是当我将EAX与1进行比较时使用CMP然后JE使用标签拒绝使用。有什么帮助吗?

这是有缺陷的代码:

format PE console
;win32a.inc, win32w.inc, win64a.inc and win64w.inc
include 'win32w.inc'
entry main

; Define macros for things like printing strings ('print') and pausing (well, waiting for the user to press enter ('pause'))
macro print str {
      push str
      call [printf]
      push 0
}
macro pause {
      invoke system,'pause>nul'
}
macro getinput prompt {
      print prompt
      invoke gets,user_input
}
macro CompareStrings str1, str2
{
      lea edi, [str2]               ; edi -> address of string2
      dec edi                       ; edi = edi - 1
      .lab1:                        ; loop through all chars and compare each of them
      inc edi                       ; ds:di --> next character in string2
      lodsb                         ; al = next char from string1. loadsb increments si automatically
      cmp [edi], al                 ; compare characters
      jne .notfound                 ; jump out of the loop if they're unequal
      cmp al, 0                     ; chars are equal, but make sure we compared the entire string
      jne .lab1                     ; if not, continue the loop
      mov eax, 1                    ; else: strings are equal -> eax = 1
      ret                           ; return; result: strings are equal
      .notfound:                    ; chars are not equal
      mov eax, 0                    ; unequal -> eax = 0
      ret                           ; return; result: strings are not equal
}

section '.text' code readable executable

main:
        print header_msg
        jmp input_loop

input_loop:
        getinput cmd_prompt
        print newline
        CompareStrings user_input,cmd_exit
        cmp eax, 1
        je exit_cmd
        jne input_loop

exit_cmd:
        print header_msg
        ret

section '.data' data readable writable
        header_msg    db      "Basic Programming Language in ASM | Version 0.0.1",13,10,0
        cmd_prompt    db      "> ",0
        newline       db      "",13,10,0
        user_input    db      "",0dh,0ah,0
        chrarr        db      '%s',0dh,0ah,0
        cmd_exit      db      'exit',0

section '.idata' import data readable

library msvcrt,"msvcrt.dll"

import msvcrt,\
       puts,"puts",\
       getchar,"getchar",\
       exit,"exit",\
       printf,"printf",\
       scanf,"scanf",\
       system,"system",\
       gets,"gets"

1 个答案:

答案 0 :(得分:1)

这个程序有很多问题,崩溃并不奇怪。

问题1

您没有为输入的字符串分配足够的空间。使用rb指令分配足够大的空间来处理输入:

user_input  rb  1024

将其放在数据部分的末尾,以便不在结果可执行文件中进行。

问题2

我不知道如何使用CPP库,但可能调用约定不是STDCALL,参数不完全是这些。例如gets可能需要某种CPP自扩展字符串,而不是简单的缓冲区地址。否则很容易溢出缓冲区,无法检查这样的问题。

因此,最好使用普通的WinAPI - ReadFile和WriteFile来读取和写入控制台。

问题3

在上述背景下,它是挑剔,但无论如何:

  push str
  call [printf]
  push 0

push 0看起来很奇怪。该怎么办?

问题4

你的风格。尝试在不使用“{”和“}”字符的情况下实现此程序。它将帮助您更好地理解汇编语言。