Skip to content

Commit

Permalink
ktls: release APIs as unstable
Browse files Browse the repository at this point in the history
  • Loading branch information
lrstewart committed Sep 21, 2023
1 parent 3758f4b commit 808dfd4
Show file tree
Hide file tree
Showing 3 changed files with 107 additions and 7 deletions.
105 changes: 105 additions & 0 deletions api/unstable/ktls.h
Original file line number Diff line number Diff line change
@@ -0,0 +1,105 @@
/*
* Copyright Amazon.com, Inc. or its affiliates. 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.
* A copy of the License is located at
*
* http://aws.amazon.com/apache2.0
*
* or in the "license" file accompanying this file. This file 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.
*/

#pragma once

#include <s2n.h>

/**
* @file ktls.h
*
* The following APIs enable applications to use kernel TLS (kTLS), meaning that
* enrypting and decrypting TLS records is handled by the kernel rather than by
* the s2n-tls library.
*
* The kTLS APIs are currently considered unstable. kTLS is a relatively new
* feature with limited and volatile support from different kernels and hardware.
*
* Currently, s2n-tls supports ktls for only very limited scenarios:
* - You must be using Linux. We have not tested with other kernels.
* - Your kernel must support kTLS. For Linux, versions >4.13 should support kTLS.
* - You must negotiate TLS1.2. TLS1.3 support is blocked on kernel support for
* TLS KeyUpdate messages.
* - You must negotiate AES128-GCM. Other ciphers will be supported soon.
*/

/**
* Enables sending using kTLS on a given connection.
*
* See above for the limitations on when kTLS can be enabled.
*
* This method must be called after the handshake completes. It may be called
* after some application data is sent and received without kTLS, but there must
* be no pending application data that requires flushing.
*
* After successfully calling this method, s2n_send, s2n_sendv, and s2n_sendv_with_offset
* will use kTLS. kTLS should result in memory and CPU savings. s2n_sendfile will
* also become available.
*
* @warning For applications using kTLS to avoid copying or allocating memory,
* s2n_sendv should be preferred over s2n_sendv_with_offset. For s2n_sendv_with_offset,
* s2n-tls may need to copy the provided iovec array to apply the offset, and may
* need to allocate memory to copy large (>16) iovec arrays.
*
* @warning Due to the uncertainty around kTLS support, the signature of this
* method is likely to change before kTLS is marked as stable.
*
* @param conn A pointer to the connection.
* @returns S2N_SUCCESS if kTLS is successfully enabled. If kTlS is not successfully
* enabled, returns S2N_FAILURE but the connection may proceed without kTLS.
*/
S2N_API int s2n_connection_ktls_enable_send(struct s2n_connection *conn);

/**
* Enables receiving using kTLS on a given connection.
*
* See above for the limitations on when kTLS can be enabled.
*
* This method must be called after the handshake completes. It may be called
* after some application data is sent and received without kTLS, but there must
* be no buffered application data that requires draining.
*
* After calling this method, s2n_recv will use kTLS. This may result in memory
* and CPU savings, but currently will still buffer and copy application data.
* We will further optimize s2n_recv for kTLS in the future.
*
* @warning Due to the uncertainty around kTLS support, the signature of this
* method is likely to change before kTLS is marked as stable.
*
* @param conn A pointer to the connection.
* @returns S2N_SUCCESS if kTLS is successfully enabled. If kTlS is not successfully
* enabled, returns S2N_FAILURE but the connection may proceed without kTLS.
*/
S2N_API int s2n_connection_ktls_enable_recv(struct s2n_connection *conn);

/**
* Sends the contents of a file as application data.
*
* s2n_sendfile should be more efficient than s2n_send because the copy between
* the file and the write socket happens inside the kernel.
*
* This method is only supported if kTLS is enabled for sending.
*
* @param conn A pointer to the connection.
* @param fd The file descriptor to read from. It must be opened for reading and
* support mmap-like operations (i.e., it cannot be a socket).
* @param offset The offset in the file to begin reading at.
* @param count The maximum number of bytes to read from the file.
* @param bytes_written Will be set to the number of bytes written if successful.
* @param blocked Will be set to the blocked status if an `S2N_ERR_T_BLOCKED` error is returned.
* @returns S2N_SUCCESS if any bytes are successfully written, S2N_FAILURE otherwise.
*/
S2N_API int s2n_sendfile(struct s2n_connection *conn, int fd, off_t offset, size_t count,
size_t *bytes_written, s2n_blocked_status *blocked);
2 changes: 1 addition & 1 deletion bindings/rust/generate/src/main.rs
Original file line number Diff line number Diff line change
Expand Up @@ -124,7 +124,7 @@ const COPYRIGHT: &str = r#"
const PRELUDE: &str = r#"
#![allow(unused_imports, non_camel_case_types)]
use libc::{iovec, FILE};
use libc::{iovec, FILE, off_t};
"#;

fn base_builder() -> bindgen::Builder {
Expand Down
7 changes: 1 addition & 6 deletions tls/s2n_ktls.h
Original file line number Diff line number Diff line change
Expand Up @@ -17,6 +17,7 @@

#include <sys/socket.h>

#include "api/unstable/ktls.h"
#include "tls/s2n_connection.h"
/* Define headers needed to enable and use kTLS.
*
Expand Down Expand Up @@ -51,12 +52,6 @@ int s2n_ktls_record_writev(struct s2n_connection *conn, uint8_t content_type,
const struct iovec *in, int in_count, size_t offs, size_t to_write);
int s2n_ktls_read_full_record(struct s2n_connection *conn, uint8_t *record_type);

/* These functions will be part of the public API. */
int s2n_connection_ktls_enable_send(struct s2n_connection *conn);
int s2n_connection_ktls_enable_recv(struct s2n_connection *conn);
int s2n_sendfile(struct s2n_connection *conn, int in_fd, off_t offset, size_t count,
size_t *bytes_written, s2n_blocked_status *blocked);

/* Testing */
typedef int (*s2n_setsockopt_fn)(int socket, int level, int option_name, const void *option_value,
socklen_t option_len);
Expand Down

0 comments on commit 808dfd4

Please sign in to comment.