• <ins id="pjuwb"></ins>
    <blockquote id="pjuwb"><pre id="pjuwb"></pre></blockquote>
    <noscript id="pjuwb"></noscript>
          <sup id="pjuwb"><pre id="pjuwb"></pre></sup>
            <dd id="pjuwb"></dd>
            <abbr id="pjuwb"></abbr>
            隨筆-162  評論-223  文章-30  trackbacks-0
               為了使nginx支持windows服務(wù),本文闡述以下主要的改進(jìn)實現(xiàn)。

            ngx_main函數(shù)
               為了在SCM服務(wù)中復(fù)用main函數(shù)的邏輯,將其重命名為ngx_main,并添加第3個參數(shù)is_scm以兼容控制臺運行方式,聲明在core/nginx.h中。   
            1#if (NGX_WIN32 && NGX_USE_SERVICE)
            2extern int  ngx_main(int argc,char *const *argv,int is_scm);
            3#endif
               定義在core/nginx.c中,供main函數(shù)和ServiceMain函數(shù)調(diào)用。
            1int ngx_cdecl
            2#if (NGX_WIN32 && NGX_USE_SERVICE)
            3    ngx_main(int argc, char *const *argv, int is_scm)
            4#else
            5    main(int argc, char *const *argv)
            6#endif    
               is_scm為非0表示以SCM服務(wù)方式運行,否則以控制臺方式運行。宏NGX_USE_SERVICE用于支持SCM服務(wù)方式的條件編譯。相比老的main函數(shù),ngx_main依次修改了以下幾方面:

               不顯示版本和幫助
               只有以控制臺方式運行時,才能查看版本和幫助,即這部分代碼當(dāng)is_scm為0時才有效。
             1#if (NGX_WIN32 && NGX_USE_SERVICE)
             2if(!is_scm){        
             3  if (ngx_show_version) {
             4      ngx_write_stderr("nginx version: " NGINX_VER NGX_LINEFEED);
             5
             6      if (ngx_show_help) {
             7        
             8      }

             9
            10      if (ngx_show_configure) {
            11          
            12      }

            13
            14      if (!ngx_test_config) {
            15          return 0;
            16      }

            17  }

            18}

            19#endif

               增加服務(wù)初始化
             1log = ngx_log_init(ngx_prefix);
             2if (log == NULL) {
             3  return 1;
             4}

             5
             6#if (NGX_WIN32 && NGX_USE_SERVICE)    
             7if(is_scm){
             8    ngx_service_init(log,argc,argv);
             9}

            10#endif
               當(dāng)is_scm為非0即nginx以服務(wù)方式運行時,調(diào)用外部函數(shù)ngx_service_init初始化。因為此時ngx_cycle還沒構(gòu)造,而ngx_service_init須將錯誤記錄在log中,所以應(yīng)在ngx_log_init完成后調(diào)用。   

               不測試配置和處理信號
               與不顯示版本和幫助同理,只有以控制臺方式運行時,這部分處理才有效。
             1#if (NGX_WIN32 && NGX_USE_SERVICE)
             2    if(!is_scm){                
             3       if (ngx_test_config) {
             4          if (!ngx_quiet_mode) {
             5               ngx_log_stderr(0"configuration file %s test is successful", cycle->conf_file.data);
             6           }

             7        
             8           return 0;
             9       }

            10
            11    if (ngx_signal) {
            12           return ngx_signal_process(cycle, ngx_signal);
            13       }
                    
            14    }

            15#endif

               設(shè)置狀態(tài)為正在運行
             1#if (NGX_WIN32 && NGX_USE_SERVICE)
             2  if(is_scm && ngx_service_update_state(ngx_cycle->log,SERVICE_RUNNING,0,0)){
             3        exit(1);
             4    }

             5#endif
             6    ngx_use_stderr = 0;
             7
             8if (ngx_process == NGX_PROCESS_SINGLE) {
             9      ngx_single_process_cycle(cycle);
            10
            11}
             else {
            12     ngx_master_process_cycle(cycle);
            13}

            14
            15   return 0;
               為了準(zhǔn)確報告服務(wù)正在運行的狀態(tài),應(yīng)在所有初始化完成后進(jìn)行,所以在最后的主循環(huán)前調(diào)用ngx_service_update_state。


               以下5個部分對應(yīng)的實現(xiàn)函數(shù),都定義在os/win32/ngx_service.c中。
            main函數(shù)
               nginx程序的啟動入口主函數(shù),可被控制臺或SCM調(diào)用,當(dāng)被SCM調(diào)用時,負(fù)責(zé)用SCM來注冊服務(wù),以及啟動服務(wù)控制調(diào)度程序。
             1void main(int argc, char *const *argv)
             2{
             3    SERVICE_TABLE_ENTRY st[] = {
             4        "nginx"ngx_service_main },
             5        { NULL, NULL }
             6    }
            ;
             7
             8  /* StartServiceCtrlDispatcher() should be called within 30 seconds */    
             9  if (StartServiceCtrlDispatcher(st) == 0{     
            10         if(ERROR_FAILED_SERVICE_CONTROLLER_CONNECT==ngx_errno){
            11            ngx_main(argc,argv,0);
            12         }

            13  }

            14}
               指定ServiceMain函數(shù)為ngx_service_main,有2種情況會以控制臺方式運行:1)在控制臺啟動nginx產(chǎn)生的master和worker進(jìn)程,2)在SCM中啟動nginx產(chǎn)生的worker進(jìn)程。當(dāng)以控制臺方式運行時,StartServiceCtrlDispatcher失敗返回ERROR_FAILED_SERVICE_CONTROLLER_CONNECT錯誤,以is_scm為0調(diào)用ngx_main函數(shù);當(dāng)以SCM服務(wù)方式運行成功時,會調(diào)用到ServiceMain函數(shù)。

            ServiceMain函數(shù)
               由SCM生成的一個邏輯線程調(diào)用。
            1static void WINAPI ngx_service_main(u_int argc, char **argv)
            2{
            3    ngx_main(argc,argv,1);    
            4    ngx_service_update_state(ngx_cycle->log,SERVICE_STOPPED,0,0);
            5}
               以is_scm為1調(diào)用ngx_main。當(dāng)ngx_main異常退出時,會調(diào)用到ngx_exit_handler;當(dāng)正常返回時直接更新服務(wù)狀態(tài)為已停止。

            服務(wù)初始化    
               由ngx_main調(diào)用。
             1void ngx_service_init(ngx_log_t *log,int argc,char *const *argv)
             2{
             3    atexit(ngx_exit_handler);
             4    
             5    ngx_service = RegisterServiceCtrlHandlerEx("nginx"ngx_service_control, NULL);
             6    if(ngx_service == INVALID_HANDLE_VALUE) {
             7        ngx_log_error(NGX_LOG_EMERG,log,ngx_errno,"RegisterServiceCtrlHandlerEx fail");
             8        exit(1);
             9    }

            10    
            11    ngx_status.dwServiceType = SERVICE_WIN32_OWN_PROCESS;
            12    ngx_status.dwControlsAccepted = SERVICE_ACCEPT_STOP|SERVICE_ACCEPT_PARAMCHANGE;
            13    ngx_status.dwWin32ExitCode = NO_ERROR;
            14    ngx_status.dwServiceSpecificExitCode = 0;
            15    
            16    /* SetServiceStatus() should be called within 80 seconds */
            17    if(ngx_service_update_state(log,SERVICE_START_PENDING,1,2000)){
            18        exit(1);
            19    }

            20}
               先注冊退出回調(diào)ngx_exit_handler,再注冊服務(wù)控制處理器ngx_service_control,最后設(shè)置服務(wù)狀態(tài)為正在啟動。 由于在nginx實現(xiàn)中,有多處出現(xiàn)異常錯誤時直接調(diào)用exit,為了簡單方便,在退出時報告服務(wù)停止的狀態(tài),因此首先使用atexit注冊了ngx_exit_handler。
            1static void ngx_exit_handler(void)
            2{
            3    ngx_service_update_state(ngx_cycle->log,SERVICE_STOPPED,0,0);
            4}

            服務(wù)控制處理器
               由SCM生成的一個邏輯線程調(diào)用。
             1static u_long WINAPI ngx_service_control(u_long control, u_long type, void *data, void *ctx)
             2{
             3  switch(control) {
             4     case SERVICE_CONTROL_STOP:        
             5      ngx_service_update_state(ngx_cycle->log, SERVICE_STOP_PENDING00);
             6      ngx_os_signal_process(ngx_cycle,"quit",ngx_pid);
             7      break;
             8  }

             9
            10  return NO_ERROR;
            11}
               當(dāng)停止或重新啟動服務(wù)時會進(jìn)入到SERVICE_CONTROL_STOP分支,設(shè)置服務(wù)狀態(tài)為正在停止,發(fā)送quit信號(在nginx中用命名event代替實現(xiàn))給master進(jìn)程。

            設(shè)置服務(wù)狀態(tài) 
             1int ngx_service_update_state(ngx_log_t *log, u_long state, u_long checkpoint,u_long waithint)
             2{
             3    ngx_status.dwCurrentState = state;
             4    ngx_status.dwCheckPoint = checkpoint;
             5    ngx_status.dwWaitHint = waithint;
             6
             7    if(SetServiceStatus(ngx_service, &ngx_status) == 0{
             8        ngx_log_error(NGX_LOG_EMERG,log,ngx_errno,"SetServiceStatus fail");
             9        return -1;
            10    }
                
            11    
            12    return 0;
            13}
               由ngx_service_init、ngx_main、ngx_service_control和ngx_exit_handler調(diào)用,分別設(shè)置正在啟動、正在運行、正在停止和已停止?fàn)顟B(tài)。
            posted on 2016-07-12 15:31 春秋十二月 閱讀(4524) 評論(0)  編輯 收藏 引用 所屬分類: Opensrc
            久久婷婷色香五月综合激情| 国产精品久久久久久久app| 久久婷婷色香五月综合激情 | 久久久久国产精品三级网| 久久久久亚洲av无码专区| 亚洲精品无码久久久久去q | 国内精品九九久久久精品| 亚洲综合伊人久久大杳蕉| 亚洲伊人久久大香线蕉综合图片| 久久亚洲AV无码精品色午夜麻豆| 久久狠狠爱亚洲综合影院| 中文精品久久久久人妻| 国产成人精品久久| 久久久久无码精品国产| 国产精品久久永久免费| 91精品国产综合久久四虎久久无码一级| .精品久久久麻豆国产精品| 91精品国产9l久久久久| 99久久伊人精品综合观看| 久久综合九色综合欧美狠狠| 久久精品国产欧美日韩| 色婷婷噜噜久久国产精品12p| 热综合一本伊人久久精品| 亚洲午夜久久久久妓女影院| 久久丫精品国产亚洲av不卡| 99国产精品久久| 久久久久亚洲AV成人网| 久久精品国产亚洲αv忘忧草 | 国内精品久久久久久久coent| 亚洲AV伊人久久青青草原| 精品综合久久久久久97| 久久97精品久久久久久久不卡| 久久国产精品国语对白| 囯产精品久久久久久久久蜜桃 | 久久综合成人网| 久久婷婷五月综合97色| 久久一区二区免费播放| 国产精品久久久久久一区二区三区| 91久久精品电影| 欧美一区二区三区久久综| 九九久久精品无码专区|