成人国产在线小视频_日韩寡妇人妻调教在线播放_色成人www永久在线观看_2018国产精品久久_亚洲欧美高清在线30p_亚洲少妇综合一区_黄色在线播放国产_亚洲另类技巧小说校园_国产主播xx日韩_a级毛片在线免费

資訊專(zhuān)欄INFORMATION COLUMN

xhprof 源碼分析

2450184176 / 2369人閱讀

摘要:下面這四個(gè)重新替換封裝非常重要,具體的方法作用已經(jīng)在下面的代碼注釋中寫(xiě)明了。使輸出的性能數(shù)據(jù)中添加數(shù)據(jù)。我們這里不考慮和內(nèi)存,那么發(fā)現(xiàn)給每隔設(shè)置了一個(gè)的起始時(shí)鐘周期。

XHProf 簡(jiǎn)要概念

重新封裝zend的原生方法

如果要檢測(cè)CPU的話,會(huì)有5ms的延遲,因?yàn)樾枰?jì)算cpu頻率

內(nèi)部使用了鏈表

源碼地址:/root/Downloads/xhprof/extension/xhprof.c

最重要的兩個(gè)結(jié)構(gòu)體
/* Xhprof"s global state.
 *
 * This structure is instantiated once.  Initialize defaults for attributes in
 * 這個(gè)結(jié)構(gòu)體只初始化一次
 * hp_init_profiler_state() Cleanup/free attributes in
 * hp_clean_profiler_state() */
typedef struct hp_global_t {

  /*       ----------   Global attributes:  -----------       */

  /* Indicates if xhprof is currently enabled 是否當(dāng)前可用 */
  int              enabled;

  /* Indicates if xhprof was ever enabled during this request 在本次請(qǐng)求過(guò)程中是否其用過(guò)xhprof */
  int              ever_enabled;

  /* Holds all the xhprof statistics */
  zval            *stats_count;

  /* Indicates the current xhprof mode or level 當(dāng)前的運(yùn)行模式和等級(jí)*/
  int              profiler_level;

  /* Top of the profile stack 堆棧中的第一個(gè)*/
  hp_entry_t      *entries;

  /* freelist of hp_entry_t chunks for reuse... */
  hp_entry_t      *entry_free_list;

  /* Callbacks for various xhprof modes 代表不同模式的回調(diào)么?*/
  hp_mode_cb       mode_cb;

  /*       ----------   Mode specific attributes:  -----------       */

  /* Global to track the time of the last sample in time and ticks */
  struct timeval   last_sample_time;
  uint64           last_sample_tsc;
  /* XHPROF_SAMPLING_INTERVAL in ticks */
  uint64           sampling_interval_tsc;

  /* This array is used to store cpu frequencies for all available logical
   * cpus.  For now, we assume the cpu frequencies will not change for power
   * saving or other reasons. If we need to worry about that in the future, we
   * can use a periodical timer to re-calculate this arrary every once in a
   * while (for example, every 1 or 5 seconds). 處理器的執(zhí)行頻率?*/
  double *cpu_frequencies;

  /* The number of logical CPUs this machine has. 邏輯cpu的數(shù)量*/
  uint32 cpu_num;

  /* The saved cpu affinity. */
  cpu_set_t prev_mask;

  /* The cpu id current process is bound to. (default 0) 當(dāng)前進(jìn)程在的處理器的id*/
  uint32 cur_cpu_id;

  /* XHProf flags */
  uint32 xhprof_flags;

  /* counter table indexed by hash value of function names.  方法的調(diào)用次數(shù)的表*/
  uint8  func_hash_counters[256];

  /* Table of ignored function names and their filter 忽略統(tǒng)計(jì)的方法的表格*/
  char  **ignored_function_names;
  uint8   ignored_function_filter[XHPROF_IGNORED_FUNCTION_FILTER_SIZE];

} hp_global_t;
typedef struct hp_entry_t {
  char                   *name_hprof;                       /* function name 方法名稱(chēng)*/
  int                     rlvl_hprof;        /* recursion level for function 方法的遞歸層級(jí)*/
  uint64                  tsc_start;         /* start value for TSC counter  開(kāi)始的時(shí)鐘周期*/
  long int                mu_start_hprof;                    /* memory usage 內(nèi)存使用量*/
  long int                pmu_start_hprof;              /* peak memory usage 內(nèi)存使用峰值*/
  struct rusage           ru_start_hprof;             /* user/sys time start */
  struct hp_entry_t      *prev_hprof;    /* ptr to prev entry being profiled 指向上一個(gè)被分析的指針*/
  uint8                   hash_code;     /* hash_code for the function name  每個(gè)方法名稱(chēng)對(duì)應(yīng)的hash*/
} hp_entry_t;
XHProf 在php中的使用

我們先看下XHProf的使用方法

save_run($data,"test");
// 我這里直接將可視化的鏈接地址打印了出來(lái),方便調(diào)試
echo "

執(zhí)行結(jié)果如下:(可以直接跳過(guò)結(jié)果,看下面,但是要記住有ct、wt這兩個(gè)值)

array(7) {
  ["test==>range"]=>
  array(2) {
    ["ct"]=>
    int(2)
    ["wt"]=>
    int(4463)
  }
  ["main()==>test"]=>
  array(2) {
    ["ct"]=>
    int(1)
    ["wt"]=>
    int(3069)
  }
  ["main()==>eval::/var/www/html/index2.php(9) : eval()"d code"]=>
  array(2) {
    ["ct"]=>
    int(1)
    ["wt"]=>
    int(16)
  }
  ["eval==>test"]=>
  array(2) {
    ["ct"]=>
    int(1)
    ["wt"]=>
    int(2614)
  }
  ["main()==>eval"]=>
  array(2) {
    ["ct"]=>
    int(1)
    ["wt"]=>
    int(2617)
  }
  ["main()==>xhprof_disable"]=>
  array(2) {
    ["ct"]=>
    int(1)
    ["wt"]=>
    int(0)
  }
  ["main()"]=>
  array(2) {
    ["ct"]=>
    int(1)
    ["wt"]=>
    int(5716)
  }
}
XHProf 源碼 xhprof_enable()

首先我們來(lái)看xhprof_enable(),這個(gè)方法定義了要接受的三個(gè)參數(shù),并且將這三個(gè)參數(shù)分別傳遞給兩個(gè)方法使用,其中最重要的是hp_begin()

/**
 * Start XHProf profiling in hierarchical mode.
 *
 * @param  long $flags  flags for hierarchical mode
 * @return void
 * @author kannan
 */
PHP_FUNCTION(xhprof_enable) {
  long  xhprof_flags = 0;                                    /* XHProf flags */
  zval *optional_array = NULL;         /* optional array arg: for future use */

  /* 
    獲取參數(shù)并且允許傳遞一個(gè)l 和z的可選參數(shù)  分別代表xhprof_flags 和 optional_array
    關(guān)于TSRMLS_CC 可以看http://www.laruence.com/2008/08/03/201.html
    另外關(guān)于zend_parse_parameters的返回值failure 代表參數(shù)的處理是否成功
   */
  if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC,
                            "|lz", &xhprof_flags, &optional_array) == FAILURE) {
    return;
  }

  /*
    從參數(shù)中獲取需要被忽略的方法
    參照手冊(cè)參數(shù)的說(shuō)明 http://php.net/manual/zh/function.xhprof-enable.php
   */
  hp_get_ignored_functions_from_arg(optional_array);


  hp_begin(XHPROF_MODE_HIERARCHICAL, xhprof_flags TSRMLS_CC);
}
hp_begin()

這個(gè)方法看起來(lái)很長(zhǎng),但是世界上邏輯很簡(jiǎn)單,主要是進(jìn)行了一些初始化。
下面一共進(jìn)行了四次replace,用來(lái)封裝zend的方法。

