From f8a314ea075745c244172173391e44c146837b87 Mon Sep 17 00:00:00 2001 From: Garrett D'Amore Date: Fri, 29 Nov 2024 00:58:33 -0500 Subject: [PATCH] performance: reference counters can use relaxed order when incrementing --- src/core/platform.h | 7 +++++-- src/platform/posix/posix_atomic.c | 7 ++++--- src/platform/posix/posix_ipc.h | 8 ++++---- src/platform/posix/posix_ipcdial.c | 10 +++++----- src/platform/posix/posix_tcp.h | 8 ++++---- src/platform/posix/posix_tcpdial.c | 8 ++++---- src/platform/windows/win_thread.c | 6 ++++-- src/supplemental/http/http_server.c | 10 +++++----- 8 files changed, 35 insertions(+), 29 deletions(-) diff --git a/src/core/platform.h b/src/core/platform.h index 0249e2a02..d2409e5b3 100644 --- a/src/core/platform.h +++ b/src/core/platform.h @@ -197,8 +197,6 @@ extern void nni_atomic_sub64(nni_atomic_u64 *, uint64_t); extern uint64_t nni_atomic_get64(nni_atomic_u64 *); extern void nni_atomic_set64(nni_atomic_u64 *, uint64_t); extern uint64_t nni_atomic_swap64(nni_atomic_u64 *, uint64_t); -extern uint64_t nni_atomic_dec64_nv(nni_atomic_u64 *); -extern void nni_atomic_inc64(nni_atomic_u64 *); // nni_atomic_cas64 is a compare and swap. The second argument is the // value to compare against, and the third is the new value. Returns @@ -218,6 +216,11 @@ extern void nni_atomic_sub(nni_atomic_int *, int); extern int nni_atomic_get(nni_atomic_int *); extern void nni_atomic_set(nni_atomic_int *, int); extern int nni_atomic_swap(nni_atomic_int *, int); + +// These versions are tuned for use as reference +// counters. Relaxed order when possible to increase +// reference count, acquire-release order for dropping +// it (where we need to check the value). extern int nni_atomic_dec_nv(nni_atomic_int *); extern void nni_atomic_dec(nni_atomic_int *); extern void nni_atomic_inc(nni_atomic_int *); diff --git a/src/platform/posix/posix_atomic.c b/src/platform/posix/posix_atomic.c index 60875845e..69c2a2c1c 100644 --- a/src/platform/posix/posix_atomic.c +++ b/src/platform/posix/posix_atomic.c @@ -105,7 +105,7 @@ nni_atomic_swap(nni_atomic_int *v, int i) void nni_atomic_inc(nni_atomic_int *v) { - atomic_fetch_add(&v->v, 1); + atomic_fetch_add_explicit(&v->v, 1, memory_order_relaxed); } void @@ -117,7 +117,7 @@ nni_atomic_dec(nni_atomic_int *v) int nni_atomic_dec_nv(nni_atomic_int *v) { - return (atomic_fetch_sub(&v->v, 1) - 1); + return (atomic_fetch_sub_explicit(&v->v, 1, memory_order_acq_rel) - 1); } bool @@ -319,10 +319,11 @@ nni_atomic_get(nni_atomic_int *v) { return (__atomic_load_n(&v->v, __ATOMIC_SEQ_CST)); } + void nni_atomic_inc(nni_atomic_int *v) { - __atomic_add_fetch(&v->v, 1, __ATOMIC_SEQ_CST); + __atomic_add_fetch(&v->v, 1, __ATOMIC_RELAXED); } void diff --git a/src/platform/posix/posix_ipc.h b/src/platform/posix/posix_ipc.h index 4ef5fa80e..98af298b0 100644 --- a/src/platform/posix/posix_ipc.h +++ b/src/platform/posix/posix_ipc.h @@ -22,12 +22,12 @@ struct nni_ipc_conn { nng_stream stream; - nni_posix_pfd * pfd; + nni_posix_pfd *pfd; nni_list readq; nni_list writeq; bool closed; nni_mtx mtx; - nni_aio * dial_aio; + nni_aio *dial_aio; nni_ipc_dialer *dialer; nng_sockaddr sa; nni_reap_node reap; @@ -39,7 +39,7 @@ struct nni_ipc_dialer { bool closed; nni_mtx mtx; nng_sockaddr sa; - nni_atomic_u64 ref; + nni_atomic_int ref; nni_atomic_bool fini; }; @@ -51,4 +51,4 @@ extern void nni_posix_ipc_dialer_rele(nni_ipc_dialer *); #endif // NNG_PLATFORM_POSIX -#endif // PLATFORM_POSIX_IPC_H \ No newline at end of file +#endif // PLATFORM_POSIX_IPC_H diff --git a/src/platform/posix/posix_ipcdial.c b/src/platform/posix/posix_ipcdial.c index 1f7728b7b..7799aca2f 100644 --- a/src/platform/posix/posix_ipcdial.c +++ b/src/platform/posix/posix_ipcdial.c @@ -1,5 +1,5 @@ // -// Copyright 2023 Staysail Systems, Inc. +// Copyright 2024 Staysail Systems, Inc. // Copyright 2018 Capitar IT Group BV // Copyright 2019 Devolutions // @@ -69,7 +69,7 @@ ipc_dialer_free(void *arg) void nni_posix_ipc_dialer_rele(ipc_dialer *d) { - if (((nni_atomic_dec64_nv(&d->ref)) != 0) || + if (((nni_atomic_dec_nv(&d->ref)) != 0) || (!nni_atomic_get_bool(&d->fini))) { return; } @@ -176,7 +176,7 @@ ipc_dialer_dial(void *arg, nni_aio *aio) return; } - nni_atomic_inc64(&d->ref); + nni_atomic_inc(&d->ref); if ((rv = nni_posix_ipc_alloc(&c, &d->sa, d)) != 0) { (void) close(fd); @@ -323,8 +323,8 @@ nni_ipc_dialer_alloc(nng_stream_dialer **dp, const nng_url *url) d->sd.sd_get = ipc_dialer_get; d->sd.sd_set = ipc_dialer_set; nni_atomic_init_bool(&d->fini); - nni_atomic_init64(&d->ref); - nni_atomic_inc64(&d->ref); + nni_atomic_init(&d->ref); + nni_atomic_inc(&d->ref); *dp = (void *) d; return (0); diff --git a/src/platform/posix/posix_tcp.h b/src/platform/posix/posix_tcp.h index a8d1308e8..20504bf55 100644 --- a/src/platform/posix/posix_tcp.h +++ b/src/platform/posix/posix_tcp.h @@ -18,12 +18,12 @@ struct nni_tcp_conn { nng_stream stream; - nni_posix_pfd * pfd; + nni_posix_pfd *pfd; nni_list readq; nni_list writeq; bool closed; nni_mtx mtx; - nni_aio * dial_aio; + nni_aio *dial_aio; nni_tcp_dialer *dialer; nni_reap_node reap; }; @@ -36,7 +36,7 @@ struct nni_tcp_dialer { struct sockaddr_storage src; size_t srclen; nni_mtx mtx; - nni_atomic_u64 ref; + nni_atomic_int ref; nni_atomic_bool fini; }; @@ -45,4 +45,4 @@ extern void nni_posix_tcp_init(nni_tcp_conn *, nni_posix_pfd *); extern void nni_posix_tcp_start(nni_tcp_conn *, int, int); extern void nni_posix_tcp_dialer_rele(nni_tcp_dialer *); -#endif // PLATFORM_POSIX_TCP_H \ No newline at end of file +#endif // PLATFORM_POSIX_TCP_H diff --git a/src/platform/posix/posix_tcpdial.c b/src/platform/posix/posix_tcpdial.c index 3f0b07685..73ba43940 100644 --- a/src/platform/posix/posix_tcpdial.c +++ b/src/platform/posix/posix_tcpdial.c @@ -41,8 +41,8 @@ nni_tcp_dialer_init(nni_tcp_dialer **dp) d->nodelay = true; nni_aio_list_init(&d->connq); nni_atomic_init_bool(&d->fini); - nni_atomic_init64(&d->ref); - nni_atomic_inc64(&d->ref); + nni_atomic_init(&d->ref); + nni_atomic_inc(&d->ref); *dp = d; return (0); } @@ -87,7 +87,7 @@ nni_tcp_dialer_fini(nni_tcp_dialer *d) void nni_posix_tcp_dialer_rele(nni_tcp_dialer *d) { - if (((nni_atomic_dec64_nv(&d->ref) != 0)) || + if (((nni_atomic_dec_nv(&d->ref) != 0)) || (!nni_atomic_get_bool(&d->fini))) { return; } @@ -200,7 +200,7 @@ nni_tcp_dial(nni_tcp_dialer *d, const nni_sockaddr *sa, nni_aio *aio) return; } - nni_atomic_inc64(&d->ref); + nni_atomic_inc(&d->ref); if ((rv = nni_posix_tcp_alloc(&c, d)) != 0) { nni_aio_finish_error(aio, rv); diff --git a/src/platform/windows/win_thread.c b/src/platform/windows/win_thread.c index a6416cd86..9e74056dd 100644 --- a/src/platform/windows/win_thread.c +++ b/src/platform/windows/win_thread.c @@ -1,5 +1,5 @@ // -// Copyright 2021 Staysail Systems, Inc. +// Copyright 2024 Staysail Systems, Inc. // Copyright 2018 Capitar IT Group BV // // This software is supplied under the terms of the MIT License, a @@ -22,6 +22,8 @@ static pfnSetThreadDescription set_thread_desc; // mingw does not define InterlockedAddNoFence64, use the mingw equivalent #if defined(__MINGW32__) || defined(__MINGW64__) #define InterlockedAddNoFence(a, b) __atomic_add_fetch(a, b, __ATOMIC_RELAXED) +#define InterlockedIncrementNoFence(a) \ + __atomic_add_fetch(a, 1, __ATOMIC_RELAXED) #define InterlockedAddNoFence64(a, b) \ __atomic_add_fetch(a, b, __ATOMIC_RELAXED) #define InterlockedIncrementAcquire64(a) \ @@ -326,7 +328,7 @@ nni_atomic_init(nni_atomic_int *v) void nni_atomic_inc(nni_atomic_int *v) { - (void) InterlockedIncrementAcquire(&v->v); + (void) InterlockedIncrementNoFence(&v->v); } int diff --git a/src/supplemental/http/http_server.c b/src/supplemental/http/http_server.c index 49203a1cc..2240492f2 100644 --- a/src/supplemental/http/http_server.c +++ b/src/supplemental/http/http_server.c @@ -31,7 +31,7 @@ struct nng_http_handler { bool host_ip; bool tree; bool tree_exclusive; - nni_atomic_u64 ref; + nni_atomic_int ref; nni_atomic_bool busy; size_t maxbody; bool getbody; @@ -107,8 +107,8 @@ nni_http_handler_init( if ((h = NNI_ALLOC_STRUCT(h)) == NULL) { return (NNG_ENOMEM); } - nni_atomic_init64(&h->ref); - nni_atomic_inc64(&h->ref); + nni_atomic_init(&h->ref); + nni_atomic_inc(&h->ref); // Default for HTTP is /. But remap it to "" for ease of matching. if ((uri == NULL) || (strlen(uri) == 0) || (strcmp(uri, "/") == 0)) { @@ -137,7 +137,7 @@ nni_http_handler_init( void nni_http_handler_fini(nni_http_handler *h) { - if (nni_atomic_dec64_nv(&h->ref) != 0) { + if (nni_atomic_dec_nv(&h->ref) != 0) { return; } if (h->dtor != NULL) { @@ -735,7 +735,7 @@ http_sconn_rxdone(void *arg) // Set a reference -- this because the callback may be running // asynchronously even after it gets removed from the server. - nni_atomic_inc64(&h->ref); + nni_atomic_inc(&h->ref); // Documented that we call this on behalf of the callback. if (nni_aio_begin(sc->cbaio) != 0) {