int_add.carbon 15 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301
  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. // --- int_add.carbon
  7. fn Add(a: i32, b: i32) -> i32 = "int.add";
  8. var arr: [i32; Add(1, 2)];
  9. fn RuntimeCall(a: i32, b: i32) -> i32 {
  10. return Add(a, b);
  11. }
  12. // --- fail_bad_decl.carbon
  13. package FailBadDecl api;
  14. fn TooFew(a: i32) -> i32 = "int.add";
  15. fn TooMany(a: i32, b: i32, c: i32) -> i32 = "int.add";
  16. fn BadReturnType(a: i32, b: i32) -> bool = "int.add";
  17. fn JustRight(a: i32, b: i32) -> i32 = "int.add";
  18. // CHECK:STDERR: fail_bad_decl.carbon:[[@LINE+3]]:20: ERROR: Array bound is not a constant.
  19. // CHECK:STDERR: var too_few: [i32; TooFew(1)];
  20. // CHECK:STDERR: ^~~~~~~
  21. var too_few: [i32; TooFew(1)];
  22. // CHECK:STDERR: fail_bad_decl.carbon:[[@LINE+3]]:21: ERROR: Array bound is not a constant.
  23. // CHECK:STDERR: var too_many: [i32; TooMany(1, 2, 3)];
  24. // CHECK:STDERR: ^~~~~~~~
  25. var too_many: [i32; TooMany(1, 2, 3)];
  26. // CHECK:STDERR: fail_bad_decl.carbon:[[@LINE+3]]:28: ERROR: Array bound is not a constant.
  27. // CHECK:STDERR: var bad_return_type: [i32; BadReturnType(1, 2)];
  28. // CHECK:STDERR: ^~~~~~~~~~~~~~
  29. var bad_return_type: [i32; BadReturnType(1, 2)];
  30. // CHECK:STDERR: fail_bad_decl.carbon:[[@LINE+6]]:21: ERROR: 3 argument(s) passed to function expecting 2 argument(s).
  31. // CHECK:STDERR: var bad_call: [i32; JustRight(1, 2, 3)];
  32. // CHECK:STDERR: ^~~~~~~~~~
  33. // CHECK:STDERR: fail_bad_decl.carbon:[[@LINE-18]]:1: Calling function declared here.
  34. // CHECK:STDERR: fn JustRight(a: i32, b: i32) -> i32 = "int.add";
  35. // CHECK:STDERR: ^~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
  36. var bad_call: [i32; JustRight(1, 2, 3)];
  37. // TODO: We should diagnose these in check rather than failing in lower.
  38. fn RuntimeCallTooFew(a: i32) -> i32 {
  39. return TooFew(a);
  40. }
  41. fn RuntimeCallTooMany(a: i32, b: i32, c: i32) -> i32 {
  42. return TooMany(a, b, c);
  43. }
  44. fn RuntimeCallBadReturnType(a: i32, b: i32) -> bool {
  45. return BadReturnType(a, b);
  46. }
  47. // --- fail_overflow.carbon
  48. package FailOverflow api;
  49. fn Add(a: i32, b: i32) -> i32 = "int.add";
  50. let a: i32 = Add(0x7FFFFFFF, 0);
  51. // CHECK:STDERR: fail_overflow.carbon:[[@LINE+3]]:14: ERROR: Integer overflow in calculation 2147483647 + 1.
  52. // CHECK:STDERR: let b: i32 = Add(0x7FFFFFFF, 1);
  53. // CHECK:STDERR: ^~~~
  54. let b: i32 = Add(0x7FFFFFFF, 1);
  55. // CHECK:STDOUT: --- int_add.carbon
  56. // CHECK:STDOUT:
  57. // CHECK:STDOUT: constants {
  58. // CHECK:STDOUT: %.1: i32 = int_literal 1 [template]
  59. // CHECK:STDOUT: %.2: i32 = int_literal 2 [template]
  60. // CHECK:STDOUT: %.3: i32 = int_literal 3 [template]
  61. // CHECK:STDOUT: %.4: type = array_type %.3, i32 [template]
  62. // CHECK:STDOUT: %.5: type = ptr_type [i32; 3] [template]
  63. // CHECK:STDOUT: }
  64. // CHECK:STDOUT:
  65. // CHECK:STDOUT: file {
  66. // CHECK:STDOUT: package: <namespace> = namespace [template] {
  67. // CHECK:STDOUT: .Add = %Add
  68. // CHECK:STDOUT: .arr = %arr
  69. // CHECK:STDOUT: .RuntimeCall = %RuntimeCall
  70. // CHECK:STDOUT: }
  71. // CHECK:STDOUT: %Add: <function> = fn_decl @Add [template] {
  72. // CHECK:STDOUT: %a.loc2_8.1: i32 = param a
  73. // CHECK:STDOUT: @Add.%a: i32 = bind_name a, %a.loc2_8.1
  74. // CHECK:STDOUT: %b.loc2_16.1: i32 = param b
  75. // CHECK:STDOUT: @Add.%b: i32 = bind_name b, %b.loc2_16.1
  76. // CHECK:STDOUT: %return.var.loc2: ref i32 = var <return slot>
  77. // CHECK:STDOUT: }
  78. // CHECK:STDOUT: %Add.ref: <function> = name_ref Add, %Add [template = %Add]
  79. // CHECK:STDOUT: %.loc4_20: i32 = int_literal 1 [template = constants.%.1]
  80. // CHECK:STDOUT: %.loc4_23: i32 = int_literal 2 [template = constants.%.2]
  81. // CHECK:STDOUT: %.loc4_19: init i32 = call %Add.ref(%.loc4_20, %.loc4_23) [template = constants.%.3]
  82. // CHECK:STDOUT: %.loc4_25: type = array_type %.loc4_19, i32 [template = constants.%.4]
  83. // CHECK:STDOUT: %arr.var: ref [i32; 3] = var arr
  84. // CHECK:STDOUT: %arr: ref [i32; 3] = bind_name arr, %arr.var
  85. // CHECK:STDOUT: %RuntimeCall: <function> = fn_decl @RuntimeCall [template] {
  86. // CHECK:STDOUT: %a.loc6_16.1: i32 = param a
  87. // CHECK:STDOUT: @RuntimeCall.%a: i32 = bind_name a, %a.loc6_16.1
  88. // CHECK:STDOUT: %b.loc6_24.1: i32 = param b
  89. // CHECK:STDOUT: @RuntimeCall.%b: i32 = bind_name b, %b.loc6_24.1
  90. // CHECK:STDOUT: %return.var.loc6: ref i32 = var <return slot>
  91. // CHECK:STDOUT: }
  92. // CHECK:STDOUT: }
  93. // CHECK:STDOUT:
  94. // CHECK:STDOUT: fn @Add(%a: i32, %b: i32) -> i32 = "int.add";
  95. // CHECK:STDOUT:
  96. // CHECK:STDOUT: fn @RuntimeCall(%a: i32, %b: i32) -> i32 {
  97. // CHECK:STDOUT: !entry:
  98. // CHECK:STDOUT: %Add.ref: <function> = name_ref Add, file.%Add [template = file.%Add]
  99. // CHECK:STDOUT: %a.ref: i32 = name_ref a, %a
  100. // CHECK:STDOUT: %b.ref: i32 = name_ref b, %b
  101. // CHECK:STDOUT: %.loc7_13.1: init i32 = call %Add.ref(%a.ref, %b.ref)
  102. // CHECK:STDOUT: %.loc7_19: i32 = value_of_initializer %.loc7_13.1
  103. // CHECK:STDOUT: %.loc7_13.2: i32 = converted %.loc7_13.1, %.loc7_19
  104. // CHECK:STDOUT: return %.loc7_13.2
  105. // CHECK:STDOUT: }
  106. // CHECK:STDOUT:
  107. // CHECK:STDOUT: --- fail_bad_decl.carbon
  108. // CHECK:STDOUT:
  109. // CHECK:STDOUT: constants {
  110. // CHECK:STDOUT: %.1: i32 = int_literal 1 [template]
  111. // CHECK:STDOUT: %.2: i32 = int_literal 2 [template]
  112. // CHECK:STDOUT: %.3: i32 = int_literal 3 [template]
  113. // CHECK:STDOUT: }
  114. // CHECK:STDOUT:
  115. // CHECK:STDOUT: file {
  116. // CHECK:STDOUT: package: <namespace> = namespace [template] {
  117. // CHECK:STDOUT: .TooFew = %TooFew
  118. // CHECK:STDOUT: .TooMany = %TooMany
  119. // CHECK:STDOUT: .BadReturnType = %BadReturnType
  120. // CHECK:STDOUT: .JustRight = %JustRight
  121. // CHECK:STDOUT: .too_few = %too_few
  122. // CHECK:STDOUT: .too_many = %too_many
  123. // CHECK:STDOUT: .bad_return_type = %bad_return_type
  124. // CHECK:STDOUT: .bad_call = %bad_call
  125. // CHECK:STDOUT: .RuntimeCallTooFew = %RuntimeCallTooFew
  126. // CHECK:STDOUT: .RuntimeCallTooMany = %RuntimeCallTooMany
  127. // CHECK:STDOUT: .RuntimeCallBadReturnType = %RuntimeCallBadReturnType
  128. // CHECK:STDOUT: }
  129. // CHECK:STDOUT: %TooFew: <function> = fn_decl @TooFew [template] {
  130. // CHECK:STDOUT: %a.loc4_11.1: i32 = param a
  131. // CHECK:STDOUT: @TooFew.%a: i32 = bind_name a, %a.loc4_11.1
  132. // CHECK:STDOUT: %return.var.loc4: ref i32 = var <return slot>
  133. // CHECK:STDOUT: }
  134. // CHECK:STDOUT: %TooMany: <function> = fn_decl @TooMany [template] {
  135. // CHECK:STDOUT: %a.loc5_12.1: i32 = param a
  136. // CHECK:STDOUT: @TooMany.%a: i32 = bind_name a, %a.loc5_12.1
  137. // CHECK:STDOUT: %b.loc5_20.1: i32 = param b
  138. // CHECK:STDOUT: @TooMany.%b: i32 = bind_name b, %b.loc5_20.1
  139. // CHECK:STDOUT: %c.loc5_28.1: i32 = param c
  140. // CHECK:STDOUT: @TooMany.%c: i32 = bind_name c, %c.loc5_28.1
  141. // CHECK:STDOUT: %return.var.loc5: ref i32 = var <return slot>
  142. // CHECK:STDOUT: }
  143. // CHECK:STDOUT: %BadReturnType: <function> = fn_decl @BadReturnType [template] {
  144. // CHECK:STDOUT: %a.loc6_18.1: i32 = param a
  145. // CHECK:STDOUT: @BadReturnType.%a: i32 = bind_name a, %a.loc6_18.1
  146. // CHECK:STDOUT: %b.loc6_26.1: i32 = param b
  147. // CHECK:STDOUT: @BadReturnType.%b: i32 = bind_name b, %b.loc6_26.1
  148. // CHECK:STDOUT: %return.var.loc6: ref bool = var <return slot>
  149. // CHECK:STDOUT: }
  150. // CHECK:STDOUT: %JustRight: <function> = fn_decl @JustRight [template] {
  151. // CHECK:STDOUT: %a.loc7_14.1: i32 = param a
  152. // CHECK:STDOUT: @JustRight.%a: i32 = bind_name a, %a.loc7_14.1
  153. // CHECK:STDOUT: %b.loc7_22.1: i32 = param b
  154. // CHECK:STDOUT: @JustRight.%b: i32 = bind_name b, %b.loc7_22.1
  155. // CHECK:STDOUT: %return.var.loc7: ref i32 = var <return slot>
  156. // CHECK:STDOUT: }
  157. // CHECK:STDOUT: %TooFew.ref: <function> = name_ref TooFew, %TooFew [template = %TooFew]
  158. // CHECK:STDOUT: %.loc12_27: i32 = int_literal 1 [template = constants.%.1]
  159. // CHECK:STDOUT: %.loc12_26: init i32 = call %TooFew.ref(%.loc12_27)
  160. // CHECK:STDOUT: %too_few.var: ref <error> = var too_few
  161. // CHECK:STDOUT: %too_few: ref <error> = bind_name too_few, %too_few.var
  162. // CHECK:STDOUT: %TooMany.ref: <function> = name_ref TooMany, %TooMany [template = %TooMany]
  163. // CHECK:STDOUT: %.loc16_29: i32 = int_literal 1 [template = constants.%.1]
  164. // CHECK:STDOUT: %.loc16_32: i32 = int_literal 2 [template = constants.%.2]
  165. // CHECK:STDOUT: %.loc16_35: i32 = int_literal 3 [template = constants.%.3]
  166. // CHECK:STDOUT: %.loc16_28: init i32 = call %TooMany.ref(%.loc16_29, %.loc16_32, %.loc16_35)
  167. // CHECK:STDOUT: %too_many.var: ref <error> = var too_many
  168. // CHECK:STDOUT: %too_many: ref <error> = bind_name too_many, %too_many.var
  169. // CHECK:STDOUT: %BadReturnType.ref: <function> = name_ref BadReturnType, %BadReturnType [template = %BadReturnType]
  170. // CHECK:STDOUT: %.loc20_42: i32 = int_literal 1 [template = constants.%.1]
  171. // CHECK:STDOUT: %.loc20_45: i32 = int_literal 2 [template = constants.%.2]
  172. // CHECK:STDOUT: %.loc20_41: init bool = call %BadReturnType.ref(%.loc20_42, %.loc20_45)
  173. // CHECK:STDOUT: %bad_return_type.var: ref <error> = var bad_return_type
  174. // CHECK:STDOUT: %bad_return_type: ref <error> = bind_name bad_return_type, %bad_return_type.var
  175. // CHECK:STDOUT: %JustRight.ref: <function> = name_ref JustRight, %JustRight [template = %JustRight]
  176. // CHECK:STDOUT: %.loc28_31: i32 = int_literal 1 [template = constants.%.1]
  177. // CHECK:STDOUT: %.loc28_34: i32 = int_literal 2 [template = constants.%.2]
  178. // CHECK:STDOUT: %.loc28_37: i32 = int_literal 3 [template = constants.%.3]
  179. // CHECK:STDOUT: %.loc28_30: init i32 = call %JustRight.ref(<invalid>) [template = <error>]
  180. // CHECK:STDOUT: %.loc28_39: type = array_type %.loc28_30, i32 [template = <error>]
  181. // CHECK:STDOUT: %bad_call.var: ref <error> = var bad_call
  182. // CHECK:STDOUT: %bad_call: ref <error> = bind_name bad_call, %bad_call.var
  183. // CHECK:STDOUT: %RuntimeCallTooFew: <function> = fn_decl @RuntimeCallTooFew [template] {
  184. // CHECK:STDOUT: %a.loc31_22.1: i32 = param a
  185. // CHECK:STDOUT: @RuntimeCallTooFew.%a: i32 = bind_name a, %a.loc31_22.1
  186. // CHECK:STDOUT: %return.var.loc31: ref i32 = var <return slot>
  187. // CHECK:STDOUT: }
  188. // CHECK:STDOUT: %RuntimeCallTooMany: <function> = fn_decl @RuntimeCallTooMany [template] {
  189. // CHECK:STDOUT: %a.loc35_23.1: i32 = param a
  190. // CHECK:STDOUT: @RuntimeCallTooMany.%a: i32 = bind_name a, %a.loc35_23.1
  191. // CHECK:STDOUT: %b.loc35_31.1: i32 = param b
  192. // CHECK:STDOUT: @RuntimeCallTooMany.%b: i32 = bind_name b, %b.loc35_31.1
  193. // CHECK:STDOUT: %c.loc35_39.1: i32 = param c
  194. // CHECK:STDOUT: @RuntimeCallTooMany.%c: i32 = bind_name c, %c.loc35_39.1
  195. // CHECK:STDOUT: %return.var.loc35: ref i32 = var <return slot>
  196. // CHECK:STDOUT: }
  197. // CHECK:STDOUT: %RuntimeCallBadReturnType: <function> = fn_decl @RuntimeCallBadReturnType [template] {
  198. // CHECK:STDOUT: %a.loc39_29.1: i32 = param a
  199. // CHECK:STDOUT: @RuntimeCallBadReturnType.%a: i32 = bind_name a, %a.loc39_29.1
  200. // CHECK:STDOUT: %b.loc39_37.1: i32 = param b
  201. // CHECK:STDOUT: @RuntimeCallBadReturnType.%b: i32 = bind_name b, %b.loc39_37.1
  202. // CHECK:STDOUT: %return.var.loc39: ref bool = var <return slot>
  203. // CHECK:STDOUT: }
  204. // CHECK:STDOUT: }
  205. // CHECK:STDOUT:
  206. // CHECK:STDOUT: fn @TooFew(%a: i32) -> i32 = "int.add";
  207. // CHECK:STDOUT:
  208. // CHECK:STDOUT: fn @TooMany(%a: i32, %b: i32, %c: i32) -> i32 = "int.add";
  209. // CHECK:STDOUT:
  210. // CHECK:STDOUT: fn @BadReturnType(%a: i32, %b: i32) -> bool = "int.add";
  211. // CHECK:STDOUT:
  212. // CHECK:STDOUT: fn @JustRight(%a: i32, %b: i32) -> i32 = "int.add";
  213. // CHECK:STDOUT:
  214. // CHECK:STDOUT: fn @RuntimeCallTooFew(%a: i32) -> i32 {
  215. // CHECK:STDOUT: !entry:
  216. // CHECK:STDOUT: %TooFew.ref: <function> = name_ref TooFew, file.%TooFew [template = file.%TooFew]
  217. // CHECK:STDOUT: %a.ref: i32 = name_ref a, %a
  218. // CHECK:STDOUT: %.loc32_16.1: init i32 = call %TooFew.ref(%a.ref)
  219. // CHECK:STDOUT: %.loc32_19: i32 = value_of_initializer %.loc32_16.1
  220. // CHECK:STDOUT: %.loc32_16.2: i32 = converted %.loc32_16.1, %.loc32_19
  221. // CHECK:STDOUT: return %.loc32_16.2
  222. // CHECK:STDOUT: }
  223. // CHECK:STDOUT:
  224. // CHECK:STDOUT: fn @RuntimeCallTooMany(%a: i32, %b: i32, %c: i32) -> i32 {
  225. // CHECK:STDOUT: !entry:
  226. // CHECK:STDOUT: %TooMany.ref: <function> = name_ref TooMany, file.%TooMany [template = file.%TooMany]
  227. // CHECK:STDOUT: %a.ref: i32 = name_ref a, %a
  228. // CHECK:STDOUT: %b.ref: i32 = name_ref b, %b
  229. // CHECK:STDOUT: %c.ref: i32 = name_ref c, %c
  230. // CHECK:STDOUT: %.loc36_17.1: init i32 = call %TooMany.ref(%a.ref, %b.ref, %c.ref)
  231. // CHECK:STDOUT: %.loc36_26: i32 = value_of_initializer %.loc36_17.1
  232. // CHECK:STDOUT: %.loc36_17.2: i32 = converted %.loc36_17.1, %.loc36_26
  233. // CHECK:STDOUT: return %.loc36_17.2
  234. // CHECK:STDOUT: }
  235. // CHECK:STDOUT:
  236. // CHECK:STDOUT: fn @RuntimeCallBadReturnType(%a: i32, %b: i32) -> bool {
  237. // CHECK:STDOUT: !entry:
  238. // CHECK:STDOUT: %BadReturnType.ref: <function> = name_ref BadReturnType, file.%BadReturnType [template = file.%BadReturnType]
  239. // CHECK:STDOUT: %a.ref: i32 = name_ref a, %a
  240. // CHECK:STDOUT: %b.ref: i32 = name_ref b, %b
  241. // CHECK:STDOUT: %.loc40_23.1: init bool = call %BadReturnType.ref(%a.ref, %b.ref)
  242. // CHECK:STDOUT: %.loc40_29: bool = value_of_initializer %.loc40_23.1
  243. // CHECK:STDOUT: %.loc40_23.2: bool = converted %.loc40_23.1, %.loc40_29
  244. // CHECK:STDOUT: return %.loc40_23.2
  245. // CHECK:STDOUT: }
  246. // CHECK:STDOUT:
  247. // CHECK:STDOUT: --- fail_overflow.carbon
  248. // CHECK:STDOUT:
  249. // CHECK:STDOUT: constants {
  250. // CHECK:STDOUT: %.1: i32 = int_literal 2147483647 [template]
  251. // CHECK:STDOUT: %.2: i32 = int_literal 0 [template]
  252. // CHECK:STDOUT: %.3: i32 = int_literal 1 [template]
  253. // CHECK:STDOUT: %.4: i32 = int_literal 2147483648 [template]
  254. // CHECK:STDOUT: }
  255. // CHECK:STDOUT:
  256. // CHECK:STDOUT: file {
  257. // CHECK:STDOUT: package: <namespace> = namespace [template] {
  258. // CHECK:STDOUT: .Add = %Add
  259. // CHECK:STDOUT: }
  260. // CHECK:STDOUT: %Add: <function> = fn_decl @Add [template] {
  261. // CHECK:STDOUT: %a.loc4_8.1: i32 = param a
  262. // CHECK:STDOUT: @Add.%a: i32 = bind_name a, %a.loc4_8.1
  263. // CHECK:STDOUT: %b.loc4_16.1: i32 = param b
  264. // CHECK:STDOUT: @Add.%b: i32 = bind_name b, %b.loc4_16.1
  265. // CHECK:STDOUT: %return.var: ref i32 = var <return slot>
  266. // CHECK:STDOUT: }
  267. // CHECK:STDOUT: %Add.ref.loc6: <function> = name_ref Add, %Add [template = %Add]
  268. // CHECK:STDOUT: %.loc6_18: i32 = int_literal 2147483647 [template = constants.%.1]
  269. // CHECK:STDOUT: %.loc6_30: i32 = int_literal 0 [template = constants.%.2]
  270. // CHECK:STDOUT: %.loc6_17.1: init i32 = call %Add.ref.loc6(%.loc6_18, %.loc6_30) [template = constants.%.1]
  271. // CHECK:STDOUT: %.loc6_32: i32 = value_of_initializer %.loc6_17.1 [template = constants.%.1]
  272. // CHECK:STDOUT: %.loc6_17.2: i32 = converted %.loc6_17.1, %.loc6_32 [template = constants.%.1]
  273. // CHECK:STDOUT: %a.loc6: i32 = bind_name a, %.loc6_17.2
  274. // CHECK:STDOUT: %Add.ref.loc10: <function> = name_ref Add, %Add [template = %Add]
  275. // CHECK:STDOUT: %.loc10_18: i32 = int_literal 2147483647 [template = constants.%.1]
  276. // CHECK:STDOUT: %.loc10_30: i32 = int_literal 1 [template = constants.%.3]
  277. // CHECK:STDOUT: %.loc10_17.1: init i32 = call %Add.ref.loc10(%.loc10_18, %.loc10_30) [template = constants.%.4]
  278. // CHECK:STDOUT: %.loc10_32: i32 = value_of_initializer %.loc10_17.1 [template = constants.%.4]
  279. // CHECK:STDOUT: %.loc10_17.2: i32 = converted %.loc10_17.1, %.loc10_32 [template = constants.%.4]
  280. // CHECK:STDOUT: %b.loc10: i32 = bind_name b, %.loc10_17.2
  281. // CHECK:STDOUT: }
  282. // CHECK:STDOUT:
  283. // CHECK:STDOUT: fn @Add(%a: i32, %b: i32) -> i32 = "int.add";
  284. // CHECK:STDOUT: