Goose  Artifact [fdc3d0d2b0]

Artifact fdc3d0d2b05bd68fc849f94370a42e37cfd05ec73041e12a34cc78d58bc43c68:

  • File bs/sema/uni-holes.cpp — part of check-in [931f31882e] at 2019-02-27 23:04:35 on branch trunk —
    • Sema: added a way to define a "three way unification" sub expression.
    • Templates: used the above to generate a proper signature for TDecls, and fixed an issue with TVars.
    (user: achavasse size: 8527)

#include "sema.h"

namespace empathy::sema
{
    void SetupHoleUnificationRules( UnificationRuleSet& ruleSet )
    {
        // Hole half-unification: Convert it to a numbered hole,
        // If the name wasn't already known, add 1 to the score's unique holes count.
        ruleSet.addHalfUnificationRule(
            TVEC( TSID( hole ), ANYTERM( h ) ),
        []( const Term& lhs, UnificationContext& c ) -> UniGen
        {
            auto lh = *HoleFromIRExpr( lhs );

            if( holds_alternative< StringId >( lh ) )
            {
                // This is a named hole: look up its name.
                const auto& name = get< StringId >( lh );
                auto holeIndex = c.getLHSHoleIndex( name );
                if( holeIndex != UnificationContext::InvalidIndex )
                {
                    if( !c.isHoleLocked( holeIndex ) )
                        co_yield { MkHole( holeIndex ), c };
                }
                else
                {
                    // This is a new name: create a new value,
                    // and increment the number of unique holes in the current score.
                    auto index = c.createValue();
                    c.setLHSHoleIndex( name, index );
                    co_yield { MkHole( index ), c };
                }
            }
            else
            {
                // This is already an indexed hole: yield it as is.
                if( !c.isHoleLocked( get< uint32_t >( lh ) ) )
                    co_yield { lhs, c };
            }
        } );

        // Hole vs anything
        ruleSet.addSymRule(
            TVEC( TSID( hole ), ANYTERM( h ) ),
            ANYTERM( t ),
        []( const Term& lhs, const Term& rhs, UnificationContext& c ) -> UniGen
        {
            auto h = *HoleFromIRExpr( lhs );
            uint32_t index = 0;

            // Remember the previous complexity count so we know how much complexity
            // is added by this particular sub-term. This is because we need
            // to be able to subtract it when updating the hole's value with a new solution.
            uint32_t oldComplexity = c.complexity();

            if( holds_alternative< uint32_t >( h ) )
                index = get< uint32_t >( h );
            else
            {
                // This is a named hole: look up its name.
                const auto& name = get< StringId >( h );

                index = c.getLHSHoleIndex( name );
                if( index == UnificationContext::InvalidIndex )
                {
                    // This is a new name: create a new value.
                    index = c.createValue();
                    c.setLHSHoleIndex( name, index );
                    auto holeExpr = MkHole( index );

                    for( auto&& [e,c] : HalfUnify( rhs, c.flip() ) )
                    {
                        c.setValue( index, move( e ), c.complexity() - oldComplexity );
                        co_yield { move( holeExpr ), c.flip() };
                    }

                    co_return;
                }
            }

            // Reject recursive hole nesting.
            if( c.isHoleLocked( index ) )
                co_return;
            c.lockHole( index );

            auto holeExpr = MkHole( index );

            auto& maybeVal = c.getValue( index );
            if( maybeVal )
            {
                for( auto&& [e,c] : Unify( *maybeVal, rhs, c ) )
                {
                    c.unlockHole( index );
                    c.setValue( index, move( e ), c.complexity() - oldComplexity );
                    co_yield { move( holeExpr ), c };
                }
            }
            else
            {
                for( auto&& [e,c] : HalfUnify( rhs, c.flip() ) )
                {
                    c.unlockHole( index );
                    c.setValue( index, move( e ), c.complexity() - oldComplexity );
                    co_yield { move( holeExpr ), c.flip() };
                }
            }
        } );

        // Hole vs hole
        ruleSet.addAsymRule(
            TVEC( TSID( hole ), ANYTERM( h1 ) ),
            TVEC( TSID( hole ), ANYTERM( h2 ) ),
        []( const Term& lhs, const Term& rhs, UnificationContext& c ) -> UniGen
        {
            auto lh = *HoleFromIRExpr( lhs );
            auto rh = *HoleFromIRExpr( rhs );

            StringId lname;
            StringId rname;

            uint32_t lindex = 0;
            uint32_t rindex = 0;

            if( holds_alternative< StringId >( lh ) )
            {
                // L is a named hole: look up its name.
                lname = get< StringId >( lh );
                lindex = c.getLHSHoleIndex( lname );
            }
            else
                lindex = get< uint32_t >( lh );

            if( holds_alternative< StringId >( rh ) )
            {
                // R is a named hole: look up its name.
                rname = get< StringId >( rh );
                rindex = c.getRHSHoleIndex( rname );
            }
            else
                rindex = get< uint32_t >( rh );

            // If neither hole currently have a value, create a new one.
            if( lindex == UnificationContext::InvalidIndex && rindex == UnificationContext::InvalidIndex )
            {
                auto index = c.createValue();
                c.setLHSHoleIndex( lname, index );
                c.setRHSHoleIndex( rname, index );

                co_yield { MkHole( index ), c };
                co_return;
            }

            // If both holes actually point to the same value, just yield it as the solution.
            if( lindex == rindex )
            {
                co_yield { MkHole( lindex ), c };
                co_return;
            }

            // If either hole doesn't have a value yet, assign it the other one's value.
            if( lindex == UnificationContext::InvalidIndex )
            {
                c.setLHSHoleIndex( lname, rindex );
                co_yield { MkHole( rindex ), c };
                co_return;
            }

            if( rindex == UnificationContext::InvalidIndex )
            {
                c.setRHSHoleIndex( rname, lindex );
                co_yield { MkHole( lindex ), c };
                co_return;
            }

            // Reject recursive hole nesting.
            if( c.isHoleLocked( lindex ) )
                co_return;
            if( c.isHoleLocked( rindex ) )
                co_return;

            c.lockHole( lindex );
            c.lockHole( rindex );

            // If either hole have an empty value, set it to a hole expression with the id of the value
            // stored in the other one. We can't just copy the value over as we would lose the dependency
            // relationship between the two holes.
            const auto& lval = c.getValue( lindex );
            const auto& rval = c.getValue( rindex );

            if( !rval )
            {
                for( auto&& [e,c] : HalfUnify( *rval, c.flip() ) )
                {
                    c.unlockHole( lindex );
                    c.unlockHole( rindex );

                    c.setValue( rindex, MkHole( lindex ), 0 );
                    co_yield { MkHole( lindex ), c };
                }
                co_return;
            }

            if( !lval )
            {
                for( auto&& [e,c] : HalfUnify( *rval, c.flip() ) )
                {
                    c.unlockHole( lindex );
                    c.unlockHole( rindex );

                    c.setValue( lindex, MkHole( rindex ), 0 );
                    co_yield { MkHole( rindex ), c.flip() };
                }
                co_return;
            }

            // Both L and R have a value: unify them, store the result in lhs,
            // replace rhs with a hole expression pointing to lhs's value.

            // Remember the previous complexity count so we know how much complexity
            // is added by this particular sub-term. This is because we need
            // to be able to subtract it when updating the hole's value with a new solution.
            uint32_t oldComplexity = c.complexity();

            for( auto&& [e,c] : Unify( *lval, *rval, c ) )
            {
                c.unlockHole( lindex );
                c.unlockHole( rindex );

                c.setValue( lindex, move( e ), c.complexity() - oldComplexity );
                c.setValue( rindex, MkHole( lindex ), 0 );
                co_yield { MkHole( lindex ), c };
            }
        } );
    }
}