blob: 623da92494c5a437313294962483b5400abb22ee [file] [log] [blame]
// Copyright 2015 The Chromium Authors
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#include "parser.h"
#include "parse_values.h"
#include <openssl/base.h>
namespace bssl::der {
Parser::Parser() {
CBS_init(&cbs_, nullptr, 0);
}
Parser::Parser(const Input& input) {
CBS_init(&cbs_, input.UnsafeData(), input.Length());
}
bool Parser::PeekTagAndValue(Tag* tag, Input* out) {
CBS peeker = cbs_;
CBS tmp_out;
size_t header_len;
unsigned tag_value;
if (!CBS_get_any_asn1_element(&peeker, &tmp_out, &tag_value, &header_len) ||
!CBS_skip(&tmp_out, header_len)) {
return false;
}
advance_len_ = CBS_len(&tmp_out) + header_len;
*tag = tag_value;
*out = Input(CBS_data(&tmp_out), CBS_len(&tmp_out));
return true;
}
bool Parser::Advance() {
if (advance_len_ == 0)
return false;
bool ret = !!CBS_skip(&cbs_, advance_len_);
advance_len_ = 0;
return ret;
}
bool Parser::HasMore() {
return CBS_len(&cbs_) > 0;
}
bool Parser::ReadRawTLV(Input* out) {
CBS tmp_out;
if (!CBS_get_any_asn1_element(&cbs_, &tmp_out, nullptr, nullptr))
return false;
*out = Input(CBS_data(&tmp_out), CBS_len(&tmp_out));
return true;
}
bool Parser::ReadTagAndValue(Tag* tag, Input* out) {
if (!PeekTagAndValue(tag, out))
return false;
BSSL_CHECK(Advance());
return true;
}
bool Parser::ReadOptionalTag(Tag tag, std::optional<Input>* out) {
if (!HasMore()) {
*out = std::nullopt;
return true;
}
Tag actual_tag;
Input value;
if (!PeekTagAndValue(&actual_tag, &value)) {
return false;
}
if (actual_tag == tag) {
BSSL_CHECK(Advance());
*out = value;
} else {
advance_len_ = 0;
*out = std::nullopt;
}
return true;
}
bool Parser::ReadOptionalTag(Tag tag, Input* out, bool* present) {
std::optional<Input> tmp_out;
if (!ReadOptionalTag(tag, &tmp_out))
return false;
*present = tmp_out.has_value();
*out = tmp_out.value_or(der::Input());
return true;
}
bool Parser::SkipOptionalTag(Tag tag, bool* present) {
Input out;
return ReadOptionalTag(tag, &out, present);
}
bool Parser::ReadTag(Tag tag, Input* out) {
Tag actual_tag;
Input value;
if (!PeekTagAndValue(&actual_tag, &value) || actual_tag != tag) {
return false;
}
BSSL_CHECK(Advance());
*out = value;
return true;
}
bool Parser::SkipTag(Tag tag) {
Input out;
return ReadTag(tag, &out);
}
// Type-specific variants of ReadTag
bool Parser::ReadConstructed(Tag tag, Parser* out) {
if (!IsConstructed(tag))
return false;
Input data;
if (!ReadTag(tag, &data))
return false;
*out = Parser(data);
return true;
}
bool Parser::ReadSequence(Parser* out) {
return ReadConstructed(kSequence, out);
}
bool Parser::ReadUint8(uint8_t* out) {
Input encoded_int;
if (!ReadTag(kInteger, &encoded_int))
return false;
return ParseUint8(encoded_int, out);
}
bool Parser::ReadUint64(uint64_t* out) {
Input encoded_int;
if (!ReadTag(kInteger, &encoded_int))
return false;
return ParseUint64(encoded_int, out);
}
std::optional<BitString> Parser::ReadBitString() {
Input value;
if (!ReadTag(kBitString, &value))
return std::nullopt;
return ParseBitString(value);
}
bool Parser::ReadGeneralizedTime(GeneralizedTime* out) {
Input value;
if (!ReadTag(kGeneralizedTime, &value))
return false;
return ParseGeneralizedTime(value, out);
}
} // namespace bssl::der