1 /**
2  * Does semantic analysis for statements.
3  *
4  * Specification: $(LINK2 https://dlang.org/spec/statement.html, Statements)
5  *
6  * Copyright:   Copyright (C) 1999-2020 by The D Language Foundation, All Rights Reserved
7  * Authors:     $(LINK2 http://www.digitalmars.com, Walter Bright)
8  * License:     $(LINK2 http://www.boost.org/LICENSE_1_0.txt, Boost License 1.0)
9  * Source:      $(LINK2 https://github.com/dlang/dmd/blob/master/src/dmd/statementsem.d, _statementsem.d)
10  * Documentation:  https://dlang.org/phobos/dmd_statementsem.html
11  * Coverage:    https://codecov.io/gh/dlang/dmd/src/master/src/dmd/statementsem.d
12  */
13 
14 module dmd.statementsem;
15 
16 import core.stdc.stdio;
17 
18 import dmd.aggregate;
19 import dmd.aliasthis;
20 import dmd.arrayop;
21 import dmd.arraytypes;
22 import dmd.blockexit;
23 import dmd.clone;
24 import dmd.cond;
25 import dmd.ctorflow;
26 import dmd.dcast;
27 import dmd.dclass;
28 import dmd.declaration;
29 import dmd.denum;
30 import dmd.dimport;
31 import dmd.dinterpret;
32 import dmd.dmodule;
33 import dmd.dscope;
34 import dmd.dsymbol;
35 import dmd.dsymbolsem;
36 import dmd.dtemplate;
37 import dmd.errors;
38 import dmd.escape;
39 import dmd.expression;
40 import dmd.expressionsem;
41 import dmd.func;
42 import dmd.globals;
43 import dmd.gluelayer;
44 import dmd.id;
45 import dmd.identifier;
46 import dmd.init;
47 import dmd.intrange;
48 import dmd.mtype;
49 import dmd.nogc;
50 import dmd.opover;
51 import dmd.root.outbuffer;
52 import dmd.root.string;
53 import dmd.semantic2;
54 import dmd.sideeffect;
55 import dmd.statement;
56 import dmd.target;
57 import dmd.tokens;
58 import dmd.typesem;
59 import dmd.visitor;
60 
61 /*****************************************
62  * CTFE requires FuncDeclaration::labtab for the interpretation.
63  * So fixing the label name inside in/out contracts is necessary
64  * for the uniqueness in labtab.
65  * Params:
66  *      sc = context
67  *      ident = statement label name to be adjusted
68  * Returns:
69  *      adjusted label name
70  */
71 private Identifier fixupLabelName(Scope* sc, Identifier ident)
72 {
73     uint flags = (sc.flags & SCOPE.contract);
74     const id = ident.toString();
75     if (flags && flags != SCOPE.invariant_ &&
76         !(id.length >= 2 && id[0] == '_' && id[1] == '_'))  // does not start with "__"
77     {
78         OutBuffer buf;
79         buf.writestring(flags == SCOPE.require ? "__in_" : "__out_");
80         buf.writestring(ident.toString());
81 
82         ident = Identifier.idPool(buf[]);
83     }
84     return ident;
85 }
86 
87 /*******************************************
88  * Check to see if statement is the innermost labeled statement.
89  * Params:
90  *      sc = context
91  *      statement = Statement to check
92  * Returns:
93  *      if `true`, then the `LabelStatement`, otherwise `null`
94  */
95 private LabelStatement checkLabeledLoop(Scope* sc, Statement statement)
96 {
97     if (sc.slabel && sc.slabel.statement == statement)
98     {
99         return sc.slabel;
100     }
101     return null;
102 }
103 
104 /***********************************************************
105  * Check an assignment is used as a condition.
106  * Intended to be use before the `semantic` call on `e`.
107  * Params:
108  *  e = condition expression which is not yet run semantic analysis.
109  * Returns:
110  *  `e` or ErrorExp.
111  */
112 private Expression checkAssignmentAsCondition(Expression e)
113 {
114     auto ec = lastComma(e);
115     if (ec.op == TOK.assign)
116     {
117         ec.error("assignment cannot be used as a condition, perhaps `==` was meant?");
118         return new ErrorExp();
119     }
120     return e;
121 }
122 
123 // Performs semantic analysis in Statement AST nodes
124 extern(C++) Statement statementSemantic(Statement s, Scope* sc)
125 {
126     scope v = new StatementSemanticVisitor(sc);
127     s.accept(v);
128     return v.result;
129 }
130 
131 private extern (C++) final class StatementSemanticVisitor : Visitor
132 {
133     alias visit = Visitor.visit;
134 
135     Statement result;
136     Scope* sc;
137 
138     this(Scope* sc)
139     {
140         this.sc = sc;
141     }
142 
143     private void setError()
144     {
145         result = new ErrorStatement();
146     }
147 
148     override void visit(Statement s)
149     {
150         result = s;
151     }
152 
153     override void visit(ErrorStatement s)
154     {
155         result = s;
156     }
157 
158     override void visit(PeelStatement s)
159     {
160         /* "peel" off this wrapper, and don't run semantic()
161          * on the result.
162          */
163         result = s.s;
164     }
165 
166     override void visit(ExpStatement s)
167     {
168         /* https://dlang.org/spec/statement.html#expression-statement
169          */
170 
171         if (s.exp)
172         {
173             //printf("ExpStatement::semantic() %s\n", exp.toChars());
174 
175             // Allow CommaExp in ExpStatement because return isn't used
176             CommaExp.allow(s.exp);
177 
178             s.exp = s.exp.expressionSemantic(sc);
179             s.exp = resolveProperties(sc, s.exp);
180             s.exp = s.exp.addDtorHook(sc);
181             if (checkNonAssignmentArrayOp(s.exp))
182                 s.exp = new ErrorExp();
183             if (auto f = isFuncAddress(s.exp))
184             {
185                 if (f.checkForwardRef(s.exp.loc))
186                     s.exp = new ErrorExp();
187             }
188             if (discardValue(s.exp))
189                 s.exp = new ErrorExp();
190 
191             s.exp = s.exp.optimize(WANTvalue);
192             s.exp = checkGC(sc, s.exp);
193             if (s.exp.op == TOK.error)
194                 return setError();
195         }
196         result = s;
197     }
198 
199     override void visit(CompileStatement cs)
200     {
201         /* https://dlang.org/spec/statement.html#mixin-statement
202          */
203 
204         //printf("CompileStatement::semantic() %s\n", exp.toChars());
205         Statements* a = cs.flatten(sc);
206         if (!a)
207             return;
208         Statement s = new CompoundStatement(cs.loc, a);
209         result = s.statementSemantic(sc);
210     }
211 
212     override void visit(CompoundStatement cs)
213     {
214         //printf("CompoundStatement::semantic(this = %p, sc = %p)\n", cs, sc);
215         version (none)
216         {
217             foreach (i, s; cs.statements)
218             {
219                 if (s)
220                     printf("[%d]: %s", i, s.toChars());
221             }
222         }
223 
224         for (size_t i = 0; i < cs.statements.dim;)
225         {
226             Statement s = (*cs.statements)[i];
227             if (s)
228             {
229                 Statements* flt = s.flatten(sc);
230                 if (flt)
231                 {
232                     cs.statements.remove(i);
233                     cs.statements.insert(i, flt);
234                     continue;
235                 }
236                 s = s.statementSemantic(sc);
237                 (*cs.statements)[i] = s;
238                 if (s)
239                 {
240                     Statement sentry;
241                     Statement sexception;
242                     Statement sfinally;
243 
244                     (*cs.statements)[i] = s.scopeCode(sc, &sentry, &sexception, &sfinally);
245                     if (sentry)
246                     {
247                         sentry = sentry.statementSemantic(sc);
248                         cs.statements.insert(i, sentry);
249                         i++;
250                     }
251                     if (sexception)
252                         sexception = sexception.statementSemantic(sc);
253                     if (sexception)
254                     {
255                         /* Returns: true if statements[] are empty statements
256                          */
257                         static bool isEmpty(const Statement[] statements)
258                         {
259                             foreach (s; statements)
260                             {
261                                 if (const cs = s.isCompoundStatement())
262                                 {
263                                     if (!isEmpty((*cs.statements)[]))
264                                         return false;
265                                 }
266                                 else
267                                     return false;
268                             }
269                             return true;
270                         }
271 
272                         if (!sfinally && isEmpty((*cs.statements)[i + 1 .. cs.statements.dim]))
273                         {
274                         }
275                         else
276                         {
277                             /* Rewrite:
278                              *      s; s1; s2;
279                              * As:
280                              *      s;
281                              *      try { s1; s2; }
282                              *      catch (Throwable __o)
283                              *      { sexception; throw __o; }
284                              */
285                             auto a = new Statements();
286                             a.pushSlice((*cs.statements)[i + 1 .. cs.statements.length]);
287                             cs.statements.setDim(i + 1);
288 
289                             Statement _body = new CompoundStatement(Loc.initial, a);
290                             _body = new ScopeStatement(Loc.initial, _body, Loc.initial);
291 
292                             Identifier id = Identifier.generateId("__o");
293 
294                             Statement handler = new PeelStatement(sexception);
295                             if (sexception.blockExit(sc.func, false) & BE.fallthru)
296                             {
297                                 auto ts = new ThrowStatement(Loc.initial, new IdentifierExp(Loc.initial, id));
298                                 ts.internalThrow = true;
299                                 handler = new CompoundStatement(Loc.initial, handler, ts);
300                             }
301 
302                             auto catches = new Catches();
303                             auto ctch = new Catch(Loc.initial, getThrowable(), id, handler);
304                             ctch.internalCatch = true;
305                             catches.push(ctch);
306 
307                             Statement st = new TryCatchStatement(Loc.initial, _body, catches);
308                             if (sfinally)
309                                 st = new TryFinallyStatement(Loc.initial, st, sfinally);
310                             st = st.statementSemantic(sc);
311 
312                             cs.statements.push(st);
313                             break;
314                         }
315                     }
316                     else if (sfinally)
317                     {
318                         if (0 && i + 1 == cs.statements.dim)
319                         {
320                             cs.statements.push(sfinally);
321                         }
322                         else
323                         {
324                             /* Rewrite:
325                              *      s; s1; s2;
326                              * As:
327                              *      s; try { s1; s2; } finally { sfinally; }
328                              */
329                             auto a = new Statements();
330                             a.pushSlice((*cs.statements)[i + 1 .. cs.statements.length]);
331                             cs.statements.setDim(i + 1);
332 
333                             auto _body = new CompoundStatement(Loc.initial, a);
334                             Statement stf = new TryFinallyStatement(Loc.initial, _body, sfinally);
335                             stf = stf.statementSemantic(sc);
336                             cs.statements.push(stf);
337                             break;
338                         }
339                     }
340                 }
341                 else
342                 {
343                     /* Remove NULL statements from the list.
344                      */
345                     cs.statements.remove(i);
346                     continue;
347                 }
348             }
349             i++;
350         }
351 
352         /* Flatten them in place
353          */
354         void flatten(Statements* statements)
355         {
356             for (size_t i = 0; i < statements.length;)
357             {
358                 Statement s = (*statements)[i];
359                 if (s)
360                 {
361                     if (auto flt = s.flatten(sc))
362                     {
363                         statements.remove(i);
364                         statements.insert(i, flt);
365                         continue;
366                     }
367                 }
368                 ++i;
369             }
370         }
371 
372         /* https://issues.dlang.org/show_bug.cgi?id=11653
373          * 'semantic' may return another CompoundStatement
374          * (eg. CaseRangeStatement), so flatten it here.
375          */
376         flatten(cs.statements);
377 
378         foreach (s; *cs.statements)
379         {
380             if (!s)
381                 continue;
382 
383             if (auto se = s.isErrorStatement())
384             {
385                 result = se;
386                 return;
387             }
388         }
389 
390         if (cs.statements.length == 1)
391         {
392             result = (*cs.statements)[0];
393             return;
394         }
395         result = cs;
396     }
397 
398     override void visit(UnrolledLoopStatement uls)
399     {
400         //printf("UnrolledLoopStatement::semantic(this = %p, sc = %p)\n", uls, sc);
401         Scope* scd = sc.push();
402         scd.sbreak = uls;
403         scd.scontinue = uls;
404 
405         Statement serror = null;
406         foreach (i, ref s; *uls.statements)
407         {
408             if (s)
409             {
410                 //printf("[%d]: %s\n", i, s.toChars());
411                 s = s.statementSemantic(scd);
412                 if (s && !serror)
413                     serror = s.isErrorStatement();
414             }
415         }
416 
417         scd.pop();
418         result = serror ? serror : uls;
419     }
420 
421     override void visit(ScopeStatement ss)
422     {
423         //printf("ScopeStatement::semantic(sc = %p)\n", sc);
424         if (ss.statement)
425         {
426             ScopeDsymbol sym = new ScopeDsymbol();
427             sym.parent = sc.scopesym;
428             sym.endlinnum = ss.endloc.linnum;
429             sc = sc.push(sym);
430 
431             Statements* a = ss.statement.flatten(sc);
432             if (a)
433             {
434                 ss.statement = new CompoundStatement(ss.loc, a);
435             }
436 
437             ss.statement = ss.statement.statementSemantic(sc);
438             if (ss.statement)
439             {
440                 if (ss.statement.isErrorStatement())
441                 {
442                     sc.pop();
443                     result = ss.statement;
444                     return;
445                 }
446 
447                 Statement sentry;
448                 Statement sexception;
449                 Statement sfinally;
450                 ss.statement = ss.statement.scopeCode(sc, &sentry, &sexception, &sfinally);
451                 assert(!sentry);
452                 assert(!sexception);
453                 if (sfinally)
454                 {
455                     //printf("adding sfinally\n");
456                     sfinally = sfinally.statementSemantic(sc);
457                     ss.statement = new CompoundStatement(ss.loc, ss.statement, sfinally);
458                 }
459             }
460 
461             sc.pop();
462         }
463         result = ss;
464     }
465 
466     override void visit(ForwardingStatement ss)
467     {
468         assert(ss.sym);
469         for (Scope* csc = sc; !ss.sym.forward; csc = csc.enclosing)
470         {
471             assert(csc);
472             ss.sym.forward = csc.scopesym;
473         }
474         sc = sc.push(ss.sym);
475         sc.sbreak = ss;
476         sc.scontinue = ss;
477         ss.statement = ss.statement.statementSemantic(sc);
478         sc = sc.pop();
479         result = ss.statement;
480     }
481 
482     override void visit(WhileStatement ws)
483     {
484         /* Rewrite as a for(;condition;) loop
485          * https://dlang.org/spec/statement.html#while-statement
486          */
487         Statement s = new ForStatement(ws.loc, null, ws.condition, null, ws._body, ws.endloc);
488         s = s.statementSemantic(sc);
489         result = s;
490     }
491 
492     override void visit(DoStatement ds)
493     {
494         /* https://dlang.org/spec/statement.html#do-statement
495          */
496         const inLoopSave = sc.inLoop;
497         sc.inLoop = true;
498         if (ds._body)
499             ds._body = ds._body.semanticScope(sc, ds, ds);
500         sc.inLoop = inLoopSave;
501 
502         if (ds.condition.op == TOK.dotIdentifier)
503             (cast(DotIdExp)ds.condition).noderef = true;
504 
505         // check in syntax level
506         ds.condition = checkAssignmentAsCondition(ds.condition);
507 
508         ds.condition = ds.condition.expressionSemantic(sc);
509         ds.condition = resolveProperties(sc, ds.condition);
510         if (checkNonAssignmentArrayOp(ds.condition))
511             ds.condition = new ErrorExp();
512         ds.condition = ds.condition.optimize(WANTvalue);
513         ds.condition = checkGC(sc, ds.condition);
514 
515         ds.condition = ds.condition.toBoolean(sc);
516 
517         if (ds.condition.op == TOK.error)
518             return setError();
519         if (ds._body && ds._body.isErrorStatement())
520         {
521             result = ds._body;
522             return;
523         }
524 
525         result = ds;
526     }
527 
528     override void visit(ForStatement fs)
529     {
530         /* https://dlang.org/spec/statement.html#for-statement
531          */
532         //printf("ForStatement::semantic %s\n", fs.toChars());
533 
534         if (fs._init)
535         {
536             /* Rewrite:
537              *  for (auto v1 = i1, v2 = i2; condition; increment) { ... }
538              * to:
539              *  { auto v1 = i1, v2 = i2; for (; condition; increment) { ... } }
540              * then lowered to:
541              *  auto v1 = i1;
542              *  try {
543              *    auto v2 = i2;
544              *    try {
545              *      for (; condition; increment) { ... }
546              *    } finally { v2.~this(); }
547              *  } finally { v1.~this(); }
548              */
549             auto ainit = new Statements();
550             ainit.push(fs._init);
551             fs._init = null;
552             ainit.push(fs);
553             Statement s = new CompoundStatement(fs.loc, ainit);
554             s = new ScopeStatement(fs.loc, s, fs.endloc);
555             s = s.statementSemantic(sc);
556             if (!s.isErrorStatement())
557             {
558                 if (LabelStatement ls = checkLabeledLoop(sc, fs))
559                     ls.gotoTarget = fs;
560                 fs.relatedLabeled = s;
561             }
562             result = s;
563             return;
564         }
565         assert(fs._init is null);
566 
567         auto sym = new ScopeDsymbol();
568         sym.parent = sc.scopesym;
569         sym.endlinnum = fs.endloc.linnum;
570         sc = sc.push(sym);
571         sc.inLoop = true;
572 
573         if (fs.condition)
574         {
575             if (fs.condition.op == TOK.dotIdentifier)
576                 (cast(DotIdExp)fs.condition).noderef = true;
577 
578             // check in syntax level
579             fs.condition = checkAssignmentAsCondition(fs.condition);
580 
581             fs.condition = fs.condition.expressionSemantic(sc);
582             fs.condition = resolveProperties(sc, fs.condition);
583             if (checkNonAssignmentArrayOp(fs.condition))
584                 fs.condition = new ErrorExp();
585             fs.condition = fs.condition.optimize(WANTvalue);
586             fs.condition = checkGC(sc, fs.condition);
587 
588             fs.condition = fs.condition.toBoolean(sc);
589         }
590         if (fs.increment)
591         {
592             CommaExp.allow(fs.increment);
593             fs.increment = fs.increment.expressionSemantic(sc);
594             fs.increment = resolveProperties(sc, fs.increment);
595             if (checkNonAssignmentArrayOp(fs.increment))
596                 fs.increment = new ErrorExp();
597             fs.increment = fs.increment.optimize(WANTvalue);
598             fs.increment = checkGC(sc, fs.increment);
599         }
600 
601         sc.sbreak = fs;
602         sc.scontinue = fs;
603         if (fs._body)
604             fs._body = fs._body.semanticNoScope(sc);
605 
606         sc.pop();
607 
608         if (fs.condition && fs.condition.op == TOK.error ||
609             fs.increment && fs.increment.op == TOK.error ||
610             fs._body && fs._body.isErrorStatement())
611             return setError();
612         result = fs;
613     }
614 
615     /*******************
616      * Determines the return type of makeTupleForeach.
617      */
618     private static template MakeTupleForeachRet(bool isDecl)
619     {
620         static if(isDecl)
621         {
622             alias MakeTupleForeachRet = Dsymbols*;
623         }
624         else
625         {
626             alias MakeTupleForeachRet = void;
627         }
628     }
629 
630     /*******************
631      * Type check and unroll `foreach` over an expression tuple as well
632      * as `static foreach` statements and `static foreach`
633      * declarations. For `static foreach` statements and `static
634      * foreach` declarations, the visitor interface is used (and the
635      * result is written into the `result` field.) For `static
636      * foreach` declarations, the resulting Dsymbols* are returned
637      * directly.
638      *
639      * The unrolled body is wrapped into a
640      *  - UnrolledLoopStatement, for `foreach` over an expression tuple.
641      *  - ForwardingStatement, for `static foreach` statements.
642      *  - ForwardingAttribDeclaration, for `static foreach` declarations.
643      *
644      * `static foreach` variables are declared as `STC.local`, such
645      * that they are inserted into the local symbol tables of the
646      * forwarding constructs instead of forwarded. For `static
647      * foreach` with multiple foreach loop variables whose aggregate
648      * has been lowered into a sequence of tuples, this function
649      * expands the tuples into multiple `STC.local` `static foreach`
650      * variables.
651      */
652     MakeTupleForeachRet!isDecl makeTupleForeach(bool isStatic, bool isDecl)(ForeachStatement fs, TupleForeachArgs!(isStatic, isDecl) args)
653     {
654         auto returnEarly()
655         {
656             static if (isDecl)
657             {
658                 return null;
659             }
660             else
661             {
662                 result = new ErrorStatement();
663                 return;
664             }
665         }
666         static if(isDecl)
667         {
668             static assert(isStatic);
669             auto dbody = args[0];
670         }
671         static if(isStatic)
672         {
673             auto needExpansion = args[$-1];
674             assert(sc);
675         }
676 
677         auto loc = fs.loc;
678         size_t dim = fs.parameters.dim;
679         static if(isStatic) bool skipCheck = needExpansion;
680         else enum skipCheck = false;
681         if (!skipCheck && (dim < 1 || dim > 2))
682         {
683             fs.error("only one (value) or two (key,value) arguments for tuple `foreach`");
684             setError();
685             return returnEarly();
686         }
687 
688         Type paramtype = (*fs.parameters)[dim - 1].type;
689         if (paramtype)
690         {
691             paramtype = paramtype.typeSemantic(loc, sc);
692             if (paramtype.ty == Terror)
693             {
694                 setError();
695                 return returnEarly();
696             }
697         }
698 
699         Type tab = fs.aggr.type.toBasetype();
700         TypeTuple tuple = cast(TypeTuple)tab;
701         static if(!isDecl)
702         {
703             auto statements = new Statements();
704         }
705         else
706         {
707             auto declarations = new Dsymbols();
708         }
709         //printf("aggr: op = %d, %s\n", fs.aggr.op, fs.aggr.toChars());
710         size_t n;
711         TupleExp te = null;
712         if (fs.aggr.op == TOK.tuple) // expression tuple
713         {
714             te = cast(TupleExp)fs.aggr;
715             n = te.exps.dim;
716         }
717         else if (fs.aggr.op == TOK.type) // type tuple
718         {
719             n = Parameter.dim(tuple.arguments);
720         }
721         else
722             assert(0);
723         foreach (j; 0 .. n)
724         {
725             size_t k = (fs.op == TOK.foreach_) ? j : n - 1 - j;
726             Expression e = null;
727             Type t = null;
728             if (te)
729                 e = (*te.exps)[k];
730             else
731                 t = Parameter.getNth(tuple.arguments, k).type;
732             Parameter p = (*fs.parameters)[0];
733             static if(!isDecl)
734             {
735                 auto st = new Statements();
736             }
737             else
738             {
739                 auto st = new Dsymbols();
740             }
741 
742             static if(isStatic) bool skip = needExpansion;
743             else enum skip = false;
744             if (!skip && dim == 2)
745             {
746                 // Declare key
747                 if (p.storageClass & (STC.out_ | STC.ref_ | STC.lazy_))
748                 {
749                     fs.error("no storage class for key `%s`", p.ident.toChars());
750                     setError();
751                     return returnEarly();
752                 }
753                 static if(isStatic)
754                 {
755                     if(!p.type)
756                     {
757                         p.type = Type.tsize_t;
758                     }
759                 }
760                 p.type = p.type.typeSemantic(loc, sc);
761                 TY keyty = p.type.ty;
762                 if (keyty != Tint32 && keyty != Tuns32)
763                 {
764                     if (global.params.isLP64)
765                     {
766                         if (keyty != Tint64 && keyty != Tuns64)
767                         {
768                             fs.error("`foreach`: key type must be `int` or `uint`, `long` or `ulong`, not `%s`", p.type.toChars());
769                             setError();
770                             return returnEarly();
771                         }
772                     }
773                     else
774                     {
775                         fs.error("`foreach`: key type must be `int` or `uint`, not `%s`", p.type.toChars());
776                         setError();
777                         return returnEarly();
778                     }
779                 }
780                 Initializer ie = new ExpInitializer(Loc.initial, new IntegerExp(k));
781                 auto var = new VarDeclaration(loc, p.type, p.ident, ie);
782                 var.storage_class |= STC.manifest;
783                 static if(isStatic) var.storage_class |= STC.local;
784                 static if(!isDecl)
785                 {
786                     st.push(new ExpStatement(loc, var));
787                 }
788                 else
789                 {
790                     st.push(var);
791                 }
792                 p = (*fs.parameters)[1]; // value
793             }
794             /***********************
795              * Declares a unrolled `foreach` loop variable or a `static foreach` variable.
796              *
797              * Params:
798              *     storageClass = The storage class of the variable.
799              *     type = The declared type of the variable.
800              *     ident = The name of the variable.
801              *     e = The initializer of the variable (i.e. the current element of the looped over aggregate).
802              *     t = The type of the initializer.
803              * Returns:
804              *     `true` iff the declaration was successful.
805              */
806             bool declareVariable(StorageClass storageClass, Type type, Identifier ident, Expression e, Type t)
807             {
808                 if (storageClass & (STC.out_ | STC.lazy_) ||
809                     storageClass & STC.ref_ && !te)
810                 {
811                     fs.error("no storage class for value `%s`", ident.toChars());
812                     setError();
813                     return false;
814                 }
815                 Declaration var;
816                 if (e)
817                 {
818                     Type tb = e.type.toBasetype();
819                     Dsymbol ds = null;
820                     if (!(storageClass & STC.manifest))
821                     {
822                         if ((isStatic || tb.ty == Tfunction || tb.ty == Tsarray || storageClass&STC.alias_) && e.op == TOK.variable)
823                             ds = (cast(VarExp)e).var;
824                         else if (e.op == TOK.template_)
825                             ds = (cast(TemplateExp)e).td;
826                         else if (e.op == TOK.scope_)
827                             ds = (cast(ScopeExp)e).sds;
828                         else if (e.op == TOK.function_)
829                         {
830                             auto fe = cast(FuncExp)e;
831                             ds = fe.td ? cast(Dsymbol)fe.td : fe.fd;
832                         }
833                         else if (e.op == TOK.overloadSet)
834                             ds = (cast(OverExp)e).vars;
835                     }
836                     else if (storageClass & STC.alias_)
837                     {
838                         fs.error("`foreach` loop variable cannot be both `enum` and `alias`");
839                         setError();
840                         return false;
841                     }
842 
843                     if (ds)
844                     {
845                         var = new AliasDeclaration(loc, ident, ds);
846                         if (storageClass & STC.ref_)
847                         {
848                             fs.error("symbol `%s` cannot be `ref`", ds.toChars());
849                             setError();
850                             return false;
851                         }
852                         if (paramtype)
853                         {
854                             fs.error("cannot specify element type for symbol `%s`", ds.toChars());
855                             setError();
856                             return false;
857                         }
858                     }
859                     else if (e.op == TOK.type)
860                     {
861                         var = new AliasDeclaration(loc, ident, e.type);
862                         if (paramtype)
863                         {
864                             fs.error("cannot specify element type for type `%s`", e.type.toChars());
865                             setError();
866                             return false;
867                         }
868                     }
869                     else
870                     {
871                         e = resolveProperties(sc, e);
872                         type = e.type;
873                         if (paramtype)
874                             type = paramtype;
875                         Initializer ie = new ExpInitializer(Loc.initial, e);
876                         auto v = new VarDeclaration(loc, type, ident, ie, storageClass);
877                         if (storageClass & STC.ref_)
878                             v.storage_class |= STC.ref_ | STC.foreach_;
879                         if (isStatic || storageClass&STC.manifest || e.isConst() ||
880                             e.op == TOK.string_ ||
881                             e.op == TOK.structLiteral ||
882                             e.op == TOK.arrayLiteral)
883                         {
884                             if (v.storage_class & STC.ref_)
885                             {
886                                 static if (!isStatic)
887                                 {
888                                     fs.error("constant value `%s` cannot be `ref`", ie.toChars());
889                                 }
890                                 else
891                                 {
892                                     if (!needExpansion)
893                                     {
894                                         fs.error("constant value `%s` cannot be `ref`", ie.toChars());
895                                     }
896                                     else
897                                     {
898                                         fs.error("constant value `%s` cannot be `ref`", ident.toChars());
899                                     }
900                                 }
901                                 setError();
902                                 return false;
903                             }
904                             else
905                                 v.storage_class |= STC.manifest;
906                         }
907                         var = v;
908                     }
909                 }
910                 else
911                 {
912                     var = new AliasDeclaration(loc, ident, t);
913                     if (paramtype)
914                     {
915                         fs.error("cannot specify element type for symbol `%s`", fs.toChars());
916                         setError();
917                         return false;
918                     }
919                 }
920                 static if (isStatic)
921                 {
922                     var.storage_class |= STC.local;
923                 }
924                 static if (!isDecl)
925                 {
926                     st.push(new ExpStatement(loc, var));
927                 }
928                 else
929                 {
930                     st.push(var);
931                 }
932                 return true;
933             }
934             static if (!isStatic)
935             {
936                 // Declare value
937                 if (!declareVariable(p.storageClass, p.type, p.ident, e, t))
938                 {
939                     return returnEarly();
940                 }
941             }
942             else
943             {
944                 if (!needExpansion)
945                 {
946                     // Declare value
947                     if (!declareVariable(p.storageClass, p.type, p.ident, e, t))
948                     {
949                         return returnEarly();
950                     }
951                 }
952                 else
953                 { // expand tuples into multiple `static foreach` variables.
954                     assert(e && !t);
955                     auto ident = Identifier.generateId("__value");
956                     declareVariable(0, e.type, ident, e, null);
957                     import dmd.cond: StaticForeach;
958                     auto field = Identifier.idPool(StaticForeach.tupleFieldName.ptr,StaticForeach.tupleFieldName.length);
959                     Expression access = new DotIdExp(loc, e, field);
960                     access = expressionSemantic(access, sc);
961                     if (!tuple) return returnEarly();
962                     //printf("%s\n",tuple.toChars());
963                     foreach (l; 0 .. dim)
964                     {
965                         auto cp = (*fs.parameters)[l];
966                         Expression init_ = new IndexExp(loc, access, new IntegerExp(loc, l, Type.tsize_t));
967                         init_ = init_.expressionSemantic(sc);
968                         assert(init_.type);
969                         declareVariable(p.storageClass, init_.type, cp.ident, init_, null);
970                     }
971                 }
972             }
973 
974             static if (!isDecl)
975             {
976                 if (fs._body) // https://issues.dlang.org/show_bug.cgi?id=17646
977                     st.push(fs._body.syntaxCopy());
978                 Statement res = new CompoundStatement(loc, st);
979             }
980             else
981             {
982                 st.append(Dsymbol.arraySyntaxCopy(dbody));
983             }
984             static if (!isStatic)
985             {
986                 res = new ScopeStatement(loc, res, fs.endloc);
987             }
988             else static if (!isDecl)
989             {
990                 auto fwd = new ForwardingStatement(loc, res);
991                 res = fwd;
992             }
993             else
994             {
995                 import dmd.attrib: ForwardingAttribDeclaration;
996                 auto res = new ForwardingAttribDeclaration(st);
997             }
998             static if (!isDecl)
999             {
1000                 statements.push(res);
1001             }
1002             else
1003             {
1004                 declarations.push(res);
1005             }
1006         }
1007 
1008         static if (!isStatic)
1009         {
1010             Statement res = new UnrolledLoopStatement(loc, statements);
1011             if (LabelStatement ls = checkLabeledLoop(sc, fs))
1012                 ls.gotoTarget = res;
1013             if (te && te.e0)
1014                 res = new CompoundStatement(loc, new ExpStatement(te.e0.loc, te.e0), res);
1015         }
1016         else static if (!isDecl)
1017         {
1018             Statement res = new CompoundStatement(loc, statements);
1019         }
1020         else
1021         {
1022             auto res = declarations;
1023         }
1024         static if (!isDecl)
1025         {
1026             result = res;
1027         }
1028         else
1029         {
1030             return res;
1031         }
1032     }
1033 
1034     override void visit(ForeachStatement fs)
1035     {
1036         /* https://dlang.org/spec/statement.html#foreach-statement
1037          */
1038 
1039         //printf("ForeachStatement::semantic() %p\n", fs);
1040 
1041         /******
1042          * Issue error if any of the ForeachTypes were not supplied and could not be inferred.
1043          * Returns:
1044          *      true if error issued
1045          */
1046         static bool checkForArgTypes(ForeachStatement fs)
1047         {
1048             bool result = false;
1049             foreach (p; *fs.parameters)
1050             {
1051                 if (!p.type)
1052                 {
1053                     fs.error("cannot infer type for `foreach` variable `%s`, perhaps set it explicitly", p.ident.toChars());
1054                     p.type = Type.terror;
1055                     result = true;
1056                 }
1057             }
1058             return result;
1059         }
1060 
1061         const loc = fs.loc;
1062         const dim = fs.parameters.dim;
1063         TypeAArray taa = null;
1064 
1065         Type tn = null;
1066         Type tnv = null;
1067 
1068         fs.func = sc.func;
1069         if (fs.func.fes)
1070             fs.func = fs.func.fes.func;
1071 
1072         VarDeclaration vinit = null;
1073         fs.aggr = fs.aggr.expressionSemantic(sc);
1074         fs.aggr = resolveProperties(sc, fs.aggr);
1075         fs.aggr = fs.aggr.optimize(WANTvalue);
1076         if (fs.aggr.op == TOK.error)
1077             return setError();
1078         Expression oaggr = fs.aggr;
1079         if (fs.aggr.type && fs.aggr.type.toBasetype().ty == Tstruct &&
1080             (cast(TypeStruct)(fs.aggr.type.toBasetype())).sym.dtor &&
1081             fs.aggr.op != TOK.type && !fs.aggr.isLvalue())
1082         {
1083             // https://issues.dlang.org/show_bug.cgi?id=14653
1084             // Extend the life of rvalue aggregate till the end of foreach.
1085             vinit = copyToTemp(STC.rvalue, "__aggr", fs.aggr);
1086             vinit.endlinnum = fs.endloc.linnum;
1087             vinit.dsymbolSemantic(sc);
1088             fs.aggr = new VarExp(fs.aggr.loc, vinit);
1089         }
1090 
1091         Dsymbol sapply = null;                  // the inferred opApply() or front() function
1092         if (!inferForeachAggregate(sc, fs.op == TOK.foreach_, fs.aggr, sapply))
1093         {
1094             const(char)* msg = "";
1095             if (fs.aggr.type && isAggregate(fs.aggr.type))
1096             {
1097                 msg = ", define `opApply()`, range primitives, or use `.tupleof`";
1098             }
1099             fs.error("invalid `foreach` aggregate `%s`%s", oaggr.toChars(), msg);
1100             return setError();
1101         }
1102 
1103         Dsymbol sapplyOld = sapply; // 'sapply' will be NULL if and after 'inferApplyArgTypes' errors
1104 
1105         /* Check for inference errors
1106          */
1107         if (!inferApplyArgTypes(fs, sc, sapply))
1108         {
1109             /**
1110              Try and extract the parameter count of the opApply callback function, e.g.:
1111              int opApply(int delegate(int, float)) => 2 args
1112              */
1113             bool foundMismatch = false;
1114             size_t foreachParamCount = 0;
1115             if (sapplyOld)
1116             {
1117                 if (FuncDeclaration fd = sapplyOld.isFuncDeclaration())
1118                 {
1119                     auto fparameters = fd.getParameterList();
1120 
1121                     if (fparameters.length == 1)
1122                     {
1123                         // first param should be the callback function
1124                         Parameter fparam = fparameters[0];
1125                         if ((fparam.type.ty == Tpointer ||
1126                              fparam.type.ty == Tdelegate) &&
1127                             fparam.type.nextOf().ty == Tfunction)
1128                         {
1129                             TypeFunction tf = cast(TypeFunction)fparam.type.nextOf();
1130                             foreachParamCount = tf.parameterList.length;
1131                             foundMismatch = true;
1132                         }
1133                     }
1134                 }
1135             }
1136 
1137             //printf("dim = %d, parameters.dim = %d\n", dim, parameters.dim);
1138             if (foundMismatch && dim != foreachParamCount)
1139             {
1140                 const(char)* plural = foreachParamCount > 1 ? "s" : "";
1141                 fs.error("cannot infer argument types, expected %d argument%s, not %d",
1142                     foreachParamCount, plural, dim);
1143             }
1144             else
1145                 fs.error("cannot uniquely infer `foreach` argument types");
1146 
1147             return setError();
1148         }
1149 
1150         Type tab = fs.aggr.type.toBasetype();
1151 
1152         if (tab.ty == Ttuple) // don't generate new scope for tuple loops
1153         {
1154             makeTupleForeach!(false,false)(fs);
1155             if (vinit)
1156                 result = new CompoundStatement(loc, new ExpStatement(loc, vinit), result);
1157             result = result.statementSemantic(sc);
1158             return;
1159         }
1160 
1161         auto sym = new ScopeDsymbol();
1162         sym.parent = sc.scopesym;
1163         sym.endlinnum = fs.endloc.linnum;
1164         auto sc2 = sc.push(sym);
1165         sc2.inLoop = true;
1166 
1167         foreach (Parameter p; *fs.parameters)
1168         {
1169             if (p.storageClass & STC.manifest)
1170             {
1171                 fs.error("cannot declare `enum` loop variables for non-unrolled foreach");
1172             }
1173             if (p.storageClass & STC.alias_)
1174             {
1175                 fs.error("cannot declare `alias` loop variables for non-unrolled foreach");
1176             }
1177         }
1178 
1179         Statement s;
1180         switch (tab.ty)
1181         {
1182         case Tarray:
1183         case Tsarray:
1184             {
1185                 if (checkForArgTypes(fs))
1186                     goto case Terror;
1187 
1188                 if (dim < 1 || dim > 2)
1189                 {
1190                     fs.error("only one or two arguments for array `foreach`");
1191                     goto case Terror;
1192                 }
1193 
1194                 // Finish semantic on all foreach parameter types.
1195                 foreach (i; 0 .. dim)
1196                 {
1197                     Parameter p = (*fs.parameters)[i];
1198                     p.type = p.type.typeSemantic(loc, sc2);
1199                     p.type = p.type.addStorageClass(p.storageClass);
1200                 }
1201 
1202                 tn = tab.nextOf().toBasetype();
1203 
1204                 if (dim == 2)
1205                 {
1206                     Type tindex = (*fs.parameters)[0].type;
1207                     if (!tindex.isintegral())
1208                     {
1209                         fs.error("foreach: key cannot be of non-integral type `%s`", tindex.toChars());
1210                         goto case Terror;
1211                     }
1212                     /* What cases to deprecate implicit conversions for:
1213                      *  1. foreach aggregate is a dynamic array
1214                      *  2. foreach body is lowered to _aApply (see special case below).
1215                      */
1216                     Type tv = (*fs.parameters)[1].type.toBasetype();
1217                     if ((tab.ty == Tarray ||
1218                          (tn.ty != tv.ty && tn.ty.isSomeChar && tv.ty.isSomeChar)) &&
1219                         !Type.tsize_t.implicitConvTo(tindex))
1220                     {
1221                         fs.deprecation("foreach: loop index implicitly converted from `size_t` to `%s`",
1222                                        tindex.toChars());
1223                     }
1224                 }
1225 
1226                 /* Look for special case of parsing char types out of char type
1227                  * array.
1228                  */
1229                 if (tn.ty.isSomeChar)
1230                 {
1231                     int i = (dim == 1) ? 0 : 1; // index of value
1232                     Parameter p = (*fs.parameters)[i];
1233                     tnv = p.type.toBasetype();
1234                     if (tnv.ty != tn.ty && tnv.ty.isSomeChar)
1235                     {
1236                         if (p.storageClass & STC.ref_)
1237                         {
1238                             fs.error("`foreach`: value of UTF conversion cannot be `ref`");
1239                             goto case Terror;
1240                         }
1241                         if (dim == 2)
1242                         {
1243                             p = (*fs.parameters)[0];
1244                             if (p.storageClass & STC.ref_)
1245                             {
1246                                 fs.error("`foreach`: key cannot be `ref`");
1247                                 goto case Terror;
1248                             }
1249                         }
1250                         goto Lapply;
1251                     }
1252                 }
1253 
1254                 foreach (i; 0 .. dim)
1255                 {
1256                     // Declare parameters
1257                     Parameter p = (*fs.parameters)[i];
1258                     VarDeclaration var;
1259 
1260                     if (dim == 2 && i == 0)
1261                     {
1262                         var = new VarDeclaration(loc, p.type.mutableOf(), Identifier.generateId("__key"), null);
1263                         var.storage_class |= STC.temp | STC.foreach_;
1264                         if (var.storage_class & (STC.ref_ | STC.out_))
1265                             var.storage_class |= STC.nodtor;
1266 
1267                         fs.key = var;
1268                         if (p.storageClass & STC.ref_)
1269                         {
1270                             if (var.type.constConv(p.type) <= MATCH.nomatch)
1271                             {
1272                                 fs.error("key type mismatch, `%s` to `ref %s`",
1273                                     var.type.toChars(), p.type.toChars());
1274                                 goto case Terror;
1275                             }
1276                         }
1277                         if (tab.ty == Tsarray)
1278                         {
1279                             TypeSArray ta = cast(TypeSArray)tab;
1280                             IntRange dimrange = getIntRange(ta.dim);
1281                             if (!IntRange.fromType(var.type).contains(dimrange))
1282                             {
1283                                 fs.error("index type `%s` cannot cover index range 0..%llu",
1284                                     p.type.toChars(), ta.dim.toInteger());
1285                                 goto case Terror;
1286                             }
1287                             fs.key.range = new IntRange(SignExtendedNumber(0), dimrange.imax);
1288                         }
1289                     }
1290                     else
1291                     {
1292                         var = new VarDeclaration(loc, p.type, p.ident, null);
1293                         var.storage_class |= STC.foreach_;
1294                         var.storage_class |= p.storageClass & (STC.in_ | STC.out_ | STC.ref_ | STC.TYPECTOR);
1295                         if (var.storage_class & (STC.ref_ | STC.out_))
1296                             var.storage_class |= STC.nodtor;
1297 
1298                         fs.value = var;
1299                         if (var.storage_class & STC.ref_)
1300                         {
1301                             if (fs.aggr.checkModifiable(sc2, 1) == Modifiable.initialization)
1302                                 var.storage_class |= STC.ctorinit;
1303 
1304                             Type t = tab.nextOf();
1305                             if (t.constConv(p.type) <= MATCH.nomatch)
1306                             {
1307                                 fs.error("argument type mismatch, `%s` to `ref %s`",
1308                                     t.toChars(), p.type.toChars());
1309                                 goto case Terror;
1310                             }
1311                         }
1312                     }
1313                 }
1314 
1315                 /* Convert to a ForStatement
1316                  *   foreach (key, value; a) body =>
1317                  *   for (T[] tmp = a[], size_t key; key < tmp.length; ++key)
1318                  *   { T value = tmp[k]; body }
1319                  *
1320                  *   foreach_reverse (key, value; a) body =>
1321                  *   for (T[] tmp = a[], size_t key = tmp.length; key--; )
1322                  *   { T value = tmp[k]; body }
1323                  */
1324                 auto id = Identifier.generateId("__r");
1325                 auto ie = new ExpInitializer(loc, new SliceExp(loc, fs.aggr, null, null));
1326                 VarDeclaration tmp;
1327                 if (fs.aggr.op == TOK.arrayLiteral &&
1328                     !((*fs.parameters)[dim - 1].storageClass & STC.ref_))
1329                 {
1330                     auto ale = cast(ArrayLiteralExp)fs.aggr;
1331                     size_t edim = ale.elements ? ale.elements.dim : 0;
1332                     auto telem = (*fs.parameters)[dim - 1].type;
1333 
1334                     // https://issues.dlang.org/show_bug.cgi?id=12936
1335                     // if telem has been specified explicitly,
1336                     // converting array literal elements to telem might make it @nogc.
1337                     fs.aggr = fs.aggr.implicitCastTo(sc, telem.sarrayOf(edim));
1338                     if (fs.aggr.op == TOK.error)
1339                         goto case Terror;
1340 
1341                     // for (T[edim] tmp = a, ...)
1342                     tmp = new VarDeclaration(loc, fs.aggr.type, id, ie);
1343                 }
1344                 else
1345                     tmp = new VarDeclaration(loc, tab.nextOf().arrayOf(), id, ie);
1346                 tmp.storage_class |= STC.temp;
1347 
1348                 Expression tmp_length = new DotIdExp(loc, new VarExp(loc, tmp), Id.length);
1349 
1350                 if (!fs.key)
1351                 {
1352                     Identifier idkey = Identifier.generateId("__key");
1353                     fs.key = new VarDeclaration(loc, Type.tsize_t, idkey, null);
1354                     fs.key.storage_class |= STC.temp;
1355                 }
1356                 else if (fs.key.type.ty != Type.tsize_t.ty)
1357                 {
1358                     tmp_length = new CastExp(loc, tmp_length, fs.key.type);
1359                 }
1360                 if (fs.op == TOK.foreach_reverse_)
1361                     fs.key._init = new ExpInitializer(loc, tmp_length);
1362                 else
1363                     fs.key._init = new ExpInitializer(loc, new IntegerExp(loc, 0, fs.key.type));
1364 
1365                 auto cs = new Statements();
1366                 if (vinit)
1367                     cs.push(new ExpStatement(loc, vinit));
1368                 cs.push(new ExpStatement(loc, tmp));
1369                 cs.push(new ExpStatement(loc, fs.key));
1370                 Statement forinit = new CompoundDeclarationStatement(loc, cs);
1371 
1372                 Expression cond;
1373                 if (fs.op == TOK.foreach_reverse_)
1374                 {
1375                     // key--
1376                     cond = new PostExp(TOK.minusMinus, loc, new VarExp(loc, fs.key));
1377                 }
1378                 else
1379                 {
1380                     // key < tmp.length
1381                     cond = new CmpExp(TOK.lessThan, loc, new VarExp(loc, fs.key), tmp_length);
1382                 }
1383 
1384                 Expression increment = null;
1385                 if (fs.op == TOK.foreach_)
1386                 {
1387                     // key += 1
1388                     increment = new AddAssignExp(loc, new VarExp(loc, fs.key), new IntegerExp(loc, 1, fs.key.type));
1389                 }
1390 
1391                 // T value = tmp[key];
1392                 IndexExp indexExp = new IndexExp(loc, new VarExp(loc, tmp), new VarExp(loc, fs.key));
1393                 indexExp.indexIsInBounds = true; // disabling bounds checking in foreach statements.
1394                 fs.value._init = new ExpInitializer(loc, indexExp);
1395                 Statement ds = new ExpStatement(loc, fs.value);
1396 
1397                 if (dim == 2)
1398                 {
1399                     Parameter p = (*fs.parameters)[0];
1400                     if ((p.storageClass & STC.ref_) && p.type.equals(fs.key.type))
1401                     {
1402                         fs.key.range = null;
1403                         auto v = new AliasDeclaration(loc, p.ident, fs.key);
1404                         fs._body = new CompoundStatement(loc, new ExpStatement(loc, v), fs._body);
1405                     }
1406                     else
1407                     {
1408                         auto ei = new ExpInitializer(loc, new IdentifierExp(loc, fs.key.ident));
1409                         auto v = new VarDeclaration(loc, p.type, p.ident, ei);
1410                         v.storage_class |= STC.foreach_ | (p.storageClass & STC.ref_);
1411                         fs._body = new CompoundStatement(loc, new ExpStatement(loc, v), fs._body);
1412                         if (fs.key.range && !p.type.isMutable())
1413                         {
1414                             /* Limit the range of the key to the specified range
1415                              */
1416                             v.range = new IntRange(fs.key.range.imin, fs.key.range.imax - SignExtendedNumber(1));
1417                         }
1418                     }
1419                 }
1420                 fs._body = new CompoundStatement(loc, ds, fs._body);
1421 
1422                 s = new ForStatement(loc, forinit, cond, increment, fs._body, fs.endloc);
1423                 if (auto ls = checkLabeledLoop(sc, fs))   // https://issues.dlang.org/show_bug.cgi?id=15450
1424                                                           // don't use sc2
1425                     ls.gotoTarget = s;
1426                 s = s.statementSemantic(sc2);
1427                 break;
1428             }
1429         case Taarray:
1430             if (fs.op == TOK.foreach_reverse_)
1431                 fs.warning("cannot use `foreach_reverse` with an associative array");
1432             if (checkForArgTypes(fs))
1433                 goto case Terror;
1434 
1435             taa = cast(TypeAArray)tab;
1436             if (dim < 1 || dim > 2)
1437             {
1438                 fs.error("only one or two arguments for associative array `foreach`");
1439                 goto case Terror;
1440             }
1441             goto Lapply;
1442 
1443         case Tclass:
1444         case Tstruct:
1445             /* Prefer using opApply, if it exists
1446              */
1447             if (sapply)
1448                 goto Lapply;
1449             {
1450                 /* Look for range iteration, i.e. the properties
1451                  * .empty, .popFront, .popBack, .front and .back
1452                  *    foreach (e; aggr) { ... }
1453                  * translates to:
1454                  *    for (auto __r = aggr[]; !__r.empty; __r.popFront()) {
1455                  *        auto e = __r.front;
1456                  *        ...
1457                  *    }
1458                  */
1459                 auto ad = (tab.ty == Tclass) ?
1460                     cast(AggregateDeclaration)(cast(TypeClass)tab).sym :
1461                     cast(AggregateDeclaration)(cast(TypeStruct)tab).sym;
1462                 Identifier idfront;
1463                 Identifier idpopFront;
1464                 if (fs.op == TOK.foreach_)
1465                 {
1466                     idfront = Id.Ffront;
1467                     idpopFront = Id.FpopFront;
1468                 }
1469                 else
1470                 {
1471                     idfront = Id.Fback;
1472                     idpopFront = Id.FpopBack;
1473                 }
1474                 auto sfront = ad.search(Loc.initial, idfront);
1475                 if (!sfront)
1476                     goto Lapply;
1477 
1478                 /* Generate a temporary __r and initialize it with the aggregate.
1479                  */
1480                 VarDeclaration r;
1481                 Statement _init;
1482                 if (vinit && fs.aggr.op == TOK.variable && (cast(VarExp)fs.aggr).var == vinit)
1483                 {
1484                     r = vinit;
1485                     _init = new ExpStatement(loc, vinit);
1486                 }
1487                 else
1488                 {
1489                     r = copyToTemp(0, "__r", fs.aggr);
1490                     r.dsymbolSemantic(sc);
1491                     _init = new ExpStatement(loc, r);
1492                     if (vinit)
1493                         _init = new CompoundStatement(loc, new ExpStatement(loc, vinit), _init);
1494                 }
1495 
1496                 // !__r.empty
1497                 Expression e = new VarExp(loc, r);
1498                 e = new DotIdExp(loc, e, Id.Fempty);
1499                 Expression condition = new NotExp(loc, e);
1500 
1501                 // __r.idpopFront()
1502                 e = new VarExp(loc, r);
1503                 Expression increment = new CallExp(loc, new DotIdExp(loc, e, idpopFront));
1504 
1505                 /* Declaration statement for e:
1506                  *    auto e = __r.idfront;
1507                  */
1508                 e = new VarExp(loc, r);
1509                 Expression einit = new DotIdExp(loc, e, idfront);
1510                 Statement makeargs, forbody;
1511                 bool ignoreRef = false; // If a range returns a non-ref front we ignore ref on foreach
1512 
1513                 Type tfront;
1514                 if (auto fd = sfront.isFuncDeclaration())
1515                 {
1516                     if (!fd.functionSemantic())
1517                         goto Lrangeerr;
1518                     tfront = fd.type;
1519                 }
1520                 else if (auto td = sfront.isTemplateDeclaration())
1521                 {
1522                     Expressions a;
1523                     if (auto f = resolveFuncCall(loc, sc, td, null, tab, &a, FuncResolveFlag.quiet))
1524                         tfront = f.type;
1525                 }
1526                 else if (auto d = sfront.toAlias().isDeclaration())
1527                 {
1528                     tfront = d.type;
1529                 }
1530                 if (!tfront || tfront.ty == Terror)
1531                     goto Lrangeerr;
1532                 if (tfront.toBasetype().ty == Tfunction)
1533                 {
1534                     auto ftt = cast(TypeFunction)tfront.toBasetype();
1535                     tfront = tfront.toBasetype().nextOf();
1536                     if (!ftt.isref)
1537                     {
1538                         // .front() does not return a ref. We ignore ref on foreach arg.
1539                         // see https://issues.dlang.org/show_bug.cgi?id=11934
1540                         if (tfront.needsDestruction()) ignoreRef = true;
1541                     }
1542                 }
1543                 if (tfront.ty == Tvoid)
1544                 {
1545                     fs.error("`%s.front` is `void` and has no value", oaggr.toChars());
1546                     goto case Terror;
1547                 }
1548 
1549                 if (dim == 1)
1550                 {
1551                     auto p = (*fs.parameters)[0];
1552                     auto ve = new VarDeclaration(loc, p.type, p.ident, new ExpInitializer(loc, einit));
1553                     ve.storage_class |= STC.foreach_;
1554                     ve.storage_class |= p.storageClass & (STC.in_ | STC.out_ | STC.ref_ | STC.TYPECTOR);
1555 
1556                     if (ignoreRef)
1557                         ve.storage_class &= ~STC.ref_;
1558 
1559                     makeargs = new ExpStatement(loc, ve);
1560                 }
1561                 else
1562                 {
1563                     auto vd = copyToTemp(STC.ref_, "__front", einit);
1564                     vd.dsymbolSemantic(sc);
1565                     makeargs = new ExpStatement(loc, vd);
1566 
1567                     // Resolve inout qualifier of front type
1568                     tfront = tfront.substWildTo(tab.mod);
1569 
1570                     Expression ve = new VarExp(loc, vd);
1571                     ve.type = tfront;
1572 
1573                     auto exps = new Expressions();
1574                     exps.push(ve);
1575                     int pos = 0;
1576                     while (exps.dim < dim)
1577                     {
1578                         pos = expandAliasThisTuples(exps, pos);
1579                         if (pos == -1)
1580                             break;
1581                     }
1582                     if (exps.dim != dim)
1583                     {
1584                         const(char)* plural = exps.dim > 1 ? "s" : "";
1585                         fs.error("cannot infer argument types, expected %d argument%s, not %d",
1586                             exps.dim, plural, dim);
1587                         goto case Terror;
1588                     }
1589 
1590                     foreach (i; 0 .. dim)
1591                     {
1592                         auto p = (*fs.parameters)[i];
1593                         auto exp = (*exps)[i];
1594                         version (none)
1595                         {
1596                             printf("[%d] p = %s %s, exp = %s %s\n", i,
1597                                 p.type ? p.type.toChars() : "?", p.ident.toChars(),
1598                                 exp.type.toChars(), exp.toChars());
1599                         }
1600                         if (!p.type)
1601                             p.type = exp.type;
1602 
1603                         auto sc = p.storageClass;
1604                         if (ignoreRef) sc &= ~STC.ref_;
1605                         p.type = p.type.addStorageClass(sc).typeSemantic(loc, sc2);
1606                         if (!exp.implicitConvTo(p.type))
1607                             goto Lrangeerr;
1608 
1609                         auto var = new VarDeclaration(loc, p.type, p.ident, new ExpInitializer(loc, exp));
1610                         var.storage_class |= STC.ctfe | STC.ref_ | STC.foreach_;
1611                         makeargs = new CompoundStatement(loc, makeargs, new ExpStatement(loc, var));
1612                     }
1613                 }
1614 
1615                 forbody = new CompoundStatement(loc, makeargs, fs._body);
1616 
1617                 s = new ForStatement(loc, _init, condition, increment, forbody, fs.endloc);
1618                 if (auto ls = checkLabeledLoop(sc, fs))
1619                     ls.gotoTarget = s;
1620 
1621                 version (none)
1622                 {
1623                     printf("init: %s\n", _init.toChars());
1624                     printf("condition: %s\n", condition.toChars());
1625                     printf("increment: %s\n", increment.toChars());
1626                     printf("body: %s\n", forbody.toChars());
1627                 }
1628                 s = s.statementSemantic(sc2);
1629                 break;
1630 
1631             Lrangeerr:
1632                 fs.error("cannot infer argument types");
1633                 goto case Terror;
1634             }
1635         case Tdelegate:
1636             if (fs.op == TOK.foreach_reverse_)
1637                 fs.deprecation("cannot use `foreach_reverse` with a delegate");
1638         Lapply:
1639             {
1640                 if (checkForArgTypes(fs))
1641                     goto case Terror;
1642 
1643                 TypeFunction tfld = null;
1644                 if (sapply)
1645                 {
1646                     FuncDeclaration fdapply = sapply.isFuncDeclaration();
1647                     if (fdapply)
1648                     {
1649                         assert(fdapply.type && fdapply.type.ty == Tfunction);
1650                         tfld = cast(TypeFunction)fdapply.type.typeSemantic(loc, sc2);
1651                         goto Lget;
1652                     }
1653                     else if (tab.ty == Tdelegate)
1654                     {
1655                         tfld = cast(TypeFunction)tab.nextOf();
1656                     Lget:
1657                         //printf("tfld = %s\n", tfld.toChars());
1658                         if (tfld.parameterList.parameters.dim == 1)
1659                         {
1660                             Parameter p = tfld.parameterList[0];
1661                             if (p.type && p.type.ty == Tdelegate)
1662                             {
1663                                 auto t = p.type.typeSemantic(loc, sc2);
1664                                 assert(t.ty == Tdelegate);
1665                                 tfld = cast(TypeFunction)t.nextOf();
1666                             }
1667                             //printf("tfld = %s\n", tfld.toChars());
1668                         }
1669                     }
1670                 }
1671 
1672                 FuncExp flde = foreachBodyToFunction(sc2, fs, tfld);
1673                 if (!flde)
1674                     goto case Terror;
1675 
1676                 // Resolve any forward referenced goto's
1677                 foreach (ScopeStatement ss; *fs.gotos)
1678                 {
1679                     GotoStatement gs = ss.statement.isGotoStatement();
1680                     if (!gs.label.statement)
1681                     {
1682                         // 'Promote' it to this scope, and replace with a return
1683                         fs.cases.push(gs);
1684                         ss.statement = new ReturnStatement(Loc.initial, new IntegerExp(fs.cases.dim + 1));
1685                     }
1686                 }
1687 
1688                 Expression e = null;
1689                 Expression ec;
1690                 if (vinit)
1691                 {
1692                     e = new DeclarationExp(loc, vinit);
1693                     e = e.expressionSemantic(sc2);
1694                     if (e.op == TOK.error)
1695                         goto case Terror;
1696                 }
1697 
1698                 if (taa)
1699                 {
1700                     // Check types
1701                     Parameter p = (*fs.parameters)[0];
1702                     bool isRef = (p.storageClass & STC.ref_) != 0;
1703                     Type ta = p.type;
1704                     if (dim == 2)
1705                     {
1706                         Type ti = (isRef ? taa.index.addMod(MODFlags.const_) : taa.index);
1707                         if (isRef ? !ti.constConv(ta) : !ti.implicitConvTo(ta))
1708                         {
1709                             fs.error("`foreach`: index must be type `%s`, not `%s`",
1710                                 ti.toChars(), ta.toChars());
1711                             goto case Terror;
1712                         }
1713                         p = (*fs.parameters)[1];
1714                         isRef = (p.storageClass & STC.ref_) != 0;
1715                         ta = p.type;
1716                     }
1717                     Type taav = taa.nextOf();
1718                     if (isRef ? !taav.constConv(ta) : !taav.implicitConvTo(ta))
1719                     {
1720                         fs.error("`foreach`: value must be type `%s`, not `%s`",
1721                             taav.toChars(), ta.toChars());
1722                         goto case Terror;
1723                     }
1724 
1725                     /* Call:
1726                      *  extern(C) int _aaApply(void*, in size_t, int delegate(void*))
1727                      *      _aaApply(aggr, keysize, flde)
1728                      *
1729                      *  extern(C) int _aaApply2(void*, in size_t, int delegate(void*, void*))
1730                      *      _aaApply2(aggr, keysize, flde)
1731                      */
1732                     __gshared FuncDeclaration* fdapply = [null, null];
1733                     __gshared TypeDelegate* fldeTy = [null, null];
1734 
1735                     ubyte i = (dim == 2 ? 1 : 0);
1736                     if (!fdapply[i])
1737                     {
1738                         auto params = new Parameters();
1739                         params.push(new Parameter(0, Type.tvoid.pointerTo(), null, null, null));
1740                         params.push(new Parameter(STC.in_, Type.tsize_t, null, null, null));
1741                         auto dgparams = new Parameters();
1742                         dgparams.push(new Parameter(0, Type.tvoidptr, null, null, null));
1743                         if (dim == 2)
1744                             dgparams.push(new Parameter(0, Type.tvoidptr, null, null, null));
1745                         fldeTy[i] = new TypeDelegate(new TypeFunction(ParameterList(dgparams), Type.tint32, LINK.d));
1746                         params.push(new Parameter(0, fldeTy[i], null, null, null));
1747                         fdapply[i] = FuncDeclaration.genCfunc(params, Type.tint32, i ? Id._aaApply2 : Id._aaApply);
1748                     }
1749 
1750                     auto exps = new Expressions();
1751                     exps.push(fs.aggr);
1752                     auto keysize = taa.index.size();
1753                     if (keysize == SIZE_INVALID)
1754                         goto case Terror;
1755                     assert(keysize < keysize.max - target.ptrsize);
1756                     keysize = (keysize + (target.ptrsize - 1)) & ~(target.ptrsize - 1);
1757                     // paint delegate argument to the type runtime expects
1758                     Expression fexp = flde;
1759                     if (!fldeTy[i].equals(flde.type))
1760                     {
1761                         fexp = new CastExp(loc, flde, flde.type);
1762                         fexp.type = fldeTy[i];
1763                     }
1764                     exps.push(new IntegerExp(Loc.initial, keysize, Type.tsize_t));
1765                     exps.push(fexp);
1766                     ec = new VarExp(Loc.initial, fdapply[i], false);
1767                     ec = new CallExp(loc, ec, exps);
1768                     ec.type = Type.tint32; // don't run semantic() on ec
1769                 }
1770                 else if (tab.ty == Tarray || tab.ty == Tsarray)
1771                 {
1772                     /* Call:
1773                      *      _aApply(aggr, flde)
1774                      */
1775                     __gshared const(char)** fntab =
1776                     [
1777                         "cc", "cw", "cd",
1778                         "wc", "cc", "wd",
1779                         "dc", "dw", "dd"
1780                     ];
1781 
1782                     const(size_t) BUFFER_LEN = 7 + 1 + 2 + dim.sizeof * 3 + 1;
1783                     char[BUFFER_LEN] fdname;
1784                     int flag;
1785 
1786                     switch (tn.ty)
1787                     {
1788                     case Tchar:     flag = 0;   break;
1789                     case Twchar:    flag = 3;   break;
1790                     case Tdchar:    flag = 6;   break;
1791                     default:
1792                         assert(0);
1793                     }
1794                     switch (tnv.ty)
1795                     {
1796                     case Tchar:     flag += 0;  break;
1797                     case Twchar:    flag += 1;  break;
1798                     case Tdchar:    flag += 2;  break;
1799                     default:
1800                         assert(0);
1801                     }
1802                     const(char)* r = (fs.op == TOK.foreach_reverse_) ? "R" : "";
1803                     int j = sprintf(fdname.ptr, "_aApply%s%.*s%llu", r, 2, fntab[flag], cast(ulong)dim);
1804                     assert(j < BUFFER_LEN);
1805 
1806                     FuncDeclaration fdapply;
1807                     TypeDelegate dgty;
1808                     auto params = new Parameters();
1809                     params.push(new Parameter(STC.in_, tn.arrayOf(), null, null, null));
1810                     auto dgparams = new Parameters();
1811                     dgparams.push(new Parameter(0, Type.tvoidptr, null, null, null));
1812                     if (dim == 2)
1813                         dgparams.push(new Parameter(0, Type.tvoidptr, null, null, null));
1814                     dgty = new TypeDelegate(new TypeFunction(ParameterList(dgparams), Type.tint32, LINK.d));
1815                     params.push(new Parameter(0, dgty, null, null, null));
1816                     fdapply = FuncDeclaration.genCfunc(params, Type.tint32, fdname.ptr);
1817 
1818                     if (tab.ty == Tsarray)
1819                         fs.aggr = fs.aggr.castTo(sc2, tn.arrayOf());
1820                     // paint delegate argument to the type runtime expects
1821                     Expression fexp = flde;
1822                     if (!dgty.equals(flde.type))
1823                     {
1824                         fexp = new CastExp(loc, flde, flde.type);
1825                         fexp.type = dgty;
1826                     }
1827                     ec = new VarExp(Loc.initial, fdapply, false);
1828                     ec = new CallExp(loc, ec, fs.aggr, fexp);
1829                     ec.type = Type.tint32; // don't run semantic() on ec
1830                 }
1831                 else if (tab.ty == Tdelegate)
1832                 {
1833                     /* Call:
1834                      *      aggr(flde)
1835                      */
1836                     if (fs.aggr.op == TOK.delegate_ && (cast(DelegateExp)fs.aggr).func.isNested() && !(cast(DelegateExp)fs.aggr).func.needThis())
1837                     {
1838                         // https://issues.dlang.org/show_bug.cgi?id=3560
1839                         fs.aggr = (cast(DelegateExp)fs.aggr).e1;
1840                     }
1841                     ec = new CallExp(loc, fs.aggr, flde);
1842                     ec = ec.expressionSemantic(sc2);
1843                     if (ec.op == TOK.error)
1844                         goto case Terror;
1845                     if (ec.type != Type.tint32)
1846                     {
1847                         fs.error("`opApply()` function for `%s` must return an `int`", tab.toChars());
1848                         goto case Terror;
1849                     }
1850                 }
1851                 else
1852                 {
1853 version (none)
1854 {
1855                     if (global.params.vsafe)
1856                     {
1857                         message(loc, "To enforce `@safe`, the compiler allocates a closure unless `opApply()` uses `scope`");
1858                     }
1859                     flde.fd.tookAddressOf = 1;
1860 }
1861 else
1862 {
1863                     if (global.params.vsafe)
1864                         ++flde.fd.tookAddressOf;  // allocate a closure unless the opApply() uses 'scope'
1865 }
1866                     assert(tab.ty == Tstruct || tab.ty == Tclass);
1867                     assert(sapply);
1868                     /* Call:
1869                      *  aggr.apply(flde)
1870                      */
1871                     ec = new DotIdExp(loc, fs.aggr, sapply.ident);
1872                     ec = new CallExp(loc, ec, flde);
1873                     ec = ec.expressionSemantic(sc2);
1874                     if (ec.op == TOK.error)
1875                         goto case Terror;
1876                     if (ec.type != Type.tint32)
1877                     {
1878                         fs.error("`opApply()` function for `%s` must return an `int`", tab.toChars());
1879                         goto case Terror;
1880                     }
1881                 }
1882                 e = Expression.combine(e, ec);
1883 
1884                 if (!fs.cases.dim)
1885                 {
1886                     // Easy case, a clean exit from the loop
1887                     e = new CastExp(loc, e, Type.tvoid); // https://issues.dlang.org/show_bug.cgi?id=13899
1888                     s = new ExpStatement(loc, e);
1889                 }
1890                 else
1891                 {
1892                     // Construct a switch statement around the return value
1893                     // of the apply function.
1894                     auto a = new Statements();
1895 
1896                     // default: break; takes care of cases 0 and 1
1897                     s = new BreakStatement(Loc.initial, null);
1898                     s = new DefaultStatement(Loc.initial, s);
1899                     a.push(s);
1900 
1901                     // cases 2...
1902                     foreach (i, c; *fs.cases)
1903                     {
1904                         s = new CaseStatement(Loc.initial, new IntegerExp(i + 2), c);
1905                         a.push(s);
1906                     }
1907 
1908                     s = new CompoundStatement(loc, a);
1909                     s = new SwitchStatement(loc, e, s, false);
1910                 }
1911                 s = s.statementSemantic(sc2);
1912                 break;
1913             }
1914             assert(0);
1915 
1916         case Terror:
1917             s = new ErrorStatement();
1918             break;
1919 
1920         default:
1921             fs.error("`foreach`: `%s` is not an aggregate type", fs.aggr.type.toChars());
1922             goto case Terror;
1923         }
1924         sc2.pop();
1925         result = s;
1926     }
1927 
1928     /*************************************
1929      * Turn foreach body into the function literal:
1930      *  int delegate(ref T param) { body }
1931      * Params:
1932      *  sc = context
1933      *  fs = ForeachStatement
1934      *  tfld = type of function literal to be created, can be null
1935      * Returns:
1936      *  Function literal created, as an expression
1937      *  null if error.
1938      */
1939     static FuncExp foreachBodyToFunction(Scope* sc, ForeachStatement fs, TypeFunction tfld)
1940     {
1941         auto params = new Parameters();
1942         foreach (i; 0 .. fs.parameters.dim)
1943         {
1944             Parameter p = (*fs.parameters)[i];
1945             StorageClass stc = STC.ref_;
1946             Identifier id;
1947 
1948             p.type = p.type.typeSemantic(fs.loc, sc);
1949             p.type = p.type.addStorageClass(p.storageClass);
1950             if (tfld)
1951             {
1952                 Parameter prm = tfld.parameterList[i];
1953                 //printf("\tprm = %s%s\n", (prm.storageClass&STC.ref_?"ref ":"").ptr, prm.ident.toChars());
1954                 stc = prm.storageClass & STC.ref_;
1955                 id = p.ident; // argument copy is not need.
1956                 if ((p.storageClass & STC.ref_) != stc)
1957                 {
1958                     if (!stc)
1959                     {
1960                         fs.error("`foreach`: cannot make `%s` `ref`", p.ident.toChars());
1961                         return null;
1962                     }
1963                     goto LcopyArg;
1964                 }
1965             }
1966             else if (p.storageClass & STC.ref_)
1967             {
1968                 // default delegate parameters are marked as ref, then
1969                 // argument copy is not need.
1970                 id = p.ident;
1971             }
1972             else
1973             {
1974                 // Make a copy of the ref argument so it isn't
1975                 // a reference.
1976             LcopyArg:
1977                 id = Identifier.generateId("__applyArg", cast(int)i);
1978 
1979                 Initializer ie = new ExpInitializer(fs.loc, new IdentifierExp(fs.loc, id));
1980                 auto v = new VarDeclaration(fs.loc, p.type, p.ident, ie);
1981                 v.storage_class |= STC.temp;
1982                 Statement s = new ExpStatement(fs.loc, v);
1983                 fs._body = new CompoundStatement(fs.loc, s, fs._body);
1984             }
1985             params.push(new Parameter(stc, p.type, id, null, null));
1986         }
1987         // https://issues.dlang.org/show_bug.cgi?id=13840
1988         // Throwable nested function inside nothrow function is acceptable.
1989         StorageClass stc = mergeFuncAttrs(STC.safe | STC.pure_ | STC.nogc, fs.func);
1990         auto tf = new TypeFunction(ParameterList(params), Type.tint32, LINK.d, stc);
1991         fs.cases = new Statements();
1992         fs.gotos = new ScopeStatements();
1993         auto fld = new FuncLiteralDeclaration(fs.loc, fs.endloc, tf, TOK.delegate_, fs);
1994         fld.fbody = fs._body;
1995         Expression flde = new FuncExp(fs.loc, fld);
1996         flde = flde.expressionSemantic(sc);
1997         fld.tookAddressOf = 0;
1998         if (flde.op == TOK.error)
1999             return null;
2000         return cast(FuncExp)flde;
2001     }
2002 
2003     override void visit(ForeachRangeStatement fs)
2004     {
2005         /* https://dlang.org/spec/statement.html#foreach-range-statement
2006          */
2007 
2008         //printf("ForeachRangeStatement::semantic() %p\n", fs);
2009         auto loc = fs.loc;
2010         fs.lwr = fs.lwr.expressionSemantic(sc);
2011         fs.lwr = resolveProperties(sc, fs.lwr);
2012         fs.lwr = fs.lwr.optimize(WANTvalue);
2013         if (!fs.lwr.type)
2014         {
2015             fs.error("invalid range lower bound `%s`", fs.lwr.toChars());
2016             return setError();
2017         }
2018 
2019         fs.upr = fs.upr.expressionSemantic(sc);
2020         fs.upr = resolveProperties(sc, fs.upr);
2021         fs.upr = fs.upr.optimize(WANTvalue);
2022         if (!fs.upr.type)
2023         {
2024             fs.error("invalid range upper bound `%s`", fs.upr.toChars());
2025             return setError();
2026         }
2027 
2028         if (fs.prm.type)
2029         {
2030             fs.prm.type = fs.prm.type.typeSemantic(loc, sc);
2031             fs.prm.type = fs.prm.type.addStorageClass(fs.prm.storageClass);
2032             fs.lwr = fs.lwr.implicitCastTo(sc, fs.prm.type);
2033 
2034             if (fs.upr.implicitConvTo(fs.prm.type) || (fs.prm.storageClass & STC.ref_))
2035             {
2036                 fs.upr = fs.upr.implicitCastTo(sc, fs.prm.type);
2037             }
2038             else
2039             {
2040                 // See if upr-1 fits in prm.type
2041                 Expression limit = new MinExp(loc, fs.upr, IntegerExp.literal!1);
2042                 limit = limit.expressionSemantic(sc);
2043                 limit = limit.optimize(WANTvalue);
2044                 if (!limit.implicitConvTo(fs.prm.type))
2045                 {
2046                     fs.upr = fs.upr.implicitCastTo(sc, fs.prm.type);
2047                 }
2048             }
2049         }
2050         else
2051         {
2052             /* Must infer types from lwr and upr
2053              */
2054             Type tlwr = fs.lwr.type.toBasetype();
2055             if (tlwr.ty == Tstruct || tlwr.ty == Tclass)
2056             {
2057                 /* Just picking the first really isn't good enough.
2058                  */
2059                 fs.prm.type = fs.lwr.type;
2060             }
2061             else if (fs.lwr.type == fs.upr.type)
2062             {
2063                 /* Same logic as CondExp ?lwr:upr
2064                  */
2065                 fs.prm.type = fs.lwr.type;
2066             }
2067             else
2068             {
2069                 scope AddExp ea = new AddExp(loc, fs.lwr, fs.upr);
2070                 if (typeCombine(ea, sc))
2071                     return setError();
2072                 fs.prm.type = ea.type;
2073                 fs.lwr = ea.e1;
2074                 fs.upr = ea.e2;
2075             }
2076             fs.prm.type = fs.prm.type.addStorageClass(fs.prm.storageClass);
2077         }
2078         if (fs.prm.type.ty == Terror || fs.lwr.op == TOK.error || fs.upr.op == TOK.error)
2079         {
2080             return setError();
2081         }
2082 
2083         /* Convert to a for loop:
2084          *  foreach (key; lwr .. upr) =>
2085          *  for (auto key = lwr, auto tmp = upr; key < tmp; ++key)
2086          *
2087          *  foreach_reverse (key; lwr .. upr) =>
2088          *  for (auto tmp = lwr, auto key = upr; key-- > tmp;)
2089          */
2090         auto ie = new ExpInitializer(loc, (fs.op == TOK.foreach_) ? fs.lwr : fs.upr);
2091         fs.key = new VarDeclaration(loc, fs.upr.type.mutableOf(), Identifier.generateId("__key"), ie);
2092         fs.key.storage_class |= STC.temp;
2093         SignExtendedNumber lower = getIntRange(fs.lwr).imin;
2094         SignExtendedNumber upper = getIntRange(fs.upr).imax;
2095         if (lower <= upper)
2096         {
2097             fs.key.range = new IntRange(lower, upper);
2098         }
2099 
2100         Identifier id = Identifier.generateId("__limit");
2101         ie = new ExpInitializer(loc, (fs.op == TOK.foreach_) ? fs.upr : fs.lwr);
2102         auto tmp = new VarDeclaration(loc, fs.upr.type, id, ie);
2103         tmp.storage_class |= STC.temp;
2104 
2105         auto cs = new Statements();
2106         // Keep order of evaluation as lwr, then upr
2107         if (fs.op == TOK.foreach_)
2108         {
2109             cs.push(new ExpStatement(loc, fs.key));
2110             cs.push(new ExpStatement(loc, tmp));
2111         }
2112         else
2113         {
2114             cs.push(new ExpStatement(loc, tmp));
2115             cs.push(new ExpStatement(loc, fs.key));
2116         }
2117         Statement forinit = new CompoundDeclarationStatement(loc, cs);
2118 
2119         Expression cond;
2120         if (fs.op == TOK.foreach_reverse_)
2121         {
2122             cond = new PostExp(TOK.minusMinus, loc, new VarExp(loc, fs.key));
2123             if (fs.prm.type.isscalar())
2124             {
2125                 // key-- > tmp
2126                 cond = new CmpExp(TOK.greaterThan, loc, cond, new VarExp(loc, tmp));
2127             }
2128             else
2129             {
2130                 // key-- != tmp
2131                 cond = new EqualExp(TOK.notEqual, loc, cond, new VarExp(loc, tmp));
2132             }
2133         }
2134         else
2135         {
2136             if (fs.prm.type.isscalar())
2137             {
2138                 // key < tmp
2139                 cond = new CmpExp(TOK.lessThan, loc, new VarExp(loc, fs.key), new VarExp(loc, tmp));
2140             }
2141             else
2142             {
2143                 // key != tmp
2144                 cond = new EqualExp(TOK.notEqual, loc, new VarExp(loc, fs.key), new VarExp(loc, tmp));
2145             }
2146         }
2147 
2148         Expression increment = null;
2149         if (fs.op == TOK.foreach_)
2150         {
2151             // key += 1
2152             //increment = new AddAssignExp(loc, new VarExp(loc, fs.key), IntegerExp.literal!1);
2153             increment = new PreExp(TOK.prePlusPlus, loc, new VarExp(loc, fs.key));
2154         }
2155         if ((fs.prm.storageClass & STC.ref_) && fs.prm.type.equals(fs.key.type))
2156         {
2157             fs.key.range = null;
2158             auto v = new AliasDeclaration(loc, fs.prm.ident, fs.key);
2159             fs._body = new CompoundStatement(loc, new ExpStatement(loc, v), fs._body);
2160         }
2161         else
2162         {
2163             ie = new ExpInitializer(loc, new CastExp(loc, new VarExp(loc, fs.key), fs.prm.type));
2164             auto v = new VarDeclaration(loc, fs.prm.type, fs.prm.ident, ie);
2165             v.storage_class |= STC.temp | STC.foreach_ | (fs.prm.storageClass & STC.ref_);
2166             fs._body = new CompoundStatement(loc, new ExpStatement(loc, v), fs._body);
2167             if (fs.key.range && !fs.prm.type.isMutable())
2168             {
2169                 /* Limit the range of the key to the specified range
2170                  */
2171                 v.range = new IntRange(fs.key.range.imin, fs.key.range.imax - SignExtendedNumber(1));
2172             }
2173         }
2174         if (fs.prm.storageClass & STC.ref_)
2175         {
2176             if (fs.key.type.constConv(fs.prm.type) <= MATCH.nomatch)
2177             {
2178                 fs.error("argument type mismatch, `%s` to `ref %s`", fs.key.type.toChars(), fs.prm.type.toChars());
2179                 return setError();
2180             }
2181         }
2182 
2183         auto s = new ForStatement(loc, forinit, cond, increment, fs._body, fs.endloc);
2184         if (LabelStatement ls = checkLabeledLoop(sc, fs))
2185             ls.gotoTarget = s;
2186         result = s.statementSemantic(sc);
2187     }
2188 
2189     override void visit(IfStatement ifs)
2190     {
2191         /* https://dlang.org/spec/statement.html#IfStatement
2192          */
2193 
2194         // check in syntax level
2195         ifs.condition = checkAssignmentAsCondition(ifs.condition);
2196 
2197         auto sym = new ScopeDsymbol();
2198         sym.parent = sc.scopesym;
2199         sym.endlinnum = ifs.endloc.linnum;
2200         Scope* scd = sc.push(sym);
2201         if (ifs.prm)
2202         {
2203             /* Declare prm, which we will set to be the
2204              * result of condition.
2205              */
2206             auto ei = new ExpInitializer(ifs.loc, ifs.condition);
2207             ifs.match = new VarDeclaration(ifs.loc, ifs.prm.type, ifs.prm.ident, ei);
2208             ifs.match.parent = scd.func;
2209             ifs.match.storage_class |= ifs.prm.storageClass;
2210             ifs.match.dsymbolSemantic(scd);
2211 
2212             auto de = new DeclarationExp(ifs.loc, ifs.match);
2213             auto ve = new VarExp(ifs.loc, ifs.match);
2214             ifs.condition = new CommaExp(ifs.loc, de, ve);
2215             ifs.condition = ifs.condition.expressionSemantic(scd);
2216 
2217             if (ifs.match.edtor)
2218             {
2219                 Statement sdtor = new DtorExpStatement(ifs.loc, ifs.match.edtor, ifs.match);
2220                 sdtor = new ScopeGuardStatement(ifs.loc, TOK.onScopeExit, sdtor);
2221                 ifs.ifbody = new CompoundStatement(ifs.loc, sdtor, ifs.ifbody);
2222                 ifs.match.storage_class |= STC.nodtor;
2223 
2224                 // the destructor is always called
2225                 // whether the 'ifbody' is executed or not
2226                 Statement sdtor2 = new DtorExpStatement(ifs.loc, ifs.match.edtor, ifs.match);
2227                 if (ifs.elsebody)
2228                     ifs.elsebody = new CompoundStatement(ifs.loc, sdtor2, ifs.elsebody);
2229                 else
2230                     ifs.elsebody = sdtor2;
2231             }
2232         }
2233         else
2234         {
2235             if (ifs.condition.op == TOK.dotIdentifier)
2236                 (cast(DotIdExp)ifs.condition).noderef = true;
2237 
2238             ifs.condition = ifs.condition.expressionSemantic(scd);
2239             ifs.condition = resolveProperties(scd, ifs.condition);
2240             ifs.condition = ifs.condition.addDtorHook(scd);
2241         }
2242         if (checkNonAssignmentArrayOp(ifs.condition))
2243             ifs.condition = new ErrorExp();
2244         ifs.condition = checkGC(scd, ifs.condition);
2245 
2246         // Convert to boolean after declaring prm so this works:
2247         //  if (S prm = S()) {}
2248         // where S is a struct that defines opCast!bool.
2249         ifs.condition = ifs.condition.toBoolean(scd);
2250 
2251         // If we can short-circuit evaluate the if statement, don't do the
2252         // semantic analysis of the skipped code.
2253         // This feature allows a limited form of conditional compilation.
2254         ifs.condition = ifs.condition.optimize(WANTvalue);
2255 
2256         // Save 'root' of two branches (then and else) at the point where it forks
2257         CtorFlow ctorflow_root = scd.ctorflow.clone();
2258 
2259         ifs.ifbody = ifs.ifbody.semanticNoScope(scd);
2260         scd.pop();
2261 
2262         CtorFlow ctorflow_then = sc.ctorflow;   // move flow results
2263         sc.ctorflow = ctorflow_root;            // reset flow analysis back to root
2264         if (ifs.elsebody)
2265             ifs.elsebody = ifs.elsebody.semanticScope(sc, null, null);
2266 
2267         // Merge 'then' results into 'else' results
2268         sc.merge(ifs.loc, ctorflow_then);
2269 
2270         ctorflow_then.freeFieldinit();          // free extra copy of the data
2271 
2272         if (ifs.condition.op == TOK.error ||
2273             (ifs.ifbody && ifs.ifbody.isErrorStatement()) ||
2274             (ifs.elsebody && ifs.elsebody.isErrorStatement()))
2275         {
2276             return setError();
2277         }
2278         result = ifs;
2279     }
2280 
2281     override void visit(ConditionalStatement cs)
2282     {
2283         //printf("ConditionalStatement::semantic()\n");
2284 
2285         // If we can short-circuit evaluate the if statement, don't do the
2286         // semantic analysis of the skipped code.
2287         // This feature allows a limited form of conditional compilation.
2288         if (cs.condition.include(sc))
2289         {
2290             DebugCondition dc = cs.condition.isDebugCondition();
2291             if (dc)
2292             {
2293                 sc = sc.push();
2294                 sc.flags |= SCOPE.debug_;
2295                 cs.ifbody = cs.ifbody.statementSemantic(sc);
2296                 sc.pop();
2297             }
2298             else
2299                 cs.ifbody = cs.ifbody.statementSemantic(sc);
2300             result = cs.ifbody;
2301         }
2302         else
2303         {
2304             if (cs.elsebody)
2305                 cs.elsebody = cs.elsebody.statementSemantic(sc);
2306             result = cs.elsebody;
2307         }
2308     }
2309 
2310     override void visit(PragmaStatement ps)
2311     {
2312         /* https://dlang.org/spec/statement.html#pragma-statement
2313          */
2314         // Should be merged with PragmaDeclaration
2315 
2316         //printf("PragmaStatement::semantic() %s\n", ps.toChars());
2317         //printf("body = %p\n", ps._body);
2318         if (ps.ident == Id.msg)
2319         {
2320             if (ps.args)
2321             {
2322                 foreach (arg; *ps.args)
2323                 {
2324                     sc = sc.startCTFE();
2325                     auto e = arg.expressionSemantic(sc);
2326                     e = resolveProperties(sc, e);
2327                     sc = sc.endCTFE();
2328 
2329                     // pragma(msg) is allowed to contain types as well as expressions
2330                     e = ctfeInterpretForPragmaMsg(e);
2331                     if (e.op == TOK.error)
2332                     {
2333                         errorSupplemental(ps.loc, "while evaluating `pragma(msg, %s)`", arg.toChars());
2334                         return setError();
2335                     }
2336                     if (auto se = e.toStringExp())
2337                     {
2338                         const slice = se.toUTF8(sc).peekString();
2339                         fprintf(stderr, "%.*s", cast(int)slice.length, slice.ptr);
2340                     }
2341                     else
2342                         fprintf(stderr, "%s", e.toChars());
2343                 }
2344                 fprintf(stderr, "\n");
2345             }
2346         }
2347         else if (ps.ident == Id.lib)
2348         {
2349             version (all)
2350             {
2351                 /* Should this be allowed?
2352                  */
2353                 ps.error("`pragma(lib)` not allowed as statement");
2354                 return setError();
2355             }
2356             else
2357             {
2358                 if (!ps.args || ps.args.dim != 1)
2359                 {
2360                     ps.error("`string` expected for library name");
2361                     return setError();
2362                 }
2363                 else
2364                 {
2365                     auto se = semanticString(sc, (*ps.args)[0], "library name");
2366                     if (!se)
2367                         return setError();
2368 
2369                     if (global.params.verbose)
2370                     {
2371                         message("library   %.*s", cast(int)se.len, se..string);
2372                     }
2373                 }
2374             }
2375         }
2376         else if (ps.ident == Id.linkerDirective)
2377         {
2378             /* Should this be allowed?
2379              */
2380             ps.error("`pragma(linkerDirective)` not allowed as statement");
2381             return setError();
2382         }
2383         else if (ps.ident == Id.startaddress)
2384         {
2385             if (!ps.args || ps.args.dim != 1)
2386                 ps.error("function name expected for start address");
2387             else
2388             {
2389                 Expression e = (*ps.args)[0];
2390                 sc = sc.startCTFE();
2391                 e = e.expressionSemantic(sc);
2392                 e = resolveProperties(sc, e);
2393                 sc = sc.endCTFE();
2394 
2395                 e = e.ctfeInterpret();
2396                 (*ps.args)[0] = e;
2397                 Dsymbol sa = getDsymbol(e);
2398                 if (!sa || !sa.isFuncDeclaration())
2399                 {
2400                     ps.error("function name expected for start address, not `%s`", e.toChars());
2401                     return setError();
2402                 }
2403                 if (ps._body)
2404                 {
2405                     ps._body = ps._body.statementSemantic(sc);
2406                     if (ps._body.isErrorStatement())
2407                     {
2408                         result = ps._body;
2409                         return;
2410                     }
2411                 }
2412                 result = ps;
2413                 return;
2414             }
2415         }
2416         else if (ps.ident == Id.Pinline)
2417         {
2418             PINLINE inlining = PINLINE.default_;
2419             if (!ps.args || ps.args.dim == 0)
2420                 inlining = PINLINE.default_;
2421             else if (!ps.args || ps.args.dim != 1)
2422             {
2423                 ps.error("boolean expression expected for `pragma(inline)`");
2424                 return setError();
2425             }
2426             else
2427             {
2428                 Expression e = (*ps.args)[0];
2429                 if (e.op != TOK.int64 || !e.type.equals(Type.tbool))
2430                 {
2431                     ps.error("pragma(inline, true or false) expected, not `%s`", e.toChars());
2432                     return setError();
2433                 }
2434 
2435                 if (e.isBool(true))
2436                     inlining = PINLINE.always;
2437                 else if (e.isBool(false))
2438                     inlining = PINLINE.never;
2439 
2440                     FuncDeclaration fd = sc.func;
2441                 if (!fd)
2442                 {
2443                     ps.error("`pragma(inline)` is not inside a function");
2444                     return setError();
2445                 }
2446                 fd.inlining = inlining;
2447             }
2448         }
2449         else if (!global.params.ignoreUnsupportedPragmas)
2450         {
2451             ps.error("unrecognized `pragma(%s)`", ps.ident.toChars());
2452             return setError();
2453         }
2454 
2455         if (ps._body)
2456         {
2457             if (ps.ident == Id.msg || ps.ident == Id.startaddress)
2458             {
2459                 ps.error("`pragma(%s)` is missing a terminating `;`", ps.ident.toChars());
2460                 return setError();
2461             }
2462             ps._body = ps._body.statementSemantic(sc);
2463         }
2464         result = ps._body;
2465     }
2466 
2467     override void visit(StaticAssertStatement s)
2468     {
2469         s.sa.semantic2(sc);
2470     }
2471 
2472     override void visit(SwitchStatement ss)
2473     {
2474         /* https://dlang.org/spec/statement.html#switch-statement
2475          */
2476 
2477         //printf("SwitchStatement::semantic(%p)\n", ss);
2478         ss.tryBody = sc.tryBody;
2479         ss.tf = sc.tf;
2480         if (ss.cases)
2481         {
2482             result = ss; // already run
2483             return;
2484         }
2485 
2486         bool conditionError = false;
2487         ss.condition = ss.condition.expressionSemantic(sc);
2488         ss.condition = resolveProperties(sc, ss.condition);
2489 
2490         Type att = null;
2491         TypeEnum te = null;
2492         while (ss.condition.op != TOK.error)
2493         {
2494             // preserve enum type for final switches
2495             if (ss.condition.type.ty == Tenum)
2496                 te = cast(TypeEnum)ss.condition.type;
2497             if (ss.condition.type.isString())
2498             {
2499                 // If it's not an array, cast it to one
2500                 if (ss.condition.type.ty != Tarray)
2501                 {
2502                     ss.condition = ss.condition.implicitCastTo(sc, ss.condition.type.nextOf().arrayOf());
2503                 }
2504                 ss.condition.type = ss.condition.type.constOf();
2505                 break;
2506             }
2507             ss.condition = integralPromotions(ss.condition, sc);
2508             if (ss.condition.op != TOK.error && ss.condition.type.isintegral())
2509                 break;
2510 
2511             auto ad = isAggregate(ss.condition.type);
2512             if (ad && ad.aliasthis && ss.condition.type != att)
2513             {
2514                 if (!att && ss.condition.type.checkAliasThisRec())
2515                     att = ss.condition.type;
2516                 if (auto e = resolveAliasThis(sc, ss.condition, true))
2517                 {
2518                     ss.condition = e;
2519                     continue;
2520                 }
2521             }
2522 
2523             if (ss.condition.op != TOK.error)
2524             {
2525                 ss.error("`%s` must be of integral or string type, it is a `%s`",
2526                     ss.condition.toChars(), ss.condition.type.toChars());
2527                 conditionError = true;
2528                 break;
2529             }
2530         }
2531         if (checkNonAssignmentArrayOp(ss.condition))
2532             ss.condition = new ErrorExp();
2533         ss.condition = ss.condition.optimize(WANTvalue);
2534         ss.condition = checkGC(sc, ss.condition);
2535         if (ss.condition.op == TOK.error)
2536             conditionError = true;
2537 
2538         bool needswitcherror = false;
2539 
2540         ss.lastVar = sc.lastVar;
2541 
2542         sc = sc.push();
2543         sc.sbreak = ss;
2544         sc.sw = ss;
2545 
2546         ss.cases = new CaseStatements();
2547         const inLoopSave = sc.inLoop;
2548         sc.inLoop = true;        // BUG: should use Scope::mergeCallSuper() for each case instead
2549         ss._body = ss._body.statementSemantic(sc);
2550         sc.inLoop = inLoopSave;
2551 
2552         if (conditionError || (ss._body && ss._body.isErrorStatement()))
2553         {
2554             sc.pop();
2555             return setError();
2556         }
2557 
2558         // Resolve any goto case's with exp
2559       Lgotocase:
2560         foreach (gcs; ss.gotoCases)
2561         {
2562             if (!gcs.exp)
2563             {
2564                 gcs.error("no `case` statement following `goto case;`");
2565                 sc.pop();
2566                 return setError();
2567             }
2568 
2569             for (Scope* scx = sc; scx; scx = scx.enclosing)
2570             {
2571                 if (!scx.sw)
2572                     continue;
2573                 foreach (cs; *scx.sw.cases)
2574                 {
2575                     if (cs.exp.equals(gcs.exp))
2576                     {
2577                         gcs.cs = cs;
2578                         continue Lgotocase;
2579                     }
2580                 }
2581             }
2582             gcs.error("`case %s` not found", gcs.exp.toChars());
2583             sc.pop();
2584             return setError();
2585         }
2586 
2587         if (ss.isFinal)
2588         {
2589             Type t = ss.condition.type;
2590             Dsymbol ds;
2591             EnumDeclaration ed = null;
2592             if (t && ((ds = t.toDsymbol(sc)) !is null))
2593                 ed = ds.isEnumDeclaration(); // typedef'ed enum
2594             if (!ed && te && ((ds = te.toDsymbol(sc)) !is null))
2595                 ed = ds.isEnumDeclaration();
2596             if (ed)
2597             {
2598               Lmembers:
2599                 foreach (es; *ed.members)
2600                 {
2601                     EnumMember em = es.isEnumMember();
2602                     if (em)
2603                     {
2604                         foreach (cs; *ss.cases)
2605                         {
2606                             if (cs.exp.equals(em.value) || (!cs.exp.type.isString() && !em.value.type.isString() && cs.exp.toInteger() == em.value.toInteger()))
2607                                 continue Lmembers;
2608                         }
2609                         ss.error("`enum` member `%s` not represented in `final switch`", em.toChars());
2610                         sc.pop();
2611                         return setError();
2612                     }
2613                 }
2614             }
2615             else
2616                 needswitcherror = true;
2617         }
2618 
2619         if (!sc.sw.sdefault && (!ss.isFinal || needswitcherror || global.params.useAssert == CHECKENABLE.on))
2620         {
2621             ss.hasNoDefault = 1;
2622 
2623             if (!ss.isFinal && (!ss._body || !ss._body.isErrorStatement()))
2624                 ss.error("`switch` statement without a `default`; use `final switch` or add `default: assert(0);` or add `default: break;`");
2625 
2626             // Generate runtime error if the default is hit
2627             auto a = new Statements();
2628             CompoundStatement cs;
2629             Statement s;
2630 
2631             if (global.params.useSwitchError == CHECKENABLE.on &&
2632                 global.params.checkAction != CHECKACTION.halt)
2633             {
2634                 if (global.params.checkAction == CHECKACTION.C)
2635                 {
2636                     /* Rewrite as an assert(0) and let e2ir generate
2637                      * the call to the C assert failure function
2638                      */
2639                     s = new ExpStatement(ss.loc, new AssertExp(ss.loc, new IntegerExp(ss.loc, 0, Type.tint32)));
2640                 }
2641                 else
2642                 {
2643                     if (!verifyHookExist(ss.loc, *sc, Id.__switch_error, "generating assert messages"))
2644                         return setError();
2645 
2646                     Expression sl = new IdentifierExp(ss.loc, Id.empty);
2647                     sl = new DotIdExp(ss.loc, sl, Id.object);
2648                     sl = new DotIdExp(ss.loc, sl, Id.__switch_error);
2649 
2650                     Expressions* args = new Expressions(2);
2651                     (*args)[0] = new StringExp(ss.loc, ss.loc.filename.toDString());
2652                     (*args)[1] = new IntegerExp(ss.loc.linnum);
2653 
2654                     sl = new CallExp(ss.loc, sl, args);
2655                     sl.expressionSemantic(sc);
2656 
2657                     s = new SwitchErrorStatement(ss.loc, sl);
2658                 }
2659             }
2660             else
2661                 s = new ExpStatement(ss.loc, new HaltExp(ss.loc));
2662 
2663             a.reserve(2);
2664             sc.sw.sdefault = new DefaultStatement(ss.loc, s);
2665             a.push(ss._body);
2666             if (ss._body.blockExit(sc.func, false) & BE.fallthru)
2667                 a.push(new BreakStatement(Loc.initial, null));
2668             a.push(sc.sw.sdefault);
2669             cs = new CompoundStatement(ss.loc, a);
2670             ss._body = cs;
2671         }
2672 
2673         if (ss.checkLabel())
2674         {
2675             sc.pop();
2676             return setError();
2677         }
2678 
2679 
2680         if (ss.condition.type.isString())
2681         {
2682             // Transform a switch with string labels into a switch with integer labels.
2683 
2684             // The integer value of each case corresponds to the index of each label
2685             // string in the sorted array of label strings.
2686 
2687             // The value of the integer condition is obtained by calling the druntime template
2688             // switch(object.__switch(cond, options...)) {0: {...}, 1: {...}, ...}
2689 
2690             // We sort a copy of the array of labels because we want to do a binary search in object.__switch,
2691             // without modifying the order of the case blocks here in the compiler.
2692 
2693             if (!verifyHookExist(ss.loc, *sc, Id.__switch, "switch cases on strings"))
2694                 return setError();
2695 
2696             size_t numcases = 0;
2697             if (ss.cases)
2698                 numcases = ss.cases.dim;
2699 
2700             for (size_t i = 0; i < numcases; i++)
2701             {
2702                 CaseStatement cs = (*ss.cases)[i];
2703                 cs.index = cast(int)i;
2704             }
2705 
2706             // Make a copy of all the cases so that qsort doesn't scramble the actual
2707             // data we pass to codegen (the order of the cases in the switch).
2708             CaseStatements *csCopy = (*ss.cases).copy();
2709 
2710             if (numcases)
2711             {
2712                 static int sort_compare(in CaseStatement* x, in CaseStatement* y) @trusted
2713                 {
2714                     auto se1 = x.exp.isStringExp();
2715                     auto se2 = y.exp.isStringExp();
2716                     return (se1 && se2) ? se1.compare(se2) : 0;
2717                 }
2718                 // Sort cases for efficient lookup
2719                 csCopy.sort!sort_compare;
2720             }
2721 
2722             // The actual lowering
2723             auto arguments = new Expressions();
2724             arguments.push(ss.condition);
2725 
2726             auto compileTimeArgs = new Objects();
2727 
2728             // The type & label no.
2729             compileTimeArgs.push(new TypeExp(ss.loc, ss.condition.type.nextOf()));
2730 
2731             // The switch labels
2732             foreach (caseString; *csCopy)
2733             {
2734                 compileTimeArgs.push(caseString.exp);
2735             }
2736 
2737             Expression sl = new IdentifierExp(ss.loc, Id.empty);
2738             sl = new DotIdExp(ss.loc, sl, Id.object);
2739             sl = new DotTemplateInstanceExp(ss.loc, sl, Id.__switch, compileTimeArgs);
2740 
2741             sl = new CallExp(ss.loc, sl, arguments);
2742             sl.expressionSemantic(sc);
2743             ss.condition = sl;
2744 
2745             auto i = 0;
2746             foreach (c; *csCopy)
2747             {
2748                 (*ss.cases)[c.index].exp = new IntegerExp(i++);
2749             }
2750 
2751             //printf("%s\n", ss._body.toChars());
2752             ss.statementSemantic(sc);
2753         }
2754 
2755         sc.pop();
2756         result = ss;
2757     }
2758 
2759     override void visit(CaseStatement cs)
2760     {
2761         SwitchStatement sw = sc.sw;
2762         bool errors = false;
2763 
2764         //printf("CaseStatement::semantic() %s\n", toChars());
2765         sc = sc.startCTFE();
2766         cs.exp = cs.exp.expressionSemantic(sc);
2767         cs.exp = resolveProperties(sc, cs.exp);
2768         sc = sc.endCTFE();
2769 
2770         if (sw)
2771         {
2772             cs.exp = cs.exp.implicitCastTo(sc, sw.condition.type);
2773             cs.exp = cs.exp.optimize(WANTvalue | WANTexpand);
2774 
2775             Expression e = cs.exp;
2776             // Remove all the casts the user and/or implicitCastTo may introduce
2777             // otherwise we'd sometimes fail the check below.
2778             while (e.op == TOK.cast_)
2779                 e = (cast(CastExp)e).e1;
2780 
2781             /* This is where variables are allowed as case expressions.
2782              */
2783             if (e.op == TOK.variable)
2784             {
2785                 VarExp ve = cast(VarExp)e;
2786                 VarDeclaration v = ve.var.isVarDeclaration();
2787                 Type t = cs.exp.type.toBasetype();
2788                 if (v && (t.isintegral() || t.ty == Tclass))
2789                 {
2790                     /* Flag that we need to do special code generation
2791                      * for this, i.e. generate a sequence of if-then-else
2792                      */
2793                     sw.hasVars = 1;
2794 
2795                     /* TODO check if v can be uninitialized at that point.
2796                      */
2797                     if (!v.isConst() && !v.isImmutable())
2798                     {
2799                         cs.deprecation("`case` variables have to be `const` or `immutable`");
2800                     }
2801 
2802                     if (sw.isFinal)
2803                     {
2804                         cs.error("`case` variables not allowed in `final switch` statements");
2805                         errors = true;
2806                     }
2807 
2808                     /* Find the outermost scope `scx` that set `sw`.
2809                      * Then search scope `scx` for a declaration of `v`.
2810                      */
2811                     for (Scope* scx = sc; scx; scx = scx.enclosing)
2812                     {
2813                         if (scx.enclosing && scx.enclosing.sw == sw)
2814                             continue;
2815                         assert(scx.sw == sw);
2816 
2817                         if (!scx.search(cs.exp.loc, v.ident, null))
2818                         {
2819                             cs.error("`case` variable `%s` declared at %s cannot be declared in `switch` body",
2820                                 v.toChars(), v.loc.toChars());
2821                             errors = true;
2822                         }
2823                         break;
2824                     }
2825                     goto L1;
2826                 }
2827             }
2828             else
2829                 cs.exp = cs.exp.ctfeInterpret();
2830 
2831             if (StringExp se = cs.exp.toStringExp())
2832                 cs.exp = se;
2833             else if (cs.exp.op != TOK.int64 && cs.exp.op != TOK.error)
2834             {
2835                 cs.error("`case` must be a `string` or an integral constant, not `%s`", cs.exp.toChars());
2836                 errors = true;
2837             }
2838 
2839         L1:
2840             foreach (cs2; *sw.cases)
2841             {
2842                 //printf("comparing '%s' with '%s'\n", exp.toChars(), cs.exp.toChars());
2843                 if (cs2.exp.equals(cs.exp))
2844                 {
2845                     cs.error("duplicate `case %s` in `switch` statement", cs.exp.toChars());
2846                     errors = true;
2847                     break;
2848                 }
2849             }
2850 
2851             sw.cases.push(cs);
2852 
2853             // Resolve any goto case's with no exp to this case statement
2854             for (size_t i = 0; i < sw.gotoCases.dim;)
2855             {
2856                 GotoCaseStatement gcs = sw.gotoCases[i];
2857                 if (!gcs.exp)
2858                 {
2859                     gcs.cs = cs;
2860                     sw.gotoCases.remove(i); // remove from array
2861                     continue;
2862                 }
2863                 i++;
2864             }
2865 
2866             if (sc.sw.tf != sc.tf)
2867             {
2868                 cs.error("`switch` and `case` are in different `finally` blocks");
2869                 errors = true;
2870             }
2871             if (sc.sw.tryBody != sc.tryBody)
2872             {
2873                 cs.error("case cannot be in different `try` block level from `switch`");
2874                 errors = true;
2875             }
2876         }
2877         else
2878         {
2879             cs.error("`case` not in `switch` statement");
2880             errors = true;
2881         }
2882 
2883         sc.ctorflow.orCSX(CSX.label);
2884         cs.statement = cs.statement.statementSemantic(sc);
2885         if (cs.statement.isErrorStatement())
2886         {
2887             result = cs.statement;
2888             return;
2889         }
2890         if (errors || cs.exp.op == TOK.error)
2891             return setError();
2892 
2893         cs.lastVar = sc.lastVar;
2894         result = cs;
2895     }
2896 
2897     override void visit(CaseRangeStatement crs)
2898     {
2899         SwitchStatement sw = sc.sw;
2900         if (sw is null)
2901         {
2902             crs.error("case range not in `switch` statement");
2903             return setError();
2904         }
2905 
2906         //printf("CaseRangeStatement::semantic() %s\n", toChars());
2907         bool errors = false;
2908         if (sw.isFinal)
2909         {
2910             crs.error("case ranges not allowed in `final switch`");
2911             errors = true;
2912         }
2913 
2914         sc = sc.startCTFE();
2915         crs.first = crs.first.expressionSemantic(sc);
2916         crs.first = resolveProperties(sc, crs.first);
2917         sc = sc.endCTFE();
2918         crs.first = crs.first.implicitCastTo(sc, sw.condition.type);
2919         crs.first = crs.first.ctfeInterpret();
2920 
2921         sc = sc.startCTFE();
2922         crs.last = crs.last.expressionSemantic(sc);
2923         crs.last = resolveProperties(sc, crs.last);
2924         sc = sc.endCTFE();
2925         crs.last = crs.last.implicitCastTo(sc, sw.condition.type);
2926         crs.last = crs.last.ctfeInterpret();
2927 
2928         if (crs.first.op == TOK.error || crs.last.op == TOK.error || errors)
2929         {
2930             if (crs.statement)
2931                 crs.statement.statementSemantic(sc);
2932             return setError();
2933         }
2934 
2935         uinteger_t fval = crs.first.toInteger();
2936         uinteger_t lval = crs.last.toInteger();
2937         if ((crs.first.type.isunsigned() && fval > lval) || (!crs.first.type.isunsigned() && cast(sinteger_t)fval > cast(sinteger_t)lval))
2938         {
2939             crs.error("first `case %s` is greater than last `case %s`", crs.first.toChars(), crs.last.toChars());
2940             errors = true;
2941             lval = fval;
2942         }
2943 
2944         if (lval - fval > 256)
2945         {
2946             crs.error("had %llu cases which is more than 256 cases in case range", lval - fval);
2947             errors = true;
2948             lval = fval + 256;
2949         }
2950 
2951         if (errors)
2952             return setError();
2953 
2954         /* This works by replacing the CaseRange with an array of Case's.
2955          *
2956          * case a: .. case b: s;
2957          *    =>
2958          * case a:
2959          *   [...]
2960          * case b:
2961          *   s;
2962          */
2963 
2964         auto statements = new Statements();
2965         for (uinteger_t i = fval; i != lval + 1; i++)
2966         {
2967             Statement s = crs.statement;
2968             if (i != lval) // if not last case
2969                 s = new ExpStatement(crs.loc, cast(Expression)null);
2970             Expression e = new IntegerExp(crs.loc, i, crs.first.type);
2971             Statement cs = new CaseStatement(crs.loc, e, s);
2972             statements.push(cs);
2973         }
2974         Statement s = new CompoundStatement(crs.loc, statements);
2975         sc.ctorflow.orCSX(CSX.label);
2976         s = s.statementSemantic(sc);
2977         result = s;
2978     }
2979 
2980     override void visit(DefaultStatement ds)
2981     {
2982         //printf("DefaultStatement::semantic()\n");
2983         bool errors = false;
2984         if (sc.sw)
2985         {
2986             if (sc.sw.sdefault)
2987             {
2988                 ds.error("`switch` statement already has a default");
2989                 errors = true;
2990             }
2991             sc.sw.sdefault = ds;
2992 
2993             if (sc.sw.tf != sc.tf)
2994             {
2995                 ds.error("`switch` and `default` are in different `finally` blocks");
2996                 errors = true;
2997             }
2998             if (sc.sw.tryBody != sc.tryBody)
2999             {
3000                 ds.error("default cannot be in different `try` block level from `switch`");
3001                 errors = true;
3002             }
3003             if (sc.sw.isFinal)
3004             {
3005                 ds.error("`default` statement not allowed in `final switch` statement");
3006                 errors = true;
3007             }
3008         }
3009         else
3010         {
3011             ds.error("`default` not in `switch` statement");
3012             errors = true;
3013         }
3014 
3015         sc.ctorflow.orCSX(CSX.label);
3016         ds.statement = ds.statement.statementSemantic(sc);
3017         if (errors || ds.statement.isErrorStatement())
3018             return setError();
3019 
3020         ds.lastVar = sc.lastVar;
3021         result = ds;
3022     }
3023 
3024     override void visit(GotoDefaultStatement gds)
3025     {
3026         /* https://dlang.org/spec/statement.html#goto-statement
3027          */
3028 
3029         gds.sw = sc.sw;
3030         if (!gds.sw)
3031         {
3032             gds.error("`goto default` not in `switch` statement");
3033             return setError();
3034         }
3035         if (gds.sw.isFinal)
3036         {
3037             gds.error("`goto default` not allowed in `final switch` statement");
3038             return setError();
3039         }
3040         result = gds;
3041     }
3042 
3043     override void visit(GotoCaseStatement gcs)
3044     {
3045         /* https://dlang.org/spec/statement.html#goto-statement
3046          */
3047 
3048         if (!sc.sw)
3049         {
3050             gcs.error("`goto case` not in `switch` statement");
3051             return setError();
3052         }
3053 
3054         if (gcs.exp)
3055         {
3056             gcs.exp = gcs.exp.expressionSemantic(sc);
3057             gcs.exp = gcs.exp.implicitCastTo(sc, sc.sw.condition.type);
3058             gcs.exp = gcs.exp.optimize(WANTvalue);
3059             if (gcs.exp.op == TOK.error)
3060                 return setError();
3061         }
3062 
3063         sc.sw.gotoCases.push(gcs);
3064         result = gcs;
3065     }
3066 
3067     override void visit(ReturnStatement rs)
3068     {
3069         /* https://dlang.org/spec/statement.html#return-statement
3070          */
3071 
3072         //printf("ReturnStatement.dsymbolSemantic() %p, %s\n", rs, rs.toChars());
3073 
3074         FuncDeclaration fd = sc.parent.isFuncDeclaration();
3075         if (fd.fes)
3076             fd = fd.fes.func; // fd is now function enclosing foreach
3077 
3078             TypeFunction tf = cast(TypeFunction)fd.type;
3079         assert(tf.ty == Tfunction);
3080 
3081         if (rs.exp && rs.exp.op == TOK.variable && (cast(VarExp)rs.exp).var == fd.vresult)
3082         {
3083             // return vresult;
3084             if (sc.fes)
3085             {
3086                 assert(rs.caseDim == 0);
3087                 sc.fes.cases.push(rs);
3088                 result = new ReturnStatement(Loc.initial, new IntegerExp(sc.fes.cases.dim + 1));
3089                 return;
3090             }
3091             if (fd.returnLabel)
3092             {
3093                 auto gs = new GotoStatement(rs.loc, Id.returnLabel);
3094                 gs.label = fd.returnLabel;
3095                 result = gs;
3096                 return;
3097             }
3098 
3099             if (!fd.returns)
3100                 fd.returns = new ReturnStatements();
3101             fd.returns.push(rs);
3102             result = rs;
3103             return;
3104         }
3105 
3106         Type tret = tf.next;
3107         Type tbret = tret ? tret.toBasetype() : null;
3108 
3109         bool inferRef = (tf.isref && (fd.storage_class & STC.auto_));
3110         Expression e0 = null;
3111 
3112         bool errors = false;
3113         if (sc.flags & SCOPE.contract)
3114         {
3115             rs.error("`return` statements cannot be in contracts");
3116             errors = true;
3117         }
3118         if (sc.os && sc.os.tok != TOK.onScopeFailure)
3119         {
3120             rs.error("`return` statements cannot be in `%s` bodies", Token.toChars(sc.os.tok));
3121             errors = true;
3122         }
3123         if (sc.tf)
3124         {
3125             rs.error("`return` statements cannot be in `finally` bodies");
3126             errors = true;
3127         }
3128 
3129         if (fd.isCtorDeclaration())
3130         {
3131             if (rs.exp)
3132             {
3133                 rs.error("cannot return expression from constructor");
3134                 errors = true;
3135             }
3136 
3137             // Constructors implicitly do:
3138             //      return this;
3139             rs.exp = new ThisExp(Loc.initial);
3140             rs.exp.type = tret;
3141         }
3142         else if (rs.exp)
3143         {
3144             fd.hasReturnExp |= (fd.hasReturnExp & 1 ? 16 : 1);
3145 
3146             FuncLiteralDeclaration fld = fd.isFuncLiteralDeclaration();
3147             if (tret)
3148                 rs.exp = inferType(rs.exp, tret);
3149             else if (fld && fld.treq)
3150                 rs.exp = inferType(rs.exp, fld.treq.nextOf().nextOf());
3151 
3152             rs.exp = rs.exp.expressionSemantic(sc);
3153             rs.exp.checkSharedAccess(sc);
3154 
3155             // for static alias this: https://issues.dlang.org/show_bug.cgi?id=17684
3156             if (rs.exp.op == TOK.type)
3157                 rs.exp = resolveAliasThis(sc, rs.exp);
3158 
3159             rs.exp = resolveProperties(sc, rs.exp);
3160             if (rs.exp.checkType())
3161                 rs.exp = new ErrorExp();
3162             if (auto f = isFuncAddress(rs.exp))
3163             {
3164                 if (fd.inferRetType && f.checkForwardRef(rs.exp.loc))
3165                     rs.exp = new ErrorExp();
3166             }
3167             if (checkNonAssignmentArrayOp(rs.exp))
3168                 rs.exp = new ErrorExp();
3169 
3170             // Extract side-effect part
3171             rs.exp = Expression.extractLast(rs.exp, e0);
3172             if (rs.exp.op == TOK.call)
3173                 rs.exp = valueNoDtor(rs.exp);
3174 
3175             if (e0)
3176                 e0 = e0.optimize(WANTvalue);
3177 
3178             /* Void-return function can have void typed expression
3179              * on return statement.
3180              */
3181             if (tbret && tbret.ty == Tvoid || rs.exp.type.ty == Tvoid)
3182             {
3183                 if (rs.exp.type.ty != Tvoid)
3184                 {
3185                     rs.error("cannot return non-void from `void` function");
3186                     errors = true;
3187                     rs.exp = new CastExp(rs.loc, rs.exp, Type.tvoid);
3188                     rs.exp = rs.exp.expressionSemantic(sc);
3189                 }
3190 
3191                 /* Replace:
3192                  *      return exp;
3193                  * with:
3194                  *      exp; return;
3195                  */
3196                 e0 = Expression.combine(e0, rs.exp);
3197                 rs.exp = null;
3198             }
3199             if (e0)
3200                 e0 = checkGC(sc, e0);
3201         }
3202 
3203         if (rs.exp)
3204         {
3205             if (fd.inferRetType) // infer return type
3206             {
3207                 if (!tret)
3208                 {
3209                     tf.next = rs.exp.type;
3210                 }
3211                 else if (tret.ty != Terror && !rs.exp.type.equals(tret))
3212                 {
3213                     int m1 = rs.exp.type.implicitConvTo(tret);
3214                     int m2 = tret.implicitConvTo(rs.exp.type);
3215                     //printf("exp.type = %s m2<-->m1 tret %s\n", exp.type.toChars(), tret.toChars());
3216                     //printf("m1 = %d, m2 = %d\n", m1, m2);
3217 
3218                     if (m1 && m2)
3219                     {
3220                     }
3221                     else if (!m1 && m2)
3222                         tf.next = rs.exp.type;
3223                     else if (m1 && !m2)
3224                     {
3225                     }
3226                     else if (rs.exp.op != TOK.error)
3227                     {
3228                         rs.error("Expected return type of `%s`, not `%s`:",
3229                                  tret.toChars(),
3230                                  rs.exp.type.toChars());
3231                         errorSupplemental((fd.returns) ? (*fd.returns)[0].loc : fd.loc,
3232                                           "Return type of `%s` inferred here.",
3233                                           tret.toChars());
3234 
3235                         errors = true;
3236                         tf.next = Type.terror;
3237                     }
3238                 }
3239 
3240                 tret = tf.next;
3241                 tbret = tret.toBasetype();
3242             }
3243 
3244             if (inferRef) // deduce 'auto ref'
3245             {
3246                 /* Determine "refness" of function return:
3247                  * if it's an lvalue, return by ref, else return by value
3248                  * https://dlang.org/spec/function.html#auto-ref-functions
3249                  */
3250 
3251                 void turnOffRef()
3252                 {
3253                     tf.isref = false;    // return by value
3254                     tf.isreturn = false; // ignore 'return' attribute, whether explicit or inferred
3255                     fd.storage_class &= ~STC.return_;
3256                 }
3257 
3258                 if (rs.exp.isLvalue())
3259                 {
3260                     /* May return by ref
3261                      */
3262                     if (checkReturnEscapeRef(sc, rs.exp, true))
3263                         turnOffRef();
3264                     else if (!rs.exp.type.constConv(tf.next))
3265                         turnOffRef();
3266                 }
3267                 else
3268                     turnOffRef();
3269 
3270                 /* The "refness" is determined by all of return statements.
3271                  * This means:
3272                  *    return 3; return x;  // ok, x can be a value
3273                  *    return x; return 3;  // ok, x can be a value
3274                  */
3275             }
3276         }
3277         else
3278         {
3279             // infer return type
3280             if (fd.inferRetType)
3281             {
3282                 if (tf.next && tf.next.ty != Tvoid)
3283                 {
3284                     if (tf.next.ty != Terror)
3285                     {
3286                         rs.error("mismatched function return type inference of `void` and `%s`", tf.next.toChars());
3287                     }
3288                     errors = true;
3289                     tf.next = Type.terror;
3290                 }
3291                 else
3292                     tf.next = Type.tvoid;
3293 
3294                     tret = tf.next;
3295                 tbret = tret.toBasetype();
3296             }
3297 
3298             if (inferRef) // deduce 'auto ref'
3299                 tf.isref = false;
3300 
3301             if (tbret.ty != Tvoid) // if non-void return
3302             {
3303                 if (tbret.ty != Terror)
3304                     rs.error("`return` expression expected");
3305                 errors = true;
3306             }
3307             else if (fd.isMain())
3308             {
3309                 // main() returns 0, even if it returns void
3310                 rs.exp = IntegerExp.literal!0;
3311             }
3312         }
3313 
3314         // If any branches have called a ctor, but this branch hasn't, it's an error
3315         if (sc.ctorflow.callSuper & CSX.any_ctor && !(sc.ctorflow.callSuper & (CSX.this_ctor | CSX.super_ctor)))
3316         {
3317             rs.error("`return` without calling constructor");
3318             errors = true;
3319         }
3320 
3321         if (sc.ctorflow.fieldinit.length)       // if aggregate fields are being constructed
3322         {
3323             auto ad = fd.isMemberLocal();
3324             assert(ad);
3325             foreach (i, v; ad.fields)
3326             {
3327                 bool mustInit = (v.storage_class & STC.nodefaultctor || v.type.needsNested());
3328                 if (mustInit && !(sc.ctorflow.fieldinit[i].csx & CSX.this_ctor))
3329                 {
3330                     rs.error("an earlier `return` statement skips field `%s` initialization", v.toChars());
3331                     errors = true;
3332                 }
3333             }
3334         }
3335         sc.ctorflow.orCSX(CSX.return_);
3336 
3337         if (errors)
3338             return setError();
3339 
3340         if (sc.fes)
3341         {
3342             if (!rs.exp)
3343             {
3344                 // Send out "case receiver" statement to the foreach.
3345                 //  return exp;
3346                 Statement s = new ReturnStatement(Loc.initial, rs.exp);
3347                 sc.fes.cases.push(s);
3348 
3349                 // Immediately rewrite "this" return statement as:
3350                 //  return cases.dim+1;
3351                 rs.exp = new IntegerExp(sc.fes.cases.dim + 1);
3352                 if (e0)
3353                 {
3354                     result = new CompoundStatement(rs.loc, new ExpStatement(rs.loc, e0), rs);
3355                     return;
3356                 }
3357                 result = rs;
3358                 return;
3359             }
3360             else
3361             {
3362                 fd.buildResultVar(null, rs.exp.type);
3363                 bool r = fd.vresult.checkNestedReference(sc, Loc.initial);
3364                 assert(!r); // vresult should be always accessible
3365 
3366                 // Send out "case receiver" statement to the foreach.
3367                 //  return vresult;
3368                 Statement s = new ReturnStatement(Loc.initial, new VarExp(Loc.initial, fd.vresult));
3369                 sc.fes.cases.push(s);
3370 
3371                 // Save receiver index for the later rewriting from:
3372                 //  return exp;
3373                 // to:
3374                 //  vresult = exp; retrun caseDim;
3375                 rs.caseDim = sc.fes.cases.dim + 1;
3376             }
3377         }
3378         if (rs.exp)
3379         {
3380             if (!fd.returns)
3381                 fd.returns = new ReturnStatements();
3382             fd.returns.push(rs);
3383         }
3384         if (e0)
3385         {
3386             if (e0.op == TOK.declaration || e0.op == TOK.comma)
3387             {
3388                 rs.exp = Expression.combine(e0, rs.exp);
3389             }
3390             else
3391             {
3392                 result = new CompoundStatement(rs.loc, new ExpStatement(rs.loc, e0), rs);
3393                 return;
3394             }
3395         }
3396         result = rs;
3397     }
3398 
3399     override void visit(BreakStatement bs)
3400     {
3401         /* https://dlang.org/spec/statement.html#break-statement
3402          */
3403 
3404         //printf("BreakStatement::semantic()\n");
3405 
3406         // If:
3407         //  break Identifier;
3408         if (bs.ident)
3409         {
3410             bs.ident = fixupLabelName(sc, bs.ident);
3411 
3412             FuncDeclaration thisfunc = sc.func;
3413 
3414             for (Scope* scx = sc; scx; scx = scx.enclosing)
3415             {
3416                 if (scx.func != thisfunc) // if in enclosing function
3417                 {
3418                     if (sc.fes) // if this is the body of a foreach
3419                     {
3420                         /* Post this statement to the fes, and replace
3421                          * it with a return value that caller will put into
3422                          * a switch. Caller will figure out where the break
3423                          * label actually is.
3424                          * Case numbers start with 2, not 0, as 0 is continue
3425                          * and 1 is break.
3426                          */
3427                         sc.fes.cases.push(bs);
3428                         result = new ReturnStatement(Loc.initial, new IntegerExp(sc.fes.cases.dim + 1));
3429                         return;
3430                     }
3431                     break; // can't break to it
3432                 }
3433 
3434                 LabelStatement ls = scx.slabel;
3435                 if (ls && ls.ident == bs.ident)
3436                 {
3437                     Statement s = ls.statement;
3438                     if (!s || !s.hasBreak())
3439                         bs.error("label `%s` has no `break`", bs.ident.toChars());
3440                     else if (ls.tf != sc.tf)
3441                         bs.error("cannot break out of `finally` block");
3442                     else
3443                     {
3444                         ls.breaks = true;
3445                         result = bs;
3446                         return;
3447                     }
3448                     return setError();
3449                 }
3450             }
3451             bs.error("enclosing label `%s` for `break` not found", bs.ident.toChars());
3452             return setError();
3453         }
3454         else if (!sc.sbreak)
3455         {
3456             if (sc.os && sc.os.tok != TOK.onScopeFailure)
3457             {
3458                 bs.error("`break` is not inside `%s` bodies", Token.toChars(sc.os.tok));
3459             }
3460             else if (sc.fes)
3461             {
3462                 // Replace break; with return 1;
3463                 result = new ReturnStatement(Loc.initial, IntegerExp.literal!1);
3464                 return;
3465             }
3466             else
3467                 bs.error("`break` is not inside a loop or `switch`");
3468             return setError();
3469         }
3470         else if (sc.sbreak.isForwardingStatement())
3471         {
3472             bs.error("must use labeled `break` within `static foreach`");
3473         }
3474         result = bs;
3475     }
3476 
3477     override void visit(ContinueStatement cs)
3478     {
3479         /* https://dlang.org/spec/statement.html#continue-statement
3480          */
3481 
3482         //printf("ContinueStatement::semantic() %p\n", cs);
3483         if (cs.ident)
3484         {
3485             cs.ident = fixupLabelName(sc, cs.ident);
3486 
3487             Scope* scx;
3488             FuncDeclaration thisfunc = sc.func;
3489 
3490             for (scx = sc; scx; scx = scx.enclosing)
3491             {
3492                 LabelStatement ls;
3493                 if (scx.func != thisfunc) // if in enclosing function
3494                 {
3495                     if (sc.fes) // if this is the body of a foreach
3496                     {
3497                         for (; scx; scx = scx.enclosing)
3498                         {
3499                             ls = scx.slabel;
3500                             if (ls && ls.ident == cs.ident && ls.statement == sc.fes)
3501                             {
3502                                 // Replace continue ident; with return 0;
3503                                 result = new ReturnStatement(Loc.initial, IntegerExp.literal!0);
3504                                 return;
3505                             }
3506                         }
3507 
3508                         /* Post this statement to the fes, and replace
3509                          * it with a return value that caller will put into
3510                          * a switch. Caller will figure out where the break
3511                          * label actually is.
3512                          * Case numbers start with 2, not 0, as 0 is continue
3513                          * and 1 is break.
3514                          */
3515                         sc.fes.cases.push(cs);
3516                         result = new ReturnStatement(Loc.initial, new IntegerExp(sc.fes.cases.dim + 1));
3517                         return;
3518                     }
3519                     break; // can't continue to it
3520                 }
3521 
3522                 ls = scx.slabel;
3523                 if (ls && ls.ident == cs.ident)
3524                 {
3525                     Statement s = ls.statement;
3526                     if (!s || !s.hasContinue())
3527                         cs.error("label `%s` has no `continue`", cs.ident.toChars());
3528                     else if (ls.tf != sc.tf)
3529                         cs.error("cannot continue out of `finally` block");
3530                     else
3531                     {
3532                         result = cs;
3533                         return;
3534                     }
3535                     return setError();
3536                 }
3537             }
3538             cs.error("enclosing label `%s` for `continue` not found", cs.ident.toChars());
3539             return setError();
3540         }
3541         else if (!sc.scontinue)
3542         {
3543             if (sc.os && sc.os.tok != TOK.onScopeFailure)
3544             {
3545                 cs.error("`continue` is not inside `%s` bodies", Token.toChars(sc.os.tok));
3546             }
3547             else if (sc.fes)
3548             {
3549                 // Replace continue; with return 0;
3550                 result = new ReturnStatement(Loc.initial, IntegerExp.literal!0);
3551                 return;
3552             }
3553             else
3554                 cs.error("`continue` is not inside a loop");
3555             return setError();
3556         }
3557         else if (sc.scontinue.isForwardingStatement())
3558         {
3559             cs.error("must use labeled `continue` within `static foreach`");
3560         }
3561         result = cs;
3562     }
3563 
3564     override void visit(SynchronizedStatement ss)
3565     {
3566         /* https://dlang.org/spec/statement.html#synchronized-statement
3567          */
3568 
3569         if (ss.exp)
3570         {
3571             ss.exp = ss.exp.expressionSemantic(sc);
3572             ss.exp = resolveProperties(sc, ss.exp);
3573             ss.exp = ss.exp.optimize(WANTvalue);
3574             ss.exp = checkGC(sc, ss.exp);
3575             if (ss.exp.op == TOK.error)
3576             {
3577                 if (ss._body)
3578                     ss._body = ss._body.statementSemantic(sc);
3579                 return setError();
3580             }
3581 
3582             ClassDeclaration cd = ss.exp.type.isClassHandle();
3583             if (!cd)
3584             {
3585                 ss.error("can only `synchronize` on class objects, not `%s`", ss.exp.type.toChars());
3586                 return setError();
3587             }
3588             else if (cd.isInterfaceDeclaration())
3589             {
3590                 /* Cast the interface to an object, as the object has the monitor,
3591                  * not the interface.
3592                  */
3593                 if (!ClassDeclaration.object)
3594                 {
3595                     ss.error("missing or corrupt object.d");
3596                     fatal();
3597                 }
3598 
3599                 Type t = ClassDeclaration.object.type;
3600                 t = t.typeSemantic(Loc.initial, sc).toBasetype();
3601                 assert(t.ty == Tclass);
3602 
3603                 ss.exp = new CastExp(ss.loc, ss.exp, t);
3604                 ss.exp = ss.exp.expressionSemantic(sc);
3605             }
3606             version (all)
3607             {
3608                 /* Rewrite as:
3609                  *  auto tmp = exp;
3610                  *  _d_monitorenter(tmp);
3611                  *  try { body } finally { _d_monitorexit(tmp); }
3612                  */
3613                 auto tmp = copyToTemp(0, "__sync", ss.exp);
3614                 tmp.dsymbolSemantic(sc);
3615 
3616                 auto cs = new Statements();
3617                 cs.push(new ExpStatement(ss.loc, tmp));
3618 
3619                 auto args = new Parameters();
3620                 args.push(new Parameter(0, ClassDeclaration.object.type, null, null, null));
3621 
3622                 FuncDeclaration fdenter = FuncDeclaration.genCfunc(args, Type.tvoid, Id.monitorenter);
3623                 Expression e = new CallExp(ss.loc, fdenter, new VarExp(ss.loc, tmp));
3624                 e.type = Type.tvoid; // do not run semantic on e
3625 
3626                 cs.push(new ExpStatement(ss.loc, e));
3627                 FuncDeclaration fdexit = FuncDeclaration.genCfunc(args, Type.tvoid, Id.monitorexit);
3628                 e = new CallExp(ss.loc, fdexit, new VarExp(ss.loc, tmp));
3629                 e.type = Type.tvoid; // do not run semantic on e
3630                 Statement s = new ExpStatement(ss.loc, e);
3631                 s = new TryFinallyStatement(ss.loc, ss._body, s);
3632                 cs.push(s);
3633 
3634                 s = new CompoundStatement(ss.loc, cs);
3635                 result = s.statementSemantic(sc);
3636             }
3637         }
3638         else
3639         {
3640             /* Generate our own critical section, then rewrite as:
3641              *  static shared align(D_CRITICAL_SECTION.alignof) byte[D_CRITICAL_SECTION.sizeof] __critsec;
3642              *  _d_criticalenter(&__critsec[0]);
3643              *  try { body } finally { _d_criticalexit(&__critsec[0]); }
3644              */
3645             auto id = Identifier.generateId("__critsec");
3646             auto t = Type.tint8.sarrayOf(target.ptrsize + target.critsecsize());
3647             auto tmp = new VarDeclaration(ss.loc, t, id, null);
3648             tmp.storage_class |= STC.temp | STC.shared_ | STC.static_;
3649             Expression tmpExp = new VarExp(ss.loc, tmp);
3650 
3651             auto cs = new Statements();
3652             cs.push(new ExpStatement(ss.loc, tmp));
3653 
3654             /* This is just a dummy variable for "goto skips declaration" error.
3655              * Backend optimizer could remove this unused variable.
3656              */
3657             auto v = new VarDeclaration(ss.loc, Type.tvoidptr, Identifier.generateId("__sync"), null);
3658             v.dsymbolSemantic(sc);
3659             cs.push(new ExpStatement(ss.loc, v));
3660 
3661             auto args = new Parameters();
3662             args.push(new Parameter(0, t.pointerTo(), null, null, null));
3663 
3664             FuncDeclaration fdenter = FuncDeclaration.genCfunc(args, Type.tvoid, Id.criticalenter, STC.nothrow_);
3665             Expression int0 = new IntegerExp(ss.loc, dinteger_t(0), Type.tint8);
3666             Expression e = new AddrExp(ss.loc, new IndexExp(ss.loc, tmpExp, int0));
3667             e = e.expressionSemantic(sc);
3668             e = new CallExp(ss.loc, fdenter, e);
3669             e.type = Type.tvoid; // do not run semantic on e
3670             cs.push(new ExpStatement(ss.loc, e));
3671 
3672             FuncDeclaration fdexit = FuncDeclaration.genCfunc(args, Type.tvoid, Id.criticalexit, STC.nothrow_);
3673             e = new AddrExp(ss.loc, new IndexExp(ss.loc, tmpExp, int0));
3674             e = e.expressionSemantic(sc);
3675             e = new CallExp(ss.loc, fdexit, e);
3676             e.type = Type.tvoid; // do not run semantic on e
3677             Statement s = new ExpStatement(ss.loc, e);
3678             s = new TryFinallyStatement(ss.loc, ss._body, s);
3679             cs.push(s);
3680 
3681             s = new CompoundStatement(ss.loc, cs);
3682             result = s.statementSemantic(sc);
3683 
3684             // set the explicit __critsec alignment after semantic()
3685             tmp.alignment = target.ptrsize;
3686         }
3687     }
3688 
3689     override void visit(WithStatement ws)
3690     {
3691         /* https://dlang.org/spec/statement.html#with-statement
3692          */
3693 
3694         ScopeDsymbol sym;
3695         Initializer _init;
3696 
3697         //printf("WithStatement::semantic()\n");
3698         ws.exp = ws.exp.expressionSemantic(sc);
3699         ws.exp = resolveProperties(sc, ws.exp);
3700         ws.exp = ws.exp.optimize(WANTvalue);
3701         ws.exp = checkGC(sc, ws.exp);
3702         if (ws.exp.op == TOK.error)
3703             return setError();
3704         if (ws.exp.op == TOK.scope_)
3705         {
3706             sym = new WithScopeSymbol(ws);
3707             sym.parent = sc.scopesym;
3708             sym.endlinnum = ws.endloc.linnum;
3709         }
3710         else if (ws.exp.op == TOK.type)
3711         {
3712             Dsymbol s = (cast(TypeExp)ws.exp).type.toDsymbol(sc);
3713             if (!s || !s.isScopeDsymbol())
3714             {
3715                 ws.error("`with` type `%s` has no members", ws.exp.toChars());
3716                 return setError();
3717             }
3718             sym = new WithScopeSymbol(ws);
3719             sym.parent = sc.scopesym;
3720             sym.endlinnum = ws.endloc.linnum;
3721         }
3722         else
3723         {
3724             Type t = ws.exp.type.toBasetype();
3725 
3726             Expression olde = ws.exp;
3727             if (t.ty == Tpointer)
3728             {
3729                 ws.exp = new PtrExp(ws.loc, ws.exp);
3730                 ws.exp = ws.exp.expressionSemantic(sc);
3731                 t = ws.exp.type.toBasetype();
3732             }
3733 
3734             assert(t);
3735             t = t.toBasetype();
3736             if (t.isClassHandle())
3737             {
3738                 _init = new ExpInitializer(ws.loc, ws.exp);
3739                 ws.wthis = new VarDeclaration(ws.loc, ws.exp.type, Id.withSym, _init);
3740                 ws.wthis.dsymbolSemantic(sc);
3741 
3742                 sym = new WithScopeSymbol(ws);
3743                 sym.parent = sc.scopesym;
3744                 sym.endlinnum = ws.endloc.linnum;
3745             }
3746             else if (t.ty == Tstruct)
3747             {
3748                 if (!ws.exp.isLvalue())
3749                 {
3750                     /* Re-write to
3751                      * {
3752                      *   auto __withtmp = exp
3753                      *   with(__withtmp)
3754                      *   {
3755                      *     ...
3756                      *   }
3757                      * }
3758                      */
3759                     auto tmp = copyToTemp(0, "__withtmp", ws.exp);
3760                     tmp.dsymbolSemantic(sc);
3761                     auto es = new ExpStatement(ws.loc, tmp);
3762                     ws.exp = new VarExp(ws.loc, tmp);
3763                     Statement ss = new ScopeStatement(ws.loc, new CompoundStatement(ws.loc, es, ws), ws.endloc);
3764                     result = ss.statementSemantic(sc);
3765                     return;
3766                 }
3767                 Expression e = ws.exp.addressOf();
3768                 _init = new ExpInitializer(ws.loc, e);
3769                 ws.wthis = new VarDeclaration(ws.loc, e.type, Id.withSym, _init);
3770                 ws.wthis.dsymbolSemantic(sc);
3771                 sym = new WithScopeSymbol(ws);
3772                 // Need to set the scope to make use of resolveAliasThis
3773                 sym.setScope(sc);
3774                 sym.parent = sc.scopesym;
3775                 sym.endlinnum = ws.endloc.linnum;
3776             }
3777             else
3778             {
3779                 ws.error("`with` expressions must be aggregate types or pointers to them, not `%s`", olde.type.toChars());
3780                 return setError();
3781             }
3782         }
3783 
3784         if (ws._body)
3785         {
3786             sym._scope = sc;
3787             sc = sc.push(sym);
3788             sc.insert(sym);
3789             ws._body = ws._body.statementSemantic(sc);
3790             sc.pop();
3791             if (ws._body && ws._body.isErrorStatement())
3792             {
3793                 result = ws._body;
3794                 return;
3795             }
3796         }
3797 
3798         result = ws;
3799     }
3800 
3801     // https://dlang.org/spec/statement.html#TryStatement
3802     override void visit(TryCatchStatement tcs)
3803     {
3804         //printf("TryCatchStatement.semantic()\n");
3805 
3806         if (!global.params.useExceptions)
3807         {
3808             tcs.error("Cannot use try-catch statements with -betterC");
3809             return setError();
3810         }
3811 
3812         if (!ClassDeclaration.throwable)
3813         {
3814             tcs.error("Cannot use try-catch statements because `object.Throwable` was not declared");
3815             return setError();
3816         }
3817 
3818         uint flags;
3819         enum FLAGcpp = 1;
3820         enum FLAGd = 2;
3821 
3822         tcs.tryBody = sc.tryBody;
3823 
3824         scope sc2 = sc.push();
3825         sc2.tryBody = tcs;
3826         tcs._body = tcs._body.semanticScope(sc, null, null);
3827         assert(tcs._body);
3828         sc2.pop();
3829 
3830         /* Even if body is empty, still do semantic analysis on catches
3831          */
3832         bool catchErrors = false;
3833         foreach (i, c; *tcs.catches)
3834         {
3835             c.catchSemantic(sc);
3836             if (c.errors)
3837             {
3838                 catchErrors = true;
3839                 continue;
3840             }
3841             auto cd = c.type.toBasetype().isClassHandle();
3842             flags |= cd.isCPPclass() ? FLAGcpp : FLAGd;
3843 
3844             // Determine if current catch 'hides' any previous catches
3845             foreach (j; 0 .. i)
3846             {
3847                 Catch cj = (*tcs.catches)[j];
3848                 const si = c.loc.toChars();
3849                 const sj = cj.loc.toChars();
3850                 if (c.type.toBasetype().implicitConvTo(cj.type.toBasetype()))
3851                 {
3852                     tcs.error("`catch` at %s hides `catch` at %s", sj, si);
3853                     catchErrors = true;
3854                 }
3855             }
3856         }
3857 
3858         if (sc.func)
3859         {
3860             sc.func.flags |= FUNCFLAG.hasCatches;
3861             if (flags == (FLAGcpp | FLAGd))
3862             {
3863                 tcs.error("cannot mix catching D and C++ exceptions in the same try-catch");
3864                 catchErrors = true;
3865             }
3866         }
3867 
3868         if (catchErrors)
3869             return setError();
3870 
3871         if (tcs._body.isErrorStatement())
3872         {
3873             result = tcs._body;
3874             return;
3875         }
3876 
3877         /* If the try body never throws, we can eliminate any catches
3878          * of recoverable exceptions.
3879          */
3880         if (!(tcs._body.blockExit(sc.func, false) & BE.throw_) && ClassDeclaration.exception)
3881         {
3882             foreach_reverse (i; 0 .. tcs.catches.dim)
3883             {
3884                 Catch c = (*tcs.catches)[i];
3885 
3886                 /* If catch exception type is derived from Exception
3887                  */
3888                 if (c.type.toBasetype().implicitConvTo(ClassDeclaration.exception.type) &&
3889                     (!c.handler || !c.handler.comeFrom()))
3890                 {
3891                     // Remove c from the array of catches
3892                     tcs.catches.remove(i);
3893                 }
3894             }
3895         }
3896 
3897         if (tcs.catches.dim == 0)
3898         {
3899             result = tcs._body.hasCode() ? tcs._body : null;
3900             return;
3901         }
3902 
3903         result = tcs;
3904     }
3905 
3906     override void visit(TryFinallyStatement tfs)
3907     {
3908         //printf("TryFinallyStatement::semantic()\n");
3909         tfs.tryBody = sc.tryBody;
3910 
3911         auto sc2 = sc.push();
3912         sc.tryBody = tfs;
3913         tfs._body = tfs._body.statementSemantic(sc);
3914         sc2.pop();
3915 
3916         sc = sc.push();
3917         sc.tf = tfs;
3918         sc.sbreak = null;
3919         sc.scontinue = null; // no break or continue out of finally block
3920         tfs.finalbody = tfs.finalbody.semanticNoScope(sc);
3921         sc.pop();
3922 
3923         if (!tfs._body)
3924         {
3925             result = tfs.finalbody;
3926             return;
3927         }
3928         if (!tfs.finalbody)
3929         {
3930             result = tfs._body;
3931             return;
3932         }
3933 
3934         auto blockexit = tfs._body.blockExit(sc.func, false);
3935 
3936         // if not worrying about exceptions
3937         if (!(global.params.useExceptions && ClassDeclaration.throwable))
3938             blockexit &= ~BE.throw_;            // don't worry about paths that otherwise may throw
3939 
3940         // Don't care about paths that halt, either
3941         if ((blockexit & ~BE.halt) == BE.fallthru)
3942         {
3943             result = new CompoundStatement(tfs.loc, tfs._body, tfs.finalbody);
3944             return;
3945         }
3946         tfs.bodyFallsThru = (blockexit & BE.fallthru) != 0;
3947         result = tfs;
3948     }
3949 
3950     override void visit(ScopeGuardStatement oss)
3951     {
3952         /* https://dlang.org/spec/statement.html#scope-guard-statement
3953          */
3954 
3955         if (oss.tok != TOK.onScopeExit)
3956         {
3957             // scope(success) and scope(failure) are rewritten to try-catch(-finally) statement,
3958             // so the generated catch block cannot be placed in finally block.
3959             // See also Catch::semantic.
3960             if (sc.os && sc.os.tok != TOK.onScopeFailure)
3961             {
3962                 // If enclosing is scope(success) or scope(exit), this will be placed in finally block.
3963                 oss.error("cannot put `%s` statement inside `%s`", Token.toChars(oss.tok), Token.toChars(sc.os.tok));
3964                 return setError();
3965             }
3966             if (sc.tf)
3967             {
3968                 oss.error("cannot put `%s` statement inside `finally` block", Token.toChars(oss.tok));
3969                 return setError();
3970             }
3971         }
3972 
3973         sc = sc.push();
3974         sc.tf = null;
3975         sc.os = oss;
3976         if (oss.tok != TOK.onScopeFailure)
3977         {
3978             // Jump out from scope(failure) block is allowed.
3979             sc.sbreak = null;
3980             sc.scontinue = null;
3981         }
3982         oss.statement = oss.statement.semanticNoScope(sc);
3983         sc.pop();
3984 
3985         if (!oss.statement || oss.statement.isErrorStatement())
3986         {
3987             result = oss.statement;
3988             return;
3989         }
3990         result = oss;
3991     }
3992 
3993     override void visit(ThrowStatement ts)
3994     {
3995         /* https://dlang.org/spec/statement.html#throw-statement
3996          */
3997 
3998         //printf("ThrowStatement::semantic()\n");
3999 
4000         if (!global.params.useExceptions)
4001         {
4002             ts.error("Cannot use `throw` statements with -betterC");
4003             return setError();
4004         }
4005 
4006         if (!ClassDeclaration.throwable)
4007         {
4008             ts.error("Cannot use `throw` statements because `object.Throwable` was not declared");
4009             return setError();
4010         }
4011 
4012         FuncDeclaration fd = sc.parent.isFuncDeclaration();
4013         fd.hasReturnExp |= 2;
4014 
4015         if (ts.exp.op == TOK.new_)
4016         {
4017             NewExp ne = cast(NewExp)ts.exp;
4018             ne.thrownew = true;
4019         }
4020 
4021         ts.exp = ts.exp.expressionSemantic(sc);
4022         ts.exp = resolveProperties(sc, ts.exp);
4023         ts.exp = checkGC(sc, ts.exp);
4024         if (ts.exp.op == TOK.error)
4025             return setError();
4026 
4027         checkThrowEscape(sc, ts.exp, false);
4028 
4029         ClassDeclaration cd = ts.exp.type.toBasetype().isClassHandle();
4030         if (!cd || ((cd != ClassDeclaration.throwable) && !ClassDeclaration.throwable.isBaseOf(cd, null)))
4031         {
4032             ts.error("can only throw class objects derived from `Throwable`, not type `%s`", ts.exp.type.toChars());
4033             return setError();
4034         }
4035 
4036         result = ts;
4037     }
4038 
4039     override void visit(DebugStatement ds)
4040     {
4041         if (ds.statement)
4042         {
4043             sc = sc.push();
4044             sc.flags |= SCOPE.debug_;
4045             ds.statement = ds.statement.statementSemantic(sc);
4046             sc.pop();
4047         }
4048         result = ds.statement;
4049     }
4050 
4051     override void visit(GotoStatement gs)
4052     {
4053         /* https://dlang.org/spec/statement.html#goto-statement
4054          */
4055 
4056         //printf("GotoStatement::semantic()\n");
4057         FuncDeclaration fd = sc.func;
4058 
4059         gs.ident = fixupLabelName(sc, gs.ident);
4060         gs.label = fd.searchLabel(gs.ident);
4061         gs.tryBody = sc.tryBody;
4062         gs.tf = sc.tf;
4063         gs.os = sc.os;
4064         gs.lastVar = sc.lastVar;
4065 
4066         if (!gs.label.statement && sc.fes)
4067         {
4068             /* Either the goto label is forward referenced or it
4069              * is in the function that the enclosing foreach is in.
4070              * Can't know yet, so wrap the goto in a scope statement
4071              * so we can patch it later, and add it to a 'look at this later'
4072              * list.
4073              */
4074             gs.label.deleted = true;
4075             auto ss = new ScopeStatement(gs.loc, gs, gs.loc);
4076             sc.fes.gotos.push(ss); // 'look at this later' list
4077             result = ss;
4078             return;
4079         }
4080 
4081         // Add to fwdref list to check later
4082         if (!gs.label.statement)
4083         {
4084             if (!fd.gotos)
4085                 fd.gotos = new GotoStatements();
4086             fd.gotos.push(gs);
4087         }
4088         else if (gs.checkLabel())
4089             return setError();
4090 
4091         result = gs;
4092     }
4093 
4094     override void visit(LabelStatement ls)
4095     {
4096         //printf("LabelStatement::semantic()\n");
4097         FuncDeclaration fd = sc.parent.isFuncDeclaration();
4098 
4099         ls.ident = fixupLabelName(sc, ls.ident);
4100         ls.tryBody = sc.tryBody;
4101         ls.tf = sc.tf;
4102         ls.os = sc.os;
4103         ls.lastVar = sc.lastVar;
4104 
4105         LabelDsymbol ls2 = fd.searchLabel(ls.ident);
4106         if (ls2.statement)
4107         {
4108             ls.error("label `%s` already defined", ls2.toChars());
4109             return setError();
4110         }
4111         else
4112             ls2.statement = ls;
4113 
4114         sc = sc.push();
4115         sc.scopesym = sc.enclosing.scopesym;
4116 
4117         sc.ctorflow.orCSX(CSX.label);
4118 
4119         sc.slabel = ls;
4120         if (ls.statement)
4121             ls.statement = ls.statement.statementSemantic(sc);
4122         sc.pop();
4123 
4124         result = ls;
4125     }
4126 
4127     override void visit(AsmStatement s)
4128     {
4129         /* https://dlang.org/spec/statement.html#asm
4130          */
4131 
4132         result = asmSemantic(s, sc);
4133     }
4134 
4135     override void visit(CompoundAsmStatement cas)
4136     {
4137         // Apply postfix attributes of the asm block to each statement.
4138         sc = sc.push();
4139         sc.stc |= cas.stc;
4140         foreach (ref s; *cas.statements)
4141         {
4142             s = s ? s.statementSemantic(sc) : null;
4143         }
4144 
4145         assert(sc.func);
4146         // use setImpure/setGC when the deprecation cycle is over
4147         PURE purity;
4148         if (!(cas.stc & STC.pure_) && (purity = sc.func.isPureBypassingInference()) != PURE.impure && purity != PURE.fwdref)
4149             cas.deprecation("`asm` statement is assumed to be impure - mark it with `pure` if it is not");
4150         if (!(cas.stc & STC.nogc) && sc.func.isNogcBypassingInference())
4151             cas.deprecation("`asm` statement is assumed to use the GC - mark it with `@nogc` if it does not");
4152         if (!(cas.stc & (STC.trusted | STC.safe)) && sc.func.setUnsafe())
4153             cas.error("`asm` statement is assumed to be `@system` - mark it with `@trusted` if it is not");
4154 
4155         sc.pop();
4156         result = cas;
4157     }
4158 
4159     override void visit(ImportStatement imps)
4160     {
4161         /* https://dlang.org/spec/module.html#ImportDeclaration
4162          */
4163 
4164         foreach (i; 0 .. imps.imports.dim)
4165         {
4166             Import s = (*imps.imports)[i].isImport();
4167             assert(!s.aliasdecls.dim);
4168             foreach (j, name; s.names)
4169             {
4170                 Identifier _alias = s.aliases[j];
4171                 if (!_alias)
4172                     _alias = name;
4173 
4174                 auto tname = new TypeIdentifier(s.loc, name);
4175                 auto ad = new AliasDeclaration(s.loc, _alias, tname);
4176                 ad._import = s;
4177                 s.aliasdecls.push(ad);
4178             }
4179 
4180             s.dsymbolSemantic(sc);
4181 
4182             // https://issues.dlang.org/show_bug.cgi?id=19942
4183             // If the module that's being imported doesn't exist, don't add it to the symbol table
4184             // for the current scope.
4185             if (s.mod !is null)
4186             {
4187                 Module.addDeferredSemantic2(s);     // https://issues.dlang.org/show_bug.cgi?id=14666
4188                 sc.insert(s);
4189 
4190                 foreach (aliasdecl; s.aliasdecls)
4191                 {
4192                     sc.insert(aliasdecl);
4193                 }
4194             }
4195         }
4196         result = imps;
4197     }
4198 }
4199 
4200 void catchSemantic(Catch c, Scope* sc)
4201 {
4202     //printf("Catch::semantic(%s)\n", ident.toChars());
4203 
4204     if (sc.os && sc.os.tok != TOK.onScopeFailure)
4205     {
4206         // If enclosing is scope(success) or scope(exit), this will be placed in finally block.
4207         error(c.loc, "cannot put `catch` statement inside `%s`", Token.toChars(sc.os.tok));
4208         c.errors = true;
4209     }
4210     if (sc.tf)
4211     {
4212         /* This is because the _d_local_unwind() gets the stack munged
4213          * up on this. The workaround is to place any try-catches into
4214          * a separate function, and call that.
4215          * To fix, have the compiler automatically convert the finally
4216          * body into a nested function.
4217          */
4218         error(c.loc, "cannot put `catch` statement inside `finally` block");
4219         c.errors = true;
4220     }
4221 
4222     auto sym = new ScopeDsymbol();
4223     sym.parent = sc.scopesym;
4224     sc = sc.push(sym);
4225 
4226     if (!c.type)
4227     {
4228         error(c.loc, "`catch` statement without an exception specification is deprecated");
4229         errorSupplemental(c.loc, "use `catch(Throwable)` for old behavior");
4230         c.errors = true;
4231 
4232         // reference .object.Throwable
4233         c.type = getThrowable();
4234     }
4235     c.type = c.type.typeSemantic(c.loc, sc);
4236     if (c.type == Type.terror)
4237         c.errors = true;
4238     else
4239     {
4240         StorageClass stc;
4241         auto cd = c.type.toBasetype().isClassHandle();
4242         if (!cd)
4243         {
4244             error(c.loc, "can only catch class objects, not `%s`", c.type.toChars());
4245             c.errors = true;
4246         }
4247         else if (cd.isCPPclass())
4248         {
4249             if (!target.cpp.exceptions)
4250             {
4251                 error(c.loc, "catching C++ class objects not supported for this target");
4252                 c.errors = true;
4253             }
4254             if (sc.func && !sc.intypeof && !c.internalCatch && sc.func.setUnsafe())
4255             {
4256                 error(c.loc, "cannot catch C++ class objects in `@safe` code");
4257                 c.errors = true;
4258             }
4259         }
4260         else if (cd != ClassDeclaration.throwable && !ClassDeclaration.throwable.isBaseOf(cd, null))
4261         {
4262             error(c.loc, "can only catch class objects derived from `Throwable`, not `%s`", c.type.toChars());
4263             c.errors = true;
4264         }
4265         else if (sc.func && !sc.intypeof && !c.internalCatch && ClassDeclaration.exception &&
4266                  cd != ClassDeclaration.exception && !ClassDeclaration.exception.isBaseOf(cd, null) &&
4267                  sc.func.setUnsafe())
4268         {
4269             error(c.loc, "can only catch class objects derived from `Exception` in `@safe` code, not `%s`", c.type.toChars());
4270             c.errors = true;
4271         }
4272         else if (global.params.ehnogc)
4273         {
4274             stc |= STC.scope_;
4275         }
4276 
4277         // DIP1008 requires destruction of the Throwable, even if the user didn't specify an identifier
4278         auto ident = c.ident;
4279         if (!ident && global.params.ehnogc)
4280             ident = Identifier.anonymous();
4281 
4282         if (ident)
4283         {
4284             c.var = new VarDeclaration(c.loc, c.type, ident, null, stc);
4285             c.var.iscatchvar = true;
4286             c.var.dsymbolSemantic(sc);
4287             sc.insert(c.var);
4288 
4289             if (global.params.ehnogc && stc & STC.scope_)
4290             {
4291                 /* Add a destructor for c.var
4292                  * try { handler } finally { if (!__ctfe) _d_delThrowable(var); }
4293                  */
4294                 assert(!c.var.edtor);           // ensure we didn't create one in callScopeDtor()
4295 
4296                 Loc loc = c.loc;
4297                 Expression e = new VarExp(loc, c.var);
4298                 e = new CallExp(loc, new IdentifierExp(loc, Id._d_delThrowable), e);
4299 
4300                 Expression ec = new IdentifierExp(loc, Id.ctfe);
4301                 ec = new NotExp(loc, ec);
4302                 Statement s = new IfStatement(loc, null, ec, new ExpStatement(loc, e), null, loc);
4303                 c.handler = new TryFinallyStatement(loc, c.handler, s);
4304             }
4305 
4306         }
4307         c.handler = c.handler.statementSemantic(sc);
4308         if (c.handler && c.handler.isErrorStatement())
4309             c.errors = true;
4310     }
4311 
4312     sc.pop();
4313 }
4314 
4315 Statement semanticNoScope(Statement s, Scope* sc)
4316 {
4317     //printf("Statement::semanticNoScope() %s\n", toChars());
4318     if (!s.isCompoundStatement() && !s.isScopeStatement())
4319     {
4320         s = new CompoundStatement(s.loc, s); // so scopeCode() gets called
4321     }
4322     s = s.statementSemantic(sc);
4323     return s;
4324 }
4325 
4326 // Same as semanticNoScope(), but do create a new scope
4327 Statement semanticScope(Statement s, Scope* sc, Statement sbreak, Statement scontinue)
4328 {
4329     auto sym = new ScopeDsymbol();
4330     sym.parent = sc.scopesym;
4331     Scope* scd = sc.push(sym);
4332     if (sbreak)
4333         scd.sbreak = sbreak;
4334     if (scontinue)
4335         scd.scontinue = scontinue;
4336     s = s.semanticNoScope(scd);
4337     scd.pop();
4338     return s;
4339 }
4340 
4341 
4342 /*******************
4343  * Determines additional argument types for makeTupleForeach.
4344  */
4345 static template TupleForeachArgs(bool isStatic, bool isDecl)
4346 {
4347     alias Seq(T...)=T;
4348     static if(isStatic) alias T = Seq!(bool);
4349     else alias T = Seq!();
4350     static if(!isDecl) alias TupleForeachArgs = T;
4351     else alias TupleForeachArgs = Seq!(Dsymbols*,T);
4352 }
4353 
4354 /*******************
4355  * Determines the return type of makeTupleForeach.
4356  */
4357 static template TupleForeachRet(bool isStatic, bool isDecl)
4358 {
4359     alias Seq(T...)=T;
4360     static if(!isDecl) alias TupleForeachRet = Statement;
4361     else alias TupleForeachRet = Dsymbols*;
4362 }
4363 
4364 
4365 /*******************
4366  * See StatementSemanticVisitor.makeTupleForeach.  This is a simple
4367  * wrapper that returns the generated statements/declarations.
4368  */
4369 TupleForeachRet!(isStatic, isDecl) makeTupleForeach(bool isStatic, bool isDecl)(Scope* sc, ForeachStatement fs, TupleForeachArgs!(isStatic, isDecl) args)
4370 {
4371     scope v = new StatementSemanticVisitor(sc);
4372     static if(!isDecl)
4373     {
4374         v.makeTupleForeach!(isStatic, isDecl)(fs, args);
4375         return v.result;
4376     }
4377     else
4378     {
4379         return v.makeTupleForeach!(isStatic, isDecl)(fs, args);
4380     }
4381 }