blob: 66afa67e320b4f6ad75d8ff2e185c2de1e105c4d [file] [log] [blame]
//===-- Holder Class for manipulating va_lists ------------------*- C++ -*-===//
//
// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
// See https://llvm.org/LICENSE.txt for license information.
// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
//
//===----------------------------------------------------------------------===//
#ifndef LLVM_LIBC_SRC___SUPPORT_ARG_LIST_H
#define LLVM_LIBC_SRC___SUPPORT_ARG_LIST_H
#include "src/__support/common.h"
#include "src/__support/macros/config.h"
#include <stdarg.h>
#include <stddef.h>
#include <stdint.h>
namespace LIBC_NAMESPACE_DECL {
namespace internal {
template <typename V, typename A>
LIBC_INLINE constexpr V align_up(V val, A align) {
return ((val + V(align) - 1) / V(align)) * V(align);
}
class ArgList {
va_list vlist;
public:
LIBC_INLINE ArgList(va_list vlist) { va_copy(this->vlist, vlist); }
LIBC_INLINE ArgList(ArgList &other) { va_copy(this->vlist, other.vlist); }
LIBC_INLINE ~ArgList() { va_end(this->vlist); }
LIBC_INLINE ArgList &operator=(ArgList &rhs) {
va_copy(vlist, rhs.vlist);
return *this;
}
template <class T> LIBC_INLINE T next_var() { return va_arg(vlist, T); }
};
// Used for testing things that use an ArgList when it's impossible to know what
// the arguments should be ahead of time. An example of this would be fuzzing,
// since a function passed a random input could request unpredictable arguments.
class MockArgList {
size_t arg_counter = 0;
public:
LIBC_INLINE MockArgList() = default;
LIBC_INLINE MockArgList(va_list) { ; }
LIBC_INLINE MockArgList(MockArgList &other) {
arg_counter = other.arg_counter;
}
LIBC_INLINE ~MockArgList() = default;
LIBC_INLINE MockArgList &operator=(MockArgList &rhs) {
arg_counter = rhs.arg_counter;
return *this;
}
template <class T> LIBC_INLINE T next_var() {
arg_counter++;
return T(arg_counter);
}
size_t read_count() const { return arg_counter; }
};
// Used by the GPU implementation to parse how many bytes need to be read from
// the variadic argument buffer.
template <bool packed> class DummyArgList {
size_t arg_counter = 0;
public:
LIBC_INLINE DummyArgList() = default;
LIBC_INLINE DummyArgList(va_list) { ; }
LIBC_INLINE DummyArgList(DummyArgList &other) {
arg_counter = other.arg_counter;
}
LIBC_INLINE ~DummyArgList() = default;
LIBC_INLINE DummyArgList &operator=(DummyArgList &rhs) {
arg_counter = rhs.arg_counter;
return *this;
}
template <class T> LIBC_INLINE T next_var() {
arg_counter = packed ? arg_counter + sizeof(T)
: align_up(arg_counter, alignof(T)) + sizeof(T);
return T(arg_counter);
}
size_t read_count() const { return arg_counter; }
};
// Used for the GPU implementation of `printf`. This models a variadic list as a
// simple array of pointers that are built manually by the implementation.
template <bool packed> class StructArgList {
void *ptr;
void *end;
public:
LIBC_INLINE StructArgList(void *ptr, size_t size)
: ptr(ptr), end(reinterpret_cast<unsigned char *>(ptr) + size) {}
LIBC_INLINE StructArgList(const StructArgList &other) {
ptr = other.ptr;
end = other.end;
}
LIBC_INLINE StructArgList() = default;
LIBC_INLINE ~StructArgList() = default;
LIBC_INLINE StructArgList &operator=(const StructArgList &rhs) {
ptr = rhs.ptr;
return *this;
}
LIBC_INLINE void *get_ptr() const { return ptr; }
template <class T> LIBC_INLINE T next_var() {
if (!packed)
ptr = reinterpret_cast<void *>(
align_up(reinterpret_cast<uintptr_t>(ptr), alignof(T)));
if (ptr >= end)
return T(-1);
// Memcpy because pointer alignment may be illegal given a packed struct.
T val;
__builtin_memcpy(&val, ptr, sizeof(T));
ptr =
reinterpret_cast<void *>(reinterpret_cast<uintptr_t>(ptr) + sizeof(T));
return val;
}
};
} // namespace internal
} // namespace LIBC_NAMESPACE_DECL
#endif // LLVM_LIBC_SRC___SUPPORT_ARG_LIST_H