/* * Copyright (c) 2021, the SerenityOS developers. * All rights reserved. * * Redistribution and use in source and binary forms, with or without * modification, are permitted provided that the following conditions are met: * * 1. Redistributions of source code must retain the above copyright notice, this * list of conditions and the following disclaimer. * * 2. Redistributions in binary form must reproduce the above copyright notice, * this list of conditions and the following disclaimer in the documentation * and/or other materials provided with the distribution. * * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE * DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR * SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER * CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, * OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. */ #include #include #include namespace Crypto::ASN1 { Result Decoder::read_tag() { auto byte_or_error = read_byte(); if (byte_or_error.is_error()) return byte_or_error.error(); auto byte = byte_or_error.value(); u8 class_ = byte & 0xc0; u8 type = byte & 0x20; u8 kind = byte & 0x1f; if (kind == 0x1f) { kind = 0; while (byte & 0x80) { auto byte_or_error = read_byte(); if (byte_or_error.is_error()) return byte_or_error.error(); byte = byte_or_error.value(); kind = (kind << 7) | (byte & 0x7f); } } return Tag { (Kind)kind, (Class)class_, (Type)type }; } Result Decoder::read_length() { auto byte_or_error = read_byte(); if (byte_or_error.is_error()) return byte_or_error.error(); auto byte = byte_or_error.value(); size_t length = byte; if (byte & 0x80) { auto count = byte & 0x7f; if (count == 0x7f) return DecodeError::InvalidInputFormat; auto data_or_error = read_bytes(count); if (data_or_error.is_error()) return data_or_error.error(); auto data = data_or_error.value(); length = 0; if (data.size() > sizeof(size_t)) return DecodeError::Overflow; for (auto&& byte : data) length = (length << 8) | (size_t)byte; } return length; } Result Decoder::read_byte() { if (m_stack.is_empty()) return DecodeError::NoInput; auto& entry = m_stack.last(); if (entry.is_empty()) return DecodeError::NotEnoughData; auto byte = entry[0]; entry = entry.slice(1); return byte; } Result Decoder::read_bytes(size_t length) { if (m_stack.is_empty()) return DecodeError::NoInput; auto& entry = m_stack.last(); if (entry.size() < length) return DecodeError::NotEnoughData; auto bytes = entry.slice(0, length); entry = entry.slice(length); return bytes; } Result Decoder::decode_boolean(ReadonlyBytes data) { if (data.size() != 1) return DecodeError::InvalidInputFormat; return data[0] == 0; } Result Decoder::decode_arbitrary_sized_integer(ReadonlyBytes data) { if (data.size() < 1) return DecodeError::NotEnoughData; if (data.size() > 1 && ((data[0] == 0xff && data[1] & 0x80) || (data[0] == 0x00 && !(data[1] & 0x80)))) { return DecodeError::InvalidInputFormat; } bool is_negative = data[0] & 0x80; if (is_negative) return DecodeError::UnsupportedFormat; return UnsignedBigInteger::import_data(data.data(), data.size()); } Result Decoder::decode_octet_string(ReadonlyBytes bytes) { return StringView { bytes.data(), bytes.size() }; } Result Decoder::decode_null(ReadonlyBytes data) { if (data.size() != 0) return DecodeError::InvalidInputFormat; return nullptr; } Result, DecodeError> Decoder::decode_object_identifier(ReadonlyBytes data) { Vector result; result.append(0); // Reserved space. u32 value = 0; for (auto&& byte : data) { if (value == 0 && byte == 0x80) return DecodeError::InvalidInputFormat; value = (value << 7) | (byte & 0x7f); if (!(byte & 0x80)) { result.append(value); value = 0; } } if (result.size() == 1 || result[1] >= 1600) return DecodeError::InvalidInputFormat; result[0] = result[1] / 40; result[1] = result[1] % 40; return result; } Result Decoder::decode_printable_string(ReadonlyBytes data) { Utf8View view { data }; if (!view.validate()) return DecodeError::InvalidInputFormat; return StringView { data }; } Result Decoder::decode_bit_string(ReadonlyBytes data) { if (data.size() < 1) return DecodeError::InvalidInputFormat; auto unused_bits = data[0]; auto total_size_in_bits = data.size() * 8; if (unused_bits > total_size_in_bits) return DecodeError::Overflow; return BitmapView { const_cast(data.offset_pointer(1)), total_size_in_bits - unused_bits }; } Result Decoder::peek() { if (m_stack.is_empty()) return DecodeError::NoInput; if (eof()) return DecodeError::EndOfStream; if (m_current_tag.has_value()) return m_current_tag.value(); auto tag_or_error = read_tag(); if (tag_or_error.is_error()) return tag_or_error.error(); m_current_tag = tag_or_error.value(); return m_current_tag.value(); } bool Decoder::eof() const { return m_stack.is_empty() || m_stack.last().is_empty(); } Optional Decoder::enter() { if (m_stack.is_empty()) return DecodeError::NoInput; auto tag_or_error = peek(); if (tag_or_error.is_error()) return tag_or_error.error(); auto tag = tag_or_error.value(); if (tag.type != Type::Constructed) return DecodeError::EnteringNonConstructedTag; auto length_or_error = read_length(); if (length_or_error.is_error()) return length_or_error.error(); auto length = length_or_error.value(); auto data_or_error = read_bytes(length); if (data_or_error.is_error()) return data_or_error.error(); m_current_tag.clear(); auto data = data_or_error.value(); m_stack.append(data); return {}; } Optional Decoder::leave() { if (m_stack.is_empty()) return DecodeError::NoInput; if (m_stack.size() == 1) return DecodeError::LeavingMainContext; m_stack.take_last(); m_current_tag.clear(); return {}; } } void AK::Formatter::format(FormatBuilder& fmtbuilder, Crypto::ASN1::DecodeError error) { using Crypto::ASN1::DecodeError; switch (error) { case DecodeError::NoInput: return Formatter::format(fmtbuilder, "DecodeError(No input provided)"); case DecodeError::NonConformingType: return Formatter::format(fmtbuilder, "DecodeError(Tried to read with a non-conforming type)"); case DecodeError::EndOfStream: return Formatter::format(fmtbuilder, "DecodeError(End of stream)"); case DecodeError::NotEnoughData: return Formatter::format(fmtbuilder, "DecodeError(Not enough data)"); case DecodeError::EnteringNonConstructedTag: return Formatter::format(fmtbuilder, "DecodeError(Tried to enter a primitive tag)"); case DecodeError::LeavingMainContext: return Formatter::format(fmtbuilder, "DecodeError(Tried to leave the main context)"); case DecodeError::InvalidInputFormat: return Formatter::format(fmtbuilder, "DecodeError(Input data contained invalid syntax/data)"); case DecodeError::Overflow: return Formatter::format(fmtbuilder, "DecodeError(Construction would overflow)"); case DecodeError::UnsupportedFormat: return Formatter::format(fmtbuilder, "DecodeError(Input data format not supported by this parser)"); default: return Formatter::format(fmtbuilder, "DecodeError(Unknown)"); } }