1*cf84ac9aSAndroid Build Coastguard Worker #ifndef STRACE_XSTRING_H
2*cf84ac9aSAndroid Build Coastguard Worker #define STRACE_XSTRING_H
3*cf84ac9aSAndroid Build Coastguard Worker
4*cf84ac9aSAndroid Build Coastguard Worker #include <stdarg.h>
5*cf84ac9aSAndroid Build Coastguard Worker #include <stdio.h>
6*cf84ac9aSAndroid Build Coastguard Worker
7*cf84ac9aSAndroid Build Coastguard Worker #include "error_prints.h"
8*cf84ac9aSAndroid Build Coastguard Worker #include "gcc_compat.h"
9*cf84ac9aSAndroid Build Coastguard Worker
10*cf84ac9aSAndroid Build Coastguard Worker /**
11*cf84ac9aSAndroid Build Coastguard Worker * Print to static buffer and die on (really unexpected) errors and overflows.
12*cf84ac9aSAndroid Build Coastguard Worker * Shouldn't be used directly; please refer to helper macros xsnprintf and
13*cf84ac9aSAndroid Build Coastguard Worker * xsprint instead.
14*cf84ac9aSAndroid Build Coastguard Worker *
15*cf84ac9aSAndroid Build Coastguard Worker * @param str String buffer to print into.
16*cf84ac9aSAndroid Build Coastguard Worker * @param size Size of the string buffer in bytes.
17*cf84ac9aSAndroid Build Coastguard Worker * @param func Function name from which this function is called.
18*cf84ac9aSAndroid Build Coastguard Worker * @param argstr Stringified arguments (including format argument).
19*cf84ac9aSAndroid Build Coastguard Worker * @param format Format string.
20*cf84ac9aSAndroid Build Coastguard Worker * @param ... Format arguments.
21*cf84ac9aSAndroid Build Coastguard Worker * @return Number of characters printed, excluding terminating null byte
22*cf84ac9aSAndroid Build Coastguard Worker * (the same as s(n)printf).
23*cf84ac9aSAndroid Build Coastguard Worker */
24*cf84ac9aSAndroid Build Coastguard Worker static inline int ATTRIBUTE_FORMAT((printf, 5, 6))
xsnprintf_(char * str,size_t size,const char * func,const char * argstr,const char * format,...)25*cf84ac9aSAndroid Build Coastguard Worker xsnprintf_(char *str, size_t size, const char *func, const char *argstr,
26*cf84ac9aSAndroid Build Coastguard Worker const char *format, ...)
27*cf84ac9aSAndroid Build Coastguard Worker {
28*cf84ac9aSAndroid Build Coastguard Worker int ret;
29*cf84ac9aSAndroid Build Coastguard Worker va_list ap;
30*cf84ac9aSAndroid Build Coastguard Worker
31*cf84ac9aSAndroid Build Coastguard Worker va_start(ap, format);
32*cf84ac9aSAndroid Build Coastguard Worker ret = vsnprintf(str, size, format, ap);
33*cf84ac9aSAndroid Build Coastguard Worker va_end(ap);
34*cf84ac9aSAndroid Build Coastguard Worker
35*cf84ac9aSAndroid Build Coastguard Worker if (ret < 0 || (unsigned int) ret >= size)
36*cf84ac9aSAndroid Build Coastguard Worker error_msg_and_die("%s: got unexpected return value %d for "
37*cf84ac9aSAndroid Build Coastguard Worker "snprintf(buf, %zu, %s)",
38*cf84ac9aSAndroid Build Coastguard Worker func, ret, size, argstr);
39*cf84ac9aSAndroid Build Coastguard Worker
40*cf84ac9aSAndroid Build Coastguard Worker return ret;
41*cf84ac9aSAndroid Build Coastguard Worker }
42*cf84ac9aSAndroid Build Coastguard Worker
43*cf84ac9aSAndroid Build Coastguard Worker /**
44*cf84ac9aSAndroid Build Coastguard Worker * snprintf that dies on (really unexpected) errors and overflows.
45*cf84ac9aSAndroid Build Coastguard Worker *
46*cf84ac9aSAndroid Build Coastguard Worker * @param str_ String buffer to print into.
47*cf84ac9aSAndroid Build Coastguard Worker * @param size_ Size of the string buffer in bytes.
48*cf84ac9aSAndroid Build Coastguard Worker * @param fmt_ Format string.
49*cf84ac9aSAndroid Build Coastguard Worker * @param ... Format arguments.
50*cf84ac9aSAndroid Build Coastguard Worker */
51*cf84ac9aSAndroid Build Coastguard Worker #define xsnprintf(str_, size_, fmt_, ...) \
52*cf84ac9aSAndroid Build Coastguard Worker xsnprintf_((str_), (size_), __func__, #fmt_ ", " #__VA_ARGS__, \
53*cf84ac9aSAndroid Build Coastguard Worker (fmt_), __VA_ARGS__)
54*cf84ac9aSAndroid Build Coastguard Worker
55*cf84ac9aSAndroid Build Coastguard Worker /**
56*cf84ac9aSAndroid Build Coastguard Worker * Print to a character array buffer and die on (really unexpected) errors and
57*cf84ac9aSAndroid Build Coastguard Worker * overflows. Buffer size is obtained with sizeof().
58*cf84ac9aSAndroid Build Coastguard Worker *
59*cf84ac9aSAndroid Build Coastguard Worker * @param str_ Character array buffer to print into.
60*cf84ac9aSAndroid Build Coastguard Worker * @param fmt_ Format string.
61*cf84ac9aSAndroid Build Coastguard Worker * @param ... Format arguments.
62*cf84ac9aSAndroid Build Coastguard Worker */
63*cf84ac9aSAndroid Build Coastguard Worker #define xsprintf(str_, fmt_, ...) \
64*cf84ac9aSAndroid Build Coastguard Worker xsnprintf((str_), sizeof(str_) + MUST_BE_ARRAY(str_), (fmt_), \
65*cf84ac9aSAndroid Build Coastguard Worker __VA_ARGS__)
66*cf84ac9aSAndroid Build Coastguard Worker
67*cf84ac9aSAndroid Build Coastguard Worker static inline size_t
get_pos_diff_(char * str,size_t size,char * pos,const char * func,const char * call)68*cf84ac9aSAndroid Build Coastguard Worker get_pos_diff_(char *str, size_t size, char *pos, const char *func,
69*cf84ac9aSAndroid Build Coastguard Worker const char *call)
70*cf84ac9aSAndroid Build Coastguard Worker {
71*cf84ac9aSAndroid Build Coastguard Worker if ((str + size) < str)
72*cf84ac9aSAndroid Build Coastguard Worker error_msg_and_die("%s: string size overflow (%p+%zu) in %s",
73*cf84ac9aSAndroid Build Coastguard Worker func, str, size, call);
74*cf84ac9aSAndroid Build Coastguard Worker
75*cf84ac9aSAndroid Build Coastguard Worker if (pos > (str + size))
76*cf84ac9aSAndroid Build Coastguard Worker error_msg_and_die("%s: got position (%p) beyond string "
77*cf84ac9aSAndroid Build Coastguard Worker "(%p+%zu) in %s",
78*cf84ac9aSAndroid Build Coastguard Worker func, pos, str, size, call);
79*cf84ac9aSAndroid Build Coastguard Worker
80*cf84ac9aSAndroid Build Coastguard Worker if (pos < str)
81*cf84ac9aSAndroid Build Coastguard Worker error_msg_and_die("%s: got position %p before string %p in %s",
82*cf84ac9aSAndroid Build Coastguard Worker func, pos, str, call);
83*cf84ac9aSAndroid Build Coastguard Worker
84*cf84ac9aSAndroid Build Coastguard Worker return pos - str;
85*cf84ac9aSAndroid Build Coastguard Worker }
86*cf84ac9aSAndroid Build Coastguard Worker
87*cf84ac9aSAndroid Build Coastguard Worker /**
88*cf84ac9aSAndroid Build Coastguard Worker * Helper function for constructing string in a character array by appending
89*cf84ac9aSAndroid Build Coastguard Worker * new formatted parts. Returns new position. Fails on error or buffer
90*cf84ac9aSAndroid Build Coastguard Worker * overflow, in line with the rest of x* functions. Obtains buffer size via
91*cf84ac9aSAndroid Build Coastguard Worker * sizeof(str_).
92*cf84ac9aSAndroid Build Coastguard Worker *
93*cf84ac9aSAndroid Build Coastguard Worker * @param str_ Character array buffer to print into.
94*cf84ac9aSAndroid Build Coastguard Worker * @param pos_ Current position.
95*cf84ac9aSAndroid Build Coastguard Worker * @param fmt_ Format string.
96*cf84ac9aSAndroid Build Coastguard Worker * @param ... Format arguments.
97*cf84ac9aSAndroid Build Coastguard Worker * @return New position.
98*cf84ac9aSAndroid Build Coastguard Worker */
99*cf84ac9aSAndroid Build Coastguard Worker #define xappendstr(str_, pos_, fmt_, ...) \
100*cf84ac9aSAndroid Build Coastguard Worker (xsnprintf((pos_), sizeof(str_) + MUST_BE_ARRAY(str_) - \
101*cf84ac9aSAndroid Build Coastguard Worker get_pos_diff_((str_), sizeof(str_), (pos_), __func__, \
102*cf84ac9aSAndroid Build Coastguard Worker "xappendstr(" #str_ ", " #pos_ ", " #fmt_ ", " \
103*cf84ac9aSAndroid Build Coastguard Worker #__VA_ARGS__ ")"), \
104*cf84ac9aSAndroid Build Coastguard Worker (fmt_), ##__VA_ARGS__) + (pos_))
105*cf84ac9aSAndroid Build Coastguard Worker
106*cf84ac9aSAndroid Build Coastguard Worker #endif /* !STRACE_XSTRING_H */
107