reference to `___chkstk_ms'

Question about which tools to use, bugs, the best way to implement a function, etc should go here. Don't forget to see if your question is answered in the wiki first! When in doubt post here.
Post Reply
robbiedobbie
Member
Member
Posts: 36
Joined: Fri May 16, 2014 2:40 pm
Libera.chat IRC: robbiedobbie

reference to `___chkstk_ms'

Post by robbiedobbie »

Recently I started developing for UEFI. So far, everything has been going great. The setup of the development environment is a bit weird and it was hard to find a right one, but once you have it running, it works fine (Except for debugging, which is still shitty since you have to tell gdb each time where the code is executed...).

My setup consists of the default mingw64 package in arch linux, which I use to output an subsystem 10 application (EFI application). Since the uefi environment does not have the standard libraries, I compile it as freestanding.

At the moment, I'm having a linking error though. For easyness I currently simply allocated a big buffer (well, not too big) on the stack (I know, bad practice). I expected that I could have issues like a stack not being big enough (even though the allocation is still not too big). However, when using a big buffer, I get errors during the linking stage. The generated code by gcc, tries to call a function ___chkstk_ms, which (I think ) tries to check the stack. I'm linking the executable with libgcc, but this has not solved the issue.

The parameters I use to compile:

Code: Select all

x86_64-w64-mingw32-gcc -ffreestanding -Wall -Wextra -std=c11 -O0 -fno-stack-protector -fpic -mno-red-zone -fno-builtin -o src/main.o -c src/main.c
x86_64-w64-mingw32-gcc -nostdlib -Wl,-dll -shared -Wl,--subsystem,10 -pie -e kernel_main -lgcc  -o bin/EFI/Boot/bootx64.efi src/main.o
The exact error that I get:

Code: Select all

src/main.o: In function `kernel_main':
/home/rob/Projects/C-C++/RobOS/src/main.c:36: undefined reference to `___chkstk_ms'
collect2: error: ld returned 1 exit status
I figured that it may be because I'm not using a self built crosscompiler. I tried to compile gcc for the x86_64-w64-mingw32 target but this kept failing with errors about not having a valid /lib/cpp environment when compiling libgcc. I may be able to work around all the errors here, but I thought to first ask here if it actually is a problem with the environment, or whether I simply forget to link a certain library or something.
User avatar
hometue
Member
Member
Posts: 100
Joined: Thu Dec 19, 2013 1:40 am
Location: Asia, Singapore

Re: reference to `___chkstk_ms'

Post by hometue »

Just a note: I don't think I have the best answer, so if anyone notices that my solution does have a problem do correct me.
Anyway it seems that there are some problems in your environment. Firstly, I don't think you should be using MinGW, you might want to use GCC instead unless Arch linux doesn't have the GCC package (yes, its probably around the same but MinGW stands for Minimalist GNU for Windows, not what I would suggest you to use...). Even if Arch linux doesn't have go grab the source from online. Next is the target you tried to build, it seems...invalid...you might want to look around for those that the developers of OS that uses UEFI uses, if I am not wrong its something else. The forum might have something...

I do understand this is rather vague, but that is all I know as of now hope it helps you. Hopefully someone with an insight into how you should be compiling your crosscompiler would answer to your problem quickly. This might not help you depending on the exact cause of your issue but it would prevent problems later on.
CookieOS. Want a cookie? Its only black and white for now though, probably as bad as my baking skills.
robbiedobbie
Member
Member
Posts: 36
Joined: Fri May 16, 2014 2:40 pm
Libera.chat IRC: robbiedobbie

Re: reference to `___chkstk_ms'

Post by robbiedobbie »

Let me explain why I'm using mingw. UEFI uses the PE executable format for the application, which is about the same as a windows executable.
There are several ways to create such executable on windows. Using the UDK (UEFI DevKit) which is extremely big and difficult. Another one is using gnu-efi, which is gcc with a crosscompiled bin-utils which provides the ability to work with the pe format. They then use a wrapper function that converts the ABI between ms standards and gcc standards.

