-
Notifications
You must be signed in to change notification settings - Fork 17
/
concat.c
70 lines (62 loc) · 1.38 KB
/
concat.c
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
/*
* concat() - allocate memory and safely concatenate strings in portable C
* (and C++ if you like).
*
* This code deals gracefully with potential integer overflows (perhaps when
* input strings are maliciously long), as well as with input strings changing
* from under it (perhaps because of misbehavior of another thread). It does
* not depend on non-portable functions such as snprintf() and asprintf().
*
* Written by Solar Designer <solar at openwall.com> and placed in the
* public domain.
*/
#include <string.h>
#include <stdarg.h>
#include <stdlib.h>
#include <limits.h>
#include "concat.h"
char *concat(const char *s1, ...)
{
va_list args;
const char *s;
char *p, *result;
size_t l, m, n;
m = n = strlen(s1);
va_start(args, s1);
while ((s = va_arg(args, char *))) {
l = strlen(s);
if ((m += l) < l)
break;
}
va_end(args);
if (s || m >= INT_MAX)
return NULL;
result = (char *)malloc(m + 1);
if (!result)
return NULL;
memcpy(p = result, s1, n);
p += n;
va_start(args, s1);
while ((s = va_arg(args, char *))) {
l = strlen(s);
if ((n += l) < l || n > m)
break;
memcpy(p, s, l);
p += l;
}
va_end(args);
if (s || m != n || p != result + n) {
free(result);
return NULL;
}
*p = 0;
return result;
}
#ifdef TEST
#include <stdio.h>
int main(int argc, char **argv)
{
puts(concat(argv[0], argv[1], argv[2], argv[3], NULL));
return 0;
}
#endif