Microsoft MVP성태의 닷넷 이야기
VC++: 47. Apache Module에 대한 'F5 디버그 (Start with debugging)' [링크 복사], [링크+제목 복사],
조회: 32181
글쓴 사람
정성태 (techsharer at outlook.com)
홈페이지
첨부 파일
 
(연관된 글이 7개 있습니다.)
Apache Module에 대한 'F5 디버그 (Start with debugging)'

역시, 디버깅은 Visual Studio IDE 내에서 "F5 (Start with debugging)" 키를 이용한 디버깅이 되어야 생산성이 급격히 높아지죠. "Apache Module" 개발에 대해서도 물론 그렇게 할 수 있습니다.

사실, 방법도 그리 어렵지 않은데요. 지난번 IDE 내에서의 컴파일에 이어서 계속 이야기를 진행해 보겠습니다.

윈도우에서 Apache Module 컴파일 (VC++)
; https://www.sysnet.pe.kr/2/0/1051

컴파일된 결과물이 mod_example.so라는 DLL 형식이기 때문에 "F5" 키를 이용하여 디버깅하려면 우선, EXE 파일 지정이 필요합니다. 당연히 아파치 실행파일인 httpd.exe가 이에 해당할테니, 다음과 같이 프로젝트 속성 창 화면에서 httpd.exe 경로를 지정해 줍니다.

how_to_debug_module_handler_1.png

그다음, 빌드된 결과물이 곧바로 반영되어 아파치에 올라가야 할 테니 출력 폴더를 apache의 module 폴더로 해주면 됩니다.

how_to_debug_module_handler_2.png

일단은, 우리가 아는 지식대로라면 이 정도면 정상적으로 'F5 디버깅'이 가능해야 합니다. x_register_hooks 함수와 x_handler 함수에 BP(BreakPoint)를 걸고 'F5' 키를 누르면 httpd.exe 프로세스가 시작되면서 x_register_hooks 함수에서 정상적으로 BP가 동작하는 것을 확인할 수 있습니다.

그런데, 아쉽게도 x_handler 함수에서는 멈추지 않습니다.




x_handler에서 멈추지 않는 이유가 뭘까요? 아파치는 httpd.exe 프로세스를 부모/자식 관계로 실행합니다. 예전의 IIS 5와 비교하자면 inetinfo.exe와 aspnet_wp.exe로 나뉜 것과 비슷한 예입니다. 그리고, Visual Studio는 'F5 디버깅' 시에 첫 번째 프로세스에 대해서만 디버깅을 하고 있으며 그 프로세스가 생성한 2차(자식) 프로세스에는 디버깅 기능을 제공하지 않습니다. 이 때문에, 실제적인 요청을 처리하는 2차 httpd.exe 프로세스에서 실행되는 x_handler 함수의 BP가 동작하지 않는 것입니다.

이렇게 되면, x_handler를 디버깅하기 위해서는 실행 후에 디버거를 붙이는 수밖에는 없어 보이는데요. 물론, 그렇게 하면 'F5 디버깅'과는 비교할 수 없는 생산성 차이가 납니다.

뭐 좋은 방법이 없을까요? ^^

다행히, 지난번에 써둔 글이 생각나는군요. ^^

보호 모드로 응용 프로그램 디버깅하는 방법
; https://www.sysnet.pe.kr/2/0/682

보호 모드로 응용 프로그램 디버깅하는 방법 - 두 번째 이야기
; https://www.sysnet.pe.kr/2/0/683

아하~~~ ^^ 매크로를 이용하면 됩니다. 일단, 현재 "보호 모드로 응용 프로그램 디버깅하는 방법 - 두 번째 이야기" 글에 정리된 매크로를 그대로 가져다가 httpd.exe 모듈 디버깅에 사용하면 다음과 같은 오류가 발생합니다.

httpd.exe: System.Runtime.InteropServices.COMException (0x800704DF): An attempt was made to perform an initialization operation when initialization has already been completed. 
    at Microsoft.VisualBasic.CompilerServices.LateBinding.InternalLateCall(Object o, Type objType, String name, Object[] args, String[] paramnames, Boolean[] CopyBack, Boolean IgnoreReturn) 
    at Microsoft.VisualBasic.CompilerServices.NewLateBinding.LateCall(Object Instance, Type Type, String MemberName, Object[] Arguments, String[] ArgumentNames, Type[] TypeArguments, Boolean[] CopyBack, Boolean IgnoreReturn) 
    at MyMacros.F5Debug.OnEventArrived(Object sender, EventArrivedEventArgs e) in vsmacros://...[생략].../MyMacros/MyMacros.vsmacros/F5Debug:line 98 

원인은, 그 매크로가 부모 자식이 모두 동일한 이름으로 실행되는 경우에 대한 배려가 없기 때문입니다. 이 때문에 이미 Visual Studio가 디버깅하고 있는 부모 httpd.exe 프로세스를 다시 "Process.Attach()" 하려고 시도하는 경우가 발생해서 "An attempt was made to perform an initialization operation when initialization has already been completed."와 같은 오류 메시지를 내는 것입니다.

