Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Adding nameof utility function #256

Open
wants to merge 2 commits into
base: main
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
3 changes: 2 additions & 1 deletion cpp11test/src/cpp11.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -5,6 +5,7 @@
#include <Rcpp.h>
using namespace Rcpp;
#include "cpp11/declarations.hpp"
#include <R_ext/Visibility.h>

// add.cpp
SEXP cpp11_add_vec_for_(cpp11::writable::doubles x, double num);
Expand Down Expand Up @@ -444,7 +445,7 @@ static const R_CallMethodDef CallEntries[] = {
};
}

extern "C" void R_init_cpp11test(DllInfo* dll){
extern "C" attribute_visible void R_init_cpp11test(DllInfo* dll){
R_registerRoutines(dll, NULL, CallEntries, NULL, NULL);
R_useDynamicSymbols(dll, FALSE);
R_forceSymbols(dll, TRUE);
Expand Down
21 changes: 21 additions & 0 deletions cpp11test/src/test-nameof.cpp
Original file line number Diff line number Diff line change
@@ -0,0 +1,21 @@
#include "cpp11/nameof.hpp"

#include <testthat.h>

struct Global;

namespace ns {
struct Namespaced;
} // namespace ns

context("nameof-C++") {
test_that("nameof works") {
expect_true(cpp11::nameof<Global>() == "Global");
expect_true(cpp11::nameof<ns::Namespaced>() == "ns::Namespaced");
expect_true(cpp11::nameof<ns::Namespaced>(/*strip_namespace=*/true) == "Namespaced");

// namespace for local scope structs is compiler dependent
struct Local;
expect_true(cpp11::nameof<Local>(/*strip_namespace=*/true) == "Local");
}
}
82 changes: 82 additions & 0 deletions inst/include/cpp11/nameof.hpp
Original file line number Diff line number Diff line change
@@ -0,0 +1,82 @@
/// Distributed under the Boost Software License, Version 1.0. (See
/// http://www.boost.org/LICENSE_1_0.txt)
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Is this copied from a boost library ? Should this be acknowledged in the DESCRIPTION file ?


#include <array>
#include <string>
#include <tuple>
#include <utility>

namespace cpp11 {
namespace detail {

#ifdef _MSC_VER
#define ARROW_PRETTY_FUNCTION __FUNCSIG__
#else
#define ARROW_PRETTY_FUNCTION __PRETTY_FUNCTION__
#endif

template <typename T>
const char* raw() {
return ARROW_PRETTY_FUNCTION;
}

template <typename T>
size_t raw_sizeof() {
return sizeof(ARROW_PRETTY_FUNCTION);
}

#undef ARROW_PRETTY_FUNCTION

constexpr bool starts_with(char const* haystack, char const* needle) {
return needle[0] == '\0' ||
(haystack[0] == needle[0] && starts_with(haystack + 1, needle + 1));
}

constexpr size_t search(char const* haystack, char const* needle) {
return haystack[0] == '\0' || starts_with(haystack, needle)
? 0
: search(haystack + 1, needle) + 1;
}

const size_t typename_prefix = search(raw<double>(), "double");

template <typename T>
size_t struct_class_prefix() {
#ifdef _MSC_VER
return starts_with(raw<T>() + typename_prefix, "struct ")
? 7
: starts_with(raw<T>() + typename_prefix, "class ") ? 6 : 0;
#else
return 0;
#endif
}

template <typename T>
size_t typename_length() {
// raw_sizeof<T>() - raw_sizeof<double>() ==
// (length of T's name) - strlen("double")
// (length of T's name) ==
// raw_sizeof<T>() - raw_sizeof<double>() + strlen("double")
return raw_sizeof<T>() - struct_class_prefix<T>() - raw_sizeof<double>() + 6;
}

template <typename T>
const char* typename_begin() {
return raw<T>() + struct_class_prefix<T>() + typename_prefix;
}

} // namespace detail

template <typename T>
std::string nameof(bool strip_namespace = false) {
std::string name{detail::typename_begin<T>(), detail::typename_length<T>()};
if (strip_namespace) {
auto i = name.find_last_of("::");
if (i != std::string::npos) {
name = name.substr(i + 1);
}
}
return name;
}

} // namespace cpp11