//===-- 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 #include #include namespace LIBC_NAMESPACE_DECL { namespace internal { template 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 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 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 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 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 class StructArgList { void *ptr; void *end; public: LIBC_INLINE StructArgList(void *ptr, size_t size) : ptr(ptr), end(reinterpret_cast(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 LIBC_INLINE T next_var() { if (!packed) ptr = reinterpret_cast( align_up(reinterpret_cast(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(reinterpret_cast(ptr) + sizeof(T)); return val; } }; } // namespace internal } // namespace LIBC_NAMESPACE_DECL #endif // LLVM_LIBC_SRC___SUPPORT_ARG_LIST_H