Skip to content

Commit 7f4a60f

Browse files
committed
fix(sema): do not stack an implicit else branch
1 parent 7d72130 commit 7f4a60f

File tree

5 files changed

+67
-30
lines changed

5 files changed

+67
-30
lines changed

regression-tests/pure2-last-use.cpp2

+14
Original file line numberDiff line numberDiff line change
@@ -174,6 +174,20 @@ draw: () = {
174174
_ = (pos).vertex();
175175
}
176176

177+
enum_0: () = {
178+
underlying_type: int;
179+
if true { }
180+
underlying_type = 0;
181+
}
182+
183+
enum_1: () = {
184+
underlying_type: int;
185+
if true {
186+
underlying_type = 0;
187+
}
188+
underlying_type = 0;
189+
}
190+
177191
union: type = {
178192
destroy: (inout this) = { }
179193
operator=: (move this) = {

regression-tests/test-results/pure2-last-use.cpp

+26-6
Original file line numberDiff line numberDiff line change
@@ -24,11 +24,11 @@ class issue_857_3;
2424
class issue_869;
2525

2626

27-
#line 177 "pure2-last-use.cpp2"
27+
#line 191 "pure2-last-use.cpp2"
2828
class cpp2_union;
2929

3030

31-
#line 185 "pure2-last-use.cpp2"
31+
#line 199 "pure2-last-use.cpp2"
3232
class my_string;
3333

3434

@@ -168,6 +168,12 @@ auto issue_888(std::string r, int size) -> void;
168168
auto draw() -> void;
169169

170170
#line 177 "pure2-last-use.cpp2"
171+
auto enum_0() -> void;
172+
173+
#line 183 "pure2-last-use.cpp2"
174+
auto enum_1() -> void;
175+
176+
#line 191 "pure2-last-use.cpp2"
171177
class cpp2_union {
172178
public: auto destroy() & -> void;
173179
public: ~cpp2_union() noexcept;
@@ -176,7 +182,7 @@ class cpp2_union {
176182
public: auto operator=(cpp2_union const&) -> void = delete;
177183

178184

179-
#line 183 "pure2-last-use.cpp2"
185+
#line 197 "pure2-last-use.cpp2"
180186
};
181187

182188
class my_string {
@@ -386,17 +392,31 @@ auto draw() -> void{
386392
static_cast<void>(CPP2_UFCS_MOVE(vertex)((std::move(pos))));
387393
}
388394

389-
#line 178 "pure2-last-use.cpp2"
395+
auto enum_0() -> void{
396+
cpp2::deferred_init<int> underlying_type;
397+
if (true) {}
398+
underlying_type.construct(0);
399+
}
400+
401+
auto enum_1() -> void{
402+
cpp2::deferred_init<int> underlying_type;
403+
if (true) {
404+
underlying_type.construct(0);
405+
}
406+
underlying_type.construct(0);
407+
}
408+
409+
#line 192 "pure2-last-use.cpp2"
390410
auto cpp2_union::destroy() & -> void{}
391411
cpp2_union::~cpp2_union() noexcept{
392412
destroy();
393413
static_cast<void>(std::move((*this)));
394414
}
395415

396-
#line 190 "pure2-last-use.cpp2"
416+
#line 204 "pure2-last-use.cpp2"
397417
auto main(int const argc_, char** argv_) -> int{
398418
auto const args = cpp2::make_args(argc_, argv_);
399-
#line 191 "pure2-last-use.cpp2"
419+
#line 205 "pure2-last-use.cpp2"
400420
issue_683(args);
401421
issue_847_2(std::vector<std::unique_ptr<int>>());
402422
}

source/reflect.h

+14-14
Original file line numberDiff line numberDiff line change
@@ -1360,7 +1360,7 @@ auto basic_enum(
13601360
std::vector<value_member_info> enumerators {};
13611361
cpp2::i64 min_value {};
13621362
cpp2::i64 max_value {};
1363-
std::string underlying_type {};
1363+
cpp2::deferred_init<std::string> underlying_type;
13641364

13651365
CPP2_UFCS(reserve_names)(t, "operator=", "operator<=>");
13661366
if (bitwise) {
@@ -1369,7 +1369,7 @@ auto basic_enum(
13691369

13701370
// 1. Gather: The names of all the user-written members, and find/compute the type
13711371

1372-
underlying_type = CPP2_UFCS(get_argument)(t, 0);// use the first template argument, if there was one
1372+
underlying_type.construct(CPP2_UFCS(get_argument)(t, 0));// use the first template argument, if there was one
13731373

13741374
auto found_non_numeric {false};
13751375
{
@@ -1415,23 +1415,23 @@ std::string value = "-1";
14151415

14161416
// Compute the default underlying type, if it wasn't explicitly specified
14171417
#line 913 "reflect.h2"
1418-
if (underlying_type == "")
1418+
if (underlying_type.value() == "")
14191419
{
14201420
CPP2_UFCS(require)(t, !(std::move(found_non_numeric)),
14211421
"if you write an enumerator with a non-numeric-literal value, you must specify the enumeration's underlying type");
14221422

14231423
if (!(bitwise)) {
14241424
if (cpp2::cmp_greater_eq(min_value,std::numeric_limits<cpp2::i8>::min()) && cpp2::cmp_less_eq(max_value,std::numeric_limits<cpp2::i8>::max())) {
1425-
underlying_type = "i8";
1425+
underlying_type.value() = "i8";
14261426
}
14271427
else {if (cpp2::cmp_greater_eq(min_value,std::numeric_limits<cpp2::i16>::min()) && cpp2::cmp_less_eq(max_value,std::numeric_limits<cpp2::i16>::max())) {
1428-
underlying_type = "i16";
1428+
underlying_type.value() = "i16";
14291429
}
14301430
else {if (cpp2::cmp_greater_eq(min_value,std::numeric_limits<cpp2::i32>::min()) && cpp2::cmp_less_eq(max_value,std::numeric_limits<cpp2::i32>::max())) {
1431-
underlying_type = "i32";
1431+
underlying_type.value() = "i32";
14321432
}
14331433
else {if (cpp2::cmp_greater_eq(std::move(min_value),std::numeric_limits<cpp2::i64>::min()) && cpp2::cmp_less_eq(std::move(max_value),std::numeric_limits<cpp2::i64>::max())) {
1434-
underlying_type = "i64";
1434+
underlying_type.value() = "i64";
14351435
}
14361436
else {
14371437
CPP2_UFCS(error)(t, "values are outside the range representable by the largest supported underlying signed type (i64)");
@@ -1440,16 +1440,16 @@ std::string value = "-1";
14401440
else {
14411441
auto umax {std::move(max_value) * cpp2::as_<cpp2::u64, 2>()};
14421442
if (cpp2::cmp_less_eq(umax,std::numeric_limits<cpp2::u8>::max())) {
1443-
underlying_type = "u8";
1443+
underlying_type.value() = "u8";
14441444
}
14451445
else {if (cpp2::cmp_less_eq(umax,std::numeric_limits<cpp2::u16>::max())) {
1446-
underlying_type = "u16";
1446+
underlying_type.value() = "u16";
14471447
}
14481448
else {if (cpp2::cmp_less_eq(std::move(umax),std::numeric_limits<cpp2::u32>::max())) {
1449-
underlying_type = "u32";
1449+
underlying_type.value() = "u32";
14501450
}
14511451
else {
1452-
underlying_type = "u64";
1452+
underlying_type.value() = "u64";
14531453
}}}
14541454
}
14551455
}
@@ -1462,9 +1462,9 @@ std::string value = "-1";
14621462
CPP2_UFCS(remove_marked_members)(t);
14631463

14641464
// Generate all the common material: value and common functions
1465-
CPP2_UFCS(add_member)(t, " _value : " + cpp2::to_string(underlying_type) + ";");
1466-
CPP2_UFCS(add_member)(t, " private operator= : (implicit out this, _val: i64) == _value = cpp2::unsafe_narrow<" + cpp2::to_string(underlying_type) + ">(_val);");
1467-
CPP2_UFCS(add_member)(t, " get_raw_value : (this) -> " + cpp2::to_string(std::move(underlying_type)) + " == _value;");
1465+
CPP2_UFCS(add_member)(t, " _value : " + cpp2::to_string(underlying_type.value()) + ";");
1466+
CPP2_UFCS(add_member)(t, " private operator= : (implicit out this, _val: i64) == _value = cpp2::unsafe_narrow<" + cpp2::to_string(underlying_type.value()) + ">(_val);");
1467+
CPP2_UFCS(add_member)(t, " get_raw_value : (this) -> " + cpp2::to_string(std::move(underlying_type.value())) + " == _value;");
14681468
CPP2_UFCS(add_member)(t, " operator= : (out this, that) == { }");
14691469
CPP2_UFCS(add_member)(t, " operator<=> : (this, that) -> std::strong_ordering;");
14701470

source/reflect.h2

+1-1
Original file line numberDiff line numberDiff line change
@@ -859,7 +859,7 @@ basic_enum: (
859859
enumerators : std::vector<value_member_info> = ();
860860
min_value : i64 = ();
861861
max_value : i64 = ();
862-
underlying_type : std::string = ();
862+
underlying_type : std::string;
863863

864864
t.reserve_names( "operator=", "operator<=>" );
865865
if bitwise {

source/sema.h

+12-9
Original file line numberDiff line numberDiff line change
@@ -148,7 +148,7 @@ struct selection_sym {
148148
struct compound_sym {
149149
bool start = false;
150150
compound_statement_node const* compound = {};
151-
enum kind { is_scope, is_true, is_false } kind_ = is_scope;
151+
enum kind { is_scope, is_implicit_scope, is_true, is_false } kind_ = is_scope;
152152

153153
compound_sym(
154154
bool s,
@@ -848,13 +848,7 @@ class sema
848848
return true;
849849
}
850850

851-
// If this is a member variable in a constructor, the name doesn't
852-
// appear lexically right in the constructor, so prepending "this."
853-
// to the printed name might make the error more readable to the programmer
854851
auto name = decl->identifier->to_string();
855-
if (decl->declaration->parent_is_type()) {
856-
name += " (aka this." + name + ")";
857-
}
858852

859853
struct stack_entry{
860854
int pos; // start of this selection statement
@@ -1079,7 +1073,11 @@ class sema
10791073
auto const& sym = std::get<symbol::active::compound>(symbols[pos].sym);
10801074

10811075
// If we're in a selection
1082-
if (std::ssize(selection_stack) > 0) {
1076+
if (
1077+
std::ssize(selection_stack) > 0
1078+
&& sym.kind_ != compound_sym::kind::is_implicit_scope
1079+
)
1080+
{
10831081
// If this is a compound start with the current selection's depth
10841082
// plus one, it's the start of one of the branches of that selection
10851083
if (
@@ -1995,7 +1993,12 @@ class sema
19951993
&& active_selections.back()->false_branch.get() == &n
19961994
)
19971995
{
1998-
kind = compound_sym::is_false;
1996+
if (active_selections.back()->has_source_false_branch) {
1997+
kind = compound_sym::is_false;
1998+
}
1999+
else {
2000+
kind = compound_sym::is_implicit_scope;
2001+
}
19992002
}
20002003
}
20012004
return kind;

0 commit comments

Comments
 (0)