aboutsummaryrefslogtreecommitdiff
diff options
context:
space:
mode:
-rw-r--r--src/libexpr/eval.cc5
-rw-r--r--src/libexpr/gc-alloc.hh26
2 files changed, 28 insertions, 3 deletions
diff --git a/src/libexpr/eval.cc b/src/libexpr/eval.cc
index 06b1f27f5..c0e7a9a2e 100644
--- a/src/libexpr/eval.cc
+++ b/src/libexpr/eval.cc
@@ -777,8 +777,7 @@ static void copyContextToValue(Value & v, const NixStringContext & context)
{
if (!context.empty()) {
size_t n = 0;
- v.string.context = (const char * *)
- gcAllocBytes((context.size() + 1) * sizeof(char *));
+ v.string.context = gcAllocType<char const *>(context.size() + 1);
for (auto & i : context)
v.string.context[n++] = gcCopyStringIfNeeded(i.to_string());
v.string.context[n] = 0;
@@ -834,7 +833,7 @@ void EvalState::mkList(Value & v, size_t size)
{
v.mkList(size);
if (size > 2)
- v.bigList.elems = (Value * *) gcAllocBytes(size * sizeof(Value *));
+ v.bigList.elems = gcAllocType<Value *>(size);
nrListElems += size;
}
diff --git a/src/libexpr/gc-alloc.hh b/src/libexpr/gc-alloc.hh
index f691bfc4b..04ac28ea8 100644
--- a/src/libexpr/gc-alloc.hh
+++ b/src/libexpr/gc-alloc.hh
@@ -18,6 +18,8 @@
#include <gc/gc_allocator.h>
#include <gc/gc_cpp.h>
+#include "checked-arithmetic.hh"
+
/// calloc, transparently GC-enabled.
#define LIX_GC_CALLOC(size) GC_MALLOC(size)
@@ -104,6 +106,30 @@ inline void * gcAllocBytes(size_t n)
return ptr;
}
+/// Typed, safe wrapper around calloc() (transparently GC-enabled). Allocates
+/// enough for the requested count of the specified type. Also checks for
+/// nullptr (and throws @ref std::bad_alloc), and casts the void pointer to
+/// a pointer of the specified type, for type-convenient goodness.
+template<typename T>
+[[gnu::always_inline]]
+inline T * gcAllocType(size_t howMany = 1)
+{
+ // NOTE: size_t * size_t, which can definitely overflow.
+ // Unsigned integer overflow is definitely a bug, but isn't undefined
+ // behavior, so we can just check if we overflowed after the fact.
+ // However, people can and do request zero sized allocations, so we need
+ // to check that neither of our multiplicands were zero before complaining
+ // about it.
+ auto checkedSz = checked::Checked<size_t>(howMany) * sizeof(T);
+ size_t sz = checkedSz.valueWrapping();
+ if (checkedSz.overflowed()) {
+ // Congrats, you done did an overflow.
+ throw std::bad_alloc();
+ }
+
+ return static_cast<T *>(gcAllocBytes(sz));
+}
+
/// GC-transparently allocates a buffer for a C-string of @ref size *bytes*,
/// meaning you should include the size needed by the NUL terminator in the
/// passed size. Memory allocated with this function must never contain other