// Part of the Carbon Language project, under the Apache License v2.0 with LLVM // Exceptions. See /LICENSE for license information. // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception // // AUTOUPDATE class Class; // CHECK:STDERR: fail_incomplete.carbon:[[@LINE+6]]:4: ERROR: Cannot declare a member of incomplete class `Class`. // CHECK:STDERR: fn Class.Function() {} // CHECK:STDERR: ^~~~~ // CHECK:STDERR: fail_incomplete.carbon:[[@LINE-5]]:1: Class was forward declared here. // CHECK:STDERR: class Class; // CHECK:STDERR: ^~~~~~~~~~~~ fn Class.Function() {} fn CallClassFunction() { // CHECK:STDERR: fail_incomplete.carbon:[[@LINE+6]]:3: ERROR: Member access into incomplete class `Class`. // CHECK:STDERR: Class.Function(); // CHECK:STDERR: ^~~~~~~~~~~~~~ // CHECK:STDERR: fail_incomplete.carbon:[[@LINE-14]]:1: Class was forward declared here. // CHECK:STDERR: class Class; // CHECK:STDERR: ^~~~~~~~~~~~ Class.Function(); } // CHECK:STDERR: fail_incomplete.carbon:[[@LINE+6]]:17: ERROR: Variable has incomplete type `Class`. // CHECK:STDERR: var global_var: Class; // CHECK:STDERR: ^~~~~ // CHECK:STDERR: fail_incomplete.carbon:[[@LINE-23]]:1: Class was forward declared here. // CHECK:STDERR: class Class; // CHECK:STDERR: ^~~~~~~~~~~~ var global_var: Class; // CHECK:STDERR: fail_incomplete.carbon:[[@LINE+6]]:24: ERROR: Function returns incomplete type `Class`. // CHECK:STDERR: fn ConvertFromStruct() -> Class { return {}; } // CHECK:STDERR: ^~~~~~~~ // CHECK:STDERR: fail_incomplete.carbon:[[@LINE-31]]:1: Class was forward declared here. // CHECK:STDERR: class Class; // CHECK:STDERR: ^~~~~~~~~~~~ fn ConvertFromStruct() -> Class { return {}; } // TODO: Once the `->` operator is supported: // TODO: fn G(p: Class*) -> i32 { // TODO: return p->n; // TODO: } fn MemberAccess(p: Class*) -> i32 { // CHECK:STDERR: fail_incomplete.carbon:[[@LINE+6]]:11: ERROR: Member access into object of incomplete type `Class`. // CHECK:STDERR: return (*p).n; // CHECK:STDERR: ^~ // CHECK:STDERR: fail_incomplete.carbon:[[@LINE-45]]:1: Class was forward declared here. // CHECK:STDERR: class Class; // CHECK:STDERR: ^~~~~~~~~~~~ return (*p).n; } // CHECK:STDERR: fail_incomplete.carbon:[[@LINE+6]]:20: ERROR: Function returns incomplete type `Class`. // CHECK:STDERR: fn Copy(p: Class*) -> Class { // CHECK:STDERR: ^~~~~~~~ // CHECK:STDERR: fail_incomplete.carbon:[[@LINE-54]]:1: Class was forward declared here. // CHECK:STDERR: class Class; // CHECK:STDERR: ^~~~~~~~~~~~ fn Copy(p: Class*) -> Class { return *p; } fn Let(p: Class*) { // CHECK:STDERR: fail_incomplete.carbon:[[@LINE+6]]:10: ERROR: `let` binding has incomplete type `Class`. // CHECK:STDERR: let c: Class = *p; // CHECK:STDERR: ^~~~~ // CHECK:STDERR: fail_incomplete.carbon:[[@LINE-65]]:1: Class was forward declared here. // CHECK:STDERR: class Class; // CHECK:STDERR: ^~~~~~~~~~~~ let c: Class = *p; } fn TakeIncomplete(c: Class); // TODO: We should allow this, and only reject calls to the function. // // CHECK:STDERR: fail_incomplete.carbon:[[@LINE+6]]:23: ERROR: Function returns incomplete type `Class`. // CHECK:STDERR: fn ReturnIncomplete() -> Class; // CHECK:STDERR: ^~~~~~~~ // CHECK:STDERR: fail_incomplete.carbon:[[@LINE-78]]:1: Class was forward declared here. // CHECK:STDERR: class Class; // CHECK:STDERR: ^~~~~~~~~~~~ fn ReturnIncomplete() -> Class; fn CallTakeIncomplete(p: Class*) { // CHECK:STDERR: fail_incomplete.carbon:[[@LINE+9]]:3: ERROR: Forming value of incomplete type `Class`. // CHECK:STDERR: TakeIncomplete(*p); // CHECK:STDERR: ^~~~~~~~~~~~~~~ // CHECK:STDERR: fail_incomplete.carbon:[[@LINE-87]]:1: Class was forward declared here. // CHECK:STDERR: class Class; // CHECK:STDERR: ^~~~~~~~~~~~ // CHECK:STDERR: fail_incomplete.carbon:[[@LINE-19]]:1: Initializing parameter 1 of function declared here. // CHECK:STDERR: fn TakeIncomplete(c: Class); // CHECK:STDERR: ^~~~~~~~~~~~~~~~~~~~~~~~~~~~ TakeIncomplete(*p); // CHECK:STDERR: fail_incomplete.carbon:[[@LINE+9]]:3: ERROR: Forming value of incomplete type `Class`. // CHECK:STDERR: TakeIncomplete({}); // CHECK:STDERR: ^~~~~~~~~~~~~~~ // CHECK:STDERR: fail_incomplete.carbon:[[@LINE-98]]:1: Class was forward declared here. // CHECK:STDERR: class Class; // CHECK:STDERR: ^~~~~~~~~~~~ // CHECK:STDERR: fail_incomplete.carbon:[[@LINE-30]]:1: Initializing parameter 1 of function declared here. // CHECK:STDERR: fn TakeIncomplete(c: Class); // CHECK:STDERR: ^~~~~~~~~~~~~~~~~~~~~~~~~~~~ TakeIncomplete({}); } fn CallReturnIncomplete() { ReturnIncomplete(); } // CHECK:STDOUT: --- fail_incomplete.carbon // CHECK:STDOUT: // CHECK:STDOUT: constants { // CHECK:STDOUT: %Class: type = class_type @Class [template] // CHECK:STDOUT: %.1: type = struct_type {} [template] // CHECK:STDOUT: %.2: type = ptr_type Class [template] // CHECK:STDOUT: %.3: type = tuple_type () [template] // CHECK:STDOUT: } // CHECK:STDOUT: // CHECK:STDOUT: file { // CHECK:STDOUT: package: = namespace [template] { // CHECK:STDOUT: .Class = %Class.decl // CHECK:STDOUT: .CallClassFunction = %CallClassFunction // CHECK:STDOUT: .global_var = %global_var // CHECK:STDOUT: .ConvertFromStruct = %ConvertFromStruct // CHECK:STDOUT: .MemberAccess = %MemberAccess // CHECK:STDOUT: .Copy = %Copy // CHECK:STDOUT: .Let = %Let // CHECK:STDOUT: .TakeIncomplete = %TakeIncomplete // CHECK:STDOUT: .ReturnIncomplete = %ReturnIncomplete // CHECK:STDOUT: .CallTakeIncomplete = %CallTakeIncomplete // CHECK:STDOUT: .CallReturnIncomplete = %CallReturnIncomplete // CHECK:STDOUT: } // CHECK:STDOUT: %Class.decl: type = class_decl @Class [template = constants.%Class] {} // CHECK:STDOUT: %.loc15: = fn_decl @.1 [template] {} // CHECK:STDOUT: %CallClassFunction: = fn_decl @CallClassFunction [template] {} // CHECK:STDOUT: %Class.ref.loc33: type = name_ref Class, %Class.decl [template = constants.%Class] // CHECK:STDOUT: %global_var.var: ref = var global_var // CHECK:STDOUT: %global_var: ref = bind_name global_var, %global_var.var // CHECK:STDOUT: %ConvertFromStruct: = fn_decl @ConvertFromStruct [template] { // CHECK:STDOUT: %Class.ref.loc41: type = name_ref Class, %Class.decl [template = constants.%Class] // CHECK:STDOUT: %return.var.loc41: ref Class = var // CHECK:STDOUT: } // CHECK:STDOUT: %MemberAccess: = fn_decl @MemberAccess [template] { // CHECK:STDOUT: %Class.ref.loc48: type = name_ref Class, %Class.decl [template = constants.%Class] // CHECK:STDOUT: %.loc48: type = ptr_type Class [template = constants.%.2] // CHECK:STDOUT: %p.loc48_17.1: Class* = param p // CHECK:STDOUT: @MemberAccess.%p: Class* = bind_name p, %p.loc48_17.1 // CHECK:STDOUT: %return.var.loc48: ref i32 = var // CHECK:STDOUT: } // CHECK:STDOUT: %Copy: = fn_decl @Copy [template] { // CHECK:STDOUT: %Class.ref.loc64_12: type = name_ref Class, %Class.decl [template = constants.%Class] // CHECK:STDOUT: %.loc64: type = ptr_type Class [template = constants.%.2] // CHECK:STDOUT: %p.loc64_9.1: Class* = param p // CHECK:STDOUT: @Copy.%p: Class* = bind_name p, %p.loc64_9.1 // CHECK:STDOUT: %Class.ref.loc64_23: type = name_ref Class, %Class.decl [template = constants.%Class] // CHECK:STDOUT: %return.var.loc64: ref Class = var // CHECK:STDOUT: } // CHECK:STDOUT: %Let: = fn_decl @Let [template] { // CHECK:STDOUT: %Class.ref.loc68: type = name_ref Class, %Class.decl [template = constants.%Class] // CHECK:STDOUT: %.loc68: type = ptr_type Class [template = constants.%.2] // CHECK:STDOUT: %p.loc68_8.1: Class* = param p // CHECK:STDOUT: @Let.%p: Class* = bind_name p, %p.loc68_8.1 // CHECK:STDOUT: } // CHECK:STDOUT: %TakeIncomplete: = fn_decl @TakeIncomplete [template] { // CHECK:STDOUT: %Class.ref.loc78: type = name_ref Class, %Class.decl [template = constants.%Class] // CHECK:STDOUT: %c.loc78_19.1: Class = param c // CHECK:STDOUT: @TakeIncomplete.%c: Class = bind_name c, %c.loc78_19.1 // CHECK:STDOUT: } // CHECK:STDOUT: %ReturnIncomplete: = fn_decl @ReturnIncomplete [template] { // CHECK:STDOUT: %Class.ref.loc88: type = name_ref Class, %Class.decl [template = constants.%Class] // CHECK:STDOUT: %return.var.loc88: ref Class = var // CHECK:STDOUT: } // CHECK:STDOUT: %CallTakeIncomplete: = fn_decl @CallTakeIncomplete [template] { // CHECK:STDOUT: %Class.ref.loc90: type = name_ref Class, %Class.decl [template = constants.%Class] // CHECK:STDOUT: %.loc90: type = ptr_type Class [template = constants.%.2] // CHECK:STDOUT: %p.loc90_23.1: Class* = param p // CHECK:STDOUT: @CallTakeIncomplete.%p: Class* = bind_name p, %p.loc90_23.1 // CHECK:STDOUT: } // CHECK:STDOUT: %CallReturnIncomplete: = fn_decl @CallReturnIncomplete [template] {} // CHECK:STDOUT: } // CHECK:STDOUT: // CHECK:STDOUT: class @Class; // CHECK:STDOUT: // CHECK:STDOUT: fn @.1() { // CHECK:STDOUT: !entry: // CHECK:STDOUT: return // CHECK:STDOUT: } // CHECK:STDOUT: // CHECK:STDOUT: fn @CallClassFunction() { // CHECK:STDOUT: !entry: // CHECK:STDOUT: %Class.ref: type = name_ref Class, file.%Class.decl [template = constants.%Class] // CHECK:STDOUT: %Function.ref: = name_ref Function, [template = ] // CHECK:STDOUT: return // CHECK:STDOUT: } // CHECK:STDOUT: // CHECK:STDOUT: fn @ConvertFromStruct() -> { // CHECK:STDOUT: !entry: // CHECK:STDOUT: %.loc41: {} = struct_literal () // CHECK:STDOUT: return // CHECK:STDOUT: } // CHECK:STDOUT: // CHECK:STDOUT: fn @MemberAccess(%p: Class*) -> i32 { // CHECK:STDOUT: !entry: // CHECK:STDOUT: %p.ref: Class* = name_ref p, %p // CHECK:STDOUT: %.loc55: ref Class = deref %p.ref // CHECK:STDOUT: return // CHECK:STDOUT: } // CHECK:STDOUT: // CHECK:STDOUT: fn @Copy(%p: Class*) -> { // CHECK:STDOUT: !entry: // CHECK:STDOUT: %p.ref: Class* = name_ref p, %p // CHECK:STDOUT: %.loc65: ref Class = deref %p.ref // CHECK:STDOUT: return // CHECK:STDOUT: } // CHECK:STDOUT: // CHECK:STDOUT: fn @Let(%p: Class*) { // CHECK:STDOUT: !entry: // CHECK:STDOUT: %Class.ref: type = name_ref Class, file.%Class.decl [template = constants.%Class] // CHECK:STDOUT: %p.ref: Class* = name_ref p, %p // CHECK:STDOUT: %.loc75: ref Class = deref %p.ref // CHECK:STDOUT: %c: = bind_name c, // CHECK:STDOUT: return // CHECK:STDOUT: } // CHECK:STDOUT: // CHECK:STDOUT: fn @TakeIncomplete(%c: Class); // CHECK:STDOUT: // CHECK:STDOUT: fn @ReturnIncomplete() -> ; // CHECK:STDOUT: // CHECK:STDOUT: fn @CallTakeIncomplete(%p: Class*) { // CHECK:STDOUT: !entry: // CHECK:STDOUT: %TakeIncomplete.ref.loc100: = name_ref TakeIncomplete, file.%TakeIncomplete [template = file.%TakeIncomplete] // CHECK:STDOUT: %p.ref: Class* = name_ref p, %p // CHECK:STDOUT: %.loc100_18: ref Class = deref %p.ref // CHECK:STDOUT: %.loc100_17: init () = call %TakeIncomplete.ref.loc100() // CHECK:STDOUT: %TakeIncomplete.ref.loc111: = name_ref TakeIncomplete, file.%TakeIncomplete [template = file.%TakeIncomplete] // CHECK:STDOUT: %.loc111_19: {} = struct_literal () // CHECK:STDOUT: %.loc111_17: init () = call %TakeIncomplete.ref.loc111() // CHECK:STDOUT: return // CHECK:STDOUT: } // CHECK:STDOUT: // CHECK:STDOUT: fn @CallReturnIncomplete() { // CHECK:STDOUT: !entry: // CHECK:STDOUT: %ReturnIncomplete.ref: = name_ref ReturnIncomplete, file.%ReturnIncomplete [template = file.%ReturnIncomplete] // CHECK:STDOUT: %.loc115: init = call %ReturnIncomplete.ref() // CHECK:STDOUT: return // CHECK:STDOUT: } // CHECK:STDOUT: