• Home
  • Features
  • Pricing
  • Docs
  • Announcements
  • Sign In

randombit / botan / 13257645065

11 Feb 2025 07:10AM UTC coverage: 91.65% (-0.009%) from 91.659%
13257645065

Pull #4647

github

web-flow
Merge b9f3a0603 into f372b5a9e
Pull Request #4647: Avoid using mem_ops.h or assert.h in public headers

94860 of 103502 relevant lines covered (91.65%)

11460675.71 hits per line

Source File
Press 'n' to go to next uncovered line, 'b' for previous

77.92
/src/lib/filters/hex_filt.cpp
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 <botan/mem_ops.h>
13
#include <algorithm>
14

15
namespace Botan {
16

17
/**
18
* Size used for internal buffer in hex encoder/decoder
19
*/
20
const size_t HEX_CODEC_BUFFER_SIZE = 256;
21

22
/*
23
* Hex_Encoder Constructor
24
*/
25
Hex_Encoder::Hex_Encoder(bool breaks, size_t length, Case c) : m_casing(c), m_line_length(breaks ? length : 0) {
18✔
26
   m_in.resize(HEX_CODEC_BUFFER_SIZE);
10✔
27
   m_out.resize(2 * m_in.size());
10✔
28
   m_counter = m_position = 0;
10✔
29
}
10✔
30

31
/*
32
* Hex_Encoder Constructor
33
*/
34
Hex_Encoder::Hex_Encoder(Case c) : m_casing(c), m_line_length(0) {
×
35
   m_in.resize(HEX_CODEC_BUFFER_SIZE);
×
36
   m_out.resize(2 * m_in.size());
×
37
   m_counter = m_position = 0;
×
38
}
×
39

40
/*
41
* Encode and send a block
42
*/
43
void Hex_Encoder::encode_and_send(const uint8_t block[], size_t length) {
37✔
44
   hex_encode(cast_uint8_ptr_to_char(m_out.data()), block, length, m_casing == Uppercase);
37✔
45

46
   if(m_line_length == 0) {
37✔
47
      send(m_out, 2 * length);
35✔
48
   } else {
49
      size_t remaining = 2 * length, offset = 0;
2✔
50
      while(remaining) {
9✔
51
         size_t sent = std::min(m_line_length - m_counter, remaining);
7✔
52
         send(&m_out[offset], sent);
7✔
53
         m_counter += sent;
7✔
54
         remaining -= sent;
7✔
55
         offset += sent;
7✔
56
         if(m_counter == m_line_length) {
7✔
57
            send('\n');
6✔
58
            m_counter = 0;
6✔
59
         }
60
      }
61
   }
62
}
37✔
63

64
/*
65
* Convert some data into hex format
66
*/
67
void Hex_Encoder::write(const uint8_t input[], size_t length) {
37✔
68
   const size_t initial_fill = std::min(m_in.size() - m_position, length);
37✔
69
   copy_mem(&m_in[m_position], input, initial_fill);
37✔
70

71
   if(m_position + length >= m_in.size()) {
37✔
72
      encode_and_send(m_in.data(), m_in.size());
×
73
      input += (m_in.size() - m_position);
×
74
      length -= (m_in.size() - m_position);
×
75
      while(length >= m_in.size()) {
×
76
         encode_and_send(input, m_in.size());
×
77
         input += m_in.size();
×
78
         length -= m_in.size();
×
79
      }
80
      copy_mem(m_in.data(), input, length);
×
81
      m_position = 0;
×
82
   }
83
   m_position += length;
37✔
84
}
37✔
85

86
/*
87
* Flush buffers
88
*/
89
void Hex_Encoder::end_msg() {
37✔
90
   encode_and_send(m_in.data(), m_position);
37✔
91
   if(m_counter && m_line_length) {
37✔
92
      send('\n');
1✔
93
   }
94
   m_counter = m_position = 0;
37✔
95
}
37✔
96

97
/*
98
* Hex_Decoder Constructor
99
*/
100
Hex_Decoder::Hex_Decoder(Decoder_Checking c) : m_checking(c) {
7✔
101
   m_in.resize(HEX_CODEC_BUFFER_SIZE);
7✔
102
   m_out.resize(m_in.size() / 2);
7✔
103
   m_position = 0;
7✔
104
}
7✔
105

106
/*
107
* Convert some data from hex format
108
*/
109
void Hex_Decoder::write(const uint8_t input[], size_t length) {
23✔
110
   while(length) {
46✔
111
      size_t to_copy = std::min<size_t>(length, m_in.size() - m_position);
23✔
112
      copy_mem(&m_in[m_position], input, to_copy);
23✔
113
      m_position += to_copy;
23✔
114

115
      size_t consumed = 0;
23✔
116
      size_t written =
23✔
117
         hex_decode(m_out.data(), cast_uint8_ptr_to_char(m_in.data()), m_position, consumed, m_checking != FULL_CHECK);
23✔
118

119
      send(m_out, written);
23✔
120

121
      if(consumed != m_position) {
23✔
122
         copy_mem(m_in.data(), m_in.data() + consumed, m_position - consumed);
×
123
         m_position = m_position - consumed;
×
124
      } else {
125
         m_position = 0;
23✔
126
      }
127

128
      length -= to_copy;
23✔
129
      input += to_copy;
23✔
130
   }
131
}
23✔
132

133
/*
134
* Flush buffers
135
*/
136
void Hex_Decoder::end_msg() {
23✔
137
   size_t consumed = 0;
23✔
138
   size_t written =
23✔
139
      hex_decode(m_out.data(), cast_uint8_ptr_to_char(m_in.data()), m_position, consumed, m_checking != FULL_CHECK);
23✔
140

141
   send(m_out, written);
23✔
142

143
   const bool not_full_bytes = consumed != m_position;
23✔
144

145
   m_position = 0;
23✔
146

147
   if(not_full_bytes) {
23✔
148
      throw Invalid_Argument("Hex_Decoder: Input not full bytes");
×
149
   }
150
}
23✔
151

152
}  // namespace Botan
STATUS · Troubleshooting · Open an Issue · Sales · Support · CAREERS · ENTERPRISE · START FREE · SCHEDULE DEMO
ANNOUNCEMENTS · TWITTER · TOS & SLA · Supported CI Services · What's a CI service? · Automated Testing

© 2026 Coveralls, Inc