Coverage Report

Created: 2025-12-31 06:37

next uncovered line (L), next uncovered region (R), next uncovered branch (B)
/src/libidn2/lib/tr46map.c
Line
Count
Source
1
/* tr46map.c - implementation of IDNA2008 TR46 functions
2
   Copyright (C) 2016-2017 Tim Rühsen
3
4
   Libidn2 is free software: you can redistribute it and/or modify it
5
   under the terms of either:
6
7
     * the GNU Lesser General Public License as published by the Free
8
       Software Foundation; either version 3 of the License, or (at
9
       your option) any later version.
10
11
   or
12
13
     * the GNU General Public License as published by the Free
14
       Software Foundation; either version 2 of the License, or (at
15
       your option) any later version.
16
17
   or both in parallel, as here.
18
19
   This program is distributed in the hope that it will be useful,
20
   but WITHOUT ANY WARRANTY; without even the implied warranty of
21
   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
22
   GNU General Public License for more details.
23
24
   You should have received copies of the GNU General Public License and
25
   the GNU Lesser General Public License along with this program.  If
26
   not, see <http://www.gnu.org/licenses/>.
27
*/
28
29
#include <config.h>
30
31
#include <stdint.h>
32
#include <stdlib.h>   /* bsearch */
33
#include <string.h>   /* memset */
34
35
#include "tr46map.h"
36
#include "tr46map_data.c"
37
38
41.5k
#define countof(a) (sizeof(a)/sizeof(*(a)))
39
40
static void
41
_fill_map (uint32_t c, const uint8_t *p, IDNAMap *map)
42
101M
{
43
101M
  uint32_t value;
44
45
101M
  if (c <= 0xFF)
46
99.7M
    {
47
99.7M
      map->cp1 = *p++;
48
99.7M
      map->range = *p++;
49
99.7M
    }
50
1.36M
  else if (c <= 0xFFFF)
51
1.29M
    {
52
1.29M
      map->cp1 = (p[0] << 8) | p[1];
53
1.29M
      map->range = (p[2] << 8) | p[3];
54
1.29M
      p += 4;
55
1.29M
    }
56
68.2k
  else
57
68.2k
    {
58
68.2k
      map->cp1 = (p[0] << 16) | (p[1] << 8) | p[2];
59
68.2k
      map->range = (p[3] << 8) | p[4];
60
68.2k
      p += 5;
61
68.2k
    }
62
63
101M
  value = (p[0] << 16) | (p[1] << 8) | p[2];
64
65
  /* deconstruct value, construction was
66
   *   value = (((map->nmappings << 14) | map->offset) << 3) | map->flag_index; */
67
101M
  map->flag_index = value & 0x7;
68
101M
  map->offset = (value >> 3) & 0x3FFF;
69
101M
  map->nmappings = (value >> 17) & 0x1F;
70
101M
}
71
72
static int
73
_compare_idna_map (const uint32_t *c, const uint8_t *p)
74
88.0M
{
75
88.0M
  IDNAMap map;
76
77
88.0M
  _fill_map (*c, p, &map);
78
79
88.0M
  if (*c < map.cp1)
80
64.7M
    return -1;
81
23.3M
  if (*c > map.cp1 + map.range)
82
10.3M
    return 1;
83
12.9M
  return 0;
84
23.3M
}
85
86
/*
87
static int
88
_compare_idna_map(uint32_t *c, IDNAMap *m2)
89
{
90
  if (*c < m2->cp1)
91
    return -1;
92
  if (*c > m2->cp1 + m2->range)
93
    return 1;
94
  return 0;
95
}
96
97
IDNAMap
98
*get_idna_map(uint32_t c)
99
{
100
  return bsearch(&c, idna_map, countof(idna_map), sizeof(IDNAMap), (int(*)(const void *, const void *))_compare_idna_map);
101
}
102
*/
103
104
int
105
get_idna_map (uint32_t c, IDNAMap *map)
106
12.9M
{
107
12.9M
  uint8_t *p;
108
109
12.9M
  if (c <= 0xFF)
110
12.8M
    p =
111
12.8M
      (uint8_t *) bsearch (&c, idna_map_8, sizeof (idna_map_8) / 5, 5,
112
12.8M
         (int (*)(const void *, const void *))
113
12.8M
         _compare_idna_map);
114
104k
  else if (c <= 0xFFFF)
115
99.3k
    p =
116
99.3k
      (uint8_t *) bsearch (&c, idna_map_16, sizeof (idna_map_16) / 7, 7,
117
99.3k
         (int (*)(const void *, const void *))
118
99.3k
         _compare_idna_map);
119
5.62k
  else if (c <= 0xFFFFFF)
120
5.62k
    p =
121
5.62k
      (uint8_t *) bsearch (&c, idna_map_24, sizeof (idna_map_24) / 8, 8,
122
5.62k
         (int (*)(const void *, const void *))
123
5.62k
         _compare_idna_map);
124
0
  else
125
0
    p = NULL;
126
127
12.9M
  if (!p)
128
0
    {
129
0
      memset (map, 0, sizeof (IDNAMap));
130
0
      return -1;
131
0
    }
132
133
12.9M
  _fill_map (c, p, map);
134
12.9M
  return 0;
135
12.9M
}
136
137
int
138
map_is (const IDNAMap *map, unsigned flags)
139
55.1M
{
140
55.1M
  return (idna_flags[map->flag_index] & flags) == flags;
141
55.1M
}
142
143
static int G_GNUC_IDN2_ATTRIBUTE_PURE
144
_compare_nfcqc_map (uint32_t *c, NFCQCMap *m2)
145
272k
{
146
272k
  if (*c < m2->cp1)
147
134k
    return -1;
148
138k
  if (*c > m2->cp2)
149
137k
    return 1;
150
1.03k
  return 0;
151
138k
}
152
153
NFCQCMap *
154
get_nfcqc_map (uint32_t c)
155
41.5k
{
156
41.5k
  return (NFCQCMap *) bsearch (&c, nfcqc_map, countof (nfcqc_map),
157
41.5k
             sizeof (NFCQCMap),
158
41.5k
             (int (*)(const void *, const void *))
159
41.5k
             _compare_nfcqc_map);
160
41.5k
}
161
162
/* copy 'n' codepoints from mapdata stream */
163
int
164
get_map_data (uint32_t *dst, const IDNAMap *map)
165
23.6k
{
166
23.6k
  int n = map->nmappings;
167
23.6k
  const uint8_t *src = mapdata + map->offset;
168
169
75.9k
  for (; n > 0; n--)
170
52.3k
    {
171
52.3k
      uint32_t cp = 0;
172
52.3k
      do
173
86.8k
  cp = (cp << 7) | (*src & 0x7F);
174
86.8k
      while (*src++ & 0x80);
175
52.3k
      *dst++ = cp;
176
52.3k
    }
177
178
23.6k
  return map->nmappings;
179
23.6k
}