blob: f5dcf44b885c910b02280f46204c18a9ca91ee1b [file] [log] [blame]
// Copyright 2025 The BoringSSL Authors
//
// 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
//
// 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.
#include "der_trailing_data.h"
#include <optional>
#include <openssl/bytestring.h>
static bool RewriteWithTrailingData(CBB *cbb, CBS *cbs,
std::optional<size_t> *rewrite_counter) {
CBS contents;
CBS_ASN1_TAG tag;
if (!CBS_get_any_asn1(cbs, &contents, &tag)) {
return false;
}
if (!rewrite_counter->has_value() || (tag & CBS_ASN1_CONSTRUCTED) == 0) {
return CBB_add_asn1_element(cbb, tag, CBS_data(&contents),
CBS_len(&contents));
}
CBB child;
if (!CBB_add_asn1(cbb, &child, tag)) {
return false;
}
if (rewrite_counter->value() == 0) {
*rewrite_counter = std::nullopt;
return CBB_add_bytes(&child, CBS_data(&contents), CBS_len(&contents)) &&
// Add a BER EOC, which is always invalid in DER.
CBB_add_u8(&child, 0) && //
CBB_add_u8(&child, 0) && //
CBB_flush(cbb);
}
*rewrite_counter = rewrite_counter->value() - 1;
while (CBS_len(&contents) != 0) {
if (!RewriteWithTrailingData(&child, &contents, rewrite_counter)) {
return false;
}
}
return CBB_flush(cbb);
}
bool TestDERTrailingData(
bssl::Span<const uint8_t> in,
std::function<void(bssl::Span<const uint8_t>, size_t)> func) {
for (size_t elem_to_rewrite = 0; true; elem_to_rewrite++) {
std::optional<size_t> rewrite_counter = elem_to_rewrite;
CBS cbs = in;
bssl::ScopedCBB cbb;
if (!CBB_init(cbb.get(), in.size() + /* EOC */ 2 +
/* in case lengths get larger */ 8) ||
!RewriteWithTrailingData(cbb.get(), &cbs, &rewrite_counter) ||
CBS_len(&cbs) != 0) {
return false;
}
// We have exhausted every constructed element.
if (rewrite_counter.has_value()) {
return true;
}
func(bssl::Span(CBB_data(cbb.get()), CBB_len(cbb.get())), elem_to_rewrite);
}
}