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:
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?
- Ereignisse
- {
- use epoll;
- worker_connections 10240;
- } 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[] = {
-
{ ngx_string(
"worker_connections"- ),
-
NGX_EVENT_CONF|NGX_CONF_TAKE1, ngx_event_connections,
-
0,
-
0,
-
NULL } ,
-
...(hier weggelassen)
-
-
{ ngx_string(
"connections"- ),
-
NGX_EVENT_CONF|NGX_CONF_TAKE1, ngx_event_connections,
- 0,
-
0,
-
NULL },
-
-
{ ngx_string(
"use"- ),
-
NGX_EVENT_CONF|NGX_CONF_TAKE1, ngx_event_use,
-
0,
-
0,
-
NULL },
-
ngx_null_command
-
};
- Wenn Ereignisse analysiert werden, wird die folgende Funktion zurückgerufen:
[cpp] view
Klartext?
- src/event/ngx_event.c
- statisch char *
- ngx_events_block(ngx_conf_t *cf, ngx_command_t *cmd, void *conf)
- {
- char *rv;
- void ***ctx;
- ngx_uint_t i;
- ngx_conf_t pcf;
- ngx_event_module_t *m;
-
- /* zähle die Anzahl der Ereignismodule und richte deren Indizes ein.*/
- //计算event模块数量,并且记录
- ngx_event_max _module = 0 ;
- for (i = 0; ngx_modules[i]; i++) {
- if (ngx_modules[i]->type != NGX_EVENT_MODULE) {
- Weiter;
- }
-
- ngx_modules[i]->ctx_index = ngx_event_max_module++ ;
- }
-
- ctx = ngx_pcalloc(cf->pool, sizeof(void *));
- if (ctx == NULL) {
- return NGX_CONF_ERROR;
- }
- //为每一个event模块分配空间,用来保存响应配置结构的地址
- //共分配了ngx_event_max_module个空间
- *ctx = ngx_pcalloc(cf->pool, ngx_event_max_module * sizeof(void *));
- if (*ctx == NULL) {
- return NGX_CONF_ERROR;
- }
-
- *(void **) conf = ctx;
-
- for (i = 0; ngx_modules[i]; i++) {
- if (ngx_modules[i]->type != NGX_EVENT_MODULE) {
- weiter;
- }
-
- m = ngx_modules[i]->ctx;
- //Rufen Sie die Hook-Funktion creat_conf jedes Moduls in einer Schleife auf, um die Konfigurationsstruktur zu erstellen
-
->create_conf) { (*ctx)[ngx_modules[i]->ctx_index] = m->create_conf(cf->cycle);
-
if- ((*ctx)[ngx_modules[i]->ctx_index] == NULL) { >
- }
- pcf = *cf;
- cf->ctx = ctx;
- cf->module_type = NGX_EVENT_MODULE; F; eine Blockanweisung, viele andere Anweisungen können unter der Ereignisdomäne konfiguriert werden,
- //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.
rv = ngx_conf_parse(cf, NULL); *cf = pcf; -
- if (rv != NGX_CONF_OK)
- Rückgabe rv;
- für
(i = 0; ngx_modules[i]; i++) {
- if (ngx_modules[i]-> Typ != NGX_EVENT_MODULE) {
- } Die Funktion init_conf eines Ereignismoduls initialisiert die Konfigurationsstruktur
-
if
- (m->init_conf) { rv = m->init_conf (cf->cycle, (*ctx)[ngx_modules[i]->ctx_index]);
- > rv;
> -
zurück NGX_CONF_OK
}
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?
- src/event/ngx_event.c
- statisch ngx_int_t
- ngx_event_module_init(ngx_cycle_t *cycle)
- {
- void ***cf;
- u_char *shared;
- size_t size, cl;
- ngx_shm_t shm;
- ngx_time_t *tp;
- ngx_core_conf_t *ccf;
- ngx_event_conf_t *ecf;
-
- //判断ngx_events_module是否调用过初始化conf操作
- cf = ngx_get_conf(cycle->conf_ctx, ngx_events_module);
-
- if (cf == NULL) {
- ngx_log_error(NGX_LOG_EMERG, cycle->log, 0,
- „kein Abschnitt „Ereignisse“ in der Konfiguration“);
- return NGX_ERROR;
- }
-
- //获取ngx_event_core_module模块的配置结构
- ecf = (*cf)[ngx_event_core_module.ctx_index];
-
- //查看是否是event中的模块,例如use 。。。。
- if (!ngx_test_config && ngx_process <= NGX_PROCESS_MASTER) { > ngx_log_error (NGX_LOG_NOTICE, Cycle->log, 0,
-
🎜>
, ecf->name); > } -
//Erhalten Sie die Konfigurationsstruktur des ngx_core_module-Moduls
ccf = (ngx_core_conf_t *) ngx_get_conf(cycle->conf_ctx, ngx_core_module); -
- // Holen Sie sich den timer_resolution-Parameter aus dem Konfigurationsstruktur des Moduls ngx_core_module
ngx_timer_resolution = ccf-> ;timer_resolution; -
- #if !(NGX_WIN32)
{ - ngx_int_t limit;
-
- //Ermitteln Sie die maximale Anzahl von Dateien, die der aktuelle Prozess öffnen kann getrlimit
- if
(getrlimit(RLIMIT_NOFILE, &rlmt) == -1) {
- ngx_log_error(NGX_LOG_ALERT, Cycle->log, ngx_errno, );
- }else {
- //Wenn die Anzahl der ngx_event_core_module-Modulverbindungen größer als das aktuelle (weiche) Limit ist
-
/Oder die Anzahl der ngx_event_core_module-Verbindungen ist größer als die maximale Anzahl der Verbindungen von ngx_core_module
- if (ecf->connections > ( ngx_uint_t) rlmt.rlim_cur.
- || ecf-> ;connections > ; (ngx_uint_t) ccf->rlimit_nofile)) {
- limit = (ccf->rlimit_nofile == NGX_ CONF_UNSET) ?
- (ngx_int_t) rlmt.rlim_cur : ccf->rlimit_nofile; 🎜> ngx_log_error (ngx_log_warn, cycle- & gt; log, 0,
-
ecf- & gt; Verbindungen, Limit); - >#endif /* !(NGX_WIN32) */
-
- //Wenn Sie nach dem Schließen des Master-Prozesses zu
zurückkehren - //Da das Schließen des Master-Prozesses ein Einzelprozess ist Arbeitsmodus,
- // Nach dem Betrieb sind keine Implementierungssperren und andere Aufgaben für den gemeinsamen Speicher erforderlich.
- if (ccf->master == 0) {
- , es ist nicht erforderlich, es erneut zu erstellen
- if
(ngx_accept_mutex_ptr) {
- return NGX_OK;
- }
- /* cl sollte gleich oder größer als die Cache-Zeilengröße sein */
-
- cl = 128;
- //Dieser gemeinsame Speicher wird in drei Abschnitte unterteilt size = cl
/* ngx_accept_mutex */-
-
+ cl
/ * ngx_connection_counter * /-
🎜 >
- // Gemeinsamen Speicher vorbereiten, die Größe ist size, benannt nginx_shared_zone,
-
shm.size = size; 🎜> shm.name.data = (u_char * ) "nginx_shared_zone"
- ; shm.log =cycle->log; > shm.addr
- if (ngx_shm_alloc(&shm) != NGX_OK) { 🎜>//Startadresse abrufen und speichern
-
-
shared = shm.addr; //accept mutex ruft den ersten CL-Speicher des Shared Memory ab
- ngx_accept_mutex_ptr = (ngx_atomic_t *) shared; >
-
-
Die Implementierung von Accept Mutex hängt davon ab, ob es atomare Operationen unterstützt
- Der erhaltene gemeinsame Speicher wird zur Implementierung des Accept-Mutex verwendet; andernfalls wird die Dateisperre verwendet
- um den Accept-Mutex zu implementieren.
- Die Funktion des Accept-Mutex besteht darin, Panikgruppen zu vermeiden und Arbeiter zu implementieren Prozesslastausgleich.
-
-
*/
- if (ngx_shmtx_create(&ngx_accept_mutex, shared, Cycle->lock_file.data)
- = NGX_OK)
- {
-
🎜> } >
ngx_connection_counter = (ngx_atomic_t *) (shared + 1 * cl); -
( void) ngx_atomic_cmp_set(ngx_connection_counter, 0, 1); , Cycle->log, 0,
"counter: %p, %d"- ,
-
ngx_connection_counter, *ngx_connection_counter); 🎜> ngx_temp_number = (ngx_atomic_t *) (shared + 2 * cl); >
-
tp = ngx_timeofday(); 🎜>
-
- returnNGX_OK;
- }
-
4.ngx_event_process_init
- 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.
- [cpp]-Ansicht
Klartext?
- src/event/ngx_event.c
- statischngx_int_t >
ngx_event_process_init(ngx_cycle_t *cycle) { ngx_uint_t m, i; -
ngx_event_t *rev, *wev; ngx_listening_t *ls; ngx_connection_t *c, *next, *old; >-
ngx_core_conf_t *ccf; ngx_event_module_t *module ;
-
-
//Wie zuvor, Holen Sie sich die Konfigurationsstruktur des Antwortmoduls -
-
ccf = (ngx_core_conf_t *) ngx_get_conf(cycle->conf_ctx, ngx_core_module);
-
ecf = ngx_event_get_conf(cycle-&g t;conf_ctx, ngx_event_core_module);
-
>
- //Öffnen Sie einfach den Accept-Mutex
-
if (ccf->master && ccf->worker_processes > 1 && ecf->accept_mutex) {
-
ngx_use_accept_mutex = 1; // Mutex verwenden
-
ngx_accept_mutex_held = 0; //Ob Mutex akzeptiert werden soll
-
ngx_accept_mutex_delay = ecf->accept_mutex_delay;//Nachdem die Konkurrenz um den Mutex fehlgeschlagen ist, warten Sie auf das nächste Konkurrenzintervall
-
} - else {
ngx_use_accept_mutex = 0; } (NGX_THREADS)-
//Reden wir vorerst nicht über Threads- >
#endif-
-
//- Zähler initialisieren, wobei a Ein rot-schwarzer Baum zur Wartung des Timers, der später ausführlich erläutert wird
if- (ngx_event_timer_init(cycle- >log ). for (m = 0; ngx_modules[m]; m++) {
- //Wie hier bereits erwähnt, überspringen Sie Nicht-NGX_EVENT_MODULE-Module
- if (ngx_modules[m]->type != NGX_EVENT_MODULE) {
- weiter;
}- //Module, die durch Nichtverwendungskonfigurationsanweisungen angegeben werden, werden übersprungen, Linux verwendet standardmäßig epoll
- > ; 🎜> Weil Nginx viele Ereignismodule implementiert, wie zum Beispiel: epoll, poll, select, dqueue, aio
- (Diese Module befinden sich im Verzeichnis src/event/modules), sodass Nginx das Zeitmodul abstrahiert,
- , um verschiedene Systeme zu erleichtern. Die Verwendung verschiedener Ereignismodelle erleichtert auch die Erweiterung neuer Zeit Unser Fokus sollte auf Epoll liegen.
-
- 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.
- Der Aktionstyp ist ngx_event_action_t, befindet sich in src/event/ngx_event.h
-
- Diese spezifischen Inhalte werden im nächsten Abschnitt erläutert.
-
- if (module->actions.init(cycle, ngx_timer_resolution) != NGX_OK) {
} -
-
//Einige Inhalte hier weglassen
- //Global erstellen Das Array ngx_connection_t speichert alle Verbindungen
- //Da dieser Prozess in jedem Worker-Prozess ausgeführt wird, hat jeder Worker jeweils sein eigenes Verbindungsarray
cycle->connections = ngx_alloc(- sizeof(ngx_connection_t) *cycle ->connection_n,cycle->log);
- if (cycle->connections == NULL) {
🎜> } -
c =. Zyklus ->Verbindungen; - //Erstellen Sie eine Reihe von Leseereignissen
-
Cycle->read_events = ngx_alloc(sizeof(ngx_event_t) * Cycle->connection_n,
-
🎜> if
- (cycle->read_events == NULL) {
-
return
- NGX_ERROR;
- }
- rev = Cycle->read_events ;
- für (i = 0; i connection_n; i++) {
- rev[i].closed = 1;
-
rev[i].instance = 1; #if ( NGX_THREADS)
- rev[i].lock = &c[i].lock
- rev [i].own_lock = &c[i].lock;
-
#endif
- > }
- //创建一个写事件数组
- cycle->write_events = ngx_alloc (sizeof(ngx_event_t) * cycle->connection_n,
- cycle->log);
- if (cycle->write_events == NULL) {
- return NGX_ERROR;
- }
-
- wev = cycle->write_events;
- for (i = 0; i < cycle->connection_n; i++) {
- wev[i].closed = 1;
- #if (NGX_THREADS)
- wev[i].lock = &c[ i].lock;
- wev[i].own_lock = &c[i].lock;
- #endif
- }
-
- i = cycle->connection_n;
- next = NULL;
- //初始化整个connection数组
- tun {
- i--;
-
- c[i].data = next;
- c[i].read = &cycle->read_events[i];
- c[i].write = &cycle->write_events[i];
- c[i].fd = (ngx_socket_t) -1;
-
- next = &c[i];
-
- #if (NGX_THREADS)
- c[i].lock = 0;
- #endif
- }while (i);
- 🎜>
- Cycle->free_connection_n = Cycle-> Verbindung_n;
-
- /* für jede Hörbuchse */
- //Weisen Sie für jeden Listening-Socket, also einen Steckplatz, eine Verbindung aus dem Verbindungsarray zu
- ls =cycle->listening.elts;
- for
(i = 0; i listening.nelts; i++) { - //Holen Sie sich ein neues von conneciton Connection solt
- c = ngx_get_connection(ls[i].fd,cycle->log);
-
- if
(c == NULL) { -
>return NGX_ERROR;
- }
-
- c->log = &ls[i].log
-
-
c->listening = &ls[i]; 🎜>
-
rev = c->read;
-
->log = c->log
-
rev->accept = 1;
//Lesezeit auftritt, Anrufannahme-
-
- #if (NGX_HAVE_DEFERRED_ACCEPT)
- >
-
if (!(ngx_event_flags & NGX_USE_
IOC- P_EVENT)) { if
- (ls[i]. previous) {
-
- * das alte Zyklusleseereignis-Array */
-
- old = ls[i]. previous->connection;
- if
- (ngx_del_event (old-& gt; read, ngx_read_event, ngx_close_event) == ngx_error)
- ) -1;
🎜>- }
-
- //Registrieren Sie die Rückruffunktion, um auf Socket-Poisoning-Ereignisse zu warten ngx_event_accept
- rev->handler = ngx_event_accept;
- /Accept_mutex verwendet, den Listening-Socket vorerst nicht in epoll einfügen, sondern
- //Warten Sie, bis der Arbeiter den Accept-Mutex ergreift, und fügen Sie dann Epoll ein, um Panikattacken zu vermeiden
- if
(ngx_use_accept_mutex) { 🎜>-
>
- if (ngx_event_flags & NGX_USE_RTSIG_EVENT) {
-
turn
- NGX_ERROR;
-
>
-
- 🎜> }
- #endif
-
- }
-
return
- NGX_OK; }
-
Bis jetzt ist die ereignisgesteuerte Initialisierung abgeschlossen.
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. -