forked from dherman/es4
-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathast-query.sml
132 lines (117 loc) · 4.45 KB
/
ast-query.sml
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
(* -*- mode: sml; mode: font-lock; tab-width: 4; insert-tabs-mode: nil; indent-tabs-mode: nil -*- *)
(*
* The following licensing terms and conditions apply and must be
* accepted in order to use the Reference Implementation:
*
* 1. This Reference Implementation is made available to all
* interested persons on the same terms as Ecma makes available its
* standards and technical reports, as set forth at
* http://www.ecma-international.org/publications/.
*
* 2. All liability and responsibility for any use of this Reference
* Implementation rests with the user, and not with any of the parties
* who contribute to, or who own or hold any copyright in, this Reference
* Implementation.
*
* 3. THIS REFERENCE IMPLEMENTATION IS PROVIDED BY THE COPYRIGHT
* HOLDERS AND CONTRIBUTORS "AS IS" AND ANY EXPRESS OR IMPLIED
* WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF
* MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
* DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE
* LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
* CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
* SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR
* BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY,
* WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE
* OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN
* IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
*
* End of Terms and Conditions
*
* Copyright (c) 2007 Adobe Systems Inc., The Mozilla Foundation, Opera
* Software ASA, and others.
*)
(*
* This module contains only functions that depend directly on Ast. It
* provides common infrastructure for inquiring about AST nodes in concise
* terms, without resorting to code duplication or awkward cyclical module
* dependencies.
*)
structure AstQuery = struct
val doTrace = ref false
fun trace ss = if (!doTrace) then LogErr.log ("[ast-query] " :: ss) else ()
fun error ss = LogErr.astError ss
fun findType (t:Ast.TYPE)
(q:Ast.TYPE -> ('a option))
(kind:string)
: 'a =
case q t of
SOME a => a
| NONE =>
(case t of
Ast.UnionType (t::ts) =>
let
fun f ty = case q ty of
NONE => false
| SOME _ => true
in
case q t of
NONE => findType (Ast.UnionType ts) q kind
| SOME a =>
if List.exists f ts
then error ["multiple ", kind,
" type expressions found in union, ",
"need unique one: ", LogErr.ty t]
else a
end
| _ => error ["unable to find ", kind,
" type expression in ", LogErr.ty t])
fun needInstanceType (t:Ast.TYPE)
: Ast.TYPE =
let
fun isInstanceType ty =
case ty of
Ast.InstanceType t => SOME ty
| Ast.AppType (c, _) => (case isInstanceType c of
NONE => NONE
| SOME _ => SOME ty)
| _ => NONE
in
findType t isInstanceType "instance"
end
fun needFunctionType (t:Ast.TYPE)
: Ast.FUNCTION_TYPE =
let
fun isFunctionType ty =
case ty of
Ast.FunctionType t => SOME t
| _ => NONE
in
findType t isFunctionType "function"
end
fun resultTyOfFuncTy t =
case (#result (needFunctionType t)) of
SOME t => t
| NONE => error ["resultTyOfFuncTy: none found"]
val minArgsOfFuncTy = (#minArgs) o needFunctionType
fun queryFuncTy (q:Ast.FUNCTION_TYPE -> 'a)
(funcTy:Ast.TYPE)
: 'a =
q (needFunctionType funcTy)
val paramTysOfFuncTy = queryFuncTy (#params)
val funcTyHasRest = queryFuncTy (#hasRest)
fun singleParamTyOfFuncTy (ty:Ast.TYPE)
: Ast.TYPE =
let
val funcTy = needFunctionType ty
in
case (#params funcTy) of
[t] => t
| _ => error ["singleParamTyOfFuncTy: non-unique parameter"]
end
fun funcIsAbstract (func:Ast.FUNC)
: bool =
case func of
Ast.Func { block = NONE, ... } => true
| _ => false
end