Coverage Report

Created: 2026-04-12 07:27

next uncovered line (L), next uncovered region (R), next uncovered branch (B)
/src/vlc/modules/demux/xiph_metadata.c
Line
Count
Source
1
/*****************************************************************************
2
 * xiph_metadata.h: Vorbis Comment parser
3
 *****************************************************************************
4
 * Copyright © 2008-2013 VLC authors and VideoLAN
5
 *
6
 * Authors: Laurent Aimar <fenrir _AT_ videolan _DOT_ org>
7
 *          Jean-Baptiste Kempf <jb@videolan.org>
8
 *
9
 * This program is free software; you can redistribute it and/or modify it
10
 * under the terms of the GNU Lesser General Public License as published by
11
 * the Free Software Foundation; either version 2.1 of the License, or
12
 * (at your option) any later version.
13
 *
14
 * This program is distributed in the hope that it will be useful,
15
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
16
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
17
 * GNU Lesser General Public License for more details.
18
 *
19
 * You should have received a copy of the GNU Lesser General Public License
20
 * along with this program; if not, write to the Free Software Foundation,
21
 * Inc., 51 Franklin Street, Fifth Floor, Boston MA 02110-1301, USA.
22
 *****************************************************************************/
23
24
#ifdef HAVE_CONFIG_H
25
# include "config.h"
26
#endif
27
28
#include <assert.h>
29
30
#include <vlc_common.h>
31
#include <vlc_arrays.h>
32
#include <vlc_charset.h>
33
#include <vlc_strings.h>
34
#include <vlc_arrays.h>
35
#include <vlc_input.h>
36
#include "xiph_metadata.h"
37
#include "../meta_engine/ID3Pictures.h"
38
39
input_attachment_t* ParseFlacPicture( const uint8_t *p_data, size_t size,
40
    int i_attachments, int *i_cover_score, int *i_cover_idx )
41
2.43k
{
42
2.43k
    uint32_t type, len;
43
44
2.43k
    if( size < 8 )
45
419
        return NULL;
46
2.01k
#define RM(x) \
47
11.3k
    do { \
48
11.3k
        assert(size >= (x)); \
49
11.3k
        size -= (x); \
50
11.3k
        p_data += (x); \
51
11.3k
    } while (0)
52
53
2.01k
    type = GetDWBE( p_data );
54
2.01k
    RM(4);
55
2.01k
    len = GetDWBE( p_data );
56
2.01k
    RM(4);
57
58
2.01k
    if( size < len )
59
202
        return NULL;
60
61
1.81k
    char *mime = strndup( (const char *)p_data, len );
62
1.81k
    if( unlikely(mime == NULL) )
63
0
        return NULL;
64
1.81k
    RM(len);
65
66
1.81k
    if( size < 4 )
67
178
    {
68
178
        free( mime );
69
178
        return NULL;
70
178
    }
71
72
1.63k
    len = GetDWBE( p_data );
73
1.63k
    RM(4);
74
75
1.63k
    if( size < len )
76
194
    {
77
194
        free( mime );
78
194
        return NULL;
79
194
    }
80
81
1.43k
    input_attachment_t *p_attachment = NULL;
82
1.43k
    char *description = strndup( (const char *)p_data, len );
83
1.43k
    if( unlikely(description == NULL) )
84
0
        goto error;
85
1.43k
    RM(len);
86
87
1.43k
    EnsureUTF8( description );
88
89
1.43k
    if( size < 20 )
90
235
        goto error;
91
92
1.20k
    RM(4 * 4); /* skip */
93
94
1.20k
    len = GetDWBE( p_data );
95
1.20k
    RM(4);
96
97
1.20k
    if( size < len )
98
87
        goto error;
99
100
    /* printf( "Picture type=%"PRIu32" mime=%s description='%s' "
101
               "file length=%zu\n", type, mime, description, len ); */
102
103
1.11k
    char name[7 + (sizeof (i_attachments) * 3) + 4 + 1];
104
105
1.11k
    snprintf( name, sizeof (name), "picture%u", i_attachments );
106
107
1.11k
    if( !strcasecmp( mime, "image/jpeg" ) )
108
7
        strcat( name, ".jpg" );
109
1.10k
    else if( !strcasecmp( mime, "image/png" ) )
110
13
        strcat( name, ".png" );
111
112
1.11k
    p_attachment = vlc_input_attachment_New( name, mime, description, p_data,
113
1.11k
                                             size /* XXX: len instead? */ );
114
115
1.11k
    if( type < ARRAY_SIZE(ID3v2_cover_scores) &&
116
579
        *i_cover_score < ID3v2_cover_scores[type] )
117
15
    {
118
15
        *i_cover_idx = i_attachments;
119
15
        *i_cover_score = ID3v2_cover_scores[type];
120
15
    }
121
122
1.43k
error:
123
1.43k
    free( mime );
124
1.43k
    free( description );
125
1.43k
    return p_attachment;
126
1.11k
}
127
128
#undef RM
129
#define RM(x) \
130
63.8k
    do { \
131
63.8k
        i_data -= (x); \
132
63.8k
        p_data += (x); \
133
63.8k
    } while (0)