The third option is to use mingw to create the binary (since it normally creates windows binaries, meant to run on windows), and use the same headers as gnu-efi, only without the ABI conversions, since mingw already uses the microsoft conventions. Since this is in my opinion the easiest approach, I chose this last option. I definitely want to keep doing this, since it also allows me to compile the same source directly from visual studio from time to time...

Now, it could be that the target I tried to build is incorrect, however, it did produce a working compiler. The only problem is compiling the libgcc for that target. Since I'm not particularly looking for building the crosscompiler, but simply for fixing the error I got, I haven't investigated this further. I will do that if it actually turns out to be my environment, since it will take me some work to figur out. I haven't done this yet since it may as well be some standard lib that I have to link which I forgot or something.
In other words, I'm simply looking for confirmation whether my way of compiling and linking is correct for a UEFI application.
jnc100
Member
Member
Posts: 775
Joined: Mon Apr 09, 2007 12:10 pm
Location: London, UK
Contact:

Re: reference to `___chkstk_ms'

Post by jnc100 »

According to a quick google, ___chkstk_ms should be defined in libgcc. You probably want the '-lgcc' option at the end of your link command, after the object files (rather than before).

If that does not work, try adding '-fno-stack-check -fno-stack-protector -mno-stack-arg-probe' to your CFLAGS.

Regards,
John.
robbiedobbie
Member
Member
Posts: 36
Joined: Fri May 16, 2014 2:40 pm
Libera.chat IRC: robbiedobbie

Re: reference to `___chkstk_ms'

Post by robbiedobbie »

Wow, I feel stupid now. I did find the correlation with libgcc, however I assumed it had to be something different since I had it specified in the parameters. Placing the -lgcc at the end of the command line worked.

Thanks jnc100 for all the help already!
jnc100
Member
Member
Posts: 775
Joined: Mon Apr 09, 2007 12:10 pm
Location: London, UK
Contact:

Re: reference to `___chkstk_ms'

Post by jnc100 »

No problem. Out of interest, do you know exactly what change to your source caused it to emit a dependency on ___chkstk_ms? I only ask because I'm unable to reproduce it here and would like to test out whether the stack protector in libgcc still works correctly in a UEFI environment before adding it to the wiki article.

Regards,
John.
robbiedobbie
Member
Member
Posts: 36
Joined: Fri May 16, 2014 2:40 pm
Libera.chat IRC: robbiedobbie

Re: reference to `___chkstk_ms'

Post by robbiedobbie »

I simply declared a big array on the stack. Inside of a function, simply declare the following: EFI_MEMORY_DESCRIPTOR descriptors[150];

Obviously this depends on the amount of other data on the stack. In my case, I triggered it when using more than 90 as size ;)
User avatar
Combuster
Member
Member
Posts: 9301
Joined: Wed Oct 18, 2006 3:45 am
Libera.chat IRC: [com]buster
Location: On the balcony, where I can actually keep 1½m distance
Contact:

Re: reference to `___chkstk_ms'

Post by Combuster »

I.e. provoking windows-specific code to poke the stack in chunks as to not make it believe you smashed the stack instead. I guess that's a particular problem of not having a native EFI toolchain.

But large stack arrays are often a signature of unhealthy code anyway...
"Certainly avoid yourself. He is a newbie and might not realize it. You'll hate his code deeply a few years down the road." - Sortie
[ My OS ] [ VDisk/SFS ]
robbiedobbie
Member
Member
Posts: 36
Joined: Fri May 16, 2014 2:40 pm
Libera.chat IRC: robbiedobbie

Re: reference to `___chkstk_ms'

Post by robbiedobbie »

