Microsoft MVP성태의 닷넷 이야기
글쓴 사람
정성태 (techsharer at outlook.com)
홈페이지
첨부 파일
(연관된 글이 3개 있습니다.)

마이크로소프트의 CoreCLR 프로파일러 리눅스 예제를 Visual Studio F5 원격 디버깅하는 방법

CoreCLR 프로파일러 예제를 리눅스 운영체제에서 빌드해 보고,

마이크로소프트의 CoreCLR 프로파일러 예제 빌드 방법
; https://www.sysnet.pe.kr/2/0/11294

이것을 Visual Studio의 CMake 프로젝트로 변경도 해봤는데요.

마이크로소프트의 CoreCLR 프로파일러 예제를 Visual Studio CMake로 빌드하는 방법
; https://www.sysnet.pe.kr/2/0/11842

F5 디버깅이 안 된다는 것은 차라리 리눅스에서 개발하는 것이 나을 정도이므로 별로 효용성이 없습니다. 하지만, 이 프로젝트를 비주얼 스튜디오의 리눅스 프로젝트로 변경하면,

Visual Studio 2019 - 리눅스 프로젝트를 이용한 공유/실행(so/out) 프로그램 개발 환경 설정
; https://www.sysnet.pe.kr/2/0/11844

F5 디버깅이 가능합니다. 변경 과정도 그다지 어렵지 않습니다. 그냥 리눅스 프로젝트 생성 후 위의 글에 설명한 데로 so 프로젝트로 변경하고 빌드에 필요한 dotnet core include 파일들을 포함하면 됩니다. include 파일들은 vcxproj 파일에 다음과 같은 식으로 일괄 포함하는 것도 가능하지만,

<ItemGroup>
    <ClInclude Include=".\cor\**\*.h" />
</ItemGroup>

일일이 추가해도 상관없습니다. 그다음 리눅스 측에서 빌드할 때 cor 폴더의 header 파일들을 찾을 수 있도록 "C/C++" / "General"의 "Additional Include Directories"에 다음의 4개 폴더를 등록합니다.

./cor/src/inc
./cor/src/pal/inc/rt
./cor/src/pal/inc
./cor/src/pal/prebuilt/inc

실제로 dotnet core 헤더 파일들에는 하위 폴더들이 많은데 프로파일러 예제 빌드로는 저 4개의 폴더만 있으면 됩니다. 그리고 "C/C++" / "Preprocessor"의 "Preprocessor Definitions"에 "BIT64;PAL_STDCPP_COMPAT;PLATFORM_UNIX" 매크로 상수도 추가한 후 경고를 없애기 위해 "C/C++" / "Command Line"의 "Additional Options"에 다음의 옵션도 넣어줍니다.

-Wno-undef -Wno-invalid-noreturn -fms-extensions -Wno-conversion -Wno-pragma-pack -Wno-unused-variable

마지막으로, "C/C++" / "General"의 "C++ Compiler"를 clang++로 변경하면 끝!

(첨부 파일은 이렇게 변경한 프로젝트 파일입니다.)




그럼 F5 디버깅을 해봐야 할 텐데요. 이를 위해 /etc/profile에 CORECLR 관련 환경 변수를 등록하든가,

$ sudo nano /etc/profile
 
export CORECLR_PROFILER={cf0d821e-299b-5307-a3d8-b283c03916dd}
export CORECLR_ENABLE_PROFILING=1
export CORECLR_PROFILER_PATH=~/projects/corprofiler/bin/x64/Debug/libcorprofiler.so

$ sudo source /etc/profile

아니면 프로젝트 속성 창의 "Debugging"에 "Pre-Launch Command"로 다음과 같이 환경 변수를 등록하면 됩니다.

export CORECLR_ENABLE_PROFILING=1; export CORECLR_PROFILER={cf0d821e-299b-5307-a3d8-b283c03916dd};export CORECLR_PROFILER_PATH=libcorprofiler.so;

(위에서는 CORECLR_PROFILER_PATH에 경로를 생략했는데 이렇게 하려면 /etc/ld.so.conf에 경로를 등록해야 합니다.)

그다음 실행 파일을 /usr/bin/dotnet으로, 인자로 닷넷 예제 프로젝트의 빌드된 DLL을 주면,

Program: /usr/bin/dotnet
Program Arguments: webapp.dll
Working Directory: ~/coreapp

이후 비주얼 스튜디오에서 F5 키를 눌러 원격 디버깅을 진행할 수 있습니다. 오~~~ 이제야 개발할 맛이 납니다. ^^




참고로, 기본 프로파일러 예제는 의존성이 많은데요,

