c++ - 在 cpp 文件中运行 asm 过程

标签 c++ assembly visual-studio-2013 masm irvine32

我正在尝试从 cpp 文件中的 asm 文件运行过程,但出现这样的错误:

Error   1   error LNK2019: unresolved external symbol _calka referenced in function _main   D:\Addem\main.obj   Project
Error   2   error LNK1120: 1 unresolved externals   D:\Addem\Debug\Project.exe  1   1   Project

这是我的 main.cpp:

#include <iostream>
using namespace std; 

extern "C" void calka();

int main()
{
    calka();

    system("Pause");
    return 0;
}

还有我的 calka.asm

; The calka Subroutine    (calka.asm)
public _calka                  

INCLUDELIB kernel32.lib
INCLUDELIB user32.lib
INCLUDELIB Irvine32.lib
INCLUDE Irvine32.inc
INCLUDE macros.inc



.data
BUFFER_SIZE = 50
buffer BYTE BUFFER_SIZE DUP(?)
filename    BYTE 80 DUP(0)
fileHandle  HANDLE ?
Rozmiar dw ?
Znak db ?
Suma dw ?
Minus db 0
wynik dw ?
temp dd ?

.code
_calka proc near

; Let user input a filename.
    mWrite "Podaj nazwe pliku (z rozszerzeniem): "
    mov edx,OFFSET filename
    mov ecx,SIZEOF filename
    call    ReadString

; Open the file for input.
    mov edx,OFFSET filename
    call    OpenInputFile
    mov fileHandle,eax

; Check for errors.
    cmp eax,INVALID_HANDLE_VALUE        ; error opening file?
    jne file_ok                 ; no: skip
    mWrite <"Cannot open file",0dh,0ah>
    jmp quit                        ; and quit
file_ok:

; Read the file into a buffer.
    mov edx,OFFSET buffer
    mov ecx,BUFFER_SIZE
    call    ReadFromFile
    jnc check_buffer_size           ; error reading?
    mWrite "Error reading file. "       ; yes: show error message
    call    WriteWindowsMsg
    jmp close_file

check_buffer_size:
    cmp eax,BUFFER_SIZE         ; buffer large enough?
    jb  buf_size_ok             ; yes
    mWrite <"Error: Buffer too small for the file",0dh,0ah>
    jmp quit                        ; and quit

buf_size_ok:    
    mov buffer[eax],0       ; insert null terminator
    ;mWrite "File size: "
    ;call   WriteDec            ; display file size
    ;call   Crlf

; Display the buffer.


; Pytanie o wielkosc przedzialu

    mWrite "Podaj wielkosc przedzialu h na osi x: "
    mov edx,OFFSET Rozmiar
    mov ecx,SIZEOF Rozmiar
    call    ReadDec

    mov Rozmiar, ax


    ;mWrite <"Buffer:",0dh,0ah,0dh,0ah>
    ;mov    edx,OFFSET buffer   ; display the buffer
    ;call   WriteString
    ;call   Crlf

    mov ecx, 15
    mov ebx, OFFSET buffer

mov si, 0
        jmp odczyt1
odczyt:
        mov ebx, temp
        inc ebx

odczyt1:
        mov cl, byte ptr [ebx]
        mov Znak, cl

        mov temp, ebx

        cmp Znak, '$' ;koniec pliku
        je koniec
        cmp Znak, 3bh ;srednik - nastepna liczba
        je nastepna
        cmp Znak, 20h ;spacja - ponowne wczytanie
        je odczyt
        cmp Znak, '-'
        je Zmien_znak


        mov al, Znak
        sub al, 30h
        mov bl, al
        mov bh, 0
        mov ax, 10
        mul si
        add ax, bx
        mov si, ax
        jmp odczyt

Zmien_znak:
        mov Minus, 1
        jmp odczyt

nastepna:
        cmp Minus, 0
        je dodaj
        jne odejm
dodaj:
        add Suma, si
        mov si, 0
        jmp odczyt
odejm:
        sub Suma, si
        mov Minus, 0
        mov si, 0
        jmp odczyt

koniec:
        cmp Minus, 0
        je dod
        jne minu
