This module contains the second semantic checking pass over the AST. Necessary because the old way had some inherent problems. Performs:
- effect+exception tracking
- "usage before definition" checking
- checking of captured entities
- detection of whether a procedure captures something and, if necessary, adjusting the calling convention
- also now calls the "lift destructor logic" at strategic positions, this is about to be put into the spec:
We treat assignment and sinks and destruction as identical.
In the construct let/var x = expr() x's type is marked.
In x = y the type of x is marked.
For every sink parameter of type T T is marked.
For every call f() the return type of f() is marked.
Captured entities
If a local is used inside procedure X but the local does not belong to X, the local needs to be captured. Turning the procedure into a top-level procedure with a hidden environment parameter is later performed by the lambdalifting transformation, but semantic analysis already needs to know which procedures require the closure calling convention, as this information is part of the procedure's type.
Imports
-
../ast/ast, ../ast/renderer, ../ast/astalgo, ../ast/types, ../ast/idents, ../ast/wordrecg, ../ast/errorreporting, ../ast/errorhandling, ../ast/lineinfos, ../ast/trees, ../front/options, ../front/msgs, ../modules/modulegraphs, ../modules/magicsys, ../utils/debugutils, ../utils/idioms, varpartitions, typeallowed, guards, semdata, nilcheck, ../ast/reports_sem, ../ast/report_enums, liftdestructors
Procs
proc canCaptureFrom(captor, target: PSym): bool {....raises: [], tags: [].}
-
Tests if the captor routine is allowed to capture a local from target, taking the compile-time/run-time boundary into account:
1) attempting to capture a local defined outside an inner macro from
inside the macro is illegal
2) closing over a local defined inside a compile-time-only routine from a
routine than can also be used at run-time is only valid if the chain of enclosing routines leading up to target are all compile-time-only
- a compile-time-only routine closing over a run-time location is illegal
proc checkForSink(config: ConfigRef; idgen: IdGenerator; owner: PSym; arg: PNode) {. ...raises: [Exception, ERecoverableError], tags: [RootEffect].}
- Source Edit
proc checkMethodEffects(g: ModuleGraph; disp, branch: PSym) {. ...raises: [Exception, ERecoverableError], tags: [RootEffect].}
- checks for consistent effects for multi methods. Source Edit
proc setEffectsForProcType(g: ModuleGraph; t: PType; n: PNode; s: PSym = nil) {. ...raises: [Exception, ERecoverableError], tags: [RootEffect].}
- Source Edit