From 2c1c21be0ce86a8f0df5566e0ea9821590c1cc7d Mon Sep 17 00:00:00 2001 From: mlugg Date: Wed, 11 Sep 2024 09:01:05 +0100 Subject: [PATCH] AstGen: do not allow unlabeled `break` to exit a labeled switch `break`ing from something which isn't a loop should always be opt-in. This was a bug in #21257. --- lib/std/zig/AstGen.zig | 4 ++-- test/behavior/switch.zig | 11 +++++++++++ 2 files changed, 13 insertions(+), 2 deletions(-) diff --git a/lib/std/zig/AstGen.zig b/lib/std/zig/AstGen.zig index 675fe095a26e..f90ccad79a17 100644 --- a/lib/std/zig/AstGen.zig +++ b/lib/std/zig/AstGen.zig @@ -7811,9 +7811,7 @@ fn switchExpr( const switch_block = try parent_gz.makeBlockInst(switch_tag, node); if (switch_full.label_token) |label_token| { - block_scope.break_block = switch_block.toOptional(); block_scope.continue_block = switch_block.toOptional(); - // `break_result_info` already set above block_scope.continue_result_info = .{ .rl = if (any_payload_is_ref) .{ .ref_coerced_ty = raw_operand_ty_ref } @@ -7825,6 +7823,8 @@ fn switchExpr( .token = label_token, .block_inst = switch_block, }; + // `break` can target this via `label.block_inst` + // `break_result_info` already set by `setBreakResultInfo` } // We re-use this same scope for all cases, including the special prong, if any. diff --git a/test/behavior/switch.zig b/test/behavior/switch.zig index f1ded573a029..d7f9af9d1d13 100644 --- a/test/behavior/switch.zig +++ b/test/behavior/switch.zig @@ -985,3 +985,14 @@ test "labeled switch with break" { comptime assert(comptime_val); } + +test "unlabeled break ignores switch" { + const result = while (true) { + _ = s: switch (@as(u32, 1)) { + 1 => continue :s 123, + else => |x| break x, + }; + comptime unreachable; // control flow never breaks from the switch + }; + try expect(result == 123); +}