下面這四個(gè)重新替換封裝非常重要,具體的方法作用已經(jīng)在下面的代碼注釋中寫(xiě)明了。

zend_compile_file => hp_compile_file

zend_compile_string => hp_compile_string

zend_execute => hp_execute

zend_execute_internal => hp_execute_internal

/**
 * This function gets called once when xhprof gets enabled.
 * 這個(gè)方法在enable的時(shí)候調(diào)用一次
 * It replaces all the functions like zend_execute, zend_execute_internal,
 * etc that needs to be instrumented with their corresponding proxies.
 * 他用來(lái)替換zend的一些需要被代理的方法意思就是xhprof劫持了原生方法
 * hp_begin(XHPROF_MODE_HIERARCHICAL, xhprof_flags TSRMLS_CC);
 *
 * level 等級(jí)
 * xhprof_flags 運(yùn)行方式
 */
static void hp_begin(long level, long xhprof_flags TSRMLS_DC) {
  /*
    如果xhprof 沒(méi)有開(kāi)啟,也就是沒(méi)有調(diào)用enable方法,那么走這里買(mǎi)的邏輯,這個(gè)是通過(guò)hp_globals來(lái)判斷的,
   */
  if (!hp_globals.enabled) {
    int hp_profile_flag = 1;

    hp_globals.enabled      = 1; /* 這里修改了enbale狀態(tài),保證enable在整個(gè)請(qǐng)求過(guò)程中只會(huì)被第一次調(diào)用觸發(fā) */
    hp_globals.xhprof_flags = (uint32)xhprof_flags; /* 格式化為32位的無(wú)符號(hào)整數(shù) */

    /*
      下面一共進(jìn)行了四次replace,用來(lái)封裝zend的方法
      1. zend_compile_file => hp_compile_file
      zend_compile_file負(fù)責(zé)將要執(zhí)行的腳本文件編譯成由ZE的基本指令序列構(gòu)成的op codes , 然后將op codes交由zend_execute執(zhí)行,從而得到我們腳本的結(jié)果。
      http://www.laruence.com/2008/08/14/250.html

      2. zend_compile_string => hp_compile_string
      這個(gè)是把php代碼編譯成為opcode的過(guò)程
      http://www.phpchina.com/portal.php?mod=view&aid=40347

      3. zend_execute => hp_execute
      zend_compile_file() zend_compile_file() is the wrapper for the lexer, parser, and code generator. It compiles a file and returns a zend_op_array.
      zend_execute() After a file is compiled, its zend_op_array is executed by zend_execute(). 
      http://php.find-info.ru/php/016/ch23lev1sec2.html

      4. zend_execute_internal => hp_execute_internal
      There is also a companion zend_execute_internal() function, which executes internal functions.

     */

    /* Replace zend_compile with our proxy 先對(duì)其進(jìn)行了備份_,通過(guò)加入_下劃線的方式,然后使用hp_compile_file來(lái)替換*/
    _zend_compile_file = zend_compile_file;
    zend_compile_file  = hp_compile_file;

    /* Replace zend_compile_string with our proxy */
    _zend_compile_string = zend_compile_string;
    zend_compile_string = hp_compile_string;

    /* Replace zend_execute with our proxy */
#if PHP_VERSION_ID < 50500
    _zend_execute = zend_execute;
    zend_execute  = hp_execute;
#else
    _zend_execute_ex = zend_execute_ex;
    zend_execute_ex  = hp_execute_ex;
#endif

    /* Replace zend_execute_internal with our proxy */
    _zend_execute_internal = zend_execute_internal;
    /* 
      XHPROF_FLAGS_NO_BUILTINGS 是用來(lái)標(biāo)識(shí),不需要統(tǒng)計(jì)內(nèi)置函數(shù)性能 
      通過(guò)位運(yùn)算&來(lái)判斷是否用戶(hù)傳遞的flags包含了NO_BUILTINGS

      除此之外還包含一下三種flags
      1. HPROF_FLAGS_NO_BUILTINS (integer) 使得跳過(guò)所有內(nèi)置(內(nèi)部)函數(shù)。
      2. XHPROF_FLAGS_CPU (integer) 使輸出的性能數(shù)據(jù)中添加 CPU 數(shù)據(jù)。
      3. XHPROF_FLAGS_MEMORY (integer) 使輸出的性能數(shù)據(jù)中添加內(nèi)存數(shù)據(jù)。
    */
    if (!(hp_globals.xhprof_flags & XHPROF_FLAGS_NO_BUILTINS)) {
      /* if NO_BUILTINS is not set (i.e. user wants to profile builtins),
       * then we intercept internal (builtin) function calls.
       * 如果沒(méi)有設(shè)置的話,那么就代表用戶(hù)想分析內(nèi)置函數(shù)性能,并且我們就會(huì)攔截內(nèi)置的方法請(qǐng)求
       */
      zend_execute_internal = hp_execute_internal;
    }

    /* Initialize with the dummy mode first Having these dummy callbacks saves
     * us from checking if any of the callbacks are NULL everywhere. 
     * 首先來(lái)初始化一下這些方法,可以避免在回調(diào)方法為NULL的時(shí)候*/
    hp_globals.mode_cb.init_cb     = hp_mode_dummy_init_cb;
    hp_globals.mode_cb.exit_cb     = hp_mode_dummy_exit_cb;
    hp_globals.mode_cb.begin_fn_cb = hp_mode_dummy_beginfn_cb;
    hp_globals.mode_cb.end_fn_cb   = hp_mode_dummy_endfn_cb;

    /* Register the appropriate callback functions Override just a subset of
     * all the callbacks is OK. 根據(jù)不同的處理模式,簡(jiǎn)單還是詳細(xì)*/
    switch(level) {
      /* 一般都是使用的這個(gè)模式,所以我們專(zhuān)注看這個(gè)mode */
      case XHPROF_MODE_HIERARCHICAL:
        hp_globals.mode_cb.begin_fn_cb = hp_mode_hier_beginfn_cb;
        hp_globals.mode_cb.end_fn_cb   = hp_mode_hier_endfn_cb;
        break;
      case XHPROF_MODE_SAMPLED:
        hp_globals.mode_cb.init_cb     = hp_mode_sampled_init_cb;
        hp_globals.mode_cb.begin_fn_cb = hp_mode_sampled_beginfn_cb;
        hp_globals.mode_cb.end_fn_cb   = hp_mode_sampled_endfn_cb;
        break;
    }

    /* one time initializations 初始化分析器,內(nèi)部搞定了cpu頻率、initcb、可忽略的方法*/
    hp_init_profiler_state(level TSRMLS_CC);

    /* start profiling from fictitious main() */
    BEGIN_PROFILING(&hp_globals.entries, ROOT_SYMBOL, hp_profile_flag);
  }
}
hp_init_profiler_state()
/**
 * Initialize profiler state
 * 初始化分析器狀態(tài)
 *
 * 這里最開(kāi)始的時(shí)候傳遞進(jìn)來(lái)的level是XHPROF_MODE_HIERARCHICAL
 * 
 * @author kannan, veeve
 */
