Igor Sysoev | 7300977 | 2003-02-06 17:21:13 +0000 | [diff] [blame] | 1 | |
Igor Sysoev | e4a2526 | 2003-06-06 14:59:20 +0000 | [diff] [blame^] | 2 | /* |
| 3 | * Copyright (C) 2002-2003 Igor Sysoev, http://sysoev.ru |
| 4 | */ |
| 5 | |
| 6 | |
Igor Sysoev | 7300977 | 2003-02-06 17:21:13 +0000 | [diff] [blame] | 7 | #include <ngx_config.h> |
Igor Sysoev | 7300977 | 2003-02-06 17:21:13 +0000 | [diff] [blame] | 8 | #include <ngx_core.h> |
Igor Sysoev | 7300977 | 2003-02-06 17:21:13 +0000 | [diff] [blame] | 9 | #include <ngx_event.h> |
Igor Sysoev | 7300977 | 2003-02-06 17:21:13 +0000 | [diff] [blame] | 10 | |
| 11 | |
Igor Sysoev | e4a2526 | 2003-06-06 14:59:20 +0000 | [diff] [blame^] | 12 | typedef struct { |
| 13 | int threads; |
| 14 | } ngx_iocp_conf_t; |
Igor Sysoev | 7300977 | 2003-02-06 17:21:13 +0000 | [diff] [blame] | 15 | |
| 16 | |
Igor Sysoev | e4a2526 | 2003-06-06 14:59:20 +0000 | [diff] [blame^] | 17 | static int ngx_iocp_init(ngx_log_t *log); |
| 18 | static void ngx_iocp_done(ngx_log_t *log); |
| 19 | static int ngx_iocp_add_event(ngx_event_t *ev, int event, u_int key); |
| 20 | static int ngx_iocp_process_events(ngx_log_t *log); |
| 21 | static void *ngx_iocp_create_conf(ngx_pool_t *pool); |
| 22 | static char *ngx_iocp_init_conf(ngx_pool_t *pool, void *conf); |
Igor Sysoev | 7300977 | 2003-02-06 17:21:13 +0000 | [diff] [blame] | 23 | |
| 24 | |
Igor Sysoev | e4a2526 | 2003-06-06 14:59:20 +0000 | [diff] [blame^] | 25 | static ngx_str_t iocp_name = ngx_string("iocp"); |
| 26 | |
| 27 | static ngx_command_t ngx_iocp_commands[] = { |
| 28 | |
| 29 | {ngx_string("iocp_threads"), |
| 30 | NGX_EVENT_CONF|NGX_CONF_TAKE1, |
| 31 | ngx_conf_set_num_slot, |
| 32 | 0, |
| 33 | offsetof(ngx_iocp_conf_t, threads), |
| 34 | NULL}, |
| 35 | |
| 36 | ngx_null_command |
| 37 | }; |
| 38 | |
| 39 | |
| 40 | ngx_event_module_t ngx_iocp_module_ctx = { |
| 41 | &iocp_name, |
| 42 | ngx_iocp_create_conf, /* create configuration */ |
| 43 | ngx_iocp_init_conf, /* init configuration */ |
| 44 | |
| 45 | { |
| 46 | ngx_iocp_add_event, /* add an event */ |
| 47 | NULL, /* delete an event */ |
| 48 | NULL, /* enable an event */ |
| 49 | NULL, /* disable an event */ |
| 50 | NULL, /* add an connection */ |
| 51 | NULL, /* delete an connection */ |
| 52 | ngx_iocp_process_events, /* process the events */ |
| 53 | ngx_iocp_init, /* init the events */ |
| 54 | ngx_iocp_done /* done the events */ |
| 55 | } |
| 56 | |
| 57 | }; |
| 58 | |
| 59 | ngx_module_t ngx_iocp_module = { |
| 60 | NGX_MODULE, |
| 61 | &ngx_iocp_module_ctx, /* module context */ |
| 62 | ngx_iocp_commands, /* module directives */ |
| 63 | NGX_EVENT_MODULE, /* module type */ |
| 64 | NULL /* init module */ |
| 65 | }; |
| 66 | |
| 67 | |
| 68 | static HANDLE iocp; |
| 69 | |
| 70 | |
| 71 | static int ngx_iocp_init(ngx_log_t *log) |
Igor Sysoev | 7300977 | 2003-02-06 17:21:13 +0000 | [diff] [blame] | 72 | { |
Igor Sysoev | e4a2526 | 2003-06-06 14:59:20 +0000 | [diff] [blame^] | 73 | ngx_iocp_conf_t *cf; |
| 74 | |
| 75 | cf = ngx_event_get_conf(ngx_iocp_module); |
| 76 | |
| 77 | iocp = CreateIoCompletionPort(INVALID_HANDLE_VALUE, NULL, 0, cf->threads); |
Igor Sysoev | 7300977 | 2003-02-06 17:21:13 +0000 | [diff] [blame] | 78 | |
| 79 | if (iocp == NULL) { |
| 80 | ngx_log_error(NGX_LOG_EMERG, log, ngx_errno, |
| 81 | "CreateIoCompletionPort() failed"); |
| 82 | return NGX_ERROR; |
| 83 | } |
| 84 | |
Igor Sysoev | e4a2526 | 2003-06-06 14:59:20 +0000 | [diff] [blame^] | 85 | if (ngx_event_timer_init(log) == NGX_ERROR) { |
Igor Sysoev | 7300977 | 2003-02-06 17:21:13 +0000 | [diff] [blame] | 86 | return NGX_ERROR; |
| 87 | } |
| 88 | |
Igor Sysoev | e4a2526 | 2003-06-06 14:59:20 +0000 | [diff] [blame^] | 89 | ngx_event_actions = ngx_iocp_module_ctx.actions; |
Igor Sysoev | 7300977 | 2003-02-06 17:21:13 +0000 | [diff] [blame] | 90 | |
| 91 | ngx_event_flags = NGX_HAVE_AIO_EVENT|NGX_HAVE_IOCP_EVENT; |
| 92 | |
| 93 | return NGX_OK; |
| 94 | } |
| 95 | |
| 96 | |
Igor Sysoev | e4a2526 | 2003-06-06 14:59:20 +0000 | [diff] [blame^] | 97 | static void ngx_iocp_done(ngx_log_t *log) |
| 98 | { |
| 99 | if (CloseHandle(iocp) == -1) { |
| 100 | ngx_log_error(NGX_LOG_ALERT, log, ngx_errno, |
| 101 | "iocp CloseHandle() failed"); |
| 102 | } |
| 103 | |
| 104 | ngx_event_timer_done(log); |
| 105 | } |
| 106 | |
| 107 | |
| 108 | static int ngx_iocp_add_event(ngx_event_t *ev, int event, u_int key) |
Igor Sysoev | 7300977 | 2003-02-06 17:21:13 +0000 | [diff] [blame] | 109 | { |
| 110 | ngx_connection_t *c; |
| 111 | |
| 112 | c = (ngx_connection_t *) ev->data; |
| 113 | |
Igor Sysoev | e4a2526 | 2003-06-06 14:59:20 +0000 | [diff] [blame^] | 114 | ngx_log_debug(ev->log, "iocp add: %d, %08x:%08x" _ c->fd _ key _ &ev->ovlp); |
Igor Sysoev | 7300977 | 2003-02-06 17:21:13 +0000 | [diff] [blame] | 115 | |
Igor Sysoev | e4a2526 | 2003-06-06 14:59:20 +0000 | [diff] [blame^] | 116 | if (CreateIoCompletionPort((HANDLE) c->fd, iocp, key, 0) == NULL) { |
Igor Sysoev | 7300977 | 2003-02-06 17:21:13 +0000 | [diff] [blame] | 117 | ngx_log_error(NGX_LOG_ALERT, ev->log, ngx_errno, |
| 118 | "CreateIoCompletionPort() failed"); |
| 119 | return NGX_ERROR; |
| 120 | } |
| 121 | |
| 122 | return NGX_OK; |
| 123 | } |
| 124 | |
| 125 | |
Igor Sysoev | e4a2526 | 2003-06-06 14:59:20 +0000 | [diff] [blame^] | 126 | static int ngx_iocp_process_events(ngx_log_t *log) |
Igor Sysoev | 7300977 | 2003-02-06 17:21:13 +0000 | [diff] [blame] | 127 | { |
| 128 | int rc; |
Igor Sysoev | e4a2526 | 2003-06-06 14:59:20 +0000 | [diff] [blame^] | 129 | u_int key; |
Igor Sysoev | 7300977 | 2003-02-06 17:21:13 +0000 | [diff] [blame] | 130 | size_t bytes; |
| 131 | ngx_err_t err; |
| 132 | ngx_msec_t timer, delta; |
Igor Sysoev | e4a2526 | 2003-06-06 14:59:20 +0000 | [diff] [blame^] | 133 | ngx_event_t *ev; |
Igor Sysoev | 7300977 | 2003-02-06 17:21:13 +0000 | [diff] [blame] | 134 | ngx_event_ovlp_t *ovlp; |
| 135 | |
Igor Sysoev | 7300977 | 2003-02-06 17:21:13 +0000 | [diff] [blame] | 136 | timer = ngx_event_find_timer(); |
| 137 | |
| 138 | if (timer) { |
| 139 | delta = ngx_msec(); |
| 140 | |
| 141 | } else { |
| 142 | timer = INFINITE; |
| 143 | delta = 0; |
| 144 | } |
| 145 | |
| 146 | ngx_log_debug(log, "iocp timer: %d" _ timer); |
| 147 | |
Igor Sysoev | e4a2526 | 2003-06-06 14:59:20 +0000 | [diff] [blame^] | 148 | rc = GetQueuedCompletionStatus(iocp, &bytes, (LPDWORD) &key, |
Igor Sysoev | 7300977 | 2003-02-06 17:21:13 +0000 | [diff] [blame] | 149 | (LPOVERLAPPED *) &ovlp, timer); |
Igor Sysoev | e4a2526 | 2003-06-06 14:59:20 +0000 | [diff] [blame^] | 150 | |
| 151 | ngx_log_debug(log, "iocp: %d, %d:%08x:%08x" _ rc _ bytes _ key _ ovlp); |
| 152 | |
Igor Sysoev | 7300977 | 2003-02-06 17:21:13 +0000 | [diff] [blame] | 153 | if (rc == 0) { |
Igor Sysoev | 7300977 | 2003-02-06 17:21:13 +0000 | [diff] [blame] | 154 | err = ngx_errno; |
| 155 | |
| 156 | if (ovlp == NULL) { |
| 157 | if (err != WAIT_TIMEOUT) { |
| 158 | ngx_log_error(NGX_LOG_ALERT, log, err, |
| 159 | "GetQueuedCompletionStatus() failed"); |
| 160 | |
| 161 | return NGX_ERROR; |
| 162 | } |
| 163 | |
| 164 | } else { |
| 165 | ovlp->error = err; |
| 166 | } |
| 167 | } |
| 168 | |
| 169 | if (timer != INFINITE) { |
| 170 | delta = ngx_msec() - delta; |
Igor Sysoev | dc479b4 | 2003-03-20 16:09:44 +0000 | [diff] [blame] | 171 | ngx_event_expire_timers(delta); |
Igor Sysoev | 7300977 | 2003-02-06 17:21:13 +0000 | [diff] [blame] | 172 | } |
| 173 | |
| 174 | if (ovlp) { |
| 175 | ev = ovlp->event; |
| 176 | |
| 177 | ngx_log_debug(log, "iocp ev: %08x" _ ev); |
| 178 | |
Igor Sysoev | e4a2526 | 2003-06-06 14:59:20 +0000 | [diff] [blame^] | 179 | switch (key) { |
| 180 | case NGX_IOCP_IO: |
Igor Sysoev | 7300977 | 2003-02-06 17:21:13 +0000 | [diff] [blame] | 181 | ev->ready = 1; |
| 182 | ev->available = bytes; |
Igor Sysoev | e4a2526 | 2003-06-06 14:59:20 +0000 | [diff] [blame^] | 183 | break; |
| 184 | |
| 185 | case NGX_IOCP_ACCEPT: |
| 186 | break; |
Igor Sysoev | 7300977 | 2003-02-06 17:21:13 +0000 | [diff] [blame] | 187 | } |
| 188 | |
Igor Sysoev | e4a2526 | 2003-06-06 14:59:20 +0000 | [diff] [blame^] | 189 | ngx_log_debug(log, "iocp ev handler: %08x" _ ev->event_handler); |
Igor Sysoev | 7300977 | 2003-02-06 17:21:13 +0000 | [diff] [blame] | 190 | |
Igor Sysoev | 1c13c66 | 2003-05-20 15:37:55 +0000 | [diff] [blame] | 191 | ev->event_handler(ev); |
Igor Sysoev | 7300977 | 2003-02-06 17:21:13 +0000 | [diff] [blame] | 192 | } |
| 193 | |
Igor Sysoev | 7300977 | 2003-02-06 17:21:13 +0000 | [diff] [blame] | 194 | return NGX_OK; |
| 195 | } |
Igor Sysoev | e4a2526 | 2003-06-06 14:59:20 +0000 | [diff] [blame^] | 196 | |
| 197 | |
| 198 | static void *ngx_iocp_create_conf(ngx_pool_t *pool) |
| 199 | { |
| 200 | ngx_iocp_conf_t *cf; |
| 201 | |
| 202 | ngx_test_null(cf, ngx_palloc(pool, sizeof(ngx_iocp_conf_t)), |
| 203 | NGX_CONF_ERROR); |
| 204 | |
| 205 | cf->threads = NGX_CONF_UNSET; |
| 206 | |
| 207 | return cf; |
| 208 | } |
| 209 | |
| 210 | |
| 211 | static char *ngx_iocp_init_conf(ngx_pool_t *pool, void *conf) |
| 212 | { |
| 213 | ngx_iocp_conf_t *cf = conf; |
| 214 | |
| 215 | ngx_conf_init_value(cf->threads, 0); |
| 216 | |
| 217 | return NGX_CONF_OK; |
| 218 | } |