[llvm] r261819 - [Support] Add a fancy helper function to get a static name for a type.

David Majnemer via llvm-commits llvm-commits at lists.llvm.org
Wed Feb 24 20:31:38 PST 2016


I think the handling for __INTEL_COMPILER is not quite right.

Intel's compiler for Windows, icl, also defines __INTEL_COMPILER.
While icl defines __PRETTY_FUNCTION__, it provides behavior compatible with
__FUNCSIG__.
Given that icc defines __GNUC__ and icl defines _MSC_VER, I'd suggest
removing the defined(__INTEL_COMPILER).  I think the other checks will do
the right thing.


On Wed, Feb 24, 2016 at 7:58 PM, Chandler Carruth via llvm-commits <
llvm-commits at lists.llvm.org> wrote:

> Author: chandlerc
> Date: Wed Feb 24 21:58:21 2016
> New Revision: 261819
>
> URL: http://llvm.org/viewvc/llvm-project?rev=261819&view=rev
> Log:
> [Support] Add a fancy helper function to get a static name for a type.
>
> This extracts the type name from __PRETTY_FUNCTION__ for compilers that
> support it (I've opted Clang, GCC, and ICC into this as I've tested that
> they work) and from __FUNCSIG__ which is very similar on MSVC. The
> routine falls back gracefully on a stub "UNKNOWN_TYPE" string with
> compilers or formats it doesn't understand.
>
> This should be enough for a lot of common cases in LLVM where the real
> goal is just to log or print a type name as a debugging aid, and save
> a ton of boilerplate in the process. Notably, I'm planning to use this
> to remove all the getName() boiler plate from the new pass manager.
>
> The design and implementation is based on a bunch of advice and
> discussion with Richard Smith and experimenting with most versions of
> Clang and GCC. David Majnemer also provided excellent advice on how best
> to do this with MSVC. Richard also checked that ICC does something
> reasonable and I'll watch the build bots for other compilers. It'd be
> great if someone could contribute logic for xlC and/or other toolchains.
>
> Differential Revision: http://reviews.llvm.org/D17565
>
> Added:
>     llvm/trunk/include/llvm/Support/TypeName.h
>     llvm/trunk/unittests/Support/TypeNameTest.cpp
> Modified:
>     llvm/trunk/unittests/Support/CMakeLists.txt
>
> Added: llvm/trunk/include/llvm/Support/TypeName.h
> URL:
> http://llvm.org/viewvc/llvm-project/llvm/trunk/include/llvm/Support/TypeName.h?rev=261819&view=auto
>
> ==============================================================================
> --- llvm/trunk/include/llvm/Support/TypeName.h (added)
> +++ llvm/trunk/include/llvm/Support/TypeName.h Wed Feb 24 21:58:21 2016
> @@ -0,0 +1,65 @@
> +//===- TypeName.h -----------------------------------------------*- C++
> -*-===//
> +//
> +//                     The LLVM Compiler Infrastructure
> +//
> +// This file is distributed under the University of Illinois Open Source
> +// License. See LICENSE.TXT for details.
> +//
>
> +//===----------------------------------------------------------------------===//
> +
> +#ifndef LLVM_SUPPORT_TYPENAME_H
> +#define LLVM_SUPPORT_TYPENAME_H
> +
> +#include "llvm/ADT/StringRef.h"
> +
> +namespace llvm {
> +
> +/// We provide a function which tries to compute the (demangled) name of
> a type
> +/// statically.
> +///
> +/// This routine may fail on some platforms or for particularly unusual
> types.
> +/// Do not use it for anything other than logging and debugging aids. It
> isn't
> +/// portable or dependendable in any real sense.
> +///
> +/// The returned StringRef will point into a static storage duration
> string.
> +/// However, it may not be null terminated and may be some strangely
> aligned
> +/// inner substring of a larger string.
> +template <typename DesiredTypeName>
> +inline StringRef getTypeName() {
> +#if defined(__clang__) || defined(__GNUC__) || defined(__INTEL_COMPILER)
> +  StringRef Name = __PRETTY_FUNCTION__;
> +
> +  StringRef Key = "DesiredTypeName = ";
> +  Name = Name.substr(Name.find(Key));
> +  assert(!Name.empty() && "Unable to find the template parameter!");
> +  Name = Name.drop_front(Key.size());
> +
> +  assert(Name.endswith("]") && "Name doesn't end in the substitution
> key!");
> +  return Name.drop_back(1);
> +#elif defined(_MSC_VER)
> +  StringRef Name = __FUNCSIG__;
> +
> +  StringRef Key = "getTypeName<";
> +  Name = Name.substr(Name.find(Key));
> +  assert(!Name.empty() && "Unable to find the function name!");
> +  Name = Name.drop_front(Key.size());
> +
> +  for (StringRef Prefix : {"class ", "struct ", "union ", "enum "})
> +    if (Name.startswith(Prefix)) {
> +      Name = Name.drop_front(Prefix.size());
> +      break;
> +    }
> +
> +  auto AnglePos = Name.rfind('>');
> +  assert(AnglePos != StringRef::npos && "Unable to find the closing
> '>'!");
> +  return Name.substr(0, AnglePos);
> +#else
> +  // No known technique for statically extracting a type name on this
> compiler.
> +  // We return a string that is unlikely to look like any type in LLVM.
> +  return "UNKNOWN_TYPE";
> +#endif
> +}
> +
> +}
> +
> +#endif
>
> Modified: llvm/trunk/unittests/Support/CMakeLists.txt
> URL:
> http://llvm.org/viewvc/llvm-project/llvm/trunk/unittests/Support/CMakeLists.txt?rev=261819&r1=261818&r2=261819&view=diff
>
> ==============================================================================
> --- llvm/trunk/unittests/Support/CMakeLists.txt (original)
> +++ llvm/trunk/unittests/Support/CMakeLists.txt Wed Feb 24 21:58:21 2016
> @@ -42,6 +42,7 @@ add_llvm_unittest(SupportTests
>    ThreadPool.cpp
>    TimerTest.cpp
>    TimeValueTest.cpp
> +  TypeNameTest.cpp
>    TrailingObjectsTest.cpp
>    UnicodeTest.cpp
>    YAMLIOTest.cpp
>
> Added: llvm/trunk/unittests/Support/TypeNameTest.cpp
> URL:
> http://llvm.org/viewvc/llvm-project/llvm/trunk/unittests/Support/TypeNameTest.cpp?rev=261819&view=auto
>
> ==============================================================================
> --- llvm/trunk/unittests/Support/TypeNameTest.cpp (added)
> +++ llvm/trunk/unittests/Support/TypeNameTest.cpp Wed Feb 24 21:58:21 2016
> @@ -0,0 +1,49 @@
> +//===- TypeNameTest.cpp
> ---------------------------------------------------===//
> +//
> +//                     The LLVM Compiler Infrastructure
> +//
> +// This file is distributed under the University of Illinois Open Source
> +// License. See LICENSE.TXT for details.
> +//
>
> +//===----------------------------------------------------------------------===//
> +
> +#include "llvm/Support/TypeName.h"
> +#include "llvm/Support/raw_ostream.h"
> +#include "gtest/gtest.h"
> +
> +using namespace llvm;
> +
> +namespace {
> +namespace N1 {
> +struct S1 {};
> +class C1 {};
> +union U1 {};
> +}
> +
> +TEST(TypeNameTest, Names) {
> +  struct S2 {};
> +
> +  StringRef S1Name = getTypeName<N1::S1>();
> +  StringRef C1Name = getTypeName<N1::C1>();
> +  StringRef U1Name = getTypeName<N1::U1>();
> +  StringRef S2Name = getTypeName<S2>();
> +
> +#if defined(__clang__) || defined(__GNUC__) || defined(__INTEL_COMPILER)
> ||    \
> +    defined(_MSC_VER)
> +  EXPECT_TRUE(S1Name.endswith("::N1::S1")) << S1Name.str();
> +  EXPECT_TRUE(C1Name.endswith("::N1::C1")) << C1Name.str();
> +  EXPECT_TRUE(U1Name.endswith("::N1::U1")) << U1Name.str();
> +#ifdef __clang__
> +  EXPECT_TRUE(S2Name.endswith("S2")) << S2Name.str();
> +#else
> +  EXPECT_TRUE(S2Name.endswith("::S2")) << S2Name.str();
> +#endif
> +#else
> +  EXPECT_EQ("UNKNOWN_TYPE", S1Name);
> +  EXPECT_EQ("UNKNOWN_TYPE", C1Name);
> +  EXPECT_EQ("UNKNOWN_TYPE", U1Name);
> +  EXPECT_EQ("UNKNOWN_TYPE", S2Name);
> +#endif
> +}
> +
> +} // end anonymous namespace
>
>
> _______________________________________________
> llvm-commits mailing list
> llvm-commits at lists.llvm.org
> http://lists.llvm.org/cgi-bin/mailman/listinfo/llvm-commits
>
-------------- next part --------------
An HTML attachment was scrubbed...
URL: <http://lists.llvm.org/pipermail/llvm-commits/attachments/20160224/ef7d11d8/attachment.html>


More information about the llvm-commits mailing list