WSL should allow you to say "protocol Foo : Bar { ... }"
[WebKit-https.git] / Tools / WebGPUShadingLanguageRI / ProtocolDecl.js
1 /*
2  * Copyright (C) 2017 Apple Inc. All rights reserved.
3  *
4  * Redistribution and use in source and binary forms, with or without
5  * modification, are permitted provided that the following conditions
6  * are met:
7  * 1. Redistributions of source code must retain the above copyright
8  *    notice, this list of conditions and the following disclaimer.
9  * 2. Redistributions in binary form must reproduce the above copyright
10  *    notice, this list of conditions and the following disclaimer in the
11  *    documentation and/or other materials provided with the distribution.
12  *
13  * THIS SOFTWARE IS PROVIDED BY APPLE INC. ``AS IS'' AND ANY
14  * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
15  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
16  * PURPOSE ARE DISCLAIMED.  IN NO EVENT SHALL APPLE INC. OR
17  * CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
18  * EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
19  * PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR
20  * PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY
21  * OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
22  * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
23  * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. 
24  */
25 "use strict";
26
27 class ProtocolDecl extends Protocol {
28     constructor(origin, name)
29     {
30         super(origin, name);
31         this.extends = [];
32         this._signatures = [];
33         this._signatureMap = new Map();
34         this._typeVariable = new TypeVariable(origin, name, null);
35         this.isPrimitive = false;
36     }
37     
38     addExtends(protocol)
39     {
40         this.extends.push(protocol);
41     }
42     
43     add(signature)
44     {
45         if (!(signature instanceof ProtocolFuncDecl))
46             throw new Error("Signature isn't a ProtocolFuncDecl but a " + signature.constructor.name);
47         
48         signature.protocolDecl = this;
49         this._signatures.push(signature);
50         let overloads = this._signatureMap.get(signature.name);
51         if (!overloads)
52             this._signatureMap.set(signature.name, overloads = []);
53         overloads.push(signature);
54     }
55     
56     get signatures() { return this._signatures; }
57     signaturesByName(name) { return this._signatureMap.get(name); }
58     get typeVariable() { return this._typeVariable; }
59     
60     signaturesByNameWithTypeVariable(name, typeVariable)
61     {
62         let substitution = new Substitution([this.typeVariable], [typeVariable]);
63         let result = this.signaturesByName(name);
64         if (!result)
65             return null;
66         return result.map(signature => signature.visit(substitution));
67     }
68     
69     inherits(otherProtocol)
70     {
71         if (!otherProtocol)
72             return {result: true};
73         
74         if (otherProtocol instanceof ProtocolRef)
75             otherProtocol = otherProtocol.protocolDecl;
76         
77         for (let otherSignature of otherProtocol.signatures) {
78             let signatures = this.signaturesByName(otherSignature.name);
79             if (!signatures)
80                 return {result: false, reason: "Protocol " + this.name + " does not have a function named " + otherSignature.name + " (looking at signature " + otherSignature + ")"};
81             let overload = resolveOverloadImpl(
82                 signatures, [],
83                 otherSignature.parameterTypes,
84                 otherSignature.returnTypeForOverloadResolution);
85             if (!overload.func)
86                 return {result: false, reason: "Did not find matching signature for " + otherSignature + " in " + this.name + (overload.failures.length ? " (tried: " + overload.failures.join("; ") + ")" : "")};
87             let substitutedReturnType =
88                 overload.func.returnType.substituteToUnification(
89                     overload.func.typeParameters, overload.unificationContext);
90             if (!substitutedReturnType.equals(otherSignature.returnType))
91                 return {result: false, reason: "Return type mismatch between " + otherSignature.returnType + " and " + substitutedReturnType};
92         }
93         return {result: true};
94     }
95     
96     hasHeir(type)
97     {
98         let substitution = new Substitution([this._typeVariable], [type]);
99         let signatures = this.signatures;
100         for (let signature of signatures) {
101             signature = signature.visit(substitution);
102             let overload = resolveOverloadImpl(signature.possibleOverloads, signature.typeParameters, signature.parameterTypes, signature.returnTyupeForOverloadResolution);
103             if (!overload.func)
104                 return {result: false, reason: "Did not find matching signature for " + signature + " with type " + type + (overload.failures.length ? " (tried: " + overload.failures.join("; ") + ")" : "")};
105             
106             let substitutedReturnType = overload.func.returnType.substituteToUnification(
107                 overload.func.typeParameters, overload.unificationContext);
108             if (!substitutedReturnType.equals(signature.returnType))
109                 return {result: false, reason: "Return type mismatch between " + signature.returnType + " and " + substitutedReturnType};
110         }
111         return {result: true};
112     }
113     
114     toString()
115     {
116         return "protocol " + this.name + " { " + this.signatures.join("; ") + "; }";
117     }
118 }
119
120