Coverage Report

Created: 2026-03-24 06:37

next uncovered line (L), next uncovered region (R), next uncovered branch (B)
/src/libidn2/lib/idna.c
Line
Count
Source
1
/* idna.c - implementation of high-level IDNA processing function
2
   Copyright (C) 2011-2025 Simon Josefsson
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 <stdlib.h>   /* free */
32
#include <errno.h>    /* errno */
33
34
#include "idn2.h"
35
#include "bidi.h"
36
#include "tables.h"
37
#include "context.h"
38
#include "tr46map.h"
39
40
#include <unitypes.h>
41
#include <unictype.h>   /* uc_is_general_category, UC_CATEGORY_M */
42
#include <uninorm.h>    /* u32_normalize */
43
#include <unistr.h>   /* u8_to_u32 */
44
45
#include "idna.h"
46
47
/*
48
 * NFC Quick Check from
49
 * http://unicode.org/reports/tr15/#Detecting_Normalization_Forms
50
 *
51
 * They say, this is much faster than 'brute force' normalization.
52
 * Strings are very likely already in NFC form.
53
 */
54
G_GNUC_IDN2_ATTRIBUTE_PURE static int
55
_isNFC (uint32_t *label, size_t len)
56
4.75k
{
57
4.75k
  int lastCanonicalClass = 0;
58
4.75k
  int result = 1;
59
4.75k
  size_t it;
60
61
44.0k
  for (it = 0; it < len; it++)
62
40.0k
    {
63
40.0k
      uint32_t ch = label[it];
64
65
      // supplementary code point
66
40.0k
      if (ch >= 0x10000)
67
1.74k
  it++;
68
69
40.0k
      int canonicalClass = uc_combining_class (ch);
70
40.0k
      if (lastCanonicalClass > canonicalClass && canonicalClass != 0)
71
81
  return 0;
72
73
40.0k
      NFCQCMap *map = get_nfcqc_map (ch);
74
40.0k
      if (map)
75
707
  {
76
707
    if (map->check)
77
707
      return 0;
78
0
    result = -1;
79
0
  }
80
81
39.3k
      lastCanonicalClass = canonicalClass;
82
39.3k
    }
83
84
3.97k
  return result;
85
4.75k
}
86
87
int
88
_idn2_u8_to_u32_nfc (const uint8_t *src, size_t srclen,
89
         uint32_t **out, size_t *outlen, int nfc)
