Coverage Report

Created: 2026-02-20 06:21

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
0
{
57
0
  int lastCanonicalClass = 0;
58
0
  int result = 1;
59
0
  size_t it;
60
61
0
  for (it = 0; it < len; it++)
62
0
    {
63
0
      uint32_t ch = label[it];
64
65
      // supplementary code point
66
0
      if (ch >= 0x10000)
67
0
  it++;
68
69
0
      int canonicalClass = uc_combining_class (ch);
70
0
      if (lastCanonicalClass > canonicalClass && canonicalClass != 0)
71
0
  return 0;
72
73
0
      NFCQCMap *map = get_nfcqc_map (ch);
74
0
      if (map)
75
0
  {
76
0
    if (map->check)
77
0
      return 0;
78
0
    result = -1;
79
0
  }
80
81
0
      lastCanonicalClass = canonicalClass;
82
0
    }
83
84
0
  return result;
85
0
}
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
1.62M
{
91
1.62M
  uint32_t *p;
92
1.62M
  size_t plen;
93
94
1.62M
  p = u8_to_u32 (src, srclen, NULL, &plen);
95
1.62M
  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
1.62M
  if (nfc && !_isNFC (p, plen))
103
0
    {
104
0
      size_t tmplen;
105
0
      uint32_t *tmp = u32_normalize (UNINORM_NFC, p, plen, NULL, &tmplen);
106
0
      free (p);
107
0
      if (tmp == NULL)
108
0
  {
109
0
    if (errno == ENOMEM)
110
0
      return IDN2_MALLOC;
111
0
    return IDN2_NFC;
112
0
  }
113
114
0
      p = tmp;
115
0
      plen = tmplen;
116
0
    }
117
118
1.62M
  *out = p;
119
1.62M
  *outlen = plen;
120
1.62M
  return IDN2_OK;
121
1.62M
}
122
123
bool
124
_idn2_ascii_p (const uint8_t *src, size_t srclen)
125
3.14M
{
126
3.14M
  size_t i;
127
128
3.35M
  for (i = 0; i < srclen; i++)
129
1.83M
    if (src[i] >= 0x80)
130
1.62M
      return false;
131
132
1.51M
  return true;
133
3.14M
}
134
135
int
136
_idn2_label_test (int what, const uint32_t *label, size_t llen)
137
3.56M
{
138
3.56M
  if (what & TEST_NFC)
139
3.56M
    {
140
3.56M
      size_t plen;
141
3.56M
      uint32_t *p = u32_normalize (UNINORM_NFC, label, llen,
142
3.56M
           NULL, &plen);
143
3.56M
      int ok;
144
3.56M
      if (p == NULL)
145
0
  {
146
0
    if (errno == ENOMEM)
147
0
      return IDN2_MALLOC;
148
0
    return IDN2_NFC;
149
0
  }
150
3.56M
      ok = llen == plen && memcmp (label, p, plen * sizeof (*label)) == 0;
151
3.56M
      free (p);
152
3.56M
      if (!ok)
153
1.17k
  return IDN2_NOT_NFC;
154
3.56M
    }
155
156
3.56M
  if (what & TEST_2HYPHEN)
157
3.56M
    {
158
3.56M
      if (llen >= 4 && label[2] == '-' && label[3] == '-')
159
363
  return IDN2_2HYPHEN;
160
3.56M
    }
161
162
3.56M
  if (what & TEST_HYPHEN_STARTEND)
163
1.93M
    {
164
1.93M
      if (llen > 0 && (label[0] == '-' || label[llen - 1] == '-'))
165
602
  return IDN2_HYPHEN_STARTEND;
166
1.93M
    }
167
168
3.56M
  if (what & TEST_LEADING_COMBINING)
169
3.56M
    {
170
3.56M
      if (llen > 0 && uc_is_general_category (label[0], UC_CATEGORY_M))
171
2.48k
  return IDN2_LEADING_COMBINING;
172
3.56M
    }
173
174
3.55M
  if (what & TEST_DISALLOWED)
175
1.62M
    {
176
1.62M
      size_t i;
177
3.53M
      for (i = 0; i < llen; i++)
178
1.91M
  if (_idn2_disallowed_p (label[i]))
179
22.6k
    {
180
22.6k
      if ((what & (TEST_TRANSITIONAL | TEST_NONTRANSITIONAL)) &&
181
22.6k
    (what & TEST_ALLOW_STD3_DISALLOWED))
182
22.6k
        {
183
22.6k
    IDNAMap map;
184
22.6k
    get_idna_map (label[i], &map);
185
22.6k
    if (map_is (&map, TR46_FLG_DISALLOWED_STD3_VALID) ||
186
4.86k
        map_is (&map, TR46_FLG_DISALLOWED_STD3_MAPPED))
187
19.1k
      continue;
188
189
22.6k
        }
190
191
3.50k
      return IDN2_DISALLOWED;
192
22.6k
    }
193
1.62M
    }
194
195
3.55M
  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
3.55M
  if (what & TEST_CONTEXTJ_RULE)
204
1.62M
    {
205
1.62M
      size_t i;
206
1.62M
      int rc;
207
208
3.52M
      for (i = 0; i < llen; i++)
209
1.90M
  {
210
1.90M
    rc = _idn2_contextj_rule (label, llen, i);
211
1.90M
    if (rc != IDN2_OK)
212
3.01k
      return rc;
213
1.90M
  }
214
1.62M
    }
215
216
3.55M
  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
3.55M
  if (what & TEST_CONTEXTO_WITH_RULE)
225
1.61M
    {
226
1.61M
      size_t i;
227
3.51M
      for (i = 0; i < llen; i++)
228
1.89M
  if (_idn2_contexto_p (label[i])
229
289k
      && !_idn2_contexto_with_rule (label[i]))
230
0
    return IDN2_CONTEXTO_NO_RULE;
231
1.61M
    }
232
233
3.55M
  if (what & TEST_CONTEXTO_RULE)
234
0
    {
235
0
      size_t i;
236
0
      int rc;
237
238
0
      for (i = 0; i < llen; i++)
239
0
  {
240
0
    rc = _idn2_contexto_rule (label, llen, i);
241
0
    if (rc != IDN2_OK)
242
0
      return rc;
243
0
  }
244
0
    }
245
246
3.55M
  if (what & TEST_UNASSIGNED)
247
1.61M
    {
248
1.61M
      size_t i;
249
3.51M
      for (i = 0; i < llen; i++)
250
1.89M
  if (_idn2_unassigned_p (label[i]))
251
0
    return IDN2_UNASSIGNED;
252
1.61M
    }
253
254
3.55M
  if (what & TEST_BIDI)
255
1.61M
    {
256
1.61M
      int rc = _idn2_bidi (label, llen);
257
1.61M
      if (rc != IDN2_OK)
258
6.17k
  return rc;
259
1.61M
    }
260
261
3.54M
  if (what & (TEST_TRANSITIONAL | TEST_NONTRANSITIONAL))
262
3.54M
    {
263
3.54M
      size_t i;
264
3.54M
      int transitional = what & TEST_TRANSITIONAL;
265
266
      /* TR46: 4. The label must not contain a U+002E ( . ) FULL STOP */
267
7.49M
      for (i = 0; i < llen; i++)
268
3.94M
  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
7.49M
      for (i = 0; i < llen; i++)
276
3.94M
  {
277
3.94M
    IDNAMap map;
278
279
3.94M
    get_idna_map (label[i], &map);
280
281
3.94M
    if (map_is (&map, TR46_FLG_VALID) ||
282
42.6k
        (!transitional && map_is (&map, TR46_FLG_DEVIATION)))
283
3.91M
      continue;
284
285
35.5k
    if (what & TEST_ALLOW_STD3_DISALLOWED &&
286
35.5k
        (map_is (&map, TR46_FLG_DISALLOWED_STD3_VALID) ||
287
3.41k
         map_is (&map, TR46_FLG_DISALLOWED_STD3_MAPPED)))
288
34.8k
      continue;
289
290
689
    return transitional ? IDN2_INVALID_TRANSITIONAL :
291
689
      IDN2_INVALID_NONTRANSITIONAL;
292
35.5k
  }
293
3.54M
    }
294
295
3.54M
  return IDN2_OK;
296
3.54M
}