mirror of https://github.com/F-Stack/f-stack.git
Nginx: support kernel network stack, so we can do what fstack can't do,
e.g. unix socket, ipc (with APP on kernel network stack), packet from kernel network stack. 1. Add a new directive kernel_network_stack : Syntax: kernel_network_stack on | off; Default: kernel_network_stack off; Context: http, server This directive is available only when NGX_HAVE_FF_STACK is defined. Determines whether server should run on kernel network stack or fstack. 2. Use a simpler and more effective solution to discriminate fstack fd(file descriptor, only socket for now) from kernel fd.
This commit is contained in:
parent
1b895bd9c2
commit
3ce9eefdd7
|
@ -41,6 +41,10 @@ http {
|
||||||
listen 80;
|
listen 80;
|
||||||
server_name localhost;
|
server_name localhost;
|
||||||
|
|
||||||
|
# bulid server on kernel network stack
|
||||||
|
#
|
||||||
|
#kernel_network_stack on;
|
||||||
|
|
||||||
#charset koi8-r;
|
#charset koi8-r;
|
||||||
|
|
||||||
access_log /dev/null;
|
access_log /dev/null;
|
||||||
|
|
|
@ -376,6 +376,10 @@ ngx_set_inherited_sockets(ngx_cycle_t *cycle)
|
||||||
return NGX_OK;
|
return NGX_OK;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#if (NGX_HAVE_FSTACK)
|
||||||
|
extern int
|
||||||
|
fstack_territory(int domain, int type, int protocol);
|
||||||
|
#endif
|
||||||
|
|
||||||
ngx_int_t
|
ngx_int_t
|
||||||
ngx_open_listening_sockets(ngx_cycle_t *cycle)
|
ngx_open_listening_sockets(ngx_cycle_t *cycle)
|
||||||
|
@ -404,6 +408,36 @@ ngx_open_listening_sockets(ngx_cycle_t *cycle)
|
||||||
ls = cycle->listening.elts;
|
ls = cycle->listening.elts;
|
||||||
for (i = 0; i < cycle->listening.nelts; i++) {
|
for (i = 0; i < cycle->listening.nelts; i++) {
|
||||||
|
|
||||||
|
#if (NGX_HAVE_FSTACK)
|
||||||
|
|
||||||
|
if (ngx_process <= NGX_PROCESS_MASTER) {
|
||||||
|
|
||||||
|
/* process master, kernel network stack*/
|
||||||
|
if (!ls[i].belong_to_aeds) {
|
||||||
|
/* We should continue to process the listening socket,
|
||||||
|
if it is not supported by fstack.*/
|
||||||
|
if (fstack_territory(ls[i].sockaddr->sa_family, ls[i].type, 0)) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} else if (NGX_PROCESS_WORKER == ngx_process) {
|
||||||
|
/* process worker, fstack */
|
||||||
|
if (ls[i].belong_to_aeds) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!fstack_territory(ls[i].sockaddr->sa_family, ls[i].type, 0)) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
ngx_log_error(NGX_LOG_ALERT, cycle->log, 0,
|
||||||
|
"unexpected process type: %d, ignored",
|
||||||
|
ngx_process);
|
||||||
|
exit(1);
|
||||||
|
}
|
||||||
|
|
||||||
|
#endif
|
||||||
|
|
||||||
if (ls[i].ignore) {
|
if (ls[i].ignore) {
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
@ -971,6 +1005,15 @@ ngx_close_listening_sockets(ngx_cycle_t *cycle)
|
||||||
ls = cycle->listening.elts;
|
ls = cycle->listening.elts;
|
||||||
for (i = 0; i < cycle->listening.nelts; i++) {
|
for (i = 0; i < cycle->listening.nelts; i++) {
|
||||||
|
|
||||||
|
#if (NGX_HAVE_FSTACK)
|
||||||
|
|
||||||
|
// No need to deal with, just skip
|
||||||
|
if (fstack_territory(ls[i].sockaddr->sa_family, ls[i].type, 0)) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
#endif //(NGX_HAVE_FSTACK)
|
||||||
|
|
||||||
c = ls[i].connection;
|
c = ls[i].connection;
|
||||||
|
|
||||||
if (c) {
|
if (c) {
|
||||||
|
@ -1032,6 +1075,9 @@ ngx_get_connection(ngx_socket_t s, ngx_log_t *log)
|
||||||
ngx_uint_t instance;
|
ngx_uint_t instance;
|
||||||
ngx_event_t *rev, *wev;
|
ngx_event_t *rev, *wev;
|
||||||
ngx_connection_t *c;
|
ngx_connection_t *c;
|
||||||
|
#if (NGX_HAVE_FSTACK)
|
||||||
|
ngx_atomic_uint_t success;
|
||||||
|
#endif
|
||||||
|
|
||||||
/* disable warning: Win32 SOCKET is u_int while UNIX socket is int */
|
/* disable warning: Win32 SOCKET is u_int while UNIX socket is int */
|
||||||
|
|
||||||
|
@ -1042,7 +1088,35 @@ ngx_get_connection(ngx_socket_t s, ngx_log_t *log)
|
||||||
s, ngx_cycle->files_n);
|
s, ngx_cycle->files_n);
|
||||||
return NULL;
|
return NULL;
|
||||||
}
|
}
|
||||||
|
#if (NGX_HAVE_FSTACK)
|
||||||
|
#ifndef unlikely
|
||||||
|
#define unlikely(x) __builtin_expect((x),0)
|
||||||
|
#endif
|
||||||
|
/* move ngx_cycle->free_connections atomically */
|
||||||
|
do {
|
||||||
|
/* Restore n as it may change every loop */
|
||||||
|
c = ngx_cycle->free_connections;
|
||||||
|
|
||||||
|
if (c == NULL) {
|
||||||
|
ngx_drain_connections((ngx_cycle_t *) ngx_cycle);
|
||||||
|
c = ngx_cycle->free_connections;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (c == NULL) {
|
||||||
|
ngx_log_error(NGX_LOG_ALERT, log, 0,
|
||||||
|
"%ui worker_connections are not enough",
|
||||||
|
ngx_cycle->connection_n);
|
||||||
|
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
success = ngx_atomic_cmp_set(&ngx_cycle->free_connections, c,
|
||||||
|
c->data);
|
||||||
|
|
||||||
|
} while (unlikely(success == 0));
|
||||||
|
|
||||||
|
ngx_memory_barrier();
|
||||||
|
#else
|
||||||
c = ngx_cycle->free_connections;
|
c = ngx_cycle->free_connections;
|
||||||
|
|
||||||
if (c == NULL) {
|
if (c == NULL) {
|
||||||
|
@ -1059,6 +1133,7 @@ ngx_get_connection(ngx_socket_t s, ngx_log_t *log)
|
||||||
}
|
}
|
||||||
|
|
||||||
ngx_cycle->free_connections = c->data;
|
ngx_cycle->free_connections = c->data;
|
||||||
|
#endif
|
||||||
ngx_cycle->free_connection_n--;
|
ngx_cycle->free_connection_n--;
|
||||||
|
|
||||||
if (ngx_cycle->files && ngx_cycle->files[s] == NULL) {
|
if (ngx_cycle->files && ngx_cycle->files[s] == NULL) {
|
||||||
|
@ -1098,8 +1173,25 @@ ngx_get_connection(ngx_socket_t s, ngx_log_t *log)
|
||||||
void
|
void
|
||||||
ngx_free_connection(ngx_connection_t *c)
|
ngx_free_connection(ngx_connection_t *c)
|
||||||
{
|
{
|
||||||
c->data = ngx_cycle->free_connections;
|
#if (NGX_HAVE_FSTACK)
|
||||||
|
ngx_atomic_uint_t success;
|
||||||
|
|
||||||
|
/* move ngx_cycle->free_connections atomically */
|
||||||
|
do {
|
||||||
|
/* Restore n as it may change every loop */
|
||||||
|
c->data = ngx_cycle->free_connections;
|
||||||
|
|
||||||
|
success = ngx_atomic_cmp_set(&ngx_cycle->free_connections, c->data,
|
||||||
|
c);
|
||||||
|
|
||||||
|
} while (unlikely(success == 0));
|
||||||
|
|
||||||
|
ngx_memory_barrier();
|
||||||
|
|
||||||
|
#else
|
||||||
ngx_cycle->free_connections = c;
|
ngx_cycle->free_connections = c;
|
||||||
|
|
||||||
|
#endif
|
||||||
ngx_cycle->free_connection_n++;
|
ngx_cycle->free_connection_n++;
|
||||||
|
|
||||||
if (ngx_cycle->files && ngx_cycle->files[c->fd] == c) {
|
if (ngx_cycle->files && ngx_cycle->files[c->fd] == c) {
|
||||||
|
@ -1129,7 +1221,11 @@ ngx_close_connection(ngx_connection_t *c)
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!c->shared) {
|
if (!c->shared) {
|
||||||
|
#if (NGX_HAVE_FSTACK)
|
||||||
|
if (ngx_event_actions.del_conn) {
|
||||||
|
#else
|
||||||
if (ngx_del_conn) {
|
if (ngx_del_conn) {
|
||||||
|
#endif
|
||||||
ngx_del_conn(c, NGX_CLOSE_EVENT);
|
ngx_del_conn(c, NGX_CLOSE_EVENT);
|
||||||
|
|
||||||
} else {
|
} else {
|
||||||
|
|
|
@ -87,6 +87,9 @@ struct ngx_listening_s {
|
||||||
int fastopen;
|
int fastopen;
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
#if (NGX_HAVE_FSTACK)
|
||||||
|
unsigned belong_to_aeds:1;
|
||||||
|
#endif
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
||||||
|
|
|
@ -609,11 +609,9 @@ ngx_init_cycle(ngx_cycle_t *old_cycle)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
#if (!NGX_HAVE_FSTACK)
|
|
||||||
if (ngx_open_listening_sockets(cycle) != NGX_OK) {
|
if (ngx_open_listening_sockets(cycle) != NGX_OK) {
|
||||||
goto failed;
|
goto failed;
|
||||||
}
|
}
|
||||||
#endif
|
|
||||||
|
|
||||||
if (!ngx_test_config) {
|
if (!ngx_test_config) {
|
||||||
ngx_configure_listening_sockets(cycle);
|
ngx_configure_listening_sockets(cycle);
|
||||||
|
|
|
@ -4512,7 +4512,11 @@ ngx_tcp_connect(ngx_resolver_connection_t *rec)
|
||||||
|
|
||||||
c->number = ngx_atomic_fetch_add(ngx_connection_counter, 1);
|
c->number = ngx_atomic_fetch_add(ngx_connection_counter, 1);
|
||||||
|
|
||||||
|
#if (NGX_HAVE_FSTACK)
|
||||||
|
if (ngx_event_actions.add_conn) {
|
||||||
|
#else
|
||||||
if (ngx_add_conn) {
|
if (ngx_add_conn) {
|
||||||
|
#endif
|
||||||
if (ngx_add_conn(c) == NGX_ERROR) {
|
if (ngx_add_conn(c) == NGX_ERROR) {
|
||||||
goto failed;
|
goto failed;
|
||||||
}
|
}
|
||||||
|
@ -4564,7 +4568,11 @@ ngx_tcp_connect(ngx_resolver_connection_t *rec)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#if (NGX_HAVE_FSTACK)
|
||||||
|
if (ngx_event_actions.add_conn) {
|
||||||
|
#else
|
||||||
if (ngx_add_conn) {
|
if (ngx_add_conn) {
|
||||||
|
#endif
|
||||||
if (rc == -1) {
|
if (rc == -1) {
|
||||||
|
|
||||||
/* NGX_EINPROGRESS */
|
/* NGX_EINPROGRESS */
|
||||||
|
|
|
@ -36,7 +36,9 @@
|
||||||
static void * ngx_ff_channel_create_conf(ngx_cycle_t *cycle);
|
static void * ngx_ff_channel_create_conf(ngx_cycle_t *cycle);
|
||||||
static char * ngx_ff_channel_init_conf(ngx_cycle_t *cycle,
|
static char * ngx_ff_channel_init_conf(ngx_cycle_t *cycle,
|
||||||
void *conf);
|
void *conf);
|
||||||
static ngx_int_t ngx_ff_epoll_init(ngx_cycle_t *cycle);
|
static ngx_int_t ngx_ff_channel_init_process(ngx_cycle_t *cycle);
|
||||||
|
static void ngx_ff_channel_exit_process(ngx_cycle_t *cycle);
|
||||||
|
static ngx_int_t ngx_ff_epoll_init(ngx_cycle_t *cycle, ngx_msec_t timer);
|
||||||
static ngx_int_t ngx_ff_epoll_add_event(ngx_event_t *ev,
|
static ngx_int_t ngx_ff_epoll_add_event(ngx_event_t *ev,
|
||||||
ngx_int_t event, ngx_uint_t flags);
|
ngx_int_t event, ngx_uint_t flags);
|
||||||
static ngx_int_t ngx_ff_epoll_del_event(ngx_event_t *ev,
|
static ngx_int_t ngx_ff_epoll_del_event(ngx_event_t *ev,
|
||||||
|
@ -49,10 +51,15 @@ static void ngx_ff_worker_channel_handler(ngx_event_t *ev);
|
||||||
static void *ngx_ff_channel_thread_main(void *args);
|
static void *ngx_ff_channel_thread_main(void *args);
|
||||||
static ngx_int_t ngx_ff_add_channel_event(ngx_cycle_t *cycle,
|
static ngx_int_t ngx_ff_add_channel_event(ngx_cycle_t *cycle,
|
||||||
ngx_fd_t fd, ngx_int_t event, ngx_event_handler_pt handler);
|
ngx_fd_t fd, ngx_int_t event, ngx_event_handler_pt handler);
|
||||||
static ngx_int_t ngx_ff_process_channel_events(ngx_cycle_t *cycle);
|
static void ngx_ff_process_events_and_timers(ngx_cycle_t *cycle);
|
||||||
|
|
||||||
ngx_int_t ngx_ff_start_worker_channel(ngx_cycle_t *cycle,
|
ngx_int_t ngx_ff_start_worker_channel(ngx_cycle_t *cycle,
|
||||||
ngx_fd_t fd, ngx_int_t event);
|
ngx_fd_t fd, ngx_int_t event);
|
||||||
|
|
||||||
|
void ngx_aeds_cancel_timers(void);
|
||||||
|
void ngx_aeds_expire_timers(void);
|
||||||
|
ngx_msec_t ngx_aeds_find_timer(void);
|
||||||
|
void ngx_aeds_cancel_timers(void);
|
||||||
|
|
||||||
struct channel_thread_args {
|
struct channel_thread_args {
|
||||||
ngx_cycle_t *cycle;
|
ngx_cycle_t *cycle;
|
||||||
|
@ -69,6 +76,9 @@ static struct epoll_event *event_list;
|
||||||
static ngx_uint_t nevents;
|
static ngx_uint_t nevents;
|
||||||
static ngx_connection_t *channel_connection;
|
static ngx_connection_t *channel_connection;
|
||||||
|
|
||||||
|
#include <semaphore.h>
|
||||||
|
static sem_t sem;
|
||||||
|
|
||||||
typedef struct {
|
typedef struct {
|
||||||
ngx_uint_t events;
|
ngx_uint_t events;
|
||||||
} ngx_ff_channel_conf_t;
|
} ngx_ff_channel_conf_t;
|
||||||
|
@ -91,10 +101,10 @@ ngx_module_t ngx_ff_channel_module = {
|
||||||
NGX_CORE_MODULE, /* module type */
|
NGX_CORE_MODULE, /* module type */
|
||||||
NULL, /* init master */
|
NULL, /* init master */
|
||||||
NULL, /* init module */
|
NULL, /* init module */
|
||||||
ngx_ff_epoll_init, /* init process */
|
ngx_ff_channel_init_process, /* init process */
|
||||||
NULL, /* init thread */
|
NULL, /* init thread */
|
||||||
NULL, /* exit thread */
|
NULL, /* exit thread */
|
||||||
NULL, /* exit process */
|
ngx_ff_channel_exit_process, /* exit process */
|
||||||
NULL, /* exit master */
|
NULL, /* exit master */
|
||||||
NGX_MODULE_V1_PADDING
|
NGX_MODULE_V1_PADDING
|
||||||
};
|
};
|
||||||
|
@ -119,8 +129,33 @@ ngx_ff_channel_init_conf(ngx_cycle_t *cycle, void *conf)
|
||||||
return NGX_CONF_OK;
|
return NGX_CONF_OK;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
static ngx_int_t ngx_ff_channel_init_process(ngx_cycle_t *cycle)
|
||||||
|
{
|
||||||
|
if (sem_init(&sem, 0, 0) != 0)
|
||||||
|
{
|
||||||
|
return NGX_ERROR;
|
||||||
|
}
|
||||||
|
|
||||||
|
return NGX_OK;
|
||||||
|
}
|
||||||
|
|
||||||
|
static void ngx_ff_channel_exit_process(ngx_cycle_t *cycle)
|
||||||
|
{
|
||||||
|
struct timespec ts;
|
||||||
|
|
||||||
|
if (clock_gettime( CLOCK_REALTIME,&ts ) < 0)
|
||||||
|
return;
|
||||||
|
|
||||||
|
//5s
|
||||||
|
ts.tv_sec += 4;
|
||||||
|
|
||||||
|
(void) sem_timedwait(&sem, &ts);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
static ngx_int_t
|
static ngx_int_t
|
||||||
ngx_ff_epoll_init(ngx_cycle_t *cycle)
|
ngx_ff_epoll_init(ngx_cycle_t *cycle, ngx_msec_t timer)
|
||||||
{
|
{
|
||||||
if (ep == -1) {
|
if (ep == -1) {
|
||||||
ep = epoll_create(1);
|
ep = epoll_create(1);
|
||||||
|
@ -413,6 +448,7 @@ ngx_ff_epoll_process_events(ngx_cycle_t *cycle,
|
||||||
|
|
||||||
if ((revents & EPOLLIN) && rev->active) {
|
if ((revents & EPOLLIN) && rev->active) {
|
||||||
rev->ready = 1;
|
rev->ready = 1;
|
||||||
|
rev->available = 1;
|
||||||
rev->handler(rev);
|
rev->handler(rev);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -531,6 +567,8 @@ ngx_ff_add_channel_event(ngx_cycle_t *cycle, ngx_fd_t fd,
|
||||||
rev->channel = 1;
|
rev->channel = 1;
|
||||||
wev->channel = 1;
|
wev->channel = 1;
|
||||||
|
|
||||||
|
rev->belong_to_aeds = wev->belong_to_aeds = 1;
|
||||||
|
|
||||||
ev = (event == NGX_READ_EVENT) ? rev : wev;
|
ev = (event == NGX_READ_EVENT) ? rev : wev;
|
||||||
ev->handler = handler;
|
ev->handler = handler;
|
||||||
|
|
||||||
|
@ -634,21 +672,48 @@ ngx_ff_channel_thread_main(void *args)
|
||||||
}
|
}
|
||||||
|
|
||||||
for (;;) {
|
for (;;) {
|
||||||
ngx_ff_process_channel_events(cycle);
|
ngx_ff_process_events_and_timers(cycle);
|
||||||
if (thread_quit) {
|
if (thread_quit) {
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
ngx_aeds_cancel_timers();
|
||||||
|
|
||||||
ngx_free(cta);
|
ngx_free(cta);
|
||||||
|
|
||||||
return NULL;
|
return NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
static ngx_int_t
|
static void
|
||||||
ngx_ff_process_channel_events(ngx_cycle_t *cycle)
|
ngx_ff_process_events_and_timers(ngx_cycle_t *cycle)
|
||||||
{
|
{
|
||||||
return ngx_ff_epoll_process_events(cycle, 500, NGX_UPDATE_TIME);
|
ngx_uint_t flags;
|
||||||
|
ngx_msec_t timer, delta;
|
||||||
|
|
||||||
|
timer = ngx_aeds_find_timer();
|
||||||
|
flags = NGX_UPDATE_TIME;
|
||||||
|
|
||||||
|
/* handle signals from master in case of network inactivity */
|
||||||
|
|
||||||
|
if (timer == NGX_TIMER_INFINITE || timer > 500) {
|
||||||
|
timer = 500;
|
||||||
|
}
|
||||||
|
|
||||||
|
delta = ngx_current_msec;
|
||||||
|
|
||||||
|
(void) ngx_ff_epoll_process_events(cycle, timer, flags);
|
||||||
|
|
||||||
|
delta = ngx_current_msec - delta;
|
||||||
|
|
||||||
|
ngx_event_process_posted(cycle, &ngx_posted_accept_events_of_aeds);
|
||||||
|
|
||||||
|
if (delta) {
|
||||||
|
ngx_aeds_expire_timers();
|
||||||
|
}
|
||||||
|
|
||||||
|
ngx_event_process_posted(cycle, &ngx_posted_events_of_aeds);
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
ngx_int_t
|
ngx_int_t
|
||||||
|
@ -681,4 +746,17 @@ ngx_ff_start_worker_channel(ngx_cycle_t *cycle, ngx_fd_t fd,
|
||||||
return NGX_OK;
|
return NGX_OK;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
ngx_event_actions_t ngx_event_actions_dy = {
|
||||||
|
ngx_ff_epoll_add_event, /* add an event */
|
||||||
|
ngx_ff_epoll_del_event, /* delete an event */
|
||||||
|
ngx_ff_epoll_add_event, /* enable an event */
|
||||||
|
ngx_ff_epoll_add_event, /* disable an event */
|
||||||
|
NULL, /* add an connection */
|
||||||
|
NULL, /* delete an connection */
|
||||||
|
NULL, /* trigger a notify */
|
||||||
|
ngx_ff_epoll_process_events, /* process the events */
|
||||||
|
ngx_ff_epoll_init, /* init the events */
|
||||||
|
NULL, /* done the events */
|
||||||
|
};
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
|
|
@ -129,6 +129,31 @@ static int inited;
|
||||||
real_##func; \
|
real_##func; \
|
||||||
})
|
})
|
||||||
|
|
||||||
|
extern intptr_t ngx_max_sockets;
|
||||||
|
|
||||||
|
/*-
|
||||||
|
* Make sockfd assigned by the fstack plus the value of maximum kernel socket.
|
||||||
|
* so we can tell them apart according to different scopes.
|
||||||
|
* Solve the condominium ownership at Application Layer and obtain more freedom.
|
||||||
|
* fstack tried to do this by 'fd_reserve', unfortunately, it doesn't work well.
|
||||||
|
*/
|
||||||
|
static inline int convert_ffd(int sockfd) {
|
||||||
|
return sockfd + ngx_max_sockets;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Restore socket fd. */
|
||||||
|
static inline int restore_ffd(int sockfd) {
|
||||||
|
if(sockfd <= ngx_max_sockets) {
|
||||||
|
return sockfd;
|
||||||
|
}
|
||||||
|
|
||||||
|
return sockfd - ngx_max_sockets;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Tell whether a 'sockfd' belongs to fstack. */
|
||||||
|
static inline int is_ffd(int sockfd) {
|
||||||
|
return sockfd >= ngx_max_sockets;
|
||||||
|
}
|
||||||
|
|
||||||
// proc_type, 1: primary, 0: secondary.
|
// proc_type, 1: primary, 0: secondary.
|
||||||
int
|
int
|
||||||
|
@ -151,8 +176,15 @@ ff_mod_init(const char *conf, int proc_id, int proc_type) {
|
||||||
}
|
}
|
||||||
|
|
||||||
rc = ff_init(ff_argc, ff_argv);
|
rc = ff_init(ff_argc, ff_argv);
|
||||||
if (rc == 0)
|
if (rc == 0) {
|
||||||
|
/* Ensure that the socket we converted does not exceed the maximum value of 'int' */
|
||||||
|
if(ngx_max_sockets + (unsigned)ff_getmaxfd() > INT_MAX)
|
||||||
|
{
|
||||||
|
rc = -1;
|
||||||
|
}
|
||||||
|
|
||||||
inited = 1;
|
inited = 1;
|
||||||
|
}
|
||||||
|
|
||||||
for (i = 0; i < ff_argc; i++) {
|
for (i = 0; i < ff_argc; i++) {
|
||||||
free(ff_argv[i]);
|
free(ff_argv[i]);
|
||||||
|
@ -163,9 +195,23 @@ ff_mod_init(const char *conf, int proc_id, int proc_type) {
|
||||||
return rc;
|
return rc;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/*-
|
||||||
|
* Verify whether the socket is supported by fstack or not.
|
||||||
|
*/
|
||||||
|
int
|
||||||
|
fstack_territory(int domain, int type, int protocol)
|
||||||
|
{
|
||||||
|
if ((AF_INET != domain) || (SOCK_STREAM != type && SOCK_DGRAM != type)) {
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
int
|
int
|
||||||
socket(int domain, int type, int protocol)
|
socket(int domain, int type, int protocol)
|
||||||
{
|
{
|
||||||
|
int sock;
|
||||||
if (unlikely(inited == 0)) {
|
if (unlikely(inited == 0)) {
|
||||||
return SYSCALL(socket)(domain, type, protocol);
|
return SYSCALL(socket)(domain, type, protocol);
|
||||||
}
|
}
|
||||||
|
@ -174,249 +220,233 @@ socket(int domain, int type, int protocol)
|
||||||
return SYSCALL(socket)(domain, type, protocol);
|
return SYSCALL(socket)(domain, type, protocol);
|
||||||
}
|
}
|
||||||
|
|
||||||
return ff_socket(domain, type, protocol);
|
sock = ff_socket(domain, type, protocol);
|
||||||
|
|
||||||
|
if (sock != -1) {
|
||||||
|
sock = convert_ffd(sock);
|
||||||
|
}
|
||||||
|
|
||||||
|
return sock;
|
||||||
}
|
}
|
||||||
|
|
||||||
int
|
int
|
||||||
bind(int sockfd, const struct sockaddr *addr, socklen_t addrlen)
|
bind(int sockfd, const struct sockaddr *addr, socklen_t addrlen)
|
||||||
{
|
{
|
||||||
if (unlikely(inited == 0)) {
|
if(is_ffd(sockfd)){
|
||||||
return SYSCALL(bind)(sockfd, addr, addrlen);
|
sockfd = restore_ffd(sockfd);
|
||||||
}
|
assert(ff_fdisused(sockfd));
|
||||||
|
|
||||||
if (ff_fdisused(sockfd)) {
|
|
||||||
return ff_bind(sockfd, (struct linux_sockaddr *)addr, addrlen);
|
return ff_bind(sockfd, (struct linux_sockaddr *)addr, addrlen);
|
||||||
} else {
|
|
||||||
return SYSCALL(bind)(sockfd, addr, addrlen);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
return SYSCALL(bind)(sockfd, addr, addrlen);
|
||||||
}
|
}
|
||||||
|
|
||||||
int
|
int
|
||||||
connect(int sockfd, const struct sockaddr *addr, socklen_t addrlen)
|
connect(int sockfd, const struct sockaddr *addr, socklen_t addrlen)
|
||||||
{
|
{
|
||||||
if (unlikely(inited == 0)) {
|
if(is_ffd(sockfd)){
|
||||||
return SYSCALL(connect)(sockfd, addr, addrlen);
|
sockfd = restore_ffd(sockfd);
|
||||||
}
|
assert(ff_fdisused(sockfd));
|
||||||
|
|
||||||
if (ff_fdisused(sockfd)) {
|
|
||||||
return ff_connect(sockfd, (struct linux_sockaddr *)addr, addrlen);
|
return ff_connect(sockfd, (struct linux_sockaddr *)addr, addrlen);
|
||||||
} else {
|
|
||||||
return SYSCALL(connect)(sockfd, addr, addrlen);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
return SYSCALL(connect)(sockfd, addr, addrlen);
|
||||||
}
|
}
|
||||||
|
|
||||||
ssize_t
|
ssize_t
|
||||||
send(int sockfd, const void *buf, size_t len, int flags)
|
send(int sockfd, const void *buf, size_t len, int flags)
|
||||||
{
|
{
|
||||||
if (unlikely(inited == 0)) {
|
if(is_ffd(sockfd)){
|
||||||
return SYSCALL(send)(sockfd, buf, len, flags);
|
sockfd = restore_ffd(sockfd);
|
||||||
}
|
assert(ff_fdisused(sockfd));
|
||||||
|
|
||||||
if (ff_fdisused(sockfd)) {
|
|
||||||
return ff_send(sockfd, buf, len, flags);
|
return ff_send(sockfd, buf, len, flags);
|
||||||
} else {
|
|
||||||
return SYSCALL(send)(sockfd, buf, len, flags);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
return SYSCALL(send)(sockfd, buf, len, flags);
|
||||||
}
|
}
|
||||||
|
|
||||||
ssize_t
|
ssize_t
|
||||||
sendto(int sockfd, const void *buf, size_t len, int flags,
|
sendto(int sockfd, const void *buf, size_t len, int flags,
|
||||||
const struct sockaddr *dest_addr, socklen_t addrlen)
|
const struct sockaddr *dest_addr, socklen_t addrlen)
|
||||||
{
|
{
|
||||||
if (unlikely(inited == 0)) {
|
if(is_ffd(sockfd)){
|
||||||
return SYSCALL(sendto)(sockfd, buf, len, flags, dest_addr, addrlen);
|
sockfd = restore_ffd(sockfd);
|
||||||
}
|
assert(ff_fdisused(sockfd));
|
||||||
|
|
||||||
if (ff_fdisused(sockfd)) {
|
|
||||||
return ff_sendto(sockfd, buf, len, flags,
|
return ff_sendto(sockfd, buf, len, flags,
|
||||||
(struct linux_sockaddr *)dest_addr, addrlen);
|
(struct linux_sockaddr *)dest_addr, addrlen);
|
||||||
} else {
|
|
||||||
return SYSCALL(sendto)(sockfd, buf, len, flags, dest_addr, addrlen);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
return SYSCALL(sendto)(sockfd, buf, len, flags, dest_addr, addrlen);
|
||||||
}
|
}
|
||||||
|
|
||||||
ssize_t
|
ssize_t
|
||||||
sendmsg(int sockfd, const struct msghdr *msg, int flags)
|
sendmsg(int sockfd, const struct msghdr *msg, int flags)
|
||||||
{
|
{
|
||||||
if (unlikely(inited == 0)) {
|
if(is_ffd(sockfd)){
|
||||||
return SYSCALL(sendmsg)(sockfd, msg, flags);
|
sockfd = restore_ffd(sockfd);
|
||||||
}
|
assert(ff_fdisused(sockfd));
|
||||||
|
|
||||||
if (ff_fdisused(sockfd)) {
|
|
||||||
return ff_sendmsg(sockfd, msg, flags);
|
return ff_sendmsg(sockfd, msg, flags);
|
||||||
} else {
|
|
||||||
return SYSCALL(sendmsg)(sockfd, msg, flags);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
return SYSCALL(sendmsg)(sockfd, msg, flags);
|
||||||
}
|
}
|
||||||
|
|
||||||
ssize_t
|
ssize_t
|
||||||
recv(int sockfd, void *buf, size_t len, int flags)
|
recv(int sockfd, void *buf, size_t len, int flags)
|
||||||
{
|
{
|
||||||
if (unlikely(inited == 0)) {
|
if(is_ffd(sockfd)){
|
||||||
return SYSCALL(recv)(sockfd, buf, len, flags);
|
sockfd = restore_ffd(sockfd);
|
||||||
}
|
assert(ff_fdisused(sockfd));
|
||||||
|
|
||||||
if (ff_fdisused(sockfd)) {
|
|
||||||
return ff_recv(sockfd, buf, len, flags);
|
return ff_recv(sockfd, buf, len, flags);
|
||||||
} else {
|
|
||||||
return SYSCALL(recv)(sockfd, buf, len, flags);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
return SYSCALL(recv)(sockfd, buf, len, flags);
|
||||||
}
|
}
|
||||||
|
|
||||||
int
|
int
|
||||||
listen(int sockfd, int backlog)
|
listen(int sockfd, int backlog)
|
||||||
{
|
{
|
||||||
if (unlikely(inited == 0)) {
|
if(is_ffd(sockfd)){
|
||||||
return SYSCALL(listen)(sockfd, backlog);
|
sockfd = restore_ffd(sockfd);
|
||||||
}
|
assert(ff_fdisused(sockfd));
|
||||||
|
|
||||||
if (ff_fdisused(sockfd)) {
|
|
||||||
return ff_listen(sockfd, backlog);
|
return ff_listen(sockfd, backlog);
|
||||||
} else {
|
|
||||||
return SYSCALL(listen)(sockfd, backlog);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
return SYSCALL(listen)(sockfd, backlog);
|
||||||
}
|
}
|
||||||
|
|
||||||
int
|
int
|
||||||
getsockopt(int sockfd, int level, int optname,
|
getsockopt(int sockfd, int level, int optname,
|
||||||
void *optval, socklen_t *optlen)
|
void *optval, socklen_t *optlen)
|
||||||
{
|
{
|
||||||
if (unlikely(inited == 0)) {
|
if(is_ffd(sockfd)){
|
||||||
return SYSCALL(getsockopt)(sockfd, level, optname, optval, optlen);
|
sockfd = restore_ffd(sockfd);
|
||||||
}
|
assert(ff_fdisused(sockfd));
|
||||||
|
|
||||||
if (ff_fdisused(sockfd)) {
|
|
||||||
return ff_getsockopt(sockfd, level, optname, optval, optlen);
|
return ff_getsockopt(sockfd, level, optname, optval, optlen);
|
||||||
} else {
|
|
||||||
return SYSCALL(getsockopt)(sockfd, level, optname, optval, optlen);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
return SYSCALL(getsockopt)(sockfd, level, optname, optval, optlen);
|
||||||
}
|
}
|
||||||
|
|
||||||
int
|
int
|
||||||
setsockopt (int sockfd, int level, int optname,
|
setsockopt (int sockfd, int level, int optname,
|
||||||
const void *optval, socklen_t optlen)
|
const void *optval, socklen_t optlen)
|
||||||
{
|
{
|
||||||
if (unlikely(inited == 0)) {
|
if(is_ffd(sockfd)){
|
||||||
return SYSCALL(setsockopt)(sockfd, level, optname, optval, optlen);
|
sockfd = restore_ffd(sockfd);
|
||||||
}
|
assert(ff_fdisused(sockfd));
|
||||||
|
|
||||||
if (ff_fdisused(sockfd)) {
|
|
||||||
return ff_setsockopt(sockfd, level, optname, optval, optlen);
|
return ff_setsockopt(sockfd, level, optname, optval, optlen);
|
||||||
} else {
|
|
||||||
return SYSCALL(setsockopt)(sockfd, level, optname, optval, optlen);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
return SYSCALL(setsockopt)(sockfd, level, optname, optval, optlen);
|
||||||
}
|
}
|
||||||
|
|
||||||
int
|
int
|
||||||
accept(int sockfd, struct sockaddr *addr, socklen_t *addrlen)
|
accept(int sockfd, struct sockaddr *addr, socklen_t *addrlen)
|
||||||
{
|
{
|
||||||
if (unlikely(inited == 0)) {
|
int rc;
|
||||||
return SYSCALL(accept)(sockfd, addr, addrlen);
|
if(is_ffd(sockfd)){
|
||||||
}
|
sockfd = restore_ffd(sockfd);
|
||||||
|
assert(ff_fdisused(sockfd));
|
||||||
|
rc = ff_accept(sockfd, (struct linux_sockaddr *)addr, addrlen);
|
||||||
|
if (rc != -1) {
|
||||||
|
rc = convert_ffd(rc);
|
||||||
|
}
|
||||||
|
|
||||||
if (ff_fdisused(sockfd)) {
|
return rc;
|
||||||
return ff_accept(sockfd, (struct linux_sockaddr *)addr, addrlen);
|
|
||||||
} else {
|
|
||||||
return SYSCALL(accept)(sockfd, addr, addrlen);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
return SYSCALL(accept)(sockfd, addr, addrlen);
|
||||||
}
|
}
|
||||||
|
|
||||||
int
|
int
|
||||||
accept4(int sockfd, struct sockaddr *addr, socklen_t *addrlen, int flags)
|
accept4(int sockfd, struct sockaddr *addr, socklen_t *addrlen, int flags)
|
||||||
{
|
{
|
||||||
if (unlikely(inited == 0)) {
|
int rc;
|
||||||
return SYSCALL(accept4)(sockfd, addr, addrlen, flags);
|
if(is_ffd(sockfd)){
|
||||||
}
|
sockfd = restore_ffd(sockfd);
|
||||||
|
assert(ff_fdisused(sockfd));
|
||||||
|
rc = ff_accept(sockfd, (struct linux_sockaddr *)addr, addrlen);
|
||||||
|
if (rc != -1) {
|
||||||
|
rc = convert_ffd(rc);
|
||||||
|
}
|
||||||
|
|
||||||
if (ff_fdisused(sockfd)) {
|
return rc;
|
||||||
return ff_accept(sockfd, (struct linux_sockaddr *)addr, addrlen);
|
|
||||||
} else {
|
|
||||||
return SYSCALL(accept4)(sockfd, addr, addrlen, flags);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
return SYSCALL(accept4)(sockfd, addr, addrlen, flags);
|
||||||
}
|
}
|
||||||
|
|
||||||
int
|
int
|
||||||
close(int sockfd)
|
close(int sockfd)
|
||||||
{
|
{
|
||||||
if (unlikely(inited == 0)) {
|
if(is_ffd(sockfd)){
|
||||||
return SYSCALL(close)(sockfd);
|
sockfd = restore_ffd(sockfd);
|
||||||
}
|
assert(ff_fdisused(sockfd));
|
||||||
|
|
||||||
if (ff_fdisused(sockfd)) {
|
|
||||||
return ff_close(sockfd);
|
return ff_close(sockfd);
|
||||||
} else {
|
|
||||||
return SYSCALL(close)(sockfd);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
return SYSCALL(close)(sockfd);
|
||||||
}
|
}
|
||||||
|
|
||||||
ssize_t
|
ssize_t
|
||||||
writev(int sockfd, const struct iovec *iov, int iovcnt)
|
writev(int sockfd, const struct iovec *iov, int iovcnt)
|
||||||
{
|
{
|
||||||
if (unlikely(inited == 0)) {
|
if(is_ffd(sockfd)){
|
||||||
return SYSCALL(writev)(sockfd, iov, iovcnt);
|
sockfd = restore_ffd(sockfd);
|
||||||
}
|
assert(ff_fdisused(sockfd));
|
||||||
|
|
||||||
if (ff_fdisused(sockfd)) {
|
|
||||||
return ff_writev(sockfd, iov, iovcnt);
|
return ff_writev(sockfd, iov, iovcnt);
|
||||||
} else {
|
|
||||||
return SYSCALL(writev)(sockfd, iov, iovcnt);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
return SYSCALL(writev)(sockfd, iov, iovcnt);
|
||||||
}
|
}
|
||||||
|
|
||||||
ssize_t
|
ssize_t
|
||||||
readv(int sockfd, const struct iovec *iov, int iovcnt)
|
readv(int sockfd, const struct iovec *iov, int iovcnt)
|
||||||
{
|
{
|
||||||
if (unlikely(inited == 0)) {
|
if(is_ffd(sockfd)){
|
||||||
return SYSCALL(readv)(sockfd, iov, iovcnt);
|
sockfd = restore_ffd(sockfd);
|
||||||
}
|
assert(ff_fdisused(sockfd));
|
||||||
|
|
||||||
if (ff_fdisused(sockfd)) {
|
|
||||||
return ff_readv(sockfd, iov, iovcnt);
|
return ff_readv(sockfd, iov, iovcnt);
|
||||||
} else {
|
|
||||||
return SYSCALL(readv)(sockfd, iov, iovcnt);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
return SYSCALL(readv)(sockfd, iov, iovcnt);
|
||||||
}
|
}
|
||||||
|
|
||||||
ssize_t
|
ssize_t
|
||||||
read(int sockfd, void *buf, size_t count)
|
read(int sockfd, void *buf, size_t count)
|
||||||
{
|
{
|
||||||
if (unlikely(inited == 0)) {
|
if(is_ffd(sockfd)){
|
||||||
return SYSCALL(read)(sockfd, buf, count);
|
sockfd = restore_ffd(sockfd);
|
||||||
}
|
assert(ff_fdisused(sockfd));
|
||||||
|
|
||||||
if (ff_fdisused(sockfd)) {
|
|
||||||
return ff_read(sockfd, buf, count);
|
return ff_read(sockfd, buf, count);
|
||||||
} else {
|
|
||||||
return SYSCALL(read)(sockfd, buf, count);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
return SYSCALL(read)(sockfd, buf, count);
|
||||||
}
|
}
|
||||||
|
|
||||||
ssize_t
|
ssize_t
|
||||||
write(int sockfd, const void *buf, size_t count)
|
write(int sockfd, const void *buf, size_t count)
|
||||||
{
|
{
|
||||||
if (unlikely(inited == 0)) {
|
if(is_ffd(sockfd)){
|
||||||
return SYSCALL(write)(sockfd, buf, count);
|
sockfd = restore_ffd(sockfd);
|
||||||
}
|
assert(ff_fdisused(sockfd));
|
||||||
|
|
||||||
if (ff_fdisused(sockfd)) {
|
|
||||||
return ff_write(sockfd, buf, count);
|
return ff_write(sockfd, buf, count);
|
||||||
} else {
|
|
||||||
return SYSCALL(write)(sockfd, buf, count);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
return SYSCALL(write)(sockfd, buf, count);
|
||||||
}
|
}
|
||||||
|
|
||||||
int
|
int
|
||||||
ioctl(int sockfd, int request, void *p)
|
ioctl(int sockfd, int request, void *p)
|
||||||
{
|
{
|
||||||
if (unlikely(inited == 0)) {
|
if(is_ffd(sockfd)){
|
||||||
return SYSCALL(ioctl)(sockfd, request, p);
|
sockfd = restore_ffd(sockfd);
|
||||||
}
|
assert(ff_fdisused(sockfd));
|
||||||
|
|
||||||
if (ff_fdisused(sockfd)) {
|
|
||||||
return ff_ioctl(sockfd, request, p);
|
return ff_ioctl(sockfd, request, p);
|
||||||
} else {
|
|
||||||
return SYSCALL(ioctl)(sockfd, request, p);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
return SYSCALL(ioctl)(sockfd, request, p);
|
||||||
}
|
}
|
||||||
|
|
||||||
int
|
int
|
||||||
|
@ -429,6 +459,28 @@ int
|
||||||
kevent(int kq, const struct kevent *changelist, int nchanges,
|
kevent(int kq, const struct kevent *changelist, int nchanges,
|
||||||
struct kevent *eventlist, int nevents, const struct timespec *timeout)
|
struct kevent *eventlist, int nevents, const struct timespec *timeout)
|
||||||
{
|
{
|
||||||
|
struct kevent *kev;
|
||||||
|
int i = 0;
|
||||||
|
for(i = 0; i < nchanges; i++) {
|
||||||
|
kev = (struct kevent *)&changelist[i];
|
||||||
|
switch (kev->filter) {
|
||||||
|
|
||||||
|
case EVFILT_READ:
|
||||||
|
case EVFILT_WRITE:
|
||||||
|
case EVFILT_VNODE:
|
||||||
|
assert(is_ffd(kev->ident));
|
||||||
|
//assert(ff_fdisused(kev->ident));
|
||||||
|
kev->ident = restore_ffd(kev->ident);
|
||||||
|
break;
|
||||||
|
case EVFILT_AIO:
|
||||||
|
case EVFILT_PROC:
|
||||||
|
case EVFILT_SIGNAL:
|
||||||
|
case EVFILT_TIMER:
|
||||||
|
case EVFILT_USER:
|
||||||
|
default:
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
return ff_kevent(kq, changelist, nchanges, eventlist, nevents, timeout);
|
return ff_kevent(kq, changelist, nchanges, eventlist, nevents, timeout);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -441,4 +493,3 @@ gettimeofday(struct timeval *tv, struct timezone *tz)
|
||||||
|
|
||||||
return ff_gettimeofday(tv, tz);
|
return ff_gettimeofday(tv, tz);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -564,6 +564,11 @@ ngx_timer_signal_handler(int signo)
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
#if (NGX_HAVE_FSTACK)
|
||||||
|
|
||||||
|
extern ngx_event_actions_t ngx_event_actions_dy;
|
||||||
|
|
||||||
|
#endif
|
||||||
|
|
||||||
static ngx_int_t
|
static ngx_int_t
|
||||||
ngx_event_process_init(ngx_cycle_t *cycle)
|
ngx_event_process_init(ngx_cycle_t *cycle)
|
||||||
|
@ -602,6 +607,11 @@ ngx_event_process_init(ngx_cycle_t *cycle)
|
||||||
ngx_queue_init(&ngx_posted_accept_events);
|
ngx_queue_init(&ngx_posted_accept_events);
|
||||||
ngx_queue_init(&ngx_posted_events);
|
ngx_queue_init(&ngx_posted_events);
|
||||||
|
|
||||||
|
#if (NGX_HAVE_FSTACK)
|
||||||
|
ngx_queue_init(&ngx_posted_accept_events_of_aeds);
|
||||||
|
ngx_queue_init(&ngx_posted_events_of_aeds);
|
||||||
|
#endif
|
||||||
|
|
||||||
if (ngx_event_timer_init(cycle->log) == NGX_ERROR) {
|
if (ngx_event_timer_init(cycle->log) == NGX_ERROR) {
|
||||||
return NGX_ERROR;
|
return NGX_ERROR;
|
||||||
}
|
}
|
||||||
|
@ -625,6 +635,13 @@ ngx_event_process_init(ngx_cycle_t *cycle)
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#if (NGX_HAVE_FSTACK)
|
||||||
|
if (ngx_event_actions_dy.init(cycle, ngx_timer_resolution) != NGX_OK) {
|
||||||
|
/* fatal */
|
||||||
|
exit(2);
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
|
||||||
#if !(NGX_WIN32)
|
#if !(NGX_WIN32)
|
||||||
|
|
||||||
if (ngx_timer_resolution && !(ngx_event_flags & NGX_USE_TIMER_EVENT)) {
|
if (ngx_timer_resolution && !(ngx_event_flags & NGX_USE_TIMER_EVENT)) {
|
||||||
|
@ -663,8 +680,14 @@ ngx_event_process_init(ngx_cycle_t *cycle)
|
||||||
|
|
||||||
cycle->files_n = (ngx_uint_t) rlmt.rlim_cur;
|
cycle->files_n = (ngx_uint_t) rlmt.rlim_cur;
|
||||||
|
|
||||||
|
#if (NGX_HAVE_FSTACK)
|
||||||
|
cycle->files = ngx_calloc(sizeof(ngx_connection_t *) * cycle->files_n * 2,
|
||||||
|
cycle->log);
|
||||||
|
#else
|
||||||
cycle->files = ngx_calloc(sizeof(ngx_connection_t *) * cycle->files_n,
|
cycle->files = ngx_calloc(sizeof(ngx_connection_t *) * cycle->files_n,
|
||||||
cycle->log);
|
cycle->log);
|
||||||
|
#endif
|
||||||
|
|
||||||
if (cycle->files == NULL) {
|
if (cycle->files == NULL) {
|
||||||
return NGX_ERROR;
|
return NGX_ERROR;
|
||||||
}
|
}
|
||||||
|
@ -757,6 +780,12 @@ ngx_event_process_init(ngx_cycle_t *cycle)
|
||||||
rev->log = c->log;
|
rev->log = c->log;
|
||||||
rev->accept = 1;
|
rev->accept = 1;
|
||||||
|
|
||||||
|
#if (NGX_HAVE_FSTACK)
|
||||||
|
/* Note when nginx running on fstack,
|
||||||
|
make sure that add the right fd to kqueue !! */
|
||||||
|
c->read->belong_to_aeds = c->write->belong_to_aeds = ls[i].belong_to_aeds;
|
||||||
|
#endif
|
||||||
|
|
||||||
#if (NGX_HAVE_DEFERRED_ACCEPT)
|
#if (NGX_HAVE_DEFERRED_ACCEPT)
|
||||||
rev->deferred_accept = ls[i].deferred_accept;
|
rev->deferred_accept = ls[i].deferred_accept;
|
||||||
#endif
|
#endif
|
||||||
|
|
|
@ -142,6 +142,10 @@ struct ngx_event_s {
|
||||||
uint32_t padding[NGX_EVENT_T_PADDING];
|
uint32_t padding[NGX_EVENT_T_PADDING];
|
||||||
#endif
|
#endif
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
#if (NGX_HAVE_FSTACK)
|
||||||
|
unsigned belong_to_aeds:1;
|
||||||
|
#endif
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
||||||
|
@ -198,7 +202,9 @@ extern ngx_event_actions_t ngx_event_actions;
|
||||||
#if (NGX_HAVE_EPOLLRDHUP)
|
#if (NGX_HAVE_EPOLLRDHUP)
|
||||||
extern ngx_uint_t ngx_use_epoll_rdhup;
|
extern ngx_uint_t ngx_use_epoll_rdhup;
|
||||||
#endif
|
#endif
|
||||||
|
#if (NGX_HAVE_FSTACK)
|
||||||
|
extern ngx_event_actions_t ngx_event_actions_dy;
|
||||||
|
#endif
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* The event filter requires to read/write the whole data:
|
* The event filter requires to read/write the whole data:
|
||||||
|
@ -407,16 +413,59 @@ extern ngx_uint_t ngx_use_epoll_rdhup;
|
||||||
#define NGX_CLEAR_EVENT 0 /* dummy declaration */
|
#define NGX_CLEAR_EVENT 0 /* dummy declaration */
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
#if (NGX_HAVE_FSTACK)
|
||||||
|
|
||||||
#define ngx_process_events ngx_event_actions.process_events
|
static inline ngx_int_t
|
||||||
#define ngx_done_events ngx_event_actions.done
|
ngx_add_event(ngx_event_t *ev, ngx_int_t event, ngx_uint_t flags) {
|
||||||
|
if (1 == ev->belong_to_aeds) {
|
||||||
|
return ngx_event_actions_dy.add(ev, event, flags);
|
||||||
|
} else {
|
||||||
|
return ngx_event_actions.add(ev, event, flags);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
#define ngx_add_event ngx_event_actions.add
|
static inline ngx_int_t
|
||||||
#define ngx_del_event ngx_event_actions.del
|
ngx_del_event(ngx_event_t *ev, ngx_int_t event, ngx_uint_t flags) {
|
||||||
#define ngx_add_conn ngx_event_actions.add_conn
|
if (1 == ev->belong_to_aeds) {
|
||||||
#define ngx_del_conn ngx_event_actions.del_conn
|
return ngx_event_actions_dy.del(ev, event, flags);
|
||||||
|
} else {
|
||||||
|
return ngx_event_actions.del(ev, event, flags);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
#define ngx_notify ngx_event_actions.notify
|
static inline ngx_int_t ngx_add_conn(ngx_connection_t *c)
|
||||||
|
{
|
||||||
|
return ngx_event_actions.add_conn(c);
|
||||||
|
}
|
||||||
|
|
||||||
|
static inline ngx_int_t ngx_del_conn(
|
||||||
|
ngx_connection_t *c, ngx_uint_t flags) {
|
||||||
|
return ngx_event_actions.del_conn(c, flags);
|
||||||
|
}
|
||||||
|
|
||||||
|
static inline ngx_int_t ngx_notify(ngx_event_handler_pt handler) {
|
||||||
|
return ngx_event_actions.notify(handler);
|
||||||
|
}
|
||||||
|
|
||||||
|
static inline ngx_int_t ngx_process_events(
|
||||||
|
ngx_cycle_t *cycle, ngx_msec_t timer, ngx_uint_t flags)
|
||||||
|
{
|
||||||
|
return ngx_event_actions.process_events(cycle, timer, flags);
|
||||||
|
}
|
||||||
|
|
||||||
|
#else
|
||||||
|
|
||||||
|
#define ngx_process_events ngx_event_actions.process_events
|
||||||
|
#define ngx_done_events ngx_event_actions.done
|
||||||
|
|
||||||
|
#define ngx_add_event ngx_event_actions.add
|
||||||
|
#define ngx_del_event ngx_event_actions.del
|
||||||
|
#define ngx_add_conn ngx_event_actions.add_conn
|
||||||
|
#define ngx_del_conn ngx_event_actions.del_conn
|
||||||
|
|
||||||
|
#define ngx_notify ngx_event_actions.notify
|
||||||
|
|
||||||
|
#endif
|
||||||
|
|
||||||
#define ngx_add_timer ngx_event_add_timer
|
#define ngx_add_timer ngx_event_add_timer
|
||||||
#define ngx_del_timer ngx_event_del_timer
|
#define ngx_del_timer ngx_event_del_timer
|
||||||
|
|
|
@ -230,6 +230,10 @@ ngx_event_accept(ngx_event_t *ev)
|
||||||
rev = c->read;
|
rev = c->read;
|
||||||
wev = c->write;
|
wev = c->write;
|
||||||
|
|
||||||
|
#if (NGX_HAVE_FSTACK)
|
||||||
|
rev->belong_to_aeds = wev->belong_to_aeds = ev->belong_to_aeds;
|
||||||
|
#endif
|
||||||
|
|
||||||
wev->ready = 1;
|
wev->ready = 1;
|
||||||
|
|
||||||
if (ngx_event_flags & NGX_USE_IOCP_EVENT) {
|
if (ngx_event_flags & NGX_USE_IOCP_EVENT) {
|
||||||
|
@ -296,7 +300,11 @@ ngx_event_accept(ngx_event_t *ev)
|
||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
#if (NGX_HAVE_FSTACK)
|
||||||
|
if (ngx_event_actions.add_conn && (ngx_event_flags & NGX_USE_EPOLL_EVENT) == 0) {
|
||||||
|
#else
|
||||||
if (ngx_add_conn && (ngx_event_flags & NGX_USE_EPOLL_EVENT) == 0) {
|
if (ngx_add_conn && (ngx_event_flags & NGX_USE_EPOLL_EVENT) == 0) {
|
||||||
|
#endif
|
||||||
if (ngx_add_conn(c) == NGX_ERROR) {
|
if (ngx_add_conn(c) == NGX_ERROR) {
|
||||||
ngx_close_accepted_connection(c);
|
ngx_close_accepted_connection(c);
|
||||||
return;
|
return;
|
||||||
|
@ -432,6 +440,7 @@ ngx_event_recvmsg(ngx_event_t *ev)
|
||||||
- ngx_cycle->free_connection_n;
|
- ngx_cycle->free_connection_n;
|
||||||
|
|
||||||
c = ngx_get_connection(lc->fd, ev->log);
|
c = ngx_get_connection(lc->fd, ev->log);
|
||||||
|
|
||||||
if (c == NULL) {
|
if (c == NULL) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
|
@ -181,7 +181,11 @@ ngx_event_connect_peer(ngx_peer_connection_t *pc)
|
||||||
|
|
||||||
c->number = ngx_atomic_fetch_add(ngx_connection_counter, 1);
|
c->number = ngx_atomic_fetch_add(ngx_connection_counter, 1);
|
||||||
|
|
||||||
|
#if (NGX_HAVE_FSTACK)
|
||||||
|
if (ngx_event_actions.add_conn) {
|
||||||
|
#else
|
||||||
if (ngx_add_conn) {
|
if (ngx_add_conn) {
|
||||||
|
#endif
|
||||||
if (ngx_add_conn(c) == NGX_ERROR) {
|
if (ngx_add_conn(c) == NGX_ERROR) {
|
||||||
goto failed;
|
goto failed;
|
||||||
}
|
}
|
||||||
|
@ -233,7 +237,11 @@ ngx_event_connect_peer(ngx_peer_connection_t *pc)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#if (NGX_HAVE_FSTACK)
|
||||||
|
if (ngx_event_actions.add_conn) {
|
||||||
|
#else
|
||||||
if (ngx_add_conn) {
|
if (ngx_add_conn) {
|
||||||
|
#endif
|
||||||
if (rc == -1) {
|
if (rc == -1) {
|
||||||
|
|
||||||
/* NGX_EINPROGRESS */
|
/* NGX_EINPROGRESS */
|
||||||
|
|
|
@ -13,6 +13,11 @@
|
||||||
ngx_queue_t ngx_posted_accept_events;
|
ngx_queue_t ngx_posted_accept_events;
|
||||||
ngx_queue_t ngx_posted_events;
|
ngx_queue_t ngx_posted_events;
|
||||||
|
|
||||||
|
#if (NGX_HAVE_FSTACK)
|
||||||
|
ngx_queue_t ngx_posted_accept_events_of_aeds;
|
||||||
|
ngx_queue_t ngx_posted_events_of_aeds;
|
||||||
|
#endif
|
||||||
|
|
||||||
|
|
||||||
void
|
void
|
||||||
ngx_event_process_posted(ngx_cycle_t *cycle, ngx_queue_t *posted)
|
ngx_event_process_posted(ngx_cycle_t *cycle, ngx_queue_t *posted)
|
||||||
|
|
|
@ -14,6 +14,34 @@
|
||||||
#include <ngx_event.h>
|
#include <ngx_event.h>
|
||||||
|
|
||||||
|
|
||||||
|
#if (NGX_HAVE_FSTACK)
|
||||||
|
#define ngx_post_event(ev, q) \
|
||||||
|
\
|
||||||
|
if (!(ev)->posted) { \
|
||||||
|
(ev)->posted = 1; \
|
||||||
|
if (1 == (ev)->belong_to_aeds) { \
|
||||||
|
if (q == &ngx_posted_events) { \
|
||||||
|
ngx_queue_insert_tail( \
|
||||||
|
&ngx_posted_events_of_aeds, &(ev)->queue); \
|
||||||
|
} else if (q == &ngx_posted_accept_events) { \
|
||||||
|
ngx_queue_insert_tail( \
|
||||||
|
&ngx_posted_accept_events_of_aeds, &(ev)->queue); \
|
||||||
|
} else { \
|
||||||
|
ngx_log_error(NGX_LOG_EMERG, (ev)->log, 0, \
|
||||||
|
"ngx_post_event: unkowned posted queue"); \
|
||||||
|
exit(1); \
|
||||||
|
} \
|
||||||
|
} else { \
|
||||||
|
ngx_queue_insert_tail(q, &(ev)->queue); \
|
||||||
|
} \
|
||||||
|
\
|
||||||
|
ngx_log_debug1(NGX_LOG_DEBUG_CORE, (ev)->log, 0, "post event %p", ev);\
|
||||||
|
\
|
||||||
|
} else { \
|
||||||
|
ngx_log_debug1(NGX_LOG_DEBUG_CORE, (ev)->log, 0, \
|
||||||
|
"update posted event %p", ev); \
|
||||||
|
}
|
||||||
|
#else
|
||||||
#define ngx_post_event(ev, q) \
|
#define ngx_post_event(ev, q) \
|
||||||
\
|
\
|
||||||
if (!(ev)->posted) { \
|
if (!(ev)->posted) { \
|
||||||
|
@ -26,6 +54,7 @@
|
||||||
ngx_log_debug1(NGX_LOG_DEBUG_CORE, (ev)->log, 0, \
|
ngx_log_debug1(NGX_LOG_DEBUG_CORE, (ev)->log, 0, \
|
||||||
"update posted event %p", ev); \
|
"update posted event %p", ev); \
|
||||||
}
|
}
|
||||||
|
#endif
|
||||||
|
|
||||||
|
|
||||||
#define ngx_delete_posted_event(ev) \
|
#define ngx_delete_posted_event(ev) \
|
||||||
|
@ -44,5 +73,9 @@ void ngx_event_process_posted(ngx_cycle_t *cycle, ngx_queue_t *posted);
|
||||||
extern ngx_queue_t ngx_posted_accept_events;
|
extern ngx_queue_t ngx_posted_accept_events;
|
||||||
extern ngx_queue_t ngx_posted_events;
|
extern ngx_queue_t ngx_posted_events;
|
||||||
|
|
||||||
|
#if (NGX_HAVE_FSTACK)
|
||||||
|
extern ngx_queue_t ngx_posted_accept_events_of_aeds;
|
||||||
|
extern ngx_queue_t ngx_posted_events_of_aeds;
|
||||||
|
#endif
|
||||||
|
|
||||||
#endif /* _NGX_EVENT_POSTED_H_INCLUDED_ */
|
#endif /* _NGX_EVENT_POSTED_H_INCLUDED_ */
|
||||||
|
|
|
@ -13,6 +13,11 @@
|
||||||
ngx_rbtree_t ngx_event_timer_rbtree;
|
ngx_rbtree_t ngx_event_timer_rbtree;
|
||||||
static ngx_rbtree_node_t ngx_event_timer_sentinel;
|
static ngx_rbtree_node_t ngx_event_timer_sentinel;
|
||||||
|
|
||||||
|
#if (NGX_HAVE_FSTACK)
|
||||||
|
ngx_rbtree_t ngx_aeds_timer_rbtree;
|
||||||
|
static ngx_rbtree_node_t ngx_aeds_timer_sentinel;
|
||||||
|
#endif
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* the event timer rbtree may contain the duplicate keys, however,
|
* the event timer rbtree may contain the duplicate keys, however,
|
||||||
* it should not be a problem, because we use the rbtree to find
|
* it should not be a problem, because we use the rbtree to find
|
||||||
|
@ -25,22 +30,54 @@ ngx_event_timer_init(ngx_log_t *log)
|
||||||
ngx_rbtree_init(&ngx_event_timer_rbtree, &ngx_event_timer_sentinel,
|
ngx_rbtree_init(&ngx_event_timer_rbtree, &ngx_event_timer_sentinel,
|
||||||
ngx_rbtree_insert_timer_value);
|
ngx_rbtree_insert_timer_value);
|
||||||
|
|
||||||
|
#if (NGX_HAVE_FSTACK)
|
||||||
|
|
||||||
|
ngx_rbtree_init(&ngx_aeds_timer_rbtree, &ngx_aeds_timer_sentinel,
|
||||||
|
ngx_rbtree_insert_timer_value);
|
||||||
|
|
||||||
|
#endif
|
||||||
|
|
||||||
return NGX_OK;
|
return NGX_OK;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
#if (NGX_HAVE_FSTACK)
|
||||||
|
|
||||||
|
ngx_msec_t
|
||||||
|
ngx_event_find_timer_internal(
|
||||||
|
ngx_rbtree_t *rbtree, ngx_rbtree_node_t *sentinel);
|
||||||
ngx_msec_t
|
ngx_msec_t
|
||||||
ngx_event_find_timer(void)
|
ngx_event_find_timer(void)
|
||||||
{
|
{
|
||||||
|
return ngx_event_find_timer_internal(&ngx_event_timer_rbtree, &ngx_event_timer_sentinel);
|
||||||
|
}
|
||||||
|
|
||||||
|
ngx_msec_t
|
||||||
|
ngx_aeds_find_timer(void)
|
||||||
|
{
|
||||||
|
return ngx_event_find_timer_internal(&ngx_aeds_timer_rbtree, &ngx_aeds_timer_sentinel);
|
||||||
|
}
|
||||||
|
|
||||||
|
ngx_msec_t
|
||||||
|
ngx_event_find_timer_internal(
|
||||||
|
ngx_rbtree_t *rbtree, ngx_rbtree_node_t *rbtree_sentinel)
|
||||||
|
{
|
||||||
|
#else
|
||||||
|
ngx_msec_t
|
||||||
|
ngx_event_find_timer(void)
|
||||||
|
{
|
||||||
|
ngx_rbtree_t * rbtree = &ngx_event_timer_rbtree;
|
||||||
|
ngx_rbtree_node_t *rbtree_sentinel = &ngx_event_timer_sentinel;
|
||||||
|
#endif
|
||||||
ngx_msec_int_t timer;
|
ngx_msec_int_t timer;
|
||||||
ngx_rbtree_node_t *node, *root, *sentinel;
|
ngx_rbtree_node_t *node, *root, *sentinel;
|
||||||
|
|
||||||
if (ngx_event_timer_rbtree.root == &ngx_event_timer_sentinel) {
|
if (rbtree->root == rbtree_sentinel) {
|
||||||
return NGX_TIMER_INFINITE;
|
return NGX_TIMER_INFINITE;
|
||||||
}
|
}
|
||||||
|
|
||||||
root = ngx_event_timer_rbtree.root;
|
root = rbtree->root;
|
||||||
sentinel = ngx_event_timer_rbtree.sentinel;
|
sentinel = rbtree->sentinel;
|
||||||
|
|
||||||
node = ngx_rbtree_min(root, sentinel);
|
node = ngx_rbtree_min(root, sentinel);
|
||||||
|
|
||||||
|
@ -50,16 +87,39 @@ ngx_event_find_timer(void)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
#if (NGX_HAVE_FSTACK)
|
||||||
|
|
||||||
|
void
|
||||||
|
ngx_event_expire_timers_internal(ngx_rbtree_t *rbtree);
|
||||||
|
|
||||||
void
|
void
|
||||||
ngx_event_expire_timers(void)
|
ngx_event_expire_timers(void)
|
||||||
{
|
{
|
||||||
|
ngx_event_expire_timers_internal(&ngx_event_timer_rbtree);
|
||||||
|
}
|
||||||
|
|
||||||
|
void
|
||||||
|
ngx_aeds_expire_timers(void)
|
||||||
|
{
|
||||||
|
ngx_event_expire_timers_internal(&ngx_aeds_timer_rbtree);
|
||||||
|
}
|
||||||
|
|
||||||
|
void
|
||||||
|
ngx_event_expire_timers_internal(ngx_rbtree_t *rbtree)
|
||||||
|
{
|
||||||
|
#else
|
||||||
|
void
|
||||||
|
ngx_event_expire_timers(void)
|
||||||
|
{
|
||||||
|
ngx_rbtree_t * rbtree = &ngx_event_timer_rbtree;
|
||||||
|
#endif
|
||||||
ngx_event_t *ev;
|
ngx_event_t *ev;
|
||||||
ngx_rbtree_node_t *node, *root, *sentinel;
|
ngx_rbtree_node_t *node, *root, *sentinel;
|
||||||
|
|
||||||
sentinel = ngx_event_timer_rbtree.sentinel;
|
sentinel = rbtree->sentinel;
|
||||||
|
|
||||||
for ( ;; ) {
|
for ( ;; ) {
|
||||||
root = ngx_event_timer_rbtree.root;
|
root = rbtree->root;
|
||||||
|
|
||||||
if (root == sentinel) {
|
if (root == sentinel) {
|
||||||
return;
|
return;
|
||||||
|
@ -79,7 +139,7 @@ ngx_event_expire_timers(void)
|
||||||
"event timer del: %d: %M",
|
"event timer del: %d: %M",
|
||||||
ngx_event_ident(ev->data), ev->timer.key);
|
ngx_event_ident(ev->data), ev->timer.key);
|
||||||
|
|
||||||
ngx_rbtree_delete(&ngx_event_timer_rbtree, &ev->timer);
|
ngx_rbtree_delete(rbtree, &ev->timer);
|
||||||
|
|
||||||
#if (NGX_DEBUG)
|
#if (NGX_DEBUG)
|
||||||
ev->timer.left = NULL;
|
ev->timer.left = NULL;
|
||||||
|
@ -96,16 +156,39 @@ ngx_event_expire_timers(void)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
#if (NGX_HAVE_FSTACK)
|
||||||
|
|
||||||
|
void
|
||||||
|
ngx_event_cancel_timers_internal(ngx_rbtree_t *rbtree);
|
||||||
|
|
||||||
void
|
void
|
||||||
ngx_event_cancel_timers(void)
|
ngx_event_cancel_timers(void)
|
||||||
{
|
{
|
||||||
|
ngx_event_cancel_timers_internal(&ngx_event_timer_rbtree);
|
||||||
|
}
|
||||||
|
|
||||||
|
void
|
||||||
|
ngx_aeds_cancel_timers(void)
|
||||||
|
{
|
||||||
|
ngx_event_cancel_timers_internal(&ngx_aeds_timer_rbtree);
|
||||||
|
}
|
||||||
|
|
||||||
|
void
|
||||||
|
ngx_event_cancel_timers_internal(ngx_rbtree_t *rbtree)
|
||||||
|
{
|
||||||
|
#else
|
||||||
|
void
|
||||||
|
ngx_event_cancel_timers(void)
|
||||||
|
{
|
||||||
|
ngx_rbtree_t * rbtree = &ngx_event_timer_rbtree;
|
||||||
|
#endif
|
||||||
ngx_event_t *ev;
|
ngx_event_t *ev;
|
||||||
ngx_rbtree_node_t *node, *root, *sentinel;
|
ngx_rbtree_node_t *node, *root, *sentinel;
|
||||||
|
|
||||||
sentinel = ngx_event_timer_rbtree.sentinel;
|
sentinel = rbtree->sentinel;
|
||||||
|
|
||||||
for ( ;; ) {
|
for ( ;; ) {
|
||||||
root = ngx_event_timer_rbtree.root;
|
root = rbtree->root;
|
||||||
|
|
||||||
if (root == sentinel) {
|
if (root == sentinel) {
|
||||||
return;
|
return;
|
||||||
|
@ -123,7 +206,7 @@ ngx_event_cancel_timers(void)
|
||||||
"event timer cancel: %d: %M",
|
"event timer cancel: %d: %M",
|
||||||
ngx_event_ident(ev->data), ev->timer.key);
|
ngx_event_ident(ev->data), ev->timer.key);
|
||||||
|
|
||||||
ngx_rbtree_delete(&ngx_event_timer_rbtree, &ev->timer);
|
ngx_rbtree_delete(rbtree, &ev->timer);
|
||||||
|
|
||||||
#if (NGX_DEBUG)
|
#if (NGX_DEBUG)
|
||||||
ev->timer.left = NULL;
|
ev->timer.left = NULL;
|
||||||
|
|
|
@ -27,6 +27,9 @@ void ngx_event_cancel_timers(void);
|
||||||
|
|
||||||
extern ngx_rbtree_t ngx_event_timer_rbtree;
|
extern ngx_rbtree_t ngx_event_timer_rbtree;
|
||||||
|
|
||||||
|
#if (NGX_HAVE_FSTACK)
|
||||||
|
extern ngx_rbtree_t ngx_aeds_timer_rbtree;
|
||||||
|
#endif
|
||||||
|
|
||||||
static ngx_inline void
|
static ngx_inline void
|
||||||
ngx_event_del_timer(ngx_event_t *ev)
|
ngx_event_del_timer(ngx_event_t *ev)
|
||||||
|
@ -35,8 +38,20 @@ ngx_event_del_timer(ngx_event_t *ev)
|
||||||
"event timer del: %d: %M",
|
"event timer del: %d: %M",
|
||||||
ngx_event_ident(ev->data), ev->timer.key);
|
ngx_event_ident(ev->data), ev->timer.key);
|
||||||
|
|
||||||
|
#if (NGX_HAVE_FSTACK)
|
||||||
|
|
||||||
|
if(ev->belong_to_aeds){
|
||||||
|
ngx_rbtree_delete(&ngx_aeds_timer_rbtree, &ev->timer);
|
||||||
|
} else {
|
||||||
|
ngx_rbtree_delete(&ngx_event_timer_rbtree, &ev->timer);
|
||||||
|
}
|
||||||
|
|
||||||
|
#else
|
||||||
|
|
||||||
ngx_rbtree_delete(&ngx_event_timer_rbtree, &ev->timer);
|
ngx_rbtree_delete(&ngx_event_timer_rbtree, &ev->timer);
|
||||||
|
|
||||||
|
#endif
|
||||||
|
|
||||||
#if (NGX_DEBUG)
|
#if (NGX_DEBUG)
|
||||||
ev->timer.left = NULL;
|
ev->timer.left = NULL;
|
||||||
ev->timer.right = NULL;
|
ev->timer.right = NULL;
|
||||||
|
@ -81,8 +96,20 @@ ngx_event_add_timer(ngx_event_t *ev, ngx_msec_t timer)
|
||||||
"event timer add: %d: %M:%M",
|
"event timer add: %d: %M:%M",
|
||||||
ngx_event_ident(ev->data), timer, ev->timer.key);
|
ngx_event_ident(ev->data), timer, ev->timer.key);
|
||||||
|
|
||||||
|
#if (NGX_HAVE_FSTACK)
|
||||||
|
|
||||||
|
if(ev->belong_to_aeds){
|
||||||
|
ngx_rbtree_insert(&ngx_aeds_timer_rbtree, &ev->timer);
|
||||||
|
} else {
|
||||||
|
ngx_rbtree_insert(&ngx_event_timer_rbtree, &ev->timer);
|
||||||
|
}
|
||||||
|
|
||||||
|
#else
|
||||||
|
|
||||||
ngx_rbtree_insert(&ngx_event_timer_rbtree, &ev->timer);
|
ngx_rbtree_insert(&ngx_event_timer_rbtree, &ev->timer);
|
||||||
|
|
||||||
|
#endif
|
||||||
|
|
||||||
ev->timer_set = 1;
|
ev->timer_set = 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -1775,6 +1775,10 @@ ngx_http_add_listening(ngx_conf_t *cf, ngx_http_conf_addr_t *addr)
|
||||||
ls->reuseport = addr->opt.reuseport;
|
ls->reuseport = addr->opt.reuseport;
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
#if (NGX_HAVE_FSTACK)
|
||||||
|
ls->belong_to_aeds = cscf->kernel_network_stack;
|
||||||
|
#endif
|
||||||
|
|
||||||
return ls;
|
return ls;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -293,6 +293,17 @@ static ngx_command_t ngx_http_core_commands[] = {
|
||||||
0,
|
0,
|
||||||
NULL },
|
NULL },
|
||||||
|
|
||||||
|
#if (NGX_HAVE_FSTACK)
|
||||||
|
|
||||||
|
{ ngx_string("kernel_network_stack"),
|
||||||
|
NGX_HTTP_MAIN_CONF|NGX_HTTP_SRV_CONF|NGX_CONF_FLAG,
|
||||||
|
ngx_conf_set_flag_slot,
|
||||||
|
NGX_HTTP_SRV_CONF_OFFSET,
|
||||||
|
offsetof(ngx_http_core_srv_conf_t, kernel_network_stack),
|
||||||
|
NULL },
|
||||||
|
|
||||||
|
#endif
|
||||||
|
|
||||||
{ ngx_string("types_hash_max_size"),
|
{ ngx_string("types_hash_max_size"),
|
||||||
NGX_HTTP_MAIN_CONF|NGX_HTTP_SRV_CONF|NGX_HTTP_LOC_CONF|NGX_CONF_TAKE1,
|
NGX_HTTP_MAIN_CONF|NGX_HTTP_SRV_CONF|NGX_HTTP_LOC_CONF|NGX_CONF_TAKE1,
|
||||||
ngx_conf_set_num_slot,
|
ngx_conf_set_num_slot,
|
||||||
|
@ -3444,6 +3455,7 @@ ngx_http_core_create_srv_conf(ngx_conf_t *cf)
|
||||||
cscf->ignore_invalid_headers = NGX_CONF_UNSET;
|
cscf->ignore_invalid_headers = NGX_CONF_UNSET;
|
||||||
cscf->merge_slashes = NGX_CONF_UNSET;
|
cscf->merge_slashes = NGX_CONF_UNSET;
|
||||||
cscf->underscores_in_headers = NGX_CONF_UNSET;
|
cscf->underscores_in_headers = NGX_CONF_UNSET;
|
||||||
|
cscf->kernel_network_stack = NGX_CONF_UNSET;
|
||||||
|
|
||||||
return cscf;
|
return cscf;
|
||||||
}
|
}
|
||||||
|
@ -3487,6 +3499,12 @@ ngx_http_core_merge_srv_conf(ngx_conf_t *cf, void *parent, void *child)
|
||||||
ngx_conf_merge_value(conf->underscores_in_headers,
|
ngx_conf_merge_value(conf->underscores_in_headers,
|
||||||
prev->underscores_in_headers, 0);
|
prev->underscores_in_headers, 0);
|
||||||
|
|
||||||
|
#if (NGX_HAVE_FSTACK)
|
||||||
|
/* By default, we set up a server on fstack */
|
||||||
|
ngx_conf_merge_value(conf->kernel_network_stack,
|
||||||
|
prev->kernel_network_stack, 0);
|
||||||
|
#endif
|
||||||
|
|
||||||
if (conf->server_names.nelts == 0) {
|
if (conf->server_names.nelts == 0) {
|
||||||
/* the array has 4 empty preallocated elements, so push cannot fail */
|
/* the array has 4 empty preallocated elements, so push cannot fail */
|
||||||
sn = ngx_array_push(&conf->server_names);
|
sn = ngx_array_push(&conf->server_names);
|
||||||
|
|
|
@ -199,6 +199,10 @@ typedef struct {
|
||||||
ngx_flag_t merge_slashes;
|
ngx_flag_t merge_slashes;
|
||||||
ngx_flag_t underscores_in_headers;
|
ngx_flag_t underscores_in_headers;
|
||||||
|
|
||||||
|
#if (NGX_HAVE_FSTACK)
|
||||||
|
ngx_flag_t kernel_network_stack; /* kernel_network_stack */
|
||||||
|
#endif
|
||||||
|
|
||||||
unsigned listen:1;
|
unsigned listen:1;
|
||||||
#if (NGX_PCRE)
|
#if (NGX_PCRE)
|
||||||
unsigned captures:1;
|
unsigned captures:1;
|
||||||
|
|
|
@ -345,6 +345,10 @@ ngx_mail_optimize_servers(ngx_conf_t *cf, ngx_array_t *ports)
|
||||||
ls->ipv6only = addr[i].opt.ipv6only;
|
ls->ipv6only = addr[i].opt.ipv6only;
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
#if (NGX_HAVE_FSTACK)
|
||||||
|
ls->belong_to_aeds = cscf->kernel_network_stack;
|
||||||
|
#endif
|
||||||
|
|
||||||
mport = ngx_palloc(cf->pool, sizeof(ngx_mail_port_t));
|
mport = ngx_palloc(cf->pool, sizeof(ngx_mail_port_t));
|
||||||
if (mport == NULL) {
|
if (mport == NULL) {
|
||||||
return NGX_CONF_ERROR;
|
return NGX_CONF_ERROR;
|
||||||
|
|
|
@ -112,6 +112,10 @@ typedef struct {
|
||||||
|
|
||||||
ngx_str_t server_name;
|
ngx_str_t server_name;
|
||||||
|
|
||||||
|
#if (NGX_HAVE_FSTACK)
|
||||||
|
ngx_flag_t kernel_network_stack; /* kernel_network_stack */
|
||||||
|
#endif
|
||||||
|
|
||||||
u_char *file_name;
|
u_char *file_name;
|
||||||
ngx_uint_t line;
|
ngx_uint_t line;
|
||||||
|
|
||||||
|
|
|
@ -64,6 +64,17 @@ static ngx_command_t ngx_mail_core_commands[] = {
|
||||||
offsetof(ngx_mail_core_srv_conf_t, server_name),
|
offsetof(ngx_mail_core_srv_conf_t, server_name),
|
||||||
NULL },
|
NULL },
|
||||||
|
|
||||||
|
#if (NGX_HAVE_FSTACK)
|
||||||
|
|
||||||
|
{ ngx_string("kernel_network_stack"),
|
||||||
|
NGX_HTTP_MAIN_CONF|NGX_HTTP_SRV_CONF|NGX_CONF_FLAG,
|
||||||
|
ngx_conf_set_flag_slot,
|
||||||
|
NGX_HTTP_SRV_CONF_OFFSET,
|
||||||
|
offsetof(ngx_http_core_srv_conf_t, kernel_network_stack),
|
||||||
|
NULL },
|
||||||
|
|
||||||
|
#endif
|
||||||
|
|
||||||
{ ngx_string("error_log"),
|
{ ngx_string("error_log"),
|
||||||
NGX_MAIL_MAIN_CONF|NGX_MAIL_SRV_CONF|NGX_CONF_1MORE,
|
NGX_MAIL_MAIN_CONF|NGX_MAIL_SRV_CONF|NGX_CONF_1MORE,
|
||||||
ngx_mail_core_error_log,
|
ngx_mail_core_error_log,
|
||||||
|
@ -167,6 +178,9 @@ ngx_mail_core_create_srv_conf(ngx_conf_t *cf)
|
||||||
|
|
||||||
cscf->file_name = cf->conf_file->file.name.data;
|
cscf->file_name = cf->conf_file->file.name.data;
|
||||||
cscf->line = cf->conf_file->line;
|
cscf->line = cf->conf_file->line;
|
||||||
|
#if (NGX_HAVE_FSTACK)
|
||||||
|
cscf->kernel_network_stack = NGX_CONF_UNSET;
|
||||||
|
#endif
|
||||||
|
|
||||||
return cscf;
|
return cscf;
|
||||||
}
|
}
|
||||||
|
@ -206,6 +220,12 @@ ngx_mail_core_merge_srv_conf(ngx_conf_t *cf, void *parent, void *child)
|
||||||
|
|
||||||
ngx_conf_merge_ptr_value(conf->resolver, prev->resolver, NULL);
|
ngx_conf_merge_ptr_value(conf->resolver, prev->resolver, NULL);
|
||||||
|
|
||||||
|
#if (NGX_HAVE_FSTACK)
|
||||||
|
/* By default, we set up a server on fstack */
|
||||||
|
ngx_conf_merge_value(conf->kernel_network_stack,
|
||||||
|
prev->kernel_network_stack, 0);
|
||||||
|
#endif
|
||||||
|
|
||||||
return NGX_CONF_OK;
|
return NGX_CONF_OK;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -223,7 +223,11 @@ ngx_add_channel_event(ngx_cycle_t *cycle, ngx_fd_t fd, ngx_int_t event,
|
||||||
|
|
||||||
ev->handler = handler;
|
ev->handler = handler;
|
||||||
|
|
||||||
|
#if (NGX_HAVE_FSTACK)
|
||||||
|
if (ngx_event_actions.add_conn && (ngx_event_flags & NGX_USE_EPOLL_EVENT) == 0) {
|
||||||
|
#else
|
||||||
if (ngx_add_conn && (ngx_event_flags & NGX_USE_EPOLL_EVENT) == 0) {
|
if (ngx_add_conn && (ngx_event_flags & NGX_USE_EPOLL_EVENT) == 0) {
|
||||||
|
#endif
|
||||||
if (ngx_add_conn(c) == NGX_ERROR) {
|
if (ngx_add_conn(c) == NGX_ERROR) {
|
||||||
ngx_free_connection(c);
|
ngx_free_connection(c);
|
||||||
return NGX_ERROR;
|
return NGX_ERROR;
|
||||||
|
|
|
@ -737,9 +737,7 @@ ngx_master_process_exit(ngx_cycle_t *cycle)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
#if (!NGX_HAVE_FSTACK)
|
|
||||||
ngx_close_listening_sockets(cycle);
|
ngx_close_listening_sockets(cycle);
|
||||||
#endif
|
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Copy ngx_cycle->log related data to the special static exit cycle,
|
* Copy ngx_cycle->log related data to the special static exit cycle,
|
||||||
|
|
|
@ -512,6 +512,10 @@ ngx_stream_optimize_servers(ngx_conf_t *cf, ngx_array_t *ports)
|
||||||
ls->reuseport = addr[i].opt.reuseport;
|
ls->reuseport = addr[i].opt.reuseport;
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
#if (NGX_HAVE_FSTACK)
|
||||||
|
ls->belong_to_aeds = cscf->kernel_network_stack;
|
||||||
|
#endif
|
||||||
|
|
||||||
stport = ngx_palloc(cf->pool, sizeof(ngx_stream_port_t));
|
stport = ngx_palloc(cf->pool, sizeof(ngx_stream_port_t));
|
||||||
if (stport == NULL) {
|
if (stport == NULL) {
|
||||||
return NGX_CONF_ERROR;
|
return NGX_CONF_ERROR;
|
||||||
|
|
|
@ -185,6 +185,10 @@ typedef struct {
|
||||||
|
|
||||||
ngx_msec_t proxy_protocol_timeout;
|
ngx_msec_t proxy_protocol_timeout;
|
||||||
|
|
||||||
|
#if (NGX_HAVE_FSTACK)
|
||||||
|
ngx_flag_t kernel_network_stack; /* kernel_network_stack */
|
||||||
|
#endif
|
||||||
|
|
||||||
ngx_uint_t listen; /* unsigned listen:1; */
|
ngx_uint_t listen; /* unsigned listen:1; */
|
||||||
} ngx_stream_core_srv_conf_t;
|
} ngx_stream_core_srv_conf_t;
|
||||||
|
|
||||||
|
|
|
@ -56,6 +56,17 @@ static ngx_command_t ngx_stream_core_commands[] = {
|
||||||
0,
|
0,
|
||||||
NULL },
|
NULL },
|
||||||
|
|
||||||
|
#if (NGX_HAVE_FSTACK)
|
||||||
|
|
||||||
|
{ ngx_string("kernel_network_stack"),
|
||||||
|
NGX_HTTP_MAIN_CONF|NGX_HTTP_SRV_CONF|NGX_CONF_FLAG,
|
||||||
|
ngx_conf_set_flag_slot,
|
||||||
|
NGX_HTTP_SRV_CONF_OFFSET,
|
||||||
|
offsetof(ngx_http_core_srv_conf_t, kernel_network_stack),
|
||||||
|
NULL },
|
||||||
|
|
||||||
|
#endif
|
||||||
|
|
||||||
{ ngx_string("error_log"),
|
{ ngx_string("error_log"),
|
||||||
NGX_STREAM_MAIN_CONF|NGX_STREAM_SRV_CONF|NGX_CONF_1MORE,
|
NGX_STREAM_MAIN_CONF|NGX_STREAM_SRV_CONF|NGX_CONF_1MORE,
|
||||||
ngx_stream_core_error_log,
|
ngx_stream_core_error_log,
|
||||||
|
@ -425,6 +436,9 @@ ngx_stream_core_create_srv_conf(ngx_conf_t *cf)
|
||||||
cscf->tcp_nodelay = NGX_CONF_UNSET;
|
cscf->tcp_nodelay = NGX_CONF_UNSET;
|
||||||
cscf->preread_buffer_size = NGX_CONF_UNSET_SIZE;
|
cscf->preread_buffer_size = NGX_CONF_UNSET_SIZE;
|
||||||
cscf->preread_timeout = NGX_CONF_UNSET_MSEC;
|
cscf->preread_timeout = NGX_CONF_UNSET_MSEC;
|
||||||
|
#if (NGX_HAVE_FSTACK)
|
||||||
|
cscf->kernel_network_stack = NGX_CONF_UNSET;
|
||||||
|
#endif
|
||||||
|
|
||||||
return cscf;
|
return cscf;
|
||||||
}
|
}
|
||||||
|
@ -483,6 +497,12 @@ ngx_stream_core_merge_srv_conf(ngx_conf_t *cf, void *parent, void *child)
|
||||||
ngx_conf_merge_msec_value(conf->preread_timeout,
|
ngx_conf_merge_msec_value(conf->preread_timeout,
|
||||||
prev->preread_timeout, 30000);
|
prev->preread_timeout, 30000);
|
||||||
|
|
||||||
|
#if (NGX_HAVE_FSTACK)
|
||||||
|
/* By default, we set up a server on fstack */
|
||||||
|
ngx_conf_merge_value(conf->kernel_network_stack,
|
||||||
|
prev->kernel_network_stack, 0);
|
||||||
|
#endif
|
||||||
|
|
||||||
return NGX_CONF_OK;
|
return NGX_CONF_OK;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -4134,5 +4134,13 @@ ff_fdused_range(int max)
|
||||||
fdalloc(td, 0, &result);
|
fdalloc(td, 0, &result);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
int
|
||||||
|
ff_getmaxfd(void)
|
||||||
|
{
|
||||||
|
struct thread *td = curthread;
|
||||||
|
return getmaxfd(td);
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
|
|
@ -117,6 +117,7 @@ int ff_gettimeofday(struct timeval *tv, struct timezone *tz);
|
||||||
/* Tests if fd is used by F-Stack */
|
/* Tests if fd is used by F-Stack */
|
||||||
extern int ff_fdisused(int fd);
|
extern int ff_fdisused(int fd);
|
||||||
|
|
||||||
|
extern int ff_getmaxfd(void);
|
||||||
|
|
||||||
/* route api begin */
|
/* route api begin */
|
||||||
enum FF_ROUTE_CTL {
|
enum FF_ROUTE_CTL {
|
||||||
|
|
|
@ -43,4 +43,5 @@ ff_route_ctl
|
||||||
ff_rtioctl
|
ff_rtioctl
|
||||||
ff_gettimeofday
|
ff_gettimeofday
|
||||||
ff_fdisused
|
ff_fdisused
|
||||||
|
ff_getmaxfd
|
||||||
ff_ngctl
|
ff_ngctl
|
||||||
|
|
|
@ -32,5 +32,6 @@
|
||||||
|
|
||||||
void ff_fdused_range(int max);
|
void ff_fdused_range(int max);
|
||||||
int ff_fdisused(int fd);
|
int ff_fdisused(int fd);
|
||||||
|
int ff_getmaxfd(void);
|
||||||
|
|
||||||
#endif /* _FSTACK_SYS_FILEDESC_H_ */
|
#endif /* _FSTACK_SYS_FILEDESC_H_ */
|
||||||
|
|
Loading…
Reference in New Issue