KItinerary

uperdecoder.h
1/*
2 SPDX-FileCopyrightText: 2022 Volker Krause <vkrause@kde.org>
3 SPDX-License-Identifier: LGPL-2.0-or-later
4*/
5
6#ifndef KITINERARY_UPERDECODER_H
7#define KITINERARY_UPERDECODER_H
8
9#include "bitvectorview.h"
10
11#include <QMetaEnum>
12
13namespace KItinerary {
14
15/** Decoder for data encoded according to X.691 ASN.1 Unaligned Packed Encoding Rules (UPER). */
16class UPERDecoder
17{
18public:
19 explicit UPERDecoder(BitVectorView data);
20 ~UPERDecoder();
21
22 using size_type = BitVectorView::size_type;
23
24 size_type offset() const;
25 void seek(size_type index);
26
27 /** Read constrained whole number from the current position.
28 * @see X.691 §11.5
29 */
30 int64_t readConstrainedWholeNumber(int64_t minimum, int64_t maximum);
31
32 /** Read unconstrained whole number.
33 * @see X.691 §11.8
34 */
36
37 /** Read length determinant.
38 * @see X.691 §11.9
39 */
40 size_type readLengthDeterminant();
41
42 /** Read UTF-8 string. */
44
45 /** Read boolean value. */
46 bool readBoolean();
47
48 /** Read an unconstrained IA5String (7 bit ASCII).
49 * @see X.691 §30
50 */
52
53 /** Read length-constrained IA5String (7 bit ASCII).
54 * @see X.691 §30
55 */
56 QByteArray readIA5String(size_type minLenght, size_type maxLength);
57
58 /** Read unconstrained octet string (8 bit data).
59 * @see X.691 §17
60 */
62
63 /** Read an @tparam N sized bitmap. */
64 template <std::size_t N>
65 inline std::bitset<N> readBitset()
66 {
67 auto result = m_data.bitsetAt<N>(m_idx);
68 m_idx += N;
69 return result;
70 }
71
72 /** Read a sequence-of field with unrestricted size.
73 * @see X.691 §20
74 */
75 template <typename T>
77 {
78 const auto size = readLengthDeterminant();
79 QList<T> result;
80 result.reserve(size);
81 for (size_type i = 0; i < size; ++i) {
82 T element;
83 element.decode(*this);
84 result.push_back(std::move(element));
85 }
86 return result;
87 }
88 QList<int> readSequenceOfConstrainedWholeNumber(int64_t minimum, int64_t maximum);
89 QList<int> readSequenceOfUnconstrainedWholeNumber();
90 QList<QByteArray> readSequenceOfIA5String();
91 QList<QString> readSequenceOfUtf8String();
92
93 /** Read an Object Identifier value.
94 * @see X.691 §24
95 */
97
98 /** Read enumerated value.
99 * @see X.691 §14
100 */
101 template <typename T>
102 inline T readEnumerated()
103 {
104 const auto me = QMetaEnum::fromType<T>();
105 const auto idx = readConstrainedWholeNumber(0, me.keyCount() - 1);
106 return static_cast<T>(me.value(idx));
107 }
108 template <typename T>
109 inline T readEnumeratedWithExtensionMarker()
110 {
111 if (readBoolean()) {
112 setError("CHOICE with extension marker set not implemented.");
113 return {};
114 }
115 return readEnumerated<T>();
116 }
117
118 /** Read a choice value.
119 * @see X.691 §23
120 */
121 template <typename... Ts>
123 {
124 if (readBoolean()) {
125 setError("CHOICE with extension marker set not implemented.");
126 return {};
127 }
128 constexpr auto count = sizeof...(Ts);
129 const auto choiceIdx = readConstrainedWholeNumber(0, count - 1);
130 if (choiceIdx > (int)count) {
131 setError("Invalid CHOICE index.");
132 return {};
133 }
134 return readChoiceElement<Ts...>(choiceIdx);
135 }
136
137 /** Reading at any point encountered an error.
138 * As uPER gives us no way to recover from that, everything
139 * read has to be considered invalid.
140 */
141 bool hasError() const;
142 QByteArray errorMessage() const;
143
144 /** Put the decoder into the error state. */
145 void setError(const char *msg);
146
147private:
148 QByteArray readIA5StringData(size_type len);
149 [[nodiscard]] uint64_t readObjectIdentifierComponent(size_type &numBytes);
150
151 template <typename T, typename T1, typename... Ts>
152 inline QVariant readChoiceElement(int choiceIdx)
153 {
154 if (choiceIdx == 0) {
155 return readChoiceElement<T>(choiceIdx);
156 }
157 return readChoiceElement<T1, Ts...>(choiceIdx - 1);
158 }
159 template <typename T>
160 inline QVariant readChoiceElement([[maybe_unused]] int choiceIdx)
161 {
162 assert(choiceIdx == 0);
163 T value;
164 value.decode(*this);
165 return QVariant::fromValue(value);
166 }
167
168 BitVectorView m_data;
169 size_type m_idx = {};
170 QByteArray m_error;
171};
172
173}
174
175#endif // KITINERARY_UPERDECODER_H
Non-owning bit-level view for working with data that isn't byte-aligned.
QByteArray readObjectIdentifier()
Read an Object Identifier value.
QList< T > readSequenceOf()
Read a sequence-of field with unrestricted size.
Definition uperdecoder.h:76
T readEnumerated()
Read enumerated value.
int64_t readUnconstrainedWholeNumber()
Read unconstrained whole number.
void setError(const char *msg)
Put the decoder into the error state.
QString readUtf8String()
Read UTF-8 string.
QVariant readChoiceWithExtensionMarker()
Read a choice value.
QByteArray readOctetString()
Read unconstrained octet string (8 bit data).
bool readBoolean()
Read boolean value.
bool hasError() const
Reading at any point encountered an error.
QByteArray readIA5String()
Read an unconstrained IA5String (7 bit ASCII).
size_type readLengthDeterminant()
Read length determinant.
std::bitset< N > readBitset()
Read an.
Definition uperdecoder.h:65
int64_t readConstrainedWholeNumber(int64_t minimum, int64_t maximum)
Read constrained whole number from the current position.
Classes for reservation/travel data models, data extraction and data augmentation.
Definition berelement.h:17
void push_back(parameter_type value)
void reserve(qsizetype size)
QMetaEnum fromType()
QVariant fromValue(T &&value)
This file is part of the KDE documentation.
Documentation copyright © 1996-2025 The KDE developers.
Generated on Fri Apr 4 2025 12:02:17 by doxygen 1.13.2 written by Dimitri van Heesch, © 1997-2006

KDE's Doxygen guidelines are available online.