성태의 닷넷 이야기
홈 주인
모아 놓은 자료
프로그래밍
질문/답변
사용자 관리
사용자
메뉴
아티클
외부 아티클
유용한 코드
온라인 기능
MathJax 입력기
최근 덧글
[정성태] Roll A Lisp In C - Reading ; https...
[정성태] Java - How to use the Foreign Funct...
[정성태] 제가 큰 실수를 했군요. ^^; Delegate를 통한 Bein...
[정성태] Working with Rust Libraries from C#...
[정성태] Detecting blocking calls using asyn...
[정성태] 아쉽게도, 커뮤니티는 아니고 개인 블로그입니다. ^^
[정성태] 질문이 잘 이해가 안 됩니다. 우선, 해당 소스코드에서 ILis...
[양승조
] var대신 dinamic으로 선언해서 해결은 했습니다. 맞는 해...
[양승조
] 또 막혔습니다. ㅠㅠ var list = props[i].Ge...
[양승조
] 아. 감사합니다. 어제는 안됐던것 같은데....정신을 차려야겠네...
글쓰기
제목
이름
암호
전자우편
HTML
홈페이지
유형
제니퍼 .NET
닷넷
COM 개체 관련
스크립트
VC++
VS.NET IDE
Windows
Team Foundation Server
디버깅 기술
오류 유형
개발 환경 구성
웹
기타
Linux
Java
DDK
Math
Phone
Graphics
사물인터넷
부모글 보이기/감추기
내용
<div style='display: inline'> <div style='font-family: 맑은 고딕, Consolas; font-size: 20pt; color: #006699; text-align: center; font-weight: bold'>소스 코드로부터 php5apache2_2.dll 생성하는 방법</div> <br /> 지난번에, PHP 환경을 소스 코드로부터 빌드하는 방법을 알아봤었습니다.<br /> <br /> <pre style='margin: 10px 0px 10px 10px; padding: 10px 0px 10px 10px; width: 800px; background-color: #fbedbb; overflow-x: scroll; font-family: Consolas, Verdana;' > PHP 소스를 윈도우 환경에서 빌드하기 ; <a target='tab' href='http://www.sysnet.pe.kr/2/0/1042'>http://www.sysnet.pe.kr/2/0/1042</a> </pre> <br /> 그때도 언급했지만 PHP의 빌드된 버전을 다운로드한 것과 소스 코드로부터 빌드한 결과물에 상당한 차이가 있었습니다. 그중에서, 아파치의 httpd.conf에 등록해 줘야 하는 php5apache2_2.dll 파일이 누락되어 있다고 했는데요. 이제는 제법 모듈에 대한 이해도 했으니,<br /> <br /> <pre style='margin: 10px 0px 10px 10px; padding: 10px 0px 10px 10px; width: 800px; background-color: #fbedbb; overflow-x: scroll; font-family: Consolas, Verdana;' > 윈도우에서 Apache Module 컴파일 (VC++) ; <a target='tab' href='http://www.sysnet.pe.kr/2/0/1051'>http://www.sysnet.pe.kr/2/0/1051</a> 윈도우에서 Apache Module - Content Handler 컴파일 ; <a target='tab' href='http://www.sysnet.pe.kr/2/0/1056'>http://www.sysnet.pe.kr/2/0/1056</a> </pre> <br /> php5apache2_2.dll을 소스 코드로부터 생성하는 방법을 알아봐야겠습니다. ^^<br /> <br /> <hr style='width: 50%' /><br /> <br /> 이를 위해 우선 소스 코드를 찾아야 하는데, <a target='tab' href='http://www.php.net/downloads.php'>php-5.3.6.tar.gz</a>을 풀어놓은 폴더들을 보니 "D:\phpbuild\php53dev\vc9\x86\php-5.3.6\sapi\apache2handler"가 바로 php5apache2_2.dll에 대한 소스 코드라고 예상되더군요. 해당 폴더에 보면, config.w32라는 파일이 있는데 아래와 같은 내용을 포함하고 있습니다.<br /> <br /> <pre style='margin: 10px 0px 10px 10px; padding: 10px 0px 10px 10px; width: 800px; background-color: #fbedbb; overflow-x: scroll; font-family: Consolas, Verdana;' > // vim:ft=javascript // $Id: config.w32 259731 2008-05-14 03:13:17Z auroraeosrose $ <b style='COLOR: blue'>ARG_ENABLE('apache2handler', 'Build Apache 2.x handler', 'no');</b> if (PHP_APACHE2HANDLER != "no") { if (PHP_ZTS == "no") { WARNING("Apache2 module requires an --enable-zts build of PHP on windows"); } else if (<b style='COLOR: blue'>CHECK_HEADER_ADD_INCLUDE("httpd.h", "CFLAGS_APACHE2HANDLER", PHP_PHP_BUILD + "\\include\\apache2")</b> && <b style='COLOR: blue'>CHECK_LIB("libhttpd.lib", "apache2handler", PHP_PHP_BUILD + "\\lib\\apache2")</b> && CHECK_LIB("libapr.lib", "apache2handler", PHP_PHP_BUILD + "\\lib\\apache2") && CHECK_LIB("libaprutil.lib", "apache2handler", PHP_PHP_BUILD + "\\lib\\apache2") ) { SAPI('apache2handler', 'mod_php5.c sapi_apache2.c apache_config.c php_functions.c', <b style='COLOR: blue'>'php' + PHP_VERSION + 'apache2.dll'</b>, '/D PHP_APACHE2_EXPORTS /I win32'); } else { WARNING("Could not find apache2 libraries/headers"); } } <b style='COLOR: blue'>ARG_ENABLE('apache2-2handler', 'Build Apache 2.2.x handler', 'no');</b> if (PHP_APACHE2_2HANDLER != "no") { if (PHP_ZTS == "no") { WARNING("Apache2 module requires an --enable-zts build of PHP on windows"); } else if (<b style='COLOR: blue'>CHECK_HEADER_ADD_INCLUDE("httpd.h", "CFLAGS_APACHE2_2HANDLER", PHP_PHP_BUILD + "\\include\\apache2_2")</b> && <b style='COLOR: blue'>CHECK_LIB("libhttpd.lib", "apache2_2handler", PHP_PHP_BUILD + "\\lib\\apache2_2")</b> && CHECK_LIB("libapr-1.lib", "apache2_2handler", PHP_PHP_BUILD + "\\lib\\apache2_2") && CHECK_LIB("libaprutil-1.lib", "apache2_2handler", PHP_PHP_BUILD + "\\lib\\apache2_2") ) { SAPI('apache2_2handler', 'mod_php5.c sapi_apache2.c apache_config.c php_functions.c', <b style='COLOR: blue'>'php' + PHP_VERSION + 'apache2_2.dll'</b>, '/D PHP_APACHE2_EXPORTS /I win32', 'sapi\\apache2_2handler'); } else { WARNING("Could not find apache2.2 libraries/headers"); } } </pre> <br /> config.w32 파일이 어떻게 해석되는지는 정확히 알 수 없으나, 대충의 감각으로 보면 'apache2handler' 옵션이 설정되어 있으면 php5apache2.dll 파일이 생성되고, apache2-2handler 옵션이 설정되어 있으면 php5apache2_2.dll을 생성해 내는 것으로 보입니다. 이걸 보면서 <a target='tab' href='http://www.sysnet.pe.kr/2/0/1042'>지난번 글의 마지막에 옮겨두었던 configure --help 출력물</a>이 생각나더군요.<br /> <br /> <pre style='margin: 10px 0px 10px 10px; padding: 10px 0px 10px 10px; width: 800px; background-color: #fbedbb; overflow-x: scroll; font-family: Consolas, Verdana;' > D:\phpbuild\php53dev\vc9\x86\php-5.3.6><b style='COLOR: blue'>configure --help</b> Options that enable extensions and SAPI will accept 'yes' or 'no' as a parameter. They also accept 'shared' as a synonym for 'yes' and request a shared build of that module. Not all modules can be built as shared modules; configure will display [shared] after the module name if can be built that way. ... [생략] ... --with-apache-libs Where to find Apache 1.3 libraries --enable-apache2filter Build Apache 2.x filter --enable-apache2-2filter Build Apache 2.2.x filter <b style='COLOR: blue'>--enable-apache2handler Build Apache 2.x handler</b> <b style='COLOR: blue'>--enable-apache2-2handler Build Apache 2.2.x handler</b> --with-apache-hooks Build Apache 1.3.x (hooks) version of PHP --disable-cgi Build CGI version of PHP --disable-cli Build CLI version of PHP ... [생략] ... </pre> <br /> 그럴 듯하지요. ^^ 그래서, 이번에는 PHP 소스 코드 빌드 과정에서 위의 옵션을 포함해 보았습니다. (<a target='tab' href='http://www.sysnet.pe.kr/2/0/1042'>지난번에 이미 한번 빌드를 해서 환경 구축</a>을 했으므로 그 부분은 생략합니다.)<br /> <br /> 먼저, "시작" / "Microsoft Windows SDKK v6.1" / "CMD Shell"을 실행시키고, 다음과 같이 일련의 명령을 내려 보았는데,<br /> <br /> <pre style='margin: 10px 0px 10px 10px; padding: 10px 0px 10px 10px; width: 800px; background-color: #fbedbb; overflow-x: scroll; font-family: Consolas, Verdana;' > D:\Program Files\Microsoft SDKs\Windows\v6.1><b style='COLOR: blue'>setenv /x86 /xp /release</b> D:\Program Files\Microsoft SDKs\Windows\v6.1><b style='COLOR: blue'>cd /d d:\phpbuild</b> d:\phpbuild><b style='COLOR: blue'>.\bin\phpsdk_setvars.bat</b> d:\phpbuild>REM phpsdk.bat d:\phpbuild><b style='COLOR: blue'>cd D:\phpbuild\php53dev\vc9\x86\php-5.3.6</b> D:\phpbuild\php53dev\vc9\x86\php-5.3.6><b style='COLOR: blue'>configure --disable-all --enable-cli --enable-apache2-2handler --enable-apache2handler</b> Saving configure options to config.nice.bat Checking for cl.exe ... <in default path> Detected compiler MSVC9 (Visual C++ 2008) Detected 32-bit compiler ...[생략]... ------------------------------------------- | | | ------------------------------------------- | Build type | Release | | Thread Safety | Yes | | Compiler | MSVC9 (Visual C++ 2008) | | Architecture | x86 | ------------------------------------------- Type 'nmake' to build PHP D:\phpbuild\php53dev\vc9\x86\php-5.3.6><b style='COLOR: blue'>nmake</b> Microsoft (R) Program Maintenance Utility Version 9.00.30729.01 Copyright (C) Microsoft Corporation. All rights reserved. Recreating build dirs Creating library Release_TS\php5ts.lib and object Release_TS\php5ts.exp SAPI sapi\cli build complete </pre> <br /> 결과는 실망스럽게도 아파치 핸들러 관련 DLL이 전혀 생성되지 않았습니다. 혹시 위의 글을 보고, 뭐가 잘못되었는지 아시는 분이 계시다면 댓글 부탁드립니다. ^^<br /> <br /> <hr style='width: 50%' /><br /> <br /> PHP 빌드 중에 자연스럽게 php5apache2.dll / php5apache2_2.dll 파일을 생성하는 것은 일단 포기하고, D:\phpbuild\php53dev\vc9\x86\php-5.3.6\sapi\apache2handler 폴더에 있는 파일들을 가지고 직접 빌드하기로 했습니다. 보니까, makefile은 없고 php5apache2.dsp 파일이 보여서 Visual Studio 2008로 열었습니다.<br /> <br /> 로드한 후, 빌드 설정을 다음과 같이 "Debug_TS"에서 "Release_TS"로 변경해 줍니다. 만약, "Debug_TS"로 빌드하고 싶다면 <a target='tab' href='http://www.sysnet.pe.kr/2/0/1042'>기존 PHP 모듈을 Debug 모드로 빌드</a>해 두는 것이 좋습니다. (그래야만 Debug_TS에서의 모든 경로 설정이 자연스럽게 연결됩니다.)<br /> <br /> <img alt='how_to_build_phphandler_1.png' src='/SysWebRes/bbs/how_to_build_phphandler_1.png' /><br /> <br /> 곧바로 빌드하면 httpd.h/apr_strings.h 헤더 파일들을 찾을 수 없다고 나옵니다.<br /> <br /> <pre style='margin: 10px 0px 10px 10px; padding: 10px 0px 10px 10px; width: 800px; background-color: #fbedbb; overflow-x: scroll; font-family: Consolas, Verdana;' > fatal error C1083: Cannot open include file: '<b style='COLOR: blue'>httpd.h</b>': No such file or directory d:\phpbuild\php53dev\vc9\x86\php-5.3.6\sapi\apache2handler2\php_apache.h fatal error C1083: Cannot open include file: '<b style='COLOR: blue'>apr_strings.h</b>': No such file or directory d:\phpbuild\php53dev\vc9\x86\php-5.3.6\sapi\apache2handler2\php_functions.c </pre> <br /> 옛날 같으면 당황했겠지만 ^^ <a target='tab' href='http://www.sysnet.pe.kr/2/0/1048'>아파치 소스 코드를 빌드해 본 경험으로</a> 어디서 찾을 수 있을지 바로 감이 왔습니다. 즉, 아파치 빌드 결과물의 include 폴더에 모두 있는 파일이었으므로 (lib 폴더와 함께) 다음과 같이 복사했습니다.<br /> <br /> <img alt='how_to_build_phphandler_2.png' src='/SysWebRes/bbs/how_to_build_phphandler_2.png' /><br /> <br /> <pre style='margin: 10px 0px 10px 10px; padding: 10px 0px 10px 10px; width: 800px; background-color: #fbedbb; overflow-x: scroll; font-family: Consolas, Verdana;' > D:\httpd_build\Apache22\include ==> D:\phpbuild\php53dev\vc9\x86\deps\include\apache2_2 D:\httpd_build\Apache22\lib ==> D:\phpbuild\php53dev\vc9\x86\deps\lib\apache2_2 </pre> <br /> 이제, Visual Studio에 헤더 및 라이브러리 파일을 찾는 경로를 알려주면 되겠군요. ^^<br /> <br /> [그림: Include 경로 포함 - "Configuration Properties" / "C/C++" / "Additional Include Directories"]<br /> <img alt='how_to_build_phphandler_3.png' src='/SysWebRes/bbs/how_to_build_phphandler_3.png' /><br /> <br /> [그림: Lib 경로 포함 - "Configuration Properties" / "Linker" / "Additional Library Directories"]<br /> <img alt='how_to_build_phphandler_4.png' src='/SysWebRes/bbs/how_to_build_phphandler_4.png' /><br /> <br /> 이렇게 설정하고 다시 빌드하면 다음과 같은 오류가 발생합니다.<br /> <br /> <pre style='margin: 10px 0px 10px 10px; padding: 10px 0px 10px 10px; width: 800px; background-color: #fbedbb; overflow-x: scroll; font-family: Consolas, Verdana;' > fatal error LNK1181: cannot open input file 'libapr.lib' </pre> <br /> 이상하군요. 아파치 빌드 결과물에는 libapr-1.lib 파일명으로 되어 있는 것을 왜 PHP 핸들러에서는 libarp.lib로 찾는지 모르겠습니다. 일단은, 프로젝트 속성 창에서 이름을 다음과 같이 바꿔주었습니다.<br /> <br /> [그림: Lib 파일명 변경 - "Configuration Properties" / "Linker" / "Input" / "Additional Dependencies"]<br /> <img alt='how_to_build_phphandler_5.png' src='/SysWebRes/bbs/how_to_build_phphandler_5.png' /><br /> <br /> 이제 빌드하고 나면 정상적으로 "D:\phpbuild\php53dev\vc9\x86\php-5.3.6\Release_TS" 폴더에 php5apache2.dll 파일명으로 DLL이 생성됩니다. 기왕 하는 김에 이름도 바꿔주면 좋겠군요. 프로젝트 속성창에서 "Linker" / "Output File" 설정을 ".\..\..\Release_TS/php5apache2_2.dll"으로 바꿔 주고 다시 빌드를 했습니다.<br /> <br /> 마침내, "D:\phpbuild\php53dev\vc9\x86\php-5.3.6\Release_TS" 폴더(디버그인 경우 Debug_TS)에 php5apache2_2.dll 파일을 얻어내는 데 성공했습니다. 휴~~~ 힘들군요. ^^;<br /> <br /> <hr style='width: 50%' /><br /> <br /> 물론 테스트를 해봐야겠지요. ^^<br /> <br /> <a target='tab' href='http://www.sysnet.pe.kr/2/0/1048'>기존에 아파치 소스 코드로부터 빌드했던 테스트 폴더</a>인 "D:\httpd_build\Apache22" 하위에 "phpmodule"을 두고 <a target='tab' href='http://www.sysnet.pe.kr/2/0/1042'>이전에 빌드한 php</a> 및 이 글을 통해 새롭게 얻은 php5apache2_2.dll 파일을 복사하고 테스트용 PHP를 방문하면 정상적으로 로드되는 것을 확인할 수 있습니다.<br /> <br /> 디버깅까지 한번 해볼까요? ^^<br /> <br /> 역시, <a target='tab' href='http://www.sysnet.pe.kr/2/0/1053'>아파치 모듈 디버깅을 했던 것</a>처럼, 이제 프로젝트의 php5apache2_2.dll 빌드 결과물에 대한 출력 폴더를 "D:\httpd_build\Apache22\phpmodule"로 설정하고 디버깅 프로세스를 httpd.exe로 지정합니다.<br /> <br /> 참고로, 디버그 모드로 php5apache2_2.dll을 빌드하는 경우, 의존성이 다음과 같습니다.<br /> <br /> <ul> <li>PHP5TS_DEBUG.DLL</li> <li>LIBHTTPD.DLL</li> <li>LIBAPR-1.DLL</li> <li>LIBAPRUTIL-1.DLL</li> </ul> <br /> "D:\httpd_build\Apache22\bin" 폴더에 해당 DLL들이 모두 있는지 확인한 다음 "Shift + F2" 키를 누르면 정상적으로 디버깅이 진행되는 것을 확인할 수 있습니다.<br /> <br /> PHP 페이지를 웹 브라우저를 통해서 방문할 때마다 실행되는 메서드 - 즉 핸들러를 찾아볼까요?<br /> <br /> <a target='tab' href='http://www.sysnet.pe.kr/2/0/1056'>Apache Module 컴파일 글</a>을 읽으셨다면 어렵지 않습니다. mod_php5.c에 있는 php5_module을 통해서,<br /> <br /> <pre style='margin: 10px 0px 10px 10px; padding: 10px 0px 10px 10px; width: 800px; background-color: #fbedbb; overflow-x: scroll; font-family: Consolas, Verdana;' > AP_MODULE_DECLARE_DATA module <b style='COLOR: blue'>php5_module</b> = { STANDARD20_MODULE_STUFF, create_php_config, /* create per-directory config structure */ merge_php_config, /* merge per-directory config structures */ NULL, /* create per-server config structure */ NULL, /* merge per-server config structures */ php_dir_cmds, /* command apr_table_t */ <b style='COLOR: blue'>php_ap2_register_hook</b> /* register hooks */ }; </pre> <br /> php_ap2_register_hook 함수에 handler가 지정될 것임을 알 수 있고, 이어서 sapi_apache2.c 파일의 php_ap2_register_hook 함수로부터,<br /> <br /> <pre style='margin: 10px 0px 10px 10px; padding: 10px 0px 10px 10px; width: 800px; background-color: #fbedbb; overflow-x: scroll; font-family: Consolas, Verdana;' > void <b style='COLOR: blue'>php_ap2_register_hook</b>(apr_pool_t *p) { ap_hook_pre_config(php_pre_config, NULL, NULL, APR_HOOK_MIDDLE); ap_hook_post_config(php_apache_server_startup, NULL, NULL, APR_HOOK_MIDDLE); ap_hook_handler(<b style='COLOR: blue'>php_handler</b>, NULL, NULL, APR_HOOK_MIDDLE); ap_hook_child_init(php_apache_child_init, NULL, NULL, APR_HOOK_MIDDLE); } </pre> <br /> ap_hook_handler 함수 호출을 통해 php_handler가 우리가 찾던 PHP 웹 페이지 방문 시 실행되는 함수임을 알 수 있고, BP(BreakPoint)를 걸어서 확인할 수 있습니다. ^^<br /> <br /> <img alt='how_to_build_phphandler_6.png' src='/SysWebRes/bbs/how_to_build_phphandler_6.png' /><br /> <br /> <a target='tab' href='http://www.sysnet.pe.kr/bbs/DownloadAttachment.aspx?fid=600&boardid=331301885'>첨부한 파일은 위와 같은 과정을 거쳐 생성된 프로젝트</a>를 포함합니다.<br /> <br /><br /><hr /><span style='color: Maroon'>[이 글에 대해서 여러분들과 의견을 공유하고 싶습니다. 틀리거나 미흡한 부분 또는 의문 사항이 있으시면 언제든 댓글 남겨주십시오.]</span> </div>
첨부파일
스팸 방지용 인증 번호
3835
(왼쪽의 숫자를 입력해야 합니다.)