dod:
        add Suma, si
        mov si, 0
        jmp kon
minu:
        sub Suma, si
        mov Minus, 0
        mov si, 0
kon:
        mov ax, Suma
        mul Rozmiar

    mWrite "Wynik calkowania: "
    cmp ax,0
    ;mov ebx, eax
    jg plus
    mWrite "-"
    mov bx, 0
    sub bx, ax
    mov ax, bx

plus:
    call WriteDec


close_file:
    mov eax,fileHandle
    call    CloseFile

    ; Zeby aplikacja sie nie zamknela, heheszki :D
    mov edx,OFFSET Rozmiar
    mov ecx,SIZEOF Rozmiar
    call    ReadString

quit:
    exit
_calka ENDP

END

我已经尝试了所有我能专注的事情——将 obj 文件添加到链接器等等。我只想输入文件名,从中读取日期并计算它。 Calka.asm 运行良好,除非我尝试让它在我的 cpp 文件中运行。

我做错了什么?

最佳答案

Irvine32.inc 包括 SmallWin.inc 和设置 .MODEL flat, stdcall。因此,默认情况下,每个过程都将被修饰,就好像调用约定是STDCALL一样。 . _calka 变为 _calka@0。对于 CDECL calling convention main.cpp 需要您必须使用 PROC C 覆盖过程的默认调用约定。现在下划线将自动添加。 TLDR:calka proc C

最好的是,您创建一个新项目。我将描述 Windows 10 上的 Microsoft Visual Studio 2015 RC 的过程。其他 VS 版本的过程非常相似。

1) 启动 Visual Studio 2015 RC 并选择 FILE - New - Project

Screenshot

2) 在下一个窗口中选择 Win 32 Console Application

Screenshot

3) 您会得到确认。单击下一步>

Screenshot

4) 在下一个窗口中选择
[X] 控制台应用
[X] 空项目
点击完成

Screenshot

5) 确保项目在解决方案资源管理器中突出显示并选择 PROJECT - Build Customizations...

Screenshot

6) 在下一个窗口中勾选 masm(.targets,.props) 并单击 OK

Screenshot

7) 创建 main.cpp:PROJECT - 添加新项

Screenshot

8) 在下一个窗口中选择C++File(.cpp),为其命名并点击Add

Screenshot

9) 创建 calka.asm:PROJECT - 添加新项。在下一个窗口中选择 C++File(.cpp) 和 - 重要! - 以 .asm 扩展名为其命名。单击添加

Screenshot

10) 现在检查 .asm 文件是否具有正确的属性。在解决方案资源管理器中,右键单击该文件并选择 Properties

Screenshot

11) 在属性页中你应该看到
从构建中排除(空)或否
项目构建 Microsoft 宏汇编器
单击确定

Screenshot

12) 现在您可以用内容填充文件。

Screenshot

您不需要 system("Pause"),因为 CTRL-F5 会构建、运行和暂停应用程序。

.asm文件中删除

INCLUDELIB kernel32.lib
INCLUDELIB user32.lib

并让 Visual Studio 自动包含它们。添加到

INCLUDELIB Irvine32.lib
INCLUDE Irvine32.inc
INCLUDE macros.inc

文件的完整路径。这比更改 VS 的设置要好。

改变

_calka proc near
...
_calka ENDP

calka PROC C
...
calka ENDP

13) 使用 CTRL-F5 构建并运行。

Screenshot

应用程序将在新窗口中打开。

关于c++ - 在 cpp 文件中运行 asm 过程,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/31037176/

相关文章:

c++ - wcsncpy_s 函数会导致缓冲区溢出吗?

c++ - 从 C++ 调用 putchar 地址

c# - Monogame 内容管道已停止工作

c - 如何将参数传递给 ZwCreateThreadEx 例程?

c - MIPS 汇编 - if 条件与减法

c# - 如何在 C# Socket 上从 android 连续接收位图?

c++ - 为什么不使用 "new"运算符创建的对象得到相同的地址

c++ - 收到错误LNK2019和错误LNK1120

c++ - 将绘图窗口 "deeper"的控制传递给程序?

windows - 我是否需要在程序集中复制命令行参数