commit: 3472fc87566576da097be49d35da77fe5ab4e760 Author: Patrick McLean <chutzpah <AT> gentoo <DOT> org> AuthorDate: Fri Sep 9 06:42:39 2016 +0000 Commit: Patrick McLean <chutzpah <AT> gentoo <DOT> org> CommitDate: Fri Sep 9 06:43:14 2016 +0000 URL: https://gitweb.gentoo.org/repo/gentoo.git/commit/?id=3472fc87
net-misc/openssh: Update the ctr-mt patch to the final version upstream Package-Manager: portage-2.3.0 ...ssh-7.3_p1-hpn-cipher-ctr-mt-no-deadlocks.patch | 157 ++++++++++++--------- 1 file changed, 92 insertions(+), 65 deletions(-) diff --git a/net-misc/openssh/files/openssh-7.3_p1-hpn-cipher-ctr-mt-no-deadlocks.patch b/net-misc/openssh/files/openssh-7.3_p1-hpn-cipher-ctr-mt-no-deadlocks.patch index cac4406..19bd08b 100644 --- a/net-misc/openssh/files/openssh-7.3_p1-hpn-cipher-ctr-mt-no-deadlocks.patch +++ b/net-misc/openssh/files/openssh-7.3_p1-hpn-cipher-ctr-mt-no-deadlocks.patch @@ -1,5 +1,5 @@ diff --git a/cipher-ctr-mt.c b/cipher-ctr-mt.c -index fdc9b2f..0b35881 100644 +index fdc9b2f..3ffe6d1 100644 --- a/cipher-ctr-mt.c +++ b/cipher-ctr-mt.c @@ -127,7 +127,7 @@ struct kq { @@ -11,23 +11,27 @@ index fdc9b2f..0b35881 100644 pthread_mutex_t lock; pthread_cond_t cond; u_char pad1[CACHELINE_LEN]; -@@ -141,6 +141,9 @@ struct ssh_aes_ctr_ctx +@@ -141,6 +141,11 @@ struct ssh_aes_ctr_ctx STATS_STRUCT(stats); u_char aes_counter[AES_BLOCK_SIZE]; pthread_t tid[CIPHER_THREADS]; + pthread_rwlock_t tid_lock; ++#ifdef __APPLE__ + pthread_rwlock_t stop_lock; + int exit_flag; ++#endif /* __APPLE__ */ int state; int qidx; int ridx; -@@ -187,6 +190,23 @@ thread_loop_cleanup(void *x) +@@ -187,6 +192,57 @@ thread_loop_cleanup(void *x) pthread_mutex_unlock((pthread_mutex_t *)x); } ++#ifdef __APPLE__ +/* Check if we should exit, we are doing both cancel and exit condition -+ * since OSX seems to misbehave with cancel sometimes, so we want to have -+ * a backup to make sure that everything exits properly ++ * since on OSX threads seem to occasionally fail to notice when they have ++ * been cancelled. We want to have a backup to make sure that we won't hang ++ * when the main process join()-s the cancelled thread. + */ +static void +thread_loop_check_exit(struct ssh_aes_ctr_ctx *c) @@ -38,14 +42,46 @@ index fdc9b2f..0b35881 100644 + exit_flag = c->exit_flag; + pthread_rwlock_unlock(&c->stop_lock); + -+ if (exit_flag == TRUE) ++ if (exit_flag) + pthread_exit(NULL); +} ++#else ++# define thread_loop_check_exit(s) ++#endif /* __APPLE__ */ ++ ++/* ++ * Helper function to terminate the helper threads ++ */ ++static void ++stop_and_join_pregen_threads(struct ssh_aes_ctr_ctx *c) ++{ ++ int i; ++ ++#ifdef __APPLE__ ++ /* notify threads that they should exit */ ++ pthread_rwlock_wrlock(&c->stop_lock); ++ c->exit_flag = TRUE; ++ pthread_rwlock_unlock(&c->stop_lock); ++#endif /* __APPLE__ */ ++ ++ /* Cancel pregen threads */ ++ for (i = 0; i < CIPHER_THREADS; i++) { ++ pthread_cancel(c->tid[i]); ++ } ++ for (i = 0; i < NUMKQ; i++) { ++ pthread_mutex_lock(&c->q[i].lock); ++ pthread_cond_broadcast(&c->q[i].cond); ++ pthread_mutex_unlock(&c->q[i].lock); ++ } ++ for (i = 0; i < CIPHER_THREADS; i++) { ++ pthread_join(c->tid[i], NULL); ++ } ++} + /* * The life of a pregen thread: * Find empty keystream queues and fill them using their counter. -@@ -201,6 +221,7 @@ thread_loop(void *x) +@@ -201,6 +257,7 @@ thread_loop(void *x) struct kq *q; int i; int qidx; @@ -53,7 +89,7 @@ index fdc9b2f..0b35881 100644 /* Threads stats on cancellation */ STATS_INIT(stats); -@@ -211,11 +232,15 @@ thread_loop(void *x) +@@ -211,11 +268,15 @@ thread_loop(void *x) /* Thread local copy of AES key */ memcpy(&key, &c->aes_ctx, sizeof(key)); @@ -70,7 +106,7 @@ index fdc9b2f..0b35881 100644 q = &c->q[0]; pthread_mutex_lock(&q->lock); if (q->qstate == KQINIT) { -@@ -245,12 +270,16 @@ thread_loop(void *x) +@@ -245,12 +306,16 @@ thread_loop(void *x) /* Check if I was cancelled, also checked in cond_wait */ pthread_testcancel(); @@ -87,7 +123,7 @@ index fdc9b2f..0b35881 100644 pthread_cond_wait(&q->cond, &q->lock); } pthread_cleanup_pop(0); -@@ -268,6 +297,7 @@ thread_loop(void *x) +@@ -268,6 +333,7 @@ thread_loop(void *x) * can see that it's being filled. */ q->qstate = KQFILLING; @@ -95,7 +131,7 @@ index fdc9b2f..0b35881 100644 pthread_mutex_unlock(&q->lock); for (i = 0; i < KQLEN; i++) { AES_encrypt(q->ctr, q->keys[i], &key); -@@ -279,7 +309,7 @@ thread_loop(void *x) +@@ -279,7 +345,7 @@ thread_loop(void *x) ssh_ctr_add(q->ctr, KQLEN * (NUMKQ - 1), AES_BLOCK_SIZE); q->qstate = KQFULL; STATS_FILL(stats); @@ -104,7 +140,7 @@ index fdc9b2f..0b35881 100644 pthread_mutex_unlock(&q->lock); } -@@ -371,6 +401,7 @@ ssh_aes_ctr(EVP_CIPHER_CTX *ctx, u_char *dest, const u_char *src, +@@ -371,6 +437,7 @@ ssh_aes_ctr(EVP_CIPHER_CTX *ctx, u_char *dest, const u_char *src, pthread_cond_wait(&q->cond, &q->lock); } q->qstate = KQDRAINING; @@ -112,40 +148,34 @@ index fdc9b2f..0b35881 100644 pthread_mutex_unlock(&q->lock); /* Mark consumed queue empty and signal producers */ -@@ -397,6 +428,9 @@ ssh_aes_ctr_init(EVP_CIPHER_CTX *ctx, const u_char *key, const u_char *iv, +@@ -397,6 +464,11 @@ ssh_aes_ctr_init(EVP_CIPHER_CTX *ctx, const u_char *key, const u_char *iv, if ((c = EVP_CIPHER_CTX_get_app_data(ctx)) == NULL) { c = xmalloc(sizeof(*c)); + pthread_rwlock_init(&c->tid_lock, NULL); ++#ifdef __APPLE__ + pthread_rwlock_init(&c->stop_lock, NULL); + c->exit_flag = FALSE; ++#endif /* __APPLE__ */ c->state = HAVE_NONE; for (i = 0; i < NUMKQ; i++) { -@@ -409,11 +443,22 @@ ssh_aes_ctr_init(EVP_CIPHER_CTX *ctx, const u_char *key, const u_char *iv, +@@ -409,11 +481,9 @@ ssh_aes_ctr_init(EVP_CIPHER_CTX *ctx, const u_char *key, const u_char *iv, } if (c->state == (HAVE_KEY | HAVE_IV)) { +- /* Cancel pregen threads */ +- for (i = 0; i < CIPHER_THREADS; i++) +- pthread_cancel(c->tid[i]); +- for (i = 0; i < CIPHER_THREADS; i++) +- pthread_join(c->tid[i], NULL); + /* tell the pregen threads to exit */ -+ pthread_rwlock_wrlock(&c->stop_lock); -+ c->exit_flag = TRUE; -+ pthread_rwlock_unlock(&c->stop_lock); -+ - /* Cancel pregen threads */ - for (i = 0; i < CIPHER_THREADS; i++) - pthread_cancel(c->tid[i]); -+ for (i = 0; i < NUMKQ; i++) { -+ pthread_mutex_lock(&c->q[i].lock); -+ pthread_cond_broadcast(&c->q[i].cond); -+ pthread_mutex_unlock(&c->q[i].lock); -+ } - for (i = 0; i < CIPHER_THREADS; i++) - pthread_join(c->tid[i], NULL); ++ stop_and_join_pregen_threads(c); + /* Start over getting key & iv */ c->state = HAVE_NONE; } -@@ -444,10 +489,12 @@ ssh_aes_ctr_init(EVP_CIPHER_CTX *ctx, const u_char *key, const u_char *iv, +@@ -444,10 +514,12 @@ ssh_aes_ctr_init(EVP_CIPHER_CTX *ctx, const u_char *key, const u_char *iv, /* Start threads */ for (i = 0; i < CIPHER_THREADS; i++) { debug("spawned a thread"); @@ -159,29 +189,26 @@ index fdc9b2f..0b35881 100644 pthread_cond_wait(&c->q[0].cond, &c->q[0].lock); pthread_mutex_unlock(&c->q[0].lock); } -@@ -463,10 +510,21 @@ ssh_aes_ctr_thread_destroy(EVP_CIPHER_CTX *ctx) +@@ -461,15 +533,10 @@ void + ssh_aes_ctr_thread_destroy(EVP_CIPHER_CTX *ctx) + { struct ssh_aes_ctr_ctx *c; - int i; - c = EVP_CIPHER_CTX_get_app_data(ctx); +- int i; + -+ /* notify threads that they should exit */ -+ pthread_rwlock_wrlock(&c->stop_lock); -+ c->exit_flag = TRUE; -+ pthread_rwlock_unlock(&c->stop_lock); + c = EVP_CIPHER_CTX_get_app_data(ctx); +- /* destroy threads */ +- for (i = 0; i < CIPHER_THREADS; i++) { +- pthread_cancel(c->tid[i]); +- } +- for (i = 0; i < CIPHER_THREADS; i++) { +- pthread_join(c->tid[i], NULL); +- } + - /* destroy threads */ - for (i = 0; i < CIPHER_THREADS; i++) { - pthread_cancel(c->tid[i]); - } -+ for (i = 0; i < NUMKQ; i++) { -+ pthread_mutex_lock(&c->q[i].lock); -+ pthread_cond_broadcast(&c->q[i].cond); -+ pthread_mutex_unlock(&c->q[i].lock); -+ } - for (i = 0; i < CIPHER_THREADS; i++) { - pthread_join(c->tid[i], NULL); - } -@@ -481,7 +539,9 @@ ssh_aes_ctr_thread_reconstruction(EVP_CIPHER_CTX *ctx) ++ stop_and_join_pregen_threads(c); + } + + void +@@ -481,7 +548,9 @@ ssh_aes_ctr_thread_reconstruction(EVP_CIPHER_CTX *ctx) /* reconstruct threads */ for (i = 0; i < CIPHER_THREADS; i++) { debug("spawned a thread"); @@ -191,23 +218,23 @@ index fdc9b2f..0b35881 100644 } } -@@ -496,9 +556,19 @@ ssh_aes_ctr_cleanup(EVP_CIPHER_CTX *ctx) +@@ -489,18 +558,13 @@ static int + ssh_aes_ctr_cleanup(EVP_CIPHER_CTX *ctx) + { + struct ssh_aes_ctr_ctx *c; +- int i; + + if ((c = EVP_CIPHER_CTX_get_app_data(ctx)) != NULL) { + #ifdef CIPHER_THREAD_STATS debug("main thread: %u drains, %u waits", c->stats.drains, c->stats.waits); #endif -+ /* tell the pregen threads to exit */ -+ pthread_rwlock_wrlock(&c->stop_lock); -+ c->exit_flag = TRUE; -+ pthread_rwlock_unlock(&c->stop_lock); -+ - /* Cancel pregen threads */ - for (i = 0; i < CIPHER_THREADS; i++) - pthread_cancel(c->tid[i]); -+ for (i = 0; i < NUMKQ; i++) { -+ pthread_mutex_lock(&c->q[i].lock); -+ pthread_cond_broadcast(&c->q[i].cond); -+ pthread_mutex_unlock(&c->q[i].lock); -+ } - for (i = 0; i < CIPHER_THREADS; i++) - pthread_join(c->tid[i], NULL); +- /* Cancel pregen threads */ +- for (i = 0; i < CIPHER_THREADS; i++) +- pthread_cancel(c->tid[i]); +- for (i = 0; i < CIPHER_THREADS; i++) +- pthread_join(c->tid[i], NULL); ++ stop_and_join_pregen_threads(c); + memset(c, 0, sizeof(*c)); + free(c);