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
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
|
/*
* Copyright (c) 2018 Eugene Syromyatnikov <evgsyr@gmail.com>
* All rights reserved.
*
* SPDX-License-Identifier: LGPL-2.1-or-later
*/
#ifndef STRACE_XSTRING_H
# define STRACE_XSTRING_H
# include <stdarg.h>
# include <stdio.h>
# include "error_prints.h"
# include "gcc_compat.h"
/**
* Print to static buffer and die on (really unexpected) errors and overflows.
* Shouldn't be used directly; please refer to helper macros xsnprintf and
* xsprint instead.
*
* @param str String buffer to print into.
* @param size Size of the string buffer in bytes.
* @param func Function name from which this function is called.
* @param argstr Stringified arguments (including format argument).
* @param format Format string.
* @param ... Format arguments.
* @return Number of characters printed, excluding terminating null byte
* (the same as s(n)printf).
*/
static inline int ATTRIBUTE_FORMAT((printf, 5, 6))
xsnprintf_(char *str, size_t size, const char *func, const char *argstr,
const char *format, ...)
{
int ret;
va_list ap;
va_start(ap, format);
ret = vsnprintf(str, size, format, ap);
va_end(ap);
if (ret < 0 || (unsigned int) ret >= size)
error_msg_and_die("%s: got unexpected return value %d for "
"snprintf(buf, %zu, %s)",
func, ret, size, argstr);
return ret;
}
/**
* snprintf that dies on (really unexpected) errors and overflows.
*
* @param str_ String buffer to print into.
* @param size_ Size of the string buffer in bytes.
* @param fmt_ Format string.
* @param ... Format arguments.
*/
# define xsnprintf(str_, size_, fmt_, ...) \
xsnprintf_((str_), (size_), __func__, #fmt_ ", " #__VA_ARGS__, \
(fmt_), __VA_ARGS__)
/**
* Print to a character array buffer and die on (really unexpected) errors and
* overflows. Buffer size is obtained with sizeof().
*
* @param str_ Character array buffer to print into.
* @param fmt_ Format string.
* @param ... Format arguments.
*/
# define xsprintf(str_, fmt_, ...) \
xsnprintf((str_), sizeof(str_) + MUST_BE_ARRAY(str_), (fmt_), \
__VA_ARGS__)
static inline size_t
get_pos_diff_(char *str, size_t size, char *pos, const char *func,
const char *call)
{
if ((str + size) < str)
error_msg_and_die("%s: string size overflow (%p+%zu) in %s",
func, str, size, call);
if (pos > (str + size))
error_msg_and_die("%s: got position (%p) beyond string "
"(%p+%zu) in %s",
func, pos, str, size, call);
if (pos < str)
error_msg_and_die("%s: got position %p before string %p in %s",
func, pos, str, call);
return pos - str;
}
/**
* Helper function for constructing string in a character array by appending
* new formatted parts. Returns new position. Fails on error or buffer
* overflow, in line with the rest of x* functions. Obtains buffer size via
* sizeof(str_).
*
* @param str_ Character array buffer to print into.
* @param pos_ Current position.
* @param fmt_ Format string.
* @param ... Format arguments.
* @return New position.
*/
# define xappendstr(str_, pos_, fmt_, ...) \
(xsnprintf((pos_), sizeof(str_) + MUST_BE_ARRAY(str_) - \
get_pos_diff_((str_), sizeof(str_), (pos_), __func__, \
"xappendstr(" #str_ ", " #pos_ ", " #fmt_ ", " \
#__VA_ARGS__ ")"), \
(fmt_), ##__VA_ARGS__) + (pos_))
#endif /* !STRACE_XSTRING_H */
|