tail-padding.cpp 3.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778
  1. // RUN: %clang_cc1 -triple %itanium_abi_triple -emit-llvm -o - %s | FileCheck %s
  2. // PR36992
  3. namespace Implicit {
  4. struct A { char c; A(const A&); };
  5. struct B { int n; char c[3]; ~B(); };
  6. struct C : B, virtual A {};
  7. static_assert(sizeof(C) == sizeof(void*) + 8);
  8. C f(C c) { return c; }
  9. // CHECK: define {{.*}} @_ZN8Implicit1CC1EOS0_
  10. // CHECK: call {{.*}} @_ZN8Implicit1AC2ERKS0_(
  11. // Note: this must memcpy 7 bytes, not 8, to avoid trampling over the virtual base class.
  12. // CHECK: call void @llvm.memcpy.p0i8.p0i8.i{{32|64}}(i8* {{.*}}, i8* {{.*}}, i{{32|64}} 7, i1 false)
  13. // CHECK: store i32 {{.*}} @_ZTVN8Implicit1CE
  14. }
  15. namespace InitWithinNVSize {
  16. // This is the same as the previous test, except that the A base lies
  17. // entirely within the nvsize of C. This makes it valid to copy at the
  18. // full width.
  19. struct A { char c; A(const A&); };
  20. struct B { int n; char c[3]; ~B(); };
  21. struct C : B, virtual A { char x; };
  22. static_assert(sizeof(C) > sizeof(void*) + 8);
  23. C f(C c) { return c; }
  24. // CHECK: define {{.*}} @_ZN16InitWithinNVSize1CC1EOS0_
  25. // CHECK: call {{.*}} @_ZN16InitWithinNVSize1AC2ERKS0_(
  26. // This copies over the 'C::x' member, but that's OK because we've not initialized it yet.
  27. // CHECK: call void @llvm.memcpy.p0i8.p0i8.i{{32|64}}(i8* {{.*}}, i8* {{.*}}, i{{32|64}} 8, i1 false)
  28. // CHECK: store i32 {{.*}} @_ZTVN16InitWithinNVSize1CE
  29. // CHECK: store i8
  30. }
  31. namespace NoUniqueAddr {
  32. struct A { char c; A(const A&); };
  33. struct B { int n; char c[3]; ~B(); };
  34. struct C : virtual A { B b; };
  35. struct D : virtual A { [[no_unique_address]] B b; };
  36. struct E : virtual A { [[no_unique_address]] B b; char x; };
  37. static_assert(sizeof(C) == sizeof(void*) + 8 + alignof(void*));
  38. static_assert(sizeof(D) == sizeof(void*) + 8);
  39. static_assert(sizeof(E) == sizeof(void*) + 8 + alignof(void*));
  40. // CHECK: define {{.*}} @_ZN12NoUniqueAddr1CC1EOS0_
  41. // CHECK: call void @_ZN12NoUniqueAddr1AC2ERKS0_(
  42. // CHECK: store i32 {{.*}} @_ZTVN12NoUniqueAddr1CE
  43. // Copy the full size of B.
  44. // CHECK: call void @llvm.memcpy.p0i8.p0i8.i64(i8* {{.*}}, i8* {{.*}}, i64 8, i1 false)
  45. C f(C c) { return c; }
  46. // CHECK: define {{.*}} @_ZN12NoUniqueAddr1DC1EOS0_
  47. // CHECK: call void @_ZN12NoUniqueAddr1AC2ERKS0_(
  48. // CHECK: store i32 {{.*}} @_ZTVN12NoUniqueAddr1DE
  49. // Copy just the data size of B, to avoid overwriting the A base class.
  50. // CHECK: call void @llvm.memcpy.p0i8.p0i8.i64(i8* {{.*}}, i8* {{.*}}, i64 7, i1 false)
  51. D f(D d) { return d; }
  52. // CHECK: define {{.*}} @_ZN12NoUniqueAddr1EC1EOS0_
  53. // CHECK: call void @_ZN12NoUniqueAddr1AC2ERKS0_(
  54. // CHECK: store i32 {{.*}} @_ZTVN12NoUniqueAddr1EE
  55. // We can copy the full size of B here. (As it happens, we fold the copy of 'x' into
  56. // this memcpy, so we're copying 8 bytes either way.)
  57. // CHECK: call void @llvm.memcpy.p0i8.p0i8.i64(i8* {{.*}}, i8* {{.*}}, i64 8, i1 false)
  58. E f(E e) { return e; }
  59. struct F : virtual A {
  60. F(const F &o) : A(o), b(o.b) {}
  61. [[no_unique_address]] B b;
  62. };
  63. // CHECK: define {{.*}} @_ZN12NoUniqueAddr1FC1ERKS0_
  64. // CHECK: call void @_ZN12NoUniqueAddr1AC2ERKS0_(
  65. // CHECK: store i32 {{.*}} @_ZTVN12NoUniqueAddr1FE
  66. // CHECK: call void @llvm.memcpy.p0i8.p0i8.i64(i8* {{.*}}, i8* {{.*}}, i64 7, i1 false)
  67. F f(F x) { return x; }
  68. }