From ff1c044ada6d651581d93e9893ea750c6f0c93df Mon Sep 17 00:00:00 2001 From: Daniel Sanders Date: Thu, 24 Apr 2014 09:58:52 +0000 Subject: [PATCH] [mips] Correct size_t and ptrdiff_t for N32. Summary: Correct size_t to be unsigned int and ptrdiff_t to be signed long. The types were the correct size before this change but the exact type matters for name mangling and exception handling in C++. Reviewers: atanasyan Reviewed By: atanasyan Differential Revision: http://reviews.llvm.org/D3470 llvm-svn: 207093 --- clang/lib/Basic/Targets.cpp | 32 ++++++++---- .../test/CodeGenCXX/mips-size_t-ptrdiff_t.cpp | 49 +++++++++++++++++++ 2 files changed, 71 insertions(+), 10 deletions(-) create mode 100644 clang/test/CodeGenCXX/mips-size_t-ptrdiff_t.cpp diff --git a/clang/lib/Basic/Targets.cpp b/clang/lib/Basic/Targets.cpp index 108b197daefc..59f5a018b0bf 100644 --- a/clang/lib/Basic/Targets.cpp +++ b/clang/lib/Basic/Targets.cpp @@ -5777,32 +5777,44 @@ class Mips64TargetInfoBase : public MipsTargetInfoBase { public: Mips64TargetInfoBase(const llvm::Triple &Triple) : MipsTargetInfoBase(Triple, "n64", "mips64r2") { - LongWidth = LongAlign = 64; - PointerWidth = PointerAlign = 64; LongDoubleWidth = LongDoubleAlign = 128; LongDoubleFormat = &llvm::APFloat::IEEEquad; if (getTriple().getOS() == llvm::Triple::FreeBSD) { LongDoubleWidth = LongDoubleAlign = 64; LongDoubleFormat = &llvm::APFloat::IEEEdouble; } + setN64ABITypes(); SuitableAlign = 128; MaxAtomicPromoteWidth = MaxAtomicInlineWidth = 64; } + + void setN64ABITypes() { + LongWidth = LongAlign = 64; + PointerWidth = PointerAlign = 64; + SizeType = UnsignedLong; + PtrDiffType = SignedLong; + } + + void setN32ABITypes() { + LongWidth = LongAlign = 32; + PointerWidth = PointerAlign = 32; + SizeType = UnsignedInt; + PtrDiffType = SignedInt; + } + bool setABI(const std::string &Name) override { if (Name == "n32") { - LongWidth = LongAlign = 32; - PointerWidth = PointerAlign = 32; + setN32ABITypes(); ABI = Name; return true; - } else if (Name == "n64") { - ABI = Name; - return true; - } else if (Name == "64") { + } else if (Name == "n64" || Name == "64") { + setN64ABITypes(); ABI = "n64"; return true; - } else - return false; + } + return false; } + void getTargetDefines(const LangOptions &Opts, MacroBuilder &Builder) const override { MipsTargetInfoBase::getTargetDefines(Opts, Builder); diff --git a/clang/test/CodeGenCXX/mips-size_t-ptrdiff_t.cpp b/clang/test/CodeGenCXX/mips-size_t-ptrdiff_t.cpp new file mode 100644 index 000000000000..7c94ea0d54f5 --- /dev/null +++ b/clang/test/CodeGenCXX/mips-size_t-ptrdiff_t.cpp @@ -0,0 +1,49 @@ +// RUN: %clang_cc1 -x c++ -emit-llvm -triple=mips-unknown-linux-gnu < %s | FileCheck --check-prefix=O32 %s +// RUN: %clang_cc1 -x c++ -emit-llvm -triple=mips64-unknown-linux-gnu -target-abi n32 < %s | FileCheck --check-prefix=N32 %s +// RUN: %clang_cc1 -x c++ -emit-llvm -triple=mips64-unknown-linux-gnu -target-abi 64 < %s | FileCheck --check-prefix=N64 %s + +// Test that the size_t is correct for the ABI. It's not sufficient to be the +// correct size, it must be the same type for correct name mangling. + +long *alloc_long() { + long *rv = new long; // size_t is implicit in the new operator + return rv; +} +// O32-LABEL: define i32* @_Z10alloc_longv() +// O32: call noalias i8* @_Znwj(i32 4) + +// N32-LABEL: define i32* @_Z10alloc_longv() +// N32: call noalias i8* @_Znwj(i32 4) + +// N64-LABEL: define i64* @_Z10alloc_longv() +// N64: call noalias i8* @_Znwm(i64 8) + +long *alloc_long_array() { + long *rv = new long[2]; + return rv; +} + +// O32-LABEL: define i32* @_Z16alloc_long_arrayv() +// O32: call noalias i8* @_Znaj(i32 8) + +// N32-LABEL: define i32* @_Z16alloc_long_arrayv() +// N32: call noalias i8* @_Znaj(i32 8) + +// N64-LABEL: define i64* @_Z16alloc_long_arrayv() +// N64: call noalias i8* @_Znam(i64 16) + +#include + +void size_t_arg(size_t a) { +} + +// O32-LABEL: _Z10size_t_argj +// N32-LABEL: _Z10size_t_argj +// N64-LABEL: _Z10size_t_argm + +void ptrdiff_t_arg(ptrdiff_t a) { +} + +// O32-LABEL: _Z13ptrdiff_t_argi +// N32-LABEL: _Z13ptrdiff_t_argi +// N64-LABEL: _Z13ptrdiff_t_argl