source: src/ControlStruct/LabelFixer.cc @ 721f17a

ADTaaron-thesisarm-ehast-experimentalcleanup-dtorsctordeferred_resndemanglerenumforall-pointer-decaygc_noraiijacob/cs343-translationjenkins-sandboxmemorynew-astnew-ast-unique-exprnew-envno_listpersistent-indexerpthread-emulationqualifiedEnumresolv-newstringwith_gc
Last change on this file since 721f17a was 1869adf, checked in by Rob Schluntz <rschlunt@…>, 9 years ago

fix label name in label address expression

  • Property mode set to 100644
File size: 7.5 KB
Line 
1//
2// Cforall Version 1.0.0 Copyright (C) 2015 University of Waterloo
3//
4// The contents of this file are covered under the licence agreement in the
5// file "LICENCE" distributed with Cforall.
6//
7// LabelFixer.cc --
8//
9// Author           : Rodolfo G. Esteves
10// Created On       : Mon May 18 07:44:20 2015
11// Last Modified By : Rob Schluntz
12// Last Modified On : Wed Jun 24 16:24:34 2015
13// Update Count     : 141
14//
15
16#include <list>
17#include <cassert>
18
19#include "LabelFixer.h"
20#include "MLEMutator.h"
21#include "SynTree/Expression.h"
22#include "SynTree/Statement.h"
23#include "SynTree/Declaration.h"
24#include "utility.h"
25
26#include <iostream>
27
28namespace ControlStruct {
29        LabelFixer::Entry::Entry( Statement *to, Statement *from ) : definition ( to ) {
30                if ( from != 0 ) {
31                        UsageLoc loc; loc.stmt = from;
32                        usage.push_back( loc );
33                }
34        }
35
36        LabelFixer::Entry::Entry( Statement *to, Expression *from ) : definition ( to ) {
37                if ( from != 0 ) {
38                        UsageLoc loc; loc.expr = from;
39                        usage.push_back( loc );
40                }
41        }
42
43
44        bool LabelFixer::Entry::insideLoop() {
45                return ( dynamic_cast< ForStmt * > ( definition ) ||
46                        dynamic_cast< WhileStmt * > ( definition )  );
47        }
48
49        void LabelFixer::Entry::UsageLoc::accept( Visitor & visitor ) {
50                if ( dynamic_cast< Statement * >( stmt ) ) {
51                        stmt->accept( visitor );
52                } else {
53                        expr->accept( visitor );
54                }
55        }
56
57        LabelFixer::LabelFixer( LabelGenerator *gen ) : generator ( gen ) {
58                if ( generator == 0 )
59                        generator = LabelGenerator::getGenerator();
60        }
61
62        void LabelFixer::visit( FunctionDecl *functionDecl ) {
63                maybeAccept( functionDecl->get_statements(), *this );
64
65                MLEMutator mlemut( resolveJumps(), generator );
66                functionDecl->acceptMutator( mlemut );
67        }
68
69        // prune to at most one label definition for each statement
70        void LabelFixer::visit( Statement *stmt ) {
71                currentStatement = stmt;
72                std::list< Label > &labels = stmt->get_labels();
73
74                if ( ! labels.empty() ) {
75                        // only remember one label for each statement
76                        Label current = setLabelsDef( labels, stmt );
77                        labels.clear();
78                        labels.push_front( current );
79                } // if
80        }
81
82        void LabelFixer::visit( BranchStmt *branchStmt ) {
83                visit ( ( Statement * )branchStmt );
84
85                // for labeled branches, add an entry to the label table
86                Label target = branchStmt->get_target();
87                if ( target != "" ) {
88                        setLabelsUsg( target, branchStmt );
89                }
90        }
91
92        void LabelFixer::visit( UntypedExpr *untyped ) {
93                if ( NameExpr * func = dynamic_cast< NameExpr * >( untyped->get_function() ) ) {
94                        if ( func->get_name() == "&&" ) {
95                                NameExpr * arg = dynamic_cast< NameExpr * >( untyped->get_args().front() );
96                                Label target = arg->get_name();
97                                assert( target != "" );
98                                setLabelsUsg( target, untyped );
99                        } else {
100                                Visitor::visit( untyped );
101                        }
102                }
103        }
104
105
106        // sets the definition of the labelTable entry to be the provided
107        // statement for every label in the list parameter. Happens for every kind of statement
108        Label LabelFixer::setLabelsDef( std::list< Label > &llabel, Statement *definition ) {
109                assert( definition != 0 );
110                assert( llabel.size() > 0 );
111
112                Entry * e = new Entry( definition );
113
114                for ( std::list< Label >::iterator i = llabel.begin(); i != llabel.end(); i++ ) {
115                        if ( labelTable.find( *i ) == labelTable.end() ) {
116                                // all labels on this statement need to use the same entry, so this should only be created once
117                                // undefined and unused until now, add an entry
118                                labelTable[ *i ] =  e;
119                        } else if ( labelTable[ *i ]->defined() ) {
120                                // defined twice, error
121                                throw SemanticError( "Duplicate definition of label: " + *i );
122                        }       else {
123                                // used previously, but undefined until now -> link with this entry
124                                Entry * oldEntry = labelTable[ *i ];
125                                e->add_uses( *oldEntry );
126                                labelTable[ *i ] = e;
127                        } // if
128                } // for
129
130                // produce one of the labels attached to this statement to be
131                // temporarily used as the canonical label
132                return labelTable[ llabel.front() ]->get_label();
133        }
134
135        // Remember all uses of a label.
136        template< typename UsageNode >
137        void LabelFixer::setLabelsUsg( Label orgValue, UsageNode *use ) {
138                assert( use != 0 );
139
140                if ( labelTable.find( orgValue ) != labelTable.end() ) {
141                        // the label has been defined or used before
142                        labelTable[ orgValue ]->add_use( use );
143                } else {
144                        labelTable[ orgValue ] = new Entry( 0, use );
145                }
146        }
147
148        class LabelGetter : public Visitor {
149                public:
150                LabelGetter( Label &label ) : label( label ) {}
151
152                virtual void visit( BranchStmt * branchStmt ) {
153                        label = branchStmt->get_target();
154                }
155
156                virtual void visit( UntypedExpr * untyped ) {
157                        NameExpr * name = dynamic_cast< NameExpr * >( untyped->get_function() );
158                        assert( name );
159                        assert( name->get_name() == "&&" );
160                        NameExpr * arg = dynamic_cast< NameExpr * >( untyped->get_args().front() );
161                        assert( arg );
162                        label = arg->get_name();
163                }               
164
165                private:
166                        Label &label;
167        };
168
169        class LabelSetter : public Visitor {
170                public:
171                LabelSetter( Label label ) : label( label ) {}
172
173                virtual void visit( BranchStmt * branchStmt ) {
174                        branchStmt->set_target( label );
175                }
176
177                virtual void visit( UntypedExpr * untyped ) {
178                        NameExpr * name = dynamic_cast< NameExpr * >( untyped->get_function() );
179                        assert( name );
180                        assert( name->get_name() == "&&" );
181                        NameExpr * arg = dynamic_cast< NameExpr * >( untyped->get_args().front() );
182                        assert( arg );
183                        arg->set_name( label );
184                }
185
186        private:
187                Label label;
188        };
189
190        // Ultimately builds a table that maps a label to its defining statement.
191        // In the process,
192        std::map<Label, Statement * > *LabelFixer::resolveJumps() throw ( SemanticError ) {
193                std::map< Statement *, Entry * > def_us;
194
195                // combine the entries for all labels that target the same location
196                for ( std::map< Label, Entry *>::iterator i = labelTable.begin(); i != labelTable.end(); ++i ) {
197                        Entry *e = i->second;
198
199                        if ( def_us.find ( e->get_definition() ) == def_us.end() ) {
200                                def_us[ e->get_definition() ] = e;
201                        } else if ( e->used() ) {
202                                def_us[ e->get_definition() ]->add_uses( *e );
203                        }
204                }
205
206                // create a unique label for each target location.
207                for ( std::map< Statement *, Entry * >::iterator i = def_us.begin(); i != def_us.end(); ++i ) {
208                        Statement *to = (*i).first;
209                        Entry * entry = (*i).second;
210                        std::list< Entry::UsageLoc > &from = entry->get_uses();
211
212                        // no label definition found
213                        if ( to == 0 ) {
214                                Label undef;
215                                LabelGetter getLabel( undef );
216                                from.back().accept( getLabel );
217                                // Label undef = getLabel( from.back()->get_target() );
218                                throw SemanticError ( "'" + undef + "' label not defined");
219                        } // if
220
221                        // generate a new label, and attach it to its defining statement as the only label on that statement
222                        Label finalLabel = generator->newLabel( to->get_labels().back() );
223                        entry->set_label( finalLabel );
224
225                        to->get_labels().clear();
226                        to->get_labels().push_back( finalLabel );
227
228                        // redirect each of the source branch statements to the new target label
229                        for ( std::list< Entry::UsageLoc >::iterator j = from.begin(); j != from.end(); ++j ) {
230                                LabelSetter setLabel( finalLabel );
231                                (*j).accept( setLabel );
232                                // setLabel( *j, finalLabel );
233
234                                // BranchStmt *jump = *j;
235                                // assert( jump != 0 );
236                                // jump->set_target( finalLabel );
237                        } // for
238                } // for
239
240                // create a table where each label maps to its defining statement
241                std::map< Label, Statement * > *ret = new std::map< Label, Statement * >();
242                for ( std::map< Statement *, Entry * >::iterator i = def_us.begin(); i != def_us.end(); ++i ) {
243                        (*ret)[ (*i).second->get_label() ] = (*i).first;
244                }
245
246                return ret;
247        }
248}  // namespace ControlStruct
249
250// Local Variables: //
251// tab-width: 4 //
252// mode: c++ //
253// compile-command: "make install" //
254// End: //
Note: See TracBrowser for help on using the repository browser.