The QUIC protocol features a Version field to enable the protocol to evolve and have future versions, multiple of which may be supported by a given implementation. MsQuic is no exception and currently supports Draft-29 and Version 1 of the QUIC protocol. By default, MsQuic clients start all connections with Version 1. MsQuic servers support Version 1 and Draft-29.
The Version Negotiation Extension is supported in MsQuic and is keeping pace with changes in the standard. It is enabled by default on both MsQuic client and server.
An application may decide that it needs a specific feature only availble in one version of QUIC. The application may also wish to change the order of preference of supported version in MsQuic. Both scenarios are supported via the QUIC_VERSION_SETTINGS
struct. Since there are three different version lists, the client MUST set all three to be the same.
The first version in the list of FullyDeployedVersions
will always be the initial version MsQuic starts the connection with.
Warning A client may only set a version that MsQuic supports. Any other value will cause
SetParam
to fail.
Use the following code snippet to change the default initial version, and only support a single QUIC version. It must be used before ConnectionStart
is called:
QUIC_VERSION_SETTINGS Settings = { 0 };
const uint32_t SupportedVersion = 0xff00001dU; // This is the Draft-29 version in HOST byte order. If the server does not support this, the connection will fail.
Settings.AcceptableVersionsList = &SupportedVersion;
Settings.AcceptableVersionsListLength = 1;
Settings.OfferedVersionsList = &SupportedVersion;
Settings.OfferedVersionsListLength = 1;
Settings.FullyDeployedVersionsList = &SupportedVersion;
Settings.FullyDeployedVersionsListLength = 1;
MsQuic->SetParam(
Connection,
QUIC_PARAM_CONN_VERSION_SETTINGS,
sizeof(Settings),
&Settings);
Changing the order of supported versions is the same as above, with the following change:
QUIC_VERSION_SETTINGS Settings = { 0 };
const uint32_t SupportedVersions[2] = {
0xff00001dU, // This is the Draft-29 version in HOST byte order. It will be used first.
0x00000001U // QUIC version 1 in HOST byte order. It will be used if a VN packet is received.
};
Settings.AcceptableVersionsList = SupportedVersions;
Settings.AcceptableVersionsListLength = 2;
Settings.OfferedVersionsList = SupportedVersions;
Settings.OfferedVersionsListLength = 2;
Settings.FullyDeployedVersionsList = SupportedVersions;
Settings.FullyDeployedVersionsListLength = 2;
The QUIC_VERSION_SETTINGS
can be set on a single QUIC_CONNECTION
, as well as a QUIC_CONFIGURATION
with SetParam
.
A server application may also want to restrict the QUIC versions it supports to ensure a specific feature is available, or to prevent older versions of QUIC from being used.
Configuring the QUIC versions on a MsQuic server is similar to configuring them on a client, however, the setting for server MUST be set globally, and not on the QUIC_CONFIGURATION
used for the QUIC_LISTENER
or QUIC_CONNECTION
.
If a server is not in a fleet, or the operator/application does not ever need to change QUIC versions, then all three lists in QUIC_VERSION_SETTINGS
MUST be the same.
If a server is deployed in a fleet, and the server operator wishes to change the supported QUIC versions, the Version Negotiation specification details how that should be done, quoted here:
When adding support for a new version:
- The first step is to progressively add support for the new version to all server instances. This step updates the Acceptable Versions but not the Offered Versions nor the Fully-Deployed Versions. Once all server instances have been updated, operators wait for at least one MSL to allow any in-flight Version Negotiation packets to arrive.
- Then, the second step is to progressively add the new version to Offered Versions on all server instances. Once complete, operators wait for at least another MSL.
- Finally, the third step is to progressively add the new version to Fully-Deployed Versions on all server instances.
When removing support for a version:
- The first step is to progressively remove the version from Fully-Deployed Versions on all server instances. Once it has been removed on all server instances, operators wait for at least one MSL to allow any in-flight Version Negotiation packets to arrive.
- Then, the second step is to progressively remove the version from Offered Versions on all server instances. Once complete, operators wait for at least another MSL.
- Finally, the third step is to progressively remove support for the version from all server instances. That step updates the Acceptable Versions.
Note that this opens connections to version downgrades (but only for partially-deployed versions) during the update window, since those could be due to clients communicating with both updated and non-updated server instances.
This snippet should execute before the server's QUIC_CONFIGURATION
is created:
QUIC_VERSION_SETTINGS Settings = { 0 };
const uint32_t SupportedVersions[2] = {
0xff00001dU, // This is the Draft-29 version in HOST byte order. It will be preferred over Version 1.
0x00000001U // QUIC version 1 in HOST byte order. It will be used if a client starts with Version 1, instead of Draft-29.
};
Settings.AcceptableVersionsList = SupportedVersion;
Settings.AcceptableVersionsListLength = 2;
Settings.OfferedVersionsList = SupportedVersion;
Settings.OfferedVersionsListLength = 2;
Settings.FullyDeployedVersionsList = SupportedVersion;
Settings.FullyDeployedVersionsListLength = 2;
MsQuic->SetParam(
NULL,
QUIC_PARAM_GLOBAL_VERSION_SETTINGS,
sizeof(Settings),
&Settings);
MsQuic supports setting the Acceptable Versions, Offered Versions, and Fully-Deployed Versions lists via the Windows registry. These settings are global for all servers and clients on the machine.
The registry settings are overridden by settings specified in the code.
The registry values must be created under the HKLM\System\CurrentControlSet\Services\MsQuic\Parameters
key.
Each list is stored in the registry as a REG_BINARY
type, with the version numbers in little-endian (host) order.
The registry value for Acceptable Versions must be named AcceptableVersions
.
The registry value for Offered Versions must be named OfferedVersions
.
The registry value for Fully-Deployed Versions must be named FullyDeployedVersions
.
Here's a sample .reg file that creates all three lists with QUIC version 2 first and QUIC version 1 after, in little endian order, and enables version negotiation:
Windows Registry Editor Version 5.00
[HKEY_LOCAL_MACHINE\SYSTEM\CurrentControlSet\Services\MsQuic\Parameters]
"AcceptableVersions"=hex:cf,43,33,6b,01,00,00,00
"OfferedVersions"=hex:cf,43,33,6b,01,00,00,00
"FullyDeployedVersions"=hex:cf,43,33,6b,01,00,00,00
"VersionNegotiationExtEnabled"=dword:00000001
The Version Negotiation Extension is off by default in our officially-released binaries, but can be enabled via registry or Settings. The Version Negotiated Extension has been standardized and is present in MsQuic since version 2.3.
The Version Negotiation Extension is enabled on client when QUIC_VERSION_SETTINGS
are set on the QUIC_CONFIGURATION
or QUIC_CONNECTION
via SetParam
.
This setting MUST be set before ConnectionStart
to take effect.
Enabling the Version Negotiation Extension on server follows the same restrictions as setting the QUIC version on server, i.e. it MUST be set globally, using SetParam
before the QUIC_CONFIGURATION
is opened for the server. It is set automatically when QUIC_VERSION_SETTINGS
are set, except via registry.