그래서 이런 문제를 보완한 다음의 매크로 소스 코드를 가져다 사용하시면 됩니다.

Imports System
Imports EnvDTE
Imports EnvDTE80
Imports EnvDTE90
Imports System.Diagnostics
Imports System.Management

Public Module F5Debug

    Public Sub RunAndAttachTo()

        DTE.Solution.SolutionBuild.Build(True)
        DTE.Solution.SolutionBuild.Debug()

        AttachTo()

    End Sub

    Dim f_watcher As ManagementEventWatcher

    Public Sub AttachTo()

        Dim pol As String
        pol = "1"

        Dim queryString As String
        queryString = "SELECT *" & _
                        "  FROM __InstanceOperationEvent " & _
                        "WITHIN  " & pol & _
                        " WHERE TargetInstance ISA 'Win32_Process'"

        Dim scope As String
        scope = "\\.\root\CIMV2"

        If (f_watcher Is Nothing) Then
        Else
            f_watcher.Stop()
            f_watcher = Nothing
        End If

        f_watcher = New ManagementEventWatcher(scope, queryString)

        ' 새로 생성되는 프로세스를 감시한다.

        AddHandler f_watcher.EventArrived, AddressOf OnEventArrived
        f_watcher.Start()

    End Sub

    Private Sub OnEventArrived(ByVal sender As Object, _
        ByVal e As EventArrivedEventArgs)

        Dim rootWatcher As ManagementEventWatcher
        rootWatcher = sender

        Dim debuggeeProcessId As Integer

        Dim processes As EnvDTE.Processes
        processes = DTE.Debugger.DebuggedProcesses

        ' 1차 디버깅 프로세스의 ID를 구한다.
        ' 즉 psexec.exe의 ID를 구한다.
        If (processes.Count <> 0) Then
            For Each procDebuggee As EnvDTE.Process In processes
                debuggeeProcessId = procDebuggee.ProcessID
            Next
        End If

        If (debuggeeProcessId = 0) Then
            Exit Sub
        End If

        ' 새로 생성된 프로세스의 부모 프로세스 ID를 구한다.
        ' 즉 현재 주 디버기 process의 하위에 생성된 프로세스를 구별해 낸다.
        Dim parentProcessIdOfnewProcess As Integer
        Dim targetInstance As ManagementBaseObject
        Dim processId As Integer
        targetInstance = e.NewEvent.Properties("TargetInstance").Value

        parentProcessIdOfnewProcess = targetInstance.Properties("ParentProcessId").Value
        processId = targetInstance.Properties("ProcessID").Value

        If (parentProcessIdOfnewProcess = debuggeeProcessId) Then
            rootWatcher.Stop()

            Dim newProcessName As String
            newProcessName = targetInstance.Properties("Name").Value.ToString()

            For Each proc In DTE.Debugger.LocalProcesses

                Dim proc3 As EnvDTE90.Process3
                proc3 = proc

                If (proc3.ProcessID = processId) Then
                    Try
                        proc.Attach()
                        Debug.Print(newProcessName & ": Attached successfully!")
                    Catch x As Exception
                        Debug.Print(newProcessName & ": " & x.ToString())
                    Finally
                    End Try

                    Exit For
                End If
            Next

        End If

    End Sub

End Module

위의 매크로를 이제 "보호 모드로 응용 프로그램 디버깅하는 방법" 글에서 설명한 것처럼 특정 단축키에 할당하거나 툴바에 새로운 아이콘으로 등록해서 편하게 사용할 수 있습니다. 제 경우에는 "Shift + F2" 키로 할당해 두었고, 아파치 모듈 프로젝트에서 "Shift + F2" 조합을 눌러 다음과 같이 디버깅 상태로 진입하는 것이 가능했습니다.

how_to_debug_module_handler_3.png

오히려, 주의해야 할 것은 종료 과정입니다. 전통적으로 "Shift + F5 (Stop Debugging)" 키를 누르면 현재 디버깅 중인 프로세스를 빠져나오게 되는데요. 이렇게 하면 부모/자식 관계로 생성되는 아파치의 경우에는 부모 프로세스만 종료되고 자식 프로세스가 살아 남아 다음번 아파치 프로그램 실행 시에 부모 프로세스가 초기화 도중 그냥 종료를 해버립니다.

이런 문제를 방지하기 위해서는 "Shift + F5" 키를 눌러 디버깅 세션을 빠져나오면 안되고, 반드시 현재 실행 중인 아파치 콘솔을 종료해 주어야 합니다. (만약 "Shift + F5" 키를 누른 경우라면, 작업 관리자에서 남아 있는 httpd.exe 프로세스를 직접 종료시켜야 합니다.)

어쨌든, 이제 남은 것은 Visual Studio에서 인텔리센스와 F5(Shift + F2)에 해당하는 디버깅 기능을 이용하여 아파치 모듈을 좀 더 쉽게 개발하는 일이겠죠.

(다들... 이렇게 개발하고 있는 거 맞죠? 설마... printf 문 찍으면서 디버깅하고 있는 거 아니죠? ^^)



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

[연관 글]






[최초 등록일: ]
[최종 수정일: 6/26/2021]

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

