// RUN: %target-swift-frontend -emit-ir -swift-version 5 -O -primary-file %s | %FileCheck %s --check-prefix=CHECK --check-prefix=CHECK-%target-ptrsize // // REQUIRES: OS=macosx || OS=ios || OS=tvos || OS=watchos // This tests the optimality of the IR generated for the new os log APIs. This // is not testing the output of a specific optimization pass (which has separate // tests) but that all optimizations together result in optimal IR. If this test // fails, it implies that some expected optimizations fail to get triggered on // os log APIs. TODO: eventually these optimization should also happen in Onone // mode. import OSLogTestHelper import Foundation // CHECK-LABEL: define hidden swiftcc void @"${{.*}}testSimpleInterpolation func testSimpleInterpolation() { _osLogTestHelper("Minimum integer value: \(Int.min)") // CHECK: entry: // Ignore some code related to the default argument and string literal invariant // checks. // CHECK: tail call swiftcc i1 @"${{.*}}isLoggingEnabled{{.*}}"() // CHECK-NEXT: br i1 {{%.*}}, label %[[ENABLED:[0-9]+]], label %[[NOT_ENABLED:[0-9]+]] // CHECK: [[NOT_ENABLED]]: // CHECK-NEXT: tail call void @swift_release // CHECK-NEXT: ret void // CHECK: [[ENABLED]]: // // Header bytes. // // CHECK-64: [[BUFFER:%.+]] = tail call noalias i8* @swift_slowAlloc(i64 12 // CHECK-32: [[BUFFER:%.+]] = tail call noalias i8* @swift_slowAlloc(i32 8 // CHECK-NEXT: store i8 0, i8* [[BUFFER]], align 1 // CHECK-NEXT: [[OFFSET1:%.+]] = getelementptr inbounds i8, i8* [[BUFFER]], i{{.*}} 1 // CHECK-NEXT: store i8 1, i8* [[OFFSET1]], align 1 // // Argument bytes. // // CHECK-NEXT: [[OFFSET2:%.+]] = getelementptr inbounds i8, i8* [[BUFFER]], i{{.*}} 2 // CHECK-NEXT: store i8 0, i8* [[OFFSET2]], align 1 // CHECK-NEXT: [[OFFSET3:%.+]] = getelementptr inbounds i8, i8* [[BUFFER]], i{{.*}} 3 // CHECK-64-NEXT: store i8 8, i8* [[OFFSET3]], align 1 // CHECK-32-NEXT: store i8 4, i8* [[OFFSET3]], align 1 // CHECK-NEXT: bitcast %swift.refcounted* %{{.*}} to %swift.opaque* // CHECK-NEXT: [[OFFSET4:%.+]] = getelementptr inbounds i8, i8* [[BUFFER]], i{{.*}} 4 // CHECK-NEXT: [[BITCASTED:%.+]] = bitcast i8* [[OFFSET4]] to i{{.*}}* // CHECK-64-NEXT: store i64 -9223372036854775808, i64* [[BITCASTED]], align 1 // CHECK-32-NEXT: store i32 -2147483648, i32* [[BITCASTED]], align 1 // CHECK-64-NEXT: tail call swiftcc void @"${{.*}}_os_log_impl_test{{.*}}"({{.*}}, {{.*}}, {{.*}}, {{.*}}, i8* getelementptr inbounds ([27 x i8], [27 x i8]* @{{.*}}, i64 0, i64 0), i8* {{(nonnull )?}}[[BUFFER]], i32 12) // CHECK-32-NEXT: tail call swiftcc void @"${{.*}}_os_log_impl_test{{.*}}"({{.*}}, {{.*}}, {{.*}}, {{.*}}, i8* getelementptr inbounds ([27 x i8], [27 x i8]* @{{.*}}, i32 0, i32 0), i8* {{(nonnull )?}}[[BUFFER]], i32 8) // CHECK-NEXT: tail call void @swift_slowDealloc(i8* {{(nonnull )?}}[[BUFFER]] // CHECK-NEXT: br label %[[NOT_ENABLED]] } // CHECK-LABEL: define hidden swiftcc void @"${{.*}}testInterpolationWithMultipleArguments func testInterpolationWithMultipleArguments() { let privateID: Int32 = 0x79abcdef let filePermissions: Int32 = 0o777 let pid: Int32 = 122225 _osLogTestHelper( """ Access prevented: process \(pid, privacy: .public) initiated by \ user: \(privateID, privacy: .private) attempted resetting \ permissions to \(filePermissions) """) // CHECK: entry: // CHECK: tail call swiftcc i1 @"${{.*}}isLoggingEnabled{{.*}}"() // CHECK-NEXT: br i1 {{%.*}}, label %[[ENABLED:[0-9]+]], label %[[NOT_ENABLED:[0-9]+]] // CHECK: [[NOT_ENABLED]]: // CHECK-NEXT: tail call void @swift_release // CHECK-NEXT: ret void // CHECK: [[ENABLED]]: // // Header bytes. // // CHECK: [[BUFFER:%.+]] = tail call noalias i8* @swift_slowAlloc(i{{.*}} 20 // CHECK-NEXT: store i8 1, i8* [[BUFFER]], align 1 // CHECK-NEXT: [[OFFSET1:%.+]] = getelementptr inbounds i8, i8* [[BUFFER]], i{{.*}} 1 // CHECK-NEXT: store i8 3, i8* [[OFFSET1]], align 1 // // First argument bytes. // // CHECK-NEXT: [[OFFSET2:%.+]] = getelementptr inbounds i8, i8* [[BUFFER]], i{{.*}} 2 // CHECK-NEXT: store i8 2, i8* [[OFFSET2]], align 1 // CHECK-NEXT: [[OFFSET3:%.+]] = getelementptr inbounds i8, i8* [[BUFFER]], i{{.*}} 3 // CHECK-NEXT: store i8 4, i8* [[OFFSET3]], align 1 // CHECK-NEXT: [[OFFSET4:%.+]] = getelementptr inbounds i8, i8* [[BUFFER]], i{{.*}} 4 // CHECK-NEXT: [[BITCASTED:%.+]] = bitcast i8* [[OFFSET4]] to i32* // CHECK-NEXT: store i32 122225, i32* [[BITCASTED]], align 1 // // Second argument // // CHECK-NEXT: [[OFFSET12:%.+]] = getelementptr inbounds i8, i8* [[BUFFER]], i{{.*}} 8 // CHECK-NEXT: store i8 1, i8* [[OFFSET12]], align 1 // CHECK-NEXT: [[OFFSET13:%.+]] = getelementptr inbounds i8, i8* [[BUFFER]], i{{.*}} 9 // CHECK-NEXT: store i8 4, i8* [[OFFSET13]], align 1 // CHECK-NEXT: [[OFFSET14:%.+]] = getelementptr inbounds i8, i8* [[BUFFER]], i{{.*}} 10 // CHECK-NEXT: [[BITCASTED2:%.+]] = bitcast i8* [[OFFSET14]] to i32* // CHECK-NEXT: store i32 2041302511, i32* [[BITCASTED2]], align 1 // // Third argument // // CHECK-NEXT: [[OFFSET22:%.+]] = getelementptr inbounds i8, i8* [[BUFFER]], i{{.*}} 14 // CHECK-NEXT: store i8 0, i8* [[OFFSET22]], align 1 // CHECK-NEXT: [[OFFSET23:%.+]] = getelementptr inbounds i8, i8* [[BUFFER]], i{{.*}} 15 // CHECK-NEXT: store i8 4, i8* [[OFFSET23]], align 1 // CHECK-NEXT: bitcast %swift.refcounted* %{{.*}} to %swift.opaque* // CHECK-NEXT: [[OFFSET24:%.+]] = getelementptr inbounds i8, i8* [[BUFFER]], i{{.*}} 16 // CHECK-NEXT: [[BITCASTED3:%.+]] = bitcast i8* [[OFFSET24]] to i32* // CHECK-NEXT: store i32 511, i32* [[BITCASTED3]], align 1 // // os_log_impl call. // CHECK-NEXT: tail call swiftcc void @"${{.*}}_os_log_impl_test{{.*}}"({{.*}}, {{.*}}, {{.*}}, {{.*}}, i8* getelementptr inbounds ([106 x i8], [106 x i8]* @{{.*}}, i{{.*}} 0, i{{.*}} 0), i8* {{(nonnull )?}}[[BUFFER]], i32 20) // CHECK-NEXT: tail call void @swift_slowDealloc(i8* {{(nonnull )?}}[[BUFFER]] // CHECK-NEXT: br label %[[NOT_ENABLED]] } // CHECK-LABEL: define hidden swiftcc void @"${{.*}}testNSObjectInterpolation func testNSObjectInterpolation(nsArray: NSArray) { _osLogTestHelper("NSArray: \(nsArray, privacy: .public)") // TODO: check why the ARC optimizer cannot eliminate the many retain/release pairs here. // CHECK: entry: // CHECK-NEXT: bitcast %TSo7NSArrayC* %0 to i8* // CHECK-NEXT: tail call i8* @llvm.objc.retain // CHECK-NEXT: [[NSARRAY_ARG:%.+]] = tail call i8* @llvm.objc.retain // CHECK: tail call swiftcc i1 @"${{.*}}isLoggingEnabled{{.*}}"() // CHECK-NEXT: br i1 {{%.*}}, label %[[ENABLED:[0-9]+]], label %[[NOT_ENABLED:[0-9]+]] // CHECK: [[NOT_ENABLED]]: // CHECK-NEXT: tail call void @swift_release // CHECK-NEXT: tail call void @llvm.objc.release // CHECK-NEXT: br label %[[EXIT:[0-9]+]] // CHECK: [[EXIT]]: // CHECK-NEXT: tail call void @llvm.objc.release(i8* [[NSARRAY_ARG]]) // CHECK-NEXT: ret void // CHECK: [[ENABLED]]: // // Header bytes. // // CHECK-64: [[BUFFER:%.+]] = tail call noalias i8* @swift_slowAlloc(i64 12 // CHECK-32: [[BUFFER:%.+]] = tail call noalias i8* @swift_slowAlloc(i32 8 // CHECK-NEXT: store i8 2, i8* [[BUFFER]], align 1 // CHECK-NEXT: [[OFFSET1:%.+]] = getelementptr inbounds i8, i8* [[BUFFER]], i{{.*}} 1 // CHECK-NEXT: store i8 1, i8* [[OFFSET1]], align 1 // // Argument bytes. // // CHECK-NEXT: [[OFFSET2:%.+]] = getelementptr inbounds i8, i8* [[BUFFER]], i{{.*}} 2 // CHECK-NEXT: store i8 66, i8* [[OFFSET2]], align 1 // CHECK-NEXT: [[OFFSET3:%.+]] = getelementptr inbounds i8, i8* [[BUFFER]], i{{.*}} 3 // CHECK-64-NEXT: store i8 8, i8* [[OFFSET3]], align 1 // CHECK-32-NEXT: store i8 4, i8* [[OFFSET3]], align 1 // CHECK-NEXT: tail call void @llvm.objc.release // CHECK-NEXT: bitcast %swift.refcounted* %{{.*}} to %swift.opaque* // CHECK-NEXT: [[OFFSET4:%.+]] = getelementptr inbounds i8, i8* [[BUFFER]], i{{.*}} 4 // CHECK-NEXT: [[BITCASTED_DEST:%.+]] = bitcast i8* [[OFFSET4]] to %TSo7NSArrayC** // CHECK-NEXT: [[BITCASTED_SRC:%.+]] = bitcast i8* [[NSARRAY_ARG]] to %TSo7NSArrayC* // CHECK-NEXT: store %TSo7NSArrayC* [[BITCASTED_SRC]], %TSo7NSArrayC** [[BITCASTED_DEST]], align 1 // CHECK-64-NEXT: tail call swiftcc void @"${{.*}}_os_log_impl_test{{.*}}"({{.*}}, {{.*}}, {{.*}}, {{.*}}, i8* getelementptr inbounds ([20 x i8], [20 x i8]* @{{.*}}, i64 0, i64 0), i8* {{(nonnull )?}}[[BUFFER]], i32 12) // CHECK-32-NEXT: tail call swiftcc void @"${{.*}}_os_log_impl_test{{.*}}"({{.*}}, {{.*}}, {{.*}}, {{.*}}, i8* getelementptr inbounds ([20 x i8], [20 x i8]* @{{.*}}, i32 0, i32 0), i8* {{(nonnull )?}}[[BUFFER]], i32 8) // CHECK-NEXT: tail call void @swift_slowDealloc(i8* {{(nonnull )?}}[[BUFFER]] // CHECK-NEXT: tail call void @swift_release // CHECK-NEXT: br label %[[EXIT]] } // CHECK-LABEL: define hidden swiftcc void @"${{.*}}testFloatInterpolation func testFloatInterpolation(doubleValue: Double) { _osLogTestHelper("Double value: \(doubleValue)") // CHECK: entry: // CHECK: tail call swiftcc i1 @"${{.*}}isLoggingEnabled{{.*}}"() // CHECK-NEXT: br i1 {{%.*}}, label %[[ENABLED:[0-9]+]], label %[[NOT_ENABLED:[0-9]+]] // CHECK: [[NOT_ENABLED]]: // CHECK-NEXT: tail call void @swift_release // CHECK-NEXT: ret void // CHECK: [[ENABLED]]: // // Header bytes. // // CHECK: [[BUFFER:%.+]] = tail call noalias i8* @swift_slowAlloc(i{{.*}} 12 // CHECK-NEXT: store i8 0, i8* [[BUFFER]], align 1 // CHECK-NEXT: [[OFFSET1:%.+]] = getelementptr inbounds i8, i8* [[BUFFER]], i{{.*}} 1 // CHECK-NEXT: store i8 1, i8* [[OFFSET1]], align 1 // // Argument bytes. // // CHECK-NEXT: [[OFFSET2:%.+]] = getelementptr inbounds i8, i8* [[BUFFER]], i{{.*}} 2 // CHECK-NEXT: store i8 0, i8* [[OFFSET2]], align 1 // CHECK-NEXT: [[OFFSET3:%.+]] = getelementptr inbounds i8, i8* [[BUFFER]], i{{.*}} 3 // CHECK-NEXT: store i8 8, i8* [[OFFSET3]], align 1 // CHECK-NEXT: bitcast %swift.refcounted* %{{.*}} to %swift.opaque* // CHECK-NEXT: [[OFFSET4:%.+]] = getelementptr inbounds i8, i8* [[BUFFER]], i{{.*}} 4 // CHECK-NEXT: [[BITCASTED:%.+]] = bitcast i8* [[OFFSET4]] to double* // CHECK-NEXT: store double %0, double* [[BITCASTED]], align 1 // CHECK-NEXT: tail call swiftcc void @"${{.*}}_os_log_impl_test{{.*}}"({{.*}}, {{.*}}, {{.*}}, {{.*}}, i8* getelementptr inbounds ([17 x i8], [17 x i8]* @{{.*}}, i{{.*}} 0, i{{.*}} 0), i8* {{(nonnull )?}}[[BUFFER]], i32 12) // CHECK-NEXT: tail call void @swift_slowDealloc(i8* {{(nonnull )?}}[[BUFFER]] // CHECK-NEXT: br label %[[NOT_ENABLED]] } // This test checks that the precision and alignment are optimally "stored" into the // byte buffer at the right positions. // CHECK-LABEL: define hidden swiftcc void @"${{.*}}testDynamicPrecisionAndAlignment func testDynamicPrecisionAndAlignment() { _osLogTestHelper( """ Maximum Int64 value: \ \(Int32.max, format: .decimal(minDigits: 10), align: .left(columns: 5)) """) // CHECK: entry: // CHECK: tail call swiftcc i1 @"${{.*}}isLoggingEnabled{{.*}}"() // CHECK-NEXT: br i1 {{%.*}}, label %[[ENABLED:[0-9]+]], label %[[NOT_ENABLED:[0-9]+]] // CHECK: [[NOT_ENABLED]]: // CHECK-NEXT: tail call void @swift_release // CHECK-NEXT: ret void // CHECK: [[ENABLED]]: // // Header bytes. // // CHECK: [[BUFFER:%.+]] = tail call noalias i8* @swift_slowAlloc(i{{.*}} 20 // CHECK-NEXT: store i8 0, i8* [[BUFFER]], align 1 // CHECK-NEXT: [[OFFSET1:%.+]] = getelementptr inbounds i8, i8* [[BUFFER]], i{{.*}} 1 // CHECK-NEXT: store i8 3, i8* [[OFFSET1]], align 1 // // First argument bytes. // // CHECK-NEXT: [[OFFSET2:%.+]] = getelementptr inbounds i8, i8* [[BUFFER]], i{{.*}} 2 // CHECK-NEXT: store i8 16, i8* [[OFFSET2]], align 1 // CHECK-NEXT: [[OFFSET3:%.+]] = getelementptr inbounds i8, i8* [[BUFFER]], i{{.*}} 3 // CHECK-NEXT: store i8 4, i8* [[OFFSET3]], align 1 // CHECK-NEXT: [[OFFSET4:%.+]] = getelementptr inbounds i8, i8* [[BUFFER]], i{{.*}} 4 // CHECK-NEXT: [[BITCASTED:%.+]] = bitcast i8* [[OFFSET4]] to i32* // CHECK-NEXT: store i32 5, i32* [[BITCASTED]], align 1 // // Second argument // // CHECK-NEXT: [[OFFSET12:%.+]] = getelementptr inbounds i8, i8* [[BUFFER]], i{{.*}} 8 // CHECK-NEXT: store i8 16, i8* [[OFFSET12]], align 1 // CHECK-NEXT: [[OFFSET13:%.+]] = getelementptr inbounds i8, i8* [[BUFFER]], i{{.*}} 9 // CHECK-NEXT: store i8 4, i8* [[OFFSET13]], align 1 // CHECK-NEXT: [[OFFSET14:%.+]] = getelementptr inbounds i8, i8* [[BUFFER]], i{{.*}} 10 // CHECK-NEXT: [[BITCASTED2:%.+]] = bitcast i8* [[OFFSET14]] to i32* // CHECK-NEXT: store i32 10, i32* [[BITCASTED2]], align 1 // // Third argument // // CHECK-NEXT: [[OFFSET22:%.+]] = getelementptr inbounds i8, i8* [[BUFFER]], i{{.*}} 14 // CHECK-NEXT: store i8 0, i8* [[OFFSET22]], align 1 // CHECK-NEXT: [[OFFSET23:%.+]] = getelementptr inbounds i8, i8* [[BUFFER]], i{{.*}} 15 // CHECK-NEXT: store i8 4, i8* [[OFFSET23]], align 1 // CHECK-NEXT: bitcast %swift.refcounted* %{{.*}} to %swift.opaque* // CHECK-NEXT: [[OFFSET24:%.+]] = getelementptr inbounds i8, i8* [[BUFFER]], i{{.*}} 16 // CHECK-NEXT: [[BITCASTED3:%.+]] = bitcast i8* [[OFFSET24]] to i32* // CHECK-NEXT: store i32 2147483647, i32* [[BITCASTED3]], align 1 // // os_log_impl call. // CHECK-NEXT: tail call swiftcc void @"${{.*}}_os_log_impl_test{{.*}}"({{.*}}, {{.*}}, {{.*}}, {{.*}}, i8* getelementptr inbounds ([28 x i8], [28 x i8]* @{{.*}}, i{{.*}} 0, i{{.*}} 0), i8* {{(nonnull )?}}[[BUFFER]], i32 20) // CHECK-NEXT: tail call void @swift_slowDealloc(i8* {{(nonnull )?}}[[BUFFER]] // CHECK-NEXT: br label %[[NOT_ENABLED]] } // TODO: add test for String. It is more complicated due to more complex logic // in string serialization.