mirror of
https://git.haproxy.org/git/haproxy.git/
synced 2025-11-24 12:20:59 +01:00
CLEANUP: include: replace hand-rolled offsetof to avoid UB
The C standard specifies that it's undefined behavior to dereference
NULL (even if you use & right after). The hand-rolled offsetof idiom
&(((s*)NULL)->f) is thus technically undefined. This clutters the
output of UBSan and is simple to fix: just use the real offsetof when
it's available.
Note that there's no clear statement about this point in the spec,
only several points which together converge to this:
- From N3220, 6.5.3.4:
A postfix expression followed by the -> operator and an identifier
designates a member of a structure or union object. The value is
that of the named member of the object to which the first expression
points, and is an lvalue.
- From N3220, 6.3.2.1:
An lvalue is an expression (with an object type other than void) that
potentially designates an object; if an lvalue does not designate an
object when it is evaluated, the behavior is undefined.
- From N3220, 6.5.4.4 p3:
The unary & operator yields the address of its operand. If the
operand has type "type", the result has type "pointer to type". If
the operand is the result of a unary * operator, neither that operator
nor the & operator is evaluated and the result is as if both were
omitted, except that the constraints on the operators still apply and
the result is not an lvalue. Similarly, if the operand is the result
of a [] operator, neither the & operator nor the unary * that is
implied by the [] is evaluated and the result is as if the & operator
were removed and the [] operator were changed to a + operator.
=> In short, this is saying that C guarantees these identities:
1. &(*p) is equivalent to p
2. &(p[n]) is equivalent to p + n
As a consequence, &(*p) doesn't result in the evaluation of *p, only
the evaluation of p (and similar for []). There is no corresponding
special carve-out for ->.
See also: https://pvs-studio.com/en/blog/posts/cpp/0306/
After this patch, HAProxy can run without crashing after building w/
clang-19 -fsanitize=undefined -fno-sanitize=function,alignment
This commit is contained in:
parent
d3b46cca7b
commit
1e48ec7f6c
@ -350,7 +350,7 @@
|
|||||||
* <type> which has its member <name> stored at address <ptr>.
|
* <type> which has its member <name> stored at address <ptr>.
|
||||||
*/
|
*/
|
||||||
#ifndef container_of
|
#ifndef container_of
|
||||||
#define container_of(ptr, type, name) ((type *)(((char *)(ptr)) - ((long)&((type *)0)->name)))
|
#define container_of(ptr, type, name) ((type *)(((char *)(ptr)) - offsetof(type, name)))
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
/* returns a pointer to the structure of type <type> which has its member <name>
|
/* returns a pointer to the structure of type <type> which has its member <name>
|
||||||
@ -359,7 +359,7 @@
|
|||||||
#ifndef container_of_safe
|
#ifndef container_of_safe
|
||||||
#define container_of_safe(ptr, type, name) \
|
#define container_of_safe(ptr, type, name) \
|
||||||
({ void *__p = (ptr); \
|
({ void *__p = (ptr); \
|
||||||
__p ? (type *)((char *)__p - ((long)&((type *)0)->name)) : (type *)0; \
|
__p ? (type *)((char *)__p - offsetof(type, name)) : (type *)0; \
|
||||||
})
|
})
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
|||||||
@ -97,7 +97,7 @@
|
|||||||
* since it's used only once.
|
* since it's used only once.
|
||||||
* Example: LIST_ELEM(cur_node->args.next, struct node *, args)
|
* Example: LIST_ELEM(cur_node->args.next, struct node *, args)
|
||||||
*/
|
*/
|
||||||
#define LIST_ELEM(lh, pt, el) ((pt)(((const char *)(lh)) - ((size_t)&((pt)NULL)->el)))
|
#define LIST_ELEM(lh, pt, el) ((pt)(((const char *)(lh)) - offsetof(typeof(*(pt)NULL), el)))
|
||||||
|
|
||||||
/* checks if the list head <lh> is empty or not */
|
/* checks if the list head <lh> is empty or not */
|
||||||
#define LIST_ISEMPTY(lh) ((lh)->n == (lh))
|
#define LIST_ISEMPTY(lh) ((lh)->n == (lh))
|
||||||
|
|||||||
@ -87,7 +87,7 @@ struct mt_list {
|
|||||||
*
|
*
|
||||||
* return MT_LIST_ELEM(cur_node->args.next, struct node *, args)
|
* return MT_LIST_ELEM(cur_node->args.next, struct node *, args)
|
||||||
*/
|
*/
|
||||||
#define MT_LIST_ELEM(a, t, m) ((t)(size_t)(((size_t)(a)) - ((size_t)&((t)NULL)->m)))
|
#define MT_LIST_ELEM(a, t, m) ((t)(size_t)(((size_t)(a)) - offsetof(typeof(*(t)NULL), m)))
|
||||||
|
|
||||||
|
|
||||||
/* Returns a pointer of type <t> to a structure following the element which
|
/* Returns a pointer of type <t> to a structure following the element which
|
||||||
|
|||||||
Loading…
x
Reference in New Issue
Block a user