$ ldd ~/projects/corprofiler/bin/x64/Debug/libcorprofiler.so
        linux-vdso.so.1 (0x00007ffda3b38000)
        libstdc++.so.6 => /usr/lib/x86_64-linux-gnu/libstdc++.so.6 (0x00007fa7b5cda000)
        libgcc_s.so.1 => /lib/x86_64-linux-gnu/libgcc_s.so.1 (0x00007fa7b5ac2000)
        libc.so.6 => /lib/x86_64-linux-gnu/libc.so.6 (0x00007fa7b56d1000)
        libm.so.6 => /lib/x86_64-linux-gnu/libm.so.6 (0x00007fa7b5333000)
        /lib64/ld-linux-x86-64.so.2 (0x00007fa7b627b000)

다음의 글에 따라 어느 정도 줄이는 것이 가능합니다.

Linux 응용 프로그램의 (C++) so 의존성 줄이기(ReleaseMinDependency)
; https://www.sysnet.pe.kr/2/0/11845

export된 symbol을 보면 재미있는 점이 있는데요.

$ nm ~/projects/corprofiler/bin/x64/Debug/libcorprofiler.so
0000000000011380 V CLSID_CLR_v1_MetaData
0000000000011390 V CLSID_CLR_v2_MetaData
0000000000011470 V CLSID_Cor
0000000000011480 V CLSID_CorMetaDataDispenser
0000000000011490 V CLSID_CorMetaDataDispenserReg
0000000000011330 V CLSID_CorMetaDataDispenserRuntime
00000000000114a0 V CLSID_CorMetaDataReg
0000000000217008 d DW.ref.__gxx_personality_v0
000000000000e410 T DecoderGetOnDiskSize
000000000000e290 T DecoderInit
000000000000e240 T DllCanUnloadNow
000000000000e170 T DllGetClassObject
0000000000217018 D EnterMethodAddress
000000000001419c r GCC_except_table1
...[생략]...

저렇게 윈도우에서의 프로파일러와 유사한 COM 접근 방식을 리눅스에서도 그대로 흉내 내고 있습니다. ^^ 실제로 소스코드를 봐도 vtable 기반 하의 COM 접근 방식을 동일하게 구현하고 있습니다.




dotnet core의 헤더 파일들을 모두 포함시키면 다음과 같은 식의 빌드 오류가 발생할 수 있습니다.

1>  Task "MultiToolTask" skipped, due to false condition; ('%(Midl.ExcludedFromBuild)'!='true' and '$(UseMultiToolTask)' == 'true') was evaluated as (''!='true' and '' == 'true').
1>  Using "MIDL" task from assembly "C:\Program Files (x86)\Microsoft Visual Studio\2019\Preview\MSBuild\Microsoft\VC\v160\Microsoft.Build.CppTasks.Common.dll".
1>  Task "MIDL"
1>    midl.exe  cor\src\inc\clrdata.idl
1>    Tracking command:
1>    C:\Program Files (x86)\Microsoft Visual Studio\2019\Preview\MSBuild\Current\Bin\Tracker.exe /d "C:\Program Files (x86)\MSBuild\15.0\FileTracker\FileTracker32.dll" /r C:\BUILD_VS_CLR_SAMPLES\CORPROFILER\CORPROFILER\COR\SRC\INC\CLRDATA.IDL /b MSBuildConsole_CancelEvent9ebc403f3abc41d38c14e3110ec31c70  /c midl.exe   cor\src\inc\clrdata.idl
1>    TRACKER : error TRK0005: Failed to locate: "midl.exe". The system cannot find the file specified.
1>

왜냐하면, 포함된 폴더에는 헤더 파일뿐만 아니라 idl, rc, cpp 등의 파일도 있기 때문인데요, 따라서 그 파일들은 빌드에서 제외해야 합니다.




리눅스 프로젝트 빌드 시 다음과 같은 오류가 발생한다면?

1>  C:\Program Files (x86)\Microsoft Visual Studio\2019\Preview\MSBuild\Microsoft\VC\v160\Application Type\Linux\1.0\Linux.targets(151,5): error : Current project architecture 'ARM' is incompatible with the remote system architecture 'x64' ('x64'). Please switch the project architecture to 'x64' in Configuration Manager.

현재 비주얼 스튜디오의 빌드 타겟이 ARM으로 설정되어 있기 때문입니다. x64로 바꿔주면 정상적으로 빌드가 됩니다.




[이 글에 대해서 여러분들과 의견을 공유하고 싶습니다. 틀리거나 미흡한 부분 또는 의문 사항이 있으시면 언제든 댓글 남겨주십시오.]

[연관 글]






[최초 등록일: ]
[최종 수정일: 3/26/2019]

