在CPP文件运行ASM程序(Running asm procedure in cpp file)

2019-10-23 12:25发布

我试图从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文件工作。

我究竟做错了什么?

Answer 1:

Irvine32.inc包括SmallWin.inc以及设定.MODEL flat, stdcall 。 所以,每道工序将默认为,如果调用约定是装饰STDCALL 。 _calka变得_calka@0 。 对于CDECL调用约定需要main.cpp ,你必须覆盖与程序的默认调用约定PROC C 。 现在,下划线会自动添加。 TLDR: calka proc C

最好的是,你创建一个新的项目。 我会形容为Microsoft Visual Studio 2015 RC的程序在Windows 10的过程对于其他版本VS颇为相似。

1)启动Visual Studio 2015年RC,然后选择FILE - New - Project

2)在接下来的窗口选择Win 32 Console Application

3)你得到了确认。 点击Next >

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

5)现在选择PROJECT - Build Customizations...

6)在下一个窗口中打勾masm(.targets,.props)然后点击OK

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

8)在接下来的窗口中选择C++File(.cpp)给它一个名称,然后单击Add

9)创建calka.asm: PROJECT - Add New Item 。 在接下来的窗口中选择C++File(.cpp)和-重要! -给它一个名称.asm扩展。 点击Add

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

11)在属性页面上,你会看到
排除从生成(空)或否
项目建设Microsoft宏汇编
点击OK

12)现在,你可以填写这些文件的内容。

你不需要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运行。

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



文章来源: Running asm procedure in cpp file