The MASM Forum

General => The Campus => Topic started by: cyrus on September 15, 2024, 12:28:17 PM

Title: LINK : error LNK2001: unresolved external symbol _mainCRTStartup
Post by: cyrus on September 15, 2024, 12:28:17 PM
I know this seems silly. I've been programming in assembly for a bit. started with 64-bit actually and now doing 32-bit stuff. For context, I'm not using masm32.exe. I'm actually using ml.exe from Visual Studio 2022. I have written programs with GNU as 32 as well. For that, I get functions through PEB from kernel32 and LoadLibrary and don't have any issues running it on any system since I am using an archane as.exe and ld.exe from mingw  :smiley: .

For my particular program written in ml format, I am using EXTERNDEF for my functions which works fine. I have tried using PEB but I am not able to retrieve LoadLibrary for some reason. I debugged it and it just fails so I know this ml is going through a different route or something is blocking me from retrieving LoadLibrary so I can get ucrtbase.dll and msvcrt.dll to get my functions. My goal is to use my program on systems that may not have Visual Studio installed  :rolleyes: . Therefore, I noticed out of 4 system, 2 have failed with the typical VCRUNTIME140.DLL not found. I know this has to do with CRT. I am not sure how to statically link my program. I'm using the following directive:

.386   
.model flat, c
option casemap :none
Title: Re: LINK : error LNK2001: unresolved external symbol _mainCRTStartup
Post by: cyrus on September 15, 2024, 12:33:22 PM
I do in fact. In my code:

entry:
main proc

[everything between]

end entry
endp

I have also omitted entry and end entry and simply referenced it when compiling by appending /entry:main
Title: Re: LINK : error LNK2001: unresolved external symbol _mainCRTStartup
Post by: zedd151 on September 15, 2024, 12:35:22 PM
Okay, it is something different than what I was thinking... sorry.


But but, wait


Should be
main proc
... ... ...
main endp

end main

And for link.exe...

/entry:main
Title: Re: LINK : error LNK2001: unresolved external symbol _mainCRTStartup
Post by: cyrus on September 15, 2024, 12:39:07 PM
I appreciate the response. I'm just confused about why I am not able to build a static binary. With ml64.exe I don't have this issue but I can't code my program in 64-bits because the functions I need are written in 32-bits.
Title: Re: LINK : error LNK2001: unresolved external symbol _mainCRTStartup
Post by: cyrus on September 15, 2024, 12:39:54 PM
Quote from: zedd151 on September 15, 2024, 12:35:22 PMOkay, it is something different than what I was thinking... sorry.


But but, wait


Should be

main proc
... ... ...
main endp

end

Yeah but I mentioned in my post above yours, I did try omitting that but it did not work  :smiley:
Title: Re: LINK : error LNK2001: unresolved external symbol _mainCRTStartup
Post by: zedd151 on September 15, 2024, 12:41:44 PM
Can you post or attach your source code?
Or you could make a simple test program that only calls MessageBox and ExitProcess, to see if that will assemble and link.

I assume from what you posted above, that you are using Visual Studio?
And just for clarification, are you working with 32 or 64 bit for this project?
Another thing, you don't need to put 'entry' anywhere in the source code.
Title: Re: LINK : error LNK2001: unresolved external symbol _mainCRTStartup
Post by: cyrus on September 15, 2024, 01:23:15 PM
; ml.exe /c /coff print-exit.asm && link.exe /SUBSYSTEM:console print-exit.obj

.386   
.model flat, c
option casemap :none


.data
; including these will cause an error when running on a non-visual studio installed system
includelib ucrt.lib
includelib msvcrt.lib   ; strstr is in this? lol
includelib legacy_stdio_definitions.lib
includelib kernel32.lib


EXTERNDEF printf: PROC
EXTERNDEF fopen: PROC
EXTERNDEF fread: PROC
EXTERNDEF fwrite: PROC
EXTERNDEF fclose: PROC
EXTERNDEF exit: PROC
EXTERNDEF ExitProcess: PROC

hello db "hello",10,0

.code


main proc

push offset hello
call printf
add esp, 4

call exit

    call ExitProcess

main endp

end

