Coverage Report

Created: 2021-10-13 08:49

/src/botan/src/lib/utils/charset.cpp
Line
Count
Source
1
/*
2
* Character Set Handling
3
* (C) 1999-2007,2021 Jack Lloyd
4
*
5
* Botan is released under the Simplified BSD License (see license.txt)
6
*/
7
8
#include <botan/internal/charset.h>
9
#include <botan/internal/loadstor.h>
10
#include <botan/exceptn.h>
11
12
namespace Botan {
13
14
namespace {
15
16
void append_utf8_for(std::string& s, uint32_t c)
17
113k
   {
18
113k
   if(c >= 0xD800 && c < 0xE000)
19
494
      throw Decoding_Error("Invalid Unicode character");
20
21
113k
   if(c <= 0x7F)
22
50.2k
      {
23
50.2k
      const uint8_t b0 = static_cast<uint8_t>(c);
24
50.2k
      s.push_back(static_cast<char>(b0));
25
50.2k
      }
26
62.9k
   else if(c <= 0x7FF)
27
45.4k
      {
28
45.4k
      const uint8_t b0 = 0xC0 | static_cast<uint8_t>(c >> 6);
29
45.4k
      const uint8_t b1 = 0x80 | static_cast<uint8_t>(c & 0x3F);
30
45.4k
      s.push_back(static_cast<char>(b0));
31
45.4k
      s.push_back(static_cast<char>(b1));
32
45.4k
      }
33
17.4k
   else if(c <= 0xFFFF)
34
14.0k
      {
35
14.0k
      const uint8_t b0 = 0xE0 | static_cast<uint8_t>(c >> 12);
36
14.0k
      const uint8_t b1 = 0x80 | static_cast<uint8_t>((c >> 6) & 0x3F);
37
14.0k
      const uint8_t b2 = 0x80 | static_cast<uint8_t>(c & 0x3F);
38
14.0k
      s.push_back(static_cast<char>(b0));
39
14.0k
      s.push_back(static_cast<char>(b1));
40
14.0k
      s.push_back(static_cast<char>(b2));
41
14.0k
      }
42
3.45k
   else if(c <= 0x10FFFF)
43
2.66k
      {
44
2.66k
      const uint8_t b0 = 0xF0 | static_cast<uint8_t>(c >> 18);
45
2.66k
      const uint8_t b1 = 0x80 | static_cast<uint8_t>((c >> 12) & 0x3F);
46
2.66k
      const uint8_t b2 = 0x80 | static_cast<uint8_t>((c >> 6) & 0x3F);
47
2.66k
      const uint8_t b3 = 0x80 | static_cast<uint8_t>(c & 0x3F);
48
2.66k
      s.push_back(static_cast<char>(b0));
49
2.66k
      s.push_back(static_cast<char>(b1));
50
2.66k
      s.push_back(static_cast<char>(b2));
51
2.66k
      s.push_back(static_cast<char>(b3));
52
2.66k
      }
53
786
   else
54
786
      throw Decoding_Error("Invalid Unicode character");
55
56
113k
   }
57
58
}
59
60
std::string ucs2_to_utf8(const uint8_t ucs2[], size_t len)
61
4.27k
   {
62
4.27k
   if(len % 2 != 0)
63
233
      throw Decoding_Error("Invalid length for UCS-2 string");
64
65
4.03k
   const size_t chars = len / 2;
66
67
4.03k
   std::string s;
68
23.9k
   for(size_t i = 0; i != chars; ++i)
69
19.8k
      {
70
19.8k
      const uint32_t c = load_be<uint16_t>(ucs2, i);
71
19.8k
      append_utf8_for(s, c);
72
19.8k
      }
73
74
4.03k
   return s;
75
4.27k
   }
76
77
std::string ucs4_to_utf8(const uint8_t ucs4[], size_t len)
78
2.70k
   {
79
2.70k
   if(len % 4 != 0)
80
242
      throw Decoding_Error("Invalid length for UCS-4 string");
81
82
2.46k
   const size_t chars = len / 4;
83
84
2.46k
   std::string s;
85
6.75k
   for(size_t i = 0; i != chars; ++i)
86
4.29k
      {
87
4.29k
      const uint32_t c = load_be<uint32_t>(ucs4, i);
88
4.29k
      append_utf8_for(s, c);
89
4.29k
      }
90
91
2.46k
   return s;
92
2.70k
   }
93
94
/*
95
* Convert from ISO 8859-1 to UTF-8
96
*/
97
std::string latin1_to_utf8(const uint8_t chars[], size_t len)
98
6.09k
   {
99
6.09k
   std::string s;
100
95.6k
   for(size_t i = 0; i != len; ++i)
101
89.5k
      {
102
89.5k
      const uint32_t c = static_cast<uint8_t>(chars[i]);
103
89.5k
      append_utf8_for(s, c);
104
89.5k
      }
105
6.09k
   return s;
106
6.09k
   }
107
108
}
109