compound_field.carbon 8.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166
  1. // Part of the Carbon Language project, under the Apache License v2.0 with LLVM
  2. // Exceptions. See /LICENSE for license information.
  3. // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
  4. //
  5. // AUTOUPDATE
  6. base class Base {
  7. var a: i32;
  8. var b: i32;
  9. var c: i32;
  10. }
  11. class Derived {
  12. extend base: Base;
  13. var d: i32;
  14. var e: i32;
  15. }
  16. fn AccessDerived(d: Derived) -> i32 {
  17. return d.(Derived.d);
  18. }
  19. fn AccessBase(d: Derived) -> i32 {
  20. return d.(Base.b);
  21. }
  22. fn AccessDerivedIndirect(p: Derived*) -> i32* {
  23. return &p->(Derived.d);
  24. }
  25. fn AccessBaseIndirect(p: Derived*) -> i32* {
  26. return &p->(Base.b);
  27. }
  28. // CHECK:STDOUT: --- compound_field.carbon
  29. // CHECK:STDOUT:
  30. // CHECK:STDOUT: constants {
  31. // CHECK:STDOUT: %Base: type = class_type @Base [template]
  32. // CHECK:STDOUT: %.1: type = unbound_element_type Base, i32 [template]
  33. // CHECK:STDOUT: %.2: type = struct_type {.a: i32, .b: i32, .c: i32} [template]
  34. // CHECK:STDOUT: %Derived: type = class_type @Derived [template]
  35. // CHECK:STDOUT: %.3: type = ptr_type {.a: i32, .b: i32, .c: i32} [template]
  36. // CHECK:STDOUT: %.4: type = unbound_element_type Derived, Base [template]
  37. // CHECK:STDOUT: %.5: type = unbound_element_type Derived, i32 [template]
  38. // CHECK:STDOUT: %.6: type = struct_type {.base: Base, .d: i32, .e: i32} [template]
  39. // CHECK:STDOUT: %.7: type = struct_type {.base: {.a: i32, .b: i32, .c: i32}*, .d: i32, .e: i32} [template]
  40. // CHECK:STDOUT: %.8: type = ptr_type {.base: {.a: i32, .b: i32, .c: i32}*, .d: i32, .e: i32} [template]
  41. // CHECK:STDOUT: %.9: type = ptr_type {.base: Base, .d: i32, .e: i32} [template]
  42. // CHECK:STDOUT: %.10: type = ptr_type Derived [template]
  43. // CHECK:STDOUT: %.11: type = ptr_type i32 [template]
  44. // CHECK:STDOUT: }
  45. // CHECK:STDOUT:
  46. // CHECK:STDOUT: file {
  47. // CHECK:STDOUT: package: <namespace> = namespace [template] {
  48. // CHECK:STDOUT: .Base = %Base.decl
  49. // CHECK:STDOUT: .Derived = %Derived.decl
  50. // CHECK:STDOUT: .AccessDerived = %AccessDerived
  51. // CHECK:STDOUT: .AccessBase = %AccessBase
  52. // CHECK:STDOUT: .AccessDerivedIndirect = %AccessDerivedIndirect
  53. // CHECK:STDOUT: .AccessBaseIndirect = %AccessBaseIndirect
  54. // CHECK:STDOUT: }
  55. // CHECK:STDOUT: %Base.decl: type = class_decl @Base [template = constants.%Base] {}
  56. // CHECK:STDOUT: %Derived.decl: type = class_decl @Derived [template = constants.%Derived] {}
  57. // CHECK:STDOUT: %AccessDerived: <function> = fn_decl @AccessDerived [template] {
  58. // CHECK:STDOUT: %Derived.ref.loc20: type = name_ref Derived, %Derived.decl [template = constants.%Derived]
  59. // CHECK:STDOUT: %d.loc20_18.1: Derived = param d
  60. // CHECK:STDOUT: @AccessDerived.%d: Derived = bind_name d, %d.loc20_18.1
  61. // CHECK:STDOUT: %return.var.loc20: ref i32 = var <return slot>
  62. // CHECK:STDOUT: }
  63. // CHECK:STDOUT: %AccessBase: <function> = fn_decl @AccessBase [template] {
  64. // CHECK:STDOUT: %Derived.ref.loc24: type = name_ref Derived, %Derived.decl [template = constants.%Derived]
  65. // CHECK:STDOUT: %d.loc24_15.1: Derived = param d
  66. // CHECK:STDOUT: @AccessBase.%d: Derived = bind_name d, %d.loc24_15.1
  67. // CHECK:STDOUT: %return.var.loc24: ref i32 = var <return slot>
  68. // CHECK:STDOUT: }
  69. // CHECK:STDOUT: %AccessDerivedIndirect: <function> = fn_decl @AccessDerivedIndirect [template] {
  70. // CHECK:STDOUT: %Derived.ref.loc28: type = name_ref Derived, %Derived.decl [template = constants.%Derived]
  71. // CHECK:STDOUT: %.loc28_36: type = ptr_type Derived [template = constants.%.10]
  72. // CHECK:STDOUT: %p.loc28_26.1: Derived* = param p
  73. // CHECK:STDOUT: @AccessDerivedIndirect.%p: Derived* = bind_name p, %p.loc28_26.1
  74. // CHECK:STDOUT: %.loc28_45: type = ptr_type i32 [template = constants.%.11]
  75. // CHECK:STDOUT: %return.var.loc28: ref i32* = var <return slot>
  76. // CHECK:STDOUT: }
  77. // CHECK:STDOUT: %AccessBaseIndirect: <function> = fn_decl @AccessBaseIndirect [template] {
  78. // CHECK:STDOUT: %Derived.ref.loc32: type = name_ref Derived, %Derived.decl [template = constants.%Derived]
  79. // CHECK:STDOUT: %.loc32_33: type = ptr_type Derived [template = constants.%.10]
  80. // CHECK:STDOUT: %p.loc32_23.1: Derived* = param p
  81. // CHECK:STDOUT: @AccessBaseIndirect.%p: Derived* = bind_name p, %p.loc32_23.1
  82. // CHECK:STDOUT: %.loc32_42: type = ptr_type i32 [template = constants.%.11]
  83. // CHECK:STDOUT: %return.var.loc32: ref i32* = var <return slot>
  84. // CHECK:STDOUT: }
  85. // CHECK:STDOUT: }
  86. // CHECK:STDOUT:
  87. // CHECK:STDOUT: class @Base {
  88. // CHECK:STDOUT: %.loc8: <unbound element of class Base> = field_decl a, element0 [template]
  89. // CHECK:STDOUT: %.loc9: <unbound element of class Base> = field_decl b, element1 [template]
  90. // CHECK:STDOUT: %.loc10: <unbound element of class Base> = field_decl c, element2 [template]
  91. // CHECK:STDOUT:
  92. // CHECK:STDOUT: !members:
  93. // CHECK:STDOUT: .Self = constants.%Base
  94. // CHECK:STDOUT: .a = %.loc8
  95. // CHECK:STDOUT: .b = %.loc9
  96. // CHECK:STDOUT: .c = %.loc10
  97. // CHECK:STDOUT: }
  98. // CHECK:STDOUT:
  99. // CHECK:STDOUT: class @Derived {
  100. // CHECK:STDOUT: %Base.ref: type = name_ref Base, file.%Base.decl [template = constants.%Base]
  101. // CHECK:STDOUT: %.loc14: <unbound element of class Derived> = base_decl Base, element0 [template]
  102. // CHECK:STDOUT: %.loc16: <unbound element of class Derived> = field_decl d, element1 [template]
  103. // CHECK:STDOUT: %.loc17: <unbound element of class Derived> = field_decl e, element2 [template]
  104. // CHECK:STDOUT:
  105. // CHECK:STDOUT: !members:
  106. // CHECK:STDOUT: .Self = constants.%Derived
  107. // CHECK:STDOUT: .base = %.loc14
  108. // CHECK:STDOUT: .d = %.loc16
  109. // CHECK:STDOUT: .e = %.loc17
  110. // CHECK:STDOUT: extend name_scope1
  111. // CHECK:STDOUT: }
  112. // CHECK:STDOUT:
  113. // CHECK:STDOUT: fn @AccessDerived(%d: Derived) -> i32 {
  114. // CHECK:STDOUT: !entry:
  115. // CHECK:STDOUT: %d.ref.loc21_10: Derived = name_ref d, %d
  116. // CHECK:STDOUT: %Derived.ref: type = name_ref Derived, file.%Derived.decl [template = constants.%Derived]
  117. // CHECK:STDOUT: %d.ref.loc21_20: <unbound element of class Derived> = name_ref d, @Derived.%.loc16 [template = @Derived.%.loc16]
  118. // CHECK:STDOUT: %.loc21_11.1: ref i32 = class_element_access %d.ref.loc21_10, element1
  119. // CHECK:STDOUT: %.loc21_11.2: i32 = bind_value %.loc21_11.1
  120. // CHECK:STDOUT: return %.loc21_11.2
  121. // CHECK:STDOUT: }
  122. // CHECK:STDOUT:
  123. // CHECK:STDOUT: fn @AccessBase(%d: Derived) -> i32 {
  124. // CHECK:STDOUT: !entry:
  125. // CHECK:STDOUT: %d.ref: Derived = name_ref d, %d
  126. // CHECK:STDOUT: %Base.ref: type = name_ref Base, file.%Base.decl [template = constants.%Base]
  127. // CHECK:STDOUT: %b.ref: <unbound element of class Base> = name_ref b, @Base.%.loc9 [template = @Base.%.loc9]
  128. // CHECK:STDOUT: %.loc25_11.1: ref Base = class_element_access %d.ref, element0
  129. // CHECK:STDOUT: %.loc25_10: ref Base = converted %d.ref, %.loc25_11.1
  130. // CHECK:STDOUT: %.loc25_11.2: ref i32 = class_element_access %.loc25_10, element1
  131. // CHECK:STDOUT: %.loc25_11.3: i32 = bind_value %.loc25_11.2
  132. // CHECK:STDOUT: return %.loc25_11.3
  133. // CHECK:STDOUT: }
  134. // CHECK:STDOUT:
  135. // CHECK:STDOUT: fn @AccessDerivedIndirect(%p: Derived*) -> i32* {
  136. // CHECK:STDOUT: !entry:
  137. // CHECK:STDOUT: %p.ref: Derived* = name_ref p, %p
  138. // CHECK:STDOUT: %Derived.ref: type = name_ref Derived, file.%Derived.decl [template = constants.%Derived]
  139. // CHECK:STDOUT: %d.ref: <unbound element of class Derived> = name_ref d, @Derived.%.loc16 [template = @Derived.%.loc16]
  140. // CHECK:STDOUT: %.loc29_12.1: ref Derived = deref %p.ref
  141. // CHECK:STDOUT: %.loc29_12.2: ref i32 = class_element_access %.loc29_12.1, element1
  142. // CHECK:STDOUT: %.loc29_10: i32* = addr_of %.loc29_12.2
  143. // CHECK:STDOUT: return %.loc29_10
  144. // CHECK:STDOUT: }
  145. // CHECK:STDOUT:
  146. // CHECK:STDOUT: fn @AccessBaseIndirect(%p: Derived*) -> i32* {
  147. // CHECK:STDOUT: !entry:
  148. // CHECK:STDOUT: %p.ref: Derived* = name_ref p, %p
  149. // CHECK:STDOUT: %Base.ref: type = name_ref Base, file.%Base.decl [template = constants.%Base]
  150. // CHECK:STDOUT: %b.ref: <unbound element of class Base> = name_ref b, @Base.%.loc9 [template = @Base.%.loc9]
  151. // CHECK:STDOUT: %.loc33_12.1: ref Derived = deref %p.ref
  152. // CHECK:STDOUT: %.loc33_12.2: ref Base = class_element_access %.loc33_12.1, element0
  153. // CHECK:STDOUT: %.loc33_12.3: ref Base = converted %.loc33_12.1, %.loc33_12.2
  154. // CHECK:STDOUT: %.loc33_12.4: ref i32 = class_element_access %.loc33_12.3, element1
  155. // CHECK:STDOUT: %.loc33_10: i32* = addr_of %.loc33_12.4
  156. // CHECK:STDOUT: return %.loc33_10
  157. // CHECK:STDOUT: }
  158. // CHECK:STDOUT: