1

嗨,我正在尝试将我编写的这个 C 函数转换为汇编代码(SPARC arc),但不断收到分段错误(核心转储)错误。该函数应该将字符串转换为长值。我知道我正确地完成了 C 函数,因为我已经对其进行了测试,并且它通过了所有测试。这是我要转换的 C 函数。

long strToLong(char *str, int base )
{
  char *endptr;
  errno = 0;
  long num;
  num = strtol(str, &endptr, base);
  char holder[BUFSIZ];

  /*if *endptr isn't '\0' then strtol didn't get to the end of the string
  *so it wasn't a valid integer*/
  if(*endptr != '\0')
  {
    (void)fprintf(stderr, "\n\t""'%s'" "  is not an integer\n", str);
     return 0;
  }
  /*if errno isn't 0 then an error occured so print out an error message*/
  else if(errno != 0)
  { 
    (void)snprintf(holder, BUFSIZ,"\n\tConverting \"%s\" base \"%d\"\n caused an         ", str, base);
    perror(holder);
    return 0;
  }

  /*if no errors, then it was a valid integer so return the integer*/
  return num;
}

这是我写的汇编代码。

.global strToULong

.section ".data"    

fmt1:    .asciz    "%s  is not an integer\n"    !ascii format

fmt2:
    .asciz    "Converting %s base %d\n"    !ascii format

.section ".text"

strToULong:
    save    %sp, -( 92 + 1028) & -8, %sp
    add    %fp, -4, %o1
    mov    %i0, %o0
    mov    %i2, %o2
    call    strtol
    nop
    mov    %o0, %l0
    ld    [%fp - 4], %l1
    ldub    [%l1], %l1
    cmp    %l1, %g0
    be    errno
    nop

    set    stdError, %o0
    ld    [%o0], %o0
    set    fmt1, %o1
    mov    %i0, %o2
    call    fprintf, 3
    nop

errno:
    mov    %l0, %i0
    set    errno, %l2
    ld    [%l2], %l2
    cmp    %l2, %g0
    be    return
    nop

    add    %fp, -1028, %o0
    mov    1024, %o1
    set    fmt2, %o2
    mov    %i0, %o3
    mov    %i1, %o4
    call    snprintf
    nop
    add    %fp, -1028, %o0
    call    perror
    nop
    mov    0, %i0

return:
    ret
    restore

我调试了程序,当我调用 strtol 函数时出现了段错误。我不确定我做错了什么,我认为我正确地传递了参数,但我仍然得到那个错误。哦,在我的主要声明中,我声明了类似 FILE* StdError = stderr 的内容,以将 stderr 作为参数传递给 fprintf。

任何帮助都将不胜感激。

4

1 回答 1

1

你有几个不同的问题,但你可以很容易解决的一个问题是你访问 errno 的方式。你不应该有一个名为 errno 的分支,因为你还有一个全局 var errno,这会引起一些混乱。

由于 errno 是一个全局变量,将 errno 加载到寄存器中应该如下所示:

    set errno, %l0    !sets %o0 to a pointer to errno
    st %g0,[%l0]      !set errno to 0

    set %i0, %o0      !move str to output
    set endptr, %01   !move endptr to output (defined above in .section ".data")
    set %i1, %o2      !move base to output

    call strtol,2     !calls strtol with given inputs
    nop

    ld  [%l0], %l1    !loads the value at errno into %l1

在这个例子中,%l0 保存了一个指向 errno 的指针,%l1 保存了调用 strtol 后 errno 的值。然后您可以使用以下方法进行错误检查:

    cmp %l1,%g0       !compares errno to 0
    be endif
    nop

    /* if body */

endif:

    /* continued code */

这应该可以帮助您解决一些问题。当你在写汇编时,用 ! 来评论它真的很有帮助。在每一行之后,以及顶部的一个块,您可以在其中跟踪本地变量。组装看起来并不那么简单,所以它真的很有帮助。

于 2012-05-08T22:42:47.673 回答