mirror of
https://github.com/nghttp2/nghttp2.git
synced 2025-12-07 18:48:54 +08:00
Merge pull request #2309 from nghttp2/nghttpx-rework-quic-conn
nghttpx: Rework QUIC connection handling
This commit is contained in:
@@ -148,6 +148,7 @@ if(ENABLE_APP)
|
||||
shrpx_http3_upstream.cc
|
||||
http3.cc
|
||||
quic.cc
|
||||
siphash.cc
|
||||
)
|
||||
endif()
|
||||
add_library(nghttpx_static STATIC ${NGHTTPX_SRCS})
|
||||
@@ -190,6 +191,9 @@ if(ENABLE_APP)
|
||||
base64_test.cc
|
||||
${CMAKE_SOURCE_DIR}/tests/munit/munit.c
|
||||
)
|
||||
if(ENABLE_HTTP3)
|
||||
list(APPEND NGHTTPX_UNITTEST_SOURCES siphash_test.cc)
|
||||
endif()
|
||||
add_executable(nghttpx-unittest EXCLUDE_FROM_ALL
|
||||
${NGHTTPX_UNITTEST_SOURCES}
|
||||
$<TARGET_OBJECTS:llhttp>
|
||||
|
||||
@@ -189,7 +189,8 @@ NGHTTPX_SRCS += \
|
||||
shrpx_quic_connection_handler.cc shrpx_quic_connection_handler.h \
|
||||
shrpx_http3_upstream.cc shrpx_http3_upstream.h \
|
||||
http3.cc http3.h \
|
||||
quic.cc quic.h
|
||||
quic.cc quic.h \
|
||||
siphash.cc siphash.h
|
||||
endif # ENABLE_HTTP3
|
||||
|
||||
noinst_LIBRARIES = libnghttpx.a
|
||||
@@ -229,6 +230,9 @@ nghttpx_unittest_SOURCES = shrpx-unittest.cc \
|
||||
base64_test.cc base64_test.h \
|
||||
$(top_srcdir)/tests/munit/munit.c $(top_srcdir)/tests/munit/munit.h \
|
||||
$(top_srcdir)/tests/munit/munitxx.h
|
||||
if ENABLE_HTTP3
|
||||
nghttpx_unittest_SOURCES += siphash_test.cc siphash_test.h
|
||||
endif # ENABLE_HTTP3
|
||||
nghttpx_unittest_CPPFLAGS = ${AM_CPPFLAGS} \
|
||||
-I$(top_srcdir)/tests/munit \
|
||||
-DNGHTTP2_SRC_DIR=\"$(top_srcdir)/src\"
|
||||
|
||||
@@ -45,18 +45,31 @@
|
||||
#include "tls.h"
|
||||
#include "shrpx_router_test.h"
|
||||
#include "shrpx_log.h"
|
||||
#ifdef ENABLE_HTTP3
|
||||
# include "siphash_test.h"
|
||||
#endif // ENABLE_HTTP3
|
||||
|
||||
int main(int argc, char *argv[]) {
|
||||
shrpx::create_config();
|
||||
|
||||
const MunitSuite suites[] = {
|
||||
shrpx::tls_suite, shrpx::downstream_suite,
|
||||
shrpx::config_suite, shrpx::worker_suite,
|
||||
shrpx::http_suite, shrpx::router_suite,
|
||||
shrpx::http2_suite, shrpx::util_suite,
|
||||
gzip_suite, buffer_suite,
|
||||
memchunk_suite, template_suite,
|
||||
base64_suite, {},
|
||||
shrpx::tls_suite,
|
||||
shrpx::downstream_suite,
|
||||
shrpx::config_suite,
|
||||
shrpx::worker_suite,
|
||||
shrpx::http_suite,
|
||||
shrpx::router_suite,
|
||||
shrpx::http2_suite,
|
||||
shrpx::util_suite,
|
||||
gzip_suite,
|
||||
buffer_suite,
|
||||
memchunk_suite,
|
||||
template_suite,
|
||||
base64_suite,
|
||||
#ifdef ENABLE_HTTP3
|
||||
siphash_suite,
|
||||
#endif // ENABLE_HTTP3
|
||||
{},
|
||||
};
|
||||
const MunitSuite suite = {
|
||||
"", nullptr, suites, 1, MUNIT_SUITE_OPTION_NONE,
|
||||
|
||||
@@ -38,31 +38,32 @@
|
||||
#ifdef NGHTTP2_OPENSSL_IS_WOLFSSL
|
||||
# include <wolfssl/options.h>
|
||||
# include <wolfssl/openssl/evp.h>
|
||||
# include <wolfssl/openssl/rand.h>
|
||||
#else // !NGHTTP2_OPENSSL_IS_WOLFSSL
|
||||
# include <openssl/evp.h>
|
||||
# include <openssl/rand.h>
|
||||
#endif // !NGHTTP2_OPENSSL_IS_WOLFSSL
|
||||
|
||||
#include <ngtcp2/ngtcp2.h>
|
||||
|
||||
#include "siphash.h"
|
||||
#include "template.h"
|
||||
#include "network.h"
|
||||
|
||||
using namespace nghttp2;
|
||||
|
||||
namespace std {
|
||||
template <> struct hash<ngtcp2_cid> {
|
||||
std::size_t operator()(const ngtcp2_cid &cid) const noexcept {
|
||||
// FNV-1a 64bits variant
|
||||
constexpr uint64_t basis = 0xCBF29CE484222325ULL;
|
||||
const uint8_t *p = cid.data, *end = cid.data + cid.datalen;
|
||||
uint64_t h = basis;
|
||||
|
||||
for (; p != end;) {
|
||||
h ^= *p++;
|
||||
h *= basis;
|
||||
}
|
||||
|
||||
return static_cast<size_t>(h);
|
||||
hash() {
|
||||
auto s = as_writable_uint8_span(std::span{key});
|
||||
assert(RAND_bytes(s.data(), s.size()) == 1);
|
||||
}
|
||||
|
||||
std::size_t operator()(const ngtcp2_cid &cid) const noexcept {
|
||||
return static_cast<size_t>(siphash24(key, {cid.data, cid.datalen}));
|
||||
}
|
||||
|
||||
std::array<uint64_t, 2> key;
|
||||
};
|
||||
} // namespace std
|
||||
|
||||
|
||||
114
src/siphash.cc
Normal file
114
src/siphash.cc
Normal file
@@ -0,0 +1,114 @@
|
||||
/* Copyright 2019 The BoringSSL Authors
|
||||
*
|
||||
* Permission to use, copy, modify, and/or distribute this software for any
|
||||
* purpose with or without fee is hereby granted, provided that the above
|
||||
* copyright notice and this permission notice appear in all copies.
|
||||
*
|
||||
* THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
|
||||
* WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
|
||||
* MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY
|
||||
* SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
|
||||
* WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN ACTION
|
||||
* OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF OR IN
|
||||
* CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE. */
|
||||
|
||||
/*
|
||||
* nghttp2 - HTTP/2 C Library
|
||||
*
|
||||
* Copyright (c) 2025 nghttp2 contributors
|
||||
*
|
||||
* Permission is hereby granted, free of charge, to any person obtaining
|
||||
* a copy of this software and associated documentation files (the
|
||||
* "Software"), to deal in the Software without restriction, including
|
||||
* without limitation the rights to use, copy, modify, merge, publish,
|
||||
* distribute, sublicense, and/or sell copies of the Software, and to
|
||||
* permit persons to whom the Software is furnished to do so, subject to
|
||||
* the following conditions:
|
||||
*
|
||||
* The above copyright notice and this permission notice shall be
|
||||
* included in all copies or substantial portions of the Software.
|
||||
*
|
||||
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
|
||||
* EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
|
||||
* MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
|
||||
* NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
|
||||
* LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
|
||||
* OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
|
||||
* WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
|
||||
*/
|
||||
#include <cstdint>
|
||||
#include <cstring>
|
||||
|
||||
#include "siphash.h"
|
||||
|
||||
namespace {
|
||||
auto CRYPTO_load_u64_le(std::span<const uint8_t, sizeof(uint64_t)> in) {
|
||||
uint64_t v;
|
||||
|
||||
memcpy(&v, in.data(), sizeof(v));
|
||||
|
||||
if constexpr (std::endian::native == std::endian::big) {
|
||||
return byteswap(v);
|
||||
}
|
||||
|
||||
return v;
|
||||
}
|
||||
} // namespace
|
||||
|
||||
namespace {
|
||||
constexpr void siphash_round(uint64_t v[4]) {
|
||||
v[0] += v[1];
|
||||
v[2] += v[3];
|
||||
v[1] = std::rotl(v[1], 13);
|
||||
v[3] = std::rotl(v[3], 16);
|
||||
v[1] ^= v[0];
|
||||
v[3] ^= v[2];
|
||||
v[0] = std::rotl(v[0], 32);
|
||||
v[2] += v[1];
|
||||
v[0] += v[3];
|
||||
v[1] = std::rotl(v[1], 17);
|
||||
v[3] = std::rotl(v[3], 21);
|
||||
v[1] ^= v[2];
|
||||
v[3] ^= v[0];
|
||||
v[2] = std::rotl(v[2], 32);
|
||||
}
|
||||
} // namespace
|
||||
|
||||
uint64_t siphash24(std::span<const uint64_t, 2> key,
|
||||
std::span<const uint8_t> input) {
|
||||
const auto orig_input_len = input.size();
|
||||
uint64_t v[]{
|
||||
key[0] ^ UINT64_C(0x736f6d6570736575),
|
||||
key[1] ^ UINT64_C(0x646f72616e646f6d),
|
||||
key[0] ^ UINT64_C(0x6c7967656e657261),
|
||||
key[1] ^ UINT64_C(0x7465646279746573),
|
||||
};
|
||||
|
||||
while (input.size() >= sizeof(uint64_t)) {
|
||||
auto m = CRYPTO_load_u64_le(input.first<sizeof(uint64_t)>());
|
||||
v[3] ^= m;
|
||||
siphash_round(v);
|
||||
siphash_round(v);
|
||||
v[0] ^= m;
|
||||
|
||||
input = input.subspan(sizeof(uint64_t));
|
||||
}
|
||||
|
||||
std::array<uint8_t, sizeof(uint64_t)> last_block{};
|
||||
std::ranges::copy(input, std::begin(last_block));
|
||||
last_block.back() = orig_input_len & 0xff;
|
||||
|
||||
auto last_block_word = CRYPTO_load_u64_le(last_block);
|
||||
v[3] ^= last_block_word;
|
||||
siphash_round(v);
|
||||
siphash_round(v);
|
||||
v[0] ^= last_block_word;
|
||||
|
||||
v[2] ^= 0xff;
|
||||
siphash_round(v);
|
||||
siphash_round(v);
|
||||
siphash_round(v);
|
||||
siphash_round(v);
|
||||
|
||||
return v[0] ^ v[1] ^ v[2] ^ v[3];
|
||||
}
|
||||
61
src/siphash.h
Normal file
61
src/siphash.h
Normal file
@@ -0,0 +1,61 @@
|
||||
/* Copyright 2019 The BoringSSL Authors
|
||||
*
|
||||
* Permission to use, copy, modify, and/or distribute this software for any
|
||||
* purpose with or without fee is hereby granted, provided that the above
|
||||
* copyright notice and this permission notice appear in all copies.
|
||||
*
|
||||
* THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
|
||||
* WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
|
||||
* MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY
|
||||
* SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
|
||||
* WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN ACTION
|
||||
* OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF OR IN
|
||||
* CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE. */
|
||||
|
||||
/*
|
||||
* nghttp2 - HTTP/2 C Library
|
||||
*
|
||||
* Copyright (c) 2025 nghttp2 contributors
|
||||
*
|
||||
* Permission is hereby granted, free of charge, to any person obtaining
|
||||
* a copy of this software and associated documentation files (the
|
||||
* "Software"), to deal in the Software without restriction, including
|
||||
* without limitation the rights to use, copy, modify, merge, publish,
|
||||
* distribute, sublicense, and/or sell copies of the Software, and to
|
||||
* permit persons to whom the Software is furnished to do so, subject to
|
||||
* the following conditions:
|
||||
*
|
||||
* The above copyright notice and this permission notice shall be
|
||||
* included in all copies or substantial portions of the Software.
|
||||
*
|
||||
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
|
||||
* EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
|
||||
* MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
|
||||
* NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
|
||||
* LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
|
||||
* OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
|
||||
* WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
|
||||
*/
|
||||
#ifndef SIPHASH_H
|
||||
#define SIPHASH_H
|
||||
|
||||
#include <bit>
|
||||
#include <concepts>
|
||||
#include <algorithm>
|
||||
#include <span>
|
||||
|
||||
// SipHash is a fast, secure PRF that is often used for hash tables.
|
||||
|
||||
// siphash24 implements SipHash-2-4. See
|
||||
// https://131002.net/siphash/siphash.pdf
|
||||
uint64_t siphash24(std::span<const uint64_t, 2> key,
|
||||
std::span<const uint8_t> input);
|
||||
|
||||
// Define here to be usable in tests.
|
||||
template <std::integral T> T byteswap(T v) {
|
||||
auto c = std::bit_cast<std::array<uint8_t, sizeof(T)>>(v);
|
||||
std::ranges::reverse(c);
|
||||
return std::bit_cast<T>(c);
|
||||
}
|
||||
|
||||
#endif // SIPHASH_H
|
||||
68
src/siphash_test.cc
Normal file
68
src/siphash_test.cc
Normal file
@@ -0,0 +1,68 @@
|
||||
/*
|
||||
* nghttp2 - HTTP/2 C Library
|
||||
*
|
||||
* Copyright (c) 2025 nghttp2 contributors
|
||||
*
|
||||
* Permission is hereby granted, free of charge, to any person obtaining
|
||||
* a copy of this software and associated documentation files (the
|
||||
* "Software"), to deal in the Software without restriction, including
|
||||
* without limitation the rights to use, copy, modify, merge, publish,
|
||||
* distribute, sublicense, and/or sell copies of the Software, and to
|
||||
* permit persons to whom the Software is furnished to do so, subject to
|
||||
* the following conditions:
|
||||
*
|
||||
* The above copyright notice and this permission notice shall be
|
||||
* included in all copies or substantial portions of the Software.
|
||||
*
|
||||
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
|
||||
* EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
|
||||
* MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
|
||||
* NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
|
||||
* LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
|
||||
* OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
|
||||
* WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
|
||||
*/
|
||||
#include "siphash_test.h"
|
||||
|
||||
#include <cstring>
|
||||
#include <array>
|
||||
#include <numeric>
|
||||
|
||||
#include "munitxx.h"
|
||||
|
||||
#include <nghttp2/nghttp2.h>
|
||||
|
||||
#include "siphash.h"
|
||||
|
||||
namespace nghttp2 {
|
||||
|
||||
namespace {
|
||||
const MunitTest tests[]{
|
||||
munit_void_test(test_siphash),
|
||||
munit_test_end(),
|
||||
};
|
||||
} // namespace
|
||||
|
||||
const MunitSuite siphash_suite{
|
||||
"/siphash", tests, nullptr, 1, MUNIT_SUITE_OPTION_NONE,
|
||||
};
|
||||
|
||||
void test_siphash(void) {
|
||||
std::array<uint8_t, 16> key_bytes;
|
||||
std::iota(std::begin(key_bytes), std::end(key_bytes), 0);
|
||||
|
||||
std::array<uint64_t, 2> key;
|
||||
memcpy(key.data(), key_bytes.data(), key_bytes.size());
|
||||
|
||||
if constexpr (std::endian::native == std::endian::big) {
|
||||
key[0] = byteswap(key[0]);
|
||||
key[1] = byteswap(key[1]);
|
||||
}
|
||||
|
||||
std::array<uint8_t, 15> input;
|
||||
std::iota(std::begin(input), std::end(input), 0);
|
||||
|
||||
assert_uint64(0xa129ca6149be45e5ull, ==, siphash24(key, input));
|
||||
}
|
||||
|
||||
} // namespace nghttp2
|
||||
44
src/siphash_test.h
Normal file
44
src/siphash_test.h
Normal file
@@ -0,0 +1,44 @@
|
||||
/*
|
||||
* nghttp2 - HTTP/2 C Library
|
||||
*
|
||||
* Copyright (c) 2025 nghttp2 contributors
|
||||
*
|
||||
* Permission is hereby granted, free of charge, to any person obtaining
|
||||
* a copy of this software and associated documentation files (the
|
||||
* "Software"), to deal in the Software without restriction, including
|
||||
* without limitation the rights to use, copy, modify, merge, publish,
|
||||
* distribute, sublicense, and/or sell copies of the Software, and to
|
||||
* permit persons to whom the Software is furnished to do so, subject to
|
||||
* the following conditions:
|
||||
*
|
||||
* The above copyright notice and this permission notice shall be
|
||||
* included in all copies or substantial portions of the Software.
|
||||
*
|
||||
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
|
||||
* EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
|
||||
* MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
|
||||
* NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
|
||||
* LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
|
||||
* OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
|
||||
* WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
|
||||
*/
|
||||
#ifndef SIPHASH_TEST_H
|
||||
#define SIPHASH_TEST_H
|
||||
|
||||
#ifdef HAVE_CONFIG_H
|
||||
# include <config.h>
|
||||
#endif // HAVE_CONFIG_H
|
||||
|
||||
#define MUNIT_ENABLE_ASSERT_ALIASES
|
||||
|
||||
#include "munit.h"
|
||||
|
||||
namespace nghttp2 {
|
||||
|
||||
extern const MunitSuite siphash_suite;
|
||||
|
||||
munit_void_test_decl(test_siphash)
|
||||
|
||||
} // namespace nghttp2
|
||||
|
||||
#endif // SIPHASH_TEST_H
|
||||
@@ -482,6 +482,17 @@ as_uint8_span(std::span<T, N> s) noexcept {
|
||||
{reinterpret_cast<const uint8_t *>(s.data()), s.size_bytes()};
|
||||
}
|
||||
|
||||
template <typename T, std::size_t N>
|
||||
[[nodiscard]] std::span<uint8_t, N == std::dynamic_extent ? std::dynamic_extent
|
||||
: N * sizeof(T)>
|
||||
as_writable_uint8_span(std::span<T, N> s) noexcept {
|
||||
return std::span < uint8_t,
|
||||
N == std::dynamic_extent
|
||||
? std::dynamic_extent
|
||||
: N * sizeof(T) >
|
||||
{reinterpret_cast<uint8_t *>(s.data()), s.size_bytes()};
|
||||
}
|
||||
|
||||
inline int run_app(std::function<int(int, char **)> app, int argc,
|
||||
char **argv) {
|
||||
try {
|
||||
|
||||
Reference in New Issue
Block a user