void hp_init_profiler_state(int level TSRMLS_DC) {
  /* Setup globals */
  if (!hp_globals.ever_enabled) {
    /* 如果之前沒(méi)有開(kāi)啟過(guò)xhprof,那么將這個(gè)值初始化為1,現(xiàn)在就算開(kāi)啟了 */
    hp_globals.ever_enabled  = 1;
    /* 堆棧的第一個(gè)設(shè)置空 */
    hp_globals.entries = NULL;
  }
  /* 分析器的等級(jí) */
  hp_globals.profiler_level  = (int) level;

  /* Init stats_count 初始化統(tǒng)計(jì)數(shù)量 */
  if (hp_globals.stats_count) {
    /* 釋放這個(gè)內(nèi)存 */
    zval_dtor(hp_globals.stats_count);
    /* 通知垃圾回收機(jī)制來(lái)回收這個(gè)內(nèi)存 */
    FREE_ZVAL(hp_globals.stats_count);
  }
  /* 創(chuàng)建一個(gè)zval變量,并且初始化為數(shù)組 參考 http://www.cunmou.com/phpbook/8.3.md */
  MAKE_STD_ZVAL(hp_globals.stats_count);
  array_init(hp_globals.stats_count);

  /* NOTE(cjiang): some fields such as cpu_frequencies take relatively longer
   * to initialize, (5 milisecond per logical cpu right now), therefore we
   * calculate them lazily. 一些字段初始化起來(lái)要花費(fèi)非常長(zhǎng)的時(shí)間,那么我們要懶計(jì)算,就是放到后面計(jì)算*/
  if (hp_globals.cpu_frequencies == NULL) {
    get_all_cpu_frequencies();
    restore_cpu_affinity(&hp_globals.prev_mask);
  }

  /* bind to a random cpu so that we can use rdtsc instruction. 這里竟然是隨機(jī)綁定一個(gè)cpu*/
  bind_to_cpu((int) (rand() % hp_globals.cpu_num));

  /* Call current mode"s init cb  根據(jù)不同的模式,調(diào)用初始方法,看line:1933*/
  hp_globals.mode_cb.init_cb(TSRMLS_C);

  /* Set up filter of functions which may be ignored during profiling 設(shè)置被過(guò)濾的方法*/
  hp_ignored_functions_filter_init();
}
get_cpu_frequency()

在上面的方法中調(diào)用了一個(gè)get_all_cpu_frequencies(),這個(gè)方法內(nèi)部調(diào)用了一個(gè)get_cpu_frequency很有意思,因?yàn)檫@個(gè)方法將導(dǎo)致如果開(kāi)啟CPU的檢測(cè),那么會(huì)有5ms的延遲

/**
 * This is a microbenchmark to get cpu frequency the process is running on. The
 * returned value is used to convert TSC counter values to microseconds.
 *
 * @return double.
 * @author cjiang
 */
static double get_cpu_frequency() {
  struct timeval start;
  struct timeval end;

  /* gettimeofday 獲取當(dāng)前的時(shí)間,并且放到start中 */
  if (gettimeofday(&start, 0)) {
    perror("gettimeofday");
    return 0.0;
  }
  uint64 tsc_start = cycle_timer();
  /* Sleep for 5 miliseconds. Comparaing with gettimeofday"s  few microseconds
   * execution time, this should be enough. 
   * 這個(gè)是為了獲取CPU的執(zhí)行頻率,用5000微秒的時(shí)間中cpu的執(zhí)行次數(shù),來(lái)得到每秒cpu能執(zhí)行的頻率
   * TSC 自從啟動(dòng)CPU開(kāi)始記錄的時(shí)鐘周期
   * */
  usleep(5000);
  if (gettimeofday(&end, 0)) {
    perror("gettimeofday");
    return 0.0;
  }
  uint64 tsc_end = cycle_timer();
  /* 時(shí)鐘周期的數(shù)量除以微秒時(shí)間間隔的數(shù)量得到cpu頻率 */
  return (tsc_end - tsc_start) * 1.0 / (get_us_interval(&start, &end));
}
BEGIN_PROFILING 重要!

這個(gè)就是分析的邏輯,他的要點(diǎn)在于生成了一個(gè)單項(xiàng)鏈表。

