객체 소개
xhprof: PHP 성능 분석 도구, Facebook에서 개발한 PHP 확장으로, PHP 스크립트 실행 중 함수/클래스 메서드 호출 체인은 물론 각 호출에 소요되는 시간과 메모리 사용량 및 기타 데이터
snappy: Google에서 개발한 문자열 압축/압축 해제 도구의 장점은 빠른 압축 속도입니다.
목표
xhprof 확장에 두 개의 PHP 함수 xhprof_compress($str) 및 xhprof_uncompress($str)를 제공하여 압축 및 압축 해제 기능을 구현합니다.
특정 프로세스
1. xhprof 소스 코드를 php-5.6.24/ext/xhprof
에 다운로드합니다. 2. 일반적인 방법을 따릅니다./configure, make, xhprof.so가 작동하는지 확인합니다. 적절하게
3. Snappy 소스 코드를 php-5.6.24/ext/xhprof/snappy
에 다운로드합니다. php-5.6.24/ext/xhprof/php_xhprof.h 파일을 편집하고 다음 줄을 추가합니다. 🎜>
PHP_FUNCTION(xhprof_compress);
PHP_FUNCTION(xhprof_uncompress);
...
... #include "snappy/snappy-c.h" ... zend_function_entry xhprof_functions[] = { ... PHP_FE(xhprof_compress, NULL) PHP_FE(xhprof_uncompress, NULL) ... }; PHP_FUNCTION(xhprof_compress) { zval *data; char *output; size_t output_len; if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "z", &data) == FAILURE) { RETURN_FALSE; } if (Z_TYPE_P(data) != IS_STRING) { zend_error(E_WARNING, "snappy_compress : expects parameter to be string."); RETURN_FALSE; } output_len = snappy_max_compressed_length(Z_STRLEN_P(data)); output = (char *)emalloc(output_len); if (!output) { zend_error(E_WARNING, "snappy_compress : memory error"); RETURN_FALSE; } if (snappy_compress(Z_STRVAL_P(data), Z_STRLEN_P(data), output, &output_len) == SNAPPY_OK) { #if ZEND_MODULE_API_NO >= 20141001 RETVAL_STRINGL(output, output_len); #else RETVAL_STRINGL(output, output_len, 1); #endif } else { RETVAL_FALSE; } efree(output); } PHP_FUNCTION(xhprof_uncompress) { zval *data; char *output = NULL; size_t output_len; if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "z", &data) == FAILURE) { RETURN_FALSE; } if (Z_TYPE_P(data) != IS_STRING) { zend_error(E_WARNING, "snappy_uncompress : expects parameter to be string."); RETURN_FALSE; } if (snappy_uncompressed_length(Z_STRVAL_P(data), Z_STRLEN_P(data), &output_len) != SNAPPY_OK) { zend_error(E_WARNING, "snappy_uncompress : output length error"); RETURN_FALSE; } output = (char *)emalloc(output_len); if (!output) { zend_error(E_WARNING, "snappy_uncompress : memory error"); RETURN_FALSE; } if (snappy_uncompress(Z_STRVAL_P(data), Z_STRLEN_P(data), output, &output_len) == SNAPPY_OK) { #if ZEND_MODULE_API_NO >= 20141001 RETVAL_STRINGL(output, output_len); #else RETVAL_STRINGL(output, output_len, 1); #endif } else { zend_error(E_WARNING, "snappy_uncompress : data error"); RETVAL_FALSE; } efree(output); } ...위 코드는 원본이 아니며 snappy의 PHP 확장: php-ext-snappy에서 가져왔습니다.
6. snappy의 컴파일 방법 지정
config.m4 코드 완성 다음과 같이 snappy를 xhprof.so에서 직접 편집합니다. 더 좋은 방법은 먼저 현재 호스트에 snappy가 설치되어 있는지 확인하는 것입니다. 그렇다면 링크 라이브러리 경로를 지정하면 snappy 관련 기능을 xhprof.so에 컴파일할 필요가 없습니다.
PHP_ARG_ENABLE(xhprof, whether to enable xhprof support, [ --enable-xhprof Enable xhprof support]) dnl compiler C++: PHP_REQUIRE_CXX() dnl snappy SNAPPY_MAJOR="1" SNAPPY_MINOR="1" SNAPPY_PATCHLEVEL="3" AC_PROG_CXX AC_LANG([C++]) AC_C_BIGENDIAN AC_CHECK_HEADERS([stdint.h stddef.h sys/mman.h sys/resource.h windows.h byteswap.h sys/byteswap. h sys/endian.h sys/time.h]) AC_CHECK_FUNC([mmap]) AC_MSG_CHECKING([if the compiler supports __builtin_expect]) AC_TRY_COMPILE(, [ return __builtin_expect(1, 1) ? 1 : 0 ], [ snappy_have_builtin_expect=yes AC_MSG_RESULT([yes]) ], [ snappy_have_builtin_expect=no AC_MSG_RESULT([no]) ]) if test x$snappy_have_builtin_expect = xyes ; then AC_DEFINE([HAVE_BUILTIN_EXPECT], [1], [Define to 1 if the compiler supports __builtin_expect.]) fi AC_MSG_CHECKING([if the compiler supports __builtin_ctzll]) AC_TRY_COMPILE(, [ return (__builtin_ctzll(0x100000000LL) == 32) ? 1 : 0 ], [ snappy_have_builtin_ctz=yes AC_MSG_RESULT([yes]) ], [ snappy_have_builtin_ctz=no AC_MSG_RESULT([no]) ]) if test x$snappy_have_builtin_ctz = xyes ; then AC_DEFINE([HAVE_BUILTIN_CTZ], [1], [Define to 1 if the compiler supports __builtin_ctz and friends.]) fi if test "$ac_cv_header_stdint_h" = "yes"; then AC_SUBST([ac_cv_have_stdint_h], [1]) else AC_SUBST([ac_cv_have_stdint_h], [0]) fi if test "$ac_cv_header_stddef_h" = "yes"; then AC_SUBST([ac_cv_have_stddef_h], [1]) else AC_SUBST([ac_cv_have_stddef_h], [0]) fi if test "$ac_cv_header_sys_uio_h" = "yes"; then AC_SUBST([ac_cv_have_sys_uio_h], [1]) else AC_SUBST([ac_cv_have_sys_uio_h], [0]) fi AC_SUBST([SNAPPY_MAJOR]) AC_SUBST([SNAPPY_MINOR]) AC_SUBST([SNAPPY_PATCHLEVEL]) AC_CONFIG_FILES([snappy/snappy-stubs-public.h]) AC_OUTPUT dnl Check for stdc++ LIBNAME=stdc++ AC_MSG_CHECKING([for stdc++]) AC_LANG_SAVE AC_LANG_CPLUSPLUS AC_TRY_COMPILE( [ #include <string> using namespace std; ],[ string dummy; ],[ AC_MSG_RESULT(yes) PHP_ADD_LIBRARY($LIBNAME, , XHPROF_SHARED_LIBADD) ],[ AC_MSG_ERROR([wrong stdc++ library not found]) ]) AC_LANG_RESTORE PHP_SUBST(XHPROF_SHARED_LIBADD) dnl Sources SNAPPY_SOURCES="snappy/snappy-c.cc snappy/snappy.cc snappy/snappy-stubs-internal.cc snappy/ snappy-sinksource.cc" if test "$PHP_XHPROF" != "no"; then PHP_NEW_EXTENSION(xhprof, xhprof.c $SNAPPY_SOURCES, $ext_shared) fi위 코드는 원본이 아니며 snappy의 PHP 확장: php-ext-snappy에서 가져온 것입니다
7. phpize를 다시 실행한 후 make를 실행하여 새로운 xhprof.so를 생성합니다
8. nm xhprof.so 명령을 실행하여 snappy 관련 함수가 성공적으로 컴파일되었는지 확인합니다. 아래와 같이 _snappy_compress, _snappy_uncompress 등의 메모리 주소가 모두 비어 있지 않아 컴파일되었음을 나타냅니다. xhprof.so.
➜ xhprof nm modules/xhprof.so 0000000000008a98 s GCC_except_table10 0000000000008ad8 s GCC_except_table11 0000000000008b24 s GCC_except_table12 0000000000008b70 s GCC_except_table13 0000000000008bb0 s GCC_except_table14 0000000000008bfc s GCC_except_table16 0000000000008c48 s GCC_except_table18 0000000000008cbc s GCC_except_table19 0000000000008d4c s GCC_except_table23 00000000000088f4 s GCC_except_table5 0000000000008934 s GCC_except_table7 0000000000008a0c s GCC_except_table8 0000000000008a4c s GCC_except_table9 ... 0000000000002510 T _restore_cpu_affinity 0000000000003f00 T _snappy_compress 0000000000003f60 T _snappy_max_compressed_length 0000000000003f70 T _snappy_uncompress 0000000000003fe0 T _snappy_uncompressed_length 0000000000004000 T _snappy_validate_compressed_buffer ... U _zend_register_long_constant U _zend_unregister_ini_entries 00000000000013a0 T _zif_xhprof_compress 0000000000001090 T _zif_xhprof_disable 0000000000001350 T _zif_xhprof_dump 0000000000001040 T _zif_xhprof_enable 0000000000001200 T _zif_xhprof_sample_disable 00000000000011e0 T _zif_xhprof_sample_enable 0000000000001470 T _zif_xhprof_uncompress 0000000000001700 T _zm_activate_xhprof 0000000000001720 T _zm_deactivate_xhprof 0000000000001960 T _zm_info_xhprof 0000000000001660 T _zm_shutdown_xhprof 0000000000001560 T _zm_startup_xhprof 0000000000001f00 t _zval_dump U _zval_used_for_init9. PHP 스크립트에서는 xhprof_compress() 및 xhrof_uncompress()를 사용할 수 있습니다