Coverage Report

Created: 2026-01-10 06:10

next uncovered line (L), next uncovered region (R), next uncovered branch (B)
/src/avahi/avahi-common/alternative.c
Line
Count
Source
1
/***
2
  This file is part of avahi.
3
4
  avahi is free software; you can redistribute it and/or modify it
5
  under the terms of the GNU Lesser General Public License as
6
  published by the Free Software Foundation; either version 2.1 of the
7
  License, or (at your option) any later version.
8
9
  avahi is distributed in the hope that it will be useful, but WITHOUT
10
  ANY WARRANTY; without even the implied warranty of MERCHANTABILITY
11
  or FITNESS FOR A PARTICULAR PURPOSE. See the GNU Lesser General
12
  Public License for more details.
13
14
  You should have received a copy of the GNU Lesser General Public
15
  License along with avahi; if not, write to the Free Software
16
  Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307
17
  USA.
18
***/
19
20
#ifdef HAVE_CONFIG_H
21
#include <config.h>
22
#endif
23
24
#include <limits.h>
25
#include <string.h>
26
#include <stdlib.h>
27
#include <ctype.h>
28
#include <assert.h>
29
30
#include "alternative.h"
31
#include "malloc.h"
32
#include "domain.h"
33
#include "utf8.h"
34
35
1.10k
static void drop_incomplete_utf8(char *c) {
36
1.10k
    char *e;
37
38
1.10k
    e = strchr(c, 0) - 1;
39
40
1.18k
    while (e >= c) {
41
42
1.04k
        if (avahi_utf8_valid(c))
43
973
            break;
44
45
1.04k
        assert(*e & 128);
46
73
        *e = 0;
47
48
73
        e--;
49
73
    }
50
1.10k
}
51
52
937
char *avahi_alternative_host_name(const char *s) {
53
937
    char label[AVAHI_LABEL_MAX], alternative[AVAHI_LABEL_MAX*4+1];
54
937
    char *alt, *r, *ret;
55
937
    const char *e;
56
937
    size_t len;
57
58
937
    assert(s);
59
60
937
    if (!avahi_is_valid_host_name(s))
61
524
        return NULL;
62
63
413
    if (!avahi_unescape_label(&s, label, sizeof(label)))
64
0
        return NULL;
65
66
413
    if ((e = strrchr(label, '-'))) {
67
111
        const char *p;
68
69
111
        e++;
70
71
732
        for (p = e; *p; p++)
72
640
            if (!isdigit(*p)) {
73
19
                e = NULL;
74
19
                break;
75
19
            }
76
77
111
        if (e && (*e == '0' || *e == 0))
78
7
            e = NULL;
79
111
    }
80
81
413
    if (e) {
82
85
        char *c, *m;
83
85
        int n;
84
85
85
        n = atoi(e);
86
85
        if (n == INT_MAX)
87
2
            n = 1;
88
83
        else
89
83
            n++;
90
85
        if (!(m = avahi_strdup_printf("%i", n)))
91
0
            return NULL;
92
93
85
        len = e-label-1;
94
95
85
        if (len >= AVAHI_LABEL_MAX-1-strlen(m)-1)
96
7
            len = AVAHI_LABEL_MAX-1-strlen(m)-1;
97
98
85
        if (!(c = avahi_strndup(label, len))) {
99
0
            avahi_free(m);
100
0
            return NULL;
101
0
        }
102
103
85
        drop_incomplete_utf8(c);
104
105
85
        r = avahi_strdup_printf("%s-%s", c, m);
106
85
        avahi_free(c);
107
85
        avahi_free(m);
108
109
328
    } else {
110
328
        char *c;
111
112
328
        if (!(c = avahi_strndup(label, AVAHI_LABEL_MAX-1-2)))
113
0
            return NULL;
114
115
328
        drop_incomplete_utf8(c);
116
117
328
        r = avahi_strdup_printf("%s-2", c);
118
328
        avahi_free(c);
119
328
    }
120
121
413
    alt = alternative;
122
413
    len = sizeof(alternative);
123
413
    ret = avahi_escape_label(r, strlen(r), &alt, &len);
124
125
413
    avahi_free(r);
126
413
    r = avahi_strdup(ret);
127
128
413
    assert(avahi_is_valid_host_name(r));
129
130
413
    return r;
131
413
}
132
133
937
char *avahi_alternative_service_name(const char *s) {
134
937
    const char *e;
135
937
    char *r;
136
137
937
    assert(s);
138
139
937
    if (!avahi_is_valid_service_name(s))
140
243
        return NULL;
141
142
694
    if ((e = strstr(s, " #"))) {
143
112
        const char *n, *p;
144
112
        e += 2;
145
146
172
        while ((n = strstr(e, " #")))
147
60
            e = n + 2;
148
149
738
        for (p = e; *p; p++)
150
643
            if (!isdigit(*p)) {
151
17
                e = NULL;
152
17
                break;
153
17
            }
154
155
112
        if (e && (*e == '0' || *e == 0))
156
10
            e = NULL;
157
112
    }
158
159
694
    if (e) {
160
85
        char *c, *m;
161
85
        size_t l;
162
85
        int n;
163
164
85
        n = atoi(e);
165
85
        if (n == INT_MAX)
166
2
            n = 1;
167
83
        else
168
83
            n++;
169
85
        if (!(m = avahi_strdup_printf("%i", n)))
170
0
            return NULL;
171
172
85
        l = e-s-2;
173
174
85
        if (l >= AVAHI_LABEL_MAX-1-strlen(m)-2)
175
7
            l = AVAHI_LABEL_MAX-1-strlen(m)-2;
176
177
85
        if (!(c = avahi_strndup(s, l))) {
178
0
            avahi_free(m);
179
0
            return NULL;
180
0
        }
181
182
85
        drop_incomplete_utf8(c);
183
184
85
        r = avahi_strdup_printf("%s #%s", c, m);
185
85
        avahi_free(c);
186
85
        avahi_free(m);
187
609
    } else {
188
609
        char *c;
189
190
609
        if (!(c = avahi_strndup(s, AVAHI_LABEL_MAX-1-3)))
191
0
            return NULL;
192
193
609
        drop_incomplete_utf8(c);
194
195
609
        r = avahi_strdup_printf("%s #2", c);
196
609
        avahi_free(c);
197
609
    }
198
199
694
    assert(avahi_is_valid_service_name(r));
200
201
694
    return r;
202
694
}