运行 cpp 文件中的 asm 程序

Running asm procedure in cpp file

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

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。因此,默认情况下每个过程都将被修饰,就好像 main.cpp 需要调用约定 STDCALL. _calka becomes _calka@0. For the CDECL calling convention 您必须使用 PROC C 覆盖过程的默认调用约定。现在下划线将自动添加。 TLDR:calka proc C.

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

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

2) 在接下来的 window 中选择 Win 32 Console Application

3) 您会得到确认。单击 Next >

4) 在接下来的window中选择
[X] 控制台应用程序
[X] 空项目
单击 Finish

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

6) 在接下来的 window 中勾选 masm(.targets,.props) 然后点击 OK.

7) 创建 main.cpp: PROJECT - Add New Item.

8) 在接下来的window中选择C++File(.cpp),给它起个名字然后点击Add

9) 创建 calka.asm:PROJECT - Add New Item。在接下来的 window 中选择 C++File(.cpp) 并且 - 重要! - 给它起一个扩展名为 .asm 的名字。单击 Add

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

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

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

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

.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.

构建并 运行

应用程序将在新 window 中打开。