134
135
136
typedef struct chapters_array_t
137
{
138
    unsigned int i_size;
139
    seekpoint_t ** pp_chapters;
140
} chapters_array_t;
141
142
static seekpoint_t * getChapterEntry( unsigned int i_index, chapters_array_t *p_array )
143
876
{
144
876
    if ( i_index > 4096 ) return NULL;
145
774
    if ( i_index >= p_array->i_size )
146
760
    {
147
760
        unsigned int i_newsize = p_array->i_size;
148
1.82k
        while( i_index >= i_newsize ) i_newsize += 50;
149
150
760
        if ( !p_array->pp_chapters )
151
745
        {
152
745
            p_array->pp_chapters = calloc( i_newsize, sizeof( seekpoint_t * ) );
153
745
            if ( !p_array->pp_chapters ) return NULL;
154
745
            p_array->i_size = i_newsize;
155
745
        } else {
156
15
            seekpoint_t **tmp = calloc( i_newsize, sizeof( seekpoint_t * ) );
157
15
            if ( !tmp ) return NULL;
158
15
            memcpy( tmp, p_array->pp_chapters, p_array->i_size * sizeof( seekpoint_t * ) );
159
15
            free( p_array->pp_chapters );
160
15
            p_array->pp_chapters = tmp;
161
15
            p_array->i_size = i_newsize;
162
15
        }
163
760
    }
164
774
    if ( !p_array->pp_chapters[i_index] )
165
767
        p_array->pp_chapters[i_index] = vlc_seekpoint_New();
166
774
    return p_array->pp_chapters[i_index];
167
774
}
168
169
5.99k
#define XIPHMETA_Title        (1 << 0)
170
5.57k
#define XIPHMETA_Artist       (1 << 1)
171
7.71k
#define XIPHMETA_Genre        (1 << 2)
172
260
#define XIPHMETA_Copyright    (1 << 3)
173
804
#define XIPHMETA_Album        (1 << 4)
174
9.67k
#define XIPHMETA_TrackNum     (1 << 5)
175
825
#define XIPHMETA_Description  (1 << 6)
176
231
#define XIPHMETA_Rating       (1 << 7)
177
5.85k
#define XIPHMETA_Date         (1 << 8)
178
300
#define XIPHMETA_Language     (1 << 9)
179
176
#define XIPHMETA_Publisher    (1 << 10)
180
1.17k
#define XIPHMETA_EncodedBy    (1 << 11)
181
627
#define XIPHMETA_TrackTotal   (1 << 12)
182
183
static char * xiph_ExtractCueSheetMeta( const char *psz_line,
184
                                        const char *psz_tag, int i_tag,
185
                                        bool b_quoted )
186
19.4k
{
187
19.4k
    if( !strncasecmp( psz_line, psz_tag, i_tag ) )
188
496
    {
189
496
        if( !b_quoted )
190
300
            return strdup( &psz_line[i_tag] );
191
192
        /* Unquote string value */
193
196
        char *psz_value = malloc( strlen( psz_line ) - i_tag + 1 );
194
196
        if( psz_value )
195
196
        {
196
196
            char *psz_out = psz_value;
197
196
            psz_line += i_tag;
198
196
            bool b_escaped = false;
199
3.72k
            while( *psz_line )
200
3.52k
            {
201
3.52k
                switch( *psz_line )
202
3.52k
                {
203
368
                    case '\\':
204
368
                        if( b_escaped )
205
93
                        {
206
93
                            b_escaped = false;
207
93
                            *(psz_out++) = *psz_line;
208
93
                        }
209
275
                        else
210
275
                        {
211
275
                            b_escaped = true;
212
275
                        }
213
368
                        break;
214
310
                    case '"':
215
310
                        if( b_escaped )
216
180
                        {
217
180
                            b_escaped = false;
218
180
                            *(psz_out++) = *psz_line;
219
180
                        }
220
310
                        break;
221
2.84k
                    default:
222
2.84k
                        *(psz_out++) = *psz_line;
223
2.84k
                        break;
224
3.52k
                }
225
3.52k
                psz_line++;
226
3.52k
            }
227
196
            *psz_out = 0;
228
196
            return psz_value;
229
196
        }
230
196
    }
231
18.9k
    return NULL;
232
19.4k
}
233
234
static void xiph_ParseCueSheetMeta( unsigned *pi_flags, vlc_meta_t *p_meta,
235
                                    const char *psz_line,
236
                                    int *pi_seekpoint, seekpoint_t ***ppp_seekpoint,
237
                                    seekpoint_t **pp_tmppoint, bool *pb_valid )
