[MemProf] Templatize CallStackRadixTreeBuilder (NFC) (#117014)
[llvm-project.git] / flang / docs / ImplementingASemanticCheck.md
blob598ef696ad14bf43e400b2c055247bf76c1c200a
1 <!--===- docs/ImplementingASemanticCheck.md 
2   
3    Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
4    See https://llvm.org/LICENSE.txt for license information.
5    SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
6   
7 -->
8 # How to implement a Sematic Check in Flang
10 ```{contents}
11 ---
12 local:
13 ---
14 ```
16 I recently added a semantic check to the Flang compiler front end.  This document
17 describes my thought process and the resulting implementation.
19 For more information about the compiler, start with the 
20 [compiler overview](Overview.md).
22 ## Problem definition
24 In the 2018 Fortran standard, section 11.1.7.4.3, paragraph 2, states that:
26 ```
27 Except for the incrementation of the DO variable that occurs in step (3), the DO variable 
28 shall neither be redefined nor become undefined while the DO construct is active.
29 ```
30 One of the ways that DO variables might be redefined is if they are passed to
31 functions with dummy arguments whose `INTENT` is `INTENT(OUT)` or
32 `INTENT(INOUT)`.  I implemented this semantic check.  Specifically, I changed
33 the compiler to emit an error message if an active DO variable was passed to a
34 dummy argument of a FUNCTION with INTENT(OUT).  Similarly, I had the compiler
35 emit a warning if an active DO variable was passed to a dummy argument with
36 INTENT(INOUT).  Previously, I had implemented similar checks for SUBROUTINE
37 calls.
39 ## Creating a test
41 My first step was to create a test case to cause the problem.  I called it testfun.f90 and used it to check the behavior of other Fortran compilers.  Here's the initial version:
43 ```fortran
44   subroutine s()
45     Integer :: ivar, jvar
47     do ivar = 1, 10
48       jvar = intentOutFunc(ivar) ! Error since ivar is a DO variable
49     end do
51   contains
52     function intentOutFunc(dummyArg)
53       integer, intent(out) :: dummyArg
54       integer  :: intentOutFunc
56       dummyArg = 216
57     end function intentOutFunc
58   end subroutine s
59 ```
61 I verified that other Fortran compilers produced an error message at the point
62 of the call to `intentOutFunc()`:
64 ```fortran
65       jvar = intentOutFunc(ivar) ! Error since ivar is a DO variable
66 ```
69 I also used this program to produce a parse tree for the program using the command:
70 ```bash
71   flang -fc1 -fdebug-dump-parse-tree testfun.f90
72 ```
74 Here's the relevant fragment of the parse tree produced by the compiler:
76 ```
77 | | ExecutionPartConstruct -> ExecutableConstruct -> DoConstruct
78 | | | NonLabelDoStmt
79 | | | | LoopControl -> LoopBounds
80 | | | | | Scalar -> Name = 'ivar'
81 | | | | | Scalar -> Expr = '1_4'
82 | | | | | | LiteralConstant -> IntLiteralConstant = '1'
83 | | | | | Scalar -> Expr = '10_4'
84 | | | | | | LiteralConstant -> IntLiteralConstant = '10'
85 | | | Block
86 | | | | ExecutionPartConstruct -> ExecutableConstruct -> ActionStmt -> AssignmentStmt = 'jvar=intentoutfunc(ivar)'
87 | | | | | Variable -> Designator -> DataRef -> Name = 'jvar'
88 | | | | | Expr = 'intentoutfunc(ivar)'
89 | | | | | | FunctionReference -> Call
90 | | | | | | | ProcedureDesignator -> Name = 'intentoutfunc'
91 | | | | | | | ActualArgSpec
92 | | | | | | | | ActualArg -> Expr = 'ivar'
93 | | | | | | | | | Designator -> DataRef -> Name = 'ivar'
94 | | | EndDoStmt -> 
95 ```
97 Note that this fragment of the tree only shows four `parser::Expr` nodes,
98 but the full parse tree also contained a fifth `parser::Expr` node for the
99 constant 216 in the statement:
101 ```fortran
102       dummyArg = 216
104 ## Analysis and implementation planning
106 I then considered what I needed to do.  I needed to detect situations where an
107 active DO variable was passed to a dummy argument with `INTENT(OUT)` or
108 `INTENT(INOUT)`.  Once I detected such a situation, I needed to produce a
109 message that highlighted the erroneous source code.  
111 ### Deciding where to add the code to the compiler
112 This new semantic check would depend on several types of information -- the
113 parse tree, source code location information, symbols, and expressions.  Thus I
114 needed to put my new code in a place in the compiler after the parse tree had
115 been created, name resolution had already happened, and expression semantic
116 checking had already taken place.
118 Most semantic checks for statements are implemented by walking the parse tree
119 and performing analysis on the nodes they visit.  My plan was to use this
120 method.  The infrastructure for walking the parse tree for statement semantic
121 checking is implemented in the files `lib/Semantics/semantics.cpp`.
122 Here's a fragment of the declaration of the framework's parse tree visitor from
123 `lib/Semantics/semantics.cpp`:
125 ```C++
126   // A parse tree visitor that calls Enter/Leave functions from each checker
127   // class C supplied as template parameters. Enter is called before the node's
128   // children are visited, Leave is called after. No two checkers may have the
129   // same Enter or Leave function. Each checker must be constructible from
130   // SemanticsContext and have BaseChecker as a virtual base class.
131   template<typename... C> class SemanticsVisitor : public virtual C... {
132   public:
133     using C::Enter...;
134     using C::Leave...;
135     using BaseChecker::Enter;
136     using BaseChecker::Leave;
137     SemanticsVisitor(SemanticsContext &context)
138       : C{context}..., context_{context} {}
139       ...
143 Since FUNCTION calls are a kind of expression, I was planning to base my
144 implementation on the contents of `parser::Expr` nodes.  I would need to define
145 either an `Enter()` or `Leave()` function whose parameter was a `parser::Expr`
146 node.  Here's the declaration I put into `lib/Semantics/check-do.h`:
148 ```C++
149   void Leave(const parser::Expr &);
151 The `Enter()` functions get called at the time the node is first visited --
152 that is, before its children.  The `Leave()` function gets called after the
153 children are visited.  For my check the visitation order didn't matter, so I
154 arbitrarily chose to implement the `Leave()` function to visit the parse tree
155 node.
157 Since my semantic check was focused on DO CONCURRENT statements, I added it to
158 the file `lib/Semantics/check-do.cpp` where most of the semantic checking for
159 DO statements already lived.
161 ### Taking advantage of prior work
162 When implementing a similar check for SUBROUTINE calls, I created a utility
163 functions in `lib/Semantics/semantics.cpp` to emit messages if
164 a symbol corresponding to an active DO variable was being potentially modified:
166 ```C++
167   void WarnDoVarRedefine(const parser::CharBlock &location, const Symbol &var);
168   void CheckDoVarRedefine(const parser::CharBlock &location, const Symbol &var);
171 The first function is intended for dummy arguments of `INTENT(INOUT)` and
172 the second for `INTENT(OUT)`.
174 Thus I needed three pieces of
175 information -- 
176 1. the source location of the erroneous text, 
177 2. the `INTENT` of the associated dummy argument, and
178 3. the relevant symbol passed as the actual argument.
180 The first and third are needed since they're required to call the utility
181 functions.  The second is needed to determine whether to call them.
183 ### Finding the source location
184 The source code location information that I'd need for the error message must
185 come from the parse tree.  I looked in the file
186 `include/flang/Parser/parse-tree.h` and determined that a `struct Expr`
187 contained source location information since it had the field `CharBlock
188 source`.  Thus, if I visited a `parser::Expr` node, I could get the source
189 location information for the associated expression.
191 ### Determining the `INTENT`
192 I knew that I could find the `INTENT` of the dummy argument associated with the
193 actual argument from the function called `dummyIntent()` in the class
194 `evaluate::ActualArgument` in the file `include/flang/Evaluate/call.h`.  So
195 if I could find an `evaluate::ActualArgument` in an expression, I could
196   determine the `INTENT` of the associated dummy argument.  I knew that it was
197   valid to call `dummyIntent()` because the data on which `dummyIntent()`
198   depends is established during semantic processing for expressions, and the
199   semantic processing for expressions happens before semantic checking for DO
200   constructs.
202 In my prior work on checking the INTENT of arguments for SUBROUTINE calls,
203 the parse tree held a node for the call (a `parser::CallStmt`) that contained
204 an `evaluate::ProcedureRef` node.
205 ```C++
206   struct CallStmt {
207     WRAPPER_CLASS_BOILERPLATE(CallStmt, Call);
208     mutable std::unique_ptr<evaluate::ProcedureRef,
209         common::Deleter<evaluate::ProcedureRef>>
210         typedCall;  // filled by semantics
211   };
213 The `evaluate::ProcedureRef` contains a list of `evaluate::ActualArgument`
214 nodes.  I could then find the INTENT of a dummy argument from the
215 `evaluate::ActualArgument` node.
217 For a FUNCTION call, though, there is no similar way to get from a parse tree
218 node to an `evaluate::ProcedureRef` node.  But I knew that there was an
219 existing framework used in DO construct semantic checking that traversed an
220 `evaluate::Expr` node collecting `semantics::Symbol` nodes.  I guessed that I'd
221 be able to use a similar framework to traverse an `evaluate::Expr`  node to
222 find all of the `evaluate::ActualArgument` nodes.  
224 Note that the compiler has multiple types called `Expr`.  One is in the
225 `parser` namespace.  `parser::Expr` is defined in the file
226 `include/flang/Parser/parse-tree.h`.  It represents a parsed expression that
227 maps directly to the source code and has fields that specify any operators in
228 the expression, the operands, and the source position of the expression.
230 Additionally, in the namespace `evaluate`, there are `evaluate::Expr<T>`
231 template classes defined in the file `include/flang/Evaluate/expression.h`.
232 These are parameterized over the various types of Fortran and constitute a
233 suite of strongly-typed representations of valid Fortran expressions of type
234 `T` that have been fully elaborated with conversion operations and subjected to
235 constant folding.  After an expression has undergone semantic analysis, the
236 field `typedExpr` in the `parser::Expr` node is filled in with a pointer that
237 owns an instance of `evaluate::Expr<SomeType>`, the most general representation
238 of an analyzed expression.
240 All of the declarations associated with both FUNCTION and SUBROUTINE calls are
241 in `include/flang/Evaluate/call.h`.  An `evaluate::FunctionRef` inherits from
242 an `evaluate::ProcedureRef` which contains the list of
243 `evaluate::ActualArgument` nodes.  But the relationship between an
244 `evaluate::FunctionRef` node and its associated arguments is not relevant.  I
245 only needed to find the `evaluate::ActualArgument` nodes in an expression.
246 They hold all of the information I needed.
248 So my plan was to start with the `parser::Expr` node and extract its
249 associated `evaluate::Expr` field.  I would then traverse the
250 `evaluate::Expr` tree collecting all of the `evaluate::ActualArgument`
251 nodes.  I would look at each of these nodes to determine the `INTENT` of
252 the associated dummy argument.
254 This combination of the traversal framework and `dummyIntent()` would give
255 me the `INTENT` of all of the dummy arguments in a FUNCTION call.  Thus, I
256 would have the second piece of information I needed.
258 ### Determining if the actual argument is a variable
259 I also guessed that I could determine if the `evaluate::ActualArgument`
260 consisted of a variable.  
262 Once I had a symbol for the variable, I could call one of the functions:
263 ```C++
264   void WarnDoVarRedefine(const parser::CharBlock &, const Symbol &);
265   void CheckDoVarRedefine(const parser::CharBlock &, const Symbol &);
267 to emit the messages.
269 If my plans worked out, this would give me the three pieces of information I
270 needed -- the source location of the erroneous text, the `INTENT` of the dummy
271 argument, and a symbol that I could use to determine whether the actual
272 argument was an active DO variable.
274 ## Implementation
276 ### Adding a parse tree visitor
277 I started my implementation by adding a visitor for `parser::Expr` nodes.
278 Since this analysis is part of DO construct checking, I did this in
279 `lib/Semantics/check-do.cpp`.  I added a print statement to the visitor to
280 verify that my new code was actually getting executed.  
282 In `lib/Semantics/check-do.h`, I added the declaration for the visitor:
284 ```C++
285   void Leave(const parser::Expr &);
288 In `lib/Semantics/check-do.cpp`, I added an (almost empty) implementation:
290 ```C++
291   void DoChecker::Leave(const parser::Expr &) {
292     std::cout << "In Leave for parser::Expr\n";
293   }
296 I then built the compiler with these changes and ran it on my test program.
297 This time, I made sure to invoke semantic checking.  Here's the command I used:
298 ```bash
299   flang -fc1 -fdebug-unparse-with-symbols testfun.f90
302 This produced the output:
305   In Leave for parser::Expr
306   In Leave for parser::Expr
307   In Leave for parser::Expr
308   In Leave for parser::Expr
309   In Leave for parser::Expr
312 This made sense since the parse tree contained five `parser::Expr` nodes.
313 So far, so good.  Note that a `parse::Expr` node has a field with the
314 source position of the associated expression (`CharBlock source`).  So I
315 now had one of the three pieces of information needed to detect and report
316 errors.
318 ### Collecting the actual arguments
319 To get the `INTENT` of the dummy arguments and the `semantics::Symbol` associated with the
320 actual argument, I needed to find all of the actual arguments embedded in an
321 expression that contained a FUNCTION call.  So my next step was to write the
322 framework to walk the `evaluate::Expr` to gather all of the
323 `evaluate::ActualArgument` nodes.  The code that I planned to model it on
324 was the existing infrastructure that collected all of the `semantics::Symbol` nodes from an
325 `evaluate::Expr`.  I found this implementation in
326 `lib/Evaluate/tools.cpp`:
328 ```C++
329   struct CollectSymbolsHelper
330     : public SetTraverse<CollectSymbolsHelper, semantics::SymbolSet> {
331     using Base = SetTraverse<CollectSymbolsHelper, semantics::SymbolSet>;
332     CollectSymbolsHelper() : Base{*this} {}
333     using Base::operator();
334     semantics::SymbolSet operator()(const Symbol &symbol) const {
335       return {symbol};
336     }
337   };
338   template<typename A> semantics::SymbolSet CollectSymbols(const A &x) {
339     return CollectSymbolsHelper{}(x);
340   }
343 Note that the `CollectSymbols()` function returns a `semantics::Symbolset`,
344 which is declared in `include/flang/Semantics/symbol.h`:
346 ```C++
347   using SymbolSet = std::set<SymbolRef>;
350 This infrastructure yields a collection based on `std::set<>`.  Using an
351 `std::set<>` means that if the same object is inserted twice, the
352 collection only gets one copy.  This was the behavior that I wanted.
354 Here's a sample invocation of `CollectSymbols()` that I found:
355 ```C++
356     if (const auto *expr{GetExpr(parsedExpr)}) {
357       for (const Symbol &symbol : evaluate::CollectSymbols(*expr)) {
360 I noted that a `SymbolSet` did not actually contain an
361 `std::set<Symbol>`.  This wasn't surprising since we don't want to put the
362 full `semantics::Symbol` objects into the set.  Ideally, we would be able to create an
363 `std::set<Symbol &>` (a set of C++ references to symbols).  But C++ doesn't
364 support sets that contain references.  This limitation is part of the rationale
365 for the Flang implementation of type `common::Reference`, which is defined in
366   `include/flang/Common/reference.h`.
368 `SymbolRef`, the specialization of the template `common::Reference` for
369 `semantics::Symbol`, is declared in the file
370 `include/flang/Semantics/symbol.h`:
372 ```C++
373   using SymbolRef = common::Reference<const Symbol>;
376 So to implement something that would collect `evaluate::ActualArgument`
377 nodes from an `evaluate::Expr`, I first defined the required types
378 `ActualArgumentRef` and `ActualArgumentSet`.  Since these are being
379 used exclusively for DO construct semantic checking (currently), I put their
380 definitions into `lib/Semantics/check-do.cpp`:
383 ```C++
384   namespace Fortran::evaluate {
385     using ActualArgumentRef = common::Reference<const ActualArgument>;
386   }
389   using ActualArgumentSet = std::set<evaluate::ActualArgumentRef>;
392 Since `ActualArgument` is in the namespace `evaluate`, I put the
393 definition for `ActualArgumentRef` in that namespace, too.
395 I then modeled the code to create an `ActualArgumentSet` after the code to
396 collect a `SymbolSet` and put it into `lib/Semantics/check-do.cpp`:
399 ```C++
400   struct CollectActualArgumentsHelper
401     : public evaluate::SetTraverse<CollectActualArgumentsHelper,
402           ActualArgumentSet> {
403     using Base = SetTraverse<CollectActualArgumentsHelper, ActualArgumentSet>;
404     CollectActualArgumentsHelper() : Base{*this} {}
405     using Base::operator();
406     ActualArgumentSet operator()(const evaluate::ActualArgument &arg) const {
407       return ActualArgumentSet{arg};
408     }
409   };
411   template<typename A> ActualArgumentSet CollectActualArguments(const A &x) {
412     return CollectActualArgumentsHelper{}(x);
413   }
415   template ActualArgumentSet CollectActualArguments(const SomeExpr &);
418 Unfortunately, when I tried to build this code, I got an error message saying
419 `std::set` requires the `<` operator to be defined for its contents.
420 To fix this, I added a definition for `<`.  I didn't care how `<` was
421 defined, so I just used the address of the object:
423 ```C++
424   inline bool operator<(ActualArgumentRef x, ActualArgumentRef y) {
425     return &*x < &*y;
426   }
429 I was surprised when this did not make the error message saying that I needed
430 the `<` operator go away.  Eventually, I figured out that the definition of
431 the `<` operator needed to be in the `evaluate` namespace.  Once I put
432 it there, everything compiled successfully.  Here's the code that worked:
434 ```C++
435   namespace Fortran::evaluate {
436   using ActualArgumentRef = common::Reference<const ActualArgument>;
438   inline bool operator<(ActualArgumentRef x, ActualArgumentRef y) {
439     return &*x < &*y;
440   }
441   }
444 I then modified my visitor for the parser::Expr to invoke my new collection
445 framework.  To verify that it was actually doing something, I printed out the
446 number of `evaluate::ActualArgument` nodes that it collected.  Note the
447 call to `GetExpr()` in the invocation of `CollectActualArguments()`.  I
448 modeled this on similar code that collected a `SymbolSet` described above:
450 ```C++
451   void DoChecker::Leave(const parser::Expr &parsedExpr) {
452     std::cout << "In Leave for parser::Expr\n";
453     ActualArgumentSet argSet{CollectActualArguments(GetExpr(parsedExpr))};
454     std::cout << "Number of arguments: " << argSet.size() << "\n";
455   }
458 I compiled and tested this code on my little test program.  Here's the output that I got:
460   In Leave for parser::Expr
461   Number of arguments: 0
462   In Leave for parser::Expr
463   Number of arguments: 0
464   In Leave for parser::Expr
465   Number of arguments: 0
466   In Leave for parser::Expr
467   Number of arguments: 1
468   In Leave for parser::Expr
469   Number of arguments: 0
472 So most of the `parser::Expr`nodes contained no actual arguments, but the
473 fourth expression in the parse tree walk contained a single argument.  This may
474 seem wrong since the third `parser::Expr` node in the file contains the
475 `FunctionReference` node along with the arguments that we're gathering.
476 But since the tree walk function is being called upon leaving a
477 `parser::Expr` node, the function visits the `parser::Expr` node
478 associated with the `parser::ActualArg` node before it visits the
479 `parser::Expr` node associated with the `parser::FunctionReference`
480 node.
482 So far, so good.
484 ### Finding the `INTENT` of the dummy argument
485 I now wanted to find the `INTENT` of the dummy argument associated with the
486 arguments in the set.  As mentioned earlier, the type
487 `evaluate::ActualArgument` has a member function called `dummyIntent()`
488 that gives this value.  So I augmented my code to print out the `INTENT`:
490 ```C++
491   void DoChecker::Leave(const parser::Expr &parsedExpr) {
492     std::cout << "In Leave for parser::Expr\n";
493     ActualArgumentSet argSet{CollectActualArguments(GetExpr(parsedExpr))};
494     std::cout << "Number of arguments: " << argSet.size() << "\n";
495     for (const evaluate::ActualArgumentRef &argRef : argSet) {
496       common::Intent intent{argRef->dummyIntent()};
497       switch (intent) {
498         case common::Intent::In: std::cout << "INTENT(IN)\n"; break;
499         case common::Intent::Out: std::cout << "INTENT(OUT)\n"; break;
500         case common::Intent::InOut: std::cout << "INTENT(INOUT)\n"; break;
501         default: std::cout << "default INTENT\n";
502       }
503     }
504   }
507 I then rebuilt my compiler and ran it on my test case.  This produced the following output:
510   In Leave for parser::Expr
511   Number of arguments: 0
512   In Leave for parser::Expr
513   Number of arguments: 0
514   In Leave for parser::Expr
515   Number of arguments: 0
516   In Leave for parser::Expr
517   Number of arguments: 1
518   INTENT(OUT)
519   In Leave for parser::Expr
520   Number of arguments: 0
523 I then modified my test case to convince myself that I was getting the correct
524 `INTENT` for `IN`, `INOUT`, and default cases.
526 So far, so good.
528 ### Finding the symbols for arguments that are variables
529 The third and last piece of information I needed was to determine if a variable
530 was being passed as an actual argument.  In such cases, I wanted to get the
531 symbol table node (`semantics::Symbol`) for the variable.  My starting point was the
532 `evaluate::ActualArgument` node.  
534 I was unsure of how to do this, so I browsed through existing code to look for
535 how it treated `evaluate::ActualArgument` objects.  Since most of the code that deals with the `evaluate` namespace is in the lib/Evaluate directory, I looked there.  I ran `grep` on all of the `.cpp` files looking for
536 uses of `ActualArgument`.  One of the first hits I got was in `lib/Evaluate/call.cpp` in the definition of `ActualArgument::GetType()`:
538 ```C++
539 std::optional<DynamicType> ActualArgument::GetType() const {
540   if (const Expr<SomeType> *expr{UnwrapExpr()}) {
541     return expr->GetType();
542   } else if (std::holds_alternative<AssumedType>(u_)) {
543     return DynamicType::AssumedType();
544   } else {
545     return std::nullopt;
546   }
550 I noted the call to `UnwrapExpr()` that yielded a value of
551 `Expr<SomeType>`.  So I guessed that I could use this member function to
552 get an `evaluate::Expr<SomeType>` on which I could perform further analysis.
554 I also knew that the header file `include/flang/Evaluate/tools.h` held many
555 utility functions for dealing with `evaluate::Expr` objects.  I was hoping to
556 find something that would determine if an `evaluate::Expr` was a variable.  So
557 I searched for `IsVariable` and got a hit immediately.  
558 ```C++
559   template<typename A> bool IsVariable(const A &x) {
560     if (auto known{IsVariableHelper{}(x)}) {
561       return *known;
562     } else {
563       return false;
564     }
565   }
568 But I actually needed more than just the knowledge that an `evaluate::Expr` was
569 a variable.  I needed the `semantics::Symbol` associated with the variable.  So
570 I searched in `include/flang/Evaluate/tools.h` for functions that returned a
571 `semantics::Symbol`.  I found the following:
573 ```C++
574 // If an expression is simply a whole symbol data designator,
575 // extract and return that symbol, else null.
576 template<typename A> const Symbol *UnwrapWholeSymbolDataRef(const A &x) {
577   if (auto dataRef{ExtractDataRef(x)}) {
578     if (const SymbolRef * p{std::get_if<SymbolRef>(&dataRef->u)}) {
579       return &p->get();
580     }
581   }
582   return nullptr;
586 This was exactly what I wanted.  DO variables must be whole symbols.  So I
587 could try to extract a whole `semantics::Symbol` from the `evaluate::Expr` in my
588 `evaluate::ActualArgument`.  If this extraction resulted in a `semantics::Symbol`
589 that wasn't a `nullptr`, I could then conclude if it was a variable that I
590 could pass to existing functions that would determine if it was an active DO
591 variable.
593 I then modified the compiler to perform the analysis that I'd guessed would
594 work:
596 ```C++
597   void DoChecker::Leave(const parser::Expr &parsedExpr) {
598     std::cout << "In Leave for parser::Expr\n";
599     ActualArgumentSet argSet{CollectActualArguments(GetExpr(parsedExpr))};
600     std::cout << "Number of arguments: " << argSet.size() << "\n";
601     for (const evaluate::ActualArgumentRef &argRef : argSet) {
602       if (const SomeExpr * argExpr{argRef->UnwrapExpr()}) {
603         std::cout << "Got an unwrapped Expr\n";
604         if (const Symbol * var{evaluate::UnwrapWholeSymbolDataRef(*argExpr)}) {
605           std::cout << "Found a whole variable: " << *var << "\n";
606         }
607       }
608       common::Intent intent{argRef->dummyIntent()};
609       switch (intent) {
610         case common::Intent::In: std::cout << "INTENT(IN)\n"; break;
611         case common::Intent::Out: std::cout << "INTENT(OUT)\n"; break;
612         case common::Intent::InOut: std::cout << "INTENT(INOUT)\n"; break;
613         default: std::cout << "default INTENT\n";
614       }
615     }
616   }
617 ```  
619 Note the line that prints out the symbol table entry for the variable:
621 ```C++
622           std::cout << "Found a whole variable: " << *var << "\n";
623 ```  
625 The compiler defines the "<<" operator for `semantics::Symbol`, which is handy
626 for analyzing the compiler's behavior.
628 Here's the result of running the modified compiler on my Fortran test case:
631   In Leave for parser::Expr
632   Number of arguments: 0
633   In Leave for parser::Expr
634   Number of arguments: 0
635   In Leave for parser::Expr
636   Number of arguments: 0
637   In Leave for parser::Expr
638   Number of arguments: 1
639   Got an unwrapped Expr
640   Found a whole variable: ivar: ObjectEntity type: INTEGER(4)
641   INTENT(OUT)
642   In Leave for parser::Expr
643   Number of arguments: 0
646 Sweet.
648 ### Emitting the messages
649 At this point, using the source location information from the original
650 `parser::Expr`, I had enough information to plug into the exiting
651 interfaces for emitting messages for active DO variables.  I modified the
652 compiler code accordingly:
655 ```C++
656   void DoChecker::Leave(const parser::Expr &parsedExpr) {
657     std::cout << "In Leave for parser::Expr\n";
658     ActualArgumentSet argSet{CollectActualArguments(GetExpr(parsedExpr))};
659     std::cout << "Number of arguments: " << argSet.size() << "\n";
660     for (const evaluate::ActualArgumentRef &argRef : argSet) {
661       if (const SomeExpr * argExpr{argRef->UnwrapExpr()}) {
662         std::cout << "Got an unwrapped Expr\n";
663         if (const Symbol * var{evaluate::UnwrapWholeSymbolDataRef(*argExpr)}) {
664           std::cout << "Found a whole variable: " << *var << "\n";
665           common::Intent intent{argRef->dummyIntent()};
666           switch (intent) {
667             case common::Intent::In: std::cout << "INTENT(IN)\n"; break;
668             case common::Intent::Out: 
669               std::cout << "INTENT(OUT)\n"; 
670               context_.CheckDoVarRedefine(parsedExpr.source, *var);
671               break;
672             case common::Intent::InOut: 
673               std::cout << "INTENT(INOUT)\n"; 
674               context_.WarnDoVarRedefine(parsedExpr.source, *var);
675               break;
676             default: std::cout << "default INTENT\n";
677           }
678         }
679       }
680     }
681   }
682 ```  
684 I then ran this code on my test case, and miraculously, got the following
685 output:
688   In Leave for parser::Expr
689   Number of arguments: 0
690   In Leave for parser::Expr
691   Number of arguments: 0
692   In Leave for parser::Expr
693   Number of arguments: 0
694   In Leave for parser::Expr
695   Number of arguments: 1
696   Got an unwrapped Expr
697   Found a whole variable: ivar: ObjectEntity type: INTEGER(4)
698   INTENT(OUT)
699   In Leave for parser::Expr
700   Number of arguments: 0
701   testfun.f90:6:12: error: Cannot redefine DO variable 'ivar'
702         jvar = intentOutFunc(ivar)
703                ^^^^^^^^^^^^^^^^^^^
704   testfun.f90:5:6: Enclosing DO construct
705       do ivar = 1, 10
706          ^^^^
709 Even sweeter.
711 ## Improving the test case
712 At this point, my implementation seemed to be working.  But I was concerned
713 about the limitations of my test case.  So I augmented it to include arguments
714 other than `INTENT(OUT)` and more complex expressions.  Luckily, my
715 augmented test did not reveal any new problems.   
717 Here's the test I ended up with:
719 ```Fortran
720   subroutine s()
722     Integer :: ivar, jvar
724     ! This one is OK
725     do ivar = 1, 10
726       jvar = intentInFunc(ivar)
727     end do
729     ! Error for passing a DO variable to an INTENT(OUT) dummy
730     do ivar = 1, 10
731       jvar = intentOutFunc(ivar)
732     end do
734     ! Error for passing a DO variable to an INTENT(OUT) dummy, more complex 
735     ! expression
736     do ivar = 1, 10
737       jvar = 83 + intentInFunc(intentOutFunc(ivar))
738     end do
740     ! Warning for passing a DO variable to an INTENT(INOUT) dummy
741     do ivar = 1, 10
742       jvar = intentInOutFunc(ivar)
743     end do
745   contains
746     function intentInFunc(dummyArg)
747       integer, intent(in) :: dummyArg
748       integer  :: intentInFunc
750       intentInFunc = 343
751     end function intentInFunc
753     function intentOutFunc(dummyArg)
754       integer, intent(out) :: dummyArg
755       integer  :: intentOutFunc
757       dummyArg = 216
758       intentOutFunc = 343
759     end function intentOutFunc
761     function intentInOutFunc(dummyArg)
762       integer, intent(inout) :: dummyArg
763       integer  :: intentInOutFunc
765       dummyArg = 216
766       intentInOutFunc = 343
767     end function intentInOutFunc
769   end subroutine s
772 ## Submitting the pull request
773 At this point, my implementation seemed functionally complete, so I stripped out all of the debug statements, ran `clang-format` on it and reviewed it
774 to make sure that the names were clear.  Here's what I ended up with:
776 ```C++
777   void DoChecker::Leave(const parser::Expr &parsedExpr) {
778     ActualArgumentSet argSet{CollectActualArguments(GetExpr(parsedExpr))};
779     for (const evaluate::ActualArgumentRef &argRef : argSet) {
780       if (const SomeExpr * argExpr{argRef->UnwrapExpr()}) {
781         if (const Symbol * var{evaluate::UnwrapWholeSymbolDataRef(*argExpr)}) {
782           common::Intent intent{argRef->dummyIntent()};
783           switch (intent) {
784             case common::Intent::Out: 
785               context_.CheckDoVarRedefine(parsedExpr.source, *var);
786               break;
787             case common::Intent::InOut: 
788               context_.WarnDoVarRedefine(parsedExpr.source, *var);
789               break;
790             default:; // INTENT(IN) or default intent
791           }
792         }
793       }
794     }
795   }
798 I then created a pull request to get review comments.  
800 ## Responding to pull request comments
801 I got feedback suggesting that I use an `if` statement rather than a
802 `case` statement.  Another comment reminded me that I should look at the
803 code I'd previously writted to do a similar check for SUBROUTINE calls to see
804 if there was an opportunity to share code.  This examination resulted in
805   converting my existing code to the following pair of functions:
808 ```C++
809   static void CheckIfArgIsDoVar(const evaluate::ActualArgument &arg,
810       const parser::CharBlock location, SemanticsContext &context) {
811     common::Intent intent{arg.dummyIntent()};
812     if (intent == common::Intent::Out || intent == common::Intent::InOut) {
813       if (const SomeExpr * argExpr{arg.UnwrapExpr()}) {
814         if (const Symbol * var{evaluate::UnwrapWholeSymbolDataRef(*argExpr)}) {
815           if (intent == common::Intent::Out) {
816             context.CheckDoVarRedefine(location, *var);
817           } else {
818             context.WarnDoVarRedefine(location, *var);  // INTENT(INOUT)
819           }
820         }
821       }
822     }
823   }
825   void DoChecker::Leave(const parser::Expr &parsedExpr) {
826     if (const SomeExpr * expr{GetExpr(parsedExpr)}) {
827       ActualArgumentSet argSet{CollectActualArguments(*expr)};
828       for (const evaluate::ActualArgumentRef &argRef : argSet) {
829         CheckIfArgIsDoVar(*argRef, parsedExpr.source, context_);
830       }
831     }
832   }
835 The function `CheckIfArgIsDoVar()` was shared with the checks for DO
836 variables being passed to SUBROUTINE calls.
838 At this point, my pull request was approved, and I merged it and deleted the
839 associated branch.