/*
 * Start profiling - called just before calling the actual function
 * 開(kāi)始分析,只在正式方法調(diào)用之前要調(diào)用
 * NOTE:  PLEASE MAKE SURE TSRMLS_CC IS AVAILABLE IN THE CONTEXT
 *        OF THE FUNCTION WHERE THIS MACRO IS CALLED.
 *        TSRMLS_CC CAN BE MADE AVAILABLE VIA TSRMLS_DC IN THE
 *        CALLING FUNCTION OR BY CALLING TSRMLS_FETCH()
 *        TSRMLS_FETCH() IS RELATIVELY EXPENSIVE.
 * entries 這里傳遞進(jìn)來(lái)的是hp_entry_t的一個(gè)指向指針的地址
 * 這個(gè)地方實(shí)際上生成的是一個(gè)單鏈表,都是用prev_hprof 來(lái)進(jìn)行關(guān)聯(lián)
 *
 * 這里do while(0) 是用來(lái)封裝宏的
 * 
 */
#define BEGIN_PROFILING(entries, symbol, profile_curr)                  
  do {                                                                  
    /* Use a hash code to filter most of the string comparisons. */     
    uint8 hash_code  = hp_inline_hash(symbol);                          
    /* 判斷這個(gè)方法是否是需要忽略的方法,如果不是需要被忽略的,那么進(jìn)行分析 */     
    profile_curr = !hp_ignore_entry(hash_code, symbol);                 
    if (profile_curr) {                                                 
      /* 返回一個(gè)指針(地址),開(kāi)辟了一個(gè)內(nèi)存空間給cur_entry,包括了hash_code、方法名稱(chēng)、堆棧指針 */     
      hp_entry_t *cur_entry = hp_fast_alloc_hprof_entry();              
      (cur_entry)->hash_code = hash_code;                               
      (cur_entry)->name_hprof = symbol;                                 
      /* 這里的*entries 指向的是指針hp_global_t.entires 堆棧的首地址  */     
      (cur_entry)->prev_hprof = (*(entries));                           
      /* Call the universal callback*/                                 
      hp_mode_common_beginfn((entries), (cur_entry) TSRMLS_CC);         
      /* Call the mode"s beginfn callback 這個(gè)方法除卻cpu和mem 只是設(shè)置了tsc_Start */                            
      hp_globals.mode_cb.begin_fn_cb((entries), (cur_entry) TSRMLS_CC); 
      /* Update entries linked list */                                  
      (*(entries)) = (cur_entry);                                       
    }                                                                   
  } while (0)

我們可以看上面的鏈表在生成的過(guò)程中,調(diào)用了 hp_globals.mode_cb.begin_fn_cb方法。我們這里不考慮CPU和內(nèi)存,那么發(fā)現(xiàn)給每隔current設(shè)置了一個(gè)tsc的起始時(shí)鐘周期。

/**
 * XHPROF_MODE_HIERARCHICAL"s begin function callback
 *
 * @author kannan
 */
void hp_mode_hier_beginfn_cb(hp_entry_t **entries,
                             hp_entry_t  *current  TSRMLS_DC) {
  /* Get start tsc counter */
  current->tsc_start = cycle_timer();

  /* Get CPU usage 如果要計(jì)算cpu的話*/
  if (hp_globals.xhprof_flags & XHPROF_FLAGS_CPU) {
    getrusage(RUSAGE_SELF, &(current->ru_start_hprof));
  }

  /* Get memory usage 如果要計(jì)算內(nèi)存的話*/
  if (hp_globals.xhprof_flags & XHPROF_FLAGS_MEMORY) {
    current->mu_start_hprof  = zend_memory_usage(0 TSRMLS_CC);
    current->pmu_start_hprof = zend_memory_peak_usage(0 TSRMLS_CC);
  }
}
hp_execute 代碼執(zhí)行部分

每次有代碼執(zhí)行的時(shí)候,都會(huì)走這個(gè)地方,這段代碼主要是在執(zhí)行zend_execute的前后,粉分別調(diào)用了BEGIN_PROFILINGEND_PROFILING

