성태의 닷넷 이야기
홈 주인
모아 놓은 자료
프로그래밍
질문/답변
사용자 관리
사용자
메뉴
아티클
외부 아티클
유용한 코드
온라인 기능
MathJax 입력기
최근 덧글
[정성태] How can I tell whether two programs...
[정성태] The case of the fail-fast crashes c...
[정성태] Creating Docker multi-arch images f...
[정성태] BinaryFormatter removed from .NET 9...
[정성태] Extending the Windows Shell Progres...
[우광현] 와..... 범위를 잡았으니 클라이언트가 해당 범위를 확인해본다...
[정성태] 딱히, 그것 이상으로 더 설명할 내용이 없습니다. 동적 포...
[정성태] If Windows 3.11 required a 32-bit p...
[정성태] What is a hard error, and what make...
[괴물신인] 질문작성자인데 이 글을 이제봤네요 ㄷㄷ 이 글처럼 타입별로 인...
글쓰기
제목
이름
암호
전자우편
HTML
홈페이지
유형
제니퍼 .NET
닷넷
COM 개체 관련
스크립트
VC++
VS.NET IDE
Windows
Team Foundation Server
디버깅 기술
오류 유형
개발 환경 구성
웹
기타
Linux
Java
DDK
Math
Phone
Graphics
사물인터넷
부모글 보이기/감추기
내용
<div style='display: inline'> <h1 style='font-family: Malgun Gothic, Consolas; font-size: 20pt; color: #006699; text-align: center; font-weight: bold'>Linux 응용 프로그램의 (C++) so 의존성 줄이기(ReleaseMinDependency)</h1> <p> Visual C++의 경우 msvcr[version].dll에 대한 정적 링크를 다음과 같이 하면,<br /> <br /> <pre style='margin: 10px 0px 10px 10px; padding: 10px 0px 10px 10px; background-color: #fbedbb; overflow: auto; font-family: Consolas, Verdana;' > 수동으로 구성해 본 VC++프로젝트 설정: ReleaseMinDependency ; <a target='tab' href='http://www.sysnet.pe.kr/2/0/800'>http://www.sysnet.pe.kr/2/0/800</a> (* Visual Studio 버전에 따라 설정 옵션이 차이가 있습니다.) </pre> <br /> 순수하게 윈도우 시스템 dll에 대한 의존성만 갖게 됩니다. 마찬가지로 Linux 응용 프로그램도 그렇게 만들고 싶었는데요. 기본적으로는 다음과 같이 libstdc++, libgcc_s.so.1, libc.so.6에 대한 C/C++ 관련 런타임이 동적으로 바인딩되어 있습니다.<br /> <br /> <pre style='margin: 10px 0px 10px 10px; padding: 10px 0px 10px 10px; background-color: #fbedbb; overflow: auto; font-family: Consolas, Verdana;' > $ ldd ./projects/testapp/bin/x64/Debug/testapp.so linux-vdso.so.1 (0x00007ffdc17d4000) <span style='color: blue; font-weight: bold'>libstdc++.so.6 => /usr/lib/x86_64-linux-gnu/libstdc++.so.6 (0x00007f71662ce000) libgcc_s.so.1 => /lib/x86_64-linux-gnu/libgcc_s.so.1 (0x00007f71660b6000) libc.so.6 => /lib/x86_64-linux-gnu/libc.so.6 (0x00007f7165cc5000) libm.so.6 => /lib/x86_64-linux-gnu/libm.so.6 (0x00007f7165927000)</span> /lib64/ld-linux-x86-64.so.2 (0x00007f716686a000) </pre> <br /> 회사의 리눅스 개발자에 의하면, 다음의 옵션을 통해 libgcc_s와 libstdc++에 대한 의존성을 제거할 수 있다고 합니다.<br /> <br /> <pre style='margin: 10px 0px 10px 10px; padding: 10px 0px 10px 10px; background-color: #fbedbb; overflow: auto; font-family: Consolas, Verdana;' > [Linker / "Command Line"의 "Additional Options"] -static-libgcc -static-libstdc++ </pre> <br /> 빌드하면 이렇게 홀쭉해진 것을 볼 수 있습니다. ^^<br /> <br /> <pre style='margin: 10px 0px 10px 10px; padding: 10px 0px 10px 10px; background-color: #fbedbb; overflow: auto; font-family: Consolas, Verdana;' > ~$ ldd ./projects/testapp/bin/x64/Debug/testapp.so linux-vdso.so.1 (0x00007ffd587e4000) <span style='color: blue; font-weight: bold'>libc.so.6 => /lib/x86_64-linux-gnu/libc.so.6 (0x00007f2f521c9000)</span> /lib64/ld-linux-x86-64.so.2 (0x00007f2f527e7000) </pre> <br /> 그런데, CRT(C Runtime) 라이브러리인 libc.so.6에 대한 의존성을 제거할 수가 없습니다. 그래서 Visual C++와는 달리 libc 만큼은 빌드 환경과 실행 환경을 고려해야 합니다. (혹시 libc 의존성 없애는 방법을 아시는 분은 덧글 부탁드립니다. ^^)<br /> <br /> <hr style='width: 50%' /><br /> <br /> 검색해 보면, libc 의존성을 없애기 위해 "-static" 옵션을 주면 된다는 글들이 있습니다. 하지만 이 옵션을 [Linker / "Command Line"의 "Additional Options"]에 함께 주면 빌드 시 다음과 같은 식의 오류가 발생합니다.<br /> <br /> <pre style='margin: 10px 0px 10px 10px; padding: 10px 0px 10px 10px; background-color: #fbedbb; overflow: auto; font-family: Consolas, Verdana;' > 1> Linking objects 1> Invoking 'ld' 1> g++ -o "/home/usr23/projects/testapp/../testapp/bin/x64/Release/libtestapp.so" -Wl,--no-undefined -Wl,-z,relro -Wl,-z,now -Wl,-z,noexecstack -shared -static /home/usr23/projects/testapp/../testapp/obj/x64/Release/ClassFactory.o /home/usr23/projects/testapp/../testapp/obj/x64/Release/testapp.o /home/usr23/projects/testapp/../testapp/obj/x64/Release/dllmain.o /home/usr23/projects/testapp/../testapp/obj/x64/Release/ILRewriter.o 1> /usr/bin/ld : error : /usr/lib/gcc/x86_64-linux-gnu/7/crtbeginT.o: relocation R_X86_64_32 against hidden symbol `__TMC_END__' can not be used when making a shared object 1> /usr/bin/ld : error : final link failed: Nonrepresentable section on output 1> collect2 : error : ld returned 1 exit status 1> /usr/bin/ld: /usr/lib/gcc/x86_64-linux-gnu/7/crtbeginT.o: relocation R_X86_64_32 against hidden symbol `__TMC_END__' can not be used when making a shared object 1> /usr/bin/ld: final link failed: Nonrepresentable section on output 1> collect2: error: ld returned 1 exit status </pre> <br /> __TMC_END__ 오류에 관한 것을 다시 검색해 보면,<br /> <br /> <pre style='margin: 10px 0px 10px 10px; padding: 10px 0px 10px 10px; background-color: #fbedbb; overflow: auto; font-family: Consolas, Verdana;' > how to make a static binary of coreutils? ; <a target='tab' href='https://askubuntu.com/questions/530617/how-to-make-a-static-binary-of-coreutils'>https://askubuntu.com/questions/530617/how-to-make-a-static-binary-of-coreutils</a> </pre> <br /> 다음과 같이 기존 crtbeginT.o를 crtbeginS.o로 덮어씌우면 됩니다.<br /> <br /> <pre style='margin: 10px 0px 10px 10px; padding: 10px 0px 10px 10px; background-color: #fbedbb; overflow: auto; font-family: Consolas, Verdana;' > $ <span style='color: blue; font-weight: bold'>g++ --version</span> g++ (Ubuntu 7.3.0-27ubuntu1~18.04) <span style='color: blue; font-weight: bold'>7.3.0</span> Copyright (C) 2017 Free Software Foundation, Inc. This is free software; see the source for copying conditions. There is NO warranty; not even for MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. $ <span style='color: blue; font-weight: bold'>cd /usr/lib/gcc/x86_64-linux-gnu/7.3.0</span> $ sudo cp crtbeginT.o crtbeginT.orig.o $ <span style='color: blue; font-weight: bold'>sudo crtbeginS.o crtbeginT.o</span> </pre> <br /> 실제로 이렇게까지 처리하면 컴파일/링킹 오류는 발생하지 않습니다. 문제는 바이너리 의존성이 오히려 원래대로 돌아가 버립니다.<br /> <br /> <pre style='margin: 10px 0px 10px 10px; padding: 10px 0px 10px 10px; background-color: #fbedbb; overflow: auto; font-family: Consolas, Verdana;' > $ ldd ~/projects/testapp/bin/x64/Release/libtestapp.so linux-vdso.so.1 (0x00007fff71fe9000) libstdc++.so.6 => /usr/lib/x86_64-linux-gnu/libstdc++.so.6 (0x00007fa2814f0000) libc.so.6 => /lib/x86_64-linux-gnu/libc.so.6 (0x00007fa2810ff000) libm.so.6 => /lib/x86_64-linux-gnu/libm.so.6 (0x00007fa280d61000) /lib64/ld-linux-x86-64.so.2 (0x00007fa281a8a000) libgcc_s.so.1 => /lib/x86_64-linux-gnu/libgcc_s.so.1 (0x00007fa280b49000) </pre> <br /> <hr style='width: 50%' /><br /> <br /> 참고로 우분투의 경우 g++ 8 버전을 다음과 같이 설치할 수 있습니다.<br /> <br /> <pre style='margin: 10px 0px 10px 10px; padding: 10px 0px 10px 10px; background-color: #fbedbb; overflow: auto; font-family: Consolas, Verdana;' > $ sudo add-apt-repository ppa:ubuntu-toolchain-r/test $ sudo apt-get update $ sudo apt-get install gcc-8 g++-8 $ gcc-8 --version </pre> <br /> 그럼 /usr/lib/gcc/x86_64-linux-gnu/8 디렉터리에 g++ 8 버전 관련 파일들이 설치됩니다. 이렇게 설치해도 기본적으로 gcc, g++ 명령시에는 g++-8 컴파일러가 동작하지 않습니다. (이를 위해서는 별도의 처리를 해야 하는데 이건 나중에. ^^)<br /> </p><br /> <br /><hr /><span style='color: Maroon'>[이 글에 대해서 여러분들과 의견을 공유하고 싶습니다. 틀리거나 미흡한 부분 또는 의문 사항이 있으시면 언제든 댓글 남겨주십시오.]</span> </div>
첨부파일
스팸 방지용 인증 번호
1823
(왼쪽의 숫자를 입력해야 합니다.)