When using the following in my stack.yaml
to (attempt) to compile a static binary through Stack:
ghc-options:
"*": -static -optc-static -optl-static -optl-pthread -fPIC
I get this error:
usr/bin/ld: /usr/lib/gcc/x86_64-amazon-linux/4.8.3/crtbeginT.o: relocation R_X86_64_32 against `__TMC_END__' can not be used when making a shared object; recompile with -fPIC
/usr/lib/gcc/x86_64-amazon-linux/4.8.3/crtbeginT.o: could not read symbols: Bad value
collect2: error: ld returned 1 exit status
`gcc' failed in phase `Linker'. (Exit code: 1)
I'm using stack with docker enabled and a customised Amazon Linux image for building a static binary for use with AWS Lambda.
I have no prior experience with static binary compilation, so I'm a bit stumped on this linker error. Any ideas please?
Here's the gcc args using -v:
/usr/bin/gcc -fno-stack-protector -DTABLES_NEXT_TO_CODE '-Wl,--hash-size=31' -Wl,--reduce-memory-overheads -Wl,--no-as-needed -Wl,-rpath<snipped LOADS> -lHSghc<SNIPPED LOADS> -lpq -lz -lrt -lutil -ldl -lgmp
@Reid Barton is right, regarding the root cause of the problem. The real issue here is that stack is building a shared binary by default. By passing the options
-static
or ghc-options:-optl-static
we are only addressing respectively the ghc layer and ld layer.The correct way to solve the issue is to tell cabal that we want a static binary, as it pilots the build.
The answer on how to do that was provided by sausade on reddit: you just have to add the following to your .cabal file in the executable section:
This will hint to cabal that we want a static binary and as so it will not pass
-shared
to ghc.So once you have added this directive in your cabal file, you just have to run:
stack install --ghc-options="-fPIC"
and voilà!
P.S: If you don't have static libraries on your system (like with Archlinux)
ld
will fallback to dynamic linkingPrevious answer:
That's an old gcc issue that no one care to fix. https://bugs.launchpad.net/ubuntu/+source/gcc-4.4/+bug/640734
For your original problem here are two useful links to show you how to do it: