mirror of https://github.com/grpc/grpc.git
Merge pull request #2713 from ctiller/plucking-hell
Allow specific pollers to be wokenpull/2809/head
commit
8158c6c309
40 changed files with 543 additions and 814 deletions
@ -1,168 +0,0 @@ |
||||
/*
|
||||
* |
||||
* Copyright 2015, Google Inc. |
||||
* All rights reserved. |
||||
* |
||||
* Redistribution and use in source and binary forms, with or without |
||||
* modification, are permitted provided that the following conditions are |
||||
* met: |
||||
* |
||||
* * Redistributions of source code must retain the above copyright |
||||
* notice, this list of conditions and the following disclaimer. |
||||
* * Redistributions in binary form must reproduce the above |
||||
* copyright notice, this list of conditions and the following disclaimer |
||||
* in the documentation and/or other materials provided with the |
||||
* distribution. |
||||
* * Neither the name of Google Inc. nor the names of its |
||||
* contributors may be used to endorse or promote products derived from |
||||
* this software without specific prior written permission. |
||||
* |
||||
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS |
||||
* "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT |
||||
* LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR |
||||
* A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT |
||||
* OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, |
||||
* SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT |
||||
* LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, |
||||
* DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY |
||||
* THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT |
||||
* (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE |
||||
* OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. |
||||
* |
||||
*/ |
||||
|
||||
#include <grpc/support/port_platform.h> |
||||
|
||||
#ifdef GPR_POSIX_SOCKET |
||||
#include "src/core/iomgr/pollset_kick_posix.h" |
||||
|
||||
#include <errno.h> |
||||
#include <string.h> |
||||
#include <unistd.h> |
||||
|
||||
#include "src/core/iomgr/socket_utils_posix.h" |
||||
#include "src/core/iomgr/wakeup_fd_posix.h" |
||||
#include <grpc/support/alloc.h> |
||||
#include <grpc/support/log.h> |
||||
|
||||
/* This implementation is based on a freelist of wakeup fds, with extra logic to
|
||||
* handle kicks while there is no attached fd. */ |
||||
|
||||
/* TODO(klempner): Autosize this, and consider providing a way to disable the
|
||||
* cap entirely on systems with large fd limits */ |
||||
#define GRPC_MAX_CACHED_WFDS 50 |
||||
|
||||
static grpc_kick_fd_info *fd_freelist = NULL; |
||||
static int fd_freelist_count = 0; |
||||
static gpr_mu fd_freelist_mu; |
||||
|
||||
static grpc_kick_fd_info *allocate_wfd(void) { |
||||
grpc_kick_fd_info *info = NULL; |
||||
gpr_mu_lock(&fd_freelist_mu); |
||||
if (fd_freelist != NULL) { |
||||
info = fd_freelist; |
||||
fd_freelist = fd_freelist->next; |
||||
--fd_freelist_count; |
||||
} |
||||
gpr_mu_unlock(&fd_freelist_mu); |
||||
if (info == NULL) { |
||||
info = gpr_malloc(sizeof(*info)); |
||||
grpc_wakeup_fd_create(&info->wakeup_fd); |
||||
info->next = NULL; |
||||
} |
||||
return info; |
||||
} |
||||
|
||||
static void destroy_wfd(grpc_kick_fd_info *wfd) { |
||||
grpc_wakeup_fd_destroy(&wfd->wakeup_fd); |
||||
gpr_free(wfd); |
||||
} |
||||
|
||||
static void free_wfd(grpc_kick_fd_info *fd_info) { |
||||
gpr_mu_lock(&fd_freelist_mu); |
||||
if (fd_freelist_count < GRPC_MAX_CACHED_WFDS) { |
||||
fd_info->next = fd_freelist; |
||||
fd_freelist = fd_info; |
||||
fd_freelist_count++; |
||||
fd_info = NULL; |
||||
} |
||||
gpr_mu_unlock(&fd_freelist_mu); |
||||
|
||||
if (fd_info) { |
||||
destroy_wfd(fd_info); |
||||
} |
||||
} |
||||
|
||||
void grpc_pollset_kick_init(grpc_pollset_kick_state *kick_state) { |
||||
gpr_mu_init(&kick_state->mu); |
||||
kick_state->kicked = 0; |
||||
kick_state->fd_list.next = kick_state->fd_list.prev = &kick_state->fd_list; |
||||
} |
||||
|
||||
void grpc_pollset_kick_destroy(grpc_pollset_kick_state *kick_state) { |
||||
gpr_mu_destroy(&kick_state->mu); |
||||
GPR_ASSERT(kick_state->fd_list.next == &kick_state->fd_list); |
||||
} |
||||
|
||||
grpc_kick_fd_info *grpc_pollset_kick_pre_poll( |
||||
grpc_pollset_kick_state *kick_state) { |
||||
grpc_kick_fd_info *fd_info; |
||||
gpr_mu_lock(&kick_state->mu); |
||||
if (kick_state->kicked) { |
||||
kick_state->kicked = 0; |
||||
gpr_mu_unlock(&kick_state->mu); |
||||
return NULL; |
||||
} |
||||
fd_info = allocate_wfd(); |
||||
fd_info->next = &kick_state->fd_list; |
||||
fd_info->prev = fd_info->next->prev; |
||||
fd_info->next->prev = fd_info->prev->next = fd_info; |
||||
gpr_mu_unlock(&kick_state->mu); |
||||
return fd_info; |
||||
} |
||||
|
||||
void grpc_pollset_kick_consume(grpc_pollset_kick_state *kick_state, |
||||
grpc_kick_fd_info *fd_info) { |
||||
grpc_wakeup_fd_consume_wakeup(&fd_info->wakeup_fd); |
||||
} |
||||
|
||||
void grpc_pollset_kick_post_poll(grpc_pollset_kick_state *kick_state, |
||||
grpc_kick_fd_info *fd_info) { |
||||
gpr_mu_lock(&kick_state->mu); |
||||
fd_info->next->prev = fd_info->prev; |
||||
fd_info->prev->next = fd_info->next; |
||||
free_wfd(fd_info); |
||||
gpr_mu_unlock(&kick_state->mu); |
||||
} |
||||
|
||||
void grpc_pollset_kick_kick(grpc_pollset_kick_state *kick_state) { |
||||
gpr_mu_lock(&kick_state->mu); |
||||
if (kick_state->fd_list.next != &kick_state->fd_list) { |
||||
grpc_wakeup_fd_wakeup(&kick_state->fd_list.next->wakeup_fd); |
||||
} else { |
||||
kick_state->kicked = 1; |
||||
} |
||||
gpr_mu_unlock(&kick_state->mu); |
||||
} |
||||
|
||||
void grpc_pollset_kick_global_init_fallback_fd(void) { |
||||
gpr_mu_init(&fd_freelist_mu); |
||||
grpc_wakeup_fd_global_init_force_fallback(); |
||||
} |
||||
|
||||
void grpc_pollset_kick_global_init(void) { |
||||
gpr_mu_init(&fd_freelist_mu); |
||||
grpc_wakeup_fd_global_init(); |
||||
} |
||||
|
||||
void grpc_pollset_kick_global_destroy(void) { |
||||
while (fd_freelist != NULL) { |
||||
grpc_kick_fd_info *current = fd_freelist; |
||||
fd_freelist = fd_freelist->next; |
||||
destroy_wfd(current); |
||||
} |
||||
grpc_wakeup_fd_global_destroy(); |
||||
gpr_mu_destroy(&fd_freelist_mu); |
||||
} |
||||
|
||||
#endif /* GPR_POSIX_SOCKET */ |
@ -1,93 +0,0 @@ |
||||
/*
|
||||
* |
||||
* Copyright 2015, Google Inc. |
||||
* All rights reserved. |
||||
* |
||||
* Redistribution and use in source and binary forms, with or without |
||||
* modification, are permitted provided that the following conditions are |
||||
* met: |
||||
* |
||||
* * Redistributions of source code must retain the above copyright |
||||
* notice, this list of conditions and the following disclaimer. |
||||
* * Redistributions in binary form must reproduce the above |
||||
* copyright notice, this list of conditions and the following disclaimer |
||||
* in the documentation and/or other materials provided with the |
||||
* distribution. |
||||
* * Neither the name of Google Inc. nor the names of its |
||||
* contributors may be used to endorse or promote products derived from |
||||
* this software without specific prior written permission. |
||||
* |
||||
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS |
||||
* "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT |
||||
* LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR |
||||
* A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT |
||||
* OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, |
||||
* SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT |
||||
* LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, |
||||
* DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY |
||||
* THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT |
||||
* (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE |
||||
* OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. |
||||
* |
||||
*/ |
||||
|
||||
#ifndef GRPC_INTERNAL_CORE_IOMGR_POLLSET_KICK_POSIX_H |
||||
#define GRPC_INTERNAL_CORE_IOMGR_POLLSET_KICK_POSIX_H |
||||
|
||||
#include "src/core/iomgr/wakeup_fd_posix.h" |
||||
#include <grpc/support/sync.h> |
||||
|
||||
/* pollset kicking allows breaking a thread out of polling work for
|
||||
a given pollset. |
||||
writing a byte to a pipe is used as a posix-ly portable base |
||||
mechanism, and eventfds are utilized on Linux for better performance. */ |
||||
|
||||
typedef struct grpc_kick_fd_info { |
||||
grpc_wakeup_fd_info wakeup_fd; |
||||
/* used for polling list and free list */ |
||||
struct grpc_kick_fd_info *next; |
||||
/* only used when polling */ |
||||
struct grpc_kick_fd_info *prev; |
||||
} grpc_kick_fd_info; |
||||
|
||||
typedef struct grpc_pollset_kick_state { |
||||
gpr_mu mu; |
||||
int kicked; |
||||
struct grpc_kick_fd_info fd_list; |
||||
} grpc_pollset_kick_state; |
||||
|
||||
#define GRPC_POLLSET_KICK_GET_FD(kick_fd_info) \ |
||||
GRPC_WAKEUP_FD_GET_READ_FD(&(kick_fd_info)->wakeup_fd) |
||||
|
||||
/* This is an abstraction around the typical pipe mechanism for waking up a
|
||||
thread sitting in a poll() style call. */ |
||||
|
||||
void grpc_pollset_kick_global_init(void); |
||||
void grpc_pollset_kick_global_destroy(void); |
||||
|
||||
void grpc_pollset_kick_init(grpc_pollset_kick_state *kick_state); |
||||
void grpc_pollset_kick_destroy(grpc_pollset_kick_state *kick_state); |
||||
|
||||
/* Guarantees a pure posix implementation rather than a specialized one, if
|
||||
* applicable. Intended for testing. */ |
||||
void grpc_pollset_kick_global_init_fallback_fd(void); |
||||
|
||||
/* Must be called before entering poll(). If return value is NULL, this consumed
|
||||
an existing kick. Otherwise the return value is an FD to add to the poll set. |
||||
*/ |
||||
grpc_kick_fd_info *grpc_pollset_kick_pre_poll( |
||||
grpc_pollset_kick_state *kick_state); |
||||
|
||||
/* Consume an existing kick. Must be called after poll returns that the fd was
|
||||
readable, and before calling kick_post_poll. */ |
||||
void grpc_pollset_kick_consume(grpc_pollset_kick_state *kick_state, |
||||
grpc_kick_fd_info *fd_info); |
||||
|
||||
/* Must be called after pre_poll, and after consume if applicable */ |
||||
void grpc_pollset_kick_post_poll(grpc_pollset_kick_state *kick_state, |
||||
grpc_kick_fd_info *fd_info); |
||||
|
||||
/* Actually kick */ |
||||
void grpc_pollset_kick_kick(grpc_pollset_kick_state *kick_state); |
||||
|
||||
#endif /* GRPC_INTERNAL_CORE_IOMGR_POLLSET_KICK_POSIX_H */ |
@ -1,130 +0,0 @@ |
||||
/*
|
||||
* |
||||
* Copyright 2015, Google Inc. |
||||
* All rights reserved. |
||||
* |
||||
* Redistribution and use in source and binary forms, with or without |
||||
* modification, are permitted provided that the following conditions are |
||||
* met: |
||||
* |
||||
* * Redistributions of source code must retain the above copyright |
||||
* notice, this list of conditions and the following disclaimer. |
||||
* * Redistributions in binary form must reproduce the above |
||||
* copyright notice, this list of conditions and the following disclaimer |
||||
* in the documentation and/or other materials provided with the |
||||
* distribution. |
||||
* * Neither the name of Google Inc. nor the names of its |
||||
* contributors may be used to endorse or promote products derived from |
||||
* this software without specific prior written permission. |
||||
* |
||||
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS |
||||
* "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT |
||||
* LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR |
||||
* A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT |
||||
* OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, |
||||
* SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT |
||||
* LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, |
||||
* DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY |
||||
* THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT |
||||
* (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE |
||||
* OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. |
||||
* |
||||
*/ |
||||
|
||||
#include "src/core/iomgr/pollset_kick_posix.h" |
||||
|
||||
#include <grpc/support/alloc.h> |
||||
#include <grpc/support/log.h> |
||||
#include "test/core/util/test_config.h" |
||||
|
||||
static void test_allocation(void) { |
||||
grpc_pollset_kick_state state; |
||||
grpc_pollset_kick_init(&state); |
||||
grpc_pollset_kick_destroy(&state); |
||||
} |
||||
|
||||
static void test_non_kick(void) { |
||||
grpc_pollset_kick_state state; |
||||
grpc_kick_fd_info *kfd; |
||||
|
||||
grpc_pollset_kick_init(&state); |
||||
kfd = grpc_pollset_kick_pre_poll(&state); |
||||
GPR_ASSERT(kfd != NULL); |
||||
|
||||
grpc_pollset_kick_post_poll(&state, kfd); |
||||
grpc_pollset_kick_destroy(&state); |
||||
} |
||||
|
||||
static void test_basic_kick(void) { |
||||
/* Kicked during poll */ |
||||
grpc_pollset_kick_state state; |
||||
grpc_kick_fd_info *kfd; |
||||
grpc_pollset_kick_init(&state); |
||||
|
||||
kfd = grpc_pollset_kick_pre_poll(&state); |
||||
GPR_ASSERT(kfd != NULL); |
||||
|
||||
grpc_pollset_kick_kick(&state); |
||||
|
||||
/* Now hypothetically we polled and found that we were kicked */ |
||||
grpc_pollset_kick_consume(&state, kfd); |
||||
|
||||
grpc_pollset_kick_post_poll(&state, kfd); |
||||
|
||||
grpc_pollset_kick_destroy(&state); |
||||
} |
||||
|
||||
static void test_non_poll_kick(void) { |
||||
/* Kick before entering poll */ |
||||
grpc_pollset_kick_state state; |
||||
grpc_kick_fd_info *kfd; |
||||
|
||||
grpc_pollset_kick_init(&state); |
||||
|
||||
grpc_pollset_kick_kick(&state); |
||||
kfd = grpc_pollset_kick_pre_poll(&state); |
||||
GPR_ASSERT(kfd == NULL); |
||||
grpc_pollset_kick_destroy(&state); |
||||
} |
||||
|
||||
#define GRPC_MAX_CACHED_PIPES 50 |
||||
|
||||
static void test_over_free(void) { |
||||
/* Check high watermark pipe free logic */ |
||||
int i; |
||||
grpc_kick_fd_info **kfds = |
||||
gpr_malloc(sizeof(grpc_kick_fd_info *) * GRPC_MAX_CACHED_PIPES); |
||||
grpc_pollset_kick_state state; |
||||
grpc_pollset_kick_init(&state); |
||||
for (i = 0; i < GRPC_MAX_CACHED_PIPES; ++i) { |
||||
kfds[i] = grpc_pollset_kick_pre_poll(&state); |
||||
GPR_ASSERT(kfds[i] != NULL); |
||||
} |
||||
|
||||
for (i = 0; i < GRPC_MAX_CACHED_PIPES; ++i) { |
||||
grpc_pollset_kick_post_poll(&state, kfds[i]); |
||||
} |
||||
grpc_pollset_kick_destroy(&state); |
||||
gpr_free(kfds); |
||||
} |
||||
|
||||
static void run_tests(void) { |
||||
test_allocation(); |
||||
test_basic_kick(); |
||||
test_non_poll_kick(); |
||||
test_non_kick(); |
||||
test_over_free(); |
||||
} |
||||
|
||||
int main(int argc, char **argv) { |
||||
grpc_test_init(argc, argv); |
||||
|
||||
grpc_pollset_kick_global_init(); |
||||
run_tests(); |
||||
grpc_pollset_kick_global_destroy(); |
||||
|
||||
grpc_pollset_kick_global_init_fallback_fd(); |
||||
run_tests(); |
||||
grpc_pollset_kick_global_destroy(); |
||||
return 0; |
||||
} |
Loading…
Reference in new issue