Botan 3.5.0
Crypto and TLS for C&
hex_filt.cpp
Go to the documentation of this file.
1/*
2* Hex Encoder/Decoder
3* (C) 1999-2007 Jack Lloyd
4*
5* Botan is released under the Simplified BSD License (see license.txt)
6*/
7
8#include <botan/filters.h>
9
10#include <botan/exceptn.h>
11#include <botan/hex.h>
12#include <algorithm>
13
14namespace Botan {
15
16/**
17* Size used for internal buffer in hex encoder/decoder
18*/
19const size_t HEX_CODEC_BUFFER_SIZE = 256;
20
21/*
22* Hex_Encoder Constructor
23*/
24Hex_Encoder::Hex_Encoder(bool breaks, size_t length, Case c) : m_casing(c), m_line_length(breaks ? length : 0) {
25 m_in.resize(HEX_CODEC_BUFFER_SIZE);
26 m_out.resize(2 * m_in.size());
27 m_counter = m_position = 0;
28}
29
30/*
31* Hex_Encoder Constructor
32*/
33Hex_Encoder::Hex_Encoder(Case c) : m_casing(c), m_line_length(0) {
34 m_in.resize(HEX_CODEC_BUFFER_SIZE);
35 m_out.resize(2 * m_in.size());
36 m_counter = m_position = 0;
37}
38
39/*
40* Encode and send a block
41*/
42void Hex_Encoder::encode_and_send(const uint8_t block[], size_t length) {
43 hex_encode(cast_uint8_ptr_to_char(m_out.data()), block, length, m_casing == Uppercase);
44
45 if(m_line_length == 0) {
46 send(m_out, 2 * length);
47 } else {
48 size_t remaining = 2 * length, offset = 0;
49 while(remaining) {
50 size_t sent = std::min(m_line_length - m_counter, remaining);
51 send(&m_out[offset], sent);
52 m_counter += sent;
53 remaining -= sent;
54 offset += sent;
55 if(m_counter == m_line_length) {
56 send('\n');
57 m_counter = 0;
58 }
59 }
60 }
61}
62
63/*
64* Convert some data into hex format
65*/
66void Hex_Encoder::write(const uint8_t input[], size_t length) {
67 const size_t initial_fill = std::min(m_in.size() - m_position, length);
68 copy_mem(&m_in[m_position], input, initial_fill);
69
70 if(m_position + length >= m_in.size()) {
71 encode_and_send(m_in.data(), m_in.size());
72 input += (m_in.size() - m_position);
73 length -= (m_in.size() - m_position);
74 while(length >= m_in.size()) {
75 encode_and_send(input, m_in.size());
76 input += m_in.size();
77 length -= m_in.size();
78 }
79 copy_mem(m_in.data(), input, length);
80 m_position = 0;
81 }
82 m_position += length;
83}
84
85/*
86* Flush buffers
87*/
89 encode_and_send(m_in.data(), m_position);
90 if(m_counter && m_line_length) {
91 send('\n');
92 }
93 m_counter = m_position = 0;
94}
95
96/*
97* Hex_Decoder Constructor
98*/
100 m_in.resize(HEX_CODEC_BUFFER_SIZE);
101 m_out.resize(m_in.size() / 2);
102 m_position = 0;
103}
104
105/*
106* Convert some data from hex format
107*/
108void Hex_Decoder::write(const uint8_t input[], size_t length) {
109 while(length) {
110 size_t to_copy = std::min<size_t>(length, m_in.size() - m_position);
111 copy_mem(&m_in[m_position], input, to_copy);
112 m_position += to_copy;
113
114 size_t consumed = 0;
115 size_t written =
116 hex_decode(m_out.data(), cast_uint8_ptr_to_char(m_in.data()), m_position, consumed, m_checking != FULL_CHECK);
117
118 send(m_out, written);
119
120 if(consumed != m_position) {
121 copy_mem(m_in.data(), m_in.data() + consumed, m_position - consumed);
122 m_position = m_position - consumed;
123 } else {
124 m_position = 0;
125 }
126
127 length -= to_copy;
128 input += to_copy;
129 }
130}
131
132/*
133* Flush buffers
134*/
136 size_t consumed = 0;
137 size_t written =
138 hex_decode(m_out.data(), cast_uint8_ptr_to_char(m_in.data()), m_position, consumed, m_checking != FULL_CHECK);
139
140 send(m_out, written);
141
142 const bool not_full_bytes = consumed != m_position;
143
144 m_position = 0;
145
146 if(not_full_bytes) {
147 throw Invalid_Argument("Hex_Decoder: Input not full bytes");
148 }
149}
150
151} // namespace Botan
virtual void send(const uint8_t in[], size_t length)
Definition filter.cpp:27
void end_msg() override
Definition hex_filt.cpp:135
Hex_Decoder(Decoder_Checking checking=NONE)
Definition hex_filt.cpp:99
void write(const uint8_t[], size_t) override
Definition hex_filt.cpp:108
void write(const uint8_t in[], size_t length) override
Definition hex_filt.cpp:66
void end_msg() override
Definition hex_filt.cpp:88
Hex_Encoder(Case the_case)
Definition hex_filt.cpp:33
const size_t HEX_CODEC_BUFFER_SIZE
Definition hex_filt.cpp:19
void hex_encode(char output[], const uint8_t input[], size_t input_length, bool uppercase)
Definition hex.cpp:33
size_t hex_decode(uint8_t output[], const char input[], size_t input_length, size_t &input_consumed, bool ignore_ws)
Definition hex.cpp:81
const char * cast_uint8_ptr_to_char(const uint8_t *b)
Definition mem_ops.h:277
Decoder_Checking
Definition filter.h:165
@ FULL_CHECK
Definition filter.h:165
constexpr void copy_mem(T *out, const T *in, size_t n)
Definition mem_ops.h:146