Coverage Report

Created: 2025-12-14 07:00

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
38.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
7.44M
{
43
7.44M
  uint32_t value;
44
45
7.44M
  if (c <= 0xFF)
46
3.70M
    {
47
3.70M
      map->cp1 = *p++;
48
3.70M
      map->range = *p++;
49
3.70M
    }
50
3.73M
  else if (c <= 0xFFFF)
51
3.54M
    {
52
3.54M
      map->cp1 = (p[0] << 8) | p[1];
53
3.54M
      map->range = (p[2] << 8) | p[3];
54
3.54M
      p += 4;
55
3.54M
    }
56
188k
  else
57
188k
    {
58
188k
      map->cp1 = (p[0] << 16) | (p[1] << 8) | p[2];
59
188k
      map->range = (p[3] << 8) | p[4];
60
188k
      p += 5;
61
188k
    }
62
63
7.44M
  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
7.44M
  map->flag_index = value & 0x7;
68
7.44M
  map->offset = (value >> 3) & 0x3FFF;
69
7.44M
  map->nmappings = (value >> 17) & 0x1F;
70
7.44M
}
71
72
static int
73
_compare_idna_map (const uint32_t *c, const uint8_t *p)
74
6.63M
{
75
6.63M
  IDNAMap map;
76
77
6.63M
  _fill_map (*c, p, &map);
78
79
6.63M
  if (*c < map.cp1)
80
3.79M
    return -1;
81
2.84M
  if (*c > map.cp1 + map.range)
82
2.04M
    return 1;
83
804k
  return 0;
84
2.84M
}
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
804k
{
107
804k
  uint8_t *p;
108
109
804k
  if (c <= 0xFF)
110
512k
    p =
111
512k
      (uint8_t *) bsearch (&c, idna_map_8, sizeof (idna_map_8) / 5, 5,
112
512k
         (int (*)(const void *, const void *))
113
512k
         _compare_idna_map);
114
292k
  else if (c <= 0xFFFF)
115
275k
    p =
116
275k
      (uint8_t *) bsearch (&c, idna_map_16, sizeof (idna_map_16) / 7, 7,
117
275k
         (int (*)(const void *, const void *))
118
275k
         _compare_idna_map);
119
16.7k
  else if (c <= 0xFFFFFF)
120
16.7k
    p =
121
16.7k
      (uint8_t *) bsearch (&c, idna_map_24, sizeof (idna_map_24) / 8, 8,
122
16.7k
         (int (*)(const void *, const void *))
123
16.7k
         _compare_idna_map);
124
0
  else
125
0
    p = NULL;
126
127
804k
  if (!p)
128
0
    {
129
0
      memset (map, 0, sizeof (IDNAMap));
130
0
      return -1;
131
0
    }
132
133
804k
  _fill_map (c, p, map);
134
804k
  return 0;
135
804k
}
136
137
int
138
map_is (const IDNAMap *map, unsigned flags)
139
2.13M
{
140
2.13M
  return (idna_flags[map->flag_index] & flags) == flags;
141
2.13M
}
142
143
static int G_GNUC_IDN2_ATTRIBUTE_PURE
144
_compare_nfcqc_map (uint32_t *c, NFCQCMap *m2)
145
263k
{
146
263k
  if (*c < m2->cp1)
147
173k
    return -1;
148
89.4k
  if (*c > m2->cp2)
149
88.7k
    return 1;
150
688
  return 0;
151
89.4k
}
152
153
NFCQCMap *
154
get_nfcqc_map (uint32_t c)
155
38.5k
{
156
38.5k
  return (NFCQCMap *) bsearch (&c, nfcqc_map, countof (nfcqc_map),
157
38.5k
             sizeof (NFCQCMap),
158
38.5k
             (int (*)(const void *, const void *))
159
38.5k
             _compare_nfcqc_map);
160
38.5k
}
161
162
/* copy 'n' codepoints from mapdata stream */
163
int
164
get_map_data (uint32_t *dst, const IDNAMap *map)
165
86.4k
{
166
86.4k
  int n = map->nmappings;
167
86.4k
  const uint8_t *src = mapdata + map->offset;
168
169
271k
  for (; n > 0; n--)
170
184k
    {
171
184k
      uint32_t cp = 0;
172
184k
      do
173
295k
  cp = (cp << 7) | (*src & 0x7F);
174
295k
      while (*src++ & 0x80);
175
184k
      *dst++ = cp;
176
184k
    }
177
178
86.4k
  return map->nmappings;
179
86.4k
}