diff --git a/.github/workflows/main.yml b/.github/workflows/main.yml index e2c86c32..028c90df 100644 --- a/.github/workflows/main.yml +++ b/.github/workflows/main.yml @@ -3,6 +3,7 @@ on: [push, pull_request] jobs: mac: + timeout-minutes: 45 strategy: fail-fast: false matrix: @@ -62,6 +63,7 @@ jobs: linux: runs-on: ubuntu-22.04 + timeout-minutes: 25 strategy: fail-fast: false matrix: diff --git a/CMakeLists.txt b/CMakeLists.txt index bcd215cc..ffc07bad 100644 --- a/CMakeLists.txt +++ b/CMakeLists.txt @@ -2,9 +2,17 @@ cmake_minimum_required(VERSION 3.16) project(quicer) +## NIF library ABI version +## Bump manually when ABI changes +set(QUICER_ABI_VERSION 1) + SET(Erlang_EI_INCLUDE_DIRS ${Erlang_OTP_LIB_DIR}/${Erlang_EI_DIR}/include) SET(CMAKE_LIBRARY_OUTPUT_DIRECTORY ${PROJECT_BINARY_DIR}/priv/) +if(CMAKE_INSTALL_PREFIX_INITIALIZED_TO_DEFAULT) + set(CMAKE_INSTALL_PREFIX ${PROJECT_SOURCE_DIR}/priv/ CACHE PATH "..." FORCE) +endif() + # For cerl picking up the OTP_ROOT if (DEFINED ENV{Erlang_OTP_ROOT_DIR}) SET(Erlang_OTP_ROOT_DIR $ENV{Erlang_OTP_ROOT_DIR}) @@ -15,6 +23,12 @@ EXECUTE_PROCESS( ) endif() +if (DEFINED ENV{QUICER_VERSION}) + set(QUICER_VERSION $ENV{QUICER_VERSION}) +else() + set(QUICER_VERSION "0") +endif() + if (DEFINED ENV{CMAKE_BUILD_TYPE}) set(CMAKE_BUILD_TYPE $ENV{CMAKE_BUILD_TYPE}) else() @@ -64,6 +78,16 @@ set(QUIC_BUILD_PERF "OFF") set(QUIC_TLS_SECRETS_SUPPORT "ON") +configure_file( + ${CMAKE_CURRENT_SOURCE_DIR}/include/quicer_vsn.hrl.in + ${CMAKE_CURRENT_SOURCE_DIR}/include/quicer_vsn.hrl +) + +configure_file( + ${CMAKE_CURRENT_SOURCE_DIR}/c_src/quicer_vsn.h.in + ${CMAKE_CURRENT_BINARY_DIR}/c_src/quicer_vsn.h +) + # src files set(SOURCES c_src/quicer_nif.c @@ -96,7 +120,8 @@ add_compile_options(-DSO_ATTACH_REUSEPORT_CBPF=51) # for lttng, quicer_tp.h include_directories(c_src) - +# for templ files +include_directories(${CMAKE_CURRENT_BINARY_DIR}/c_src/) add_subdirectory(msquic) add_library(quicer_static STATIC ${SOURCES}) @@ -138,10 +163,9 @@ add_dependencies(quicer_static msquic) set_target_properties(quicer_nif PROPERTIES - LIBRARY_OUTPUT_DIRECTORY ${PROJECT_SOURCE_DIR}/priv/ + LIBRARY_OUTPUT_NAME quicer_nif + VERSION ${QUICER_ABI_VERSION}-${QUICER_VERSION} ) +include(GNUInstallDirs) +install(TARGETS quicer_nif LIBRARY DESTINATION ${PROJECT_SOURCE_DIR}/priv/) -if (QUIC_ENABLE_LOGGING STREQUAL "ON" AND QUIC_LOGGING_TYPE STREQUAL "lttng") - set_target_properties(msquic.lttng PROPERTIES LIBRARY_OUTPUT_DIRECTORY ${CMAKE_CURRENT_SOURCE_DIR}/priv) - set_target_properties(msquic.lttng PROPERTIES LIBRARY_OUTPUT_DIRECTORY_DEBUG ${CMAKE_CURRENT_SOURCE_DIR}/priv) -endif() diff --git a/Makefile b/Makefile index 9a323c91..576eeb4e 100644 --- a/Makefile +++ b/Makefile @@ -1,5 +1,8 @@ REBAR := rebar3 +QUICER_VERSION ?= $(shell git describe --tags --always) +export QUICER_VERSION + .PHONY: all all: compile diff --git a/build.sh b/build.sh index 65d4fa4d..8ab89935 100755 --- a/build.sh +++ b/build.sh @@ -17,6 +17,7 @@ build() { ./get-msquic.sh "$MSQUIC_VERSION" cmake -B c_build make -j "$JOBS" -C c_build + make install -C c_build ## MacOS if [ -f priv/libquicer_nif.dylib ]; then # https://developer.apple.com/forums/thread/696460 diff --git a/c_src/quicer_config.c b/c_src/quicer_config.c index f970873f..e720dc94 100644 --- a/c_src/quicer_config.c +++ b/c_src/quicer_config.c @@ -21,6 +21,7 @@ limitations under the License. #include extern QuicerRegistrationCTX *G_r_ctx; +extern pthread_mutex_t MsQuicLock; static ERL_NIF_TERM get_stream_opt(ErlNifEnv *env, QuicerStreamCTX *s_ctx, @@ -768,7 +769,15 @@ getopt3(ErlNifEnv *env, if (IS_SAME_TERM(ATOM_QUIC_GLOBAL, ctx)) { - res = get_global_opt(env, NULL, eopt); + pthread_mutex_lock(&MsQuicLock); + // In a env that while there is no allocated NIF resources (reg, conf, + // listener, conn, stream), VM may unload the module causes unloading DSO + // in parallel. + if (MsQuic) + { + res = get_global_opt(env, NULL, eopt); + } + pthread_mutex_unlock(&MsQuicLock); } else if (enif_get_resource(env, ctx, ctx_stream_t, &q_ctx)) { diff --git a/c_src/quicer_connection.c b/c_src/quicer_connection.c index a3956397..2a5fb2fd 100644 --- a/c_src/quicer_connection.c +++ b/c_src/quicer_connection.c @@ -327,7 +327,6 @@ _IRQL_requires_max_(DISPATCH_LEVEL) // @see async_connect3 status = handle_connection_event_shutdown_complete(c_ctx, Event); is_destroy = TRUE; - c_ctx->is_closed = TRUE; // client shutdown completed break; case QUIC_CONNECTION_EVENT_PEER_ADDRESS_CHANGED: @@ -372,10 +371,23 @@ _IRQL_requires_max_(DISPATCH_LEVEL) break; } enif_clear_env(env); + + QuicerConfigCTX *conf_ctx = c_ctx->config_resource; + if (is_destroy) + { + c_ctx->is_closed = TRUE; // client shutdown completed + c_ctx->Connection = NULL; + c_ctx->config_resource = NULL; + } enif_mutex_unlock(c_ctx->lock); if (is_destroy) { + MsQuic->ConnectionClose(Connection); + if (conf_ctx) + { + enif_release_resource(conf_ctx); + } destroy_c_ctx(c_ctx); } return status; @@ -434,7 +446,6 @@ ServerConnectionCallback(HQUIC Connection, // safely cleaned up. // status = handle_connection_event_shutdown_complete(c_ctx, Event); - c_ctx->is_closed = TRUE; // server shutdown_complete is_destroy = TRUE; break; case QUIC_CONNECTION_EVENT_PEER_STREAM_STARTED: @@ -481,10 +492,23 @@ ServerConnectionCallback(HQUIC Connection, break; } enif_clear_env(env); + + QuicerConfigCTX *conf_ctx = c_ctx->config_resource; + if (is_destroy) + { + c_ctx->Connection = NULL; + c_ctx->is_closed = TRUE; // server shutdown_complete + c_ctx->config_resource = NULL; + } enif_mutex_unlock(c_ctx->lock); if (is_destroy) { + MsQuic->ConnectionClose(Connection); + if (conf_ctx) + { + enif_release_resource(conf_ctx); + } destroy_c_ctx(c_ctx); } @@ -503,7 +527,7 @@ open_connectionX(ErlNifEnv *env, int argc, const ERL_NIF_TERM argv[]) ERL_NIF_TERM res = ERROR_TUPLE_2(ATOM_ERROR_INTERNAL_ERROR); QuicerRegistrationCTX *r_ctx = NULL; HQUIC registration = NULL; - ERL_NIF_TERM options = argv[1]; + ERL_NIF_TERM options = argv[0]; if (argc == 0) { @@ -799,14 +823,15 @@ async_connect3(ErlNifEnv *env, AcceptorDestroy(c_ctx->owner); c_ctx->owner = NULL; - /* Although MsQuic internally close the connection after failed to start, - we still do not need to set is_closed here, we expect callback to set - it while handling the shutdown complete event otherwise could cause - race cond. - */ - // c_ctx->is_closed = TRUE; + if (Status != QUIC_STATUS_INVALID_PARAMETER) + { + c_ctx->Connection = NULL; + } - c_ctx->Connection = NULL; + if (c_ctx->config_resource) + { + destroy_config_ctx(c_ctx->config_resource); + } res = ERROR_TUPLE_2(ATOM_CONN_START_ERROR); TP_NIF_3(start_fail, (uintptr_t)(c_ctx->Connection), Status); @@ -1056,6 +1081,11 @@ continue_connection_handshake(QuicerConnCTX *c_ctx) return QUIC_STATUS_INTERNAL_ERROR; } + if (!c_ctx->Connection) + { + return QUIC_STATUS_INVALID_STATE; + } + if (QUIC_FAILED( Status = MsQuic->ConnectionSetConfiguration( c_ctx->Connection, c_ctx->config_resource->Configuration))) diff --git a/c_src/quicer_ctx.c b/c_src/quicer_ctx.c index db6ba7eb..285e43a8 100644 --- a/c_src/quicer_ctx.c +++ b/c_src/quicer_ctx.c @@ -215,7 +215,10 @@ deinit_config_ctx(QuicerConfigCTX *config_ctx) void destroy_config_ctx(QuicerConfigCTX *config_ctx) { - enif_release_resource(config_ctx); + if (config_ctx) + { + enif_release_resource(config_ctx); + } } QuicerStreamCTX * @@ -254,10 +257,8 @@ deinit_s_ctx(QuicerStreamCTX *s_ctx) void destroy_s_ctx(QuicerStreamCTX *s_ctx) { + assert(!s_ctx->Stream); enif_free_env(s_ctx->imm_env); - // Since enif_release_resource is async call, - // we should demon the owner now! - enif_demonitor_process(s_ctx->env, s_ctx, &s_ctx->owner_mon); enif_release_resource(s_ctx); } diff --git a/c_src/quicer_listener.c b/c_src/quicer_listener.c index 2e1d1d2f..58d46496 100644 --- a/c_src/quicer_listener.c +++ b/c_src/quicer_listener.c @@ -23,6 +23,7 @@ limitations under the License. #include extern QuicerRegistrationCTX *G_r_ctx; +extern pthread_mutex_t GRegLock; BOOLEAN parse_registration(ErlNifEnv *env, ERL_NIF_TERM options, @@ -279,6 +280,7 @@ listen2(ErlNifEnv *env, __unused_parm__ int argc, const ERL_NIF_TERM argv[]) { // TLS opt error not file content error free(cacertfile); + free_certificate(&CredConfig); return ERROR_TUPLE_2(ATOM_CACERTFILE); } @@ -288,6 +290,7 @@ listen2(ErlNifEnv *env, __unused_parm__ int argc, const ERL_NIF_TERM argv[]) if (!l_ctx) { free(cacertfile); + free_certificate(&CredConfig); return ERROR_TUPLE_2(ATOM_ERROR_NOT_ENOUGH_MEMORY); } @@ -306,6 +309,10 @@ listen2(ErlNifEnv *env, __unused_parm__ int argc, const ERL_NIF_TERM argv[]) goto exit; } } + else + { // since we don't use cacertfile, free it + free(cacertfile); + } // Set owner for l_ctx if (!enif_self(env, &(l_ctx->listenerPid))) @@ -576,8 +583,15 @@ ERL_NIF_TERM get_listenersX(ErlNifEnv *env, int argc, const ERL_NIF_TERM argv[]) { QuicerRegistrationCTX *r_ctx = NULL; - if (argc == 0) + ERL_NIF_TERM res = enif_make_list(env, 0); + if (argc == 0) // use global registration { + pthread_mutex_lock(&GRegLock); + if (!G_r_ctx) + { + pthread_mutex_unlock(&GRegLock); + return res; + } r_ctx = G_r_ctx; } else @@ -587,8 +601,6 @@ get_listenersX(ErlNifEnv *env, int argc, const ERL_NIF_TERM argv[]) return ERROR_TUPLE_2(ATOM_BADARG); } } - ERL_NIF_TERM res = enif_make_list(env, 0); - enif_mutex_lock(r_ctx->lock); CXPLAT_LIST_ENTRY *Entry = r_ctx->Listeners.Flink; while (Entry != &r_ctx->Listeners) @@ -600,5 +612,9 @@ get_listenersX(ErlNifEnv *env, int argc, const ERL_NIF_TERM argv[]) } enif_mutex_unlock(r_ctx->lock); + if (argc == 0) // use global registration + { + pthread_mutex_unlock(&GRegLock); + } return res; } diff --git a/c_src/quicer_nif.c b/c_src/quicer_nif.c index 90478e3e..60876849 100644 --- a/c_src/quicer_nif.c +++ b/c_src/quicer_nif.c @@ -19,6 +19,7 @@ limitations under the License. #include #include "quicer_listener.h" +#include "quicer_vsn.h" #include #include @@ -802,11 +803,16 @@ resource_listener_down_callback(__unused_parm__ ErlNifEnv *env, if (!l_ctx->is_closed && !l_ctx->is_stopped && l_ctx->Listener) { l_ctx->is_stopped = TRUE; + /* // We only stop here, but not close it, because possible subsequent - // scenarios a. Some pid could still start the stopped listener with nif - // handle b. Some pid could still close the stopped listener with nif - // handle - // 3. We close it in resource_listener_dealloc_callback anyway. + // scenarios: + // a. Some pid could still start the stopped listener with nif + // handle. + // b. Some pid could still close the stopped listener with nif + // handle. + // c. We close it in resource_listener_dealloc_callback anyway when + // Listener term get GC. + */ MsQuic->ListenerStop(l_ctx->Listener); } enif_mutex_unlock(l_ctx->lock); @@ -836,6 +842,10 @@ resource_listener_dealloc_callback(__unused_parm__ ErlNifEnv *env, void *obj) // process is able to access the listener via any l_ctx MsQuic->ListenerClose(l_ctx->Listener); } + else + { + TP_CB_3(skip, (uintptr_t)l_ctx->Listener, 0); + } if (l_ctx->cacertfile) { @@ -880,8 +890,10 @@ resource_conn_down_callback(__unused_parm__ ErlNifEnv *env, && !enif_compare_pids(&c_ctx->owner->Pid, DeadPid)) { TP_CB_3(start, (uintptr_t)c_ctx->Connection, (uintptr_t)ctx); + enif_mutex_lock(c_ctx->lock); MsQuic->ConnectionShutdown( c_ctx->Connection, QUIC_CONNECTION_SHUTDOWN_FLAG_NONE, 0); + enif_mutex_unlock(c_ctx->lock); TP_CB_3(end, (uintptr_t)c_ctx->Connection, (uintptr_t)ctx); } } @@ -913,6 +925,7 @@ resource_stream_down_callback(__unused_parm__ ErlNifEnv *env, QUIC_STATUS status = QUIC_STATUS_SUCCESS; QuicerStreamCTX *s_ctx = ctx; + enif_mutex_lock(s_ctx->lock); if (s_ctx && s_ctx->owner && DeadPid && !enif_compare_pids(&s_ctx->owner->Pid, DeadPid)) { @@ -931,6 +944,7 @@ resource_stream_down_callback(__unused_parm__ ErlNifEnv *env, TP_CB_3(shutdown_success, (uintptr_t)s_ctx->Stream, status); } } + enif_mutex_unlock(s_ctx->lock); } void @@ -961,25 +975,21 @@ resource_reg_dealloc_callback(__unused_parm__ ErlNifEnv *env, void *obj) TP_CB_3(end, (uintptr_t)obj, 0); } -/* -** on_load is called when the NIF library is loaded and no previously loaded -*library exists for this module. -*/ -static int -on_load(ErlNifEnv *env, - __unused_parm__ void **priv_data, - __unused_parm__ ERL_NIF_TERM loadinfo) +static void +init_atoms(ErlNifEnv *env) { - int ret_val = 0; - -// init atoms in use. + // init atoms in use. #define ATOM(name, val) \ { \ (name) = enif_make_atom(env, #val); \ } INIT_ATOMS #undef ATOM +} +static void +open_resources(ErlNifEnv *env) +{ ErlNifResourceFlags flags = (ErlNifResourceFlags)(ERL_NIF_RT_CREATE | ERL_NIF_RT_TAKEOVER); @@ -1029,14 +1039,55 @@ on_load(ErlNifEnv *env, &streamInit, // init callbacks flags, NULL); +} +/* +** on_load is called when the NIF library is loaded and no previously loaded +* library exists for this module. +*/ +static int +on_load(ErlNifEnv *env, + __unused_parm__ void **priv_data, + ERL_NIF_TERM loadinfo) +{ + int ret_val = 0; + unsigned load_vsn = 0; + + TP_NIF_3(start, &MsQuic, 0); + if (!enif_get_uint(env, loadinfo, &load_vsn)) + { + load_vsn = 0; + } + + // This check avoid erlang module loaded + // incompatible NIF library + if (load_vsn != QUICER_ABI_VERSION) + { + TP_NIF_3(end, &MsQuic, 1); + return 1; // any value except 0 is error + } + + init_atoms(env); + open_resources(env); + + TP_NIF_3(end, &MsQuic, 0); return ret_val; } /* -** on_upgrade is called when the NIF library is loaded and there is old code of -*this module with a loaded NIF library. -*/ + * on_upgrade is called when the NIF library is loaded and there is old code of + * this module with a loaded NIF library. + * + * But new code could be the same as old code, that is, the same msquic + * library is mapped into process memory. To distinguish the two cases, the + * `MsQuic` API handle is checked since it is init as NULL for new loading. If + * MsQuic is NULL, then it is a new load, that two msquic libraries (new and + * old) are mapped into process memory, If MsQuic is not NULL, then it is + * already initilized and there is still one msquic library in process memory. + * + * In any case above, we return success. + */ + static int on_upgrade(ErlNifEnv *env, void **priv_data, @@ -1047,12 +1098,62 @@ on_upgrade(ErlNifEnv *env, } /* -** unload is called when the module code that the NIF library belongs to is -*purged as old. New code of the same module may or may not exist. -*/ +** on_unload is called when the module code that the NIF library belongs to is +* purged as old. +* +* New code of the same module may or may not exist. +* +* But there are three cases: +* +* Case A: No new code of the same module exists. +* arg `priv_data` is not NULL. +* +* It is ok to teardown the MsQuic with API handle and then close the +* API handle. +* +* Case B: New code of the same module exists and it uses the same NIF DSO. +* arg `priv_data` is NULL. +* +* It could be checked with `quicer:nif_mapped()` +* +* It is *NOT* ok to teardown the MsQuic since the new code is +* still using it. +* +* Case C: New code of the same module exists and it uses different NIF DSO. +* arg `priv_data` is not NULL. +* AND +* &MsQuic != the 'lib_api_ptr' in priv_data +* +* This could be checked with `quicer:nif_mapped()` +* +* It is ok to teardown the MsQuic with API handle and then close the +* API handle. + +* +* @NOTE 1. This callback will *NOT* be called when the module is purged +* while there are opening resources. +* When new code of the same module exists, the resources will be +* taken over by the new code thus it will get called for the +* old code. +* +* @NOTE 2. The `MsQuic` and `GRegistration` are in library scope. +* +* @NOTE 3: It is very important to shutdown all the MsQuic Registrations +* before return to avoid unexpected behaviour after NIF DSO is +* unmapped by OS. +* +* @NOTE 4: For safty, it is ok to dlopen the shared library by calling +* quicer:dlopen/1, so we will have a refcnt on it and it won't +* be unmapped by OS. +* +* @NOTE 5: 'same NIF DSO' means same shared library file that is managed +* by OS. +* Two copies of the same shared library in OS are different NIF DSOs. +* */ static void on_unload(__unused_parm__ ErlNifEnv *env, __unused_parm__ void *priv_data) { + // @TODO clean all the leakages before close the lib closeLib(env, 0, NULL); } @@ -1430,10 +1531,11 @@ static ErlNifFunc nif_funcs[] = { { "close_lib", 0, closeLib, 0 }, { "reg_open", 0, registration, 0 }, { "reg_open", 1, registration, 0 }, - { "reg_close", 0, deregistration, 0 }, + { "reg_close", 0, deregistration, 1 }, { "new_registration", 2, new_registration2, 0}, { "shutdown_registration", 1, shutdown_registration_x, 0}, { "shutdown_registration", 3, shutdown_registration_x, 0}, + { "close_registration", 1, close_registration, 1}, { "get_registration_name", 1, get_registration_name1, 0}, { "listen", 2, listen2, 0}, { "start_listener", 3, start_listener3, 0}, diff --git a/c_src/quicer_reg.c b/c_src/quicer_reg.c index 7e51b559..7377edda 100644 --- a/c_src/quicer_reg.c +++ b/c_src/quicer_reg.c @@ -199,6 +199,26 @@ shutdown_registration_x(ErlNifEnv *env, int argc, const ERL_NIF_TERM *argv) return ATOM_OK; } +ERL_NIF_TERM +close_registration(ErlNifEnv *env, + __unused_parm__ int argc, + const ERL_NIF_TERM argv[]) +{ + QuicerRegistrationCTX *r_ctx = NULL; + HQUIC Registration = NULL; + ERL_NIF_TERM ectx = argv[0]; + if (!enif_get_resource(env, ectx, ctx_reg_t, (void **)&r_ctx)) + { + return ERROR_TUPLE_2(ATOM_BADARG); + } + enif_mutex_lock(r_ctx->lock); + Registration = r_ctx->Registration; + r_ctx->Registration = NULL; + enif_mutex_unlock(r_ctx->lock); + MsQuic->RegistrationClose(Registration); + return ATOM_OK; +} + ERL_NIF_TERM get_registration_name1(ErlNifEnv *env, __unused_parm__ int argc, diff --git a/c_src/quicer_reg.h b/c_src/quicer_reg.h index c326530d..9fe6eaa2 100644 --- a/c_src/quicer_reg.h +++ b/c_src/quicer_reg.h @@ -29,6 +29,9 @@ new_registration2(ErlNifEnv *env, int argc, const ERL_NIF_TERM argv[]); ERL_NIF_TERM shutdown_registration_x(ErlNifEnv *env, int argc, const ERL_NIF_TERM argv[]); +ERL_NIF_TERM +close_registration(ErlNifEnv *env, int argc, const ERL_NIF_TERM argv[]); + ERL_NIF_TERM get_registration_name1(ErlNifEnv *env, int argc, const ERL_NIF_TERM argv[]); diff --git a/c_src/quicer_stream.c b/c_src/quicer_stream.c index 04e12f0f..3d59087f 100644 --- a/c_src/quicer_stream.c +++ b/c_src/quicer_stream.c @@ -130,12 +130,14 @@ ServerStreamCallback(HQUIC Stream, void *Context, QUIC_STREAM_EVENT *Event) if (is_destroy) { s_ctx->is_closed = TRUE; + s_ctx->Stream = NULL; } enif_mutex_unlock(s_ctx->lock); if (is_destroy) { + MsQuic->StreamClose(Stream); // must be called after mutex unlock destroy_s_ctx(s_ctx); } @@ -228,6 +230,8 @@ _IRQL_requires_max_(DISPATCH_LEVEL) if (is_destroy) { s_ctx->is_closed = TRUE; + s_ctx->Stream = NULL; + MsQuic->SetCallbackHandler(Stream, NULL, NULL); } enif_mutex_unlock(s_ctx->lock); @@ -235,6 +239,7 @@ _IRQL_requires_max_(DISPATCH_LEVEL) if (is_destroy) { // must be called after mutex unlock, + MsQuic->StreamClose(Stream); destroy_s_ctx(s_ctx); } return status; @@ -341,6 +346,7 @@ async_start_stream2(ErlNifEnv *env, // Now we have Stream handle s_ctx->eHandle = enif_make_resource(s_ctx->imm_env, s_ctx); + res = enif_make_copy(env, s_ctx->eHandle); // // Starts the bidirectional stream. By default, the peer is not notified of @@ -348,13 +354,13 @@ async_start_stream2(ErlNifEnv *env, // if (QUIC_FAILED(Status = MsQuic->StreamStart(s_ctx->Stream, start_flag))) { - // note, stream call back would close the stream - // destroy_s_ctx should not be called here + HQUIC Stream = s_ctx->Stream; + enif_mutex_lock(s_ctx->lock); + s_ctx->is_closed = TRUE; + enif_mutex_unlock(s_ctx->lock); + MsQuic->StreamClose(Stream); return ERROR_TUPLE_3(ATOM_STREAM_START_ERROR, ATOM_STATUS(Status)); } - - res = enif_make_copy(env, s_ctx->eHandle); - // NOTE: Set is_closed to FALSE (s_ctx->is_closed = FALSE;) // must be done in the worker callback (for // QUICER_STREAM_EVENT_MASK_START_COMPLETE) to avoid race cond. @@ -548,6 +554,7 @@ csend4(ErlNifEnv *env, int argc, const ERL_NIF_TERM argv[]) { res = ERROR_TUPLE_3(ATOM_STREAM_OPEN_ERROR, ATOM_STATUS(Status)); + s_ctx->Stream = NULL; goto ErrorExit; } @@ -664,6 +671,11 @@ send3(ErlNifEnv *env, int argc, const ERL_NIF_TERM argv[]) } enif_mutex_lock(s_ctx->lock); + if (!s_ctx->Stream) + { + res = ERROR_TUPLE_2(ATOM_CLOSED); + goto ErrorExit; + } send_ctx->s_ctx = s_ctx; @@ -724,6 +736,12 @@ recv2(ErlNifEnv *env, __unused_parm__ int argc, const ERL_NIF_TERM argv[]) TP_NIF_3(start, (uintptr_t)s_ctx->Stream, size_req); enif_mutex_lock(s_ctx->lock); + if ( !s_ctx->Stream ) + { + res = ERROR_TUPLE_2(ATOM_CLOSED); + goto Exit; + } + if (ACCEPTOR_RECV_MODE_PASSIVE != s_ctx->owner->active) { res = ERROR_TUPLE_2(ATOM_EINVAL); @@ -814,12 +832,13 @@ shutdown_stream3(ErlNifEnv *env, ret = ERROR_TUPLE_2(ATOM_BADARG); } + enif_mutex_lock(s_ctx->lock); if (QUIC_FAILED(Status = MsQuic->StreamShutdown(s_ctx->Stream, flags, app_errcode))) { ret = ERROR_TUPLE_2(ATOM_STATUS(Status)); } - + enif_mutex_unlock(s_ctx->lock); return ret; } @@ -1000,7 +1019,6 @@ handle_stream_event_start_complete(QuicerStreamCTX *s_ctx, assert(env); assert(QUIC_STREAM_EVENT_START_COMPLETE == Event->Type); // Only for Local initiated stream - s_ctx->is_closed = FALSE; if (s_ctx->event_mask & QUICER_STREAM_EVENT_MASK_START_COMPLETE) { ERL_NIF_TERM props_name[] diff --git a/c_src/quicer_vsn.h.in b/c_src/quicer_vsn.h.in new file mode 100644 index 00000000..ae953305 --- /dev/null +++ b/c_src/quicer_vsn.h.in @@ -0,0 +1,23 @@ +/*-------------------------------------------------------------------- +Copyright (c) 2023 EMQ Technologies Co., Ltd. All Rights Reserved. + +Licensed under the Apache License, Version 2.0 (the "License"); +you may not use this file except in compliance with the License. +You may obtain a copy of the License at + + http://www.apache.org/licenses/LICENSE-2.0 + +Unless required by applicable law or agreed to in writing, software +distributed under the License is distributed on an "AS IS" BASIS, +WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +See the License for the specific language governing permissions and +limitations under the License. +-------------------------------------------------------------------*/ +#ifndef __QUICER_VSN_H_ +#define __QUICER_VSN_H_ + +// clang-format off +#define QUICER_ABI_VERSION @QUICER_ABI_VERSION@ +// clang-format on + +#endif //__QUICER_VSN_H_ diff --git a/docs/internals.org b/docs/internals.org index 3bdcfc8b..134604a0 100644 --- a/docs/internals.org +++ b/docs/internals.org @@ -250,33 +250,35 @@ SYNC call in non-callback-context *** API Types (number in tracing) #+begin_verse -QUIC_TRACE_API_SET_PARAM, // 0 -QUIC_TRACE_API_GET_PARAM, // 1 -QUIC_TRACE_API_REGISTRATION_OPEN , // 2 -QUIC_TRACE_API_REGISTRATION_CLOSE, // 3 -QUIC_TRACE_API_REGISTRATION_SHUTDOWN, // 4 -QUIC_TRACE_API_CONFIGURATION_OPEN, // 5 -QUIC_TRACE_API_CONFIGURATION_CLOSE, // 6 -QUIC_TRACE_API_CONFIGURATION_LOAD_CREDENTIAL, // 7 -QUIC_TRACE_API_LISTENER_OPEN, // 8 -QUIC_TRACE_API_LISTENER_CLOSE, // 9 -QUIC_TRACE_API_LISTENER_START, // 10 -QUIC_TRACE_API_LISTENER_STOP, // 11 -QUIC_TRACE_API_CONNECTION_OPEN, // 12 -QUIC_TRACE_API_CONNECTION_CLOSE, // 13 -QUIC_TRACE_API_CONNECTION_SHUTDOWN, // 14 -QUIC_TRACE_API_CONNECTION_START, // 15 -QUIC_TRACE_API_CONNECTION_SET_CONFIGURATION, // 16 -QUIC_TRACE_API_CONNECTION_SEND_RESUMPTION_TICKET, // 17 -QUIC_TRACE_API_STREAM_OPEN, // 18 -QUIC_TRACE_API_STREAM_CLOSE, // 19 -QUIC_TRACE_API_STREAM_START, // 20 -QUIC_TRACE_API_STREAM_SHUTDOWN, // 21 -QUIC_TRACE_API_STREAM_SEND, // 22 -QUIC_TRACE_API_STREAM_RECEIVE_COMPLETE, // 23 -QUIC_TRACE_API_STREAM_RECEIVE_SET_ENABLED, // 24 -QUIC_TRACE_API_DATAGRAM_SEND, // 25 -QUIC_TRACE_API_COUNT // 26 +0 QUIC_TRACE_API_SET_PARAM, +1 QUIC_TRACE_API_GET_PARAM, +2 QUIC_TRACE_API_REGISTRATION_OPEN, +3 QUIC_TRACE_API_REGISTRATION_CLOSE, +4 QUIC_TRACE_API_REGISTRATION_SHUTDOWN, +5 QUIC_TRACE_API_CONFIGURATION_OPEN, +6 QUIC_TRACE_API_CONFIGURATION_CLOSE, +7 QUIC_TRACE_API_CONFIGURATION_LOAD_CREDENTIAL, +8 QUIC_TRACE_API_LISTENER_OPEN, +9 QUIC_TRACE_API_LISTENER_CLOSE, +10 QUIC_TRACE_API_LISTENER_START, +11 QUIC_TRACE_API_LISTENER_STOP, +12 QUIC_TRACE_API_CONNECTION_OPEN, +13 QUIC_TRACE_API_CONNECTION_CLOSE, +14 QUIC_TRACE_API_CONNECTION_SHUTDOWN, +15 QUIC_TRACE_API_CONNECTION_START, +16 QUIC_TRACE_API_CONNECTION_SET_CONFIGURATION, +17 QUIC_TRACE_API_CONNECTION_SEND_RESUMPTION_TICKET, +18 QUIC_TRACE_API_CONNECTION_COMPLETE_RESUMPTION_TICKET_VALIDATION, +19 QUIC_TRACE_API_CONNECTION_COMPLETE_CERTIFICATE_VALIDATION, +20 QUIC_TRACE_API_STREAM_OPEN, +21 QUIC_TRACE_API_STREAM_CLOSE, +22 QUIC_TRACE_API_STREAM_START, +23 QUIC_TRACE_API_STREAM_SHUTDOWN, +24 QUIC_TRACE_API_STREAM_SEND, +25 QUIC_TRACE_API_STREAM_RECEIVE_COMPLETE, +26 QUIC_TRACE_API_STREAM_RECEIVE_SET_ENABLED, +27 QUIC_TRACE_API_DATAGRAM_SEND, +28 QUIC_TRACE_API_COUNT // Must be last #+end_verse * Event handling diff --git a/get-msquic.sh b/get-msquic.sh index 63c38c00..a6327bb6 100755 --- a/get-msquic.sh +++ b/get-msquic.sh @@ -4,6 +4,29 @@ set -euo pipefail VERSION="$1" +patch_dir="patches" + +do_patch() +{ + patch_source="$1" + patch_file="${patch_dir}/$(basename ${patch_source})" + curl -f -L -o "${patch_file}" "$patch_source" + if patch -p1 -f --dry-run -s < "${patch_file}" 2>/dev/null; then + patch -p1 < "${patch_file}" + else + echo "Skip patching ${patch_file}, already applied" + fi +} + +patch_2_2_3() +{ + local patch_1="https://github.com/microsoft/msquic/commit/73a11d7bdc724432964a2d4bdc4211ed29823380.patch" + mkdir -p "$patch_dir" + echo "Patching Msquic 2.2.3" + do_patch "$patch_1" +} + + if [ ! -d msquic ]; then git clone https://github.com/microsoft/msquic.git -b "$VERSION" --recursive --depth 1 --shallow-submodules msquic fi @@ -20,3 +43,9 @@ if [ "$CURRENT_VSN" != "$VERSION" ]; then echo "undesired_msquic_version, required=$VERSION, got=$CURRENT_VSN" exit 1 fi + +## Patching +case $VERSION in + v2.2.3) + patch_2_2_3 +esac diff --git a/include/quicer_vsn.hrl.in b/include/quicer_vsn.hrl.in new file mode 100644 index 00000000..db73c6ad --- /dev/null +++ b/include/quicer_vsn.hrl.in @@ -0,0 +1,19 @@ +%%-------------------------------------------------------------------- +%% Copyright (c) 2023 EMQ Technologies Co., Ltd. All Rights Reserved. +%% +%% Licensed under the Apache License, Version 2.0 (the "License"); +%% you may not use this file except in compliance with the License. +%% You may obtain a copy of the License at +%% +%% http://www.apache.org/licenses/LICENSE-2.0 +%% +%% Unless required by applicable law or agreed to in writing, software +%% distributed under the License is distributed on an "AS IS" BASIS, +%% WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +%% See the License for the specific language governing permissions and +%% limitations under the License. +%%-------------------------------------------------------------------- +-ifndef(QUICER_VSN_HRL). +-define(QUICER_VSN_HRL, true). +-define(QUICER_ABI_VERSION, @QUICER_ABI_VERSION@). +-endif. diff --git a/src/quicer.erl b/src/quicer.erl index b067ba3a..cd531019 100644 --- a/src/quicer.erl +++ b/src/quicer.erl @@ -114,15 +114,24 @@ , open_connection/0 , get_listeners/0 , get_listeners/1 + , close_registration/1 ]). -export([ spawn_listener/3 %% start application over quic , terminate_listener/1 ]). +%% versions +-export([abi_version/0]). + -type connection_opts() :: proplists:proplist() | quicer_connection:opts(). -type listener_opts() :: proplists:proplist() | quicer_listener:listener_opts(). +%% @doc Return ABI version of the library. +-spec abi_version() -> quicer_nif:abi_version(). +abi_version() -> + quicer_nif:abi_version(). + %% @doc Quicer library must be opened before any use. %% %% This is called automatically while quicer application is started @@ -164,6 +173,12 @@ shutdown_registration(Handle) -> shutdown_registration(Handle, IsSilent, ErrCode) -> quicer_nif:shutdown_registration(Handle, IsSilent, ErrCode). +%% @doc close a registration. +-spec close_registration(Handle) -> + quicer_nif:close_registration(Handle). +close_registration(Handle) -> + quicer_nif:close_registration(Handle). + %% @doc get registration name -spec get_registration_name(Handle) -> quicer_nif:get_registration_name(Handle). diff --git a/src/quicer_nif.erl b/src/quicer_nif.erl index 73e0a628..a10aa547 100644 --- a/src/quicer_nif.erl +++ b/src/quicer_nif.erl @@ -22,6 +22,7 @@ , new_registration/2 , shutdown_registration/1 , shutdown_registration/3 + , close_registration/1 , get_registration_name/1 , listen/2 , start_listener/3 @@ -56,39 +57,64 @@ , get_listeners/1 ]). +-export([abi_version/0]). + +%% for test +-export([init/1]). + +%% @NOTE: In embedded mode, first all modules are loaded. Then all on_load functions are called. -on_load(init/0). -include_lib("kernel/include/file.hrl"). -include("quicer.hrl"). -include("quicer_types.hrl"). +-include("quicer_vsn.hrl"). +-include_lib("snabbkaffe/include/snabbkaffe.hrl"). + +-spec abi_version() -> integer(). +abi_version() -> + ?QUICER_ABI_VERSION. -spec init() -> ok. init() -> + ABIVsn = case persistent_term:get({'_quicer_overrides_', abi_version}, undefined) of + undefined -> abi_version(); + Vsn -> Vsn + end, + init(ABIVsn). + +init(ABIVsn) -> NifName = "libquicer_nif", {ok, Niflib} = locate_lib(priv_dir(), NifName), - ok = erlang:load_nif(Niflib, 0), - %% It could cause segfault if MsQuic library is not opened nor registered. - %% here we have added dummy calls, and it should cover most of cases - %% unless caller wants to call erlang:load_nif/1 and then call quicer_nif - %% without opened library to suicide. - %% - %% Note, we could do same dummy calls in nif instead but it might mess up the reference counts. - {ok, _} = open_lib(), - %% dummy reg open - case reg_open() of - ok -> ok; - {error, badarg} -> - %% already opened - ok + case erlang:load_nif(Niflib, ABIVsn) of + ok -> + %% It could cause segfault if MsQuic library is not opened nor registered. + %% here we have added dummy calls, and it should cover most of cases + %% unless caller wants to call erlang:load_nif/1 and then call quicer_nif + %% without opened library to suicide. + %% + %% Note, we could do same dummy calls in nif instead but it might mess up the reference counts. + {ok, _} = open_lib(), + %% dummy reg open + case reg_open() of + ok -> ok; + {error, badarg} -> + %% already opened + ok + end; + {error, _Reason} = Res-> + %% load fail, but beam will keep using current vsn if presents. + ?tp_ignore_side_effects_in_prod(debug, + #{module => ?MODULE, event => init, result => Res}), + Res end. - -spec open_lib() -> {ok, true} | %% opened {ok, false} | %% already opened {ok, debug} | %% opened with lttng debug library loaded (if present) {error, open_failed, atom_reason()}. open_lib() -> - LibFile = case locate_lib(priv_dir(), "libmsquic.lttng.so") of + LibFile = case locate_lib(priv_dir(), "lib/libmsquic.lttng.so") of {ok, File} -> File; {error, _} -> @@ -131,6 +157,10 @@ shutdown_registration(_Handle) -> shutdown_registration(_Handle, _IsSilent, _ErrorCode) -> erlang:nif_error(nif_library_not_loaded). +-spec close_registration(reg_handle()) -> ok | {error | badarg}. +close_registration(_Handle) -> + erlang:nif_error(nif_library_not_loaded). + -spec get_registration_name(reg_handle()) -> {ok, string()} | {error, badarg}. get_registration_name(_Handle) -> erlang:nif_error(nif_library_not_loaded). diff --git a/test/example_server_stream.erl b/test/example_server_stream.erl index 33276f03..ef168ffc 100644 --- a/test/example_server_stream.erl +++ b/test/example_server_stream.erl @@ -82,7 +82,10 @@ peer_send_aborted(Stream, ErrorCode, #{is_unidir := true, is_local := false} = S {ok, S}. peer_send_shutdown(Stream, _Flags, S) -> - ok = quicer:async_shutdown_stream(Stream, ?QUIC_STREAM_SHUTDOWN_FLAG_GRACEFUL, 0), + case quicer:async_shutdown_stream(Stream, ?QUIC_STREAM_SHUTDOWN_FLAG_GRACEFUL, 0) of + ok -> ok; + {error, _} -> ok + end, {ok, S}. send_complete(_Stream, false, S) -> diff --git a/test/quicer_SUITE.erl b/test/quicer_SUITE.erl index 49b5475a..9830c0cb 100644 --- a/test/quicer_SUITE.erl +++ b/test/quicer_SUITE.erl @@ -144,6 +144,9 @@ , tc_peercert_client_nocert/1 , tc_peercert_server/1 , tc_peercert_server_nocert/1 + + %% Versions test + , tc_abi_version/1 %% testcase to verify env works %% , tc_network/1 ]). @@ -192,7 +195,10 @@ init_per_suite(Config) -> Config. end_per_suite(_Config) -> + quicer_test_lib:report_active_connections(), application:stop(quicer), + code:purge(quicer_nif), + code:delete(quicer_nif), ok. @@ -213,6 +219,7 @@ end_per_group(_Groupname, _Config) -> %%% Testcase specific setup/teardown %%%=================================================================== init_per_testcase(_TestCase, Config) -> + quicer_test_lib:cleanup_msquic(), [{timetrap, 5000} | Config]. end_per_testcase(tc_close_lib_test, _Config) -> @@ -228,10 +235,13 @@ end_per_testcase(tc_lib_re_registration_neg, _Config) -> end_per_testcase(tc_open_listener_neg_1, _Config) -> quicer:open_lib(), quicer:reg_open(); +end_per_testcase(tc_lib_registration_neg, _Config) -> + quicer:reg_open(); end_per_testcase(_TestCase, _Config) -> quicer:terminate_listener(mqtt), - Unhandled = quicer_test_lib:receive_all(), - Unhandled =/= [] andalso ct:comment("What left in the message queue: ~p", [Unhandled]), + quicer_test_lib:report_unhandled_messages(), + quicer_test_lib:report_active_connections(fun ct:pal/2), + ct:pal("Counters ~p", [quicer:perf_counters()]), ok. %%%=================================================================== @@ -321,7 +331,6 @@ tc_open_listener_inval_reg(Config) -> quicer:reg_open(), ok. - tc_stream_client_init(Config) -> Port = select_port(), Owner = self(), @@ -340,7 +349,6 @@ tc_stream_client_init(Config) -> ct:fail("timeout") end. - tc_stream_client_send_binary(Config) -> Port = select_port(), Owner = self(), @@ -574,7 +582,10 @@ tc_stream_passive_receive_shutdown(Config) -> {ok, <<"pong">>} = quicer:recv(Stm, 0), {ok, 4} = quicer:send(Stm, <<"ping">>, ?QUIC_SEND_FLAG_FIN), {ok, <<"pong">>} = quicer:recv(Stm, 0), - {error, peer_send_shutdown} = quicer:recv(Stm, 0), + case quicer:recv(Stm, 0) of + {error, peer_send_shutdown} -> ok; + {error, closed} -> ok + end, quicer:close_connection(Conn), SPid ! done, ensure_server_exit_normal(Ref) @@ -612,7 +623,10 @@ tc_stream_passive_receive_aborted(Config) -> {ok, 4} = quicer:send(Stm, <<"ping">>), {ok, <<"ping">>} = quicer:recv(Stm, 0), {ok, 5} = quicer:send(Stm, <<"Abort">>), - {error, peer_send_aborted} = quicer:recv(Stm, 0), + case quicer:recv(Stm, 0) of + {error, peer_send_aborted} -> ok; + {error, closed} -> ok + end, quicer:close_connection(Conn), SPid ! done, ensure_server_exit_normal(Ref) @@ -698,7 +712,12 @@ tc_stream_send_after_conn_close(Config) -> %% a) Just close connection, stream is not created in QUIC %% b) Close the connection after the stream is created in QUIC ok = quicer:close_connection(Conn), - {error, stm_send_error, aborted} = quicer:send(Stm, <<"ping2">>), + case quicer:send(Stm, <<"ping2">>) of + {error, closed} -> + ok; + {error, stm_send_error, aborted} -> + ok + end, SPid ! done, ok = ensure_server_exit_normal(Ref) after 1000 -> @@ -918,7 +937,7 @@ tc_getopt(Config) -> receive {quic, <<"ping">>, Stm, _} -> ok end, ok = quicer:close_connection(Conn), %% @todo unsupp in msquic, leave it for now - {error, not_found} = quicer:getopt(Conn, param_conn_close_reason_phrase), + {error, _} = quicer:getopt(Conn, param_conn_close_reason_phrase), SPid ! done, ensure_server_exit_normal(Ref) after 5000 -> @@ -1009,9 +1028,13 @@ tc_get_stream_0rtt_length(Config) -> end, %% before stream shutdown, {error, invalid_state} = quicer:getopt(Stm, param_stream_0rtt_length), - quicer:shutdown_stream(Stm), - {ok, Val} = quicer:getopt(Stm, param_stream_0rtt_length), - ?assert(is_integer(Val)), + quicer:async_shutdown_stream(Stm), + case quicer:getopt(Stm, param_stream_0rtt_length) of + {ok, Val} -> ?assert(is_integer(Val)); + {error, invalid_state} -> ok; + {error, invalid_parameter} -> ok + end, + quicer:close_connection(Conn), SPid ! done, ensure_server_exit_normal(Ref) after 5000 -> @@ -1035,8 +1058,6 @@ tc_get_stream_ideal_sndbuff_size(Config) -> {ok, Val} = quicer:getopt(Stm, param_stream_ideal_send_buffer_size), ?assert(is_integer(Val)), ok = quicer:shutdown_stream(Stm), - {ok, Val2} = quicer:getopt(Stm, param_stream_ideal_send_buffer_size), - ?assert(is_integer(Val2)), SPid ! done, ensure_server_exit_normal(Ref) after 5000 -> @@ -1110,7 +1131,7 @@ tc_peername_v6(Config) -> tc_peername_v4(Config) -> Port = select_port(), Owner = self(), - {_SPid, _Ref} = spawn_monitor(fun() -> echo_server(Owner, Config, Port) end), + {SPid, _Ref} = spawn_monitor(fun() -> echo_server(Owner, Config, Port) end), receive listener_ready -> {ok, Conn} = quicer:connect("127.0.0.1", Port, default_conn_opts(), 5000), @@ -1122,7 +1143,8 @@ tc_peername_v4(Config) -> true = is_integer(RPort), ct:pal("addr is ~p", [Addr]), "127.0.0.1" = inet:ntoa(Addr), - ok = quicer:close_connection(Conn) + ok = quicer:close_connection(Conn), + SPid ! done %{error, _} = quicer:peername(Conn) after 5000 -> ct:fail("listener_timeout") @@ -1158,11 +1180,10 @@ tc_alpn_mismatch(Config) -> receive ok -> ct:fail("illegal connection"); - {error, transport_down} -> - ok - after 1000 -> - SPid ! done - end + {error, transport_down, #{error := 376, status := alpn_neg_failure}} -> + ok + end, + SPid ! done after 1000 -> ct:fail("timeout") end. @@ -1523,6 +1544,7 @@ tc_setopt_stream_unsupp_opts(Config) -> {ok, <<"ping">>} = quicer:recv(Stm, 0), % try to set priority out of range {error, param_error} = quicer:setopt(Stm, param_stream_priority, 65536), + quicer:shutdown_stream(Stm), SPid ! done, ensure_server_exit_normal(Ref) after 5000 -> @@ -1721,11 +1743,13 @@ tc_stream_open_flag_unidirectional(Config) -> -> ct:pal("stream is closed due to connecion idle") end, ?assert(is_integer(Rid)), - ?assert(Rid =/= 0). + ?assert(Rid =/= 0), + quicer:close_connection(Conn). tc_stream_start_flag_fail_blocked(Config) -> Port = select_port(), application:ensure_all_started(quicer), + %% Given a server with 0 allowed remote bidi stream. ListenerOpts = [{conn_acceptors, 32}, {quic_event_mask, ?QUICER_STREAM_EVENT_MASK_START_COMPLETE} | lists:keyreplace(peer_bidi_stream_count, 1, default_listen_opts(Config), {peer_bidi_stream_count,0})], ConnectionOpts = [ {conn_callback, quicer_server_conn_callback} @@ -1737,16 +1761,27 @@ tc_stream_start_flag_fail_blocked(Config) -> ct:pal("Listener Options: ~p", [Options]), {ok, _QuicApp} = quicer:spawn_listener(mqtt, Port, Options), {ok, Conn} = quicer:connect("localhost", Port, default_conn_opts(), 5000), + %% When a client tries to start a bidi stream with flag "QUIC_STREAM_START_FLAG_FAIL_BLOCKED" {ok, Stm} = quicer:start_stream(Conn, [ {active, 3}, {start_flag, ?QUIC_STREAM_START_FLAG_FAIL_BLOCKED} , {quic_event_mask, ?QUICER_STREAM_EVENT_MASK_START_COMPLETE} ]), {ok, Rid} = quicer:get_stream_rid(Stm), - {ok, 5} = quicer:async_send(Stm, <<"ping1">>), + case quicer:async_send(Stm, <<"ping1">>) of + {ok, 5} -> + ok; + {error, closed} -> + ok; + {error, stm_send_error, invalid_state} -> + %% Deps on the timing + ok + end, receive {quic, <<"ping1">>, Stm, _} -> ct:fail("Should not get ping1 due to rate limiter"); {quic, start_completed, Stm, #{status := stream_limit_reached, stream_id := StreamID}} -> + %% Then stream start should fail with reason stream_limit_reached + quicer:close_stream(Stm, ?QUIC_STREAM_SHUTDOWN_FLAG_ABORT bor ?QUIC_STREAM_SHUTDOWN_FLAG_IMMEDIATE, 0, 1000), ct:pal("Stream ~p limit reached", [StreamID]); {quic, start_completed, Stm, #{status := AtomStatus, stream_id := StreamID, is_peer_accepted := _PeerAccepted} @@ -1754,6 +1789,7 @@ tc_stream_start_flag_fail_blocked(Config) -> ct:fail("Stream ~pstart complete with unexpect reason: ~p", [StreamID, AtomStatus]) end, + %% Then stream is closed automatically receive {quic, stream_closed, Stm, _} -> ct:failed("Stream ~p is closed but shouldn't since QUIC_STREAM_START_FLAG_SHUTDOWN_ON_FAIL is unset", [Stm]); @@ -1764,8 +1800,8 @@ tc_stream_start_flag_fail_blocked(Config) -> {quic, closed, Conn, _Flags} -> ct:pal("Connecion is closed ~p", [Conn]) end, - ?assert(is_integer(Rid)), - ?assert(Rid =/= 0). + quicer:terminate_listener(mqtt), + ?assert(is_integer(Rid)). tc_stream_start_flag_immediate(Config) -> Port = select_port(), @@ -1800,6 +1836,7 @@ tc_stream_start_flag_immediate(Config) -> tc_stream_start_flag_shutdown_on_fail(Config) -> Port = select_port(), application:ensure_all_started(quicer), + %% Given a server with 0 allowed remote bidi stream. ListenerOpts = [{conn_acceptors, 32}, {quic_event_mask, ?QUICER_STREAM_EVENT_MASK_START_COMPLETE} | lists:keyreplace(peer_bidi_stream_count, 1, default_listen_opts(Config), {peer_bidi_stream_count,0})], ConnectionOpts = [ {conn_callback, quicer_server_conn_callback} @@ -1811,6 +1848,7 @@ tc_stream_start_flag_shutdown_on_fail(Config) -> ct:pal("Listener Options: ~p", [Options]), {ok, _QuicApp} = quicer:spawn_listener(mqtt, Port, Options), {ok, Conn} = quicer:connect("localhost", Port, default_conn_opts(), 5000), + %% When a client tries to start a bidi stream with flag "QUIC_STREAM_START_FLAG_FAIL_BLOCKED" unset {ok, Stm} = quicer:start_stream(Conn, [ {active, 3}, {start_flag, ?QUIC_STREAM_START_FLAG_SHUTDOWN_ON_FAIL bor ?QUIC_STREAM_START_FLAG_FAIL_BLOCKED } @@ -1819,6 +1857,7 @@ tc_stream_start_flag_shutdown_on_fail(Config) -> {ok, Rid} = quicer:get_stream_rid(Stm), case quicer:async_send(Stm, <<"ping1">>) of {ok, 5} -> ok; + {error, closed} -> ok; {error, stm_send_error, invalid_state} -> ok %% already closed end, receive @@ -1842,8 +1881,7 @@ tc_stream_start_flag_shutdown_on_fail(Config) -> Other -> ct:fail("Unexpected event ~p after stream start complete", [Other]) end, - ?assert(is_integer(Rid)), - ?assert(Rid =/= 0). + ?assert(is_integer(Rid)). tc_stream_start_flag_indicate_peer_accept_1(Config) -> Port = select_port(), @@ -2128,9 +2166,13 @@ tc_event_start_compl_client(Config) -> [{param_conn_disable_1rtt_encryption, true} | default_conn_opts()], 5000), %% Stream 1 enabled - {ok, Stm} = quicer:start_stream(Conn, [{active, true}, {quic_event_mask, ?QUICER_STREAM_EVENT_MASK_START_COMPLETE}]), + {ok, Stm} = quicer:start_stream(Conn, [{active, true}, {quic_event_mask, ?QUICER_STREAM_EVENT_MASK_START_COMPLETE}, + {start_flag, ?QUIC_STREAM_START_FLAG_SHUTDOWN_ON_FAIL} + ]), %% Stream 2 disabled - {ok, Stm2} = quicer:start_stream(Conn, [{active, true}, {quic_event_mask, 0}]), + {ok, Stm2} = quicer:start_stream(Conn, [{active, true}, {quic_event_mask, 0}, + {start_flag, ?QUIC_STREAM_START_FLAG_SHUTDOWN_ON_FAIL} + ]), {ok, 5} = quicer:async_send(Stm, <<"ping1">>), {ok, 5} = quicer:async_send(Stm, <<"ping2">>), receive @@ -2145,7 +2187,7 @@ tc_event_start_compl_client(Config) -> {quic, start_completed, Stm2, #{status := Status}} -> ct:fail("Stream ~p should NOT recv event : ~p", [Stm, Status]) - after 0 -> + after 500 -> ok end, quicer:close_connection(Conn), @@ -2323,6 +2365,7 @@ tc_direct_send_over_conn_block(Config) -> after 100 -> ct:pal("No resp from unidi Stm2") end, + quicer:close_connection(Conn), ok. tc_direct_send_over_conn_fail(Config) -> @@ -2342,16 +2385,22 @@ tc_direct_send_over_conn_fail(Config) -> [{param_conn_disable_1rtt_encryption, true} | default_conn_opts()]), %% Stream 1 enabled - {ok, Stm} = quicer:start_stream(Conn, [{active, true}, {quic_event_mask, ?QUICER_STREAM_EVENT_MASK_START_COMPLETE}]), + {ok, Stm} = quicer:start_stream(Conn, [{active, true}, {quic_event_mask, ?QUICER_STREAM_EVENT_MASK_START_COMPLETE}, + {start_flag, ?QUIC_STREAM_START_FLAG_SHUTDOWN_ON_FAIL} + ]), {ok, 5} = quicer:async_send(Stm, <<"ping1">>), quicer:shutdown_connection(Conn), %% csend over a closed conn - {error, stm_open_error, invalid_state} = - quicer:async_csend(Conn, <<"ping2">>, [{active, true}, {quic_event_mask, ?QUICER_STREAM_EVENT_MASK_START_COMPLETE}, - {open_flag, ?QUIC_STREAM_OPEN_FLAG_UNIDIRECTIONAL} - ], ?QUIC_SEND_FLAG_ALLOW_0_RTT), + + case quicer:async_csend(Conn, <<"ping22">>, [{active, true}, {quic_event_mask, ?QUICER_STREAM_EVENT_MASK_START_COMPLETE}, + {open_flag, ?QUIC_STREAM_OPEN_FLAG_UNIDIRECTIONAL} + ], ?QUIC_SEND_FLAG_ALLOW_0_RTT) of + {error, closed} -> ok; + {error, stm_open_error, invalid_parameter} -> ok; + {error, stm_open_error, invalid_state} -> ok + end, receive {quic, start_completed, Stm0, #{status := StartStatus, stream_id := StreamId2}} -> @@ -2366,6 +2415,7 @@ tc_direct_send_over_conn_fail(Config) -> #{status := StartStatusX, stream_id := StreamIdX}} when StmX =/= Stm0 -> ct:fail("Stream id: ~p started: ~p", [StreamIdX, StartStatusX]) after 100 -> + quicer:close_connection(Conn), ok end. @@ -2496,6 +2546,9 @@ tc_peercert_server_nocert(Config) -> ensure_server_exit_normal(Ref), ok. +tc_abi_version(Config) -> + ?assertEqual(1, quicer:abi_version()). + %%% ==================== %%% Internal helpers %%% ==================== @@ -2547,6 +2600,8 @@ echo_server_stm_loop(L, Conn, Stms) -> {error, cancelled} -> ct:pal("echo server: send cancelled: ~p ", [Bin]), cancelled; + {error, closed} -> + closed; {ok, _} -> ok end, @@ -2737,11 +2792,14 @@ simple_conn_server_client_cert_loop(L, Conn, Owner) -> conn_server_with(Owner, Port, Opts) -> {ok, L} = quicer:listen(Port, Opts), Owner ! listener_ready, - {ok, Conn} = quicer:accept(L, [], 10000), - {ok, Conn} = quicer:handshake(Conn), + case quicer:accept(L, [], 10000) of + {error, _} -> + quicer:close_listener(L); + {ok, Conn} -> + {ok, Conn} = quicer:handshake(Conn) + end, receive done -> - quicer:close_listener(L), - ok + quicer:close_listener(L) end. simple_stream_server(Owner, Config, Port) -> @@ -2752,7 +2810,12 @@ simple_stream_server(Owner, Config, Port) -> {ok, Conn} = quicer:handshake(Conn), receive {quic, new_stream, Stream, _Props} -> - {ok, StreamId} = quicer:get_stream_id(Stream), + StreamId = case quicer:get_stream_id(Stream) of + {ok, Stm} -> + Stm; + {error, _} -> + simple_stream_server_exit(L) + end, ct:pal("New StreamID: ~p", [StreamId]), receive {quic, shutdown, Conn, _ErrorCode} -> @@ -2761,7 +2824,7 @@ simple_stream_server(Owner, Config, Port) -> {quic, peer_send_shutdown, Stream, undefined} -> quicer:close_stream(Stream); done -> - exit(normal) + simple_stream_server_exit(L) end; {quic, shutdown, Conn, _ErrorCode} -> ct:pal("Received Conn close for ~p", [Conn]), @@ -2774,6 +2837,9 @@ simple_stream_server(Owner, Config, Port) -> done -> ok end, + simple_stream_server_exit(L). + +simple_stream_server_exit(L) -> quicer:close_listener(L). diff --git a/test/quicer_connection_SUITE.erl b/test/quicer_connection_SUITE.erl index 713a2957..f4d6d610 100644 --- a/test/quicer_connection_SUITE.erl +++ b/test/quicer_connection_SUITE.erl @@ -58,6 +58,8 @@ init_per_suite(Config) -> %% @end %%-------------------------------------------------------------------- end_per_suite(_Config) -> + code:purge(quicer_nif), + code:delete(quicer_nif), ok. %%-------------------------------------------------------------------- @@ -83,7 +85,9 @@ init_per_group(suite_reg, Config) -> %% @end %%-------------------------------------------------------------------- end_per_group(suite_reg, Config) -> - quicer:shutdown_registration(proplists:get_value(quic_registration, Config)); + Reg = proplists:get_value(quic_registration, Config), + quicer:shutdown_registration(Reg), + ok = quicer:close_registration(Reg); end_per_group(_GroupName, _Config) -> ok. @@ -96,6 +100,8 @@ end_per_group(_GroupName, _Config) -> %% @end %%-------------------------------------------------------------------- init_per_testcase(_TestCase, Config) -> + application:ensure_all_started(quicer), + quicer_test_lib:cleanup_msquic(), Config. %%-------------------------------------------------------------------- @@ -107,6 +113,8 @@ init_per_testcase(_TestCase, Config) -> %% @end %%-------------------------------------------------------------------- end_per_testcase(_TestCase, _Config) -> + quicer_test_lib:report_active_connections(), + ct:pal("Counters ~p", [quicer:perf_counters()]), ok. %%-------------------------------------------------------------------- @@ -478,7 +486,8 @@ run_tc_conn_client_bad_cert(Config)-> case quicer:send(Stm, <<"ping">>) of {ok, 4} -> ok; {error, cancelled} -> ok; - {error, stm_send_error, aborted} -> ok + {error, stm_send_error, aborted} -> ok; + {error, closed} -> ok end, receive {quic, transport_shutdown, _Ref, @@ -572,7 +581,7 @@ tc_conn_controlling_process(Config) -> tc_conn_opt_ideal_processor(Config) -> Port = select_port(), Owner = self(), - {_SPid, _Ref} = spawn_monitor(fun() -> echo_server(Owner, Config, Port) end), + {SPid, _Ref} = spawn_monitor(fun() -> echo_server(Owner, Config, Port) end), receive listener_ready -> {ok, Conn} = quicer:connect("127.0.0.1", Port, default_conn_opts(Config), 5000), @@ -580,7 +589,8 @@ tc_conn_opt_ideal_processor(Config) -> {ok, 4} = quicer:send(Stm, <<"ping">>), {ok, Processor} = quicer:getopt(Conn, param_conn_ideal_processor), ?assert(is_integer(Processor)), - ok = quicer:close_connection(Conn) + ok = quicer:close_connection(Conn), + SPid ! done after 5000 -> ct:fail("listener_timeout") end. @@ -588,7 +598,7 @@ tc_conn_opt_ideal_processor(Config) -> tc_conn_opt_share_udp_binding(Config) -> Port = select_port(), Owner = self(), - {_SPid, _Ref} = spawn_monitor(fun() -> echo_server(Owner, Config, Port) end), + {SPid, _Ref} = spawn_monitor(fun() -> echo_server(Owner, Config, Port) end), receive listener_ready -> {ok, Conn} = quicer:connect("127.0.0.1", Port, default_conn_opts(Config), 5000), @@ -598,7 +608,8 @@ tc_conn_opt_share_udp_binding(Config) -> ?assert(is_boolean(IsShared)), {error, invalid_state} = quicer:setopt(Conn, param_conn_share_udp_binding, not IsShared), {ok, IsShared} = quicer:getopt(Conn, param_conn_share_udp_binding), - ok = quicer:close_connection(Conn) + ok = quicer:close_connection(Conn), + SPid ! done after 5000 -> ct:fail("listener_timeout") end. @@ -606,7 +617,7 @@ tc_conn_opt_share_udp_binding(Config) -> tc_conn_opt_local_bidi_stream_count(Config) -> Port = select_port(), Owner = self(), - {_SPid, _Ref} = spawn_monitor(fun() -> echo_server(Owner, Config, Port) end), + {SPid, _Ref} = spawn_monitor(fun() -> echo_server(Owner, Config, Port) end), receive listener_ready -> {ok, Conn} = quicer:connect("127.0.0.1", Port, default_conn_opts(Config), 5000), @@ -615,7 +626,8 @@ tc_conn_opt_local_bidi_stream_count(Config) -> {ok, Cnt} = quicer:getopt(Conn, param_conn_local_bidi_stream_count), ?assert(is_integer(Cnt)), {error, invalid_parameter} = quicer:setopt(Conn, param_conn_local_bidi_stream_count, Cnt + 2), - ok = quicer:close_connection(Conn) + ok = quicer:close_connection(Conn), + SPid ! done after 5000 -> ct:fail("listener_timeout") end. @@ -623,7 +635,7 @@ tc_conn_opt_local_bidi_stream_count(Config) -> tc_conn_opt_local_uni_stream_count(Config) -> Port = select_port(), Owner = self(), - {_SPid, _Ref} = spawn_monitor(fun() -> echo_server(Owner, Config, Port) end), + {SPid, _Ref} = spawn_monitor(fun() -> echo_server(Owner, Config, Port) end), receive listener_ready -> {ok, Conn} = quicer:connect("127.0.0.1", Port, default_conn_opts(Config), 5000), @@ -632,7 +644,8 @@ tc_conn_opt_local_uni_stream_count(Config) -> {ok, Cnt} = quicer:getopt(Conn, param_conn_local_unidi_stream_count), ?assert(is_integer(Cnt)), {error, invalid_parameter} = quicer:setopt(Conn, param_conn_local_unidi_stream_count, Cnt + 2), - ok = quicer:close_connection(Conn) + ok = quicer:close_connection(Conn), + SPid ! done after 5000 -> ct:fail("listener_timeout") end. @@ -788,7 +801,8 @@ echo_server(Owner, Config, Port)-> end end, ct:pal("echo server stream accepted", []), - echo_server_stm_loop(L, Conn, [Stm]); + catch echo_server_stm_loop(L, Conn, [Stm]), + quicer:close_listener(L); {error, listener_start_error, 200000002} -> ct:pal("echo_server: listener_start_error", []), timer:sleep(100), @@ -815,6 +829,8 @@ echo_server_stm_loop(L, Conn, Stms) -> {error, cancelled} -> ct:pal("echo server: send cancelled: ~p ", [Bin]), cancelled; + {error, closed} -> + closed; {ok, _} -> ok end, @@ -868,8 +884,7 @@ echo_server_stm_loop(L, Conn, Stms) -> echo_server_stm_loop(L, Conn, NewStmList); done -> ct:pal("echo server shutting down", []), - quicer:async_close_connection(Conn), - quicer:close_listener(L) + quicer:async_close_connection(Conn) end. default_conn_opts(Config) -> diff --git a/test/quicer_echo_server_stream_callback.erl b/test/quicer_echo_server_stream_callback.erl index 9084f135..677ae3c0 100644 --- a/test/quicer_echo_server_stream_callback.erl +++ b/test/quicer_echo_server_stream_callback.erl @@ -94,13 +94,23 @@ handle_stream_data(Stream, Bin, _Opts, #{ sent_bytes := Cnt } = State) -> case maps:get(is_echo_new_stream, StreamOpts, false) of false -> - {ok, Size} = quicer:send(Stream, echo_msg(Bin, State)), - {ok, State#{ sent_bytes => Cnt + Size }}; + case quicer:send(Stream, echo_msg(Bin, State)) of + {ok, Size} -> + {ok, State#{ sent_bytes => Cnt + Size }}; + _ -> + %% handle error in test aborted. + {ok, State} + end; true -> %% echo reply with a new stream from server to client. {ok, EchoStream} = quicer:start_stream(Conn, StreamOpts), - {ok, Size} = quicer:send(EchoStream, echo_msg(Bin, State)), - {ok, State#{ sent_bytes => Cnt + Size, echo_stream => EchoStream }} + case quicer:send(EchoStream, echo_msg(Bin, State)) of + {ok, Size} -> + {ok, State#{ sent_bytes => Cnt + Size, echo_stream => EchoStream }}; + _ -> + %% handle error in test aborted. + {ok, State} + end end; handle_stream_data(Stream, Bin, _Opts, #{ sent_bytes := Cnt , echo_stream := _Ignore diff --git a/test/quicer_listener_SUITE.erl b/test/quicer_listener_SUITE.erl index 36ef1bc3..b295baf5 100644 --- a/test/quicer_listener_SUITE.erl +++ b/test/quicer_listener_SUITE.erl @@ -45,6 +45,7 @@ suite() -> %% @end %%-------------------------------------------------------------------- init_per_suite(Config) -> + application:ensure_all_started(quicer), quicer_test_lib:generate_tls_certs(Config), Config. @@ -79,7 +80,9 @@ init_per_group(suite_reg, Config) -> %% @end %%-------------------------------------------------------------------- end_per_group(suite_reg, Config) -> - quicer:shutdown_registration(proplists:get_value(quic_registration, Config)); + Reg = proplists:get_value(quic_registration, Config), + quicer:shutdown_registration(Reg), + ok = quicer:close_registration(Reg); end_per_group(_GroupName, _Config) -> ok. @@ -93,6 +96,7 @@ end_per_group(_GroupName, _Config) -> %%-------------------------------------------------------------------- init_per_testcase(_TestCase, Config) -> application:ensure_all_started(quicer), + quicer_test_lib:cleanup_msquic(), Config. %%-------------------------------------------------------------------- @@ -106,6 +110,7 @@ init_per_testcase(_TestCase, Config) -> end_per_testcase(_TestCase, Config) -> RegH = proplists:get_value(quic_registration, Config, global), [quicer:close_listener(L, 1000) || L <- quicer:get_listeners(RegH)], + quicer_test_lib:report_active_connections(), ok. %%-------------------------------------------------------------------- diff --git a/test/quicer_reg_SUITE.erl b/test/quicer_reg_SUITE.erl index 91d729bf..32f7b56a 100644 --- a/test/quicer_reg_SUITE.erl +++ b/test/quicer_reg_SUITE.erl @@ -90,7 +90,7 @@ init_per_testcase(_TestCase, Config) -> %%-------------------------------------------------------------------- end_per_testcase(_TestCase, _Config) -> erlang:garbage_collect(self(), [{type, major}]), - timer:sleep(1000), + quicer_test_lib:report_active_connections(), ok. %%-------------------------------------------------------------------- @@ -167,7 +167,8 @@ tc_shutdown_3_abnormal(_Config) -> {ok, Reg} = quicer:new_registration(Name, Profile), ?assertEqual({error, badarg}, quicer:shutdown_registration(Reg, 1, 2)), ?assertEqual({error, badarg}, quicer:shutdown_registration(Reg, 1, foo)), - ?assertEqual({error, badarg}, quicer:shutdown_registration(Reg, true, -1)). + ?assertEqual({error, badarg}, quicer:shutdown_registration(Reg, true, -1)), + ok = quicer:shutdown_registration(Reg). tc_shutdown_ok(_Config) -> Name = atom_to_list(?FUNCTION_NAME), diff --git a/test/quicer_snb_SUITE.erl b/test/quicer_snb_SUITE.erl index f4ab9ba3..01fb329e 100644 --- a/test/quicer_snb_SUITE.erl +++ b/test/quicer_snb_SUITE.erl @@ -97,6 +97,7 @@ init_per_suite(Config) -> %% @end %%-------------------------------------------------------------------- end_per_suite(_Config) -> + quicer_test_lib:report_active_connections(), ok. %%-------------------------------------------------------------------- @@ -135,6 +136,7 @@ init_per_testcase(tc_listener_inval_local_addr, Config) -> Config end; init_per_testcase(_TestCase, Config) -> + quicer_test_lib:cleanup_msquic(), Config. %%-------------------------------------------------------------------- @@ -148,8 +150,8 @@ init_per_testcase(_TestCase, Config) -> end_per_testcase(_TestCase, _Config) -> quicer:terminate_listener(mqtt), snabbkaffe:cleanup(), - Unhandled = quicer_test_lib:receive_all(), - Unhandled =/= [] andalso ct:comment("What left in the message queue: ~p", [Unhandled]), + quicer_test_lib:report_unhandled_messages(), + quicer_test_lib:report_active_connections(), ok. %%-------------------------------------------------------------------- @@ -414,6 +416,7 @@ tc_conn_owner_down(Config) -> begin {ok, _QuicApp} = quicer_start_listener(mqtt, Port, Options), {ok, Conn} = quicer:connect("localhost", Port, default_conn_opts(), 5000), + {ok, CRid} = quicer:get_conn_rid(Conn), {ok, Stm} = quicer:start_stream(Conn, [{active, false}]), {ok, 4} = quicer:send(Stm, <<"ping">>), {ok, <<"ping">>} = quicer:recv(Stm, 4), @@ -422,6 +425,12 @@ tc_conn_owner_down(Config) -> receive down -> ok end end), quicer:controlling_process(Conn, Pid), + + ?assert(timeout =/= ?block_until(#{ ?snk_kind := debug + , function := "connection_controlling_process" + , tag := "exit" + , resource_id := CRid + }, 1000, 1000)), Pid ! down, ?assert(timeout =/= ?block_until( @@ -438,7 +447,7 @@ tc_conn_owner_down(Config) -> #{?snk_kind := debug, event := closed, module := quicer_connection}, 1000, 1000)), ct:pal("stop listener"), ok = quicer:terminate_listener(mqtt), - {ok, CRid} = quicer:get_conn_rid(Conn), + {CRid, SRid} end, fun(Result, Trace) -> @@ -826,7 +835,8 @@ tc_conn_idle_close(Config) -> end, case quicer:send(Stm, <<"ping2">>) of {error, stm_send_error, invalid_state} -> ok; - {error, cancelled} -> ok + {error, cancelled} -> ok; + {error, closed} -> ok end, ?block_until( @@ -883,9 +893,9 @@ tc_conn_gc(Config) -> | default_stream_opts() ], Options = {ListenerOpts, ConnectionOpts, StreamOpts}, ct:pal("Listener Options: ~p", [Options]), - ?check_trace(#{timetrap => 1000}, + ?check_trace(#{timetrap => 100000}, begin - %% Spawn a process that will die without handler cleanups + %% Spawn a process that will die without handle cleanups %% The dead process should trigger a connection close %% The dead process should trigger a GC {ok, _QuicApp} = quicer_start_listener(mqtt, Port, Options), @@ -931,12 +941,25 @@ tc_conn_gc(Config) -> {ok, _} = ?block_until(#{ ?snk_kind := debug , context := "callback" , function := "resource_conn_dealloc_callback" - , resource_id := CRid + , resource_id := 0 , tag := "end"}, 5000, 1000), + timer:sleep(1000), {SRid, CRid} end, - fun({_SRid, CRid}, Trace) -> + fun({_SRid, CRid}, Trace0) -> + Trace = flush_previous_run(Trace0, fun(#{ ?snk_kind := debug + , context := "callback" + , function := "ClientConnectionCallback" + , mark := ?QUIC_CONNECTION_EVENT_CONNECTED + , resource_id := Rid + , tag := "event" + }) when Rid == CRid -> + true; + (_) -> + false + end + ), ct:pal("Trace is ~p", [Trace]), ct:pal("Target SRid: ~p, CRid: ~p", [_SRid, CRid]), %% check that at client side, GC is triggered after connection close. @@ -951,7 +974,7 @@ tc_conn_gc(Config) -> #{ ?snk_kind := debug , context := "callback" , function := "resource_conn_dealloc_callback" - , resource_id := CRid + , resource_id := 0 , tag := "end"}, Trace)), ?assert(?strict_causality(#{ ?snk_kind := debug @@ -968,23 +991,9 @@ tc_conn_gc(Config) -> , mark := ?QUIC_CONNECTION_EVENT_SHUTDOWN_COMPLETE , tag := "event"}, Trace)), - - - TraceEvents = flush_previous_run(Trace, fun(#{ ?snk_kind := debug - , context := "callback" - , function := "ClientConnectionCallback" - , mark := ?QUIC_CONNECTION_EVENT_CONNECTED - , resource_id := Rid - , tag := "event" - }) when Rid == CRid -> - true; - (_) -> - false - end - ), ?assertEqual(1, length([ E || #{ function := "resource_conn_dealloc_callback" , resource_id := Rid - , tag := "end"} = E <- TraceEvents, Rid == CRid]) + , tag := "end"} = E <- Trace, Rid == 0]) ) end), ct:pal("stop listener"), @@ -1014,13 +1023,13 @@ tc_conn_no_gc(Config) -> {ok, Conn} = quicer:connect("localhost", Port, [{idle_timeout_ms, 1000}, {verify, none}, {alpn, ["sample"]}], 5000), + {ok, CRid} = quicer:get_conn_rid(Conn), _Child = spawn_link(fun() -> {ok, Stm} = quicer:start_stream(Conn, [{active, false}]), {ok, 4} = quicer:async_send(Stm, <<"ping">>), {ok, <<"ping">>} = quicer:recv(Stm, 4), quicer:shutdown_connection(Conn, 0, 0) end), - {ok, CRid} = quicer:get_conn_rid(Conn), %% Server Process {ok, #{resource_id := SRid}} = ?block_until(#{ ?snk_kind := debug @@ -1610,10 +1619,7 @@ tc_listener_no_acceptor(Config) -> begin {ok, _QuicApp} = quicer_start_listener(mqtt, Port, Options), {error, transport_down, #{status := connection_refused}} - = quicer:connect("localhost", Port, default_conn_opts(), 5000), - ct:pal("stop listener"), - ok = quicer:terminate_listener(mqtt), - timer:sleep(5000) + = quicer:connect("localhost", Port, default_conn_opts(), 5000) end, fun(_Result, Trace) -> ct:pal("Trace is ~p", [Trace]), @@ -1631,6 +1637,8 @@ tc_listener_no_acceptor(Config) -> }, Trace)) end), + ct:pal("stop listener"), + ok = quicer:terminate_listener(mqtt), ok. %% @doc this triggers listener start fail @@ -1714,6 +1722,7 @@ tc_conn_stop_notify_acceptor(Config) -> {error, closed} -> ok; {ok, _Stream} -> ok end, + quicer:close_listener(Listener), exit({normal, Acceptors}) end), receive {SPid, ready} -> ok end, diff --git a/test/quicer_test_lib.erl b/test/quicer_test_lib.erl index a8c8cfdb..28d29695 100644 --- a/test/quicer_test_lib.erl +++ b/test/quicer_test_lib.erl @@ -30,7 +30,12 @@ select_free_port/1, flush/1, ensure_server_exit_normal/1, - ensure_server_exit_normal/2 + ensure_server_exit_normal/2, + + report_active_connections/0, + report_active_connections/1, + + report_unhandled_messages/0 ]). @@ -40,6 +45,13 @@ , default_stream_opts/0 ]). +%% cleanups +-export([ reset_global_reg/0 + , shutdown_all_listeners/0 + , cleanup_msquic/0 + ]). + + %% ct helper -export([all_tcs/1]). @@ -345,6 +357,36 @@ ensure_server_exit_normal(MonRef, Timeout) -> ct:fail("server still running", []) end. +-spec report_active_connections() -> _. +report_active_connections() -> + report_active_connections(fun ct:comment/2). +report_active_connections(LogFun) -> + erlang:garbage_collect(), + {ok, Cnts} = quicer:perf_counters(), + ActiveStrms = proplists:get_value(strm_active, Cnts), + ActiveConns = proplists:get_value(conn_active, Cnts), + 0 =/= (ActiveStrms + ActiveConns) andalso + LogFun("active conns: ~p, strms: ~p", [ActiveConns, ActiveStrms]). + +-spec report_unhandled_messages() -> ok. +report_unhandled_messages() -> + Unhandled = quicer_test_lib:receive_all(), + Unhandled =/= [] andalso + ct:comment("What left in the message queue: ~p", [Unhandled]). + +-spec cleanup_msquic() -> ok. +cleanup_msquic() -> + shutdown_all_listeners(), + reset_global_reg(), + ok. + +reset_global_reg()-> + quicer:reg_close(), + quicer:reg_open(). + +shutdown_all_listeners() -> + lists:foreach(fun quicer:shutdown_listener/1, + quicer:listeners()). %%%_* Emacs ==================================================================== %%% Local Variables: diff --git a/test/quicer_upgrade_SUITE.erl b/test/quicer_upgrade_SUITE.erl new file mode 100644 index 00000000..a15cfde9 --- /dev/null +++ b/test/quicer_upgrade_SUITE.erl @@ -0,0 +1,237 @@ +%%-------------------------------------------------------------------- +%% Copyright (c) 2023 EMQ Technologies Co., Ltd. All Rights Reserved. +%% +%% Licensed under the Apache License, Version 2.0 (the "License"); +%% you may not use this file except in compliance with the License. +%% You may obtain a copy of the License at +%% +%% http://www.apache.org/licenses/LICENSE-2.0 +%% +%% Unless required by applicable law or agreed to in writing, software +%% distributed under the License is distributed on an "AS IS" BASIS, +%% WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +%% See the License for the specific language governing permissions and +%% limitations under the License. +%%-------------------------------------------------------------------- +-module(quicer_upgrade_SUITE). + +-compile(export_all). + +-include_lib("common_test/include/ct.hrl"). +-include_lib("eunit/include/eunit.hrl"). +%%-------------------------------------------------------------------- +%% @spec suite() -> Info +%% Info = [tuple()] +%% @end +%%-------------------------------------------------------------------- +suite() -> + [{timetrap,{seconds,30}}]. + +%%-------------------------------------------------------------------- +%% @spec init_per_suite(Config0) -> +%% Config1 | {skip,Reason} | {skip_and_save,Reason,Config1} +%% Config0 = Config1 = [tuple()] +%% Reason = term() +%% @end +%%-------------------------------------------------------------------- +init_per_suite(Config) -> + %% close all listeners under global registration + [quicer:close_listener(L, 1000) || L <- quicer:get_listeners()], + Config. + +%%-------------------------------------------------------------------- +%% @spec end_per_suite(Config0) -> term() | {save_config,Config1} +%% Config0 = Config1 = [tuple()] +%% @end +%%-------------------------------------------------------------------- +end_per_suite(_Config) -> + ok. + +%%-------------------------------------------------------------------- +%% @spec init_per_group(GroupName, Config0) -> +%% Config1 | {skip,Reason} | {skip_and_save,Reason,Config1} +%% GroupName = atom() +%% Config0 = Config1 = [tuple()] +%% Reason = term() +%% @end +%%-------------------------------------------------------------------- +init_per_group(_GroupName, Config) -> + Config. + +%%-------------------------------------------------------------------- +%% @spec end_per_group(GroupName, Config0) -> +%% term() | {save_config,Config1} +%% GroupName = atom() +%% Config0 = Config1 = [tuple()] +%% @end +%%-------------------------------------------------------------------- +end_per_group(_GroupName, _Config) -> + ok. + +%%-------------------------------------------------------------------- +%% @spec init_per_testcase(TestCase, Config0) -> +%% Config1 | {skip,Reason} | {skip_and_save,Reason,Config1} +%% TestCase = atom() +%% Config0 = Config1 = [tuple()] +%% Reason = term() +%% @end +%%-------------------------------------------------------------------- +init_per_testcase(TestCase, Config) -> + case erlang:function_exported(?MODULE, TestCase, 2) of + false -> + Config; + true -> + ?MODULE:TestCase(init, Config) + end. + +%%-------------------------------------------------------------------- +%% @spec end_per_testcase(TestCase, Config0) -> +%% term() | {save_config,Config1} | {fail,Reason} +%% TestCase = atom() +%% Config0 = Config1 = [tuple()] +%% Reason = term() +%% @end +%%-------------------------------------------------------------------- +end_per_testcase(_TestCase, _Config) -> + quicer_test_lib:report_active_connections(), + ok. + +%%-------------------------------------------------------------------- +%% @spec groups() -> [Group] +%% Group = {GroupName,Properties,GroupsAndTestCases} +%% GroupName = atom() +%% Properties = [parallel | sequence | Shuffle | {RepeatType,N}] +%% GroupsAndTestCases = [Group | {group,GroupName} | TestCase] +%% TestCase = atom() +%% Shuffle = shuffle | {shuffle,{integer(),integer(),integer()}} +%% RepeatType = repeat | repeat_until_all_ok | repeat_until_all_fail | +%% repeat_until_any_ok | repeat_until_any_fail +%% N = integer() | forever +%% @end +%%-------------------------------------------------------------------- +groups() -> + []. + +%%-------------------------------------------------------------------- +%% @spec all() -> GroupsAndTestCases | {skip,Reason} +%% GroupsAndTestCases = [{group,GroupName} | TestCase] +%% GroupName = atom() +%% TestCase = atom() +%% Reason = term() +%% @end +%%-------------------------------------------------------------------- +all() -> + quicer_test_lib:all_tcs(?MODULE). + +%%-------------------------------------------------------------------- +%% @spec TestCase(Config0) -> +%% ok | exit() | {skip,Reason} | {comment,Comment} | +%% {save_config,Config1} | {skip_and_save,Reason,Config1} +%% Config0 = Config1 = [tuple()] +%% Reason = term() +%% Comment = term() +%% @end +%%-------------------------------------------------------------------- +tc_nif_module_is_loaded(_Config) -> + ?assertMatch({file, _}, code:is_loaded(quicer_nif)). + +tc_nif_module_purge(init, Config) -> + %% When we have both old code and new code + case code:load_file(quicer_nif) of + {module, quicer_nif} -> + ok; + {error, not_purged} -> + ok + end, + Config. +tc_nif_module_purge(_Config) -> + %% Then purge old code should success (test nif lib no crash) + ?assertEqual(false, code:purge(quicer_nif)). + +tc_nif_no_old_module_purge(init, Config) -> + %% Give there is no old code of quicer_nif present + code:purge(quicer_nif), + Config. +tc_nif_no_old_module_purge(_Config) -> + %% Then purge non existing old code should success (test nif lib no crash) + ?assertEqual(false, code:purge(quicer_nif)). + +tc_nif_module_reload(init, Config) -> + %% Given no old code of quicer_nif present + code:purge(quicer_nif), + Config. +tc_nif_module_reload(_Config) -> + %% When reload quicer_nif with same module file + %% Then load success + ?assertEqual({module, quicer_nif}, code:load_file(quicer_nif)). + +tc_nif_module_load_current(init, Config) -> + %% Given no new/old code of quicer_nif present + ensure_module_no_code(quicer_nif), + Config. +tc_nif_module_load_current(_Config) -> + %% When load quicer_nif, it should success + ?assertEqual({module, quicer_nif}, code:load_file(quicer_nif)). + +tc_nif_module_softpurge(init, Config) -> + %% When we have both old code and new code + case code:load_file(quicer_nif) of + {module, quicer_nif} -> + ok; + {error, not_purged} -> + ok + end, + Config. +tc_nif_module_softpurge(_Config) -> + ?assertEqual(true, code:soft_purge(quicer_nif)). + +tc_nif_module_no_reinit(_Config) -> + %% Given quicer_nif is not loaded + ensure_module_no_code(quicer_nif), + Res = {error, {reload, "NIF library already loaded (reload disallowed since OTP 20)."}}, + %% When calling quicer_nif:init/1 with ABI version 1 + %% Then it should still fail + ?assertEqual(Res, quicer_nif:init(1)). + +tc_nif_module_load_fail_dueto_mismatch_abiversion(init, Config) -> + %% Given quicer_nif is not loaded + ensure_module_no_code(quicer_nif), + %% When _quicer_overrides_ provides ABI version 0 which is reserved + persistent_term:put({'_quicer_overrides_', abi_version}, 0), + Config. +tc_nif_module_load_fail_dueto_mismatch_abiversion(_Config) -> + %% Then load quicer_nif should fail due to abi version mismatch + ?assertEqual({error, on_load_failure}, code:load_file(quicer_nif)), + persistent_term:erase({'_quicer_overrides_', abi_version}). + +tc_nif_module_upgrade_fail_dueto_mismatch_abiversion(init, Config) -> + %% Given quicer_nif has current code + ensure_module_current_vsn(quicer_nif), + %% When _quicer_overrides_ provides ABI version 0 which is reserved + persistent_term:put({'_quicer_overrides_', abi_version}, 0), + Config. +tc_nif_module_upgrade_fail_dueto_mismatch_abiversion(_Config) -> + %% Then upgrade quicer_nif should fail due to abi version mismatch + ?assertEqual({error, on_load_failure}, code:load_file(quicer_nif)), + persistent_term:erase({'_quicer_overrides_', abi_version}), + %% Then quicer_nif should still be loaded + ?assertMatch({file, _}, code:is_loaded(quicer_nif)). + +%% Helpers + +%% @doc ensure neither old nor new code is present +ensure_module_no_code(Module) -> + %% purge old if any + code:purge(Module), + %% current become old + code:delete(Module), + %% assert no current. + not_loaded = code:module_status(Module), + %% force purge this old + _ = code:purge(Module). + +%% @doc ensure only current code is present +ensure_module_current_vsn(Module) -> + ensure_module_no_code(Module), + _ = code:load_file(Module), + ok.