238
5.29k
{
239
5.29k
    VLC_UNUSED(pi_seekpoint);
240
5.29k
    VLC_UNUSED(ppp_seekpoint);
241
242
5.29k
    seekpoint_t *p_seekpoint = *pp_tmppoint;
243
5.29k
    char *psz_string;
244
245
5.29k
#define TRY_EXTRACT_CUEMETA(var, string, quoted) \
246
20.5k
    if( !(*pi_flags & XIPHMETA_##var) &&\
247
20.5k
         ( psz_string = xiph_ExtractCueSheetMeta( psz_line, string, sizeof(string) - 1, quoted ) ) )\
248
20.5k
    {\
249
417
        vlc_meta_Set( p_meta, vlc_meta_##var, psz_string );\
250
417
        free( psz_string );\
251
417
        *pi_flags |= XIPHMETA_##var;\
252
417
    }
253
254
5.29k
    TRY_EXTRACT_CUEMETA(Title, "TITLE \"", true)
255
5.19k
    else TRY_EXTRACT_CUEMETA(Genre, "REM GENRE ", false)
256
5.16k
    else TRY_EXTRACT_CUEMETA(Date, "REM DATE ", false)
257
4.89k
    else TRY_EXTRACT_CUEMETA(Artist, "PERFORMER \"", true)
258
4.87k
    else if( !strncasecmp( psz_line, "  TRACK ", 8 ) )
259
229
    {
260
229
        if( p_seekpoint )
261
44
        {
262
44
            if( *pb_valid )
263
44
                TAB_APPEND( *pi_seekpoint, *ppp_seekpoint, p_seekpoint );
264
44
            else
265
44
                vlc_seekpoint_Delete( p_seekpoint );
266
44
            *pb_valid = false;
267
44
        }
268
229
        *pp_tmppoint = p_seekpoint = vlc_seekpoint_New();
269
229
    }
270
4.64k
    else if( p_seekpoint && !strncasecmp( psz_line, "    INDEX 01 ", 13 ) )
271
5
    {
272
5
        unsigned m, s, f;
273
5
        if( sscanf( &psz_line[13], "%u:%u:%u", &m, &s, &f ) == 3 )
274
0
        {
275
0
            p_seekpoint->i_time_offset = vlc_tick_from_sec(m * 60 + s) + vlc_tick_from_samples(f, 75);
276
0
            *pb_valid = true;
277
0
        }
278
5
    }
279
4.64k
    else if( p_seekpoint && !p_seekpoint->psz_name )
280
1.50k
    {
281
1.50k
        p_seekpoint->psz_name = xiph_ExtractCueSheetMeta( psz_line, "    TITLE \"", 11, true );
282
1.50k
    }
283
5.29k
}
284
285
static void xiph_ParseCueSheet( unsigned *pi_flags, vlc_meta_t *p_meta,
286
                                const char *p_data, int i_data,
287
                                int *pi_seekpoint, seekpoint_t ***ppp_seekpoint )
288
1.78k
{
289
1.78k
    seekpoint_t *p_seekpoint = NULL;
290
1.78k
    bool b_valid = false;
291
292
1.78k
    const char *p_head = p_data;
293
1.78k
    const char *p_tail = p_head;
294
432k
    while( p_tail < p_data + i_data )
295
430k
    {
296
430k
        if( *p_tail == 0x0D )
297
5.29k
        {
298
5.29k
            char *psz = strndup( p_head, p_tail - p_head );
299
5.29k
            if( psz )
300
5.29k
            {
301
5.29k
                xiph_ParseCueSheetMeta( pi_flags, p_meta, psz,
302
5.29k
                                        pi_seekpoint, ppp_seekpoint,
303
5.29k
                                        &p_seekpoint, &b_valid );
304
5.29k
                free( psz );
305
5.29k
            }
306
5.29k
            if( *(++p_tail) == 0x0A )
307
624
                p_tail++;
308
5.29k
            p_head = p_tail;
309
5.29k
        }
310
425k
        else
311
425k
        {
312
425k
            p_tail++;
313
425k
        }
314
430k
    }
315
316
317
1.78k
    if( p_seekpoint )
318
185
    {
319
185
        if( b_valid )
320
185
            TAB_APPEND( *pi_seekpoint, *ppp_seekpoint, p_seekpoint );
321
185
        else
322
185
            vlc_seekpoint_Delete( p_seekpoint );
323
185
    }
324
1.78k
}
325
326
void vorbis_ParseComment( es_format_t *p_fmt, vlc_meta_t **pp_meta,
327
        const uint8_t *p_data, size_t i_data,
328
        int *i_attachments, input_attachment_t ***attachments,
329
        int *i_cover_score, int *i_cover_idx,
330
        int *i_seekpoint, seekpoint_t ***ppp_seekpoint )
331
14.1k
{
332
14.1k
    if( i_data < 8 )
333
1.61k
        return;
334
335
12.4k
    uint32_t vendor_length = GetDWLE(p_data); RM(4);
336
337
12.4k
    if( vendor_length > i_data )
338
1.85k
        return; /* invalid length */
339
340
10.6k
    RM(vendor_length); /* TODO: handle vendor payload */
341
342
10.6k
    if( i_data < 4 )
343
49
        return;
344
345
10.5k
    uint32_t i_comment = GetDWLE(p_data); RM(4);
346
347
10.5k
    if( i_comment > i_data || i_comment == 0 )
348
1.17k
        return; /* invalid length */
349
350
    /* */
351
9.41k
    vlc_meta_t *p_meta = *pp_meta;
352
9.41k
    if( !p_meta )
353
969
        *pp_meta = p_meta = vlc_meta_New();
354
355
9.41k
    if( unlikely( !p_meta ) )
356
0
        return;
357
358
    /* */
359
9.41k
    unsigned hasMetaFlags = 0;
360
361
9.41k
    chapters_array_t chapters_array = { 0, NULL };
362
363
22.2k
    for( ; i_comment > 0 && i_data >= 4; i_comment-- )
364
17.8k
    {
365
17.8k
        uint32_t comment_size = GetDWLE(p_data); RM(4);
366
367
17.8k
        if( comment_size > i_data )
368
5.00k
            break;
369
370
12.8k
        if( comment_size == 0 )
371
640
            continue;
372
373
12.2k
        char* psz_comment = malloc( comment_size + 1 );
374
375
12.2k
        if( unlikely( !psz_comment ) )
376
0
            goto next_comment;
377
378
12.2k
        memcpy( psz_comment, p_data, comment_size );
379
12.2k
        psz_comment[comment_size] = '\0';
380
381
12.2k
#define IF_EXTRACT(txt,var) \
382
115k
    if( !strncasecmp(psz_comment, txt, strlen(txt)) ) \
383
115k
    { \
384
4.39k
        size_t key_length = strlen(txt); \
385
4.39k
        EnsureUTF8( psz_comment + key_length ); \
386
4.39k
        const char *oldval = vlc_meta_Get( p_meta, vlc_meta_ ## var ); \
387
4.39k
        if( oldval && (hasMetaFlags & XIPHMETA_##var)) \
388
4.39k
        { \
389
424
            char * newval; \
390
424
            if( asprintf( &newval, "%s,%s", oldval, &psz_comment[key_length] ) == -1 ) \
391
424
                newval = NULL; \
392
424
            vlc_meta_Set( p_meta, vlc_meta_ ## var, newval ); \
393
424
            free( newval ); \
394
424
        } \
395
4.39k
        else \
396
4.39k
            vlc_meta_Set( p_meta, vlc_meta_ ## var, &psz_comment[key_length] ); \
397
4.39k
        hasMetaFlags |= XIPHMETA_##var; \
398
4.39k
    }
399
400
12.2k
#define IF_EXTRACT_ONCE_NUMBER(txt,var) \
401
18.5k
    if( !strncasecmp(psz_comment, txt, strlen(txt)) && !(hasMetaFlags & XIPHMETA_##var) ) \
402
18.5k
    { \
403
402
        bool isnum = true; \
404
402
        const char *num_str = &psz_comment[strlen(txt)], *c; \
405
1.58k
        for (c = num_str; isnum && *c != '\0'; c++) { \
406
1.18k
            isnum = *c >= '0' && *c <= '9'; \
407
1.18k
        } \
408
402
        if (isnum) \
409
402
        { \
410
211
            vlc_meta_Set( p_meta, vlc_meta_ ## var, num_str ); \
411
211
            hasMetaFlags |= XIPHMETA_##var; \
412
211
        } \
413
402
    }
414
415
12.2k
        IF_EXTRACT("TITLE=", Title )
416
11.8k
        else IF_EXTRACT("ARTIST=", Artist )
417
11.4k
        else IF_EXTRACT("GENRE=", Genre )
418
10.1k
        else IF_EXTRACT("COPYRIGHT=", Copyright )
419
10.0k
        else IF_EXTRACT("ALBUM=", Album )
420
9.60k
        else if( !(hasMetaFlags & XIPHMETA_TrackNum) && !strncasecmp(psz_comment, "TRACKNUMBER=", strlen("TRACKNUMBER=" ) ) )
421
177
        {
422
            /* Yeah yeah, such a clever idea, let's put xx/xx inside TRACKNUMBER
423
             * Oh, and let's not use TRACKTOTAL or TOTALTRACKS... */
424
177
            short unsigned u_track, u_total;
425
177
            int nb_values = sscanf( &psz_comment[strlen("TRACKNUMBER=")], "%hu/%hu", &u_track, &u_total );
426
177
            if( nb_values >= 1 )
427
75
            {
428
75
                char str[6];
429
75
                snprintf(str, 6, "%u", u_track);
430
75
                vlc_meta_Set( p_meta, vlc_meta_TrackNumber, str );
431
75
                hasMetaFlags |= XIPHMETA_TrackNum;
432
75
                if( nb_values >= 2 )
433
11
                {
434
11
                    snprintf(str, 6, "%u", u_total);
435
11
                    vlc_meta_Set( p_meta, vlc_meta_TrackTotal, str );
436
11
                    hasMetaFlags |= XIPHMETA_TrackTotal;
437
11
                }
438
75
            }
439
177
        }
440
9.42k
        else IF_EXTRACT_ONCE_NUMBER("TRACKTOTAL=", TrackTotal )
441
9.13k
        else IF_EXTRACT_ONCE_NUMBER("TOTALTRACKS=", TrackTotal )
442
9.02k
        else IF_EXTRACT("DESCRIPTION=", Description )
443
8.94k
        else IF_EXTRACT("COMMENT=", Description )
444
8.72k
        else IF_EXTRACT("COMMENTS=", Description )
445
8.59k
        else IF_EXTRACT("RATING=", Rating )
446
8.47k
        else IF_EXTRACT("DATE=", Date )
447
8.25k
        else if( !strncasecmp(psz_comment, "LANGUAGE=", strlen("LANGUAGE=") ) )
448
156
        {
449
156
            IF_EXTRACT("LANGUAGE=",Language)
450
156
            if( p_fmt )
451
0
            {
452
0
                free( p_fmt->psz_language );
453
0
                p_fmt->psz_language = strdup(&psz_comment[strlen("LANGUAGE=")]);
454
0
            }
455
156
        }
456
8.09k
        else IF_EXTRACT("ORGANIZATION=", Publisher )
457
8.00k
        else IF_EXTRACT("ENCODER=", EncodedBy )
458
7.27k
        else if( !strncasecmp( psz_comment, "METADATA_BLOCK_PICTURE=", strlen("METADATA_BLOCK_PICTURE=")))
459
96
        {
460
96
            if( attachments == NULL )
461
0
                goto next_comment;
462
463
96
            uint8_t *p_picture;
464
96
            size_t i_size = vlc_b64_decode_binary( &p_picture, &psz_comment[strlen("METADATA_BLOCK_PICTURE=")]);
465
96
            input_attachment_t *p_attachment = ParseFlacPicture( p_picture,
466
96
                i_size, *i_attachments, i_cover_score, i_cover_idx );
467
96
            free( p_picture );
468
96
            if( p_attachment )
469
0
            {
470
0
                TAB_APPEND_CAST( (input_attachment_t**),
471
0
                    *i_attachments, *attachments, p_attachment );
472
0
            }
473
96
        }
474
7.17k
        else if( !strncasecmp(psz_comment, "CHAPTER", 7) )
475
1.12k
        {
476
1.12k
            unsigned int i_chapt;
477
1.12k
            seekpoint_t *p_seekpoint = NULL;
478
479
15.3k
            for( int i = 0; psz_comment[i] && psz_comment[i] != '='; i++ )
480
14.1k
                if( psz_comment[i] >= 'a' && psz_comment[i] <= 'z' )
481
255
                    psz_comment[i] -= 'a' - 'A';
482
483
1.12k
            if( strstr( psz_comment, "NAME=" ) &&
484
899
                    sscanf( psz_comment, "CHAPTER%uNAME=", &i_chapt ) == 1 )
485
876
            {
486
876
                char *p = strchr( psz_comment, '=' );
487
876
                p_seekpoint = getChapterEntry( i_chapt, &chapters_array );
488
876
                if ( !p || ! p_seekpoint ) goto next_comment;
489
774
                EnsureUTF8( ++p );
490
774
                if ( ! p_seekpoint->psz_name )
491
767
                    p_seekpoint->psz_name = strdup( p );
492
774
            }
493
248
            else if( sscanf( psz_comment, "CHAPTER%u=", &i_chapt ) == 1 )
494
162
            {
495
162
                unsigned int h, m, s, ms;
496
162
                char *p = strchr( psz_comment, '=' );
497
162
                if( p && sscanf( ++p, "%u:%u:%u.%u", &h, &m, &s, &ms ) == 4 )
498
0
                {
499
0
                    p_seekpoint = getChapterEntry( i_chapt, &chapters_array );
500
0
                    if ( ! p_seekpoint ) goto next_comment;
501
0
                    p_seekpoint->i_time_offset = vlc_tick_from_sec(h * 3600 + m * 60 + s) + VLC_TICK_FROM_MS(ms);
502
0
                }
503
162
            }
504
1.12k
        }
505
6.05k
        else if( !strncasecmp(psz_comment, "cuesheet=", 9) )
506
1.78k
        {
507
1.78k
            EnsureUTF8( &psz_comment[9] );
508
1.78k
            xiph_ParseCueSheet( &hasMetaFlags, p_meta, &psz_comment[9], comment_size - 9,
509
1.78k
                                i_seekpoint, ppp_seekpoint );
510
1.78k
        }
511
4.26k
        else if( strchr( psz_comment, '=' ) )
512
3.41k
        {
513
            /* generic (PERFORMER/LICENSE/ORGANIZATION/LOCATION/CONTACT/ISRC,
514
             * undocumented tags and replay gain ) */
515
3.41k
            char *p = strchr( psz_comment, '=' );
516
3.41k
            *p++ = '\0';
517
3.41k
            EnsureUTF8( p );
518
519
30.5k
            for( int i = 0; psz_comment[i]; i++ )
520
27.1k
                if( psz_comment[i] >= 'a' && psz_comment[i] <= 'z' )
521
6.25k
                    psz_comment[i] -= 'a' - 'A';
522
523
3.41k
            vlc_meta_SetExtra( p_meta, psz_comment, p );
524
3.41k
        }
525
12.1k
#undef IF_EXTRACT
526
12.2k
next_comment:
527
12.2k
        free( psz_comment );
528
12.2k
        RM( comment_size );
529
12.2k
    }
530
9.41k
#undef RM
531
532
62.8k
    for ( unsigned int i=0; i<chapters_array.i_size; i++ )
533
53.4k
    {
534
53.4k
        if ( !chapters_array.pp_chapters[i] ) continue;
535
767
        TAB_APPEND_CAST( (seekpoint_t**), *i_seekpoint, *ppp_seekpoint,
536
767
                         chapters_array.pp_chapters[i] );
537
767
    }
538
9.41k
    free( chapters_array.pp_chapters );
539
9.41k
}
540
541
const char *FindKateCategoryName( const char *psz_tag )
542
0
{
543
0
    for( size_t i = 0; i < sizeof(Katei18nCategories)/sizeof(Katei18nCategories[0]); i++ )
544
0
    {
545
0
        if( !strcmp( psz_tag, Katei18nCategories[i].psz_tag ) )
546
0
            return Katei18nCategories[i].psz_i18n;
547
0
    }
548
0
    return N_("Unknown category");
549
0
}