1 /**
2  * Checks that a function marked `@nogc` does not invoke the Garbage Collector.
3  *
4  * Specification: $(LINK2 https://dlang.org/spec/function.html#nogc-functions, No-GC Functions)
5  *
6  * Copyright:   Copyright (C) 1999-2021 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/nogc.d, _nogc.d)
10  * Documentation:  https://dlang.org/phobos/dmd_nogc.html
11  * Coverage:    https://codecov.io/gh/dlang/dmd/src/master/src/dmd/nogc.d
12  */
13 
14 module dmd.nogc;
15 
16 import dmd.aggregate;
17 import dmd.apply;
18 import dmd.declaration;
19 import dmd.dscope;
20 import dmd.expression;
21 import dmd.func;
22 import dmd.globals;
23 import dmd.init;
24 import dmd.mtype;
25 import dmd.tokens;
26 import dmd.visitor;
27 
28 /**************************************
29  * Look for GC-allocations
30  */
31 extern (C++) final class NOGCVisitor : StoppableVisitor
32 {
33     alias visit = typeof(super).visit;
34 public:
35     FuncDeclaration f;
36     bool err;
37 
38     extern (D) this(FuncDeclaration f)
39     {
40         this.f = f;
41     }
42 
43     void doCond(Expression exp)
44     {
45         if (exp)
46             walkPostorder(exp, this);
47     }
48 
49     override void visit(Expression e)
50     {
51     }
52 
53     override void visit(DeclarationExp e)
54     {
55         // Note that, walkPostorder does not support DeclarationExp today.
56         VarDeclaration v = e.declaration.isVarDeclaration();
57         if (v && !(v.storage_class & STC.manifest) && !v.isDataseg() && v._init)
58         {
59             if (ExpInitializer ei = v._init.isExpInitializer())
60             {
61                 doCond(ei.exp);
62             }
63         }
64     }
65 
66     override void visit(CallExp e)
67     {
68         import dmd.id : Id;
69         import core.stdc.stdio : printf;
70         if (!e.f)
71             return;
72 
73         auto fd = stripHookTraceImpl(e.f);
74         if (fd.ident == Id._d_arraysetlengthT)
75         {
76             if (f.setGC())
77             {
78                 e.error("setting `length` in `@nogc` %s `%s` may cause a GC allocation",
79                     f.kind(), f.toPrettyChars());
80                 err = true;
81                 return;
82             }
83             f.printGCUsage(e.loc, "setting `length` may cause a GC allocation");
84         }
85     }
86 
87     override void visit(ArrayLiteralExp e)
88     {
89         if (e.type.ty != Tarray || !e.elements || !e.elements.dim)
90             return;
91         if (f.setGC())
92         {
93             e.error("array literal in `@nogc` %s `%s` may cause a GC allocation",
94                 f.kind(), f.toPrettyChars());
95             err = true;
96             return;
97         }
98         f.printGCUsage(e.loc, "array literal may cause a GC allocation");
99     }
100 
101     override void visit(AssocArrayLiteralExp e)
102     {
103         if (!e.keys.dim)
104             return;
105         if (f.setGC())
106         {
107             e.error("associative array literal in `@nogc` %s `%s` may cause a GC allocation",
108                 f.kind(), f.toPrettyChars());
109             err = true;
110             return;
111         }
112         f.printGCUsage(e.loc, "associative array literal may cause a GC allocation");
113     }
114 
115     override void visit(NewExp e)
116     {
117         if (e.member && !e.member.isNogc() && f.setGC())
118         {
119             // @nogc-ness is already checked in NewExp::semantic
120             return;
121         }
122         if (e.onstack)
123             return;
124         if (e.allocator)
125             return;
126         if (global.params.ehnogc && e.thrownew)
127             return;                     // separate allocator is called for this, not the GC
128         if (f.setGC())
129         {
130             e.error("cannot use `new` in `@nogc` %s `%s`",
131                 f.kind(), f.toPrettyChars());
132             err = true;
133             return;
134         }
135         f.printGCUsage(e.loc, "`new` causes a GC allocation");
136     }
137 
138     override void visit(DeleteExp e)
139     {
140         if (e.e1.op == TOK.variable)
141         {
142             VarDeclaration v = (cast(VarExp)e.e1).var.isVarDeclaration();
143             if (v && v.onstack)
144                 return; // delete for scope allocated class object
145         }
146 
147         Type tb = e.e1.type.toBasetype();
148         AggregateDeclaration ad = null;
149         switch (tb.ty)
150         {
151         case Tclass:
152             ad = (cast(TypeClass)tb).sym;
153             break;
154 
155         case Tpointer:
156             tb = (cast(TypePointer)tb).next.toBasetype();
157             if (tb.ty == Tstruct)
158                 ad = (cast(TypeStruct)tb).sym;
159             break;
160 
161         default:
162             break;
163         }
164 
165         if (f.setGC())
166         {
167             e.error("cannot use `delete` in `@nogc` %s `%s`",
168                 f.kind(), f.toPrettyChars());
169             err = true;
170             return;
171         }
172         f.printGCUsage(e.loc, "`delete` requires the GC");
173     }
174 
175     override void visit(IndexExp e)
176     {
177         Type t1b = e.e1.type.toBasetype();
178         if (t1b.ty == Taarray)
179         {
180             if (f.setGC())
181             {
182                 e.error("indexing an associative array in `@nogc` %s `%s` may cause a GC allocation",
183                     f.kind(), f.toPrettyChars());
184                 err = true;
185                 return;
186             }
187             f.printGCUsage(e.loc, "indexing an associative array may cause a GC allocation");
188         }
189     }
190 
191     override void visit(AssignExp e)
192     {
193         if (e.e1.op == TOK.arrayLength)
194         {
195             if (f.setGC())
196             {
197                 e.error("setting `length` in `@nogc` %s `%s` may cause a GC allocation",
198                     f.kind(), f.toPrettyChars());
199                 err = true;
200                 return;
201             }
202             f.printGCUsage(e.loc, "setting `length` may cause a GC allocation");
203         }
204     }
205 
206     override void visit(CatAssignExp e)
207     {
208         if (f.setGC())
209         {
210             e.error("cannot use operator `~=` in `@nogc` %s `%s`",
211                 f.kind(), f.toPrettyChars());
212             err = true;
213             return;
214         }
215         f.printGCUsage(e.loc, "operator `~=` may cause a GC allocation");
216     }
217 
218     override void visit(CatExp e)
219     {
220         if (f.setGC())
221         {
222             e.error("cannot use operator `~` in `@nogc` %s `%s`",
223                 f.kind(), f.toPrettyChars());
224             err = true;
225             return;
226         }
227         f.printGCUsage(e.loc, "operator `~` may cause a GC allocation");
228     }
229 }
230 
231 Expression checkGC(Scope* sc, Expression e)
232 {
233     FuncDeclaration f = sc.func;
234     if (e && e.op != TOK.error && f && sc.intypeof != 1 && !(sc.flags & SCOPE.ctfe) &&
235            (f.type.ty == Tfunction &&
236             (cast(TypeFunction)f.type).isnogc || (f.flags & FUNCFLAG.nogcInprocess) || global.params.vgc) &&
237            !(sc.flags & SCOPE.debug_))
238     {
239         scope NOGCVisitor gcv = new NOGCVisitor(f);
240         walkPostorder(e, gcv);
241         if (gcv.err)
242             return ErrorExp.get();
243     }
244     return e;
245 }
246 
247 /**
248  * Removes `_d_HookTraceImpl` if found from `fd`.
249  * This is needed to be able to find hooks that are called though the hook's `*Trace` wrapper.
250  * Parameters:
251  *  fd = The function declaration to remove `_d_HookTraceImpl` from
252  */
253 private FuncDeclaration stripHookTraceImpl(FuncDeclaration fd)
254 {
255     import dmd.id : Id;
256     import dmd.dsymbol : Dsymbol;
257     import dmd.root.rootobject : RootObject, DYNCAST;
258 
259     if (fd.ident != Id._d_HookTraceImpl)
260         return fd;
261 
262     // Get the Hook from the second template parameter
263     auto templateInstance = fd.parent.isTemplateInstance;
264     RootObject hook = (*templateInstance.tiargs)[1];
265     assert(hook.dyncast() == DYNCAST.dsymbol, "Expected _d_HookTraceImpl's second template parameter to be an alias to the hook!");
266     return (cast(Dsymbol)hook).isFuncDeclaration;
267 }