[libc] Add holder class for va_lists

This class is intended to be used in cases where a class is being used
on a va_list. It provides destruction and copy semantics with small
overhead. This is intended to be used in printf.

Reviewed By: sivachandra, lntue

Differential Revision: https://reviews.llvm.org/D123061
This commit is contained in:
Michael Jones 2022-04-04 11:47:22 -07:00
parent f9e0bc37db
commit 5561ab3495
4 changed files with 105 additions and 0 deletions

View File

@ -54,6 +54,12 @@ add_header_library(
integer_operations.h
)
add_header_library(
arg_list
HDRS
arg_list.h
)
# Thread support is used by other support libraries. So, we add the "threads"
# before other directories.
add_subdirectory(threads)

View File

@ -0,0 +1,31 @@
//===-- 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 <stdarg.h>
namespace __llvm_libc {
namespace internal {
class ArgList {
va_list vlist;
public:
ArgList(va_list vlist) { va_copy(this->vlist, vlist); }
ArgList(ArgList &other) { va_copy(this->vlist, other.vlist); }
~ArgList() { va_end(this->vlist); }
template <class T> T inline next_var() { return va_arg(vlist, T); }
};
} // namespace internal
} // namespace __llvm_libc
#endif // LLVM_LIBC_SRC_SUPPORT_ARG_LIST_H

View File

@ -30,6 +30,16 @@ add_libc_unittest(
libc.src.__support.str_to_float
)
add_libc_unittest(
arg_list_test
SUITE
libc_support_unittests
SRCS
arg_list_test.cpp
DEPENDS
libc.src.__support.arg_list
)
add_executable(
libc_str_to_float_comparison_test
str_to_float_comparison_test.cpp

View File

@ -0,0 +1,58 @@
//===-- Unittests for ArgList ---------------------------------------------===//
//
// 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
//
//===----------------------------------------------------------------------===//
#include "src/__support/arg_list.h"
#include "utils/UnitTest/Test.h"
int get_nth_int(int n, ...) {
va_list vlist;
va_start(vlist, n);
__llvm_libc::internal::ArgList v(vlist);
va_end(vlist);
for (int i = 0; i < n; ++i) {
v.next_var<int>();
}
return v.next_var<int>();
}
TEST(LlvmLibcArgListTest, BasicUsage) {
ASSERT_EQ(get_nth_int(5, 0, 10, 20, 30, 40, 50, 60, 70, 80, 90), 50);
}
int sum_two_nums(int first, int second, ...) {
va_list vlist;
va_start(vlist, second);
__llvm_libc::internal::ArgList v1(vlist);
va_end(vlist);
__llvm_libc::internal::ArgList v2 = v1;
int first_val;
for (int i = 0; i < first; ++i) {
v1.next_var<int>();
}
first_val = v1.next_var<int>();
int second_val;
for (int i = 0; i < second; ++i) {
v2.next_var<int>();
}
second_val = v2.next_var<int>();
return first_val + second_val;
}
TEST(LlvmLibcArgListTest, CopyConstructor) {
ASSERT_EQ(sum_two_nums(3, 1, 1, 2, 4, 8, 16, 32, 64, 128, 256, 512, 1024),
10);
ASSERT_EQ(sum_two_nums(3, 5, 1, 2, 4, 8, 16, 32, 64, 128, 256, 512, 1024),
40);
}