- File:
-
- 1 edited
-
src/ResolvExpr/SatisfyAssertions.cpp (modified) (17 diffs)
Legend:
- Unmodified
- Added
- Removed
-
src/ResolvExpr/SatisfyAssertions.cpp
r251ce80 r34b4268 16 16 #include "SatisfyAssertions.hpp" 17 17 18 #include <iostream> 18 19 #include <algorithm> 19 20 #include <cassert> … … 23 24 #include <vector> 24 25 25 #include "AdjustExprType.hpp"26 26 #include "Candidate.hpp" 27 27 #include "CandidateFinder.hpp" 28 #include "CommonType.hpp"29 28 #include "Cost.h" 30 29 #include "RenameVars.h" 31 #include "SpecCost.hpp"32 30 #include "typeops.h" 33 31 #include "Unify.h" … … 45 43 #include "SymTab/Mangler.h" 46 44 45 46 47 47 namespace ResolvExpr { 48 48 … … 65 65 ast::AssertionSet && h, ast::AssertionSet && n, ast::OpenVarSet && o, ast::UniqueId rs ) 66 66 : cdata( c ), adjType( at ), env( std::move( e ) ), have( std::move( h ) ), 67 need( std::move( n ) ), open( std::move( o ) ), resnSlot( rs ) {} 67 need( std::move( n ) ), open( std::move( o ) ), resnSlot( rs ) { 68 if (!have.empty()) { 69 std::cerr << c.id->location << ':' << c.id->name << std::endl; 70 } 71 } 68 72 }; 69 73 … … 139 143 }; 140 144 141 /// Adds a captured assertion to the symbol table 142 void addToSymbolTable( const ast::AssertionSet & have, ast::SymbolTable & symtab ) { 143 for ( auto & i : have ) { 144 if ( i.second.isUsed ) { symtab.addId( i.first->var ); } 145 } 146 } 145 enum AssertionResult {Fail, Skip, Success} ; 147 146 148 147 /// Binds a single assertion, updating satisfaction state … … 155 154 "Assertion candidate does not have a unique ID: %s", toString( candidate ).c_str() ); 156 155 157 ast::Expr * varExpr = match.cdata.combine( cand->expr->location, cand->c vtCost );156 ast::Expr * varExpr = match.cdata.combine( cand->expr->location, cand->cost ); 158 157 varExpr->result = match.adjType; 159 158 if ( match.resnSlot ) { varExpr->inferred.resnSlots().emplace_back( match.resnSlot ); } … … 165 164 166 165 /// Satisfy a single assertion 167 bool satisfyAssertion( ast::AssertionList::value_type & assn, SatState & sat, bool allowConversion = false, bool skipUnbound = false) {166 AssertionResult satisfyAssertion( ast::AssertionList::value_type & assn, SatState & sat, bool skipUnbound = false) { 168 167 // skip unused assertions 169 if ( ! assn.second.isUsed ) return true; 168 static unsigned int cnt = 0; 169 if ( ! assn.second.isUsed ) return AssertionResult::Success; 170 171 if (assn.first->var->name[1] == '|') std::cerr << ++cnt << std::endl; 170 172 171 173 // find candidates that unify with the desired type 172 AssnCandidateList matches ;174 AssnCandidateList matches, inexactMatches; 173 175 174 176 std::vector<ast::SymbolTable::IdData> candidates; … … 184 186 if (thisArgType.as<ast::PointerType>()) otypeKey = Mangle::Encoding::pointer; 185 187 else if (!isUnboundType(thisArgType)) otypeKey = Mangle::mangle(thisArgType, Mangle::Type | Mangle::NoGenericParams); 186 else if (skipUnbound) return false;188 else if (skipUnbound) return AssertionResult::Skip; 187 189 188 190 candidates = sat.symtab.specialLookupId(kind, otypeKey); … … 212 214 213 215 ast::OpenVarSet closed; 214 findOpenVars( toType, newOpen, closed, newNeed, have, FirstClosed ); 215 findOpenVars( adjType, newOpen, closed, newNeed, have, FirstOpen ); 216 if ( allowConversion ) { 217 if ( auto c = commonType( toType, adjType, newEnv, newNeed, have, newOpen, WidenMode {true, true} ) ) { 216 // findOpenVars( toType, newOpen, closed, newNeed, have, FirstClosed ); 217 findOpenVars( adjType, newOpen, closed, newNeed, have, newEnv, FirstOpen ); 218 ast::TypeEnvironment tempNewEnv {newEnv}; 219 220 if ( unifyExact( toType, adjType, tempNewEnv, newNeed, have, newOpen, WidenMode {true, true}, sat.symtab ) ) { 221 // set up binding slot for recursive assertions 222 ast::UniqueId crntResnSlot = 0; 223 if ( ! newNeed.empty() ) { 224 crntResnSlot = ++globalResnSlot; 225 for ( auto & a : newNeed ) { a.second.resnSlot = crntResnSlot; } 226 } 227 228 matches.emplace_back( 229 cdata, adjType, std::move( tempNewEnv ), std::move( have ), std::move( newNeed ), 230 std::move( newOpen ), crntResnSlot ); 231 } 232 else if ( matches.empty() ) { 233 // restore invalidated env 234 // newEnv = sat.cand->env; 235 // newNeed.clear(); 236 if ( auto c = commonType( toType, adjType, newEnv, newNeed, have, newOpen, WidenMode {true, true}, sat.symtab ) ) { 218 237 // set up binding slot for recursive assertions 219 238 ast::UniqueId crntResnSlot = 0; … … 223 242 } 224 243 225 matches.emplace_back(244 inexactMatches.emplace_back( 226 245 cdata, adjType, std::move( newEnv ), std::move( have ), std::move( newNeed ), 227 246 std::move( newOpen ), crntResnSlot ); 228 247 } 229 248 } 230 else {231 if ( unifyExact( toType, adjType, newEnv, newNeed, have, newOpen, WidenMode {true, true} ) ) {232 // set up binding slot for recursive assertions233 ast::UniqueId crntResnSlot = 0;234 if ( ! newNeed.empty() ) {235 crntResnSlot = ++globalResnSlot;236 for ( auto & a : newNeed ) { a.second.resnSlot = crntResnSlot; }237 }238 239 matches.emplace_back(240 cdata, adjType, std::move( newEnv ), std::move( have ), std::move( newNeed ),241 std::move( newOpen ), crntResnSlot );242 }243 }244 249 } 245 250 246 251 // break if no satisfying match 247 if ( matches.empty() ) return false; 252 if ( matches.empty() ) matches = std::move(inexactMatches); 253 if ( matches.empty() ) return AssertionResult::Fail; 248 254 249 255 // defer if too many satisfying matches 250 256 if ( matches.size() > 1 ) { 251 257 sat.deferred.emplace_back( assn.first, assn.second, std::move( matches ) ); 252 return true;258 return AssertionResult::Success; 253 259 } 254 260 255 261 // otherwise bind unique match in ongoing scope 256 262 AssnCandidate & match = matches.front(); 257 addToSymbolTable( match.have, sat.symtab );263 // addToSymbolTable( match.have, sat.symtab ); 258 264 sat.newNeed.insert( match.need.begin(), match.need.end() ); 259 265 sat.cand->env = std::move( match.env ); … … 261 267 262 268 bindAssertion( assn.first, assn.second, sat.cand, match, sat.inferred ); 263 return true;269 return AssertionResult::Success; 264 270 } 265 271 … … 392 398 mergeOpenVars( open, i.match.open ); 393 399 394 if ( ! env.combine( i.match.env, open ) ) return false;400 if ( ! env.combine( i.match.env, open, symtab ) ) return false; 395 401 396 402 crnt.emplace_back( i ); … … 438 444 // for each current mutually-compatible set of assertions 439 445 for ( SatState & sat : sats ) { 440 bool allowConversion = false;441 446 // stop this branch if a better option is already found 442 447 auto it = thresholds.find( pruneKey( *sat.cand ) ); … … 447 452 for (unsigned resetCount = 0; ; ++resetCount) { 448 453 ast::AssertionList next; 449 resetTyVarRenaming();450 454 // make initial pass at matching assertions 451 455 for ( auto & assn : sat.need ) { 456 resetTyVarRenaming(); 452 457 // fail early if any assertion is not satisfiable 453 if ( ! satisfyAssertion( assn, sat, allowConversion, !next.empty() ) ) { 454 next.emplace_back(assn); 455 // goto nextSat; 456 } 457 } 458 // success 459 if (next.empty()) break; 460 // fail if nothing resolves 461 else if (next.size() == sat.need.size()) { 462 if (allowConversion) { 458 auto result = satisfyAssertion( assn, sat, !next.empty() ); 459 if ( result == AssertionResult::Fail ) { 463 460 Indenter tabs{ 3 }; 464 461 std::ostringstream ss; … … 466 463 print( ss, *sat.cand, ++tabs ); 467 464 ss << (tabs-1) << "Could not satisfy assertion:\n"; 468 ast::print( ss, next[0].first, tabs );465 ast::print( ss, assn.first, tabs ); 469 466 470 467 errors.emplace_back( ss.str() ); 471 468 goto nextSat; 472 469 } 473 474 else { 475 allowConversion = true; 476 continue; 477 } 478 } 479 allowConversion = false; 470 else if ( result == AssertionResult::Skip ) { 471 next.emplace_back(assn); 472 // goto nextSat; 473 } 474 } 475 // success 476 if (next.empty()) break; 477 480 478 sat.need = std::move(next); 481 479 } … … 531 529 sat.cand->expr, std::move( compat.env ), std::move( compat.open ), 532 530 ast::AssertionSet{} /* need moved into satisfaction state */, 533 sat.cand->cost , sat.cand->cvtCost);531 sat.cand->cost ); 534 532 535 533 ast::AssertionSet nextNewNeed{ sat.newNeed }; … … 544 542 for ( DeferRef r : compat.assns ) { 545 543 AssnCandidate match = r.match; 546 addToSymbolTable( match.have, nextSymtab );544 // addToSymbolTable( match.have, nextSymtab ); 547 545 nextNewNeed.insert( match.need.begin(), match.need.end() ); 548 546
Note:
See TracChangeset
for help on using the changeset viewer.