1 # This file is part of NIT ( http://www.nitlanguage.org ).
3 # Copyright 2012 Jean Privat <jean@pryen.org>
5 # Licensed under the Apache License, Version 2.0 (the "License");
6 # you may not use this file except in compliance with the License.
7 # You may obtain a copy of the License at
9 # http://www.apache.org/licenses/LICENSE-2.0
11 # Unless required by applicable law or agreed to in writing, software
12 # distributed under the License is distributed on an "AS IS" BASIS,
13 # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
14 # See the License for the specific language governing permissions and
15 # limitations under the License.
17 # Parsing of literal values in the abstract syntax tree.
22 redef class ToolContext
23 # Parses literal values in the whole AST and produces errors if needed
24 var literal_phase
: Phase = new LiteralPhase(self, null)
27 private class LiteralPhase
30 redef fun process_nmodule
(nmodule
) do nmodule
.do_literal
(toolcontext
)
34 # Visit the module to compute the real value of the literal-related node of the AST.
35 # Warnings and errors are displayed on the toolcontext.
36 fun do_literal
(toolcontext
: ToolContext)
38 var v
= new LiteralVisitor(toolcontext
)
43 private class LiteralVisitor
46 var toolcontext
: ToolContext
50 n
.accept_literal
(self)
56 private fun accept_literal
(v
: LiteralVisitor) do end
60 # Get `self` as a `String`.
61 # Return null if not a string.
62 fun as_string
: nullable String
64 if not self isa AStringFormExpr then return null
65 return self.value
.as(not null)
68 # Get `self` as an `Int`.
69 # Return null if not an integer.
70 fun as_int
: nullable Int
72 if not self isa AIntExpr then return null
73 return self.value
.as(not null)
78 # The value of the literal int once computed.
79 var value
: nullable Int
82 redef class ADecIntExpr
83 redef fun accept_literal
(v
)
85 value
= self.n_number
.text
.remove_underscores
.to_i
89 redef class AHexIntExpr
90 redef fun accept_literal
(v
)
92 var s
= self.n_hex_number
.text
.substring_from
(2).remove_underscores
94 v
.toolcontext
.error
(location
, "Error: invalid hexadecimal literal")
101 redef class ABinIntExpr
102 redef fun accept_literal
(v
)
104 var s
= self.n_bin_number
.text
.substring_from
(2).remove_underscores
106 v
.toolcontext
.error
(location
, "Error: invalid binary literal")
113 redef class AOctIntExpr
114 redef fun accept_literal
(v
)
116 var s
= self.n_oct_number
.text
.substring_from
(2).remove_underscores
118 v
.toolcontext
.error
(location
, "Error: invalid octal literal")
125 redef class AByteExpr
126 # The value of the literal int once computed.
127 var value
: nullable Byte
130 redef class ADecByteExpr
131 redef fun accept_literal
(v
)
133 var t
= self.n_bytenum
.text
134 value
= t
.substring
(0, t
.length
- 2).remove_underscores
.to_i
.to_b
138 redef class AHexByteExpr
139 redef fun accept_literal
(v
)
141 var t
= self.n_hex_bytenum
.text
142 var s
= t
.substring
(2, t
.length
- 4).remove_underscores
144 v
.toolcontext
.error
(location
, "Error: invalid hexadecimal literal")
147 value
= s
.to_hex
.to_b
151 redef class ABinByteExpr
152 redef fun accept_literal
(v
)
154 var t
= self.n_bin_bytenum
.text
155 var s
= t
.substring
(2, t
.length
- 4).remove_underscores
157 v
.toolcontext
.error
(location
, "Error: invalid binary literal")
160 value
= s
.to_bin
.to_b
164 redef class AOctByteExpr
165 redef fun accept_literal
(v
)
167 var t
= self.n_oct_bytenum
.text
168 var s
= t
.substring
(2, t
.length
- 4).remove_underscores
170 v
.toolcontext
.error
(location
, "Error: invalid octal literal")
173 value
= s
.to_oct
.to_b
177 redef class AFloatExpr
178 # The value of the literal float once computed.
179 var value
: nullable Float
180 redef fun accept_literal
(v
)
182 self.value
= self.n_float
.text
.to_f
186 redef class ACharExpr
187 # The value of the literal char once computed.
188 var value
: nullable Char
189 redef fun accept_literal
(v
)
191 var txt
= self.n_char
.text
.unescape_nit
192 if txt
.length
!= 3 then
193 v
.toolcontext
.error
(self.hot_location
, "Syntax Error: invalid character literal `{txt}`.")
196 self.value
= txt
.chars
[1]
200 redef class AStringFormExpr
201 # The value of the literal string once computed.
202 var value
: nullable String
203 redef fun accept_literal
(v
)
205 var txt
= self.n_string
.text
208 if txt
.chars
[0] == txt
.chars
[1] and txt
.length
>= 6 then
211 if txt
.chars
[0] == '"' and txt
.chars
[3] == '\n' then behead
= 4 # ignore first \n in """
213 self.value
= txt
.substring
(behead
, txt
.length
- behead
- betail
).unescape_nit