summaryrefslogtreecommitdiff
path: root/binutils
diff options
context:
space:
mode:
authorAlan Modra <amodra@gmail.com>2021-05-15 14:36:26 +0930
committerAlan Modra <amodra@gmail.com>2021-05-15 14:36:54 +0930
commitfc5e0925d4bff79c8c036cf00803112a1ec04188 (patch)
treed71d622247fdaadaa710ece3fdce8bfb9ffc2289 /binutils
parent7c96e6120f1b9b5025629bbe995ca55d1be8f36f (diff)
_mul_overflow and get_encoded_value
A sufficiently mad compiler optimiser can take undefined behaviour according to the C standard as an opportunity to remove code. Since "data + size" might be seen to be past the end of an array, calculating such an expression is UB. _mul_overflow is infrastructure for later patches. * bucomm.h (_mul_overflow): Define. * dwarf.c (get_encoded_value): Avoid pointer UB.
Diffstat (limited to 'binutils')
-rw-r--r--binutils/ChangeLog5
-rw-r--r--binutils/bucomm.h8
-rw-r--r--binutils/dwarf.c2
3 files changed, 14 insertions, 1 deletions
diff --git a/binutils/ChangeLog b/binutils/ChangeLog
index 85d21ebfa6..74efc3376e 100644
--- a/binutils/ChangeLog
+++ b/binutils/ChangeLog
@@ -1,3 +1,8 @@
+2021-05-15 Alan Modra <amodra@gmail.com>
+
+ * bucomm.h (_mul_overflow): Define.
+ * dwarf.c (get_encoded_value): Avoid pointer UB.
+
2021-05-13 Alan Modra <amodra@gmail.com>
PR 27861
diff --git a/binutils/bucomm.h b/binutils/bucomm.h
index 78f61762ca..2769c27867 100644
--- a/binutils/bucomm.h
+++ b/binutils/bucomm.h
@@ -80,4 +80,12 @@ void *xmalloc (size_t);
void *xrealloc (void *, size_t);
+#if __GNUC__ >= 7
+#define _mul_overflow(a, b, res) __builtin_mul_overflow (a, b, res)
+#else
+/* Assumes unsigned values. Careful! Args evaluated multiple times. */
+#define _mul_overflow(a, b, res) \
+ ((*res) = (a), (*res) *= (b), (b) != 0 && (*res) / (b) != (a))
+#endif
+
#endif /* _BUCOMM_H */
diff --git a/binutils/dwarf.c b/binutils/dwarf.c
index 2794a15a1d..020b7e071e 100644
--- a/binutils/dwarf.c
+++ b/binutils/dwarf.c
@@ -178,7 +178,7 @@ get_encoded_value (unsigned char **pdata,
unsigned int size = size_of_encoded_value (encoding);
dwarf_vma val;
- if (data + size >= end)
+ if (data >= end || size > (size_t) (end - data))
{
warn (_("Encoded value extends past end of section\n"));
* pdata = end;