source: src/ResolvExpr/AlternativeFinder.cc@ 91496f3

ADT aaron-thesis arm-eh ast-experimental cleanup-dtors deferred_resn demangler enum forall-pointer-decay jacob/cs343-translation jenkins-sandbox new-ast new-ast-unique-expr new-env no_list persistent-indexer pthread-emulation qualifiedEnum resolv-new with_gc
Last change on this file since 91496f3 was 13deae88, checked in by Rob Schluntz <rschlunt@…>, 8 years ago

Convert AlternativeFinder to PassVisitor

  • Property mode set to 100644
File size: 73.8 KB
RevLine 
[a32b204]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//
[6ed1d4b]7// AlternativeFinder.cc --
[a32b204]8//
9// Author : Richard C. Bilson
10// Created On : Sat May 16 23:52:08 2015
[b128d3e]11// Last Modified By : Peter A. Buhr
12// Last Modified On : Mon Aug 28 13:47:24 2017
13// Update Count : 32
[a32b204]14//
15
[ea6332d]16#include <algorithm> // for copy
[e3e16bc]17#include <cassert> // for strict_dynamic_cast, assert, assertf
[403b388]18#include <cstddef> // for size_t
[ea6332d]19#include <iostream> // for operator<<, cerr, ostream, endl
20#include <iterator> // for back_insert_iterator, back_inserter
21#include <list> // for _List_iterator, list, _List_const_...
22#include <map> // for _Rb_tree_iterator, map, _Rb_tree_c...
[403b388]23#include <memory> // for allocator_traits<>::value_type, unique_ptr
[ea6332d]24#include <utility> // for pair
[aeb75b1]25#include <vector> // for vector
[51b73452]26
[ea6332d]27#include "Alternative.h" // for AltList, Alternative
[51b73452]28#include "AlternativeFinder.h"
[ea6332d]29#include "Common/SemanticError.h" // for SemanticError
30#include "Common/utility.h" // for deleteAll, printAll, CodeLocation
31#include "Cost.h" // for Cost, Cost::zero, operator<<, Cost...
[a8b27c6]32#include "ExplodedActual.h" // for ExplodedActual
[ea6332d]33#include "InitTweak/InitTweak.h" // for getFunctionName
34#include "RenameVars.h" // for RenameVars, global_renamer
35#include "ResolveTypeof.h" // for resolveTypeof
36#include "Resolver.h" // for resolveStmtExpr
37#include "SymTab/Indexer.h" // for Indexer
38#include "SymTab/Mangler.h" // for Mangler
39#include "SymTab/Validate.h" // for validateType
40#include "SynTree/Constant.h" // for Constant
41#include "SynTree/Declaration.h" // for DeclarationWithType, TypeDecl, Dec...
42#include "SynTree/Expression.h" // for Expression, CastExpr, NameExpr
43#include "SynTree/Initializer.h" // for SingleInit, operator<<, Designation
44#include "SynTree/SynTree.h" // for UniqueId
45#include "SynTree/Type.h" // for Type, FunctionType, PointerType
46#include "Tuples/Explode.h" // for explode
47#include "Tuples/Tuples.h" // for isTtype, handleTupleAssignment
48#include "Unify.h" // for unify
49#include "typeops.h" // for adjustExprType, polyCost, castCost
[51b73452]50
[b87a5ed]51extern bool resolvep;
[6ed1d4b]52#define PRINT( text ) if ( resolvep ) { text }
[51b73452]53//#define DEBUG_COST
54
[403b388]55using std::move;
56
57/// copies any copyable type
58template<typename T>
59T copy(const T& x) { return x; }
60
[51b73452]61namespace ResolvExpr {
[13deae88]62 struct AlternativeFinder::Finder : public WithShortCircuiting {
63 Finder( AlternativeFinder & altFinder ) : altFinder( altFinder ), indexer( altFinder.indexer ), alternatives( altFinder.alternatives ), env( altFinder.env ), targetType( altFinder.targetType ) {}
64
65 void previsit( BaseSyntaxNode * ) { visit_children = false; }
66
67 void postvisit( ApplicationExpr * applicationExpr );
68 void postvisit( UntypedExpr * untypedExpr );
69 void postvisit( AddressExpr * addressExpr );
70 void postvisit( LabelAddressExpr * labelExpr );
71 void postvisit( CastExpr * castExpr );
72 void postvisit( VirtualCastExpr * castExpr );
73 void postvisit( UntypedMemberExpr * memberExpr );
74 void postvisit( MemberExpr * memberExpr );
75 void postvisit( NameExpr * variableExpr );
76 void postvisit( VariableExpr * variableExpr );
77 void postvisit( ConstantExpr * constantExpr );
78 void postvisit( SizeofExpr * sizeofExpr );
79 void postvisit( AlignofExpr * alignofExpr );
80 void postvisit( UntypedOffsetofExpr * offsetofExpr );
81 void postvisit( OffsetofExpr * offsetofExpr );
82 void postvisit( OffsetPackExpr * offsetPackExpr );
83 void postvisit( AttrExpr * attrExpr );
84 void postvisit( LogicalExpr * logicalExpr );
85 void postvisit( ConditionalExpr * conditionalExpr );
86 void postvisit( CommaExpr * commaExpr );
87 void postvisit( ImplicitCopyCtorExpr * impCpCtorExpr );
88 void postvisit( ConstructorExpr * ctorExpr );
89 void postvisit( RangeExpr * rangeExpr );
90 void postvisit( UntypedTupleExpr * tupleExpr );
91 void postvisit( TupleExpr * tupleExpr );
92 void postvisit( TupleIndexExpr * tupleExpr );
93 void postvisit( TupleAssignExpr * tupleExpr );
94 void postvisit( UniqueExpr * unqExpr );
95 void postvisit( StmtExpr * stmtExpr );
96 void postvisit( UntypedInitExpr * initExpr );
97
98 /// Adds alternatives for anonymous members
99 void addAnonConversions( const Alternative & alt );
100 /// Adds alternatives for member expressions, given the aggregate, conversion cost for that aggregate, and name of the member
101 template< typename StructOrUnionType > void addAggMembers( StructOrUnionType *aggInst, Expression *expr, const Cost &newCost, const TypeEnvironment & env, Expression * member );
102 /// Adds alternatives for member expressions where the left side has tuple type
103 void addTupleMembers( TupleType * tupleType, Expression *expr, const Cost &newCost, const TypeEnvironment & env, Expression * member );
104 /// Adds alternatives for offsetof expressions, given the base type and name of the member
105 template< typename StructOrUnionType > void addOffsetof( StructOrUnionType *aggInst, const std::string &name );
106 /// Takes a final result and checks if its assertions can be satisfied
107 template<typename OutputIterator>
108 void validateFunctionAlternative( const Alternative &func, ArgPack& result, const std::vector<ArgPack>& results, OutputIterator out );
109 /// Finds matching alternatives for a function, given a set of arguments
110 template<typename OutputIterator>
111 void makeFunctionAlternatives( const Alternative &func, FunctionType *funcType, const ExplodedArgs& args, OutputIterator out );
112 /// Checks if assertion parameters match for a new alternative
113 template< typename OutputIterator >
114 void inferParameters( const AssertionSet &need, AssertionSet &have, const Alternative &newAlt, OpenVarSet &openVars, OutputIterator out );
115 private:
116 AlternativeFinder & altFinder;
117 const SymTab::Indexer &indexer;
118 AltList & alternatives;
119 const TypeEnvironment &env;
120 Type *& targetType;
121 };
122
[a32b204]123 Expression *resolveInVoidContext( Expression *expr, const SymTab::Indexer &indexer, TypeEnvironment &env ) {
124 CastExpr *castToVoid = new CastExpr( expr );
125
126 AlternativeFinder finder( indexer, env );
127 finder.findWithAdjustment( castToVoid );
128
129 // it's a property of the language that a cast expression has either 1 or 0 interpretations; if it has 0
130 // interpretations, an exception has already been thrown.
131 assert( finder.get_alternatives().size() == 1 );
132 CastExpr *newExpr = dynamic_cast< CastExpr* >( finder.get_alternatives().front().expr );
133 assert( newExpr );
134 env = finder.get_alternatives().front().env;
135 return newExpr->get_arg()->clone();
136 }
137
[908cc83]138 Cost sumCost( const AltList &in ) {
[89be1c68]139 Cost total = Cost::zero;
[908cc83]140 for ( AltList::const_iterator i = in.begin(); i != in.end(); ++i ) {
141 total += i->cost;
142 }
143 return total;
144 }
145
[1e8bbac9]146 void printAlts( const AltList &list, std::ostream &os, unsigned int indentAmt ) {
147 Indenter indent = { Indenter::tabsize, indentAmt };
148 for ( AltList::const_iterator i = list.begin(); i != list.end(); ++i ) {
149 i->print( os, indent );
150 os << std::endl;
[a32b204]151 }
[1e8bbac9]152 }
[d9a0e76]153
[1e8bbac9]154 namespace {
[a32b204]155 void makeExprList( const AltList &in, std::list< Expression* > &out ) {
156 for ( AltList::const_iterator i = in.begin(); i != in.end(); ++i ) {
157 out.push_back( i->expr->clone() );
158 }
159 }
[d9a0e76]160
[a32b204]161 struct PruneStruct {
162 bool isAmbiguous;
163 AltList::iterator candidate;
164 PruneStruct() {}
165 PruneStruct( AltList::iterator candidate ): isAmbiguous( false ), candidate( candidate ) {}
166 };
167
[0f19d763]168 /// Prunes a list of alternatives down to those that have the minimum conversion cost for a given return type; skips ambiguous interpretations
[a32b204]169 template< typename InputIterator, typename OutputIterator >
[d7dc824]170 void pruneAlternatives( InputIterator begin, InputIterator end, OutputIterator out ) {
[a32b204]171 // select the alternatives that have the minimum conversion cost for a particular set of result types
172 std::map< std::string, PruneStruct > selected;
173 for ( AltList::iterator candidate = begin; candidate != end; ++candidate ) {
174 PruneStruct current( candidate );
175 std::string mangleName;
[906e24d]176 {
177 Type * newType = candidate->expr->get_result()->clone();
[a32b204]178 candidate->env.apply( newType );
[906e24d]179 mangleName = SymTab::Mangler::mangle( newType );
[a32b204]180 delete newType;
181 }
182 std::map< std::string, PruneStruct >::iterator mapPlace = selected.find( mangleName );
183 if ( mapPlace != selected.end() ) {
184 if ( candidate->cost < mapPlace->second.candidate->cost ) {
185 PRINT(
[6ed1d4b]186 std::cerr << "cost " << candidate->cost << " beats " << mapPlace->second.candidate->cost << std::endl;
[7c64920]187 )
[0f19d763]188 selected[ mangleName ] = current;
[a32b204]189 } else if ( candidate->cost == mapPlace->second.candidate->cost ) {
190 PRINT(
[6ed1d4b]191 std::cerr << "marking ambiguous" << std::endl;
[7c64920]192 )
[0f19d763]193 mapPlace->second.isAmbiguous = true;
[b0837e4]194 } else {
195 PRINT(
196 std::cerr << "cost " << candidate->cost << " loses to " << mapPlace->second.candidate->cost << std::endl;
197 )
[a32b204]198 }
199 } else {
200 selected[ mangleName ] = current;
201 }
202 }
[d9a0e76]203
[0f19d763]204 // accept the alternatives that were unambiguous
205 for ( std::map< std::string, PruneStruct >::iterator target = selected.begin(); target != selected.end(); ++target ) {
206 if ( ! target->second.isAmbiguous ) {
207 Alternative &alt = *target->second.candidate;
[906e24d]208 alt.env.applyFree( alt.expr->get_result() );
[0f19d763]209 *out++ = alt;
[a32b204]210 }
[0f19d763]211 }
[d9a0e76]212 }
[a32b204]213
214 void renameTypes( Expression *expr ) {
[ad51cc2]215 renameTyVars( expr->result );
[e76acbe]216 }
[1dcd9554]217 } // namespace
[b1bead1]218
[1dcd9554]219 void referenceToRvalueConversion( Expression *& expr ) {
220 if ( dynamic_cast< ReferenceType * >( expr->get_result() ) ) {
221 // cast away reference from expr
222 expr = new CastExpr( expr, expr->get_result()->stripReferences()->clone() );
[b1bead1]223 }
[1dcd9554]224 }
[d9a0e76]225
[a32b204]226 template< typename InputIterator, typename OutputIterator >
227 void AlternativeFinder::findSubExprs( InputIterator begin, InputIterator end, OutputIterator out ) {
228 while ( begin != end ) {
229 AlternativeFinder finder( indexer, env );
230 finder.findWithAdjustment( *begin );
231 // XXX either this
232 //Designators::fixDesignations( finder, (*begin++)->get_argName() );
233 // or XXX this
234 begin++;
235 PRINT(
[6ed1d4b]236 std::cerr << "findSubExprs" << std::endl;
237 printAlts( finder.alternatives, std::cerr );
[7c64920]238 )
[0f19d763]239 *out++ = finder;
[a32b204]240 }
[d9a0e76]241 }
242
[a32b204]243 AlternativeFinder::AlternativeFinder( const SymTab::Indexer &indexer, const TypeEnvironment &env )
244 : indexer( indexer ), env( env ) {
[d9a0e76]245 }
[51b73452]246
[4e66a18]247 void AlternativeFinder::find( Expression *expr, bool adjust, bool prune, bool failFast ) {
[13deae88]248 PassVisitor<Finder> finder( *this );
249 expr->accept( finder );
[4e66a18]250 if ( failFast && alternatives.empty() ) {
[83882e9]251 PRINT(
252 std::cerr << "No reasonable alternatives for expression " << expr << std::endl;
253 )
[a32b204]254 throw SemanticError( "No reasonable alternatives for expression ", expr );
255 }
[b6fe7e6]256 if ( prune ) {
[b0837e4]257 auto oldsize = alternatives.size();
[b6fe7e6]258 PRINT(
259 std::cerr << "alternatives before prune:" << std::endl;
260 printAlts( alternatives, std::cerr );
261 )
[bd4f2e9]262 AltList pruned;
263 pruneAlternatives( alternatives.begin(), alternatives.end(), back_inserter( pruned ) );
264 if ( failFast && pruned.empty() ) {
[b6fe7e6]265 std::ostringstream stream;
266 AltList winners;
267 findMinCost( alternatives.begin(), alternatives.end(), back_inserter( winners ) );
[50377a4]268 stream << "Cannot choose between " << winners.size() << " alternatives for expression\n";
[5a824c2]269 expr->print( stream );
[50377a4]270 stream << "Alternatives are:\n";
271 printAlts( winners, stream, 1 );
[b6fe7e6]272 throw SemanticError( stream.str() );
273 }
[bd4f2e9]274 alternatives = move(pruned);
[b0837e4]275 PRINT(
276 std::cerr << "there are " << oldsize << " alternatives before elimination" << std::endl;
277 )
[b6fe7e6]278 PRINT(
279 std::cerr << "there are " << alternatives.size() << " alternatives after elimination" << std::endl;
280 )
[a32b204]281 }
[954ef5b]282 // adjust types after pruning so that types substituted by pruneAlternatives are correctly adjusted
283 for ( AltList::iterator i = alternatives.begin(); i != alternatives.end(); ++i ) {
284 if ( adjust ) {
285 adjustExprType( i->expr->get_result(), i->env, indexer );
286 }
287 }
[8e9cbb2]288
[64ac636]289 // Central location to handle gcc extension keyword, etc. for all expression types.
[8e9cbb2]290 for ( Alternative &iter: alternatives ) {
291 iter.expr->set_extension( expr->get_extension() );
[64ac636]292 iter.expr->location = expr->location;
[8e9cbb2]293 } // for
[0f19d763]294 }
[d9a0e76]295
[4e66a18]296 void AlternativeFinder::findWithAdjustment( Expression *expr ) {
297 find( expr, true );
298 }
299
300 void AlternativeFinder::findWithoutPrune( Expression * expr ) {
301 find( expr, true, false );
302 }
303
304 void AlternativeFinder::maybeFind( Expression * expr ) {
305 find( expr, true, true, false );
[d9a0e76]306 }
[a32b204]307
[13deae88]308 void AlternativeFinder::Finder::addAnonConversions( const Alternative & alt ) {
[4b0f997]309 // adds anonymous member interpretations whenever an aggregate value type is seen.
[d1685588]310 // it's okay for the aggregate expression to have reference type -- cast it to the base type to treat the aggregate as the referenced value
311 std::unique_ptr<Expression> aggrExpr( alt.expr->clone() );
312 alt.env.apply( aggrExpr->get_result() );
313 Type * aggrType = aggrExpr->get_result();
314 if ( dynamic_cast< ReferenceType * >( aggrType ) ) {
315 aggrType = aggrType->stripReferences();
316 aggrExpr.reset( new CastExpr( aggrExpr.release(), aggrType->clone() ) );
317 }
318
319 if ( StructInstType *structInst = dynamic_cast< StructInstType* >( aggrExpr->get_result() ) ) {
[4b0f997]320 NameExpr nameExpr( "" );
[d1685588]321 addAggMembers( structInst, aggrExpr.get(), alt.cost+Cost::safe, alt.env, &nameExpr );
322 } else if ( UnionInstType *unionInst = dynamic_cast< UnionInstType* >( aggrExpr->get_result() ) ) {
[4b0f997]323 NameExpr nameExpr( "" );
[d1685588]324 addAggMembers( unionInst, aggrExpr.get(), alt.cost+Cost::safe, alt.env, &nameExpr );
[4b0f997]325 } // if
326 }
[77971f6]327
[a32b204]328 template< typename StructOrUnionType >
[13deae88]329 void AlternativeFinder::Finder::addAggMembers( StructOrUnionType *aggInst, Expression *expr, const Cost &newCost, const TypeEnvironment & env, Expression * member ) {
[bf32bb8]330 // by this point, member must be a name expr
[c93bc28]331 NameExpr * nameExpr = dynamic_cast< NameExpr * >( member );
332 if ( ! nameExpr ) return;
[bf32bb8]333 const std::string & name = nameExpr->get_name();
334 std::list< Declaration* > members;
335 aggInst->lookup( name, members );
[4b0f997]336
[bf32bb8]337 for ( std::list< Declaration* >::const_iterator i = members.begin(); i != members.end(); ++i ) {
338 if ( DeclarationWithType *dwt = dynamic_cast< DeclarationWithType* >( *i ) ) {
[d9fa60a]339 alternatives.push_back( Alternative( new MemberExpr( dwt, expr->clone() ), env, newCost ) );
[bf32bb8]340 renameTypes( alternatives.back().expr );
[4b0f997]341 addAnonConversions( alternatives.back() ); // add anonymous member interpretations whenever an aggregate value type is seen as a member expression.
[bf32bb8]342 } else {
343 assert( false );
[a32b204]344 }
345 }
[d9a0e76]346 }
[a32b204]347
[13deae88]348 void AlternativeFinder::Finder::addTupleMembers( TupleType * tupleType, Expression *expr, const Cost &newCost, const TypeEnvironment & env, Expression * member ) {
[848ce71]349 if ( ConstantExpr * constantExpr = dynamic_cast< ConstantExpr * >( member ) ) {
350 // get the value of the constant expression as an int, must be between 0 and the length of the tuple type to have meaning
351 // xxx - this should be improved by memoizing the value of constant exprs
352 // during parsing and reusing that information here.
353 std::stringstream ss( constantExpr->get_constant()->get_value() );
[c93bc28]354 int val = 0;
[848ce71]355 std::string tmp;
356 if ( ss >> val && ! (ss >> tmp) ) {
357 if ( val >= 0 && (unsigned int)val < tupleType->size() ) {
358 alternatives.push_back( Alternative( new TupleIndexExpr( expr->clone(), val ), env, newCost ) );
359 } // if
360 } // if
[141b786]361 } else if ( NameExpr * nameExpr = dynamic_cast< NameExpr * >( member ) ) {
362 // xxx - temporary hack until 0/1 are int constants
363 if ( nameExpr->get_name() == "0" || nameExpr->get_name() == "1" ) {
364 std::stringstream ss( nameExpr->get_name() );
365 int val;
366 ss >> val;
367 alternatives.push_back( Alternative( new TupleIndexExpr( expr->clone(), val ), env, newCost ) );
368 }
[848ce71]369 } // if
370 }
371
[13deae88]372 void AlternativeFinder::Finder::postvisit( ApplicationExpr *applicationExpr ) {
[a32b204]373 alternatives.push_back( Alternative( applicationExpr->clone(), env, Cost::zero ) );
[d9a0e76]374 }
375
[ddf8a29]376 Cost computeConversionCost( Type * actualType, Type * formalType, const SymTab::Indexer &indexer, const TypeEnvironment & env ) {
377 PRINT(
378 std::cerr << std::endl << "converting ";
379 actualType->print( std::cerr, 8 );
380 std::cerr << std::endl << " to ";
381 formalType->print( std::cerr, 8 );
382 std::cerr << std::endl << "environment is: ";
383 env.print( std::cerr, 8 );
384 std::cerr << std::endl;
385 )
386 Cost convCost = conversionCost( actualType, formalType, indexer, env );
387 PRINT(
[d06c808]388 std::cerr << std::endl << "cost is " << convCost << std::endl;
[ddf8a29]389 )
390 if ( convCost == Cost::infinity ) {
391 return convCost;
392 }
393 convCost.incPoly( polyCost( formalType, env, indexer ) + polyCost( actualType, env, indexer ) );
[d06c808]394 PRINT(
395 std::cerr << "cost with polycost is " << convCost << std::endl;
396 )
[ddf8a29]397 return convCost;
398 }
399
400 Cost computeExpressionConversionCost( Expression *& actualExpr, Type * formalType, const SymTab::Indexer &indexer, const TypeEnvironment & env ) {
401 Cost convCost = computeConversionCost( actualExpr->result, formalType, indexer, env );
402
[bb666f64]403 // if there is a non-zero conversion cost, ignoring poly cost, then the expression requires conversion.
404 // ignore poly cost for now, since this requires resolution of the cast to infer parameters and this
405 // does not currently work for the reason stated below.
[ddf8a29]406 Cost tmpCost = convCost;
407 tmpCost.incPoly( -tmpCost.get_polyCost() );
408 if ( tmpCost != Cost::zero ) {
409 Type *newType = formalType->clone();
410 env.apply( newType );
411 actualExpr = new CastExpr( actualExpr, newType );
412 // xxx - SHOULD be able to resolve this cast, but at the moment pointers are not castable to zero_t, but are implicitly convertible. This is clearly
413 // inconsistent, once this is fixed it should be possible to resolve the cast.
414 // xxx - this isn't working, it appears because type1 (the formal type) is seen as widenable, but it shouldn't be, because this makes the conversion from DT* to DT* since commontype(zero_t, DT*) is DT*, rather than just nothing.
415
416 // AlternativeFinder finder( indexer, env );
417 // finder.findWithAdjustment( actualExpr );
418 // assertf( finder.get_alternatives().size() > 0, "Somehow castable expression failed to find alternatives." );
419 // assertf( finder.get_alternatives().size() == 1, "Somehow got multiple alternatives for known cast expression." );
420 // Alternative & alt = finder.get_alternatives().front();
421 // delete actualExpr;
422 // actualExpr = alt.expr->clone();
423 }
424 return convCost;
425 }
426
427 Cost computeApplicationConversionCost( Alternative &alt, const SymTab::Indexer &indexer ) {
[e3e16bc]428 ApplicationExpr *appExpr = strict_dynamic_cast< ApplicationExpr* >( alt.expr );
429 PointerType *pointer = strict_dynamic_cast< PointerType* >( appExpr->get_function()->get_result() );
430 FunctionType *function = strict_dynamic_cast< FunctionType* >( pointer->get_base() );
[a32b204]431
[89be1c68]432 Cost convCost = Cost::zero;
[a32b204]433 std::list< DeclarationWithType* >& formals = function->get_parameters();
434 std::list< DeclarationWithType* >::iterator formal = formals.begin();
435 std::list< Expression* >& actuals = appExpr->get_args();
[0362d42]436
[a32b204]437 for ( std::list< Expression* >::iterator actualExpr = actuals.begin(); actualExpr != actuals.end(); ++actualExpr ) {
[53e3b4a]438 Type * actualType = (*actualExpr)->get_result();
[a32b204]439 PRINT(
[6ed1d4b]440 std::cerr << "actual expression:" << std::endl;
441 (*actualExpr)->print( std::cerr, 8 );
442 std::cerr << "--- results are" << std::endl;
[53e3b4a]443 actualType->print( std::cerr, 8 );
[7c64920]444 )
[53e3b4a]445 if ( formal == formals.end() ) {
446 if ( function->get_isVarArgs() ) {
[89be1c68]447 convCost.incUnsafe();
[d06c808]448 PRINT( std::cerr << "end of formals with varargs function: inc unsafe: " << convCost << std::endl; ; )
[b1bead1]449 // convert reference-typed expressions to value-typed expressions
450 referenceToRvalueConversion( *actualExpr );
[53e3b4a]451 continue;
452 } else {
453 return Cost::infinity;
[7c64920]454 }
[53e3b4a]455 }
456 Type * formalType = (*formal)->get_type();
[ddf8a29]457 convCost += computeExpressionConversionCost( *actualExpr, formalType, indexer, alt.env );
[53e3b4a]458 ++formal; // can't be in for-loop update because of the continue
[d9a0e76]459 }
[a32b204]460 if ( formal != formals.end() ) {
461 return Cost::infinity;
[d9a0e76]462 }
463
[a32b204]464 for ( InferredParams::const_iterator assert = appExpr->get_inferParams().begin(); assert != appExpr->get_inferParams().end(); ++assert ) {
[ddf8a29]465 convCost += computeConversionCost( assert->second.actualType, assert->second.formalType, indexer, alt.env );
[a32b204]466 }
[d9a0e76]467
[a32b204]468 return convCost;
469 }
[d9a0e76]470
[8c84ebd]471 /// Adds type variables to the open variable set and marks their assertions
[a32b204]472 void makeUnifiableVars( Type *type, OpenVarSet &unifiableVars, AssertionSet &needAssertions ) {
[8c49c0e]473 for ( Type::ForallList::const_iterator tyvar = type->get_forall().begin(); tyvar != type->get_forall().end(); ++tyvar ) {
[2c57025]474 unifiableVars[ (*tyvar)->get_name() ] = TypeDecl::Data{ *tyvar };
[a32b204]475 for ( std::list< DeclarationWithType* >::iterator assert = (*tyvar)->get_assertions().begin(); assert != (*tyvar)->get_assertions().end(); ++assert ) {
[6c3a988f]476 needAssertions[ *assert ].isUsed = true;
[a32b204]477 }
[d9a0e76]478/// needAssertions.insert( needAssertions.end(), (*tyvar)->get_assertions().begin(), (*tyvar)->get_assertions().end() );
479 }
480 }
[a32b204]481
[89b686a]482 // /// Map of declaration uniqueIds (intended to be the assertions in an AssertionSet) to their parents and the number of times they've been included
483 //typedef std::unordered_map< UniqueId, std::unordered_map< UniqueId, unsigned > > AssertionParentSet;
[79970ed]484
[a1d7679]485 static const int recursionLimit = /*10*/ 4; ///< Limit to depth of recursion satisfaction
[89b686a]486 //static const unsigned recursionParentLimit = 1; ///< Limit to the number of times an assertion can recursively use itself
[51b73452]487
[a32b204]488 void addToIndexer( AssertionSet &assertSet, SymTab::Indexer &indexer ) {
489 for ( AssertionSet::iterator i = assertSet.begin(); i != assertSet.end(); ++i ) {
[6c3a988f]490 if ( i->second.isUsed ) {
[33a25f9]491 indexer.addId( i->first );
[a32b204]492 }
493 }
[d9a0e76]494 }
[79970ed]495
[a32b204]496 template< typename ForwardIterator, typename OutputIterator >
[79970ed]497 void inferRecursive( ForwardIterator begin, ForwardIterator end, const Alternative &newAlt, OpenVarSet &openVars, const SymTab::Indexer &decls, const AssertionSet &newNeed, /*const AssertionParentSet &needParents,*/
[ebf5689]498 int level, const SymTab::Indexer &indexer, OutputIterator out ) {
[a32b204]499 if ( begin == end ) {
500 if ( newNeed.empty() ) {
[6c3a988f]501 PRINT(
502 std::cerr << "all assertions satisfied, output alternative: ";
503 newAlt.print( std::cerr );
504 std::cerr << std::endl;
505 );
[a32b204]506 *out++ = newAlt;
507 return;
508 } else if ( level >= recursionLimit ) {
509 throw SemanticError( "Too many recursive assertions" );
510 } else {
511 AssertionSet newerNeed;
512 PRINT(
513 std::cerr << "recursing with new set:" << std::endl;
514 printAssertionSet( newNeed, std::cerr, 8 );
[7c64920]515 )
[89b686a]516 inferRecursive( newNeed.begin(), newNeed.end(), newAlt, openVars, decls, newerNeed, /*needParents,*/ level+1, indexer, out );
[a32b204]517 return;
518 }
519 }
520
521 ForwardIterator cur = begin++;
[6c3a988f]522 if ( ! cur->second.isUsed ) {
[89b686a]523 inferRecursive( begin, end, newAlt, openVars, decls, newNeed, /*needParents,*/ level, indexer, out );
[7933351]524 return; // xxx - should this continue? previously this wasn't here, and it looks like it should be
[a32b204]525 }
526 DeclarationWithType *curDecl = cur->first;
[7933351]527
[d9a0e76]528 PRINT(
[a32b204]529 std::cerr << "inferRecursive: assertion is ";
530 curDecl->print( std::cerr );
531 std::cerr << std::endl;
[7c64920]532 )
[a40d503]533 std::list< SymTab::Indexer::IdData > candidates;
[a32b204]534 decls.lookupId( curDecl->get_name(), candidates );
[6ed1d4b]535/// if ( candidates.empty() ) { std::cerr << "no candidates!" << std::endl; }
[a40d503]536 for ( const auto & data : candidates ) {
537 DeclarationWithType * candidate = data.id;
[a32b204]538 PRINT(
[6ed1d4b]539 std::cerr << "inferRecursive: candidate is ";
[a40d503]540 candidate->print( std::cerr );
[6ed1d4b]541 std::cerr << std::endl;
[7c64920]542 )
[79970ed]543
[0f19d763]544 AssertionSet newHave, newerNeed( newNeed );
[a32b204]545 TypeEnvironment newEnv( newAlt.env );
546 OpenVarSet newOpenVars( openVars );
[a40d503]547 Type *adjType = candidate->get_type()->clone();
[a32b204]548 adjustExprType( adjType, newEnv, indexer );
[ad51cc2]549 renameTyVars( adjType );
[a32b204]550 PRINT(
551 std::cerr << "unifying ";
552 curDecl->get_type()->print( std::cerr );
553 std::cerr << " with ";
554 adjType->print( std::cerr );
555 std::cerr << std::endl;
[7c64920]556 )
[0f19d763]557 if ( unify( curDecl->get_type(), adjType, newEnv, newerNeed, newHave, newOpenVars, indexer ) ) {
558 PRINT(
559 std::cerr << "success!" << std::endl;
[a32b204]560 )
[0f19d763]561 SymTab::Indexer newDecls( decls );
562 addToIndexer( newHave, newDecls );
563 Alternative newerAlt( newAlt );
564 newerAlt.env = newEnv;
[a40d503]565 assertf( candidate->get_uniqueId(), "Assertion candidate does not have a unique ID: %s", toString( candidate ).c_str() );
[6c3a988f]566
567 // everything with an empty idChain was pulled in by the current assertion.
568 // add current assertion's idChain + current assertion's ID so that the correct inferParameters can be found.
569 for ( auto & a : newerNeed ) {
570 if ( a.second.idChain.empty() ) {
571 a.second.idChain = cur->second.idChain;
572 a.second.idChain.push_back( curDecl->get_uniqueId() );
573 }
574 }
575
[89b686a]576 //AssertionParentSet newNeedParents( needParents );
[22cad76]577 // skip repeatingly-self-recursive assertion satisfaction
[89b686a]578 // DOESN'T WORK: grandchild nodes conflict with their cousins
579 //if ( newNeedParents[ curDecl->get_uniqueId() ][ candDecl->get_uniqueId() ]++ > recursionParentLimit ) continue;
[a40d503]580 Expression *varExpr = data.combine();
[906e24d]581 delete varExpr->get_result();
582 varExpr->set_result( adjType->clone() );
[0f19d763]583 PRINT(
[6ed1d4b]584 std::cerr << "satisfying assertion " << curDecl->get_uniqueId() << " ";
585 curDecl->print( std::cerr );
[a40d503]586 std::cerr << " with declaration " << candidate->get_uniqueId() << " ";
587 candidate->print( std::cerr );
[6ed1d4b]588 std::cerr << std::endl;
[7c64920]589 )
[6c3a988f]590 // follow the current assertion's ID chain to find the correct set of inferred parameters to add the candidate to (i.e. the set of inferred parameters belonging to the entity which requested the assertion parameter).
[df626eb]591 InferredParams * inferParameters = &newerAlt.expr->get_inferParams();
[6c3a988f]592 for ( UniqueId id : cur->second.idChain ) {
593 inferParameters = (*inferParameters)[ id ].inferParams.get();
594 }
[0f19d763]595 // XXX: this is a memory leak, but adjType can't be deleted because it might contain assertions
[a40d503]596 (*inferParameters)[ curDecl->get_uniqueId() ] = ParamEntry( candidate->get_uniqueId(), adjType->clone(), curDecl->get_type()->clone(), varExpr );
[89b686a]597 inferRecursive( begin, end, newerAlt, newOpenVars, newDecls, newerNeed, /*newNeedParents,*/ level, indexer, out );
[0f19d763]598 } else {
599 delete adjType;
600 }
[a32b204]601 }
[d9a0e76]602 }
603
[a32b204]604 template< typename OutputIterator >
[13deae88]605 void AlternativeFinder::Finder::inferParameters( const AssertionSet &need, AssertionSet &have, const Alternative &newAlt, OpenVarSet &openVars, OutputIterator out ) {
[d9a0e76]606// PRINT(
[6ed1d4b]607// std::cerr << "inferParameters: assertions needed are" << std::endl;
608// printAll( need, std::cerr, 8 );
[d9a0e76]609// )
[a32b204]610 SymTab::Indexer decls( indexer );
[e4d829b]611 // PRINT(
612 // std::cerr << "============= original indexer" << std::endl;
613 // indexer.print( std::cerr );
614 // std::cerr << "============= new indexer" << std::endl;
615 // decls.print( std::cerr );
616 // )
[0f19d763]617 addToIndexer( have, decls );
[a32b204]618 AssertionSet newNeed;
[89b686a]619 //AssertionParentSet needParents;
[74b007ba]620 PRINT(
621 std::cerr << "env is: " << std::endl;
622 newAlt.env.print( std::cerr, 0 );
623 std::cerr << std::endl;
624 )
625
[89b686a]626 inferRecursive( need.begin(), need.end(), newAlt, openVars, decls, newNeed, /*needParents,*/ 0, indexer, out );
[d9a0e76]627// PRINT(
[6ed1d4b]628// std::cerr << "declaration 14 is ";
[d9a0e76]629// Declaration::declFromId
630// *out++ = newAlt;
631// )
632 }
633
[aeb75b1]634 /// Gets a default value from an initializer, nullptr if not present
635 ConstantExpr* getDefaultValue( Initializer* init ) {
636 if ( SingleInit* si = dynamic_cast<SingleInit*>( init ) ) {
637 if ( CastExpr* ce = dynamic_cast<CastExpr*>( si->get_value() ) ) {
638 return dynamic_cast<ConstantExpr*>( ce->get_arg() );
639 }
640 }
641 return nullptr;
642 }
643
644 /// State to iteratively build a match of parameter expressions to arguments
645 struct ArgPack {
[452747a]646 std::size_t parent; ///< Index of parent pack
[403b388]647 std::unique_ptr<Expression> expr; ///< The argument stored here
648 Cost cost; ///< The cost of this argument
649 TypeEnvironment env; ///< Environment for this pack
650 AssertionSet need; ///< Assertions outstanding for this pack
651 AssertionSet have; ///< Assertions found for this pack
652 OpenVarSet openVars; ///< Open variables for this pack
653 unsigned nextArg; ///< Index of next argument in arguments list
654 unsigned tupleStart; ///< Number of tuples that start at this index
[a8b27c6]655 unsigned nextExpl; ///< Index of next exploded element
656 unsigned explAlt; ///< Index of alternative for nextExpl > 0
[403b388]657
658 ArgPack()
[ad51cc2]659 : parent(0), expr(), cost(Cost::zero), env(), need(), have(), openVars(), nextArg(0),
[a8b27c6]660 tupleStart(0), nextExpl(0), explAlt(0) {}
[aeb75b1]661
[11094d9]662 ArgPack(const TypeEnvironment& env, const AssertionSet& need, const AssertionSet& have,
[aeb75b1]663 const OpenVarSet& openVars)
[452747a]664 : parent(0), expr(), cost(Cost::zero), env(env), need(need), have(have),
[a8b27c6]665 openVars(openVars), nextArg(0), tupleStart(0), nextExpl(0), explAlt(0) {}
[11094d9]666
[452747a]667 ArgPack(std::size_t parent, Expression* expr, TypeEnvironment&& env, AssertionSet&& need,
668 AssertionSet&& have, OpenVarSet&& openVars, unsigned nextArg,
[178e4ec]669 unsigned tupleStart = 0, Cost cost = Cost::zero, unsigned nextExpl = 0,
[a8b27c6]670 unsigned explAlt = 0 )
[452747a]671 : parent(parent), expr(expr->clone()), cost(cost), env(move(env)), need(move(need)),
[403b388]672 have(move(have)), openVars(move(openVars)), nextArg(nextArg), tupleStart(tupleStart),
[a8b27c6]673 nextExpl(nextExpl), explAlt(explAlt) {}
[452747a]674
675 ArgPack(const ArgPack& o, TypeEnvironment&& env, AssertionSet&& need, AssertionSet&& have,
[73a5cadb]676 OpenVarSet&& openVars, unsigned nextArg, Cost added )
[452747a]677 : parent(o.parent), expr(o.expr ? o.expr->clone() : nullptr), cost(o.cost + added),
678 env(move(env)), need(move(need)), have(move(have)), openVars(move(openVars)),
[a8b27c6]679 nextArg(nextArg), tupleStart(o.tupleStart), nextExpl(0), explAlt(0) {}
[73a5cadb]680
[a8b27c6]681 /// true iff this pack is in the middle of an exploded argument
682 bool hasExpl() const { return nextExpl > 0; }
[aeb75b1]683
[a8b27c6]684 /// Gets the list of exploded alternatives for this pack
685 const ExplodedActual& getExpl( const ExplodedArgs& args ) const {
686 return args[nextArg-1][explAlt];
687 }
[aeb75b1]688
689 /// Ends a tuple expression, consolidating the appropriate actuals
[403b388]690 void endTuple( const std::vector<ArgPack>& packs ) {
691 // add all expressions in tuple to list, summing cost
[aeb75b1]692 std::list<Expression*> exprs;
[403b388]693 const ArgPack* pack = this;
694 if ( expr ) { exprs.push_front( expr.release() ); }
695 while ( pack->tupleStart == 0 ) {
696 pack = &packs[pack->parent];
697 exprs.push_front( pack->expr->clone() );
698 cost += pack->cost;
[aeb75b1]699 }
[403b388]700 // reset pack to appropriate tuple
701 expr.reset( new TupleExpr( exprs ) );
702 tupleStart = pack->tupleStart - 1;
703 parent = pack->parent;
[aeb75b1]704 }
[4b6ef70]705 };
[aeb75b1]706
707 /// Instantiates an argument to match a formal, returns false if no results left
[11094d9]708 bool instantiateArgument( Type* formalType, Initializer* initializer,
[178e4ec]709 const ExplodedArgs& args, std::vector<ArgPack>& results, std::size_t& genStart,
[a8b27c6]710 const SymTab::Indexer& indexer, unsigned nTuples = 0 ) {
[aeb75b1]711 if ( TupleType* tupleType = dynamic_cast<TupleType*>( formalType ) ) {
712 // formalType is a TupleType - group actuals into a TupleExpr
[403b388]713 ++nTuples;
[aeb75b1]714 for ( Type* type : *tupleType ) {
715 // xxx - dropping initializer changes behaviour from previous, but seems correct
[452747a]716 if ( ! instantiateArgument(
717 type, nullptr, args, results, genStart, indexer, nTuples ) )
[aeb75b1]718 return false;
[403b388]719 nTuples = 0;
720 }
721 // re-consititute tuples for final generation
722 for ( auto i = genStart; i < results.size(); ++i ) {
723 results[i].endTuple( results );
[aeb75b1]724 }
725 return true;
726 } else if ( TypeInstType* ttype = Tuples::isTtype( formalType ) ) {
727 // formalType is a ttype, consumes all remaining arguments
728 // xxx - mixing default arguments with variadic??
[403b388]729
730 // completed tuples; will be spliced to end of results to finish
731 std::vector<ArgPack> finalResults{};
732
[aeb75b1]733 // iterate until all results completed
[403b388]734 std::size_t genEnd;
735 ++nTuples;
736 do {
737 genEnd = results.size();
738
[aeb75b1]739 // add another argument to results
[403b388]740 for ( std::size_t i = genStart; i < genEnd; ++i ) {
[a8b27c6]741 auto nextArg = results[i].nextArg;
[452747a]742
[62194cb]743 // use next element of exploded tuple if present
[a8b27c6]744 if ( results[i].hasExpl() ) {
745 const ExplodedActual& expl = results[i].getExpl( args );
[403b388]746
[a8b27c6]747 unsigned nextExpl = results[i].nextExpl + 1;
[62194cb]748 if ( nextExpl == expl.exprs.size() ) {
[a8b27c6]749 nextExpl = 0;
750 }
[403b388]751
752 results.emplace_back(
[178e4ec]753 i, expl.exprs[results[i].nextExpl].get(), copy(results[i].env),
754 copy(results[i].need), copy(results[i].have),
755 copy(results[i].openVars), nextArg, nTuples, Cost::zero, nextExpl,
[62194cb]756 results[i].explAlt );
[452747a]757
[403b388]758 continue;
759 }
[452747a]760
[aeb75b1]761 // finish result when out of arguments
[a8b27c6]762 if ( nextArg >= args.size() ) {
[452747a]763 ArgPack newResult{
764 results[i].env, results[i].need, results[i].have,
[403b388]765 results[i].openVars };
[a8b27c6]766 newResult.nextArg = nextArg;
[403b388]767 Type* argType;
768
[7faab5e]769 if ( nTuples > 0 || ! results[i].expr ) {
[ad51cc2]770 // first iteration or no expression to clone,
[7faab5e]771 // push empty tuple expression
[403b388]772 newResult.parent = i;
773 std::list<Expression*> emptyList;
774 newResult.expr.reset( new TupleExpr( emptyList ) );
775 argType = newResult.expr->get_result();
[aeb75b1]776 } else {
[403b388]777 // clone result to collect tuple
778 newResult.parent = results[i].parent;
779 newResult.cost = results[i].cost;
780 newResult.tupleStart = results[i].tupleStart;
781 newResult.expr.reset( results[i].expr->clone() );
782 argType = newResult.expr->get_result();
783
784 if ( results[i].tupleStart > 0 && Tuples::isTtype( argType ) ) {
[452747a]785 // the case where a ttype value is passed directly is special,
[403b388]786 // e.g. for argument forwarding purposes
[452747a]787 // xxx - what if passing multiple arguments, last of which is
[403b388]788 // ttype?
[452747a]789 // xxx - what would happen if unify was changed so that unifying
790 // tuple
791 // types flattened both before unifying lists? then pass in
[403b388]792 // TupleType (ttype) below.
793 --newResult.tupleStart;
794 } else {
795 // collapse leftover arguments into tuple
796 newResult.endTuple( results );
797 argType = newResult.expr->get_result();
798 }
[aeb75b1]799 }
[403b388]800
[aeb75b1]801 // check unification for ttype before adding to final
[452747a]802 if ( unify( ttype, argType, newResult.env, newResult.need, newResult.have,
[403b388]803 newResult.openVars, indexer ) ) {
804 finalResults.push_back( move(newResult) );
[aeb75b1]805 }
[452747a]806
[aeb75b1]807 continue;
808 }
809
810 // add each possible next argument
[a8b27c6]811 for ( std::size_t j = 0; j < args[nextArg].size(); ++j ) {
812 const ExplodedActual& expl = args[nextArg][j];
[178e4ec]813
[403b388]814 // fresh copies of parent parameters for this iteration
815 TypeEnvironment env = results[i].env;
816 OpenVarSet openVars = results[i].openVars;
817
[a8b27c6]818 env.addActual( expl.env, openVars );
[11094d9]819
[a8b27c6]820 // skip empty tuple arguments by (near-)cloning parent into next gen
[62194cb]821 if ( expl.exprs.empty() ) {
[73a5cadb]822 results.emplace_back(
[452747a]823 results[i], move(env), copy(results[i].need),
[a8b27c6]824 copy(results[i].have), move(openVars), nextArg + 1, expl.cost );
[452747a]825
[403b388]826 continue;
[4b6ef70]827 }
[11094d9]828
[403b388]829 // add new result
830 results.emplace_back(
[178e4ec]831 i, expl.exprs.front().get(), move(env), copy(results[i].need),
832 copy(results[i].have), move(openVars), nextArg + 1,
[62194cb]833 nTuples, expl.cost, expl.exprs.size() == 1 ? 0 : 1, j );
[aeb75b1]834 }
835 }
836
837 // reset for next round
[403b388]838 genStart = genEnd;
839 nTuples = 0;
840 } while ( genEnd != results.size() );
841
842 // splice final results onto results
843 for ( std::size_t i = 0; i < finalResults.size(); ++i ) {
844 results.push_back( move(finalResults[i]) );
[aeb75b1]845 }
[403b388]846 return ! finalResults.empty();
[aeb75b1]847 }
[11094d9]848
[aeb75b1]849 // iterate each current subresult
[403b388]850 std::size_t genEnd = results.size();
851 for ( std::size_t i = genStart; i < genEnd; ++i ) {
[a8b27c6]852 auto nextArg = results[i].nextArg;
853
[403b388]854 // use remainder of exploded tuple if present
[a8b27c6]855 if ( results[i].hasExpl() ) {
856 const ExplodedActual& expl = results[i].getExpl( args );
[62194cb]857 Expression* expr = expl.exprs[results[i].nextExpl].get();
[452747a]858
[403b388]859 TypeEnvironment env = results[i].env;
860 AssertionSet need = results[i].need, have = results[i].have;
861 OpenVarSet openVars = results[i].openVars;
[4b6ef70]862
[62194cb]863 Type* actualType = expr->get_result();
[4b6ef70]864
865 PRINT(
866 std::cerr << "formal type is ";
867 formalType->print( std::cerr );
868 std::cerr << std::endl << "actual type is ";
869 actualType->print( std::cerr );
870 std::cerr << std::endl;
871 )
[11094d9]872
[403b388]873 if ( unify( formalType, actualType, env, need, have, openVars, indexer ) ) {
[a8b27c6]874 unsigned nextExpl = results[i].nextExpl + 1;
[62194cb]875 if ( nextExpl == expl.exprs.size() ) {
[a8b27c6]876 nextExpl = 0;
877 }
[178e4ec]878
[452747a]879 results.emplace_back(
[178e4ec]880 i, expr, move(env), move(need), move(have), move(openVars), nextArg,
[62194cb]881 nTuples, Cost::zero, nextExpl, results[i].explAlt );
[4b6ef70]882 }
883
884 continue;
[403b388]885 }
[452747a]886
[403b388]887 // use default initializers if out of arguments
[a8b27c6]888 if ( nextArg >= args.size() ) {
[aeb75b1]889 if ( ConstantExpr* cnstExpr = getDefaultValue( initializer ) ) {
890 if ( Constant* cnst = dynamic_cast<Constant*>( cnstExpr->get_constant() ) ) {
[403b388]891 TypeEnvironment env = results[i].env;
892 AssertionSet need = results[i].need, have = results[i].have;
893 OpenVarSet openVars = results[i].openVars;
894
[452747a]895 if ( unify( formalType, cnst->get_type(), env, need, have, openVars,
[403b388]896 indexer ) ) {
897 results.emplace_back(
[452747a]898 i, cnstExpr, move(env), move(need), move(have),
[a8b27c6]899 move(openVars), nextArg, nTuples );
[aeb75b1]900 }
901 }
902 }
[403b388]903
[aeb75b1]904 continue;
905 }
906
907 // Check each possible next argument
[a8b27c6]908 for ( std::size_t j = 0; j < args[nextArg].size(); ++j ) {
909 const ExplodedActual& expl = args[nextArg][j];
910
[403b388]911 // fresh copies of parent parameters for this iteration
912 TypeEnvironment env = results[i].env;
913 AssertionSet need = results[i].need, have = results[i].have;
914 OpenVarSet openVars = results[i].openVars;
915
[a8b27c6]916 env.addActual( expl.env, openVars );
[4b6ef70]917
[a8b27c6]918 // skip empty tuple arguments by (near-)cloning parent into next gen
[62194cb]919 if ( expl.exprs.empty() ) {
[73a5cadb]920 results.emplace_back(
[178e4ec]921 results[i], move(env), move(need), move(have), move(openVars),
[a8b27c6]922 nextArg + 1, expl.cost );
[73a5cadb]923
[4b6ef70]924 continue;
925 }
[aeb75b1]926
[4b6ef70]927 // consider only first exploded actual
[62194cb]928 Expression* expr = expl.exprs.front().get();
929 Type* actualType = expr->get_result()->clone();
[a585396]930
[4b6ef70]931 PRINT(
932 std::cerr << "formal type is ";
933 formalType->print( std::cerr );
934 std::cerr << std::endl << "actual type is ";
935 actualType->print( std::cerr );
936 std::cerr << std::endl;
937 )
[aeb75b1]938
[4b6ef70]939 // attempt to unify types
[403b388]940 if ( unify( formalType, actualType, env, need, have, openVars, indexer ) ) {
941 // add new result
942 results.emplace_back(
[178e4ec]943 i, expr, move(env), move(need), move(have), move(openVars), nextArg + 1,
[62194cb]944 nTuples, expl.cost, expl.exprs.size() == 1 ? 0 : 1, j );
[4b6ef70]945 }
[aeb75b1]946 }
947 }
948
949 // reset for next parameter
[403b388]950 genStart = genEnd;
[11094d9]951
[403b388]952 return genEnd != results.size();
953 }
954
955 template<typename OutputIterator>
[13deae88]956 void AlternativeFinder::Finder::validateFunctionAlternative( const Alternative &func, ArgPack& result,
[403b388]957 const std::vector<ArgPack>& results, OutputIterator out ) {
958 ApplicationExpr *appExpr = new ApplicationExpr( func.expr->clone() );
959 // sum cost and accumulate actuals
960 std::list<Expression*>& args = appExpr->get_args();
[8a62d04]961 Cost cost = func.cost;
[403b388]962 const ArgPack* pack = &result;
963 while ( pack->expr ) {
964 args.push_front( pack->expr->clone() );
965 cost += pack->cost;
966 pack = &results[pack->parent];
967 }
968 // build and validate new alternative
969 Alternative newAlt( appExpr, result.env, cost );
970 PRINT(
971 std::cerr << "instantiate function success: " << appExpr << std::endl;
972 std::cerr << "need assertions:" << std::endl;
973 printAssertionSet( result.need, std::cerr, 8 );
974 )
975 inferParameters( result.need, result.have, newAlt, result.openVars, out );
[11094d9]976 }
[aeb75b1]977
978 template<typename OutputIterator>
[13deae88]979 void AlternativeFinder::Finder::makeFunctionAlternatives( const Alternative &func,
[a8b27c6]980 FunctionType *funcType, const ExplodedArgs &args, OutputIterator out ) {
[aeb75b1]981 OpenVarSet funcOpenVars;
982 AssertionSet funcNeed, funcHave;
[3f7e12cb]983 TypeEnvironment funcEnv( func.env );
[aeb75b1]984 makeUnifiableVars( funcType, funcOpenVars, funcNeed );
[11094d9]985 // add all type variables as open variables now so that those not used in the parameter
[aeb75b1]986 // list are still considered open.
987 funcEnv.add( funcType->get_forall() );
[11094d9]988
[53e3b4a]989 if ( targetType && ! targetType->isVoid() && ! funcType->get_returnVals().empty() ) {
[ea83e00a]990 // attempt to narrow based on expected target type
991 Type * returnType = funcType->get_returnVals().front()->get_type();
[11094d9]992 if ( ! unify( returnType, targetType, funcEnv, funcNeed, funcHave, funcOpenVars,
[aeb75b1]993 indexer ) ) {
994 // unification failed, don't pursue this function alternative
[ea83e00a]995 return;
996 }
997 }
998
[aeb75b1]999 // iteratively build matches, one parameter at a time
[403b388]1000 std::vector<ArgPack> results;
1001 results.push_back( ArgPack{ funcEnv, funcNeed, funcHave, funcOpenVars } );
1002 std::size_t genStart = 0;
1003
[aeb75b1]1004 for ( DeclarationWithType* formal : funcType->get_parameters() ) {
1005 ObjectDecl* obj = strict_dynamic_cast< ObjectDecl* >( formal );
[11094d9]1006 if ( ! instantiateArgument(
[403b388]1007 obj->get_type(), obj->get_init(), args, results, genStart, indexer ) )
[aeb75b1]1008 return;
1009 }
1010
1011 if ( funcType->get_isVarArgs() ) {
[403b388]1012 // append any unused arguments to vararg pack
1013 std::size_t genEnd;
1014 do {
1015 genEnd = results.size();
1016
1017 // iterate results
1018 for ( std::size_t i = genStart; i < genEnd; ++i ) {
[a8b27c6]1019 auto nextArg = results[i].nextArg;
[452747a]1020
[403b388]1021 // use remainder of exploded tuple if present
[a8b27c6]1022 if ( results[i].hasExpl() ) {
1023 const ExplodedActual& expl = results[i].getExpl( args );
[403b388]1024
[a8b27c6]1025 unsigned nextExpl = results[i].nextExpl + 1;
[62194cb]1026 if ( nextExpl == expl.exprs.size() ) {
[a8b27c6]1027 nextExpl = 0;
1028 }
[403b388]1029
1030 results.emplace_back(
[178e4ec]1031 i, expl.exprs[results[i].nextExpl].get(), copy(results[i].env),
1032 copy(results[i].need), copy(results[i].have),
1033 copy(results[i].openVars), nextArg, 0, Cost::zero, nextExpl,
[62194cb]1034 results[i].explAlt );
[452747a]1035
[403b388]1036 continue;
1037 }
1038
1039 // finish result when out of arguments
[a8b27c6]1040 if ( nextArg >= args.size() ) {
[403b388]1041 validateFunctionAlternative( func, results[i], results, out );
[fae6f21]1042
[aeb75b1]1043 continue;
1044 }
1045
1046 // add each possible next argument
[a8b27c6]1047 for ( std::size_t j = 0; j < args[nextArg].size(); ++j ) {
1048 const ExplodedActual& expl = args[nextArg][j];
1049
[403b388]1050 // fresh copies of parent parameters for this iteration
1051 TypeEnvironment env = results[i].env;
1052 OpenVarSet openVars = results[i].openVars;
1053
[a8b27c6]1054 env.addActual( expl.env, openVars );
[d551d0a]1055
[a8b27c6]1056 // skip empty tuple arguments by (near-)cloning parent into next gen
[62194cb]1057 if ( expl.exprs.empty() ) {
[452747a]1058 results.emplace_back(
1059 results[i], move(env), copy(results[i].need),
[a8b27c6]1060 copy(results[i].have), move(openVars), nextArg + 1, expl.cost );
[178e4ec]1061
[403b388]1062 continue;
1063 }
[d551d0a]1064
[403b388]1065 // add new result
1066 results.emplace_back(
[178e4ec]1067 i, expl.exprs.front().get(), move(env), copy(results[i].need),
1068 copy(results[i].have), move(openVars), nextArg + 1, 0,
[62194cb]1069 expl.cost, expl.exprs.size() == 1 ? 0 : 1, j );
[aeb75b1]1070 }
1071 }
1072
[403b388]1073 genStart = genEnd;
1074 } while ( genEnd != results.size() );
[aeb75b1]1075 } else {
1076 // filter out results that don't use all the arguments
[403b388]1077 for ( std::size_t i = genStart; i < results.size(); ++i ) {
1078 ArgPack& result = results[i];
[a8b27c6]1079 if ( ! result.hasExpl() && result.nextArg >= args.size() ) {
[403b388]1080 validateFunctionAlternative( func, result, results, out );
[aeb75b1]1081 }
1082 }
1083 }
[d9a0e76]1084 }
1085
[13deae88]1086 void AlternativeFinder::Finder::postvisit( UntypedExpr *untypedExpr ) {
[6ccfb7f]1087 AlternativeFinder funcFinder( indexer, env );
[a32b204]1088 funcFinder.findWithAdjustment( untypedExpr->get_function() );
[6ccfb7f]1089 // if there are no function alternatives, then proceeding is a waste of time.
1090 if ( funcFinder.alternatives.empty() ) return;
1091
[aeb75b1]1092 std::vector< AlternativeFinder > argAlternatives;
[13deae88]1093 altFinder.findSubExprs( untypedExpr->begin_args(), untypedExpr->end_args(),
[aeb75b1]1094 back_inserter( argAlternatives ) );
[d9a0e76]1095
[5af62f1]1096 // take care of possible tuple assignments
1097 // if not tuple assignment, assignment is taken care of as a normal function call
[13deae88]1098 Tuples::handleTupleAssignment( altFinder, untypedExpr, argAlternatives );
[c43c171]1099
[6ccfb7f]1100 // find function operators
[4e66a18]1101 static NameExpr *opExpr = new NameExpr( "?()" );
[6ccfb7f]1102 AlternativeFinder funcOpFinder( indexer, env );
[4e66a18]1103 // it's ok if there aren't any defined function ops
1104 funcOpFinder.maybeFind( opExpr);
[6ccfb7f]1105 PRINT(
1106 std::cerr << "known function ops:" << std::endl;
[50377a4]1107 printAlts( funcOpFinder.alternatives, std::cerr, 1 );
[6ccfb7f]1108 )
1109
[a8b27c6]1110 // pre-explode arguments
1111 ExplodedArgs argExpansions;
1112 argExpansions.reserve( argAlternatives.size() );
1113
1114 for ( const AlternativeFinder& arg : argAlternatives ) {
1115 argExpansions.emplace_back();
1116 auto& argE = argExpansions.back();
1117 argE.reserve( arg.alternatives.size() );
[178e4ec]1118
[a8b27c6]1119 for ( const Alternative& actual : arg ) {
1120 argE.emplace_back( actual, indexer );
1121 }
1122 }
1123
[a32b204]1124 AltList candidates;
[91b8a17]1125 SemanticError errors;
[b1bead1]1126 for ( AltList::iterator func = funcFinder.alternatives.begin(); func != funcFinder.alternatives.end(); ++func ) {
[91b8a17]1127 try {
1128 PRINT(
1129 std::cerr << "working on alternative: " << std::endl;
1130 func->print( std::cerr, 8 );
1131 )
1132 // check if the type is pointer to function
[b1bead1]1133 if ( PointerType *pointer = dynamic_cast< PointerType* >( func->expr->get_result()->stripReferences() ) ) {
[91b8a17]1134 if ( FunctionType *function = dynamic_cast< FunctionType* >( pointer->get_base() ) ) {
[326338ae]1135 Alternative newFunc( *func );
1136 referenceToRvalueConversion( newFunc.expr );
[a8b27c6]1137 makeFunctionAlternatives( newFunc, function, argExpansions,
[aeb75b1]1138 std::back_inserter( candidates ) );
[b1bead1]1139 }
1140 } else if ( TypeInstType *typeInst = dynamic_cast< TypeInstType* >( func->expr->get_result()->stripReferences() ) ) { // handle ftype (e.g. *? on function pointer)
1141 EqvClass eqvClass;
1142 if ( func->env.lookup( typeInst->get_name(), eqvClass ) && eqvClass.type ) {
1143 if ( FunctionType *function = dynamic_cast< FunctionType* >( eqvClass.type ) ) {
[326338ae]1144 Alternative newFunc( *func );
1145 referenceToRvalueConversion( newFunc.expr );
[a8b27c6]1146 makeFunctionAlternatives( newFunc, function, argExpansions,
[aeb75b1]1147 std::back_inserter( candidates ) );
[a32b204]1148 } // if
1149 } // if
[11094d9]1150 }
[91b8a17]1151 } catch ( SemanticError &e ) {
1152 errors.append( e );
1153 }
[a32b204]1154 } // for
1155
[aeb75b1]1156 // try each function operator ?() with each function alternative
1157 if ( ! funcOpFinder.alternatives.empty() ) {
[a8b27c6]1158 // add exploded function alternatives to front of argument list
1159 std::vector<ExplodedActual> funcE;
1160 funcE.reserve( funcFinder.alternatives.size() );
1161 for ( const Alternative& actual : funcFinder ) {
1162 funcE.emplace_back( actual, indexer );
1163 }
1164 argExpansions.insert( argExpansions.begin(), move(funcE) );
[aeb75b1]1165
1166 for ( AltList::iterator funcOp = funcOpFinder.alternatives.begin();
1167 funcOp != funcOpFinder.alternatives.end(); ++funcOp ) {
1168 try {
1169 // check if type is a pointer to function
[11094d9]1170 if ( PointerType* pointer = dynamic_cast<PointerType*>(
[aeb75b1]1171 funcOp->expr->get_result()->stripReferences() ) ) {
[11094d9]1172 if ( FunctionType* function =
[aeb75b1]1173 dynamic_cast<FunctionType*>( pointer->get_base() ) ) {
1174 Alternative newFunc( *funcOp );
1175 referenceToRvalueConversion( newFunc.expr );
[a8b27c6]1176 makeFunctionAlternatives( newFunc, function, argExpansions,
[aeb75b1]1177 std::back_inserter( candidates ) );
1178 }
1179 }
1180 } catch ( SemanticError &e ) {
1181 errors.append( e );
1182 }
1183 }
1184 }
1185
[91b8a17]1186 // Implement SFINAE; resolution errors are only errors if there aren't any non-erroneous resolutions
1187 if ( candidates.empty() && ! errors.isEmpty() ) { throw errors; }
1188
[4b0f997]1189 // compute conversionsion costs
[bd4f2e9]1190 for ( Alternative& withFunc : candidates ) {
1191 Cost cvtCost = computeApplicationConversionCost( withFunc, indexer );
[a32b204]1192
1193 PRINT(
[bd4f2e9]1194 ApplicationExpr *appExpr = strict_dynamic_cast< ApplicationExpr* >( withFunc.expr );
[e3e16bc]1195 PointerType *pointer = strict_dynamic_cast< PointerType* >( appExpr->get_function()->get_result() );
1196 FunctionType *function = strict_dynamic_cast< FunctionType* >( pointer->get_base() );
[a61ad31]1197 std::cerr << "Case +++++++++++++ " << appExpr->get_function() << std::endl;
[6ed1d4b]1198 std::cerr << "formals are:" << std::endl;
1199 printAll( function->get_parameters(), std::cerr, 8 );
1200 std::cerr << "actuals are:" << std::endl;
1201 printAll( appExpr->get_args(), std::cerr, 8 );
1202 std::cerr << "bindings are:" << std::endl;
[bd4f2e9]1203 withFunc.env.print( std::cerr, 8 );
[6ed1d4b]1204 std::cerr << "cost of conversion is:" << cvtCost << std::endl;
[7c64920]1205 )
1206 if ( cvtCost != Cost::infinity ) {
[bd4f2e9]1207 withFunc.cvtCost = cvtCost;
1208 alternatives.push_back( withFunc );
[7c64920]1209 } // if
[a32b204]1210 } // for
[4b0f997]1211
[bd4f2e9]1212 candidates = move(alternatives);
[a32b204]1213
[11094d9]1214 // use a new list so that alternatives are not examined by addAnonConversions twice.
1215 AltList winners;
1216 findMinCost( candidates.begin(), candidates.end(), std::back_inserter( winners ) );
[ea83e00a]1217
[452747a]1218 // function may return struct or union value, in which case we need to add alternatives
1219 // for implicitconversions to each of the anonymous members, must happen after findMinCost
[bd4f2e9]1220 // since anon conversions are never the cheapest expression
[11094d9]1221 for ( const Alternative & alt : winners ) {
[ca946a4]1222 addAnonConversions( alt );
1223 }
[bd4f2e9]1224 spliceBegin( alternatives, winners );
[ca946a4]1225
[ea83e00a]1226 if ( alternatives.empty() && targetType && ! targetType->isVoid() ) {
1227 // xxx - this is a temporary hack. If resolution is unsuccessful with a target type, try again without a
1228 // target type, since it will sometimes succeed when it wouldn't easily with target type binding. For example,
1229 // forall( otype T ) lvalue T ?[?]( T *, ptrdiff_t );
1230 // const char * x = "hello world";
1231 // unsigned char ch = x[0];
1232 // Fails with simple return type binding. First, T is bound to unsigned char, then (x: const char *) is unified
1233 // with unsigned char *, which fails because pointer base types must be unified exactly. The new resolver should
1234 // fix this issue in a more robust way.
1235 targetType = nullptr;
[13deae88]1236 postvisit( untypedExpr );
[ea83e00a]1237 }
[a32b204]1238 }
1239
1240 bool isLvalue( Expression *expr ) {
[906e24d]1241 // xxx - recurse into tuples?
[d29fa5f]1242 return expr->result && ( expr->get_result()->get_lvalue() || dynamic_cast< ReferenceType * >( expr->get_result() ) );
[a32b204]1243 }
1244
[13deae88]1245 void AlternativeFinder::Finder::postvisit( AddressExpr *addressExpr ) {
[a32b204]1246 AlternativeFinder finder( indexer, env );
1247 finder.find( addressExpr->get_arg() );
[bd4f2e9]1248 for ( Alternative& alt : finder.alternatives ) {
1249 if ( isLvalue( alt.expr ) ) {
[452747a]1250 alternatives.push_back(
[bd4f2e9]1251 Alternative{ new AddressExpr( alt.expr->clone() ), alt.env, alt.cost } );
[a32b204]1252 } // if
1253 } // for
1254 }
1255
[13deae88]1256 void AlternativeFinder::Finder::postvisit( LabelAddressExpr * expr ) {
[bd4f2e9]1257 alternatives.push_back( Alternative{ expr->clone(), env, Cost::zero } );
[5809461]1258 }
1259
[62423350]1260 Expression * restructureCast( Expression * argExpr, Type * toType ) {
[e6cee92]1261 if ( argExpr->get_result()->size() > 1 && ! toType->isVoid() && ! dynamic_cast<ReferenceType *>( toType ) ) {
1262 // Argument expression is a tuple and the target type is not void and not a reference type.
1263 // Cast each member of the tuple to its corresponding target type, producing the tuple of those
1264 // cast expressions. If there are more components of the tuple than components in the target type,
1265 // then excess components do not come out in the result expression (but UniqueExprs ensure that
1266 // side effects will still be done).
[5ccb10d]1267 if ( Tuples::maybeImpureIgnoreUnique( argExpr ) ) {
[62423350]1268 // expressions which may contain side effects require a single unique instance of the expression.
1269 argExpr = new UniqueExpr( argExpr );
1270 }
1271 std::list< Expression * > componentExprs;
1272 for ( unsigned int i = 0; i < toType->size(); i++ ) {
1273 // cast each component
1274 TupleIndexExpr * idx = new TupleIndexExpr( argExpr->clone(), i );
1275 componentExprs.push_back( restructureCast( idx, toType->getComponent( i ) ) );
1276 }
1277 delete argExpr;
1278 assert( componentExprs.size() > 0 );
1279 // produce the tuple of casts
1280 return new TupleExpr( componentExprs );
1281 } else {
1282 // handle normally
1283 return new CastExpr( argExpr, toType->clone() );
1284 }
1285 }
1286
[13deae88]1287 void AlternativeFinder::Finder::postvisit( CastExpr *castExpr ) {
[906e24d]1288 Type *& toType = castExpr->get_result();
[7933351]1289 assert( toType );
[906e24d]1290 toType = resolveTypeof( toType, indexer );
1291 SymTab::validateType( toType, &indexer );
1292 adjustExprType( toType, env, indexer );
[a32b204]1293
1294 AlternativeFinder finder( indexer, env );
[7933351]1295 finder.targetType = toType;
[a32b204]1296 finder.findWithAdjustment( castExpr->get_arg() );
1297
1298 AltList candidates;
[452747a]1299 for ( Alternative & alt : finder.alternatives ) {
[a32b204]1300 AssertionSet needAssertions, haveAssertions;
1301 OpenVarSet openVars;
1302
1303 // It's possible that a cast can throw away some values in a multiply-valued expression. (An example is a
1304 // cast-to-void, which casts from one value to zero.) Figure out the prefix of the subexpression results
1305 // that are cast directly. The candidate is invalid if it has fewer results than there are types to cast
1306 // to.
[bd4f2e9]1307 int discardedValues = alt.expr->get_result()->size() - castExpr->get_result()->size();
[a32b204]1308 if ( discardedValues < 0 ) continue;
[7933351]1309 // xxx - may need to go into tuple types and extract relevant types and use unifyList. Note that currently, this does not
1310 // allow casting a tuple to an atomic type (e.g. (int)([1, 2, 3]))
[adcdd2f]1311 // unification run for side-effects
[452747a]1312 unify( castExpr->get_result(), alt.expr->get_result(), alt.env, needAssertions,
[bd4f2e9]1313 haveAssertions, openVars, indexer );
[452747a]1314 Cost thisCost = castCost( alt.expr->get_result(), castExpr->get_result(), indexer,
[bd4f2e9]1315 alt.env );
[7e4c4f4]1316 PRINT(
1317 std::cerr << "working on cast with result: " << castExpr->result << std::endl;
[452747a]1318 std::cerr << "and expr type: " << alt.expr->result << std::endl;
1319 std::cerr << "env: " << alt.env << std::endl;
[7e4c4f4]1320 )
[a32b204]1321 if ( thisCost != Cost::infinity ) {
[7e4c4f4]1322 PRINT(
1323 std::cerr << "has finite cost." << std::endl;
1324 )
[a32b204]1325 // count one safe conversion for each value that is thrown away
[89be1c68]1326 thisCost.incSafe( discardedValues );
[452747a]1327 Alternative newAlt( restructureCast( alt.expr->clone(), toType ), alt.env,
[bd4f2e9]1328 alt.cost, thisCost );
[452747a]1329 inferParameters( needAssertions, haveAssertions, newAlt, openVars,
[bd4f2e9]1330 back_inserter( candidates ) );
[a32b204]1331 } // if
1332 } // for
1333
1334 // findMinCost selects the alternatives with the lowest "cost" members, but has the side effect of copying the
1335 // cvtCost member to the cost member (since the old cost is now irrelevant). Thus, calling findMinCost twice
1336 // selects first based on argument cost, then on conversion cost.
1337 AltList minArgCost;
1338 findMinCost( candidates.begin(), candidates.end(), std::back_inserter( minArgCost ) );
1339 findMinCost( minArgCost.begin(), minArgCost.end(), std::back_inserter( alternatives ) );
1340 }
1341
[13deae88]1342 void AlternativeFinder::Finder::postvisit( VirtualCastExpr * castExpr ) {
[a5f0529]1343 assertf( castExpr->get_result(), "Implicate virtual cast targets not yet supported." );
1344 AlternativeFinder finder( indexer, env );
1345 // don't prune here, since it's guaranteed all alternatives will have the same type
[4e66a18]1346 finder.findWithoutPrune( castExpr->get_arg() );
[a5f0529]1347 for ( Alternative & alt : finder.alternatives ) {
1348 alternatives.push_back( Alternative(
1349 new VirtualCastExpr( alt.expr->clone(), castExpr->get_result()->clone() ),
1350 alt.env, alt.cost ) );
1351 }
1352 }
1353
[13deae88]1354 void AlternativeFinder::Finder::postvisit( UntypedMemberExpr *memberExpr ) {
[a32b204]1355 AlternativeFinder funcFinder( indexer, env );
1356 funcFinder.findWithAdjustment( memberExpr->get_aggregate() );
1357 for ( AltList::const_iterator agg = funcFinder.alternatives.begin(); agg != funcFinder.alternatives.end(); ++agg ) {
[a61ad31]1358 // it's okay for the aggregate expression to have reference type -- cast it to the base type to treat the aggregate as the referenced value
1359 std::unique_ptr<Expression> aggrExpr( agg->expr->clone() );
1360 Type * aggrType = aggrExpr->get_result();
1361 if ( dynamic_cast< ReferenceType * >( aggrType ) ) {
1362 aggrType = aggrType->stripReferences();
1363 aggrExpr.reset( new CastExpr( aggrExpr.release(), aggrType->clone() ) );
1364 }
1365 // find member of the given type
1366 if ( StructInstType *structInst = dynamic_cast< StructInstType* >( aggrExpr->get_result() ) ) {
1367 addAggMembers( structInst, aggrExpr.get(), agg->cost, agg->env, memberExpr->get_member() );
1368 } else if ( UnionInstType *unionInst = dynamic_cast< UnionInstType* >( aggrExpr->get_result() ) ) {
1369 addAggMembers( unionInst, aggrExpr.get(), agg->cost, agg->env, memberExpr->get_member() );
1370 } else if ( TupleType * tupleType = dynamic_cast< TupleType * >( aggrExpr->get_result() ) ) {
1371 addTupleMembers( tupleType, aggrExpr.get(), agg->cost, agg->env, memberExpr->get_member() );
[a32b204]1372 } // if
1373 } // for
1374 }
1375
[13deae88]1376 void AlternativeFinder::Finder::postvisit( MemberExpr *memberExpr ) {
[a32b204]1377 alternatives.push_back( Alternative( memberExpr->clone(), env, Cost::zero ) );
1378 }
1379
[13deae88]1380 void AlternativeFinder::Finder::postvisit( NameExpr *nameExpr ) {
[a40d503]1381 std::list< SymTab::Indexer::IdData > declList;
[a32b204]1382 indexer.lookupId( nameExpr->get_name(), declList );
1383 PRINT( std::cerr << "nameExpr is " << nameExpr->get_name() << std::endl; )
[a40d503]1384 for ( auto & data : declList ) {
1385 Expression * newExpr = data.combine();
1386 // xxx - add in extra cost for with-statement exprs?
1387 alternatives.push_back( Alternative( newExpr, env, Cost::zero ) );
[0f19d763]1388 PRINT(
1389 std::cerr << "decl is ";
[a40d503]1390 data.id->print( std::cerr );
[0f19d763]1391 std::cerr << std::endl;
1392 std::cerr << "newExpr is ";
[a40d503]1393 newExpr->print( std::cerr );
[0f19d763]1394 std::cerr << std::endl;
[7c64920]1395 )
[0f19d763]1396 renameTypes( alternatives.back().expr );
[4b0f997]1397 addAnonConversions( alternatives.back() ); // add anonymous member interpretations whenever an aggregate value type is seen as a name expression.
[0f19d763]1398 } // for
[a32b204]1399 }
1400
[13deae88]1401 void AlternativeFinder::Finder::postvisit( VariableExpr *variableExpr ) {
[85517ddb]1402 // not sufficient to clone here, because variable's type may have changed
1403 // since the VariableExpr was originally created.
1404 alternatives.push_back( Alternative( new VariableExpr( variableExpr->get_var() ), env, Cost::zero ) );
[a32b204]1405 }
1406
[13deae88]1407 void AlternativeFinder::Finder::postvisit( ConstantExpr *constantExpr ) {
[a32b204]1408 alternatives.push_back( Alternative( constantExpr->clone(), env, Cost::zero ) );
1409 }
1410
[13deae88]1411 void AlternativeFinder::Finder::postvisit( SizeofExpr *sizeofExpr ) {
[a32b204]1412 if ( sizeofExpr->get_isType() ) {
[322b97e]1413 Type * newType = sizeofExpr->get_type()->clone();
1414 alternatives.push_back( Alternative( new SizeofExpr( resolveTypeof( newType, indexer ) ), env, Cost::zero ) );
[a32b204]1415 } else {
1416 // find all alternatives for the argument to sizeof
1417 AlternativeFinder finder( indexer, env );
1418 finder.find( sizeofExpr->get_expr() );
1419 // find the lowest cost alternative among the alternatives, otherwise ambiguous
1420 AltList winners;
1421 findMinCost( finder.alternatives.begin(), finder.alternatives.end(), back_inserter( winners ) );
1422 if ( winners.size() != 1 ) {
1423 throw SemanticError( "Ambiguous expression in sizeof operand: ", sizeofExpr->get_expr() );
1424 } // if
1425 // return the lowest cost alternative for the argument
1426 Alternative &choice = winners.front();
[b37dba0]1427 referenceToRvalueConversion( choice.expr );
[a32b204]1428 alternatives.push_back( Alternative( new SizeofExpr( choice.expr->clone() ), choice.env, Cost::zero ) );
[47534159]1429 } // if
1430 }
1431
[13deae88]1432 void AlternativeFinder::Finder::postvisit( AlignofExpr *alignofExpr ) {
[47534159]1433 if ( alignofExpr->get_isType() ) {
[322b97e]1434 Type * newType = alignofExpr->get_type()->clone();
1435 alternatives.push_back( Alternative( new AlignofExpr( resolveTypeof( newType, indexer ) ), env, Cost::zero ) );
[47534159]1436 } else {
1437 // find all alternatives for the argument to sizeof
1438 AlternativeFinder finder( indexer, env );
1439 finder.find( alignofExpr->get_expr() );
1440 // find the lowest cost alternative among the alternatives, otherwise ambiguous
1441 AltList winners;
1442 findMinCost( finder.alternatives.begin(), finder.alternatives.end(), back_inserter( winners ) );
1443 if ( winners.size() != 1 ) {
1444 throw SemanticError( "Ambiguous expression in alignof operand: ", alignofExpr->get_expr() );
1445 } // if
1446 // return the lowest cost alternative for the argument
1447 Alternative &choice = winners.front();
[b37dba0]1448 referenceToRvalueConversion( choice.expr );
[47534159]1449 alternatives.push_back( Alternative( new AlignofExpr( choice.expr->clone() ), choice.env, Cost::zero ) );
[a32b204]1450 } // if
1451 }
1452
[2a4b088]1453 template< typename StructOrUnionType >
[13deae88]1454 void AlternativeFinder::Finder::addOffsetof( StructOrUnionType *aggInst, const std::string &name ) {
[2a4b088]1455 std::list< Declaration* > members;
1456 aggInst->lookup( name, members );
1457 for ( std::list< Declaration* >::const_iterator i = members.begin(); i != members.end(); ++i ) {
1458 if ( DeclarationWithType *dwt = dynamic_cast< DeclarationWithType* >( *i ) ) {
[79970ed]1459 alternatives.push_back( Alternative( new OffsetofExpr( aggInst->clone(), dwt ), env, Cost::zero ) );
[2a4b088]1460 renameTypes( alternatives.back().expr );
1461 } else {
1462 assert( false );
1463 }
1464 }
1465 }
[6ed1d4b]1466
[13deae88]1467 void AlternativeFinder::Finder::postvisit( UntypedOffsetofExpr *offsetofExpr ) {
[2a4b088]1468 AlternativeFinder funcFinder( indexer, env );
[85517ddb]1469 // xxx - resolveTypeof?
[2a4b088]1470 if ( StructInstType *structInst = dynamic_cast< StructInstType* >( offsetofExpr->get_type() ) ) {
1471 addOffsetof( structInst, offsetofExpr->get_member() );
1472 } else if ( UnionInstType *unionInst = dynamic_cast< UnionInstType* >( offsetofExpr->get_type() ) ) {
1473 addOffsetof( unionInst, offsetofExpr->get_member() );
1474 }
1475 }
[6ed1d4b]1476
[13deae88]1477 void AlternativeFinder::Finder::postvisit( OffsetofExpr *offsetofExpr ) {
[25a054f]1478 alternatives.push_back( Alternative( offsetofExpr->clone(), env, Cost::zero ) );
[afc1045]1479 }
1480
[13deae88]1481 void AlternativeFinder::Finder::postvisit( OffsetPackExpr *offsetPackExpr ) {
[afc1045]1482 alternatives.push_back( Alternative( offsetPackExpr->clone(), env, Cost::zero ) );
[25a054f]1483 }
1484
[a40d503]1485 namespace {
1486 void resolveAttr( SymTab::Indexer::IdData data, FunctionType *function, Type *argType, const TypeEnvironment &env, AlternativeFinder & finder ) {
1487 // assume no polymorphism
1488 // assume no implicit conversions
1489 assert( function->get_parameters().size() == 1 );
1490 PRINT(
1491 std::cerr << "resolvAttr: funcDecl is ";
1492 data.id->print( std::cerr );
1493 std::cerr << " argType is ";
1494 argType->print( std::cerr );
1495 std::cerr << std::endl;
1496 )
1497 const SymTab::Indexer & indexer = finder.get_indexer();
1498 AltList & alternatives = finder.get_alternatives();
1499 if ( typesCompatibleIgnoreQualifiers( argType, function->get_parameters().front()->get_type(), indexer, env ) ) {
1500 alternatives.push_back( Alternative( new AttrExpr( data.combine(), argType->clone() ), env, Cost::zero ) );
1501 for ( DeclarationWithType * retVal : function->returnVals ) {
1502 alternatives.back().expr->result = retVal->get_type()->clone();
1503 } // for
1504 } // if
1505 }
[a32b204]1506 }
1507
[13deae88]1508 void AlternativeFinder::Finder::postvisit( AttrExpr *attrExpr ) {
[a32b204]1509 // assume no 'pointer-to-attribute'
1510 NameExpr *nameExpr = dynamic_cast< NameExpr* >( attrExpr->get_attr() );
1511 assert( nameExpr );
[a40d503]1512 std::list< SymTab::Indexer::IdData > attrList;
[a32b204]1513 indexer.lookupId( nameExpr->get_name(), attrList );
1514 if ( attrExpr->get_isType() || attrExpr->get_expr() ) {
[a40d503]1515 for ( auto & data : attrList ) {
1516 DeclarationWithType * id = data.id;
[a32b204]1517 // check if the type is function
[a40d503]1518 if ( FunctionType *function = dynamic_cast< FunctionType* >( id->get_type() ) ) {
[a32b204]1519 // assume exactly one parameter
1520 if ( function->get_parameters().size() == 1 ) {
1521 if ( attrExpr->get_isType() ) {
[13deae88]1522 resolveAttr( data, function, attrExpr->get_type(), env, altFinder);
[a32b204]1523 } else {
1524 AlternativeFinder finder( indexer, env );
1525 finder.find( attrExpr->get_expr() );
1526 for ( AltList::iterator choice = finder.alternatives.begin(); choice != finder.alternatives.end(); ++choice ) {
[906e24d]1527 if ( choice->expr->get_result()->size() == 1 ) {
[13deae88]1528 resolveAttr(data, function, choice->expr->get_result(), choice->env, altFinder );
[a32b204]1529 } // fi
1530 } // for
1531 } // if
1532 } // if
1533 } // if
1534 } // for
1535 } else {
[a40d503]1536 for ( auto & data : attrList ) {
1537 alternatives.push_back( Alternative( data.combine(), env, Cost::zero ) );
[a32b204]1538 renameTypes( alternatives.back().expr );
1539 } // for
1540 } // if
1541 }
1542
[13deae88]1543 void AlternativeFinder::Finder::postvisit( LogicalExpr *logicalExpr ) {
[a32b204]1544 AlternativeFinder firstFinder( indexer, env );
1545 firstFinder.findWithAdjustment( logicalExpr->get_arg1() );
1546 for ( AltList::const_iterator first = firstFinder.alternatives.begin(); first != firstFinder.alternatives.end(); ++first ) {
1547 AlternativeFinder secondFinder( indexer, first->env );
1548 secondFinder.findWithAdjustment( logicalExpr->get_arg2() );
1549 for ( AltList::const_iterator second = secondFinder.alternatives.begin(); second != secondFinder.alternatives.end(); ++second ) {
1550 LogicalExpr *newExpr = new LogicalExpr( first->expr->clone(), second->expr->clone(), logicalExpr->get_isAnd() );
1551 alternatives.push_back( Alternative( newExpr, second->env, first->cost + second->cost ) );
[d9a0e76]1552 }
1553 }
1554 }
[51b73452]1555
[13deae88]1556 void AlternativeFinder::Finder::postvisit( ConditionalExpr *conditionalExpr ) {
[32b8144]1557 // find alternatives for condition
[a32b204]1558 AlternativeFinder firstFinder( indexer, env );
1559 firstFinder.findWithAdjustment( conditionalExpr->get_arg1() );
1560 for ( AltList::const_iterator first = firstFinder.alternatives.begin(); first != firstFinder.alternatives.end(); ++first ) {
[32b8144]1561 // find alternatives for true expression
[a32b204]1562 AlternativeFinder secondFinder( indexer, first->env );
1563 secondFinder.findWithAdjustment( conditionalExpr->get_arg2() );
1564 for ( AltList::const_iterator second = secondFinder.alternatives.begin(); second != secondFinder.alternatives.end(); ++second ) {
[32b8144]1565 // find alterantives for false expression
[a32b204]1566 AlternativeFinder thirdFinder( indexer, second->env );
1567 thirdFinder.findWithAdjustment( conditionalExpr->get_arg3() );
1568 for ( AltList::const_iterator third = thirdFinder.alternatives.begin(); third != thirdFinder.alternatives.end(); ++third ) {
[32b8144]1569 // unify true and false types, then infer parameters to produce new alternatives
[a32b204]1570 OpenVarSet openVars;
1571 AssertionSet needAssertions, haveAssertions;
1572 Alternative newAlt( 0, third->env, first->cost + second->cost + third->cost );
[668e971a]1573 Type* commonType = nullptr;
[906e24d]1574 if ( unify( second->expr->get_result(), third->expr->get_result(), newAlt.env, needAssertions, haveAssertions, openVars, indexer, commonType ) ) {
[a32b204]1575 ConditionalExpr *newExpr = new ConditionalExpr( first->expr->clone(), second->expr->clone(), third->expr->clone() );
[906e24d]1576 newExpr->set_result( commonType ? commonType : second->expr->get_result()->clone() );
[ddf8a29]1577 // convert both options to the conditional result type
1578 newAlt.cost += computeExpressionConversionCost( newExpr->arg2, newExpr->result, indexer, newAlt.env );
1579 newAlt.cost += computeExpressionConversionCost( newExpr->arg3, newExpr->result, indexer, newAlt.env );
[a32b204]1580 newAlt.expr = newExpr;
1581 inferParameters( needAssertions, haveAssertions, newAlt, openVars, back_inserter( alternatives ) );
1582 } // if
1583 } // for
1584 } // for
1585 } // for
1586 }
1587
[13deae88]1588 void AlternativeFinder::Finder::postvisit( CommaExpr *commaExpr ) {
[a32b204]1589 TypeEnvironment newEnv( env );
1590 Expression *newFirstArg = resolveInVoidContext( commaExpr->get_arg1(), indexer, newEnv );
1591 AlternativeFinder secondFinder( indexer, newEnv );
1592 secondFinder.findWithAdjustment( commaExpr->get_arg2() );
1593 for ( AltList::const_iterator alt = secondFinder.alternatives.begin(); alt != secondFinder.alternatives.end(); ++alt ) {
1594 alternatives.push_back( Alternative( new CommaExpr( newFirstArg->clone(), alt->expr->clone() ), alt->env, alt->cost ) );
1595 } // for
1596 delete newFirstArg;
1597 }
1598
[13deae88]1599 void AlternativeFinder::Finder::postvisit( RangeExpr * rangeExpr ) {
[32b8144]1600 // resolve low and high, accept alternatives whose low and high types unify
1601 AlternativeFinder firstFinder( indexer, env );
1602 firstFinder.findWithAdjustment( rangeExpr->get_low() );
1603 for ( AltList::const_iterator first = firstFinder.alternatives.begin(); first != firstFinder.alternatives.end(); ++first ) {
1604 AlternativeFinder secondFinder( indexer, first->env );
1605 secondFinder.findWithAdjustment( rangeExpr->get_high() );
1606 for ( AltList::const_iterator second = secondFinder.alternatives.begin(); second != secondFinder.alternatives.end(); ++second ) {
1607 OpenVarSet openVars;
1608 AssertionSet needAssertions, haveAssertions;
1609 Alternative newAlt( 0, second->env, first->cost + second->cost );
1610 Type* commonType = nullptr;
1611 if ( unify( first->expr->get_result(), second->expr->get_result(), newAlt.env, needAssertions, haveAssertions, openVars, indexer, commonType ) ) {
1612 RangeExpr *newExpr = new RangeExpr( first->expr->clone(), second->expr->clone() );
1613 newExpr->set_result( commonType ? commonType : first->expr->get_result()->clone() );
1614 newAlt.expr = newExpr;
1615 inferParameters( needAssertions, haveAssertions, newAlt, openVars, back_inserter( alternatives ) );
1616 } // if
1617 } // for
1618 } // for
1619 }
1620
[13deae88]1621 void AlternativeFinder::Finder::postvisit( UntypedTupleExpr *tupleExpr ) {
[bd4f2e9]1622 std::vector< AlternativeFinder > subExprAlternatives;
[13deae88]1623 altFinder.findSubExprs( tupleExpr->get_exprs().begin(), tupleExpr->get_exprs().end(),
[bd4f2e9]1624 back_inserter( subExprAlternatives ) );
1625 std::vector< AltList > possibilities;
[452747a]1626 combos( subExprAlternatives.begin(), subExprAlternatives.end(),
[bd4f2e9]1627 back_inserter( possibilities ) );
1628 for ( const AltList& alts : possibilities ) {
[907eccb]1629 std::list< Expression * > exprs;
[bd4f2e9]1630 makeExprList( alts, exprs );
[a32b204]1631
1632 TypeEnvironment compositeEnv;
[bd4f2e9]1633 simpleCombineEnvironments( alts.begin(), alts.end(), compositeEnv );
[452747a]1634 alternatives.push_back(
[bd4f2e9]1635 Alternative{ new TupleExpr( exprs ), compositeEnv, sumCost( alts ) } );
[a32b204]1636 } // for
[d9a0e76]1637 }
[dc2e7e0]1638
[13deae88]1639 void AlternativeFinder::Finder::postvisit( TupleExpr *tupleExpr ) {
[907eccb]1640 alternatives.push_back( Alternative( tupleExpr->clone(), env, Cost::zero ) );
1641 }
1642
[13deae88]1643 void AlternativeFinder::Finder::postvisit( ImplicitCopyCtorExpr * impCpCtorExpr ) {
[dc2e7e0]1644 alternatives.push_back( Alternative( impCpCtorExpr->clone(), env, Cost::zero ) );
1645 }
[b6fe7e6]1646
[13deae88]1647 void AlternativeFinder::Finder::postvisit( ConstructorExpr * ctorExpr ) {
[b6fe7e6]1648 AlternativeFinder finder( indexer, env );
1649 // don't prune here, since it's guaranteed all alternatives will have the same type
1650 // (giving the alternatives different types is half of the point of ConstructorExpr nodes)
[4e66a18]1651 finder.findWithoutPrune( ctorExpr->get_callExpr() );
[b6fe7e6]1652 for ( Alternative & alt : finder.alternatives ) {
1653 alternatives.push_back( Alternative( new ConstructorExpr( alt.expr->clone() ), alt.env, alt.cost ) );
1654 }
1655 }
[8f7cea1]1656
[13deae88]1657 void AlternativeFinder::Finder::postvisit( TupleIndexExpr *tupleExpr ) {
[8f7cea1]1658 alternatives.push_back( Alternative( tupleExpr->clone(), env, Cost::zero ) );
1659 }
[aa8f9df]1660
[13deae88]1661 void AlternativeFinder::Finder::postvisit( TupleAssignExpr *tupleAssignExpr ) {
[aa8f9df]1662 alternatives.push_back( Alternative( tupleAssignExpr->clone(), env, Cost::zero ) );
1663 }
[bf32bb8]1664
[13deae88]1665 void AlternativeFinder::Finder::postvisit( UniqueExpr *unqExpr ) {
[bf32bb8]1666 AlternativeFinder finder( indexer, env );
1667 finder.findWithAdjustment( unqExpr->get_expr() );
1668 for ( Alternative & alt : finder.alternatives ) {
[141b786]1669 // ensure that the id is passed on to the UniqueExpr alternative so that the expressions are "linked"
[77971f6]1670 UniqueExpr * newUnqExpr = new UniqueExpr( alt.expr->clone(), unqExpr->get_id() );
[141b786]1671 alternatives.push_back( Alternative( newUnqExpr, alt.env, alt.cost ) );
[bf32bb8]1672 }
1673 }
1674
[13deae88]1675 void AlternativeFinder::Finder::postvisit( StmtExpr *stmtExpr ) {
[722617d]1676 StmtExpr * newStmtExpr = stmtExpr->clone();
1677 ResolvExpr::resolveStmtExpr( newStmtExpr, indexer );
1678 // xxx - this env is almost certainly wrong, and needs to somehow contain the combined environments from all of the statements in the stmtExpr...
1679 alternatives.push_back( Alternative( newStmtExpr, env, Cost::zero ) );
1680 }
1681
[13deae88]1682 void AlternativeFinder::Finder::postvisit( UntypedInitExpr *initExpr ) {
[62423350]1683 // handle each option like a cast
[e4d829b]1684 AltList candidates;
[13deae88]1685 PRINT(
1686 std::cerr << "untyped init expr: " << initExpr << std::endl;
1687 )
[e4d829b]1688 // O(N^2) checks of d-types with e-types
[62423350]1689 for ( InitAlternative & initAlt : initExpr->get_initAlts() ) {
[228099e]1690 Type * toType = resolveTypeof( initAlt.type->clone(), indexer );
[62423350]1691 SymTab::validateType( toType, &indexer );
1692 adjustExprType( toType, env, indexer );
1693 // Ideally the call to findWithAdjustment could be moved out of the loop, but unfortunately it currently has to occur inside or else
1694 // polymorphic return types are not properly bound to the initialization type, since return type variables are only open for the duration of resolving
1695 // the UntypedExpr. This is only actually an issue in initialization contexts that allow more than one possible initialization type, but it is still suboptimal.
1696 AlternativeFinder finder( indexer, env );
1697 finder.targetType = toType;
1698 finder.findWithAdjustment( initExpr->get_expr() );
1699 for ( Alternative & alt : finder.get_alternatives() ) {
1700 TypeEnvironment newEnv( alt.env );
[e4d829b]1701 AssertionSet needAssertions, haveAssertions;
[62423350]1702 OpenVarSet openVars; // find things in env that don't have a "representative type" and claim those are open vars?
[13deae88]1703 PRINT(
1704 std::cerr << " @ " << toType << " " << initAlt.designation << std::endl;
1705 )
[e4d829b]1706 // It's possible that a cast can throw away some values in a multiply-valued expression. (An example is a
1707 // cast-to-void, which casts from one value to zero.) Figure out the prefix of the subexpression results
1708 // that are cast directly. The candidate is invalid if it has fewer results than there are types to cast
1709 // to.
[62423350]1710 int discardedValues = alt.expr->get_result()->size() - toType->size();
[e4d829b]1711 if ( discardedValues < 0 ) continue;
1712 // xxx - may need to go into tuple types and extract relevant types and use unifyList. Note that currently, this does not
1713 // allow casting a tuple to an atomic type (e.g. (int)([1, 2, 3]))
1714 // unification run for side-effects
[62423350]1715 unify( toType, alt.expr->get_result(), newEnv, needAssertions, haveAssertions, openVars, indexer ); // xxx - do some inspecting on this line... why isn't result bound to initAlt.type??
[e4d829b]1716
[62423350]1717 Cost thisCost = castCost( alt.expr->get_result(), toType, indexer, newEnv );
[e4d829b]1718 if ( thisCost != Cost::infinity ) {
1719 // count one safe conversion for each value that is thrown away
[89be1c68]1720 thisCost.incSafe( discardedValues );
[bb666f64]1721 Alternative newAlt( new InitExpr( restructureCast( alt.expr->clone(), toType ), initAlt.designation->clone() ), newEnv, alt.cost, thisCost );
1722 inferParameters( needAssertions, haveAssertions, newAlt, openVars, back_inserter( candidates ) );
[e4d829b]1723 }
1724 }
1725 }
1726
1727 // findMinCost selects the alternatives with the lowest "cost" members, but has the side effect of copying the
1728 // cvtCost member to the cost member (since the old cost is now irrelevant). Thus, calling findMinCost twice
1729 // selects first based on argument cost, then on conversion cost.
1730 AltList minArgCost;
1731 findMinCost( candidates.begin(), candidates.end(), std::back_inserter( minArgCost ) );
1732 findMinCost( minArgCost.begin(), minArgCost.end(), std::back_inserter( alternatives ) );
1733 }
[51b73452]1734} // namespace ResolvExpr
[a32b204]1735
1736// Local Variables: //
1737// tab-width: 4 //
1738// mode: c++ //
1739// compile-command: "make install" //
1740// End: //
Note: See TracBrowser for help on using the repository browser.