2022-04-07 18:03:55 +08:00
|
|
|
// RUN: %clang_cc1 -no-opaque-pointers -triple x86_64-unknown-freebsd -S -emit-llvm -fno-use-init-array -fobjc-runtime=gnustep-2.0 -o - %s | FileCheck %s -check-prefix=CHECK-NEW
|
|
|
|
// RUN: %clang_cc1 -no-opaque-pointers -triple x86_64-pc-windows-msvc -S -emit-llvm -fno-use-init-array -fobjc-runtime=gnustep-2.0 -o - %s | FileCheck %s -check-prefix=CHECK-WIN
|
|
|
|
// RUN: %clang_cc1 -no-opaque-pointers -triple x86_64-unknown-freebsd -S -emit-llvm -fno-use-init-array -fobjc-runtime=gnustep-1.8 -o - %s | FileCheck %s -check-prefix=CHECK-OLD
|
|
|
|
// RUN: %clang_cc1 -no-opaque-pointers -triple x86_64-unknown-freebsd -S -emit-llvm -fobjc-runtime=gnustep-2.0 -o - %s | FileCheck %s -check-prefix=CHECK-INIT_ARRAY
|
2018-05-22 18:13:06 +08:00
|
|
|
|
|
|
|
// Almost minimal Objective-C file, check that it emits calls to the correct
|
|
|
|
// runtime entry points.
|
|
|
|
@interface X @end
|
|
|
|
@implementation X @end
|
|
|
|
|
|
|
|
|
|
|
|
// Check that we emit a class ref
|
|
|
|
// CHECK-NEW: @._OBJC_REF_CLASS_X
|
|
|
|
// CHECK-NEW-SAME: section "__objc_class_refs"
|
|
|
|
|
|
|
|
// Check that we get a class ref to the defined class.
|
|
|
|
// CHECK-NEW: @._OBJC_INIT_CLASS_X = global
|
2019-02-12 04:33:22 +08:00
|
|
|
// CHECK-NEW-SAME: @._OBJC_CLASS_X, section "__objc_classes"
|
2018-05-22 18:13:06 +08:00
|
|
|
|
|
|
|
// Check that we emit the section start and end symbols as hidden globals.
|
|
|
|
// CHECK-NEW: @__start___objc_selectors = external hidden global i8*
|
|
|
|
// CHECK-NEW: @__stop___objc_selectors = external hidden global i8*
|
|
|
|
// CHECK-NEW: @__start___objc_classes = external hidden global i8*
|
|
|
|
// CHECK-NEW: @__stop___objc_classes = external hidden global i8*
|
|
|
|
// CHECK-NEW: @__start___objc_class_refs = external hidden global i8*
|
|
|
|
// CHECK-NEW: @__stop___objc_class_refs = external hidden global i8*
|
|
|
|
// CHECK-NEW: @__start___objc_cats = external hidden global i8*
|
|
|
|
// CHECK-NEW: @__stop___objc_cats = external hidden global i8*
|
|
|
|
// CHECK-NEW: @__start___objc_protocols = external hidden global i8*
|
|
|
|
// CHECK-NEW: @__stop___objc_protocols = external hidden global i8*
|
|
|
|
// CHECK-NEW: @__start___objc_protocol_refs = external hidden global i8*
|
|
|
|
// CHECK-NEW: @__stop___objc_protocol_refs = external hidden global i8*
|
|
|
|
// CHECK-NEW: @__start___objc_class_aliases = external hidden global i8*
|
|
|
|
// CHECK-NEW: @__stop___objc_class_aliases = external hidden global i8*
|
|
|
|
// CHECK-NEW: @__start___objc_constant_string = external hidden global i8*
|
|
|
|
// CHECK-NEW: @__stop___objc_constant_string = external hidden global i8*
|
|
|
|
|
|
|
|
// Check that we emit the init structure correctly, including in a comdat.
|
|
|
|
// CHECK-NEW: @.objc_init = linkonce_odr hidden global { i64, i8**, i8**, i8**, i8**, i8**, i8**, i8**, i8**, i8**, i8**, i8**, i8**, i8**, i8**, i8**, i8** } { i64 0, i8** @__start___objc_selectors, i8** @__stop___objc_selectors, i8** @__start___objc_classes, i8** @__stop___objc_classes, i8** @__start___objc_class_refs, i8** @__stop___objc_class_refs, i8** @__start___objc_cats, i8** @__stop___objc_cats, i8** @__start___objc_protocols, i8** @__stop___objc_protocols, i8** @__start___objc_protocol_refs, i8** @__stop___objc_protocol_refs, i8** @__start___objc_class_aliases, i8** @__stop___objc_class_aliases, i8** @__start___objc_constant_string, i8** @__stop___objc_constant_string }, comdat, align 8
|
|
|
|
|
|
|
|
// Check that the load function is manually inserted into .ctors.
|
2020-04-14 21:13:13 +08:00
|
|
|
// CHECK-NEW: @.objc_ctor = linkonce hidden global void ()* @.objcv2_load_function, section ".ctors", comdat
|
|
|
|
// CHECK-INIT_ARRAY: @.objc_ctor = linkonce hidden global void ()* @.objcv2_load_function, section ".init_array", comdat
|
2018-05-22 18:13:06 +08:00
|
|
|
|
|
|
|
|
|
|
|
// Make sure that we provide null versions of everything so the __start /
|
|
|
|
// __stop symbols work.
|
|
|
|
// CHECK-NEW: @.objc_null_selector = linkonce_odr hidden global { i8*, i8* } zeroinitializer, section "__objc_selectors", comdat, align 8
|
|
|
|
// CHECK-NEW: @.objc_null_category = linkonce_odr hidden global { i8*, i8*, i8*, i8*, i8*, i8*, i8* } zeroinitializer, section "__objc_cats", comdat, align 8
|
|
|
|
// CHECK-NEW: @.objc_null_protocol = linkonce_odr hidden global { i8*, i8*, i8*, i8*, i8*, i8*, i8*, i8*, i8*, i8*, i8* } zeroinitializer, section "__objc_protocols", comdat, align 8
|
|
|
|
// CHECK-NEW: @.objc_null_protocol_ref = linkonce_odr hidden global { i8* } zeroinitializer, section "__objc_protocol_refs", comdat, align 8
|
|
|
|
// CHECK-NEW: @.objc_null_class_alias = linkonce_odr hidden global { i8*, i8* } zeroinitializer, section "__objc_class_aliases", comdat, align 8
|
|
|
|
// CHECK-NEW: @.objc_null_constant_string = linkonce_odr hidden global { i8*, i32, i32, i32, i32, i8* } zeroinitializer, section "__objc_constant_string", comdat, align 8
|
|
|
|
// Make sure that the null symbols are not going to be removed, even by linking.
|
2018-08-10 20:53:13 +08:00
|
|
|
// CHECK-NEW: @llvm.used = appending global [8 x i8*] [i8* bitcast ({ { i8*, i8*, i8*, i64, i64, i64, i8*, i8*, i8*, i8*, i8*, i8*, i8*, i8*, i8*, i64, i8* }*, i8*, i8*, i64, i64, i64, i8*, i8*, i8*, i8*, i8*, i8*, i8*, i8*, i8*, i64, i8* }** @._OBJC_INIT_CLASS_X to i8*), i8* bitcast (void ()** @.objc_ctor to i8*), i8* bitcast ({ i8*, i8* }* @.objc_null_selector to i8*), i8* bitcast ({ i8*, i8*, i8*, i8*, i8*, i8*, i8* }* @.objc_null_category to i8*), i8* bitcast ({ i8*, i8*, i8*, i8*, i8*, i8*, i8*, i8*, i8*, i8*, i8* }* @.objc_null_protocol to i8*), i8* bitcast ({ i8* }* @.objc_null_protocol_ref to i8*), i8* bitcast ({ i8*, i8* }* @.objc_null_class_alias to i8*), i8* bitcast ({ i8*, i32, i32, i32, i32, i8* }* @.objc_null_constant_string to i8*)], section "llvm.metadata"
|
2018-05-22 18:13:06 +08:00
|
|
|
// Make sure that the load function and the reference to it are marked as used.
|
2018-08-10 20:53:13 +08:00
|
|
|
// CHECK-NEW: @llvm.compiler.used = appending global [1 x i8*] [i8* bitcast (void ()* @.objcv2_load_function to i8*)], section "llvm.metadata"
|
2018-05-22 18:13:06 +08:00
|
|
|
|
|
|
|
// Check that we emit the load function in a comdat and that it does the right thing.
|
|
|
|
// CHECK-NEW: define linkonce_odr hidden void @.objcv2_load_function() comdat {
|
|
|
|
// CHECK-NEW-NEXT: entry:
|
|
|
|
// CHECK-NEW-NEXT: call void @__objc_load(
|
|
|
|
// CHECK-NEW-SAME: @.objc_init
|
|
|
|
// CHECK-NEW-NEXT: ret void
|
|
|
|
|
2018-05-22 18:13:14 +08:00
|
|
|
// CHECK-OLD: @4 = internal global { i64, i64, i8*, { i64, { i8*, i8* }*, i16, i16, [4 x i8*] }* } { i64 9, i64 32, i8* getelementptr inbounds ([{{[0-9]+}} x i8], [{{[0-9]+}} x i8]* @.objc_source_file_name, i64 0, i64 0), { i64, { i8*, i8* }*, i16, i16, [4 x i8*] }* @3 }, align 8
|
2018-05-22 18:13:06 +08:00
|
|
|
// CHECK-OLD: @llvm.global_ctors = appending global [1 x { i32, void ()*, i8* }] [{ i32, void ()*, i8* } { i32 65535, void ()* @.objc_load_function, i8* null }]
|
|
|
|
|
|
|
|
// CHECK-OLD: define internal void @.objc_load_function() {
|
|
|
|
// CHECK-OLD-NEXT: entry:
|
|
|
|
// CHECK-OLD-NEXT: call void ({ i64, i64, i8*, { i64, { i8*, i8* }*, i16, i16, [4 x i8*] }* }*, ...) @__objc_exec_class({ i64, i64, i8*, { i64, { i8*, i8* }*, i16, i16, [4 x i8*] }* }* @4)
|
|
|
|
|
2018-08-10 20:53:13 +08:00
|
|
|
|
|
|
|
|
|
|
|
// Make sure all of our section boundary variables are emitted correctly.
|
2019-03-31 19:22:33 +08:00
|
|
|
// CHECK-WIN-DAG: @"__stop.objcrt$SEL" = linkonce_odr hidden global %.objc_section_sentinel zeroinitializer, section ".objcrt$SEL$z", comdat, align 8
|
|
|
|
// CHECK-WIN-DAG: @"__start_.objcrt$CLS" = linkonce_odr hidden global %.objc_section_sentinel zeroinitializer, section ".objcrt$CLS$a", comdat, align 8
|
|
|
|
// CHECK-WIN-DAG: @"__stop.objcrt$CLS" = linkonce_odr hidden global %.objc_section_sentinel zeroinitializer, section ".objcrt$CLS$z", comdat, align 8
|
|
|
|
// CHECK-WIN-DAG: @"__start_.objcrt$CLR" = linkonce_odr hidden global %.objc_section_sentinel zeroinitializer, section ".objcrt$CLR$a", comdat, align 8
|
|
|
|
// CHECK-WIN-DAG: @"__stop.objcrt$CLR" = linkonce_odr hidden global %.objc_section_sentinel zeroinitializer, section ".objcrt$CLR$z", comdat, align 8
|
|
|
|
// CHECK-WIN-DAG: @"__start_.objcrt$CAT" = linkonce_odr hidden global %.objc_section_sentinel zeroinitializer, section ".objcrt$CAT$a", comdat, align 8
|
|
|
|
// CHECK-WIN-DAG: @"__stop.objcrt$CAT" = linkonce_odr hidden global %.objc_section_sentinel zeroinitializer, section ".objcrt$CAT$z", comdat, align 8
|
|
|
|
// CHECK-WIN-DAG: @"__start_.objcrt$PCL" = linkonce_odr hidden global %.objc_section_sentinel zeroinitializer, section ".objcrt$PCL$a", comdat, align 8
|
|
|
|
// CHECK-WIN-DAG: @"__stop.objcrt$PCL" = linkonce_odr hidden global %.objc_section_sentinel zeroinitializer, section ".objcrt$PCL$z", comdat, align 8
|
|
|
|
// CHECK-WIN-DAG: @"__start_.objcrt$PCR" = linkonce_odr hidden global %.objc_section_sentinel zeroinitializer, section ".objcrt$PCR$a", comdat, align 8
|
|
|
|
// CHECK-WIN-DAG: @"__stop.objcrt$PCR" = linkonce_odr hidden global %.objc_section_sentinel zeroinitializer, section ".objcrt$PCR$z", comdat, align 8
|
|
|
|
// CHECK-WIN-DAG: @"__start_.objcrt$CAL" = linkonce_odr hidden global %.objc_section_sentinel zeroinitializer, section ".objcrt$CAL$a", comdat, align 8
|
|
|
|
// CHECK-WIN-DAG: @"__stop.objcrt$CAL" = linkonce_odr hidden global %.objc_section_sentinel zeroinitializer, section ".objcrt$CAL$z", comdat, align 8
|
|
|
|
// CHECK-WIN-DAG: @"__start_.objcrt$STR" = linkonce_odr hidden global %.objc_section_sentinel zeroinitializer, section ".objcrt$STR$a", comdat, align 8
|
|
|
|
// CHECK-WIN-DAG: @"__stop.objcrt$STR" = linkonce_odr hidden global %.objc_section_sentinel zeroinitializer, section ".objcrt$STR$z", comdat, align 8
|
|
|
|
// CHECK-WIN-DAG: @.objc_init = linkonce_odr hidden global { i64, %.objc_section_sentinel*, %.objc_section_sentinel*, %.objc_section_sentinel*, %.objc_section_sentinel*, %.objc_section_sentinel*, %.objc_section_sentinel*, %.objc_section_sentinel*, %.objc_section_sentinel*, %.objc_section_sentinel*, %.objc_section_sentinel*, %.objc_section_sentinel*, %.objc_section_sentinel*, %.objc_section_sentinel*, %.objc_section_sentinel*, %.objc_section_sentinel*, %.objc_section_sentinel* } { i64 0, %.objc_section_sentinel* @"__start_.objcrt$SEL", %.objc_section_sentinel* @"__stop.objcrt$SEL", %.objc_section_sentinel* @"__start_.objcrt$CLS", %.objc_section_sentinel* @"__stop.objcrt$CLS", %.objc_section_sentinel* @"__start_.objcrt$CLR", %.objc_section_sentinel* @"__stop.objcrt$CLR", %.objc_section_sentinel* @"__start_.objcrt$CAT", %.objc_section_sentinel* @"__stop.objcrt$CAT", %.objc_section_sentinel* @"__start_.objcrt$PCL", %.objc_section_sentinel* @"__stop.objcrt$PCL", %.objc_section_sentinel* @"__start_.objcrt$PCR", %.objc_section_sentinel* @"__stop.objcrt$PCR", %.objc_section_sentinel* @"__start_.objcrt$CAL", %.objc_section_sentinel* @"__stop.objcrt$CAL", %.objc_section_sentinel* @"__start_.objcrt$STR", %.objc_section_sentinel* @"__stop.objcrt$STR" }, comdat, align 8
|
2018-08-10 20:53:13 +08:00
|
|
|
|
|
|
|
// Make sure our init variable is in the correct section for late library init.
|
2020-04-14 21:13:13 +08:00
|
|
|
// CHECK-WIN: @.objc_ctor = linkonce hidden global void ()* @.objcv2_load_function, section ".CRT$XCLz", comdat
|
2018-08-10 20:53:13 +08:00
|
|
|
|
|
|
|
// We shouldn't have emitted any null placeholders on Windows.
|
2019-03-31 19:22:33 +08:00
|
|
|
// CHECK-WIN: @llvm.used = appending global [2 x i8*] [i8* bitcast ({ { i8*, i8*, i8*, i32, i32, i32, i8*, i8*, i8*, i8*, i8*, i8*, i8*, i8*, i8*, i32, i8* }*, i8*, i8*, i32, i32, i32, i8*, i8*, i8*, i8*, i8*, i8*, i8*, i8*, i8*, i32, i8* }** @"$_OBJC_INIT_CLASS_X" to i8*), i8* bitcast (void ()** @.objc_ctor to i8*)], section "llvm.metadata"
|
2018-08-10 20:53:13 +08:00
|
|
|
// CHECK-WIN: @llvm.compiler.used = appending global [1 x i8*] [i8* bitcast (void ()* @.objcv2_load_function to i8*)], section "llvm.metadata"
|
|
|
|
|
|
|
|
// Check our load function is in a comdat.
|
|
|
|
// CHECK-WIN: define linkonce_odr hidden void @.objcv2_load_function() comdat {
|
|
|
|
|
2019-04-30 07:05:47 +08:00
|
|
|
// Make sure we do not have dllimport on the load function
|
|
|
|
// CHECK-WIN: declare dso_local void @__objc_load
|
2018-08-10 20:53:13 +08:00
|
|
|
|