在 C 程序中使用汇编函数

Mad*_*386 2 c assembly dos

我制作的装配函数有一个小问题。

;  Im failing super hard at writing this
;  Function.

.MODEL c, small
    .DATA?
    .DATA
curpos_ PROTO C columns_:BYTE, rows_:BYTE
    .CODE
public curpos_
curpos_ PROC  C columns_:BYTE, rows_:BYTE
    mov dh, columns_
    mov dl, rows_
    mov     bh, 0
    mov     ah, 2
    int 10h
    ret
curpos_ ENDP
END
Run Code Online (Sandbox Code Playgroud)

还有我的 C 文件,其中我对汇编函数进行了原型设计。

#include<stdio.h>
#include<conio.h>
#include<math.h>
void clrscr(void);
extern char _columns, _rows;
extern void _curpos(char _columns, char _rows);
void arcradius() {
    float w;
    float h;
    float radi = w / 2.0;
    float value;
    clrscr();

  ...

_curpos(20,40);
getch();
arcradius();

}
Run Code Online (Sandbox Code Playgroud)

我遇到的问题是我的 C 程序没有为汇编函数提供正确的参数,在我的 C 文件中我使用 _curpos(20,40) 但它不使用括号中的值。相反,它使用前一个 scanf() 中的一些垃圾数字;输入。

我是否有错误声明、原型设计错​​误或忘记的事情?

我正在使用 OpenWatcom 和 MASM 6.11。

谢谢,诺亚“疯狗”布泽利

编辑:这是固定的组装功能

;  Im failing super hard at writing this
;  Function.
.MODEL small
.DATA?
.DATA
curpos PROTO C _columns:BYTE, _rows:BYTE
.CODE
public curpos
curpos PROC  C _columns:BYTE, _rows:BYTE
    push     bx
    mov dh, BYTE PTR _columns
    mov dl, BYTE PTR _rows
    mov     bh, 0
    mov     ah, 2
    int 10h
    pop     bx
    ret 4
curpos ENDP
END
Run Code Online (Sandbox Code Playgroud)

这是我的 C 原型

extern void __stdcall curpos(char columns, char rows);
Run Code Online (Sandbox Code Playgroud)

谢谢你,姆盖兹:)

Mge*_*etz 8

因此需要指定组装方法的调用约定。默认情况下打开 watcom 使用__stdcall

所以它应该看起来像这样:

extern void __stdcall curpos(char _columns, char _rows);
Run Code Online (Sandbox Code Playgroud)

这个汇编可能是正确的,但未经测试。我们手动完成所有事情,而不是依赖 MASM 的魔法来设置 BP 并计算 args 在堆栈上的位置。

_columns$ = 4                                       ; size = 1
_rows$ = 6         ; offsets relative to the frame pointer
                   ; saved-BP at [bp+0], ret addr at [bp+2], first arg at [bp+4]
_curpos@4 PROC                                      ; COMDAT
        push    bp
        mov     bp, sp        ; for access to args on the stack, [sp] isn't valid
        push    bx            ; save/restore the caller's BX
        mov     dh, BYTE PTR _columns$[bp]
        mov     dl, BYTE PTR _rows$[bp]
        mov     bh, 0         ; page=0
        mov     ah, 2
        int     10h           ; int 10h / AH=2 - BIOS Set cursor position   
        pop     bx
        pop     bp          ; no  mov sp,bp  needed, SP is already good
        ret     4           ; pop ret addr, then SP+=4
_curpos@4 ENDP
Run Code Online (Sandbox Code Playgroud)

就其价值而言,最好使用内联汇编,而不是在 asm 中进行完整实现。如果您使用内联汇编,编译器会处理所有这些,您只需将输入放入寄存器,而不是编写返回的代码。