OS X Mavericks で Clang を使用してカスタム C エントリ ポイントを設定するにはどうすればよいですか?



次のプログラムがあります:


#include <stdio.h>
int bob() {
printf("bob\n");
return 0;
}
int main() {
printf("main\n");
return 0;
}

Linux では、次の方法でカスタム エントリ ポイントを有効にできます。


gcc test.c -Wl,-e,bob

結果のプログラムを実行すると、次のようになります。


./a.out
bob

ただし、OS X では、これは機能しません:


clang test.c -Wl,-e,bob
./a.out
main

これを機能させるためにあらゆることを試みました。多分バグだと思います。 -v の出力は次のとおりです。 オプション:


clang test.c -Wl,-e,bob -v
Apple LLVM version 5.1 (clang-503.0.40) (based on LLVM 3.4svn)
Target: x86_64-apple-darwin13.3.0
Thread model: posix
"/Applications/Xcode.app/Contents/Developer/Toolchains/XcodeDefault.xctoolchain/usr/bin/clang" -cc1 -triple x86_64-apple-macosx10.9.0 -emit-obj -mrelax-all -disable-free -disable-llvm-verifier -main-file-name test.c -mrelocation-model pic -pic-level 2 -mdisable-fp-elim -masm-verbose -munwind-tables -target-cpu core2 -target-linker-version 236.3 -v -resource-dir /Applications/Xcode.app/Contents/Developer/Toolchains/XcodeDefault.xctoolchain/usr/bin/../lib/clang/5.1 -fdebug-compilation-dir /Users/mfichman/jogo -ferror-limit 19 -fmessage-length 125 -stack-protector 1 -mstackrealign -fblocks -fobjc-runtime=macosx-10.9.0 -fencode-extended-block-signature -fdiagnostics-show-option -fcolor-diagnostics -vectorize-slp -o /var/folders/4z/q41by0256hjc7s6v8ljmfpw8lywh5g/T/test-9b80a6.o -x c test.c
clang -cc1 version 5.1 based upon LLVM 3.4svn default target x86_64-apple-darwin13.3.0
#include "..." search starts here:
#include <...> search starts here:
/usr/local/include
/Applications/Xcode.app/Contents/Developer/Toolchains/XcodeDefault.xctoolchain/usr/bin/../lib/clang/5.1/include
/Applications/Xcode.app/Contents/Developer/Toolchains/XcodeDefault.xctoolchain/usr/include
/usr/include
/System/Library/Frameworks (framework directory)
/Library/Frameworks (framework directory)
End of search list.
"/Applications/Xcode.app/Contents/Developer/Toolchains/XcodeDefault.xctoolchain/usr/bin/ld" -demangle -dynamic -arch x86_64 -macosx_version_min 10.9.0 -e bob -o a.out /var/folders/4z/q41by0256hjc7s6v8ljmfpw8lywh5g/T/test-9b80a6.o -lSystem /Applications/Xcode.app/Contents/Developer/Toolchains/XcodeDefault.xctoolchain/usr/bin/../lib/clang/5.1/lib/darwin/libclang_rt.osx.a

clang であることがわかります -e を正しく渡しています ld まで 、おそらくこれは Apple の ld の問題です .もしそうなら、私は回避策に興味があります.


答え:


-e によってオーバーライドされるデフォルトのエントリ ポイント 引数が _main ではありません むしろ start _main の設定と呼び出しを担当します。 、次に _main の戻り値を渡します _exit まで .独自のエントリ ポイントを指定する場合は、これらの手順を自分で実行する必要があります。現在、この初期化を実行する方法はありませんが、_main の使用として別のメイン関数を使用します。 ツールにハードコードされています。


-e 引数が無視される理由は、10.8 での変更によるものです。このリリースより前の start の実装 crt1.o を介して各アプリケーションにリンクされていました . 10.8 では start 処理は dyld で実行でき、LC_MAIN ロード コマンドはプログラム内のメイン関数へのオフセットを指定します。このオフセットを変更することはあなたが望んでいるように見えますが、LC_MAIN 起動メソッドが使用されている場合、ld は常に _main を使用するため、これは現在不可能です。 -e を無視します 口論。独自のエントリ ポイントを指定するには、プログラム起動の古い方法を使用するように ld に指示する必要があります。これは、-no_new_main を渡すことで、10.8 以降の展開ターゲットを持つアプリケーションに対して実行できます。 リンカーに。これは、10.8 より前の展開ターゲットを持つアプリケーションのデフォルトの動作です。