#if PHP_VERSION_ID < 50500
ZEND_DLEXPORT void hp_execute (zend_op_array *ops TSRMLS_DC) {
#else
ZEND_DLEXPORT void hp_execute_ex (zend_execute_data *execute_data TSRMLS_DC) {
  zend_op_array *ops = execute_data->op_array;
#endif
  char          *func = NULL;
  int hp_profile_flag = 1;

  func = hp_get_function_name(ops TSRMLS_CC);
  if (!func) {
#if PHP_VERSION_ID < 50500
    _zend_execute(ops TSRMLS_CC);
#else
    _zend_execute_ex(execute_data TSRMLS_CC);
#endif
    return;
  }

  BEGIN_PROFILING(&hp_globals.entries, func, hp_profile_flag);
#if PHP_VERSION_ID < 50500
  _zend_execute(ops TSRMLS_CC);
#else
  _zend_execute_ex(execute_data TSRMLS_CC);
#endif
  if (hp_globals.entries) {
    END_PROFILING(&hp_globals.entries, hp_profile_flag);
  }
  efree(func);
}
END_PROFILING
hp_globals.mode_cb.end_fn_cb((entries) TSRMLS_CC);                

這段代碼最終指向了hp_mode_hier_endfn_cb,這段代碼中主要構(gòu)成了一個(gè)"==>"數(shù)據(jù)格式,并且計(jì)算了每個(gè)方法的調(diào)用次數(shù)。

void hp_mode_hier_endfn_cb(hp_entry_t **entries  TSRMLS_DC) {
  /* 整個(gè)堆棧的最后一個(gè)調(diào)用 */
  hp_entry_t   *top = (*entries);
  zval            *counts;
  struct rusage    ru_end;
  char             symbol[SCRATCH_BUF_LEN];
  long int         mu_end;
  long int         pmu_end;

  /* Get the stat array */
  hp_get_function_stack(top, 2, symbol, sizeof(symbol));
  if (!(counts = hp_mode_shared_endfn_cb(top,
                                         symbol  TSRMLS_CC))) {
    return;
  }

  if (hp_globals.xhprof_flags & XHPROF_FLAGS_CPU) {
    /* Get CPU usage */
    getrusage(RUSAGE_SELF, &ru_end);

    /* Bump CPU stats in the counts hashtable */
    hp_inc_count(counts, "cpu", (get_us_interval(&(top->ru_start_hprof.ru_utime),
                                              &(ru_end.ru_utime)) +
                              get_us_interval(&(top->ru_start_hprof.ru_stime),
                                              &(ru_end.ru_stime)))
              TSRMLS_CC);
  }

  if (hp_globals.xhprof_flags & XHPROF_FLAGS_MEMORY) {
    /* Get Memory usage */
    mu_end  = zend_memory_usage(0 TSRMLS_CC);
    pmu_end = zend_memory_peak_usage(0 TSRMLS_CC);

    /* Bump Memory stats in the counts hashtable */
    hp_inc_count(counts, "mu",  mu_end - top->mu_start_hprof    TSRMLS_CC);
    hp_inc_count(counts, "pmu", pmu_end - top->pmu_start_hprof  TSRMLS_CC);
  }
}
hp_mode_shared_endfn_cb

這個(gè)方法統(tǒng)計(jì)了調(diào)用次數(shù)和消耗時(shí)間,實(shí)際上最終所有的數(shù)據(jù)都存儲(chǔ)在hp_entry_t所構(gòu)造的鏈表中

zval * hp_mode_shared_endfn_cb(hp_entry_t *top,
                               char          *symbol  TSRMLS_DC) {
  zval    *counts;
  uint64   tsc_end;

  /* Get end tsc counter */
  tsc_end = cycle_timer();

  /* Get the stat array */
  if (!(counts = hp_hash_lookup(symbol TSRMLS_CC))) {
    return (zval *) 0;
  }

  /* Bump stats in the counts hashtable */
  hp_inc_count(counts, "ct", 1  TSRMLS_CC);

  hp_inc_count(counts, "wt", get_us_from_tsc(tsc_end - top->tsc_start,
        hp_globals.cpu_frequencies[hp_globals.cur_cpu_id]) TSRMLS_CC);
  return counts;
}

文章版權(quán)歸作者所有,未經(jīng)允許請(qǐng)勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。

轉(zhuǎn)載請(qǐng)注明本文地址:http://systransis.cn/yun/23171.html

相關(guān)文章

  • 使用PHP擴(kuò)展Xhprof分析項(xiàng)目性能實(shí)踐

    摘要:一背景項(xiàng)目即將上線,想通過(guò)一些工具來(lái)分析代碼的穩(wěn)定性和效率,想起在上個(gè)團(tuán)隊(duì)時(shí)使用過(guò)的擴(kuò)展因?yàn)閾Q了新電腦,所以需要重新編譯此擴(kuò)展,現(xiàn)將安裝與實(shí)際排查過(guò)程完整記錄下來(lái),方便自己回顧和幫助更多的讀者。作者湯青松微信日期 一、背景 項(xiàng)目即將上線,想通過(guò)一些工具來(lái)分析代碼的穩(wěn)定性和效率,想起在上個(gè)團(tuán)隊(duì)時(shí)使用過(guò)的xhprof擴(kuò)展;因?yàn)閾Q了新電腦,所以需要重新編譯此擴(kuò)展,現(xiàn)將安裝與實(shí)際排查過(guò)程完整記...

    高勝山 評(píng)論0 收藏0
  • PHP 性能追蹤及分析工具(XHPROF

    摘要:什么是開(kāi)源的輕量級(jí)性能分析工具。它報(bào)告函數(shù)級(jí)別的請(qǐng)求次數(shù)和各種指標(biāo),包括阻塞時(shí)間,時(shí)間和內(nèi)存使用情況。基于瀏覽器的性能分析用戶(hù)界面能更容易查看,或是與同行們分享成果。對(duì)于本地開(kāi)發(fā)環(huán)境來(lái)說(shuō),進(jìn)行性能分析是夠用了。 什么是 XHPROF? XHPROF:Facebook 開(kāi)源的輕量級(jí)PHP性能分析工具。 它報(bào)告函數(shù)級(jí)別的請(qǐng)求次數(shù)和各種指標(biāo),包括阻塞時(shí)間,CPU時(shí)間和內(nèi)存使用情況。 XHPr...

    raoyi 評(píng)論0 收藏0
  • php xhprof使用

    摘要:性能分析此版本為第三方擴(kuò)展官房不支持目錄為擴(kuò)展源碼安狀擴(kuò)展即可編輯啟用擴(kuò)展性能分析數(shù)據(jù)文件存放位置需要用戶(hù)有可寫(xiě)可讀權(quán)限對(duì)項(xiàng)目入口文件添加代碼在第一步后的文件夾里面生成數(shù)據(jù)文件后綴或者創(chuàng)建網(wǎng)占目錄為例在第一步后的文件夾里面訪問(wèn)上面虛擬主機(jī) xhprof php性能分析 1.clone xhprof 此版本為github第三方擴(kuò)展 (php官房不支持 php 7) https://git...

    superw 評(píng)論0 收藏0
  • 使用XHProf查找PHP性能瓶頸

    摘要:是開(kāi)發(fā)的一個(gè)測(cè)試性能的擴(kuò)展,本文記錄了在應(yīng)用中使用對(duì)進(jìn)行性能優(yōu)化,查找性能瓶頸的方法。函數(shù)用于停止性能分析,并返回分析的數(shù)據(jù)。該參數(shù)用于為剖析結(jié)果添加額外的信息,該參數(shù)的值使用以下宏,如果需要提供多個(gè)值,使用進(jìn)行分隔。 XHProf是facebook 開(kāi)發(fā)的一個(gè)測(cè)試php性能的擴(kuò)展,本文記錄了在PHP應(yīng)用中使用XHProf對(duì)PHP進(jìn)行性能優(yōu)化,查找性能瓶頸的方法。 安裝Xhprof擴(kuò)展...

    Xufc 評(píng)論0 收藏0

發(fā)表評(píng)論

0條評(píng)論

最新活動(dòng)
閱讀需要支付1元查看
<