Creative Commons License
이 저작물은 크리에이티브 커먼즈 코리아 저작자표시-비영리-변경금지 2.0 대한민국 라이센스에 따라 이용하실 수 있습니다.
by SeongTae Jeong, mailto:techsharer at outlook.com

비밀번호

댓글 작성자
 



2019-05-21 03시22분
How to debug .net core 2 app on linux server via visual studio 2017 on windows? [Music no 4]
; http://tomaszjarzynski.pl/how-debug-net-core-2-app-linux-server-via-visual-studio-2017-windows-music-4/

Debugging ASP Core on Linux with Visual Studio 2017
; https://blogs.msdn.microsoft.com/premier_developer/2017/11/10/debugging-asp-core-on-linux-with-visual-studio-2017/
정성태

1  2  3  4  5  6  7  8  9  10  11  12  13  [14]  15  ...
NoWriterDateCnt.TitleFile(s)
13279정성태3/9/20234108오류 유형: 851. 파이썬 ModuleNotFoundError: No module named '_cffi_backend'
13278정성태3/8/20234139개발 환경 구성: 669. WSL 2의 (init이 아닌) systemd 지원 [1]
13277정성태3/6/20234763개발 환경 구성: 668. 코드 사인용 인증서 신청 및 적용 방법(예: Digicert)
13276정성태3/5/20234444.NET Framework: 2102. C# 11 - ref struct/ref field를 위해 새롭게 도입된 scoped 예약어
13275정성태3/3/20234768.NET Framework: 2101. C# 11의 ref 필드 설명
13274정성태3/2/20234334.NET Framework: 2100. C# - ref 필드로 ref struct 타입을 허용하지 않는 이유
13273정성태2/28/20234056.NET Framework: 2099. C# - 관리 포인터로서의 ref 예약어 의미
13272정성태2/27/20234327오류 유형: 850. SSMS - mdf 파일을 Attach 시킬 때 Operating system error 5: "5(Access is denied.)" 에러
13271정성태2/25/20234230오류 유형: 849. Sql Server Configuration Manager가 시작 메뉴에 없는 경우
13270정성태2/24/20233845.NET Framework: 2098. dotnet build에 /p 옵션을 적용 시 유의점
13269정성태2/23/20234412스크립트: 46. 파이썬 - uvicorn의 콘솔 출력을 UDP로 전송
13268정성태2/22/20234967개발 환경 구성: 667. WSL 2 내부에서 열고 있는 UDP 서버를 호스트 측에서 접속하는 방법
13267정성태2/21/20234862.NET Framework: 2097. C# - 비동기 소켓 사용 시 메모리 해제가 finalizer 단계에서 발생하는 사례파일 다운로드1
13266정성태2/20/20234487오류 유형: 848. .NET Core/5+ - Process terminated. Couldn't find a valid ICU package installed on the system
13265정성태2/18/20234398.NET Framework: 2096. .NET Core/5+ - PublishSingleFile 유형에 대한 runtimeconfig.json 설정
13264정성태2/17/20235916스크립트: 45. 파이썬 - uvicorn 사용자 정의 Logger 작성
13263정성태2/16/20234074개발 환경 구성: 666. 최신 버전의 ilasm.exe/ildasm.exe 사용하는 방법
13262정성태2/15/20235135디버깅 기술: 191. dnSpy를 이용한 (소스 코드가 없는) 닷넷 응용 프로그램 디버깅 방법 [1]
13261정성태2/15/20234412Windows: 224. Visual Studio - 영문 폰트가 Fullwidth Latin Character로 바뀌는 문제
13260정성태2/14/20234211오류 유형: 847. ilasm.exe 컴파일 오류 - error : syntax error at token '-' in ... -inf
13259정성태2/14/20234362.NET Framework: 2095. C# - .NET5부터 도입된 CollectionsMarshal
13258정성태2/13/20234251오류 유형: 846. .NET Framework 4.8 Developer Pack 설치 실패 - 0x81f40001
13257정성태2/13/20234337.NET Framework: 2094. C# - Job에 Process 포함하는 방법 [1]파일 다운로드1
13256정성태2/10/20235177개발 환경 구성: 665. WSL 2의 네트워크 통신 방법 - 두 번째 이야기
13255정성태2/10/20234495오류 유형: 845. gihub - windows2022 이미지에서 .NET Framework 4.5.2 미만의 프로젝트에 대한 빌드 오류
13254정성태2/10/20234382Windows: 223. (WMI 쿼리를 위한) PowerShell 문자열 escape 처리
1  2  3  4  5  6  7  8  9  10  11  12  13  [14]  15  ...