Skip to content

Commit f4e24e3

Browse files
committed
Add support for interfaces extending multiple interfaces
This is a feature in TypeScript, and I didn't see much of a technical reason to disallow it. By changing interface extension such that implementsTypes and interfacePrototypes are used for base interfaces instead of extendsType and basePrototype in InterfacePrototype and Interface respectively, and by modifying the parser, existing code doesn't seem to break, and multiple base interfaces are possible (if not working already). There was also a small change to the instanceof helper generation, where arrays are now used instead of Sets, since I needed to filter for interfaces, and Set_values was used on the constructed Set regardless. However, the change also modified the order of instanceof checks as seen in instanceof.debug.wat. The instanceof.release.wat file underwent more drastic changes, but it still appears to work anyway.
1 parent 67579de commit f4e24e3

8 files changed

+320
-160
lines changed

src/compiler.ts

+13-10
Original file line numberDiff line numberDiff line change
@@ -7696,23 +7696,26 @@ export class Compiler extends DiagnosticEmitter {
76967696
), false // managedness is irrelevant here, isn't interrupted
76977697
)
76987698
);
7699-
let allInstances: Set<Class> | null;
7699+
let allInstances: Class[] | null;
77007700
if (instance.isInterface) {
7701-
allInstances = instance.implementers;
7701+
let implementers = instance.implementers;
7702+
// Ensure interfaces are filtered out, since their class IDs will never be
7703+
// seen in actual objects.
7704+
allInstances = implementers
7705+
? Set_values(implementers).filter(implementer => implementer.kind == ElementKind.Class)
7706+
: null;
77027707
} else {
7703-
allInstances = new Set();
7704-
allInstances.add(instance);
77057708
let extenders = instance.extenders;
77067709
if (extenders) {
7707-
for (let _values = Set_values(extenders), i = 0, k = _values.length; i < k; ++i) {
7708-
let extender = _values[i];
7709-
allInstances.add(extender);
7710-
}
7710+
allInstances = Set_values(extenders);
7711+
allInstances.push(instance);
7712+
} else {
7713+
allInstances = [instance];
77117714
}
77127715
}
77137716
if (allInstances) {
7714-
for (let _values = Set_values(allInstances), i = 0, k = _values.length; i < k; ++i) {
7715-
let instance = _values[i];
7717+
for (let i = 0, k = allInstances.length; i < k; ++i) {
7718+
let instance = unchecked(allInstances[i]);
77167719
stmts.push(
77177720
module.br("is_instance",
77187721
module.binary(BinaryOp.EqI32,

src/extra/ast.ts

+8-4
Original file line numberDiff line numberDiff line change
@@ -1297,12 +1297,16 @@ export class ASTBuilder {
12971297
}
12981298
sb.push(">");
12991299
}
1300-
let extendsType = node.extendsType;
1301-
if (extendsType) {
1300+
let implementsTypes = node.implementsTypes;
1301+
if (implementsTypes && implementsTypes.length > 0) {
13021302
sb.push(" extends ");
1303-
this.visitTypeNode(extendsType);
1303+
this.visitTypeNode(implementsTypes[0]);
1304+
for (let i = 1, k = implementsTypes.length; i < k; ++i) {
1305+
sb.push(", ");
1306+
this.visitTypeNode(implementsTypes[i]);
1307+
}
13041308
}
1305-
// must not have implementsTypes
1309+
// must not have extendsType
13061310
sb.push(" {\n");
13071311
let indentLevel = ++this.indentLevel;
13081312
let members = node.members;

src/parser.ts

+32-12
Original file line numberDiff line numberDiff line change
@@ -1708,20 +1708,40 @@ export class Parser extends DiagnosticEmitter {
17081708
}
17091709

17101710
let extendsType: NamedTypeNode | null = null;
1711+
let implementsTypes: NamedTypeNode[] | null = null;
17111712
if (tn.skip(Token.Extends)) {
1712-
let type = this.parseType(tn);
1713-
if (!type) return null;
1714-
if (type.kind != NodeKind.NamedType) {
1715-
this.error(
1716-
DiagnosticCode.Identifier_expected,
1717-
type.range
1718-
);
1719-
return null;
1713+
if (isInterface) {
1714+
do {
1715+
let type = this.parseType(tn);
1716+
if (!type) return null;
1717+
if (type.kind != NodeKind.NamedType) {
1718+
this.error(
1719+
DiagnosticCode.Identifier_expected,
1720+
type.range
1721+
);
1722+
return null;
1723+
}
1724+
// Note: Even though the keyword is "extends", the base interfaces are stored in
1725+
// the implementsTypes field, as that's already an array that can be used.
1726+
// When an InterfacePrototype is created, the base InterfacePrototypes are
1727+
// stored in the interfacePrototypes field for the same reason.
1728+
if (!implementsTypes) implementsTypes = [<NamedTypeNode>type];
1729+
else implementsTypes.push(<NamedTypeNode>type);
1730+
} while (tn.skip(Token.Comma));
1731+
} else {
1732+
let type = this.parseType(tn);
1733+
if (!type) return null;
1734+
if (type.kind != NodeKind.NamedType) {
1735+
this.error(
1736+
DiagnosticCode.Identifier_expected,
1737+
type.range
1738+
);
1739+
return null;
1740+
}
1741+
extendsType = <NamedTypeNode>type;
17201742
}
1721-
extendsType = <NamedTypeNode>type;
17221743
}
17231744

1724-
let implementsTypes: NamedTypeNode[] | null = null;
17251745
if (tn.skip(Token.Implements)) {
17261746
if (isInterface) {
17271747
this.error(
@@ -1757,14 +1777,14 @@ export class Parser extends DiagnosticEmitter {
17571777
let members = new Array<DeclarationStatement>();
17581778
let declaration: ClassDeclaration;
17591779
if (isInterface) {
1760-
assert(!implementsTypes);
1780+
assert(!extendsType);
17611781
declaration = Node.createInterfaceDeclaration(
17621782
identifier,
17631783
decorators,
17641784
flags,
17651785
typeParameters,
1766-
extendsType,
17671786
null,
1787+
implementsTypes,
17681788
members,
17691789
tn.range(startPos, tn.pos)
17701790
);

src/program.ts

+84-59
Original file line numberDiff line numberDiff line change
@@ -1130,7 +1130,7 @@ export class Program extends DiagnosticEmitter {
11301130
break;
11311131
}
11321132
case NodeKind.InterfaceDeclaration: {
1133-
this.initializeInterface(<InterfaceDeclaration>statement, file, queuedExtends);
1133+
this.initializeInterface(<InterfaceDeclaration>statement, file, queuedImplements);
11341134
break;
11351135
}
11361136
case NodeKind.NamespaceDeclaration: {
@@ -1303,64 +1303,45 @@ export class Program extends DiagnosticEmitter {
13031303
}
13041304
}
13051305

1306-
// resolve prototypes of extended classes or interfaces
1306+
// resolve prototypes of extended classes
13071307
let resolver = this.resolver;
13081308
for (let i = 0, k = queuedExtends.length; i < k; ++i) {
13091309
let thisPrototype = queuedExtends[i];
1310+
assert(thisPrototype.kind == ElementKind.ClassPrototype);
13101311
let extendsNode = assert(thisPrototype.extendsNode); // must be present if in queuedExtends
13111312
let baseElement = resolver.resolveTypeName(extendsNode.name, null, thisPrototype.parent);
13121313
if (!baseElement) continue;
1313-
if (thisPrototype.kind == ElementKind.ClassPrototype) {
1314-
if (baseElement.kind == ElementKind.ClassPrototype) {
1315-
let basePrototype = <ClassPrototype>baseElement;
1316-
if (basePrototype.hasDecorator(DecoratorFlags.Final)) {
1317-
this.error(
1318-
DiagnosticCode.Class_0_is_final_and_cannot_be_extended,
1319-
extendsNode.range, basePrototype.identifierNode.text
1320-
);
1321-
}
1322-
if (
1323-
basePrototype.hasDecorator(DecoratorFlags.Unmanaged) !=
1324-
thisPrototype.hasDecorator(DecoratorFlags.Unmanaged)
1325-
) {
1326-
this.error(
1327-
DiagnosticCode.Unmanaged_classes_cannot_extend_managed_classes_and_vice_versa,
1328-
Range.join(thisPrototype.identifierNode.range, extendsNode.range)
1329-
);
1330-
}
1331-
if (!thisPrototype.extends(basePrototype)) {
1332-
thisPrototype.basePrototype = basePrototype;
1333-
} else {
1334-
this.error(
1335-
DiagnosticCode._0_is_referenced_directly_or_indirectly_in_its_own_base_expression,
1336-
basePrototype.identifierNode.range,
1337-
basePrototype.identifierNode.text,
1338-
);
1339-
}
1340-
} else {
1314+
if (baseElement.kind == ElementKind.ClassPrototype) {
1315+
let basePrototype = <ClassPrototype>baseElement;
1316+
if (basePrototype.hasDecorator(DecoratorFlags.Final)) {
13411317
this.error(
1342-
DiagnosticCode.A_class_may_only_extend_another_class,
1343-
extendsNode.range
1318+
DiagnosticCode.Class_0_is_final_and_cannot_be_extended,
1319+
extendsNode.range, basePrototype.identifierNode.text
13441320
);
13451321
}
1346-
} else if (thisPrototype.kind == ElementKind.InterfacePrototype) {
1347-
if (baseElement.kind == ElementKind.InterfacePrototype) {
1348-
const basePrototype = <InterfacePrototype>baseElement;
1349-
if (!thisPrototype.extends(basePrototype)) {
1350-
thisPrototype.basePrototype = basePrototype;
1351-
} else {
1352-
this.error(
1353-
DiagnosticCode._0_is_referenced_directly_or_indirectly_in_its_own_base_expression,
1354-
basePrototype.identifierNode.range,
1355-
basePrototype.identifierNode.text,
1356-
);
1357-
}
1322+
if (
1323+
basePrototype.hasDecorator(DecoratorFlags.Unmanaged) !=
1324+
thisPrototype.hasDecorator(DecoratorFlags.Unmanaged)
1325+
) {
1326+
this.error(
1327+
DiagnosticCode.Unmanaged_classes_cannot_extend_managed_classes_and_vice_versa,
1328+
Range.join(thisPrototype.identifierNode.range, extendsNode.range)
1329+
);
1330+
}
1331+
if (!thisPrototype.extends(basePrototype)) {
1332+
thisPrototype.basePrototype = basePrototype;
13581333
} else {
13591334
this.error(
1360-
DiagnosticCode.An_interface_can_only_extend_an_interface,
1361-
extendsNode.range
1335+
DiagnosticCode._0_is_referenced_directly_or_indirectly_in_its_own_base_expression,
1336+
basePrototype.identifierNode.range,
1337+
basePrototype.identifierNode.text,
13621338
);
13631339
}
1340+
} else {
1341+
this.error(
1342+
DiagnosticCode.A_class_may_only_extend_another_class,
1343+
extendsNode.range
1344+
);
13641345
}
13651346
}
13661347

@@ -1399,7 +1380,7 @@ export class Program extends DiagnosticEmitter {
13991380
}
14001381
}
14011382

1402-
// resolve prototypes of implemented interfaces
1383+
// resolve prototypes of implemented/extended interfaces
14031384
for (let i = 0, k = queuedImplements.length; i < k; ++i) {
14041385
let thisPrototype = queuedImplements[i];
14051386
let implementsNodes = assert(thisPrototype.implementsNodes); // must be present if in queuedImplements
@@ -1411,10 +1392,23 @@ export class Program extends DiagnosticEmitter {
14111392
let interfacePrototype = <InterfacePrototype>interfaceElement;
14121393
let interfacePrototypes = thisPrototype.interfacePrototypes;
14131394
if (!interfacePrototypes) thisPrototype.interfacePrototypes = interfacePrototypes = new Array();
1414-
interfacePrototypes.push(interfacePrototype);
1395+
if (
1396+
thisPrototype.kind == ElementKind.Interface &&
1397+
thisPrototype.implements(interfacePrototype)
1398+
) {
1399+
this.error(
1400+
DiagnosticCode._0_is_referenced_directly_or_indirectly_in_its_own_base_expression,
1401+
interfacePrototype.identifierNode.range,
1402+
interfacePrototype.identifierNode.text,
1403+
);
1404+
} else {
1405+
interfacePrototypes.push(interfacePrototype);
1406+
}
14151407
} else {
14161408
this.error(
1417-
DiagnosticCode.A_class_can_only_implement_an_interface,
1409+
thisPrototype.kind == ElementKind.InterfacePrototype
1410+
? DiagnosticCode.An_interface_can_only_extend_an_interface
1411+
: DiagnosticCode.A_class_can_only_implement_an_interface,
14181412
implementsNode.range
14191413
);
14201414
}
@@ -2474,7 +2468,7 @@ export class Program extends DiagnosticEmitter {
24742468
break;
24752469
}
24762470
case NodeKind.InterfaceDeclaration: {
2477-
element = this.initializeInterface(<InterfaceDeclaration>declaration, parent, queuedExtends);
2471+
element = this.initializeInterface(<InterfaceDeclaration>declaration, parent, queuedImplements);
24782472
break;
24792473
}
24802474
case NodeKind.NamespaceDeclaration: {
@@ -2625,7 +2619,7 @@ export class Program extends DiagnosticEmitter {
26252619
/** Parent element, usually a file or namespace. */
26262620
parent: Element,
26272621
/** So far queued `extends` clauses. */
2628-
queuedExtends: ClassPrototype[],
2622+
queuedImplements: ClassPrototype[],
26292623
): InterfacePrototype | null {
26302624
let name = declaration.name.text;
26312625
let element = new InterfacePrototype(
@@ -2638,8 +2632,8 @@ export class Program extends DiagnosticEmitter {
26382632
);
26392633
if (!parent.add(name, element)) return null;
26402634

2641-
// remember interfaces that extend another interface
2642-
if (declaration.extendsType) queuedExtends.push(element);
2635+
// remember interfaces that extend other interfaces
2636+
if (declaration.implementsTypes) queuedImplements.push(element);
26432637

26442638
let memberDeclarations = declaration.members;
26452639
for (let i = 0, k = memberDeclarations.length; i < k; ++i) {
@@ -2762,7 +2756,7 @@ export class Program extends DiagnosticEmitter {
27622756
break;
27632757
}
27642758
case NodeKind.InterfaceDeclaration: {
2765-
this.initializeInterface(<InterfaceDeclaration>member, original, queuedExtends);
2759+
this.initializeInterface(<InterfaceDeclaration>member, original, queuedImplements);
27662760
break;
27672761
}
27682762
case NodeKind.NamespaceDeclaration: {
@@ -4298,6 +4292,24 @@ export class ClassPrototype extends DeclaredElement {
42984292
return false;
42994293
}
43004294

4295+
implements(other: InterfacePrototype, seen: Set<InterfacePrototype> | null = null): bool {
4296+
if (this.interfacePrototypes) {
4297+
if (!seen) seen = new Set();
4298+
let interfacePrototypes = assert(this.interfacePrototypes);
4299+
4300+
for (let i = 0, k = interfacePrototypes.length; i < k; ++i) {
4301+
let prototype = unchecked(interfacePrototypes[i]);
4302+
4303+
if (prototype == other) return true;
4304+
if (seen.has(prototype)) continue;
4305+
seen.add(prototype);
4306+
4307+
if (prototype.implements(other, seen)) return true;
4308+
}
4309+
}
4310+
return false;
4311+
}
4312+
43014313
/** Adds an element as an instance member of this one. Returns the previous element if a duplicate. */
43024314
addInstance(name: string, element: DeclaredElement): bool {
43034315
let originalDeclaration = element.declaration;
@@ -4557,9 +4569,11 @@ export class Class extends TypedElement {
45574569
// Start with the interface itself, adding this class and its extenders to
45584570
// its implementers. Repeat for the interface's bases that are indirectly
45594571
// implemented by means of being extended by the interface.
4560-
let nextIface: Interface | null = iface;
4572+
// TODO: Maybe add a fast path when `iface` has no bases?
4573+
let ifaceStack = [iface];
45614574
let extenders = this.extenders;
45624575
do {
4576+
let nextIface = assert(ifaceStack.pop());
45634577
let implementers = nextIface.implementers;
45644578
if (!implementers) nextIface.implementers = implementers = new Set();
45654579
implementers.add(this);
@@ -4569,8 +4583,19 @@ export class Class extends TypedElement {
45694583
implementers.add(extender);
45704584
}
45714585
}
4572-
nextIface = <Interface | null>nextIface.base;
4573-
} while (nextIface);
4586+
4587+
let nextIfaces = nextIface.interfaces;
4588+
if (!nextIfaces) continue;
4589+
4590+
let stackIndex = ifaceStack.length;
4591+
4592+
// Calls the internal ensureCapacity() when run in the bootstrapped compiler:
4593+
ifaceStack.length = stackIndex + nextIfaces.size;
4594+
4595+
for (let _values = Set_values(nextIfaces), i = 0, k = _values.length; i < k; ++i) {
4596+
ifaceStack[stackIndex++] = unchecked(_values[i]);
4597+
}
4598+
} while (ifaceStack.length);
45744599
}
45754600

45764601
/** Adds an interface. */
@@ -4589,7 +4614,7 @@ export class Class extends TypedElement {
45894614
if (target.isInterface) {
45904615
if (this.isInterface) {
45914616
// targetInterface = thisInterface
4592-
return this == target || this.extends(target);
4617+
return this == target || this.implements(<Interface>target);
45934618
} else {
45944619
// targetInterface = thisClass
45954620
return this.implements(<Interface>target);
@@ -4863,7 +4888,7 @@ export class Class extends TypedElement {
48634888
return true;
48644889
}
48654890

4866-
/** Tests if this class or interface extends the given class or interface. */
4891+
/** Tests if this class extends the given class. */
48674892
extends(other: Class): bool {
48684893
return other.hasExtender(this);
48694894
}

0 commit comments

Comments
 (0)