I just used it when poking around a little. I know that allocating big arrays on the stack is bad practice ;) I still wanted to figure out how to solve it since it could be a deeper problem (which it was because apparently libgcc wasn't linked in...)
jnc100
Member
Member
Posts: 775
Joined: Mon Apr 09, 2007 12:10 pm
Location: London, UK
Contact:

Re: reference to `___chkstk_ms'

Post by jnc100 »

robbiedobbie wrote:I simply declared a big array on the stack.)
Thanks, confirmed and fixed in the wiki.
Combuster wrote:I.e. provoking windows-specific code to poke the stack in chunks as to not make it believe you smashed the stack instead. I guess that's a particular problem of not having a native EFI toolchain.
I wouldn't consider a dependency on libgcc to be 'windows-specific' code, despite it using the same name as a function in MSVCRT. I'd imagine the Mingw developers haven't looked at the MS source code in order to implement it. Regardless, there are compiler options to disable it if you like.

Regards,
John.
Gigasoft
Member
Member
Posts: 856
Joined: Sat Nov 21, 2009 5:11 pm

Re: reference to `___chkstk_ms'

Post by Gigasoft »

It's Windows-specific because calling chkstk is only meaningful on Windows. Other systems do not require it.
jnc100
Member
Member
Posts: 775
Joined: Mon Apr 09, 2007 12:10 pm
Location: London, UK
Contact:

Re: reference to `___chkstk_ms'

Post by jnc100 »

Gigasoft wrote:It's Windows-specific because calling chkstk is only meaningful on Windows. Other systems do not require it.
Its an implementation of stack protection that works on the Microsoft x64 ABI (the same ABI as used in UEFI). It is not required per se under Windows as you can disable it. There also exists a GCC stack protection mechanism, which can also be enabled or disabled, which emits references to __stack_chk_guard which is defined in libssp (and incidentally cannot be compiled easily for a vanilla cross-compiler due to dependencies on libc).

My point was that the compiler is free to emit calls to its bundled libraries at will and you should link with them, and that I do not consider a dependency on libgcc to be windows-specific. However I feel we are arguing about semantics.

Regards,
John.
Gigasoft
Member
Member
Posts: 856
Joined: Sat Nov 21, 2009 5:11 pm

Re: reference to `___chkstk_ms'

Post by Gigasoft »

You must be thinking of something else. Chkstk is definitely only meaningful for Windows user mode applications, as it requires that stack pages are accessed in a decrementing manner. Although it is possible to prevent the compiler from generating chkstk calls, the resulting program will not be correct (unless you know that you have committed all the stack pages in advance). A program that fails to touch all stack pages in the correct order will crash. On other systems, there is no such requirement, and calling chkstk does nothing except waste time.
User avatar
Combuster
Member
Member
Posts: 9301
Joined: Wed Oct 18, 2006 3:45 am
Libera.chat IRC: [com]buster
Location: On the balcony, where I can actually keep 1½m distance
Contact:

Re: reference to `___chkstk_ms'

Post by Combuster »

From the libgcc source (libgcc/config/i386/cygwin.S):

Code: Select all

/* Function prologue calls __chkstk to probe the stack when allocating more
   than CHECK_STACK_LIMIT bytes in one go.  Touching the stack at 4K
   increments is necessary to ensure that the guard pages used
   by the OS virtual memory manger are allocated in correct sequence.  */
Note the mention of "guard pages" and "OS virtual memory manager"
"Certainly avoid yourself. He is a newbie and might not realize it. You'll hate his code deeply a few years down the road." - Sortie
[ My OS ] [ VDisk/SFS ]
jnc100
Member
Member
Posts: 775
Joined: Mon Apr 09, 2007 12:10 pm
Location: London, UK
Contact:

Re: reference to `___chkstk_ms'

Post by jnc100 »

My apologies, I was mistaken. I'd thought it was an implementation of a stack protector (i.e. with canary value et al). Thank you for the pointers, but I guess it is something we don't need in a UEFI environment (at least I can find no evidence that stack probing is required in x64 UEFI). Out of interest, the MSVC compiler also appears to emit these references when used for UEFI (link).

Regards,
John.
Post Reply