io.netty.incubator.codec.quic.Quiche Maven / Gradle / Ivy
The newest version!
/*
* Copyright 2020 The Netty Project
*
* The Netty Project licenses this file to you 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:
*
* https://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.
*/
package io.netty.incubator.codec.quic;
import io.netty.buffer.ByteBuf;
import io.netty.buffer.Unpooled;
import io.netty.util.collection.IntObjectHashMap;
import io.netty.util.internal.ClassInitializerUtil;
import io.netty.util.internal.NativeLibraryLoader;
import io.netty.util.internal.PlatformDependent;
import io.netty.util.internal.logging.InternalLogger;
import io.netty.util.internal.logging.InternalLoggerFactory;
import org.jetbrains.annotations.Nullable;
import javax.net.ssl.SSLException;
import javax.net.ssl.SSLHandshakeException;
import java.nio.ByteBuffer;
import java.nio.ByteOrder;
final class Quiche {
private static final InternalLogger logger = InternalLoggerFactory.getInstance(Quiche.class);
private static final boolean DEBUG_LOGGING_ENABLED = logger.isDebugEnabled();
private static final IntObjectHashMap ERROR_MAPPINGS = new IntObjectHashMap<>();
static {
// Preload all classes that will be used in the OnLoad(...) function of JNI to eliminate the possibility of a
// class-loader deadlock. This is a workaround for https://github.com/netty/netty/issues/11209.
// This needs to match all the classes that are loaded via NETTY_JNI_UTIL_LOAD_CLASS or looked up via
// NETTY_JNI_UTIL_FIND_CLASS.
ClassInitializerUtil.tryLoadClasses(Quiche.class,
// netty_quic_boringssl
byte[].class, String.class, BoringSSLCertificateCallback.class,
BoringSSLCertificateVerifyCallback.class, BoringSSLHandshakeCompleteCallback.class,
//netty_quic_quiche
QuicheLogger.class
);
try {
// First, try calling a side-effect free JNI method to see if the library was already
// loaded by the application.
quiche_version();
} catch (UnsatisfiedLinkError ignore) {
// The library was not previously loaded, load it now.
loadNativeLibrary();
}
// Let's enable debug logging for quiche if its enabled in our logger.
if (DEBUG_LOGGING_ENABLED) {
quiche_enable_debug_logging(new QuicheLogger(logger));
}
}
private static void loadNativeLibrary() {
// This needs to be kept in sync with what is defined in netty_quic_quiche.c
String libName = "netty_quiche";
ClassLoader cl = PlatformDependent.getClassLoader(Quiche.class);
if (!PlatformDependent.isAndroid()) {
libName += '_' + PlatformDependent.normalizedOs()
+ '_' + PlatformDependent.normalizedArch();
}
try {
NativeLibraryLoader.load(libName, cl);
} catch (UnsatisfiedLinkError e) {
logger.debug("Failed to load {}", libName, e);
throw e;
}
}
static final short AF_INET = (short) QuicheNativeStaticallyReferencedJniMethods.afInet();
static final short AF_INET6 = (short) QuicheNativeStaticallyReferencedJniMethods.afInet6();
static final int SIZEOF_SOCKADDR_STORAGE = QuicheNativeStaticallyReferencedJniMethods.sizeofSockaddrStorage();
static final int SIZEOF_SOCKADDR_IN = QuicheNativeStaticallyReferencedJniMethods.sizeofSockaddrIn();
static final int SIZEOF_SOCKADDR_IN6 = QuicheNativeStaticallyReferencedJniMethods.sizeofSockaddrIn6();
static final int SOCKADDR_IN_OFFSETOF_SIN_FAMILY =
QuicheNativeStaticallyReferencedJniMethods.sockaddrInOffsetofSinFamily();
static final int SOCKADDR_IN_OFFSETOF_SIN_PORT =
QuicheNativeStaticallyReferencedJniMethods.sockaddrInOffsetofSinPort();
static final int SOCKADDR_IN_OFFSETOF_SIN_ADDR =
QuicheNativeStaticallyReferencedJniMethods.sockaddrInOffsetofSinAddr();
static final int IN_ADDRESS_OFFSETOF_S_ADDR = QuicheNativeStaticallyReferencedJniMethods.inAddressOffsetofSAddr();
static final int SOCKADDR_IN6_OFFSETOF_SIN6_FAMILY =
QuicheNativeStaticallyReferencedJniMethods.sockaddrIn6OffsetofSin6Family();
static final int SOCKADDR_IN6_OFFSETOF_SIN6_PORT =
QuicheNativeStaticallyReferencedJniMethods.sockaddrIn6OffsetofSin6Port();
static final int SOCKADDR_IN6_OFFSETOF_SIN6_FLOWINFO =
QuicheNativeStaticallyReferencedJniMethods.sockaddrIn6OffsetofSin6Flowinfo();
static final int SOCKADDR_IN6_OFFSETOF_SIN6_ADDR =
QuicheNativeStaticallyReferencedJniMethods.sockaddrIn6OffsetofSin6Addr();
static final int SOCKADDR_IN6_OFFSETOF_SIN6_SCOPE_ID =
QuicheNativeStaticallyReferencedJniMethods.sockaddrIn6OffsetofSin6ScopeId();
static final int IN6_ADDRESS_OFFSETOF_S6_ADDR =
QuicheNativeStaticallyReferencedJniMethods.in6AddressOffsetofS6Addr();
static final int SIZEOF_SOCKLEN_T = QuicheNativeStaticallyReferencedJniMethods.sizeofSocklenT();
static final int SIZEOF_SIZE_T = QuicheNativeStaticallyReferencedJniMethods.sizeofSizeT();
static final int SIZEOF_TIMESPEC = QuicheNativeStaticallyReferencedJniMethods.sizeofTimespec();
static final int SIZEOF_TIME_T = QuicheNativeStaticallyReferencedJniMethods.sizeofTimeT();
static final int SIZEOF_LONG = QuicheNativeStaticallyReferencedJniMethods.sizeofLong();
static final int TIMESPEC_OFFSETOF_TV_SEC =
QuicheNativeStaticallyReferencedJniMethods.timespecOffsetofTvSec();
static final int TIMESPEC_OFFSETOF_TV_NSEC =
QuicheNativeStaticallyReferencedJniMethods.timespecOffsetofTvNsec();
static final int QUICHE_RECV_INFO_OFFSETOF_FROM =
QuicheNativeStaticallyReferencedJniMethods.quicheRecvInfoOffsetofFrom();
static final int QUICHE_RECV_INFO_OFFSETOF_FROM_LEN =
QuicheNativeStaticallyReferencedJniMethods.quicheRecvInfoOffsetofFromLen();
static final int QUICHE_RECV_INFO_OFFSETOF_TO =
QuicheNativeStaticallyReferencedJniMethods.quicheRecvInfoOffsetofTo();
static final int QUICHE_RECV_INFO_OFFSETOF_TO_LEN =
QuicheNativeStaticallyReferencedJniMethods.quicheRecvInfoOffsetofToLen();
static final int SIZEOF_QUICHE_RECV_INFO = QuicheNativeStaticallyReferencedJniMethods.sizeofQuicheRecvInfo();
static final int QUICHE_SEND_INFO_OFFSETOF_TO =
QuicheNativeStaticallyReferencedJniMethods.quicheSendInfoOffsetofTo();
static final int QUICHE_SEND_INFO_OFFSETOF_TO_LEN =
QuicheNativeStaticallyReferencedJniMethods.quicheSendInfoOffsetofToLen();
static final int QUICHE_SEND_INFO_OFFSETOF_FROM =
QuicheNativeStaticallyReferencedJniMethods.quicheSendInfoOffsetofFrom();
static final int QUICHE_SEND_INFO_OFFSETOF_FROM_LEN =
QuicheNativeStaticallyReferencedJniMethods.quicheSendInfoOffsetofFromLen();
static final int QUICHE_SEND_INFO_OFFSETOF_AT =
QuicheNativeStaticallyReferencedJniMethods.quicheSendInfoOffsetofAt();
static final int SIZEOF_QUICHE_SEND_INFO = QuicheNativeStaticallyReferencedJniMethods.sizeofQuicheSendInfo();
static final int QUICHE_PROTOCOL_VERSION = QuicheNativeStaticallyReferencedJniMethods.quiche_protocol_version();
static final int QUICHE_MAX_CONN_ID_LEN = QuicheNativeStaticallyReferencedJniMethods.quiche_max_conn_id_len();
/**
* See QUICHE_SHUTDOWN_READ.
*/
static final int QUICHE_SHUTDOWN_READ = QuicheNativeStaticallyReferencedJniMethods.quiche_shutdown_read();
/**
* See QUICHE_SHUTDOWN_WRITE.
*/
static final int QUICHE_SHUTDOWN_WRITE = QuicheNativeStaticallyReferencedJniMethods.quiche_shutdown_write();
/**
* See QUICHE_ERR_DONE.
*/
static final int QUICHE_ERR_DONE = QuicheNativeStaticallyReferencedJniMethods.quiche_err_done();
/**
* See
* QUICHE_ERR_BUFFER_TOO_SHORT.
*/
static final int QUICHE_ERR_BUFFER_TOO_SHORT =
QuicheNativeStaticallyReferencedJniMethods.quiche_err_buffer_too_short();
/**
* See
* QUICHE_ERR_UNKNOWN_VERSION.
*/
static final int QUICHE_ERR_UNKNOWN_VERSION =
QuicheNativeStaticallyReferencedJniMethods.quiche_err_unknown_version();
/**
* See
* QUICHE_ERR_INVALID_FRAME.
*/
static final int QUICHE_ERR_INVALID_FRAME = QuicheNativeStaticallyReferencedJniMethods.quiche_err_invalid_frame();
/**
* See
* QUICHE_ERR_INVALID_PACKET.
*/
static final int QUICHE_ERR_INVALID_PACKET = QuicheNativeStaticallyReferencedJniMethods.quiche_err_invalid_packet();
/**
* See
* QUICHE_ERR_INVALID_STATE.
*/
static final int QUICHE_ERR_INVALID_STATE = QuicheNativeStaticallyReferencedJniMethods.quiche_err_invalid_state();
/**
* See
* QUICHE_ERR_INVALID_STREAM_STATE.
*/
static final int QUICHE_ERR_INVALID_STREAM_STATE =
QuicheNativeStaticallyReferencedJniMethods.quiche_err_invalid_stream_state();
/**
* See
* QUICHE_ERR_INVALID_TRANSPORT_PARAM.
*/
static final int QUICHE_ERR_INVALID_TRANSPORT_PARAM =
QuicheNativeStaticallyReferencedJniMethods.quiche_err_invalid_transport_param();
/**
* See
* QUICHE_ERR_CRYPTO_FAIL.
*/
static final int QUICHE_ERR_CRYPTO_FAIL = QuicheNativeStaticallyReferencedJniMethods.quiche_err_crypto_fail();
/**
* See
* QUICHE_ERR_TLS_FAIL.
*/
static final int QUICHE_ERR_TLS_FAIL = QuicheNativeStaticallyReferencedJniMethods.quiche_err_tls_fail();
/**
* See
* QUICHE_ERR_FLOW_CONTROL.
*/
static final int QUICHE_ERR_FLOW_CONTROL = QuicheNativeStaticallyReferencedJniMethods.quiche_err_flow_control();
/**
* See
* QUICHE_ERR_STREAM_LIMIT.
*/
static final int QUICHE_ERR_STREAM_LIMIT = QuicheNativeStaticallyReferencedJniMethods.quiche_err_stream_limit();
/**
* See
* QUICHE_ERR_FINAL_SIZE.
*/
static final int QUICHE_ERR_FINAL_SIZE = QuicheNativeStaticallyReferencedJniMethods.quiche_err_final_size();
/**
* See
* QUICHE_ERR_CONGESTION_CONTROL.
*/
static final int QUICHE_ERR_CONGESTION_CONTROL =
QuicheNativeStaticallyReferencedJniMethods.quiche_err_congestion_control();
/**
* See QUICHE_ERR_STREAM_STOPPED.
*/
static final int QUICHE_ERR_STREAM_RESET =
QuicheNativeStaticallyReferencedJniMethods.quiche_err_stream_reset();
/**
* See
* QUICHE_ERR_STREAM_STOPPED.
*/
static final int QUICHE_ERR_STREAM_STOPPED =
QuicheNativeStaticallyReferencedJniMethods.quiche_err_stream_stopped();
// Too many identifiers were provided.
static final int QUICHE_ERR_ID_LIMIT =
QuicheNativeStaticallyReferencedJniMethods.quiche_err_id_limit();
// Not enough available identifiers.
static final int QUICHE_ERR_OUT_OF_IDENTIFIERS =
QuicheNativeStaticallyReferencedJniMethods.quiche_err_out_of_identifiers();
// Error in key update.
static final int QUICHE_ERR_KEY_UPDATE =
QuicheNativeStaticallyReferencedJniMethods.quiche_err_key_update();
static final int QUICHE_ERR_CRYPTO_BUFFER_EXCEEDED =
QuicheNativeStaticallyReferencedJniMethods.quiche_err_crypto_buffer_exceeded();
/**
* See
* QUICHE_CC_RENO.
*/
static final int QUICHE_CC_RENO = QuicheNativeStaticallyReferencedJniMethods.quiche_cc_reno();
/**
* See
* QUICHE_CC_CUBIC.
*/
static final int QUICHE_CC_CUBIC = QuicheNativeStaticallyReferencedJniMethods.quiche_cc_cubic();
/**
* See
* QUICHE_CC_BBR.
*/
static final int QUICHE_CC_BBR = QuicheNativeStaticallyReferencedJniMethods.quiche_cc_bbr();
static final int QUICHE_PATH_EVENT_NEW = QuicheNativeStaticallyReferencedJniMethods.quiche_path_event_new();
static final int QUICHE_PATH_EVENT_VALIDATED = QuicheNativeStaticallyReferencedJniMethods.quiche_path_event_validated();
static final int QUICHE_PATH_EVENT_FAILED_VALIDATION = QuicheNativeStaticallyReferencedJniMethods.quiche_path_event_failed_validation();
static final int QUICHE_PATH_EVENT_CLOSED = QuicheNativeStaticallyReferencedJniMethods.quiche_path_event_closed();
static final int QUICHE_PATH_EVENT_REUSED_SOURCE_CONNECTION_ID = QuicheNativeStaticallyReferencedJniMethods.quiche_path_event_reused_source_connection_id();
static final int QUICHE_PATH_EVENT_PEER_MIGRATED = QuicheNativeStaticallyReferencedJniMethods.quiche_path_event_peer_migrated();
/**
* See quiche_version.
*/
@Nullable
static native String quiche_version();
/**
* See
* quiche_version_is_supported.
*/
static native boolean quiche_version_is_supported(int version);
/**
* See quiche_header_info.
*/
static native int quiche_header_info(long bufAddr, int bufLength, int dcil, long versionAddr, long typeAddr,
long scidAddr, long scidLenAddr, long dcidAddr, long dcidLenAddr,
long tokenAddr, long tokenLenAddr);
/**
* See quiche_negotiate_version.
*/
static native int quiche_negotiate_version(
long scidAddr, int scidLen, long dcidAddr, int dcidLen, long outAddr, int outLen);
/**
* See quiche_retry.
*/
static native int quiche_retry(long scidAddr, int scidLen, long dcidAddr, int dcidLen, long newScidAddr,
int newScidLen, long tokenAddr, int tokenLen, int version, long outAddr, int outLen);
/**
* See quiche_conn_new_with_tls.
*/
static native long quiche_conn_new_with_tls(long scidAddr, int scidLen, long odcidAddr, int odcidLen,
long localAddr, int localLen,
long peerAddr, int peerLen,
long configAddr, long ssl, boolean isServer);
/**
* See
* quiche_conn_set_qlog_path.
*/
static native boolean quiche_conn_set_qlog_path(long connAddr, String path, String logTitle, String logDescription);
/**
* See quiche_conn_recv.
*/
static native int quiche_conn_recv(long connAddr, long bufAddr, int bufLen, long infoAddr);
/**
* See quiche_conn_send.
*/
static native int quiche_conn_send(long connAddr, long outAddr, int outLen, long infoAddr);
/**
* See quiche_conn_free.
*/
static native void quiche_conn_free(long connAddr);
@Nullable
static QuicConnectionCloseEvent quiche_conn_peer_error(long connAddr) {
Object[] error = quiche_conn_peer_error0(connAddr);
if (error == null) {
return null;
}
return new QuicConnectionCloseEvent((Boolean) error[0], (Integer) error[1], (byte[]) error[2]);
}
private static native Object @Nullable [] quiche_conn_peer_error0(long connAddr);
/**
* See
* quiche_conn_peer_streams_left_bidi.
*/
static native long quiche_conn_peer_streams_left_bidi(long connAddr);
/**
* See
* quiche_conn_peer_streams_left_uni.
*/
static native long quiche_conn_peer_streams_left_uni(long connAddr);
/**
* See
* quiche_conn_stream_priority.
*/
static native int quiche_conn_stream_priority(
long connAddr, long streamId, byte urgency, boolean incremental);
static native int quiche_conn_send_quantum(long connAddr);
/**
* See quiche_conn_trace_id.
*/
static native byte @Nullable [] quiche_conn_trace_id(long connAddr);
static native byte[] quiche_conn_source_id(long connAddr);
static native byte[] quiche_conn_destination_id(long connAddr);
/**
* See quiche_conn_stream_recv.
*/
static native int quiche_conn_stream_recv(long connAddr, long streamId, long outAddr, int bufLen, long finAddr);
/**
* See quiche_conn_stream_send.
*/
static native int quiche_conn_stream_send(long connAddr, long streamId, long bufAddr, int bufLen, boolean fin);
/**
* See
* quiche_conn_stream_shutdown.
*/
static native int quiche_conn_stream_shutdown(long connAddr, long streamId, int direction, long err);
/**
* See
* quiche_conn_stream_capacity.
*/
static native int quiche_conn_stream_capacity(long connAddr, long streamId);
/**
* See
* quiche_conn_stream_finished.
*/
static native boolean quiche_conn_stream_finished(long connAddr, long streamId);
/**
* See
* quiche_conn_close.
*/
static native int quiche_conn_close(long connAddr, boolean app, long err, long reasonAddr, int reasonLen);
/**
* See
* quiche_conn_is_established.
*/
static native boolean quiche_conn_is_established(long connAddr);
/**
* See
* quiche_conn_is_in_early_data.
*/
static native boolean quiche_conn_is_in_early_data(long connAddr);
/**
* See
* quiche_conn_is_closed.
*/
static native boolean quiche_conn_is_closed(long connAddr);
/**
* See
* quiche_conn_is_timed_out.
*/
static native boolean quiche_conn_is_timed_out(long connAddr);
/**
* See
* quiche_conn_stats.
* The implementation relies on all fields of
* quiche_stats being numerical.
* The assumption made allows passing primitive array rather than dealing with objects.
*/
static native long @Nullable [] quiche_conn_stats(long connAddr);
/**
* See
*
* quiche_conn_stats.
*/
static native long @Nullable [] quiche_conn_peer_transport_params(long connAddr);
/**
* See
* quiche_conn_timeout_as_nanos.
*/
static native long quiche_conn_timeout_as_nanos(long connAddr);
/**
* See
* quiche_conn_on_timeout.
*/
static native void quiche_conn_on_timeout(long connAddr);
/**
* See
* quiche_conn_readable.
*/
static native long quiche_conn_readable(long connAddr);
/**
* See
* quiche_conn_writable.
*/
static native long quiche_conn_writable(long connAddr);
/**
* See
* quiche_stream_iter_next.
*
* This method will fill the {@code streamIds} array and return the number of streams that were filled into
* the array. If the number is the same as the length of the array you should call it again until it returns
* less to ensure you process all the streams later on.
*/
static native int quiche_stream_iter_next(long iterAddr, long[] streamIds);
/**
* See
* quiche_stream_iter_free.
*
*/
static native void quiche_stream_iter_free(long iterAddr);
/**
* See
* quiche_conn_path_stats.
*/
static native Object @Nullable [] quiche_conn_path_stats(long connAddr, long streamIdx);
/**
* See
*
* quiche_conn_dgram_max_writable_len.
*/
static native int quiche_conn_dgram_max_writable_len(long connAddr);
/**
* See
*
* quiche_conn_dgram_recv_front_len.
*/
static native int quiche_conn_dgram_recv_front_len(long connAddr);
/**
* See
*
* quiche_conn_dgram_recv.
*/
static native int quiche_conn_dgram_recv(long connAddr, long buf, int size);
/**
* See
*
* quiche_conn_dgram_send.
*/
static native int quiche_conn_dgram_send(long connAddr, long buf, int size);
/**
* See
*
* quiche_conn_set_session.
*/
static native int quiche_conn_set_session(long connAddr, byte[] sessionBytes);
/**
* See
*
* quiche_conn_max_send_udp_payload_size.
*/
static native int quiche_conn_max_send_udp_payload_size(long connAddr);
static native int quiche_conn_scids_left(long connAddr);
static native long quiche_conn_new_scid(long connAddr, long scidAddr, int scidLen, byte[] resetToken, boolean retire_if_needed, long seq);
static native byte @Nullable [] quiche_conn_retired_scid_next(long connAddr);
static native long quiche_conn_path_event_next(long connAddr);
static native int quiche_path_event_type(long pathEvent);
static native void quiche_path_event_free(long pathEvent);
static native Object[] quiche_path_event_new(long pathEvent);
static native Object[] quiche_path_event_validated(long pathEvent);
static native Object[] quiche_path_event_failed_validation(long pathEvent);
static native Object[] quiche_path_event_closed(long pathEvent);
static native Object[] quiche_path_event_reused_source_connection_id(long pathEvent);
static native Object[] quiche_path_event_peer_migrated(long pathEvent);
/**
* See
* quiche_config_new.
*/
static native long quiche_config_new(int version);
/**
* See
*
* quiche_config_grease.
*/
static native void quiche_config_grease(long configAddr, boolean value);
/**
* See
*
* quiche_config_set_max_idle_timeout.
*/
static native void quiche_config_set_max_idle_timeout(long configAddr, long value);
/**
* See
*
* quiche_config_set_max_recv_udp_payload_size.
*/
static native void quiche_config_set_max_recv_udp_payload_size(long configAddr, long value);
/**
* See
*
* quiche_config_set_max_recv_udp_payload_size.
*/
static native void quiche_config_set_max_send_udp_payload_size(long configAddr, long value);
/**
* See
*
* quiche_config_set_initial_max_data.
*/
static native void quiche_config_set_initial_max_data(long configAddr, long value);
/**
* See
*
* quiche_config_set_initial_max_stream_data_bidi_local.
*/
static native void quiche_config_set_initial_max_stream_data_bidi_local(long configAddr, long value);
/**
* See
*
* quiche_config_set_initial_max_stream_data_bidi_remote.
*/
static native void quiche_config_set_initial_max_stream_data_bidi_remote(long configAddr, long value);
/**
* See
*
* quiche_config_set_initial_max_stream_data_uni.
*/
static native void quiche_config_set_initial_max_stream_data_uni(long configAddr, long value);
/**
* See
*
* quiche_config_set_initial_max_streams_bidi.
*/
static native void quiche_config_set_initial_max_streams_bidi(long configAddr, long value);
/**
* See
*
* quiche_config_set_initial_max_streams_uni.
*/
static native void quiche_config_set_initial_max_streams_uni(long configAddr, long value);
/**
* See
*
* quiche_config_set_ack_delay_exponent.
*/
static native void quiche_config_set_ack_delay_exponent(long configAddr, long value);
/**
* See
*
* quiche_config_set_max_ack_delay.
*/
static native void quiche_config_set_max_ack_delay(long configAddr, long value);
/**
* See
*
* quiche_config_set_disable_active_migration.
*/
static native void quiche_config_set_disable_active_migration(long configAddr, boolean value);
/**
* See
*
* quiche_config_set_cc_algorithm.
*/
static native void quiche_config_set_cc_algorithm(long configAddr, int algo);
/**
* See
*
* quiche_config_set_initial_congestion_window_packets
*
*/
static native void quiche_config_set_initial_congestion_window_packets(long configAddr, int numPackets);
/**
* See
*
* quiche_config_enable_hystart.
*/
static native void quiche_config_enable_hystart(long configAddr, boolean value);
/**
* See
*
* quiche_config_enable_dgram.
*/
static native void quiche_config_enable_dgram(long configAddr, boolean enable,
int recv_queue_len, int send_queue_len);
// Sets the limit of active connection IDs.
static native void quiche_config_set_active_connection_id_limit(long configAddr, long value);
// Sets the initial stateless reset token.
static native void quiche_config_set_stateless_reset_token(long configAddr, byte[] token);
/**
* See
*
* quiche_config_free.
*/
static native void quiche_config_free(long configAddr);
/**
* See
* quiche_config_new.
*/
private static native void quiche_enable_debug_logging(QuicheLogger logger);
private static native long buffer_memory_address(ByteBuffer buffer);
static native int sockaddr_cmp(long addr, long addr2);
/**
* Returns the memory address if the {@link ByteBuf} taking the readerIndex into account.
*
* @param buf the {@link ByteBuf} of which we want to obtain the memory address
* (taking its {@link ByteBuf#readerIndex()} into account).
* @return the memory address of this {@link ByteBuf}s readerIndex.
*/
static long readerMemoryAddress(ByteBuf buf) {
return memoryAddress(buf, buf.readerIndex(), buf.readableBytes());
}
/**
* Returns the memory address if the {@link ByteBuf} taking the writerIndex into account.
*
* @param buf the {@link ByteBuf} of which we want to obtain the memory address
* (taking its {@link ByteBuf#writerIndex()} into account).
* @return the memory address of this {@link ByteBuf}s writerIndex.
*/
static long writerMemoryAddress(ByteBuf buf) {
return memoryAddress(buf, buf.writerIndex(), buf.writableBytes());
}
/**
* Returns the memory address if the {@link ByteBuf} taking the offset into account.
*
* @param buf the {@link ByteBuf} of which we want to obtain the memory address
* (taking the {@code offset} into account).
* @param offset the offset of the memory address.
* @param len the length of the {@link ByteBuf}.
* @return the memory address of this {@link ByteBuf}s offset.
*/
static long memoryAddress(ByteBuf buf, int offset, int len) {
assert buf.isDirect();
if (buf.hasMemoryAddress()) {
return buf.memoryAddress() + offset;
}
return memoryAddressWithPosition(buf.internalNioBuffer(offset, len));
}
/**
* Returns the memory address of the given {@link ByteBuffer} taking its current {@link ByteBuffer#position()} into
* account.
*
* @param buf the {@link ByteBuffer} of which we want to obtain the memory address
* (taking its {@link ByteBuffer#position()} into account).
* @return the memory address of this {@link ByteBuffer}s position.
*/
static long memoryAddressWithPosition(ByteBuffer buf) {
assert buf.isDirect();
return buffer_memory_address(buf) + buf.position();
}
@SuppressWarnings("deprecation")
static ByteBuf allocateNativeOrder(int capacity) {
// Just use Unpooled as the life-time of these buffers is long.
ByteBuf buffer = Unpooled.directBuffer(capacity);
// As we use the buffers as pointers to int etc we need to ensure we use the right oder so we will
// see the right value when we read primitive values.
return PlatformDependent.BIG_ENDIAN_NATIVE_ORDER ? buffer : buffer.order(ByteOrder.LITTLE_ENDIAN);
}
static boolean shouldClose(int res) {
return res == Quiche.QUICHE_ERR_CRYPTO_FAIL || res == Quiche.QUICHE_ERR_TLS_FAIL;
}
/**
* Returns {@code true} if both {@link ByteBuffer}s have the same {@code sock_addr} stored.
*
* @param memory the first {@link ByteBuffer} which holds a {@code quiche_recv_info}.
* @param memory2 the second {@link ByteBuffer} which holds a {@code quiche_recv_info}.
* @return {@code true} if both {@link ByteBuffer}s have the same {@code sock_addr} stored, {@code false}
* otherwise.
*/
static boolean isSameAddress(ByteBuffer memory, ByteBuffer memory2, int addressOffset) {
long address1 = Quiche.memoryAddressWithPosition(memory) + addressOffset;
long address2 = Quiche.memoryAddressWithPosition(memory2) + addressOffset;
return SockaddrIn.cmp(address1, address2) == 0;
}
static void setPrimitiveValue(ByteBuffer memory, int offset, int valueType, long value) {
switch (valueType) {
case 1:
memory.put(offset, (byte) value);
break;
case 2:
memory.putShort(offset, (short) value);
break;
case 4:
memory.putInt(offset, (int) value);
break;
case 8:
memory.putLong(offset, value);
break;
default:
throw new IllegalStateException();
}
}
static long getPrimitiveValue(ByteBuffer memory, int offset, int valueType) {
switch (valueType) {
case 1:
return memory.get(offset);
case 2:
return memory.getShort(offset);
case 4:
return memory.getInt(offset);
case 8:
return memory.getLong(offset);
default:
throw new IllegalStateException();
}
}
// See https://github.com/cloudflare/quiche/commit/1d00ee1bb2256dfd99ba0cb2dfac72fe1e59407f
static {
ERROR_MAPPINGS.put(QUICHE_ERR_DONE,
new QuicTransportErrorHolder(QuicTransportError.NO_ERROR, "QUICHE_ERR_DONE"));
ERROR_MAPPINGS.put(QUICHE_ERR_INVALID_FRAME,
new QuicTransportErrorHolder(QuicTransportError.FRAME_ENCODING_ERROR, "QUICHE_ERR_INVALID_FRAME"));
ERROR_MAPPINGS.put(QUICHE_ERR_INVALID_STREAM_STATE,
new QuicTransportErrorHolder(QuicTransportError.STREAM_STATE_ERROR, "QUICHE_ERR_INVALID_STREAM_STATE"));
ERROR_MAPPINGS.put(QUICHE_ERR_INVALID_TRANSPORT_PARAM,
new QuicTransportErrorHolder(QuicTransportError.TRANSPORT_PARAMETER_ERROR, "QUICHE_ERR_INVALID_TRANSPORT_PARAM"));
ERROR_MAPPINGS.put(QUICHE_ERR_FLOW_CONTROL,
new QuicTransportErrorHolder(QuicTransportError.FLOW_CONTROL_ERROR, "QUICHE_ERR_FLOW_CONTROL"));
ERROR_MAPPINGS.put(QUICHE_ERR_STREAM_LIMIT,
new QuicTransportErrorHolder(QuicTransportError.STREAM_LIMIT_ERROR, "QUICHE_ERR_STREAM_LIMIT"));
ERROR_MAPPINGS.put(QUICHE_ERR_ID_LIMIT,
new QuicTransportErrorHolder(QuicTransportError.CONNECTION_ID_LIMIT_ERROR, "QUICHE_ERR_ID_LIMIT"));
ERROR_MAPPINGS.put(QUICHE_ERR_FINAL_SIZE,
new QuicTransportErrorHolder(QuicTransportError.FINAL_SIZE_ERROR, "QUICHE_ERR_FINAL_SIZE"));
ERROR_MAPPINGS.put(QUICHE_ERR_CRYPTO_BUFFER_EXCEEDED,
new QuicTransportErrorHolder(QuicTransportError.CRYPTO_BUFFER_EXCEEDED, "QUICHE_ERR_CRYPTO_BUFFER_EXCEEDED"));
ERROR_MAPPINGS.put(QUICHE_ERR_KEY_UPDATE,
new QuicTransportErrorHolder(QuicTransportError.KEY_UPDATE_ERROR, "QUICHE_ERR_KEY_UPDATE"));
// Should the code be something different ?
ERROR_MAPPINGS.put(QUICHE_ERR_TLS_FAIL,
new QuicTransportErrorHolder(QuicTransportError.valueOf(0x0100), "QUICHE_ERR_TLS_FAIL"));
ERROR_MAPPINGS.put(QUICHE_ERR_CRYPTO_FAIL,
new QuicTransportErrorHolder(QuicTransportError.valueOf(0x0100), "QUICHE_ERR_CRYPTO_FAIL"));
ERROR_MAPPINGS.put(QUICHE_ERR_BUFFER_TOO_SHORT,
new QuicTransportErrorHolder(QuicTransportError.PROTOCOL_VIOLATION, "QUICHE_ERR_BUFFER_TOO_SHORT"));
ERROR_MAPPINGS.put(QUICHE_ERR_UNKNOWN_VERSION, new QuicTransportErrorHolder(QuicTransportError.PROTOCOL_VIOLATION, "QUICHE_ERR_UNKNOWN_VERSION"));
ERROR_MAPPINGS.put(QUICHE_ERR_INVALID_PACKET, new QuicTransportErrorHolder(QuicTransportError.PROTOCOL_VIOLATION, "QUICHE_ERR_INVALID_PACKET"));
ERROR_MAPPINGS.put(QUICHE_ERR_INVALID_STATE, new QuicTransportErrorHolder(QuicTransportError.PROTOCOL_VIOLATION, "QUICHE_ERR_INVALID_STATE"));
ERROR_MAPPINGS.put(QUICHE_ERR_CONGESTION_CONTROL, new QuicTransportErrorHolder(QuicTransportError.PROTOCOL_VIOLATION, "QUICHE_ERR_CONGESTION_CONTROL"));
ERROR_MAPPINGS.put(QUICHE_ERR_STREAM_STOPPED, new QuicTransportErrorHolder(QuicTransportError.PROTOCOL_VIOLATION, "QUICHE_ERR_STREAM_STOPPED"));
ERROR_MAPPINGS.put(QUICHE_ERR_OUT_OF_IDENTIFIERS, new QuicTransportErrorHolder(QuicTransportError.PROTOCOL_VIOLATION, "QUICHE_ERR_OUT_OF_IDENTIFIERS"));
}
static Exception convertToException(int result) {
QuicTransportErrorHolder holder = ERROR_MAPPINGS.get(result);
if (holder == null) {
// There is no mapping to a transport error, it's something internal so throw it directly.
return new QuicException(QuicheError.valueOf(result).message());
}
Exception exception = new QuicException(holder.error + ": " + holder.quicheErrorName, holder.error);
if (result == QUICHE_ERR_TLS_FAIL) {
String lastSslError = BoringSSL.ERR_last_error();
final SSLHandshakeException sslExc = new SSLHandshakeException(lastSslError);
sslExc.initCause(exception);
return sslExc;
}
if (result == QUICHE_ERR_CRYPTO_FAIL) {
return new SSLException(exception);
}
return exception;
}
private static final class QuicTransportErrorHolder {
private final QuicTransportError error;
private final String quicheErrorName;
QuicTransportErrorHolder(QuicTransportError error, String quicheErrorName) {
this.error = error;
this.quicheErrorName = quicheErrorName;
}
}
private Quiche() { }
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy