// RUN: %clang_cc1 -std=c++20 -triple x86_64-unknown-linux-gnu -fcxx-exceptions -fexceptions -fclangir -emit-cir %s -o %t.cir // RUN: FileCheck --input-file=%t.cir %s -check-prefix=CIR // RUN: %clang_cc1 -std=c++20 -triple x86_64-unknown-linux-gnu -fcxx-exceptions -fexceptions -fclangir -emit-llvm %s -o %t-cir.ll // RUN: FileCheck --input-file=%t-cir.ll %s -check-prefix=LLVM // RUN: %clang_cc1 -std=c++20 -triple x86_64-unknown-linux-gnu -fcxx-exceptions -fexceptions -emit-llvm %s -o %t.ll // RUN: FileCheck --input-file=%t.ll %s -check-prefix=OGCG void empty_try_block_with_catch_all() { try {} catch (...) {} } // CIR: cir.func{{.*}} @_Z30empty_try_block_with_catch_allv() // CIR: cir.return // LLVM: define{{.*}} void @_Z30empty_try_block_with_catch_allv() // LLVM: ret void // OGCG: define{{.*}} void @_Z30empty_try_block_with_catch_allv() // OGCG: ret void void empty_try_block_with_catch_with_int_exception() { try {} catch (int e) {} } // CIR: cir.func{{.*}} @_Z45empty_try_block_with_catch_with_int_exceptionv() // CIR: cir.return // LLVM: define{{.*}} void @_Z45empty_try_block_with_catch_with_int_exceptionv() // LLVM: ret void // OGCG: define{{.*}} void @_Z45empty_try_block_with_catch_with_int_exceptionv() // OGCG: ret void void try_catch_with_empty_catch_all() { int a = 1; try { return; ++a; } catch (...) { } } // CIR: %[[A_ADDR:.*]] = cir.alloca !s32i, !cir.ptr, ["a", init] // CIR: %[[CONST_1:.*]] = cir.const #cir.int<1> : !s32i // CIR: cir.store{{.*}} %[[CONST_1]], %[[A_ADDR]] : !s32i, !cir.ptr, !s32i // CIR: %[[RESULT:.*]] = cir.unary(inc, %[[TMP_A]]) nsw : !s32i, !s32i // CIR: cir.store{{.*}} %[[RESULT]], %[[A_ADDR]] : !s32i, !cir.ptr // CIR: cir.yield // CIR: } // CIR: } // LLVM: %[[A_ADDR:.*]] = alloca i32, i64 1, align 4 // LLVM: store i32 1, ptr %[[A_ADDR]], align 4 // LLVM: br label %[[BB_2:.*]] // LLVM: [[BB_2]]: // LLVM: br label %[[BB_3:.*]] // LLVM: [[BB_3]]: // LLVM: ret void // LLVM: [[BB_4:.*]]: // LLVM: %[[TMP_A:.*]] = load i32, ptr %[[A_ADDR]], align 4 // LLVM: %[[RESULT:.*]] = add nsw i32 %[[TMP_A]], 1 // LLVM: store i32 %[[RESULT]], ptr %[[A_ADDR]], align 4 // LLVM: br label %[[BB_7:.*]] // LLVM: [[BB_7]]: // LLVM: br label %[[BB_8:.*]] // LLVM: [[BB_8]]: // LLVM: ret void // OGCG: %[[A_ADDR:.*]] = alloca i32, align 4 // OGCG: store i32 1, ptr %[[A_ADDR]], align 4 // OGCG: ret void void try_catch_with_empty_catch_all_2() { int a = 1; try { ++a; return; } catch (...) { } } // CIR: %[[A_ADDR:.*]] = cir.alloca !s32i, !cir.ptr, ["a", init] // CIR: %[[CONST_1:.*]] = cir.const #cir.int<1> : !s32i // CIR: cir.store{{.*}} %[[CONST_1]], %[[A_ADDR]] : !s32i, !cir.ptr // CIR: cir.scope { // CIR: cir.try { // CIR: %[[TMP_A:.*]] = cir.load{{.*}} %[[A_ADDR]] : !cir.ptr, !s32i // CIR: %[[RESULT:.*]] = cir.unary(inc, %[[TMP_A]]) nsw : !s32i, !s32i // CIR: cir.store{{.*}} %[[RESULT]], %[[A_ADDR]] : !s32i, !cir.ptr // CIR: cir.return // CIR: } // CIR: } // LLVM: %[[A_ADDR]] = alloca i32, i64 1, align 4 // LLVM: store i32 1, ptr %[[A_ADDR]], align 4 // LLVM: br label %[[BB_2:.*]] // LLVM: [[BB_2]]: // LLVM: br label %[[BB_3:.*]] // LLVM: [[BB_3]]: // LLVM: %[[TMP_A:.*]] = load i32, ptr %[[A_ADDR]], align 4 // LLVM: %[[RESULT:.*]] = add nsw i32 %[[TMP_A:.*]], 1 // LLVM: store i32 %[[RESULT]], ptr %[[A_ADDR]], align 4 // LLVM: ret void // LLVM: [[BB_6:.*]]: // LLVM: br label %[[BB_7:.*]] // LLVM: [[BB_7]]: // LLVM: ret void // OGCG: %[[A_ADDR:.*]] = alloca i32, align 4 // OGCG: store i32 1, ptr %[[A_ADDR]], align 4 // OGCG: %[[TMP_A:.*]] = load i32, ptr %[[A_ADDR]], align 4 // OGCG: %[[RESULT:.*]] = add nsw i32 %[[TMP_A]], 1 // OGCG: store i32 %[[RESULT]], ptr %[[A_ADDR]], align 4 // OGCG: ret void void try_catch_with_alloca() { try { int a; int b; int c = a + b; } catch (...) { } } // CIR: cir.scope { // CIR: %[[A_ADDR:.*]] = cir.alloca !s32i, !cir.ptr, ["a"] // CIR: %[[B_ADDR:.*]] = cir.alloca !s32i, !cir.ptr, ["b"] // CIR: %[[C_ADDR:.*]] = cir.alloca !s32i, !cir.ptr, ["c", init] // CIR: cir.try { // CIR: %[[TMP_A:.*]] = cir.load{{.*}} %[[A_ADDR]] : !cir.ptr, !s32i // CIR: %[[TMP_B:.*]] = cir.load{{.*}} %[[B_ADDR]] : !cir.ptr, !s32i // CIR: %[[RESULT:.*]] = cir.binop(add, %[[TMP_A]], %[[TMP_B]]) nsw : !s32i // CIR: cir.store{{.*}} %[[RESULT]], %[[C_ADDR]] : !s32i, !cir.ptr // CIR: cir.yield // CIR: } // CIR: } // LLVM: %[[A_ADDR:.*]] = alloca i32, i64 1, align 4 // LLVM: %[[B_ADDR:.*]] = alloca i32, i64 1, align 4 // LLVM: %[[C_ADDR:.*]] = alloca i32, i64 1, align 4 // LLVM: br label %[[LABEL_1:.*]] // LLVM: [[LABEL_1]]: // LLVM: br label %[[LABEL_2:.*]] // LLVM: [[LABEL_2]]: // LLVM: %[[TMP_A:.*]] = load i32, ptr %[[A_ADDR]], align 4 // LLVM: %[[TMP_B:.*]] = load i32, ptr %[[B_ADDR]], align 4 // LLVM: %[[RESULT:.*]] = add nsw i32 %[[TMP_A]], %[[TMP_B]] // LLVM: store i32 %[[RESULT]], ptr %[[C_ADDR]], align 4 // LLVM: br label %[[LABEL_3:.*]] // LLVM: [[LABEL_3]]: // LLVM: br label %[[LABEL_4:.*]] // LLVM: [[LABEL_4]]: // LLVM: ret void // OGCG: %[[A_ADDR:.*]] = alloca i32, align 4 // OGCG: %[[B_ADDR:.*]] = alloca i32, align 4 // OGCG: %[[C_ADDR:.*]] = alloca i32, align 4 // OGCG: %[[TMP_A:.*]] = load i32, ptr %[[A_ADDR]], align 4 // OGCG: %[[TMP_B:.*]] = load i32, ptr %[[B_ADDR]], align 4 // OGCG: %[[RESULT:.*]] = add nsw i32 %[[TMP_A]], %[[TMP_B]] // OGCG: store i32 %[[RESULT]], ptr %[[C_ADDR]], align 4 void function_with_noexcept() noexcept; void calling_noexcept_function_inside_try_block() { try { function_with_noexcept(); } catch (...) { } } // CIR: cir.scope { // CIR: cir.try { // CIR: cir.call @_Z22function_with_noexceptv() nothrow : () -> () // CIR: cir.yield // CIR: } // CIR: } // LLVM: br label %[[LABEL_1:.*]] // LLVM: [[LABEL_1]]: // LLVM: br label %[[LABEL_2:.*]] // LLVM: [[LABEL_2]]: // LLVM: call void @_Z22function_with_noexceptv() // LLVM: br label %[[LABEL_3:.*]] // LLVM: [[LABEL_3]]: // LLVM: br label %[[LABEL_4:.*]] // LLVM: [[LABEL_4]]: // LLVM: ret void // OGCG: call void @_Z22function_with_noexceptv() // OGCG: ret void