I can't even get ExitProcess from this. It somehow cannot get any functions from kernel32.lib even though I am referencing it. I tried referencing the entire path as well, didn't help. I am using 32-bits for this. The 64-bit ml.exe does not give me any issues but the functions I need are written by someone else and available in 32-bit that works because their 64-bit version doesn't compile.
Title: Re: LINK : error LNK2001: unresolved external symbol _mainCRTStartup
Post by: zedd151 on September 15, 2024, 01:28:14 PM
Before you declare the .libs, you must include the include .inc files.
You don't need to use EXTERNDEF's this way. Do you even have the Masm32 SDK installed?

Your methods are somewhat foreign to me, maybe someone else might have other suggestions for you. Basically I give up, but I tried.

All that being said, I am certain there is a simple solution that will work for you.  :smiley:

Title: Re: LINK : error LNK2001: unresolved external symbol _mainCRTStartup
Post by: cyrus on September 15, 2024, 01:34:41 PM
I don't have anything masm32 installed and I know that's kind of funny because I'm on a masm32 forum  :rolleyes: but I figured I can use this forum to find answers (and help others if possible although I'm not sure I'm ready for that yet lol) and I really do like the people on here. I'm not sure I am using ml.exe correctly and I had considered installing masm but got kind of lost because there are many versions. Also, the syntax is different in the way things are set up and I already learned so many different assembly languages and syntaxes! But I guess I might have to learn it the masm way. Would this avoid my issue with any of these VCRUNTIME140.DLL errors? I know that it is seeking something from Visual Studio but I don't get why those developers assume that every .exe is going to be running on a developer's machine! I'm able to link statically for 64-bits but not 32-bits with ml.exe. I just don't get why I can't statically compile an .exe with ml.exe
Title: Re: LINK : error LNK2001: unresolved external symbol _mainCRTStartup
Post by: NoCforMe on September 15, 2024, 02:02:43 PM
You absolutely do need the right include files (and the includelib files) so that 1) MASM (ml.exe) knows how many arguments each function takes and 2) the linker knows how to access those library functions.

I would recommend going ahead and installing MASM32; it's not painful, doesn't take up an enormous amount of space, and if you don't need to use it you can just ignore it (doesn't mess with the registry or anything intrusive like that). After you install it the include files will be in the include folder (makes sense, huh?). Try including the file masm32rt.inc, which will definitely let you access LoadLibrary().
Title: Re: LINK : error LNK2001: unresolved external symbol _mainCRTStartup
Post by: sinsi on September 15, 2024, 02:41:48 PM
Easy fix, replace ExitProcess with ExitProcess@4
You have to get the import right :biggrin:

You could also do
EXTERNDEF ExitProcess@4: PROC
ExitProcess TEXTEQU <ExitProcess@4>
...
call ExitProcess

Easiest fix, install the MASM32 SDK - it already has all the EXTERNDEFs and a lot more.
Title: Re: LINK : error LNK2001: unresolved external symbol _mainCRTStartup
Post by: cyrus on September 15, 2024, 05:02:32 PM
Quote from: sinsi on September 15, 2024, 02:41:48 PMEasy fix, replace ExitProcess with ExitProcess@4
You have to get the import right :biggrin:

You could also do
EXTERNDEF ExitProcess@4: PROC
ExitProcess TEXTEQU <ExitProcess@4>
...
call ExitProcess

Easiest fix, install the MASM32 SDK - it already has all the EXTERNDEFs and a lot more.


ARGH! jeez why on earth do they make it that painful? @4 ? that's the thing, figuring out each little niche thing about each different assembly. ml is so different than ml64. even including other files. you have to be exact with stating .386 and .model I had to look these up. ml64.exe is much easier but I want to code in 32-bits for this project and I have to give up on ml.exe from VS.

I ended up installing masm32 and playing around with invoke which is new to me. I know I can use the other format. I'm enjoying this standalone platform for development. I tested a simple program on another system and didn't see any VCRUNTIME140.DLL errors  :biggrin:

Even though I moved on to the real masm now, I'd still like to know why I have this _mainCRTStartup problem and how can I actually make my build static, or is it just not possible with this VC/VS version and I should just move on... ?
Title: Re: LINK : error LNK2001: unresolved external symbol _mainCRTStartup
Post by: NoCforMe on September 15, 2024, 05:18:05 PM
Quote from: cyrus on September 15, 2024, 05:02:32 PMARGH! jeez why on earth do they make it that painful? @4 ? that's the thing, figuring out each little niche thing about each different assembly. ml is so different than ml64. even including other files. you have to be exact with stating .386 and .model I had to look these up.

If you're doing 32-bit programming you don't have to worry about that stuff; it's already in the include files.

If you use masm32rt.inc as I suggested, it has all this stuff right near the top:

      .486                                      ; create 32 bit code
      .model flat, stdcall                      ; 32 bit memory model
      option casemap :none                      ; case sensitive

One small gotcha: since there's no .nolist directive in this file, you'll get a shit-ton of unwanted stuff (like everything that's in all the include files, since this "master" include file includes other files) if you generate a listing file. That's why I do this in all my programs:

.nolist
    include \masm32\include\masm32rt.inc
.list

Title: Re: LINK : error LNK2001: unresolved external symbol _mainCRTStartup
Post by: Vortex on September 15, 2024, 09:09:26 PM
Hi cyrus,

QuoteARGH! jeez why on earth do they make it that painful? @4 ?

_ExitProcess@4 is a an example of name mangling :

QuoteName mangling (C++ only)

QuoteName mangling is the encoding of function and variable names into unique names so that linkers can separate common names in the language. Type names may also be mangled. Name mangling is commonly used to facilitate the overloading feature and visibility within different scopes.

https://www.ibm.com/docs/en/i/7.5?topic=linkage-name-mangling-c-only

If you are trying to use UCRT ( Universal C runtime ) with Masm, here is an example :

.386
.model flat,stdcall
option casemap:none

_stdout equ 1

ExitProcess             PROTO :DWORD
printf                  PROTO C :DWORD,:VARARG
__acrt_iob_func         PROTO C :DWORD

; int __cdecl __stdio_common_vfprintf(unsigned __int64 _Options,FILE* _Stream,
;                                     char const* _Format,int _Locale,va_list);

__stdio_common_vfprintf PROTO C :DWORD,:DWORD,:DWORD,:DWORD,:DWORD,:DWORD
_strupr                 PROTO C :DWORD

includelib  \masm32\lib\kernel32.lib
includelib  ucrt.lib

.data

frmt    db '%s %s %s %u.',0
a1      db 'This',0
b1      db 'is',0
c1      db 'printf test',0

.code

start:

    invoke  _strupr,ADDR c1

    invoke  printf,ADDR frmt,ADDR a1,ADDR b1,eax,1

    invoke  ExitProcess,0


printf PROC C _format:DWORD,args:VARARG

;   #define _stdout (__acrt_iob_func(1))

    invoke  __acrt_iob_func,_stdout   

    lea     ecx,args
    invoke  __stdio_common_vfprintf,0,0,\ ; unsigned __int64 _Options
            eax,_format,0,ecx
    ret

printf ENDP

END start

https://masm32.com/board/index.php?topic=11525.0
Title: Re: LINK : error LNK2001: unresolved external symbol _mainCRTStartup
Post by: mineiro on September 15, 2024, 10:56:28 PM
I noted that you have used 10 in your example as being a new line. On windows a new line is 13,10, on linux that's 10. End os string in both are 0.
The C calling convention is equal in windows32 and linux32, so, if you try jwasm/uasm/asmc you're able to create  an executable in linux too. The main problem will be function names that exist in both O.S.
This is a snippet to try.
;ML.EXE /c /coff /Cp /nologo /I"C:\Masm32\Include" "printf.asm"
;LINK.EXE /SUBSYSTEM:CONSOLE /RELEASE /VERSION:4.0 /LIBPATH:"C:\Masm32\Lib" /OUT:"printf.exe" "printf.obj"

.386
option casemap:none
.model flat,stdcall

WIN32 EQU 1 ;define WIN32

IFDEF LIN32
NL EQU 10
ENDIF

IFDEF WIN32

include msvcrt.inc
include kernel32.inc

includelib msvcrt.lib
includelib kernel32.lib

printf textequ <crt_printf>
exit textequ <crt_exit>
NL EQU 13,10
ENDIF

.data
fmt_msg db "have %d good days",NL,0

.code
main proc

invoke printf,addr fmt_msg,10
invoke exit,0

main endp

end main
Title: Re: LINK : error LNK2001: unresolved external symbol _mainCRTStartup
Post by: cyrus on September 16, 2024, 02:19:00 AM
I'm really liking this masm32 a lot. Can't believe I haven't been using it. I've been looking at the examples. A great solid piece of software put together and this is such a great forum as well. Thank you all.

Just ffs I guess, how do I find out what the name mangling is for other functions? I tried looking it up and couldn't find it. I'm looking for FindFirstFileA and FindNextFileA. I tried adding the @4 but still having trouble linking with that.

Update: I now understand the mangling thing. When I disassemble my c++ with g++ -S -fverbose, I am able to see all those functions. These two are @8
Title: Re: LINK : error LNK2001: unresolved external symbol _mainCRTStartup
Post by: cyrus on September 16, 2024, 02:21:53 AM
Quote from: mineiro on September 15, 2024, 10:56:28 PMI noted that you have used 10 in your example as being a new line. On windows a new line is 13,10, on linux that's 10. End os string in both are 0.
The C calling convention is equal in windows32 and linux32, so, if you try jwasm/uasm/asmc you're able to create  an executable in linux too. The main problem will be function names that exist in both O.S.
This is a snippet to try.
;ML.EXE /c /coff /Cp /nologo /I"C:\Masm32\Include" "printf.asm"
;LINK.EXE /SUBSYSTEM:CONSOLE /RELEASE /VERSION:4.0 /LIBPATH:"C:\Masm32\Lib" /OUT:"printf.exe" "printf.obj"

.386
option casemap:none
.model flat,stdcall

WIN32 EQU 1 ;define WIN32

IFDEF LIN32
NL EQU 10
ENDIF

IFDEF WIN32

include msvcrt.inc
include kernel32.inc

includelib msvcrt.lib
includelib kernel32.lib

printf textequ <crt_printf>
exit textequ <crt_exit>
NL EQU 13,10
ENDIF

.data
fmt_msg db "have %d good days",NL,0

.code
main proc

invoke printf,addr fmt_msg,10
invoke exit,0

main endp

end main

Thanks for that piece of code. Yeah it's definitely better to use the IF directive, saves a lot of headache.
As for the newline, yes you are right, I am using 10,0 but I know about 13,10,0. In fact I do use that most of the time but for this particular piece of code and most for testing, I don't add 13, because I might be comparing output say from a recursive function in windows to a listing from linux. I use linux as my main OS and only use windows as a VM. So I use the cmp command in linux through a share thats mapped in windows.
Title: Re: LINK : error LNK2001: unresolved external symbol _mainCRTStartup
Post by: TimoVJL on September 16, 2024, 03:22:31 AM
Quote from: cyrus on September 16, 2024, 02:19:00 AMI'm really liking this masm32 a lot. Can't believe I haven't been using it. I've been looking at the examples. A great solid piece of software put together and this is such a great forum as well. Thank you all.

Just ffs I guess, how do I find out what the name mangling is for other functions? I tried looking it up and couldn't find it. I'm looking for FindFirstFileA and FindNextFileA. I tried adding the @4 but still having trouble linking with that.
FindFirstFileA PROTO STDCALL :DWORD,:DWORD
IFNDEF __UNICODE__
  FindFirstFile equ <FindFirstFileA>
ENDIF
so two params 2 * 4 = 8FindFirstFileA@8FindNextFileA PROTO STDCALL :DWORD,:DWORD
IFNDEF __UNICODE__
  FindNextFile equ <FindNextFileA>
ENDIF
FindNextFileA@8
Title: Re: LINK : error LNK2001: unresolved external symbol _mainCRTStartup
Post by: zedd151 on September 16, 2024, 05:38:44 AM
Quote from: cyrus on September 16, 2024, 02:19:00 AMI'm really liking this masm32 a lot. Can't believe I haven't been using it. I've been looking at the examples. A great solid piece of software ...
Absolutely. With the problems that you initially described, I pretty much knew that you didn't have it (Masm32 SDK) at that point.
Hutch put together the Masm32 SDK specifically to make learning 32 bit assembly for Windows as easy as can be.  :smiley: No need to worry about name mangling in most instances.
Using batch files (makeit.bat, for example) makes it easy to produce your executable without fumbling with VS settings too (or manually via the command line). VS is generally overkill for creating assembly programs, imho. Unless of course, there is a specific need to use it.
Title: Re: LINK : error LNK2001: unresolved external symbol _mainCRTStartup
Post by: Vortex on September 16, 2024, 06:06:02 AM
Hi cyrus,

You can also use Agner Fog's objconv to disassemble object files to study mangled symbols.
Title: Re: LINK : error LNK2001: unresolved external symbol _mainCRTStartup
Post by: NoCforMe on September 16, 2024, 08:10:04 AM
Quote from: cyrus on September 16, 2024, 02:19:00 AMI tried looking it up and couldn't find it. I'm looking for FindFirstFileA and FindNextFileA. I tried adding the @4 but still having trouble linking with that.

Couple things:
Title: Re: LINK : error LNK2001: unresolved external symbol _mainCRTStartup
Post by: sinsi on September 16, 2024, 10:35:24 AM
Quote from: NoCforMe on September 16, 2024, 08:10:04 AMAll that behind-the-scenes nonsense becomes invisible to you, and you can just have fun programming instead of endlessly scratching your head over those weird constructs ...
Some of us are masochistic enough to like the bare bones approach :biggrin: for smaller programs of course.That's why 64-bit programming is ... exciting.
Title: Re: LINK : error LNK2001: unresolved external symbol _mainCRTStartup
Post by: NoCforMe on September 16, 2024, 11:43:54 AM
Quote from: sinsi on September 16, 2024, 10:35:24 AMSome of us are masochistic enough to like the bare bones approach :biggrin: for smaller programs of course.That's why 64-bit programming is ... exciting.

Well, all in good time. First we need to soften up this n00b, lure him into thinking programming is fun before having him ride over the waterfall in a barrel ..

Besides, you know me, sinsi. I don't go in for any of that macro nonsense either ...

It's like that old saying: reality is just a crutch for those who can't handle drugs.
Title: Re: LINK : error LNK2001: unresolved external symbol _mainCRTStartup
Post by: Vortex on September 17, 2024, 02:28:56 AM
ChatGPT's explanations on name mangling :

QuoteName mangling is a technique used by compilers to handle function overloading and to ensure that functions with the same name but different parameters are uniquely identifiable. It involves encoding additional information into the names of functions to differentiate them.

In the context of Windows programming, particularly when discussing 64-bit Windows and the C++ programming language, there are a few key reasons why name mangling is not a major issue:

1. **Calling Conventions and ABI**: On 64-bit Windows (often using the Microsoft x64 calling convention), the application binary interface (ABI) specifies how functions are called and how parameters are passed. This ABI allows for clear differentiation of functions based on their signatures without relying on name mangling. The calling convention and ABI details ensure that function signatures are unique enough to avoid conflicts.

2. **Name Mangling in C++**: While name mangling is indeed used in C++ to handle function overloading and to ensure unique function names, the 64-bit Windows environment does not change the fundamental need for name mangling. However, the mangling schemes used might differ between compilers and platforms. On 64-bit Windows, Microsoft's compiler uses a specific mangling scheme, but it operates under the same principles of encoding additional function signature information into the name.

3. **Linkage and Compatibility**: The way name mangling is handled in 64-bit Windows is compatible with the existing standards and practices of C++ compilation and linking. The mangling process itself remains relevant for function overloading and linking, but the specifics of how it is implemented may vary. The general principles of name mangling are still present, but the linkage and ABI conventions of the 64-bit Windows environment help to manage the function signatures effectively.

4. **Modern Development Practices**: Modern C++ development practices and tools often provide additional layers of abstraction and linkage management that can mitigate the visibility of name mangling issues. Tools like `extern "C"` for C linkage or the use of namespace scopes can help manage and reduce potential conflicts.

In summary, name mangling is still a relevant concept in 64-bit Windows programming, particularly for C++ code. However, the 64-bit ABI and calling conventions help manage and mitigate the complexity associated with it, making it less of a prominent issue in terms of programming and linking.
Title: Re: LINK : error LNK2001: unresolved external symbol _mainCRTStartup
Post by: zedd151 on September 18, 2024, 12:15:18 PM
@cyrus
include \masm32\include\masm32rt.inc

.data

hello db "hello", 13, 10, 0

.code

main proc
    invoke StdOut, addr hello
    inkey
    invoke ExitProcess, 0
main endp

end main

batch file to build it
@echo off

set appname=test3
if exist %appname%.obj del %appname%.obj
if exist %appname%.exe del %appname%.exe

\masm32\bin\ml.exe /c /coff /nologo %appname%.asm
\masm32\bin\link.exe /SUBSYSTEM:CONSOLE /ENTRY:main /nologo %appname%.obj

dir %appname%.*

pause
Title: Re: LINK : error LNK2001: unresolved external symbol _mainCRTStartup
Post by: cyrus on October 09, 2024, 04:48:59 AM
hey guys thanks for all the replies to this thread. definitely learned a lot!