Ignore:
Timestamp:
Jan 7, 2021, 3:27:00 PM (5 years ago)
Author:
Thierry Delisle <tdelisle@…>
Branches:
ADT, arm-eh, ast-experimental, enum, forall-pointer-decay, jacob/cs343-translation, master, new-ast-unique-expr, pthread-emulation, qualifiedEnum
Children:
2b4daf2, 64aeca0
Parents:
3c64c668 (diff), eef8dfb (diff)
Note: this is a merge changeset, the changes displayed below correspond to the merge itself.
Use the (diff) links above to see all the changes relative to each parent.
Message:

Merge branch 'master' into park_unpark

File:
1 edited

Legend:

Unmodified
Added
Removed
  • src/ResolvExpr/SatisfyAssertions.cpp

    r3c64c668 r58fe85a  
    99// Author           : Aaron B. Moss
    1010// Created On       : Mon Jun 10 17:45:00 2019
    11 // Last Modified By : Aaron B. Moss
    12 // Last Modified On : Mon Jun 10 17:45:00 2019
    13 // Update Count     : 1
     11// Last Modified By : Andrew Beach
     12// Last Modified On : Tue Oct  1 13:56:00 2019
     13// Update Count     : 2
    1414//
    1515
     
    5757                ast::UniqueId resnSlot;          ///< Slot for any recursive assertion IDs
    5858
    59                 AssnCandidate( 
    60                         const ast::SymbolTable::IdData c, const ast::Type * at, ast::TypeEnvironment && e, 
     59                AssnCandidate(
     60                        const ast::SymbolTable::IdData c, const ast::Type * at, ast::TypeEnvironment && e,
    6161                        ast::AssertionSet && h, ast::AssertionSet && n, ast::OpenVarSet && o, ast::UniqueId rs )
    62                 : cdata( c ), adjType( at ), env( std::move( e ) ), have( std::move( h ) ), 
     62                : cdata( c ), adjType( at ), env( std::move( e ) ), have( std::move( h ) ),
    6363                  need( std::move( n ) ), open( std::move( o ) ), resnSlot( rs ) {}
    6464        };
     
    6969        /// Reference to a single deferred item
    7070        struct DeferRef {
    71                 const ast::DeclWithType * decl;
     71                const ast::VariableExpr * expr;
    7272                const ast::AssertionSetValue & info;
    7373                const AssnCandidate & match;
    7474        };
    75        
    76         /// Wrapper for the deferred items from a single assertion satisfaction. 
     75
     76        /// Wrapper for the deferred items from a single assertion satisfaction.
    7777        /// Acts like an indexed list of DeferRef
    7878        struct DeferItem {
    79                 const ast::DeclWithType * decl;
     79                const ast::VariableExpr * expr;
    8080                const ast::AssertionSetValue & info;
    8181                AssnCandidateList matches;
    8282
    83                 DeferItem( 
    84                         const ast::DeclWithType * d, const ast::AssertionSetValue & i, AssnCandidateList && ms )
    85                 : decl( d ), info( i ), matches( std::move( ms ) ) {}
     83                DeferItem(
     84                        const ast::VariableExpr * d, const ast::AssertionSetValue & i, AssnCandidateList && ms )
     85                : expr( d ), info( i ), matches( std::move( ms ) ) {}
    8686
    8787                bool empty() const { return matches.empty(); }
     
    8989                AssnCandidateList::size_type size() const { return matches.size(); }
    9090
    91                 DeferRef operator[] ( unsigned i ) const { return { decl, info, matches[i] }; }
     91                DeferRef operator[] ( unsigned i ) const { return { expr, info, matches[i] }; }
    9292        };
    9393
     
    117117                /// Initial satisfaction state for a candidate
    118118                SatState( CandidateRef & c, const ast::SymbolTable & syms )
    119                 : cand( c ), need(), newNeed(), deferred(), inferred(), costs{ Cost::zero }, 
     119                : cand( c ), need(), newNeed(), deferred(), inferred(), costs{ Cost::zero },
    120120                  symtab( syms ) { need.swap( c->need ); }
    121                
     121
    122122                /// Update satisfaction state for next step after previous state
    123123                SatState( SatState && o, IterateFlag )
    124                 : cand( std::move( o.cand ) ), need( o.newNeed.begin(), o.newNeed.end() ), newNeed(), 
    125                   deferred(), inferred( std::move( o.inferred ) ), costs( std::move( o.costs ) ), 
     124                : cand( std::move( o.cand ) ), need( o.newNeed.begin(), o.newNeed.end() ), newNeed(),
     125                  deferred(), inferred( std::move( o.inferred ) ), costs( std::move( o.costs ) ),
    126126                  symtab( o.symtab ) { costs.emplace_back( Cost::zero ); }
    127                
     127
    128128                /// Field-wise next step constructor
    129129                SatState(
    130                         CandidateRef && c, ast::AssertionSet && nn, InferCache && i, CostVec && cs, 
     130                        CandidateRef && c, ast::AssertionSet && nn, InferCache && i, CostVec && cs,
    131131                        ast::SymbolTable && syms )
    132                 : cand( std::move( c ) ), need( nn.begin(), nn.end() ), newNeed(), deferred(), 
     132                : cand( std::move( c ) ), need( nn.begin(), nn.end() ), newNeed(), deferred(),
    133133                  inferred( std::move( i ) ), costs( std::move( cs ) ), symtab( std::move( syms ) )
    134134                  { costs.emplace_back( Cost::zero ); }
     
    138138        void addToSymbolTable( const ast::AssertionSet & have, ast::SymbolTable & symtab ) {
    139139                for ( auto & i : have ) {
    140                         if ( i.second.isUsed ) { symtab.addId( i.first ); }
     140                        if ( i.second.isUsed ) { symtab.addId( i.first->var ); }
    141141                }
    142142        }
    143143
    144144        /// Binds a single assertion, updating satisfaction state
    145         void bindAssertion( 
    146                 const ast::DeclWithType * decl, const ast::AssertionSetValue & info, CandidateRef & cand,
     145        void bindAssertion(
     146                const ast::VariableExpr * expr, const ast::AssertionSetValue & info, CandidateRef & cand,
    147147                AssnCandidate & match, InferCache & inferred
    148148        ) {
    149149                const ast::DeclWithType * candidate = match.cdata.id;
    150                 assertf( candidate->uniqueId, 
     150                assertf( candidate->uniqueId,
    151151                        "Assertion candidate does not have a unique ID: %s", toString( candidate ).c_str() );
    152                
     152
    153153                ast::Expr * varExpr = match.cdata.combine( cand->expr->location, cand->cvtCost );
    154154                varExpr->result = match.adjType;
     
    156156
    157157                // place newly-inferred assertion in proper location in cache
    158                 inferred[ info.resnSlot ][ decl->uniqueId ] = ast::ParamEntry{
    159                         candidate->uniqueId, candidate, match.adjType, decl->get_type(), varExpr };
     158                inferred[ info.resnSlot ][ expr->var->uniqueId ] = ast::ParamEntry{
     159                        candidate->uniqueId, candidate, match.adjType, expr->result, varExpr };
    160160        }
    161161
     
    167167                // find candidates that unify with the desired type
    168168                AssnCandidateList matches;
    169                 for ( const ast::SymbolTable::IdData & cdata : sat.symtab.lookupId( assn.first->name ) ) {
     169
     170                std::vector<ast::SymbolTable::IdData> candidates;
     171                auto kind = ast::SymbolTable::getSpecialFunctionKind(assn.first->var->name);
     172                if (kind != ast::SymbolTable::SpecialFunctionKind::NUMBER_OF_KINDS) {
     173                        // prefilter special decls by argument type, if already known
     174                        ast::ptr<ast::Type> thisArgType = assn.first->result.strict_as<ast::PointerType>()->base
     175                                .strict_as<ast::FunctionType>()->params[0]
     176                                .strict_as<ast::ReferenceType>()->base;
     177                        sat.cand->env.apply(thisArgType);
     178
     179                        std::string otypeKey = "";
     180                        if (thisArgType.as<ast::PointerType>()) otypeKey = Mangle::Encoding::pointer;
     181                        else if (!isUnboundType(thisArgType)) otypeKey = Mangle::mangle(thisArgType, Mangle::Type | Mangle::NoGenericParams);
     182
     183                        candidates = sat.symtab.specialLookupId(kind, otypeKey);
     184                }
     185                else {
     186                        candidates = sat.symtab.lookupId(assn.first->var->name);
     187                }
     188                for ( const ast::SymbolTable::IdData & cdata : candidates ) {
    170189                        const ast::DeclWithType * candidate = cdata.id;
     190
     191                        // ignore deleted candidates.
     192                        // NOTE: this behavior is different from main resolver.
     193                        // further investigations might be needed to determine
     194                        // if we should implement the same rule here
     195                        // (i.e. error if unique best match is deleted)
     196                        if (candidate->isDeleted && candidate->linkage == ast::Linkage::AutoGen) continue;
    171197
    172198                        // build independent unification context for candidate
     
    174200                        ast::TypeEnvironment newEnv{ sat.cand->env };
    175201                        ast::OpenVarSet newOpen{ sat.cand->open };
    176                         ast::ptr< ast::Type > toType = assn.first->get_type();
    177                         ast::ptr< ast::Type > adjType = 
    178                                 renameTyVars( adjustExprType( candidate->get_type(), newEnv, sat.symtab ) );
     202                        ast::ptr< ast::Type > toType = assn.first->result;
     203                        ast::ptr< ast::Type > adjType =
     204                                renameTyVars( adjustExprType( candidate->get_type(), newEnv, sat.symtab ), GEN_USAGE, false );
    179205
    180206                        // only keep candidates which unify
     
    187213                                }
    188214
    189                                 matches.emplace_back( 
    190                                         cdata, adjType, std::move( newEnv ), std::move( newNeed ), std::move( have ),
     215                                matches.emplace_back(
     216                                        cdata, adjType, std::move( newEnv ), std::move( have ), std::move( newNeed ),
    191217                                        std::move( newOpen ), crntResnSlot );
    192218                        }
     
    229255                InferMatcher( InferCache & inferred ) : inferred( inferred ) {}
    230256
    231                 const ast::Expr * postmutate( const ast::Expr * expr ) {
     257                const ast::Expr * postvisit( const ast::Expr * expr ) {
    232258                        // Skip if no slots to find
    233                         if ( expr->inferred.mode != ast::Expr::InferUnion::Slots ) return expr;
    234 
     259                        if ( !expr->inferred.hasSlots() ) return expr;
     260                        // if ( expr->inferred.mode != ast::Expr::InferUnion::Slots ) return expr;
     261                        std::vector<UniqueId> missingSlots;
    235262                        // find inferred parameters for resolution slots
    236                         ast::InferredParams newInferred;
     263                        ast::InferredParams * newInferred = new ast::InferredParams();
    237264                        for ( UniqueId slot : expr->inferred.resnSlots() ) {
    238265                                // fail if no matching assertions found
    239266                                auto it = inferred.find( slot );
    240267                                if ( it == inferred.end() ) {
    241                                         assert(!"missing assertion");
     268                                        // std::cerr << "missing assertion " << slot << std::endl;
     269                                        missingSlots.push_back(slot);
     270                                        continue;
    242271                                }
    243272
     
    245274                                for ( auto & entry : it->second ) {
    246275                                        // recurse on inferParams of resolved expressions
    247                                         entry.second.expr = postmutate( entry.second.expr );
    248                                         auto res = newInferred.emplace( entry );
     276                                        entry.second.expr = postvisit( entry.second.expr );
     277                                        auto res = newInferred->emplace( entry );
    249278                                        assert( res.second && "all assertions newly placed" );
    250279                                }
     
    252281
    253282                        ast::Expr * ret = mutate( expr );
    254                         ret->inferred.set_inferParams( std::move( newInferred ) );
     283                        ret->inferred.set_inferParams( newInferred );
     284                        if (!missingSlots.empty()) ret->inferred.resnSlots() = missingSlots;
    255285                        return ret;
    256286                }
    257287        };
    258288
    259         /// Replace ResnSlots with InferParams and add alternative to output list, if it meets pruning 
     289        /// Replace ResnSlots with InferParams and add alternative to output list, if it meets pruning
    260290        /// threshold.
    261         void finalizeAssertions( 
    262                 CandidateRef & cand, InferCache & inferred, PruneMap & thresholds, CostVec && costs, 
    263                 CandidateList & out 
     291        void finalizeAssertions(
     292                CandidateRef & cand, InferCache & inferred, PruneMap & thresholds, CostVec && costs,
     293                CandidateList & out
    264294        ) {
    265295                // prune if cheaper alternative for same key has already been generated
     
    278308        }
    279309
    280         /// Combo iterator that combines candidates into an output list, merging their environments. 
    281         /// Rejects an appended candidate if environments cannot be merged. See `Common/FilterCombos.h` 
     310        /// Combo iterator that combines candidates into an output list, merging their environments.
     311        /// Rejects an appended candidate if environments cannot be merged. See `Common/FilterCombos.h`
    282312        /// for description of "combo iterator".
    283313        class CandidateEnvMerger {
     
    299329                        Cost cost;
    300330
    301                         OutType( 
    302                                 const ast::TypeEnvironment & e, const ast::OpenVarSet & o, 
     331                        OutType(
     332                                const ast::TypeEnvironment & e, const ast::OpenVarSet & o,
    303333                                const std::vector< DeferRef > & as, const ast::SymbolTable & symtab )
    304334                        : env( e ), open( o ), assns( as ), cost( Cost::zero ) {
     
    306336                                for ( const DeferRef & assn : assns ) {
    307337                                        // compute conversion cost from satisfying decl to assertion
    308                                         cost += computeConversionCost( 
    309                                                 assn.match.adjType, assn.decl->get_type(), symtab, env );
    310                                        
     338                                        cost += computeConversionCost(
     339                                                assn.match.adjType, assn.expr->result, false, symtab, env );
     340
    311341                                        // mark vars+specialization on function-type assertions
    312                                         const ast::FunctionType * func = 
     342                                        const ast::FunctionType * func =
    313343                                                GenPoly::getFunctionType( assn.match.cdata.id->get_type() );
    314344                                        if ( ! func ) continue;
    315345
    316                                         for ( const ast::DeclWithType * param : func->params ) {
    317                                                 cost.decSpec( specCost( param->get_type() ) );
     346                                        for ( const auto & param : func->params ) {
     347                                                cost.decSpec( specCost( param ) );
    318348                                        }
    319                                        
     349
    320350                                        cost.incVar( func->forall.size() );
    321                                        
    322                                         for ( const ast::TypeDecl * td : func->forall ) {
    323                                                 cost.decSpec( td->assertions.size() );
    324                                         }
     351
     352                                        cost.decSpec( func->assertions.size() );
    325353                                }
    326354                        }
     
    329357                };
    330358
    331                 CandidateEnvMerger( 
    332                         const ast::TypeEnvironment & env, const ast::OpenVarSet & open, 
     359                CandidateEnvMerger(
     360                        const ast::TypeEnvironment & env, const ast::OpenVarSet & open,
    333361                        const ast::SymbolTable & syms )
    334362                : crnt(), envs{ env }, opens{ open }, symtab( syms ) {}
     
    357385
    358386        /// Limit to depth of recursion of assertion satisfaction
    359         static const int recursionLimit = 4;
     387        static const int recursionLimit = 8;
    360388        /// Maximum number of simultaneously-deferred assertions to attempt concurrent satisfaction of
    361389        static const int deferLimit = 10;
    362390} // anonymous namespace
    363391
    364 void satisfyAssertions( 
    365         CandidateRef & cand, const ast::SymbolTable & symtab, CandidateList & out, 
     392void satisfyAssertions(
     393        CandidateRef & cand, const ast::SymbolTable & symtab, CandidateList & out,
    366394        std::vector<std::string> & errors
    367395) {
     
    389417                        if ( it != thresholds.end() && it->second < sat.costs ) goto nextSat;
    390418
    391                         // make initial pass at matching assertions
    392                         for ( auto & assn : sat.need ) {
    393                                 // fail early if any assertion is not satisfiable
    394                                 if ( ! satisfyAssertion( assn, sat ) ) {
     419                        // should a limit be imposed? worst case here is O(n^2) but very unlikely to happen.
     420                        for (unsigned resetCount = 0; ; ++resetCount) {
     421                                ast::AssertionList next;
     422                                resetTyVarRenaming();
     423                                // make initial pass at matching assertions
     424                                for ( auto & assn : sat.need ) {
     425                                        // fail early if any assertion is not satisfiable
     426                                        if ( ! satisfyAssertion( assn, sat ) ) {
     427                                                next.emplace_back(assn);
     428                                                // goto nextSat;
     429                                        }
     430                                }
     431                                // success
     432                                if (next.empty()) break;
     433                                // fail if nothing resolves
     434                                else if (next.size() == sat.need.size()) {
    395435                                        Indenter tabs{ 3 };
    396436                                        std::ostringstream ss;
     
    398438                                        print( ss, *sat.cand, ++tabs );
    399439                                        ss << (tabs-1) << "Could not satisfy assertion:\n";
    400                                         ast::print( ss, assn.first, tabs );
     440                                        ast::print( ss, next[0].first, tabs );
    401441
    402442                                        errors.emplace_back( ss.str() );
    403443                                        goto nextSat;
    404444                                }
     445                                sat.need = std::move(next);
    405446                        }
    406447
     
    408449                                // either add successful match or push back next state
    409450                                if ( sat.newNeed.empty() ) {
    410                                         finalizeAssertions( 
     451                                        finalizeAssertions(
    411452                                                sat.cand, sat.inferred, thresholds, std::move( sat.costs ), out );
    412453                                } else {
     
    421462                                ss << (tabs-1) << "Too many non-unique satisfying assignments for assertions:\n";
    422463                                for ( const auto & d : sat.deferred ) {
    423                                         ast::print( ss, d.decl, tabs );
     464                                        ast::print( ss, d.expr, tabs );
    424465                                }
    425466
     
    430471                                std::vector< CandidateEnvMerger::OutType > compatible = filterCombos(
    431472                                        sat.deferred, CandidateEnvMerger{ sat.cand->env, sat.cand->open, sat.symtab } );
    432                                
     473
    433474                                // fail early if no mutually-compatible assertion satisfaction
    434475                                if ( compatible.empty() ) {
     
    439480                                        ss << (tabs-1) << "No mutually-compatible satisfaction for assertions:\n";
    440481                                        for ( const auto& d : sat.deferred ) {
    441                                                 ast::print( ss, d.decl, tabs );
     482                                                ast::print( ss, d.expr, tabs );
    442483                                        }
    443484
     
    453494                                        // set up next satisfaction state
    454495                                        CandidateRef nextCand = std::make_shared<Candidate>(
    455                                                 sat.cand->expr, std::move( compat.env ), std::move( compat.open ), 
     496                                                sat.cand->expr, std::move( compat.env ), std::move( compat.open ),
    456497                                                ast::AssertionSet{} /* need moved into satisfaction state */,
    457498                                                sat.cand->cost, sat.cand->cvtCost );
     
    459500                                        ast::AssertionSet nextNewNeed{ sat.newNeed };
    460501                                        InferCache nextInferred{ sat.inferred };
    461                                        
     502
    462503                                        CostVec nextCosts{ sat.costs };
    463504                                        nextCosts.back() += compat.cost;
    464                                                                
     505
    465506                                        ast::SymbolTable nextSymtab{ sat.symtab };
    466507
     
    471512                                                nextNewNeed.insert( match.need.begin(), match.need.end() );
    472513
    473                                                 bindAssertion( r.decl, r.info, nextCand, match, nextInferred );
     514                                                bindAssertion( r.expr, r.info, nextCand, match, nextInferred );
    474515                                        }
    475516
    476517                                        // either add successful match or push back next state
    477518                                        if ( nextNewNeed.empty() ) {
    478                                                 finalizeAssertions( 
     519                                                finalizeAssertions(
    479520                                                        nextCand, nextInferred, thresholds, std::move( nextCosts ), out );
    480521                                        } else {
    481                                                 nextSats.emplace_back( 
    482                                                         std::move( nextCand ), std::move( nextNewNeed ), 
    483                                                         std::move( nextInferred ), std::move( nextCosts ), 
     522                                                nextSats.emplace_back(
     523                                                        std::move( nextCand ), std::move( nextNewNeed ),
     524                                                        std::move( nextInferred ), std::move( nextCosts ),
    484525                                                        std::move( nextSymtab ) );
    485526                                        }
     
    493534                nextSats.clear();
    494535        }
    495        
     536
    496537        // exceeded recursion limit if reaches here
    497538        if ( out.empty() ) {
Note: See TracChangeset for help on using the changeset viewer.