为了账号安全,请及时绑定邮箱和手机立即绑定

从Visual Studio中的asm调用C标准库函数

从Visual Studio中的asm调用C标准库函数

慕尼黑5688855 2019-11-20 14:19:59
我在Visual Studio(Win10 x64,Visual Studio 2015)中创建的asm项目中调用C函数时遇到问题。项目包含一个asm文件:.586.model flat, stdcalloption casemap:noneincludelib msvcrt.libExitProcess PROTO return:DWORDextern printf:near.datatext BYTE "Text", 0.codemain PROC    push offset text    call printf    add esp,4    invoke ExitProcess,0main ENDPend main当我构建项目时,链接器输出错误:错误_main @ 0中引用的LNK2019无法解析的外部符号_printf链接器输出参数:/OUT:"C:\Users\apple\Documents\SP_Lab7\Debug\SP_Lab7_Demo.exe“ / MANIFEST:否/ NXCOMPAT /PDB:"C:\Users\apple\Documents\SP_Lab7\Debug\SP_Lab7_Demo.pdb” / DYNAMICBASE “ kernel32.lib”“ user32.lib”“ gdi32.lib”“ winspool.lib”“ comdlg32.lib”“ advapi32.lib”“ shell32.lib”“ ole32.lib”“ oleaut32.lib”“ uuid.lib” “ odbc32.lib”“ odbccp32.lib” / MACHINE:X86 / SAFESEH:NO / INCREMENTAL:NO /PGD:"C:\Users\apple\Documents\SP_Lab7\Debug\SP_Lab7_Demo.pgd“ / SUBSYSTEM:WINDOWS / MANIFESTUAC: “ level ='asInvoker'uiAccess ='false'” /ManifestFile:"Debug\SP_Lab7_Demo.exe.intermediate.manifest“ / ERRORREPORT:PROMPT / NOLOGO / TLBID:1如果我发表评论call print,那么一切都会正常执行(甚至是Windows API函数)。有什么方法可以从asm文件中调用C函数,而无需创建包含以下内容的cpp文件<cstdio>?有可能吗?
查看完整描述

3 回答

?
叮当猫咪

TA贡献1776条经验 获得超12个赞

Microsoft 在VS 2015中重构了大部分C运行时和库。某些功能不再从C库导出(某些功能在C头文件中定义)。Microsoft具有一些兼容性库,例如legacy_stdio_definitions.lib和legacy_stdio_wide_specifiers.lib,但您也可以选择将较早的Visual Studio 2013平台工具集与较早的C库一起使用。


要更改平台工具集:下拉Project菜单;选择Properties...; 转到Configuration Properties/ General,然后更改Platform Toolset为Visual Studio 2013(v120)


查看完整回答
反对 回复 2019-11-20
?
神不在的星期二

TA贡献1963条经验 获得超6个赞

似乎可以对Visual Studio 2015 Toolset进行一些修改。


您需要将以下库添加到依赖项中:libcmt.lib,libvcruntime.lib,libucrt.lib,legacy_stdio_definitions.lib。或者,您可以includelib用来将这些库包含在程序集文件中。

为您的过程指定C调用约定mainPROC C

在文件末尾(这很重要)不要使用end main,而end只能使用。不解决此问题可能会导致意外崩溃。

尽管我们可以使用ExitProcess退出我们的应用程序,但是我们也可以将返回代码放入EAX中并做一个ret返回。该Ç运行时调用我们的main功能,并在返回时将调用关机代码为我们。

代码看起来像:


.586

.model flat, stdcall

option casemap:none


includelib libcmt.lib

includelib libvcruntime.lib

includelib libucrt.lib

includelib legacy_stdio_definitions.lib


ExitProcess PROTO return:DWORD

extern printf:NEAR


.data

text BYTE "Text", 0


.code

main PROC C                    ; Specify "C" calling convention

    push offset text

    call printf

    add  esp, 4

;   invoke ExitProcess,0       ; Since the C library called main (this function)

                               ; we can set eax to 0 and use ret`to have

                               ; the C runtime close down and return our error

                               ; code instead of invoking ExitProcess

    mov eax, 0

    ret

main ENDP

end                            ; Use `end` on a line by itself

                               ; We don't want to use `end main` as that would

                               ; make this function our program entry point

                               ; effectively skipping by the C runtime initialization


查看完整回答
反对 回复 2019-11-20
?
慕田峪9158850

TA贡献1794条经验 获得超7个赞

您可以调用C函数,但随后需要与C库链接。确切的实现方式取决于您要链接的C库。我建议您找到一个最小的C运行时,例如WCRT库。


该库可能需要初始化,并且可能需要您在其某个地方定义一堆缓冲区以进行记账。


建议您不要使用Windows API,而是使用WriteConsole函数,而不要麻烦这些麻烦。


查看完整回答
反对 回复 2019-11-20
  • 3 回答
  • 0 关注
  • 666 浏览
慕课专栏
更多

添加回答

举报

0/150
提交
取消
意见反馈 帮助中心 APP下载
官方微信