Heim  >  Artikel  >  Backend-Entwicklung  >  Nginx-Quellcode-Studiennotizen (20) – Ereignismodul eins – Initialisierung

Nginx-Quellcode-Studiennotizen (20) – Ereignismodul eins – Initialisierung

WBOY
WBOYOriginal
2016-07-29 09:11:01906Durchsuche

Nachdem ich die vorherigen Studiennotizen gelesen habe, glaube ich, dass ich ein gewisses Verständnis für den Startvorgang von Nginx habe. Von diesem Abschnitt an möchten wir uns eingehend mit den einzelnen Modulen befassen und die Hauptoperationen in jedem Modul erlernen.

Dieser Artikel stammt von: http://blog.csdn.net/lengzijian/article/details/7598996

Lass uns heute lernen. Lass uns Laden Sie das Ereignismodul herunter. Im vorherigen Start haben wir den Aufruf der Hook-Funktionen jedes Moduls mehrfach erwähnt. Erinnern wir uns zunächst an die Ausführung der Hook-Funktion des Ereignismoduls, die auch der Schritt zum Starten des Ereignismoduls ist:

nginx 源码学习笔记(二十)—— event 模块一 ——初始化

1. Conf(create_conf) erstellen:

ngx_event_create_conf()

Diese Methode erstellt hauptsächlich eine ngx_event_conf_t-Struktur und weist Speicherplatz zu.

2. Lesen Sie die Konfigurationsdatei:

Die gelesene Datei enthält beispielsweise die folgenden Zeilen:

[cpp]-Ansicht Klartext?

  1. Ereignisse 
  2. use epoll;
  3. worker_connections 10240;
  4. } Lokale Ereignisse sind eine Blockanweisung, die in geschweiften Klammern konfiguriert werden kann /event/ngx_event.c

[cpp] Ansicht Klartext?

statisch

ngx_command_t ngx_event_core_commands[] = {
  1. { ngx_string(
  2. "worker_connections"
  3. ),
  4. NGX_EVENT_CONF|NGX_CONF_TAKE1, ngx_event_connections,
  5. 0,
  6. 0,
  7. NULL } ,
  8. ...(hier weggelassen)
  9. { ngx_string(
  10. "connections"
  11. ),
  12. NGX_EVENT_CONF|NGX_CONF_TAKE1, ngx_event_connections,
  13. 0,
  14. 0,
  15. NULL },
  16. { ngx_string(
  17. "use"
  18. ),
  19. NGX_EVENT_CONF|NGX_CONF_TAKE1, ngx_event_use,
  20. 0,
  21. 0,
  22. NULL },
  23. ngx_null_command
  24. };
  25. Wenn Ereignisse analysiert werden, wird die folgende Funktion zurückgerufen:

