diff --git a/src/type-logic.ts b/src/type-logic.ts index 916d8fe7..f84354cd 100644 --- a/src/type-logic.ts +++ b/src/type-logic.ts @@ -206,11 +206,15 @@ export function meet(a: Type, b: Type): Type { // union is join. meet distributes over join. return a.of.map(t => meet(t, b)).reduce(join); } - if ( - (a.kind === 'list' && b.kind === 'list') || - (a.kind === 'normal completion' && b.kind === 'normal completion') - ) { - return { kind: a.kind, of: meet(a.of, b.of) }; + if (a.kind === 'list' && b.kind === 'list') { + return { kind: 'list', of: meet(a.of, b.of) }; + } + if (a.kind === 'normal completion' && b.kind === 'normal completion') { + const inner = meet(a.of, b.of); + if (inner.kind === 'never') { + return { kind: 'never' }; + } + return { kind: 'normal completion', of: inner }; } return { kind: 'never' }; } @@ -397,6 +401,19 @@ export function typeFromExpr(expr: Expr, biblio: Biblio): Type { } const calleeName = callee[0].contents; + // special case: `Completion` is identity + if (expr.name === 'call' && calleeName === 'Completion' && expr.arguments.length === 1) { + return typeFromExpr(expr.arguments[0], biblio); + } + // special case: `NormalCompletion` wraps its input + if ( + expr.name === 'call' && + calleeName === 'NormalCompletion' && + expr.arguments.length === 1 + ) { + return { kind: 'normal completion', of: typeFromExpr(expr.arguments[0], biblio) }; + } + const biblioEntry = biblio.byAoid(calleeName); if (biblioEntry?.signature?.return == null) { break; diff --git a/test/typecheck.js b/test/typecheck.js index a1be0d57..55b39e2b 100644 --- a/test/typecheck.js +++ b/test/typecheck.js @@ -1511,7 +1511,7 @@ describe('type system', () => { await assertTypeError( 'an ECMAScript language value', 'NormalCompletion(42)', - 'argument type (a normal completion) does not look plausibly assignable to parameter type (ECMAScript language value)', + 'argument type (a normal completion containing 42) does not look plausibly assignable to parameter type (ECMAScript language value)', [completionBiblio], ); @@ -1535,7 +1535,14 @@ describe('type system', () => { await assertTypeError( 'a Boolean', 'NormalCompletion(*false*)', - 'argument type (a normal completion) does not look plausibly assignable to parameter type (Boolean)', + 'argument type (a normal completion containing false) does not look plausibly assignable to parameter type (Boolean)', + [completionBiblio], + ); + + await assertTypeError( + 'a normal completion containing a Number', + 'NormalCompletion(*false*)', + 'argument type (a normal completion containing false) does not look plausibly assignable to parameter type (a normal completion containing Number)', [completionBiblio], );