90
20.7k
{
91
20.7k
  uint32_t *p;
92
20.7k
  size_t plen;
93
94
20.7k
  p = u8_to_u32 (src, srclen, NULL, &plen);
95
20.7k
  if (p == NULL)
96
0
    {
97
0
      if (errno == ENOMEM)
98
0
  return IDN2_MALLOC;
99
0
      return IDN2_ENCODING_ERROR;
100
0
    }
101
102
20.7k
  if (nfc && !_isNFC (p, plen))
103
788
    {
104
788
      size_t tmplen;
105
788
      uint32_t *tmp = u32_normalize (UNINORM_NFC, p, plen, NULL, &tmplen);
106
788
      free (p);
107
788
      if (tmp == NULL)
108
0
  {
109
0
    if (errno == ENOMEM)
110
0
      return IDN2_MALLOC;
111
0
    return IDN2_NFC;
112
0
  }
113
114
788
      p = tmp;
115
788
      plen = tmplen;
116
788
    }
117
118
20.7k
  *out = p;
119
20.7k
  *outlen = plen;
120
20.7k
  return IDN2_OK;
121
20.7k
}
122
123
bool
124
_idn2_ascii_p (const uint8_t *src, size_t srclen)
125
32.6k
{
126
32.6k
  size_t i;
127
128
175k
  for (i = 0; i < srclen; i++)
129
159k
    if (src[i] >= 0x80)
130
16.7k
      return false;
131
132
15.9k
  return true;
133
32.6k
}
134
135
int
136
_idn2_label_test (int what, const uint32_t *label, size_t llen)
137
48.7k
{
138
48.7k
  if (what & TEST_NFC)
139
48.7k
    {
140
48.7k
      size_t plen;
141
48.7k
      uint32_t *p = u32_normalize (UNINORM_NFC, label, llen,
142
48.7k
           NULL, &plen);
143
48.7k
      int ok;
144
48.7k
      if (p == NULL)
145
0
  {
146
0
    if (errno == ENOMEM)
147
0
      return IDN2_MALLOC;
148
0
    return IDN2_NFC;
149
0
  }
150
48.7k
      ok = llen == plen && memcmp (label, p, plen * sizeof (*label)) == 0;
151
48.7k
      free (p);
152
48.7k
      if (!ok)
153
2.24k
  return IDN2_NOT_NFC;
154
48.7k
    }
155
156
46.5k
  if (what & TEST_2HYPHEN)
157
46.5k
    {
158
46.5k
      if (llen >= 4 && label[2] == '-' && label[3] == '-')
159
388
  return IDN2_2HYPHEN;
160
46.5k
    }
161
162
46.1k
  if (what & TEST_HYPHEN_STARTEND)
163
33.4k
    {
164
33.4k
      if (llen > 0 && (label[0] == '-' || label[llen - 1] == '-'))
165
1.09k
  return IDN2_HYPHEN_STARTEND;
166
33.4k
    }
167
168
45.0k
  if (what & TEST_LEADING_COMBINING)
169
45.0k
    {
170
45.0k
      if (llen > 0 && uc_is_general_category (label[0], UC_CATEGORY_M))
171
1.14k
  return IDN2_LEADING_COMBINING;
172
45.0k
    }
173
174
43.8k
  if (what & TEST_DISALLOWED)
175
14.2k
    {
176
14.2k
      size_t i;
177
142k
      for (i = 0; i < llen; i++)
178
129k
  if (_idn2_disallowed_p (label[i]))
179
17.2k
    {
180
17.2k
      if ((what & (TEST_TRANSITIONAL | TEST_NONTRANSITIONAL)) &&
181
16.8k
    (what & TEST_ALLOW_STD3_DISALLOWED))
182
16.4k
        {
183
16.4k
    IDNAMap map;
184
16.4k
    get_idna_map (label[i], &map);
185
16.4k
    if (map_is (&map, TR46_FLG_DISALLOWED_STD3_VALID) ||
186
2.66k
        map_is (&map, TR46_FLG_DISALLOWED_STD3_MAPPED))
187
15.2k
      continue;
188
189
16.4k
        }
190
191
2.00k
      return IDN2_DISALLOWED;
192
17.2k
    }
193
14.2k
    }
194
195
41.8k
  if (what & TEST_CONTEXTJ)
196
0
    {
197
0
      size_t i;
198
0
      for (i = 0; i < llen; i++)
199
0
  if (_idn2_contextj_p (label[i]))
200
0
    return IDN2_CONTEXTJ;
201
0
    }
202
203
41.8k
  if (what & TEST_CONTEXTJ_RULE)
204
12.2k
    {
205
12.2k
      size_t i;
206
12.2k
      int rc;
207
208
130k
      for (i = 0; i < llen; i++)
209
118k
  {
210
118k
    rc = _idn2_contextj_rule (label, llen, i);
211
118k
    if (rc != IDN2_OK)
212
942
      return rc;
213
118k
  }
214
12.2k
    }
215
216
40.9k
  if (what & TEST_CONTEXTO)
217
0
    {
218
0
      size_t i;
219
0
      for (i = 0; i < llen; i++)
220
0
  if (_idn2_contexto_p (label[i]))
221
0
    return IDN2_CONTEXTO;
222
0
    }
223
224
40.9k
  if (what & TEST_CONTEXTO_WITH_RULE)
225
10.2k
    {
226
10.2k
      size_t i;
227
111k
      for (i = 0; i < llen; i++)
228
101k
  if (_idn2_contexto_p (label[i])
229
6.16k
      && !_idn2_contexto_with_rule (label[i]))
230
0
    return IDN2_CONTEXTO_NO_RULE;
231
10.2k
    }
232
233
40.9k
  if (what & TEST_CONTEXTO_RULE)
234
1.06k
    {
235
1.06k
      size_t i;
236
1.06k
      int rc;
237
238
7.83k
      for (i = 0; i < llen; i++)
239
6.91k
  {
240
6.91k
    rc = _idn2_contexto_rule (label, llen, i);
241
6.91k
    if (rc != IDN2_OK)
242
147
      return rc;
243
6.91k
  }
244
1.06k
    }
245
246
40.7k
  if (what & TEST_UNASSIGNED)
247
11.1k
    {
248
11.1k
      size_t i;
249
118k
      for (i = 0; i < llen; i++)
250
107k
  if (_idn2_unassigned_p (label[i]))
251
149
    return IDN2_UNASSIGNED;
252
11.1k
    }
253
254
40.6k
  if (what & TEST_BIDI)
255
11.0k
    {
256
11.0k
      int rc = _idn2_bidi (label, llen);
257
11.0k
      if (rc != IDN2_OK)
258
1.77k
  return rc;
259
11.0k
    }
260
261
38.8k
  if (what & (TEST_TRANSITIONAL | TEST_NONTRANSITIONAL))
262
38.2k
    {
263
38.2k
      size_t i;
264
38.2k
      int transitional = what & TEST_TRANSITIONAL;
265
266
      /* TR46: 4. The label must not contain a U+002E ( . ) FULL STOP */
267
298k
      for (i = 0; i < llen; i++)
268
260k
  if (label[i] == 0x002E)
269
0
    return IDN2_DOT_IN_LABEL;
270
271
      /* TR46: 6. Each code point in the label must only have certain status
272
       * values according to Section 5, IDNA Mapping Table:
273
       *    a. For Transitional Processing, each value must be valid.
274
       *    b. For Nontransitional Processing, each value must be either valid or deviation. */
275
293k
      for (i = 0; i < llen; i++)
276
258k
  {
277
258k
    IDNAMap map;
278
279
258k
    get_idna_map (label[i], &map);
280
281
258k
    if (map_is (&map, TR46_FLG_VALID) ||
282
40.5k
        (!transitional && map_is (&map, TR46_FLG_DEVIATION)))
283
225k
      continue;
284
285
32.9k
    if (what & TEST_ALLOW_STD3_DISALLOWED &&
286
31.8k
        (map_is (&map, TR46_FLG_DISALLOWED_STD3_VALID) ||
287
4.47k
         map_is (&map, TR46_FLG_DISALLOWED_STD3_MAPPED)))
288
29.9k
      continue;
289
290
3.00k
    return transitional ? IDN2_INVALID_TRANSITIONAL :
291
3.00k
      IDN2_INVALID_NONTRANSITIONAL;
292
32.9k
  }
293
38.2k
    }
294
295
35.8k
  return IDN2_OK;
296
38.8k
}