[cpp] view Klartext?

  1. src/event/ngx_event.c  
  2. statisch  char *  
  3. ngx_events_block(ngx_conf_t *cf, ngx_command_t *cmd, void *conf)  
  4. {  
  5.     char                 *rv;  
  6.     void               ***ctx;  
  7.     ngx_uint_t            i;  
  8.     ngx_conf_t            pcf;  
  9.     ngx_event_module_t   *m;  
  10.   
  11.     /* zähle die Anzahl der Ereignismodule und richte deren Indizes ein.*/  
  12.     //计算event模块数量,并且记录  
  13.     ngx_event_max _module = 0 ;  
  14.     for (i = 0; ngx_modules[i]; i++) {  
  15. if (ngx_modules[i]->type != NGX_EVENT_MODULE) {  
  16.             Weiter;  
  17.         }  
  18.   
  19.         ngx_modules[i]->ctx_index = ngx_event_max_module++ ;  
  20.     }  
  21.   
  22.     ctx = ngx_pcalloc(cf->pool,  sizeof(void *));  
  23.     if (ctx == NULL) {  
  24.          return NGX_CONF_ERROR;  
  25.     }  
  26.     //为每一个event模块分配空间,用来保存响应配置结构的地址  
  27.     //共分配了ngx_event_max_module个空间  
  28.     *ctx = ngx_pcalloc(cf->pool, ngx_event_max_module * sizeof(void *));  
  29.     if (*ctx == NULL) {  
  30.         return NGX_CONF_ERROR;  
  31.     }  
  32.   
  33.     *(void **) conf = ctx;  
  34.   
  35.     for (i = 0; ngx_modules[i]; i++) {
  36.         if (ngx_modules[i]->type != NGX_EVENT_MODULE) {  
  37.             weiter;  
  38.         }
  39. m = ngx_modules[i]->ctx;
  40. //Rufen Sie die Hook-Funktion creat_conf jedes Moduls in einer Schleife auf, um die Konfigurationsstruktur zu erstellen
  41. ->create_conf) { (*ctx)[ngx_modules[i]->ctx_index] = m->create_conf(cf->cycle);
  42. if
  43. ((*ctx)[ngx_modules[i]->ctx_index] == NULL) {                                                                                                    >                                                                          
  44. }
  45. pcf = *cf;
  46. cf->ctx = ctx;
  47. cf->module_type = NGX_EVENT_MODULE; F; eine Blockanweisung, viele andere Anweisungen können unter der Ereignisdomäne konfiguriert werden,
  48. //Zum Beispiel die zuvor erwähnte Verwendung usw ., beginnen Sie nun mit dem Parsen der Anweisungen im Ereignisblock und schließen Sie die Initialisierungsarbeit ab.
  49. rv = ngx_conf_parse(cf, NULL);
  50. *cf = pcf;
  51. if (rv != NGX_CONF_OK)
  52. Rückgabe rv;
  53. für
  54. (i = 0; ngx_modules[i]; i++) {
  55. if (ngx_modules[i]-> Typ != NGX_EVENT_MODULE) {
  56. }                                                    Die Funktion init_conf eines Ereignismoduls initialisiert die Konfigurationsstruktur
  57. if
  58. (m->init_conf) { rv = m->init_conf (cf->cycle, (*ctx)[ngx_modules[i]->ctx_index]);
  59.                                                                                                              > rv;
  60.                                                                                              >
  61. zurück NGX_CONF_OK
  62. }

  63. Der wichtigste Prozess in der Funktion ngx_events_block() ist der Aufruf von ngx_conf_parse(cf, NULL). Die Funktion des Aufrufs von ngx_conf_parse() besteht darin, die Analyse des Blocks events{} abzuschließen Konfigurationsdatei, also Rufen Sie die Rückruffunktionen aller Konfigurationsanweisungen darunter auf, um die Initialisierung des Parsens der Konfigurationsdatei abzuschließen. Aber hier habe ich eine persönliche Frage. Nachdem ich die Senioren gefragt habe, werde ich die Frage und Antwort angeben******.

    2. Initialisierung conf(init_conf)

    ngx_event_init_conf()

    Die Methode, die hauptsächlich die Struktur ngx_event_conf_t initialisiert.

    3.ngx_event_module_init

    Aus dem Namen geht hervor, dass es sich um den Initialisierungsvorgang des Moduls handelt, aber wenn man sich die Quelle ansieht Im Code jedes Moduls haben wir festgestellt, dass es viele gibt. Keines der Module verfügt über eine Init-Rückruffunktion. Ich habe auch Probleme mit dem Warum hier und hoffe, dass ich die Antwort finden kann, nachdem ich den gesamten Code gelernt habe.

    [cpp]-Ansicht Klartext?

  1. src/event/ngx_event.c  
  2. statisch ngx_int_t  
  3. ngx_event_module_init(ngx_cycle_t *cycle)  
  4. {  
  5.     void              ***cf;  
  6.     u_char              *shared;  
  7.     size_t               size, cl;  
  8.     ngx_shm_t            shm;  
  9.     ngx_time_t          *tp;  
  10.     ngx_core_conf_t     *ccf;  
  11.     ngx_event_conf_t    *ecf;  
  12.       
  13.     //判断ngx_events_module是否调用过初始化conf操作  
  14.     cf = ngx_get_conf(cycle->conf_ctx, ngx_events_module);  
  15.   
  16.     if (cf == NULL) {  
  17.         ngx_log_error(NGX_LOG_EMERG, cycle->log, 0,  
  18.                       „kein Abschnitt „Ereignisse“ in der Konfiguration“);  
  19.         return NGX_ERROR;  
  20.     }  
  21.       
  22.     //获取ngx_event_core_module模块的配置结构  
  23.     ecf = (*cf)[ngx_event_core_module.ctx_index];  
  24.       
  25.     //查看是否是event中的模块,例如use 。。。。
  26. if (!ngx_test_config && ngx_process <= NGX_PROCESS_MASTER) { > ngx_log_error (NGX_LOG_NOTICE, Cycle->log, 0,
  27. 🎜>
  28. , ecf->name); > }
  29. //Erhalten Sie die Konfigurationsstruktur des ngx_core_module-Moduls
  30. ccf = (ngx_core_conf_t *) ngx_get_conf(cycle->conf_ctx, ngx_core_module);
  31. // Holen Sie sich den timer_resolution-Parameter aus dem Konfigurationsstruktur des Moduls ngx_core_module
  32. ngx_timer_resolution = ccf-> ;timer_resolution;
  33. #if !(NGX_WIN32)
  34. {
  35. ngx_int_t limit;
  36. //Ermitteln Sie die maximale Anzahl von Dateien, die der aktuelle Prozess öffnen kann getrlimit
  37. if
  38. (getrlimit(RLIMIT_NOFILE, &rlmt) == -1) {
  39. ngx_log_error(NGX_LOG_ALERT, Cycle->log, ngx_errno,                                                             );
  40. }else {
  41. //Wenn die Anzahl der ngx_event_core_module-Modulverbindungen größer als das aktuelle (weiche) Limit ist
  42.                                                                                                        /Oder die Anzahl der ngx_event_core_module-Verbindungen ist größer als die maximale Anzahl der Verbindungen von ngx_core_module
  43. if (ecf->connections > ( ngx_uint_t) rlmt.rlim_cur.
  44.                                                          || ecf-> ;connections > ; (ngx_uint_t) ccf->rlimit_nofile)) {
  45. limit = (ccf->rlimit_nofile == NGX_ CONF_UNSET) ?
  46. (ngx_int_t) rlmt.rlim_cur : ccf->rlimit_nofile; 🎜> ngx_log_error (ngx_log_warn, cycle- & gt; log, 0,
  47. ecf- & gt; Verbindungen, Limit);
  48.                                                                    >#endif /* !(NGX_WIN32) */
  49. //Wenn Sie nach dem Schließen des Master-Prozesses zu
  50. zurückkehren
  51. //Da das Schließen des Master-Prozesses ein Einzelprozess ist Arbeitsmodus,
  52. // Nach dem Betrieb sind keine Implementierungssperren und andere Aufgaben für den gemeinsamen Speicher erforderlich.
  53. if (ccf->master == 0) {
  54.                                                                        , es ist nicht erforderlich, es erneut zu erstellen
  55. if
  56. (ngx_accept_mutex_ptr) {
  57. return NGX_OK;
  58. }
  59. /* cl sollte gleich oder größer als die Cache-Zeilengröße sein */
  60. cl = 128;
  61. //Dieser gemeinsame Speicher wird in drei Abschnitte unterteilt size = cl
  62. /* ngx_accept_mutex */
  63. + cl
  64. / * ngx_connection_counter * /
  65.                                                              🎜 >
  66. // Gemeinsamen Speicher vorbereiten, die Größe ist size, benannt nginx_shared_zone,
  67. shm.size = size; 🎜> shm.name.data = (u_char * ) "nginx_shared_zone"
  68. ; shm.log =cycle->log; > shm.addr
  69. if (ngx_shm_alloc(&shm) != NGX_OK) {                                                                                                                      🎜>//Startadresse abrufen und speichern
  70. shared = shm.addr; //accept mutex ruft den ersten CL-Speicher des Shared Memory ab
  71. ngx_accept_mutex_ptr = (ngx_atomic_t *) shared; >
  72. Die Implementierung von Accept Mutex hängt davon ab, ob es atomare Operationen unterstützt
  73. Der erhaltene gemeinsame Speicher wird zur Implementierung des Accept-Mutex verwendet; andernfalls wird die Dateisperre verwendet
  74. um den Accept-Mutex zu implementieren.
  75. Die Funktion des Accept-Mutex besteht darin, Panikgruppen zu vermeiden und Arbeiter zu implementieren Prozesslastausgleich.
  76. */
  77.  if (ngx_shmtx_create(&ngx_accept_mutex, shared, Cycle->lock_file.data) 
  78.                    = NGX_OK)
  79. {
  80. 🎜> } >
  81. ngx_connection_counter = (ngx_atomic_t *) (shared + 1 * cl);
  82. ( void) ngx_atomic_cmp_set(ngx_connection_counter, 0, 1); , Cycle->log, 0,
  83. "counter: %p, %d"
  84. ,
  85. ngx_connection_counter, *ngx_connection_counter); 🎜> ngx_temp_number = (ngx_atomic_t *) (shared + 2 * cl); >
  86. tp = ngx_timeofday(); 🎜>
  87. returnNGX_OK;
  88. }
  89. 4.ngx_event_process_init
  90. wurde in der vorherigen Worker-Prozessanalyse erwähnt. Nach dem Erstellen eines Worker-Prozesses initialisiert der Worker-Prozess zunächst den Prozess und ruft zu diesem Zeitpunkt die Funktion ngx_event_process_init auf.
  91. [cpp]-Ansicht Klartext?
    1. src/event/ngx_event.c
    2. statischngx_int_t >
    3. ngx_event_process_init(ngx_cycle_t *cycle)
    4. {
    5. ngx_uint_t m, i;
    6. ngx_event_t *rev, *wev;
    7. ngx_listening_t *ls;
    8. ngx_connection_t *c, *next, *old; >
    9. ngx_core_conf_t *ccf; ngx_event_module_t *module ;
    10. //Wie zuvor, Holen Sie sich die Konfigurationsstruktur des Antwortmoduls
    11. ccf = (ngx_core_conf_t *) ngx_get_conf(cycle->conf_ctx, ngx_core_module);
    12. ecf = ngx_event_get_conf(cycle-&g t;conf_ctx, ngx_event_core_module);
    13. >
    14. //Öffnen Sie einfach den Accept-Mutex
    15. if (ccf->master && ccf->worker_processes > 1 && ecf->accept_mutex) {
    16. ngx_use_accept_mutex = 1; // Mutex verwenden
    17. ngx_accept_mutex_held = 0; //Ob Mutex akzeptiert werden soll
    18. ngx_accept_mutex_delay = ecf->accept_mutex_delay;//Nachdem die Konkurrenz um den Mutex fehlgeschlagen ist, warten Sie auf das nächste Konkurrenzintervall
    19. }
    20. else {
    21. ngx_use_accept_mutex = 0;
    22. } (NGX_THREADS)
    23. //Reden wir vorerst nicht über Threads
    24. >
    25. #endif
    26. //
    27. Zähler initialisieren, wobei a Ein rot-schwarzer Baum zur Wartung des Timers, der später ausführlich erläutert wird
    28. if
    29. (ngx_event_timer_init(cycle- >log ).  for (m = 0; ngx_modules[m]; m++) { 
    30. //Wie hier bereits erwähnt, überspringen Sie Nicht-NGX_EVENT_MODULE-Module
    31. if (ngx_modules[m]->type != NGX_EVENT_MODULE) {
    32.  
    33. weiter;
    34. }
    35.  //Module, die durch Nichtverwendungskonfigurationsanweisungen angegeben werden, werden übersprungen, Linux verwendet standardmäßig epoll
    36. > ; 🎜> Weil Nginx viele Ereignismodule implementiert, wie zum Beispiel: epoll, poll, select, dqueue, aio
    37. (Diese Module befinden sich im Verzeichnis src/event/modules), sodass Nginx das Zeitmodul abstrahiert,
    38. , um verschiedene Systeme zu erleichtern. Die Verwendung verschiedener Ereignismodelle erleichtert auch die Erweiterung neuer Zeit Unser Fokus sollte auf Epoll liegen.
    39. Der Init-Callback hier ruft tatsächlich die Funktion ngx_epoll_init auf. Die Struktur module->actions kapselt alle Schnittstellenfunktionen von epoll. Nginx registriert Epoll über die Aktionsstruktur in der Ereignisabstraktionsschicht.
    40. Der Aktionstyp ist ngx_event_action_t, befindet sich in src/event/ngx_event.h
    41. Diese spezifischen Inhalte werden im nächsten Abschnitt erläutert.
    42.  
    43. if (module->actions.init(cycle, ngx_timer_resolution) != NGX_OK) {
    44.                                                                                               }  
    45.  
    46.                                                                         
    47. //Einige Inhalte hier weglassen
    48. //Global erstellen Das Array ngx_connection_t speichert alle Verbindungen
    49. //Da dieser Prozess in jedem Worker-Prozess ausgeführt wird, hat jeder Worker jeweils sein eigenes Verbindungsarray
    50. cycle->connections =
    51. ngx_alloc(
    52. sizeof(ngx_connection_t) *cycle ->connection_n,cycle->log);
    53. if (cycle->connections == NULL) {
    54.                                           🎜> } 
    55.  
    56. c =. Zyklus ->Verbindungen; 
    57.  
    58. //Erstellen Sie eine Reihe von Leseereignissen
    59. Cycle->read_events = ngx_alloc(sizeof(ngx_event_t) * Cycle->connection_n,
    60. 🎜> if
    61. (cycle->read_events == NULL) {
    62.  return
    63. NGX_ERROR;
    64. }
    65. rev = Cycle->read_events ;
    66.  für (i = 0; i connection_n; i++) {
    67. rev[i].closed = 1;
    68. rev[i].instance = 1; #if ( NGX_THREADS)
    69. rev[i].lock = &c[i].lock
    70. rev [i].own_lock = &c[i].lock;
    71. #endif
    72. > }  
    73.     //创建一个写事件数组  
    74.     cycle->write_events = ngx_alloc (sizeof(ngx_event_t) * cycle->connection_n,  
    75.                                    cycle->log);  
    76.     if (cycle->write_events == NULL) {  
    77.         return NGX_ERROR;  
    78.     }  
    79.   
    80.     wev = cycle->write_events;  
    81.     for (i = 0; i < cycle->connection_n; i++) {  
    82.         wev[i].closed = 1;  
    83. #if (NGX_THREADS)  
    84.         wev[i].lock = &c[ i].lock;  
    85.         wev[i].own_lock = &c[i].lock;  
    86. #endif  
    87.     }  
    88.   
    89.     i = cycle->connection_n;  
    90.     next = NULL;  
    91.     //初始化整个connection数组  
    92.     tun {  
    93.         i--;  
    94.   
    95.         c[i].data = next;  
    96.         c[i].read = &cycle->read_events[i];  
    97.         c[i].write = &cycle->write_events[i];  
    98.         c[i].fd = (ngx_socket_t) -1;  
    99.   
    100.         next = &c[i];  
    101.   
    102. #if (NGX_THREADS)  
    103.         c[i].lock = 0;  
    104. #endif  
    105.     }while (i);
    106. 🎜>
    107. Cycle->free_connection_n = Cycle-> Verbindung_n;
    108. /* für jede Hörbuchse */
    109. //Weisen Sie für jeden Listening-Socket, also einen Steckplatz, eine Verbindung aus dem Verbindungsarray zu
    110. ls =cycle->listening.elts;
    111. for
    112. (i = 0; i listening.nelts; i++) {
    113. //Holen Sie sich ein neues von conneciton Connection solt
    114. c = ngx_get_connection(ls[i].fd,cycle->log);
    115.  
    116.  if
    117. (c == NULL) {
    118.              >return NGX_ERROR;
    119. }
    120. c->log = &ls[i].log
    121. c->listening = &ls[i]; 🎜> 
    122. rev = c->read;
    123. ->log = c->log
    124. rev->accept = 1;
    125. //Lesezeit auftritt, Anrufannahme
    126. #if (NGX_HAVE_DEFERRED_ACCEPT)
    127.                                                                                   >
    128. if (!(ngx_event_flags & NGX_USE_
    129. IOC
    130. P_EVENT)) {  if
    131. (ls[i]. previous) {
    132.                                                                                                        
    133. * das alte Zyklusleseereignis-Array */
    134. old = ls[i]. previous->connection;
    135. if
    136. (ngx_del_event (old-& gt; read, ngx_read_event, ngx_close_event) == ngx_error)
    137.                                                                                                     ) -1;
    138.                                  🎜>
    139. }
    140. //Registrieren Sie die Rückruffunktion, um auf Socket-Poisoning-Ereignisse zu warten ngx_event_accept
    141. rev->handler = ngx_event_accept;
    142. /Accept_mutex verwendet, den Listening-Socket vorerst nicht in epoll einfügen, sondern
    143. //Warten Sie, bis der Arbeiter den Accept-Mutex ergreift, und fügen Sie dann Epoll ein, um Panikattacken zu vermeiden
    144. if
    145. (ngx_use_accept_mutex) { 🎜>
    146.                                                                                                                   >  
    147.  if (ngx_event_flags & NGX_USE_RTSIG_EVENT) {
    148.                                                                                                                                        turn
    149. NGX_ERROR; 
    150.                                       >                                                                                                                         
    151. 🎜> }  
    152. #endif 
    153. }
    154. return
    155. NGX_OK; }
    156. Bis jetzt ist die ereignisgesteuerte Initialisierung abgeschlossen.
    157. Das Obige stellt die Nginx-Quellcode-Studiennotizen (20) vor – Ereignismodul 1 – Initialisierung, einschließlich IOC und Zählerinhalt. Ich hoffe, dass es für Freunde hilfreich sein wird, die an PHP-Tutorials interessiert sind.
Stellungnahme:
Der Inhalt dieses Artikels wird freiwillig von Internetnutzern beigesteuert und das Urheberrecht liegt beim ursprünglichen Autor. Diese Website übernimmt keine entsprechende rechtliche Verantwortung. Wenn Sie Inhalte finden, bei denen der Verdacht eines Plagiats oder einer Rechtsverletzung besteht, wenden Sie sich bitte an admin@php.cn