비밀번호

댓글 작성자
 



2012-03-20 06시10분
[초보개발] 안녕하세요 아파치 모듈개발을 개발하려는 초보 개발자입니다 궁금해서 댓글을 남깁니다.

우선 그럼 아파치실행파일을 보호모드로 실행한뒤 저매크로를 F5나 F10 버튼이아닌 매크로로 만들어낸 버튼을 통해 디버깅을 하게되는건가요?? 보고 잘따라했는데 잘안되서요;;
[guest]
2012-03-20 08시10분
아파치의 경우에는 보호 모드와는 무관하기 때문에 그 부분은 신경쓰시지 않아도 됩니다. 단지, 그 링크를 예로 들었던 것은, 그 안의 매크로 함수 기능이 Parent/Child에서의 디버깅이 가능하도록 되어 있기 때문입니다. 그 매크로를 꼼꼼하게 보시고 이해하신다면, 사용하는데 크게 어려움이 없을 것입니다. ^^
정성태

1  2  3  [4]  5  6  7  8  9  10  11  12  13  14  15  ...
NoWriterDateCnt.TitleFile(s)
13843정성태12/13/20244391오류 유형: 938. Docker container 내에서 빌드 시 error MSB3021: Unable to copy file "..." to "...". Access to the path '...' is denied.
13842정성태12/12/20244534디버깅 기술: 205. Windbg - KPCR, KPRCB
13841정성태12/11/20244866오류 유형: 937. error MSB4044: The "ValidateValidArchitecture" task was not given a value for the required parameter "RemoteTarget"
13840정성태12/11/20244439오류 유형: 936. msbuild - Your project file doesn't list 'win' as a "RuntimeIdentifier"
13839정성태12/11/20244878오류 유형: 936. msbuild - error CS1617: Invalid option '12.0' for /langversion. Use '/langversion:?' to list supported values.
13838정성태12/4/20244606오류 유형: 935. Windbg - Breakpoint 0's offset expression evaluation failed.
13837정성태12/3/20245073디버깅 기술: 204. Windbg - 윈도우 핸들 테이블 (3) - Windows 10 이상인 경우
13836정성태12/3/20244628디버깅 기술: 203. Windbg - x64 가상 주소를 물리 주소로 변환 (페이지 크기가 2MB인 경우)
13835정성태12/2/20245071오류 유형: 934. Azure - rm: cannot remove '...': Directory not empty
13834정성태11/29/20245305Windows: 275. C# - CUI 애플리케이션과 Console 윈도우 (Windows 10 미만의 Classic Console 모드인 경우) [1]파일 다운로드1
13833정성태11/29/20244979개발 환경 구성: 737. Azure Web App에서 Scale-out으로 늘어난 리눅스 인스턴스에 SSH 접속하는 방법
13832정성태11/27/20244915Windows: 274. Windows 7부터 도입한 conhost.exe
13831정성태11/27/20244382Linux: 111. eBPF - BPF_MAP_TYPE_PERF_EVENT_ARRAY, BPF_MAP_TYPE_RINGBUF에 대한 다양한 용어들
13830정성태11/25/20245202개발 환경 구성: 736. 파이썬 웹 앱을 Azure App Service에 배포하기
13829정성태11/25/20245174스크립트: 67. 파이썬 - Windows 버전에서 함께 설치되는 py.exe
13828정성태11/25/20244450개발 환경 구성: 735. Azure - 압축 파일을 이용한 web app 배포 시 디렉터리 구분이 안 되는 문제파일 다운로드1
13827정성태11/25/20245105Windows: 273. Windows 환경의 파일 압축 방법 (tar, Compress-Archive)
13826정성태11/21/20245338닷넷: 2313. C# - (비밀번호 등의) Console로부터 입력받을 때 문자열 출력 숨기기(echo 끄기)파일 다운로드1
13825정성태11/21/20245672Linux: 110. eBPF / bpf2go - BPF_RINGBUF_OUTPUT / BPF_MAP_TYPE_RINGBUF 사용법
13824정성태11/20/20244751Linux: 109. eBPF / bpf2go - BPF_PERF_OUTPUT / BPF_MAP_TYPE_PERF_EVENT_ARRAY 사용법
13823정성태11/20/20245304개발 환경 구성: 734. Ubuntu에 docker, kubernetes (k3s) 설치
13822정성태11/20/20245172개발 환경 구성: 733. Windbg - VirtualBox VM의 커널 디버거 연결 시 COM 포트가 없는 경우
13821정성태11/18/20245096Linux: 108. Linux와 Windows의 프로세스/스레드 ID 관리 방식
13820정성태11/18/20245252VS.NET IDE: 195. Visual C++ - C# 프로젝트처럼 CopyToOutputDirectory 항목을 추가하는 방법
13819정성태11/15/20244490Linux: 107. eBPF - libbpf CO-RE의 CONFIG_DEBUG_INFO_BTF 빌드 여부에 대한 의존성
13818정성태11/15/20245302Windows: 272. Windows 11 24H2 - sudo 추가
1  2  3  [4]  5  6  7  8  9  10  11  12  13  14  15  ...