Coverage Report

Created: 2025-03-18 06:55

/src/libunistring/lib/uninorm/u-normalize-internal.h
Line
Count
Source (jump to first uncovered line)
1
/* Decomposition and composition of Unicode strings.
2
   Copyright (C) 2009-2024 Free Software Foundation, Inc.
3
   Written by Bruno Haible <bruno@clisp.org>, 2009.
4
5
   This file is free software: you can redistribute it and/or modify
6
   it under the terms of the GNU Lesser General Public License as
7
   published by the Free Software Foundation; either version 2.1 of the
8
   License, or (at your option) any later version.
9
10
   This file is distributed in the hope that it will be useful,
11
   but WITHOUT ANY WARRANTY; without even the implied warranty of
12
   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
13
   GNU Lesser General Public License for more details.
14
15
   You should have received a copy of the GNU Lesser General Public License
16
   along with this program.  If not, see <https://www.gnu.org/licenses/>.  */
17
18
UNIT *
19
FUNC (uninorm_t nf, const UNIT *s, size_t n,
20
      UNIT *resultbuf, size_t *lengthp)
21
194k
{
22
194k
  int (*decomposer) (ucs4_t uc, ucs4_t *decomposition) = nf->decomposer;
23
194k
  ucs4_t (*composer) (ucs4_t uc1, ucs4_t uc2) = nf->composer;
24
25
  /* The result being accumulated.  */
26
194k
  UNIT *result;
27
194k
  size_t length;
28
194k
  size_t allocated;
29
  /* The buffer for sorting.  */
30
194k
  #define SORTBUF_PREALLOCATED 64
31
194k
  struct ucs4_with_ccc sortbuf_preallocated[2 * SORTBUF_PREALLOCATED];
32
194k
  struct ucs4_with_ccc *sortbuf; /* array of size 2 * sortbuf_allocated */
33
194k
  size_t sortbuf_allocated;
34
194k
  size_t sortbuf_count;
35
36
  /* Initialize the accumulator.  */
37
194k
  if (resultbuf == NULL)
38
194k
    {
39
194k
      result = NULL;
40
194k
      allocated = 0;
41
194k
    }
42
0
  else
43
0
    {
44
0
      result = resultbuf;
45
0
      allocated = *lengthp;
46
0
    }
47
194k
  length = 0;
48
49
  /* Initialize the buffer for sorting.  */
50
194k
  sortbuf = sortbuf_preallocated;
51
194k
  sortbuf_allocated = SORTBUF_PREALLOCATED;
52
194k
  sortbuf_count = 0;
53
54
194k
  {
55
194k
    const UNIT *s_end = s + n;
56
57
194k
    for (;;)
58
2.21M
      {
59
2.21M
        int count;
60
2.21M
        ucs4_t decomposed[UC_DECOMPOSITION_MAX_LENGTH];
61
2.21M
        int decomposed_count;
62
2.21M
        int i;
63
64
2.21M
        if (s < s_end)
65
2.02M
          {
66
            /* Fetch the next character.  */
67
2.02M
            count = U_MBTOUC_UNSAFE (&decomposed[0], s, s_end - s);
68
2.02M
            decomposed_count = 1;
69
70
            /* Decompose it, recursively.
71
               It would be possible to precompute the recursive decomposition
72
               and store it in a table.  But this would significantly increase
73
               the size of the decomposition tables, because for example for
74
               U+1FC1 the recursive canonical decomposition and the recursive
75
               compatibility decomposition are different.  */
76
2.02M
            {
77
2.02M
              int curr;
78
79
4.29M
              for (curr = 0; curr < decomposed_count; )
80
2.27M
                {
81
                  /* Invariant: decomposed[0..curr-1] is fully decomposed, i.e.
82
                     all elements are atomic.  */
83
2.27M
                  ucs4_t curr_decomposed[UC_DECOMPOSITION_MAX_LENGTH];
84
2.27M
                  int curr_decomposed_count;
85
86
2.27M
                  curr_decomposed_count = decomposer (decomposed[curr], curr_decomposed);
87
2.27M
                  if (curr_decomposed_count >= 0)
88
126k
                    {
89
                      /* Move curr_decomposed[0..curr_decomposed_count-1] over
90
                         decomposed[curr], making room.  It's not worth using
91
                         memcpy() here, since the counts are so small.  */
92
126k
                      int shift = curr_decomposed_count - 1;
93
94
126k
                      if (shift < 0)
95
0
                        abort ();
96
126k
                      if (shift > 0)
97
125k
                        {
98
125k
                          int j;
99
100
125k
                          decomposed_count += shift;
101
125k
                          if (decomposed_count > UC_DECOMPOSITION_MAX_LENGTH)
102
0
                            abort ();
103
138k
                          for (j = decomposed_count - 1 - shift; j > curr; j--)
104
13.0k
                            decomposed[j + shift] = decomposed[j];
105
125k
                        }
106
379k
                      for (; shift >= 0; shift--)
107
252k
                        decomposed[curr + shift] = curr_decomposed[shift];
108
126k
                    }
109
2.14M
                  else
110
2.14M
                    {
111
                      /* decomposed[curr] is atomic.  */
112
2.14M
                      curr++;
113
2.14M
                    }
114
2.27M
                }
115
2.02M
            }
116
2.02M
          }
117
194k
        else
118
194k
          {
119
194k
            count = 0;
120
194k
            decomposed_count = 0;
121
194k
          }
122
123
2.21M
        i = 0;
124
2.21M
        for (;;)
125
4.36M
          {
126
4.36M
            ucs4_t uc;
127
4.36M
            int ccc;
128
129
4.36M
            if (s < s_end)
130
4.17M
              {
131
                /* Fetch the next character from the decomposition.  */
132
4.17M
                if (i == decomposed_count)
133
2.02M
                  break;
134
2.14M
                uc = decomposed[i];
135
2.14M
                ccc = uc_combining_class (uc);
136
2.14M
              }
137
194k
            else
138
194k
              {
139
                /* End of string reached.  */
140
194k
                uc = 0;
141
194k
                ccc = 0;
142
194k
              }
143
144
2.34M
            if (ccc == 0)
145
2.06M
              {
146
2.06M
                size_t j;
147
148
                /* Apply the canonical ordering algorithm to the accumulated
149
                   sequence of characters.  */
150
2.06M
                if (sortbuf_count > 1)
151
103k
                  gl_uninorm_decompose_merge_sort_inplace (sortbuf, sortbuf_count,
152
103k
                                                           sortbuf + sortbuf_count);
153
154
2.06M
                if (composer != NULL)
155
2.06M
                  {
156
                    /* Attempt to combine decomposed characters, as specified
157
                       in the Unicode Standard Annex #15 "Unicode Normalization
158
                       Forms".  We need to check
159
                         1. whether the first accumulated character is a
160
                            "starter" (i.e. has ccc = 0).  This is usually the
161
                            case.  But when the string starts with a
162
                            non-starter, the sortbuf also starts with a
163
                            non-starter.  Btw, this check could also be
164
                            omitted, because the composition table has only
165
                            entries (code1, code2) for which code1 is a
166
                            starter; if the first accumulated character is not
167
                            a starter, no lookup will succeed.
168
                         2. If the sortbuf has more than one character, check
169
                            for each of these characters that are not "blocked"
170
                            from the starter (i.e. have a ccc that is higher
171
                            than the ccc of the previous character) whether it
172
                            can be combined with the first character.
173
                         3. If only one character is left in sortbuf, check
174
                            whether it can be combined with the next character
175
                            (also a starter).  */
176
2.06M
                    if (sortbuf_count > 0 && sortbuf[0].ccc == 0)
177
1.86M
                      {
178
2.08M
                        for (j = 1; j < sortbuf_count; )
179
220k
                          {
180
220k
                            if (sortbuf[j].ccc > sortbuf[j - 1].ccc)
181
119k
                              {
182
119k
                                ucs4_t combined =
183
119k
                                  composer (sortbuf[0].code, sortbuf[j].code);
184
119k
                                if (combined)
185
94.2k
                                  {
186
94.2k
                                    size_t k;
187
188
94.2k
                                    sortbuf[0].code = combined;
189
                                    /* sortbuf[0].ccc = 0, still valid.  */
190
190k
                                    for (k = j + 1; k < sortbuf_count; k++)
191
96.1k
                                      sortbuf[k - 1] = sortbuf[k];
192
94.2k
                                    sortbuf_count--;
193
94.2k
                                    continue;
194
94.2k
                                  }
195
119k
                              }
196
126k
                            j++;
197
126k
                          }
198
1.86M
                        if (s < s_end && sortbuf_count == 1)
199
1.66M
                          {
200
1.66M
                            ucs4_t combined =
201
1.66M
                              composer (sortbuf[0].code, uc);
202
1.66M
                            if (combined)
203
6.63k
                              {
204
6.63k
                                uc = combined;
205
6.63k
                                ccc = 0;
206
                                /* uc could be further combined with subsequent
207
                                   characters.  So don't put it into sortbuf[0] in
208
                                   this round, only in the next round.  */
209
6.63k
                                sortbuf_count = 0;
210
6.63k
                              }
211
1.66M
                          }
212
1.86M
                      }
213
2.06M
                  }
214
215
4.10M
                for (j = 0; j < sortbuf_count; j++)
216
2.04M
                  {
217
2.04M
                    ucs4_t muc = sortbuf[j].code;
218
219
                    /* Append muc to the result accumulator.  */
220
2.04M
                    if (length < allocated)
221
1.84M
                      {
222
1.84M
                        int ret =
223
1.84M
                          U_UCTOMB (result + length, muc, allocated - length);
224
1.84M
                        if (ret == -1)
225
0
                          {
226
0
                            errno = EINVAL;
227
0
                            goto fail;
228
0
                          }
229
1.84M
                        if (ret >= 0)
230
1.84M
                          {
231
1.84M
                            length += ret;
232
1.84M
                            goto done_appending;
233
1.84M
                          }
234
1.84M
                      }
235
199k
                    {
236
199k
                      size_t old_allocated = allocated;
237
199k
                      size_t new_allocated = 2 * old_allocated;
238
199k
                      if (new_allocated < 64)
239
191k
                        new_allocated = 64;
240
199k
                      if (new_allocated < old_allocated) /* integer overflow? */
241
0
                        abort ();
242
199k
                      {
243
199k
                        UNIT *larger_result;
244
199k
                        if (result == NULL)
245
191k
                          {
246
191k
                            larger_result =
247
191k
                              (UNIT *) malloc (new_allocated * sizeof (UNIT));
248
191k
                            if (larger_result == NULL)
249
0
                              {
250
0
                                errno = ENOMEM;
251
0
                                goto fail;
252
0
                              }
253
191k
                          }
254
8.26k
                        else if (result == resultbuf)
255
0
                          {
256
0
                            larger_result =
257
0
                              (UNIT *) malloc (new_allocated * sizeof (UNIT));
258
0
                            if (larger_result == NULL)
259
0
                              {
260
0
                                errno = ENOMEM;
261
0
                                goto fail;
262
0
                              }
263
0
                            U_CPY (larger_result, resultbuf, length);
264
0
                          }
265
8.26k
                        else
266
8.26k
                          {
267
8.26k
                            larger_result =
268
8.26k
                              (UNIT *) realloc (result, new_allocated * sizeof (UNIT));
269
8.26k
                            if (larger_result == NULL)
270
0
                              {
271
0
                                errno = ENOMEM;
272
0
                                goto fail;
273
0
                              }
274
8.26k
                          }
275
199k
                        result = larger_result;
276
199k
                        allocated = new_allocated;
277
199k
                        {
278
199k
                          int ret =
279
199k
                            U_UCTOMB (result + length, muc, allocated - length);
280
199k
                          if (ret == -1)
281
0
                            {
282
0
                              errno = EINVAL;
283
0
                              goto fail;
284
0
                            }
285
199k
                          if (ret < 0)
286
0
                            abort ();
287
199k
                          length += ret;
288
199k
                          goto done_appending;
289
199k
                        }
290
199k
                      }
291
199k
                    }
292
2.04M
                   done_appending: ;
293
2.04M
                  }
294
295
                /* sortbuf is now empty.  */
296
2.06M
                sortbuf_count = 0;
297
2.06M
              }
298
299
2.34M
            if (!(s < s_end))
300
              /* End of string reached.  */
301
194k
              break;
302
303
            /* Append (uc, ccc) to sortbuf.  */
304
2.14M
            if (sortbuf_count == sortbuf_allocated)
305
1.40k
              {
306
1.40k
                struct ucs4_with_ccc *new_sortbuf;
307
308
1.40k
                sortbuf_allocated = 2 * sortbuf_allocated;
309
1.40k
                if (sortbuf_allocated < sortbuf_count) /* integer overflow? */
310
0
                  abort ();
311
1.40k
                new_sortbuf =
312
1.40k
                  (struct ucs4_with_ccc *) malloc (2 * sortbuf_allocated * sizeof (struct ucs4_with_ccc));
313
1.40k
                if (new_sortbuf == NULL)
314
0
                  {
315
0
                    errno = ENOMEM;
316
0
                    goto fail;
317
0
                  }
318
1.40k
                memcpy (new_sortbuf, sortbuf,
319
1.40k
                        sortbuf_count * sizeof (struct ucs4_with_ccc));
320
1.40k
                if (sortbuf != sortbuf_preallocated)
321
372
                  free (sortbuf);
322
1.40k
                sortbuf = new_sortbuf;
323
1.40k
              }
324
2.14M
            sortbuf[sortbuf_count].code = uc;
325
2.14M
            sortbuf[sortbuf_count].ccc = ccc;
326
2.14M
            sortbuf_count++;
327
328
2.14M
            i++;
329
2.14M
          }
330
331
2.21M
        if (!(s < s_end))
332
          /* End of string reached.  */
333
194k
          break;
334
335
2.02M
        s += count;
336
2.02M
      }
337
194k
  }
338
339
194k
  if (length == 0)
340
2.45k
    {
341
2.45k
      if (result == NULL)
342
2.45k
        {
343
          /* Return a non-NULL value.  NULL means error.  */
344
2.45k
          result = (UNIT *) malloc (1);
345
2.45k
          if (result == NULL)
346
0
            {
347
0
              errno = ENOMEM;
348
0
              goto fail;
349
0
            }
350
2.45k
        }
351
2.45k
    }
352
191k
  else if (result != resultbuf && length < allocated)
353
189k
    {
354
      /* Shrink the allocated memory if possible.  */
355
189k
      UNIT *memory;
356
357
189k
      memory = (UNIT *) realloc (result, length * sizeof (UNIT));
358
189k
      if (memory != NULL)
359
189k
        result = memory;
360
189k
    }
361
362
194k
  if (sortbuf_count > 0)
363
0
    abort ();
364
194k
  if (sortbuf != sortbuf_preallocated)
365
1.02k
    free (sortbuf);
366
367
194k
  *lengthp = length;
368
194k
  return result;
369
370
0
 fail:
371
0
  {
372
0
    int saved_errno = errno;
373
0
    if (sortbuf != sortbuf_preallocated)
374
0
      free (sortbuf);
375
0
    if (result != resultbuf)
376
0
      free (result);
377
0
    errno = saved_errno;
378
0
  }
379
0
  return NULL;
380
194k
}
Unexecuted instantiation: u8_normalize
u32_normalize
Line
Count
Source
21
194k
{
22
194k
  int (*decomposer) (ucs4_t uc, ucs4_t *decomposition) = nf->decomposer;
23
194k
  ucs4_t (*composer) (ucs4_t uc1, ucs4_t uc2) = nf->composer;
24
25
  /* The result being accumulated.  */
26
194k
  UNIT *result;
27
194k
  size_t length;
28
194k
  size_t allocated;
29
  /* The buffer for sorting.  */
30
194k
  #define SORTBUF_PREALLOCATED 64
31
194k
  struct ucs4_with_ccc sortbuf_preallocated[2 * SORTBUF_PREALLOCATED];
32
194k
  struct ucs4_with_ccc *sortbuf; /* array of size 2 * sortbuf_allocated */
33
194k
  size_t sortbuf_allocated;
34
194k
  size_t sortbuf_count;
35
36
  /* Initialize the accumulator.  */
37
194k
  if (resultbuf == NULL)
38
194k
    {
39
194k
      result = NULL;
40
194k
      allocated = 0;
41
194k
    }
42
0
  else
43
0
    {
44
0
      result = resultbuf;
45
0
      allocated = *lengthp;
46
0
    }
47
194k
  length = 0;
48
49
  /* Initialize the buffer for sorting.  */
50
194k
  sortbuf = sortbuf_preallocated;
51
194k
  sortbuf_allocated = SORTBUF_PREALLOCATED;
52
194k
  sortbuf_count = 0;
53
54
194k
  {
55
194k
    const UNIT *s_end = s + n;
56
57
194k
    for (;;)
58
2.21M
      {
59
2.21M
        int count;
60
2.21M
        ucs4_t decomposed[UC_DECOMPOSITION_MAX_LENGTH];
61
2.21M
        int decomposed_count;
62
2.21M
        int i;
63
64
2.21M
        if (s < s_end)
65
2.02M
          {
66
            /* Fetch the next character.  */
67
2.02M
            count = U_MBTOUC_UNSAFE (&decomposed[0], s, s_end - s);
68
2.02M
            decomposed_count = 1;
69
70
            /* Decompose it, recursively.
71
               It would be possible to precompute the recursive decomposition
72
               and store it in a table.  But this would significantly increase
73
               the size of the decomposition tables, because for example for
74
               U+1FC1 the recursive canonical decomposition and the recursive
75
               compatibility decomposition are different.  */
76
2.02M
            {
77
2.02M
              int curr;
78
79
4.29M
              for (curr = 0; curr < decomposed_count; )
80
2.27M
                {
81
                  /* Invariant: decomposed[0..curr-1] is fully decomposed, i.e.
82
                     all elements are atomic.  */
83
2.27M
                  ucs4_t curr_decomposed[UC_DECOMPOSITION_MAX_LENGTH];
84
2.27M
                  int curr_decomposed_count;
85
86
2.27M
                  curr_decomposed_count = decomposer (decomposed[curr], curr_decomposed);
87
2.27M
                  if (curr_decomposed_count >= 0)
88
126k
                    {
89
                      /* Move curr_decomposed[0..curr_decomposed_count-1] over
90
                         decomposed[curr], making room.  It's not worth using
91
                         memcpy() here, since the counts are so small.  */
92
126k
                      int shift = curr_decomposed_count - 1;
93
94
126k
                      if (shift < 0)
95
0
                        abort ();
96
126k
                      if (shift > 0)
97
125k
                        {
98
125k
                          int j;
99
100
125k
                          decomposed_count += shift;
101
125k
                          if (decomposed_count > UC_DECOMPOSITION_MAX_LENGTH)
102
0
                            abort ();
103
138k
                          for (j = decomposed_count - 1 - shift; j > curr; j--)
104
13.0k
                            decomposed[j + shift] = decomposed[j];
105
125k
                        }
106
379k
                      for (; shift >= 0; shift--)
107
252k
                        decomposed[curr + shift] = curr_decomposed[shift];
108
126k
                    }
109
2.14M
                  else
110
2.14M
                    {
111
                      /* decomposed[curr] is atomic.  */
112
2.14M
                      curr++;
113
2.14M
                    }
114
2.27M
                }
115
2.02M
            }
116
2.02M
          }
117
194k
        else
118
194k
          {
119
194k
            count = 0;
120
194k
            decomposed_count = 0;
121
194k
          }
122
123
2.21M
        i = 0;
124
2.21M
        for (;;)
125
4.36M
          {
126
4.36M
            ucs4_t uc;
127
4.36M
            int ccc;
128
129
4.36M
            if (s < s_end)
130
4.17M
              {
131
                /* Fetch the next character from the decomposition.  */
132
4.17M
                if (i == decomposed_count)
133
2.02M
                  break;
134
2.14M
                uc = decomposed[i];
135
2.14M
                ccc = uc_combining_class (uc);
136
2.14M
              }
137
194k
            else
138
194k
              {
139
                /* End of string reached.  */
140
194k
                uc = 0;
141
194k
                ccc = 0;
142
194k
              }
143
144
2.34M
            if (ccc == 0)
145
2.06M
              {
146
2.06M
                size_t j;
147
148
                /* Apply the canonical ordering algorithm to the accumulated
149
                   sequence of characters.  */
150
2.06M
                if (sortbuf_count > 1)
151
103k
                  gl_uninorm_decompose_merge_sort_inplace (sortbuf, sortbuf_count,
152
103k
                                                           sortbuf + sortbuf_count);
153
154
2.06M
                if (composer != NULL)
155
2.06M
                  {
156
                    /* Attempt to combine decomposed characters, as specified
157
                       in the Unicode Standard Annex #15 "Unicode Normalization
158
                       Forms".  We need to check
159
                         1. whether the first accumulated character is a
160
                            "starter" (i.e. has ccc = 0).  This is usually the
161
                            case.  But when the string starts with a
162
                            non-starter, the sortbuf also starts with a
163
                            non-starter.  Btw, this check could also be
164
                            omitted, because the composition table has only
165
                            entries (code1, code2) for which code1 is a
166
                            starter; if the first accumulated character is not
167
                            a starter, no lookup will succeed.
168
                         2. If the sortbuf has more than one character, check
169
                            for each of these characters that are not "blocked"
170
                            from the starter (i.e. have a ccc that is higher
171
                            than the ccc of the previous character) whether it
172
                            can be combined with the first character.
173
                         3. If only one character is left in sortbuf, check
174
                            whether it can be combined with the next character
175
                            (also a starter).  */
176
2.06M
                    if (sortbuf_count > 0 && sortbuf[0].ccc == 0)
177
1.86M
                      {
178
2.08M
                        for (j = 1; j < sortbuf_count; )
179
220k
                          {
180
220k
                            if (sortbuf[j].ccc > sortbuf[j - 1].ccc)
181
119k
                              {
182
119k
                                ucs4_t combined =
183
119k
                                  composer (sortbuf[0].code, sortbuf[j].code);
184
119k
                                if (combined)
185
94.2k
                                  {
186
94.2k
                                    size_t k;
187
188
94.2k
                                    sortbuf[0].code = combined;
189
                                    /* sortbuf[0].ccc = 0, still valid.  */
190
190k
                                    for (k = j + 1; k < sortbuf_count; k++)
191
96.1k
                                      sortbuf[k - 1] = sortbuf[k];
192
94.2k
                                    sortbuf_count--;
193
94.2k
                                    continue;
194
94.2k
                                  }
195
119k
                              }
196
126k
                            j++;
197
126k
                          }
198
1.86M
                        if (s < s_end && sortbuf_count == 1)
199
1.66M
                          {
200
1.66M
                            ucs4_t combined =
201
1.66M
                              composer (sortbuf[0].code, uc);
202
1.66M
                            if (combined)
203
6.63k
                              {
204
6.63k
                                uc = combined;
205
6.63k
                                ccc = 0;
206
                                /* uc could be further combined with subsequent
207
                                   characters.  So don't put it into sortbuf[0] in
208
                                   this round, only in the next round.  */
209
6.63k
                                sortbuf_count = 0;
210
6.63k
                              }
211
1.66M
                          }
212
1.86M
                      }
213
2.06M
                  }
214
215
4.10M
                for (j = 0; j < sortbuf_count; j++)
216
2.04M
                  {
217
2.04M
                    ucs4_t muc = sortbuf[j].code;
218
219
                    /* Append muc to the result accumulator.  */
220
2.04M
                    if (length < allocated)
221
1.84M
                      {
222
1.84M
                        int ret =
223
1.84M
                          U_UCTOMB (result + length, muc, allocated - length);
224
1.84M
                        if (ret == -1)
225
0
                          {
226
0
                            errno = EINVAL;
227
0
                            goto fail;
228
0
                          }
229
1.84M
                        if (ret >= 0)
230
1.84M
                          {
231
1.84M
                            length += ret;
232
1.84M
                            goto done_appending;
233
1.84M
                          }
234
1.84M
                      }
235
199k
                    {
236
199k
                      size_t old_allocated = allocated;
237
199k
                      size_t new_allocated = 2 * old_allocated;
238
199k
                      if (new_allocated < 64)
239
191k
                        new_allocated = 64;
240
199k
                      if (new_allocated < old_allocated) /* integer overflow? */
241
0
                        abort ();
242
199k
                      {
243
199k
                        UNIT *larger_result;
244
199k
                        if (result == NULL)
245
191k
                          {
246
191k
                            larger_result =
247
191k
                              (UNIT *) malloc (new_allocated * sizeof (UNIT));
248
191k
                            if (larger_result == NULL)
249
0
                              {
250
0
                                errno = ENOMEM;
251
0
                                goto fail;
252
0
                              }
253
191k
                          }
254
8.26k
                        else if (result == resultbuf)
255
0
                          {
256
0
                            larger_result =
257
0
                              (UNIT *) malloc (new_allocated * sizeof (UNIT));
258
0
                            if (larger_result == NULL)
259
0
                              {
260
0
                                errno = ENOMEM;
261
0
                                goto fail;
262
0
                              }
263
0
                            U_CPY (larger_result, resultbuf, length);
264
0
                          }
265
8.26k
                        else
266
8.26k
                          {
267
8.26k
                            larger_result =
268
8.26k
                              (UNIT *) realloc (result, new_allocated * sizeof (UNIT));
269
8.26k
                            if (larger_result == NULL)
270
0
                              {
271
0
                                errno = ENOMEM;
272
0
                                goto fail;
273
0
                              }
274
8.26k
                          }
275
199k
                        result = larger_result;
276
199k
                        allocated = new_allocated;
277
199k
                        {
278
199k
                          int ret =
279
199k
                            U_UCTOMB (result + length, muc, allocated - length);
280
199k
                          if (ret == -1)
281
0
                            {
282
0
                              errno = EINVAL;
283
0
                              goto fail;
284
0
                            }
285
199k
                          if (ret < 0)
286
0
                            abort ();
287
199k
                          length += ret;
288
199k
                          goto done_appending;
289
199k
                        }
290
199k
                      }
291
199k
                    }
292
2.04M
                   done_appending: ;
293
2.04M
                  }
294
295
                /* sortbuf is now empty.  */
296
2.06M
                sortbuf_count = 0;
297
2.06M
              }
298
299
2.34M
            if (!(s < s_end))
300
              /* End of string reached.  */
301
194k
              break;
302
303
            /* Append (uc, ccc) to sortbuf.  */
304
2.14M
            if (sortbuf_count == sortbuf_allocated)
305
1.40k
              {
306
1.40k
                struct ucs4_with_ccc *new_sortbuf;
307
308
1.40k
                sortbuf_allocated = 2 * sortbuf_allocated;
309
1.40k
                if (sortbuf_allocated < sortbuf_count) /* integer overflow? */
310
0
                  abort ();
311
1.40k
                new_sortbuf =
312
1.40k
                  (struct ucs4_with_ccc *) malloc (2 * sortbuf_allocated * sizeof (struct ucs4_with_ccc));
313
1.40k
                if (new_sortbuf == NULL)
314
0
                  {
315
0
                    errno = ENOMEM;
316
0
                    goto fail;
317
0
                  }
318
1.40k
                memcpy (new_sortbuf, sortbuf,
319
1.40k
                        sortbuf_count * sizeof (struct ucs4_with_ccc));
320
1.40k
                if (sortbuf != sortbuf_preallocated)
321
372
                  free (sortbuf);
322
1.40k
                sortbuf = new_sortbuf;
323
1.40k
              }
324
2.14M
            sortbuf[sortbuf_count].code = uc;
325
2.14M
            sortbuf[sortbuf_count].ccc = ccc;
326
2.14M
            sortbuf_count++;
327
328
2.14M
            i++;
329
2.14M
          }
330
331
2.21M
        if (!(s < s_end))
332
          /* End of string reached.  */
333
194k
          break;
334
335
2.02M
        s += count;
336
2.02M
      }
337
194k
  }
338
339
194k
  if (length == 0)
340
2.45k
    {
341
2.45k
      if (result == NULL)
342
2.45k
        {
343
          /* Return a non-NULL value.  NULL means error.  */
344
2.45k
          result = (UNIT *) malloc (1);
345
2.45k
          if (result == NULL)
346
0
            {
347
0
              errno = ENOMEM;
348
0
              goto fail;
349
0
            }
350
2.45k
        }
351
2.45k
    }
352
191k
  else if (result != resultbuf && length < allocated)
353
189k
    {
354
      /* Shrink the allocated memory if possible.  */
355
189k
      UNIT *memory;
356
357
189k
      memory = (UNIT *) realloc (result, length * sizeof (UNIT));
358
189k
      if (memory != NULL)
359
189k
        result = memory;
360
189k
    }
361
362
194k
  if (sortbuf_count > 0)
363
0
    abort ();
364
194k
  if (sortbuf != sortbuf_preallocated)
365
1.02k
    free (sortbuf);
366
367
194k
  *lengthp = length;
368
194k
  return result;
369
370
0
 fail:
371
0
  {
372
0
    int saved_errno = errno;
373
0
    if (sortbuf != sortbuf_preallocated)
374
0
      free (sortbuf);
375
0
    if (result != resultbuf)
376
0
      free (result);
377
0
    errno = saved_errno;
378
0
  }
379
0
  return NULL;
380
194k
}