Skip to content

Commit

Permalink
Merge pull request #38 from nwf/202410-tls_precise_rtos
Browse files Browse the repository at this point in the history
lib/tls: use RTOS bounds().set_inexact_up_to()
  • Loading branch information
nwf authored Oct 10, 2024
2 parents 7b503dd + c27a8e5 commit f26cb4d
Showing 1 changed file with 4 additions and 46 deletions.
50 changes: 4 additions & 46 deletions lib/tls/tls.cc
Original file line number Diff line number Diff line change
Expand Up @@ -189,56 +189,13 @@ namespace
br_ssl_engine_set_default_aes_gcm(&cc->eng);
}

/**
* Set the bounds on `buffer` to `length` if `length` is representable with
* the current alignment of `buffer`. If not, then reduce `length` until it
* is representable. This ensures that we call the TCP/IP stack with a
* precisely bounded buffer, rather than one that is slightly larger,
* avoiding potential corruption or information leaks if the TCP/IP stack
* is compromised, at the expense of more cross-compartment calls.
*/
void precisely_bound_buffer(Capability<uint8_t> &buffer, size_t &length)
{
// Make sure that the bounds are precisely representable by doing a
// smaller copy.
ptraddr_t alignmentMask = representable_alignment_mask(length);
ptraddr_t baseAddress =
static_cast<ptraddr_t>(reinterpret_cast<uintptr_t>(buffer.get()));
if ((baseAddress & alignmentMask) != baseAddress)
{
Debug::log("Buffer is not precisely representable. Base: {}, "
"alignment mask: {}, length: {}",
baseAddress,
alignmentMask,
length);
// The number of bits in the capability encoding mantissa bits.
// This is part of the capability encoding and so, ideally,
// wouldn't be hard coded here.
static constexpr size_t MantissaBits = 9;
// Shrink the length to the largest representable length that we
// can fit.
length = std::min<size_t>(
length, ((1 << MantissaBits) - 1) << __builtin_ctz(baseAddress));
Debug::log("Reducing length to {}", length);
Debug::Assert(
[=]() {
return (baseAddress & representable_alignment_mask(length)) ==
baseAddress;
},
"Reduced buffer length is not representable. Length: {}, base: "
"{}",
length,
baseAddress);
}
buffer.bounds() = length;
}

int receive_records(Timeout *t, TLSContext *connection)
{
auto *engine = &connection->clientContext->eng;
size_t length;
Capability inputBuffer = br_ssl_engine_recvrec_buf(engine, &length);
precisely_bound_buffer(inputBuffer, length);
inputBuffer.bounds().set_inexact_at_most(length);
length = inputBuffer.length();

// Remove local so that the network stack cannot capture
// this, remove load so that we cannot leak state.
Expand Down Expand Up @@ -275,7 +232,8 @@ namespace
size_t readyLength;
Capability readyBuffer =
br_ssl_engine_sendrec_buf(engine, &readyLength);
precisely_bound_buffer(readyBuffer, readyLength);
readyBuffer.bounds().set_inexact_at_most(readyLength);
readyLength = readyBuffer.length();

// Remove local so that the network stack cannot capture
// this, remove store so that we cannot leak state.
Expand Down

0 comments on commit f26cb4d

Please sign in to comment.