summaryrefslogtreecommitdiff
path: root/libavutil/mem.c
diff options
context:
space:
mode:
authorAnton Khirnov <anton@khirnov.net>2021-05-31 11:25:26 +0200
committerAnton Khirnov <anton@khirnov.net>2021-06-11 19:42:47 +0200
commit580e168a945b65100ec2c25433f33bfacfe9f7be (patch)
tree411a45b685ad4815cd06e5a992719fa9387dfcd4 /libavutil/mem.c
parent71c0ef882e37091b22bae2175efb2d5cc4225d8d (diff)
lavu/mem: un-inline av_size_mult()
There seems to be no compelling reason for it to be inline.
Diffstat (limited to 'libavutil/mem.c')
-rw-r--r--libavutil/mem.c18
1 files changed, 18 insertions, 0 deletions
diff --git a/libavutil/mem.c b/libavutil/mem.c
index a52d33d4a6..063635fb22 100644
--- a/libavutil/mem.c
+++ b/libavutil/mem.c
@@ -547,3 +547,21 @@ void av_fast_mallocz(void *ptr, unsigned int *size, size_t min_size)
{
fast_malloc(ptr, size, min_size, 1);
}
+
+int av_size_mult(size_t a, size_t b, size_t *r)
+{
+ size_t t;
+
+#if (!defined(__INTEL_COMPILER) && AV_GCC_VERSION_AT_LEAST(5,1)) || AV_HAS_BUILTIN(__builtin_mul_overflow)
+ if (__builtin_mul_overflow(a, b, &t))
+ return AVERROR(EINVAL);
+#else
+ t = a * b;
+ /* Hack inspired from glibc: don't try the division if nelem and elsize
+ * are both less than sqrt(SIZE_MAX). */
+ if ((a | b) >= ((size_t)1 << (sizeof(size_t) * 4)) && a && t / a != b)
+ return AVERROR(EINVAL);
+#endif
+ *r = t;
+ return 0;
+}