File: integer_constant.e

package info (click to toggle)
smarteiffel 1.1-11
  • links: PTS
  • area: main
  • in suites: etch, etch-m68k
  • size: 12,288 kB
  • ctags: 40,785
  • sloc: ansic: 35,791; lisp: 4,036; sh: 1,783; java: 895; ruby: 613; python: 209; makefile: 115; csh: 78; cpp: 50
file content (289 lines) | stat: -rw-r--r-- 7,196 bytes parent folder | download | duplicates (2)
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
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
-- This file is part of SmartEiffel The GNU Eiffel Compiler Tools and Libraries
--
-- SmartEiffel is  free software;  you can redistribute it and/or  modify it
-- under  the terms of the  GNU General Public License, as published by  the
-- Free Software Foundation; either version 2, or (at your option) any later
-- version.
-- SmartEiffel is distributed in the hope that it will be useful but WITHOUT 
-- ANY WARRANTY;  without  even the implied warranty  of MERCHANTABILITY  or
-- FITNESS FOR A PARTICULAR PURPOSE.  See the GNU General Public License for
-- more details.  You should have received a copy of  the GNU General Public
-- License along with SmartEiffel;  see the file COPYING.  If not,  write to
-- the Free Software Foundation,  Inc., 59 Temple Place - Suite 330,  Boston, 
-- MA 02111-1307, USA.
--
-- Copyright(C) 1994-2002: INRIA - LORIA (INRIA Lorraine) - ESIAL U.H.P.
--			   - University of Nancy 1 - FRANCE
-- Copyright(C) 2003:      INRIA - LORIA (INRIA Lorraine) - I.U.T. Charlemagne
--			   - University of Nancy 2 - FRANCE
--
--		 Dominique COLNET, Suzanne COLLIN, Olivier ZENDRA,
--			   Philippe RIBET, Cyril ADRIAN
--
-- http://SmartEiffel.loria.fr - SmartEiffel@loria.fr
--
class INTEGER_CONSTANT
   --
   -- For Manifest Constant of class INTEGER.
   --

inherit BASE_TYPE_CONSTANT

creation make

creation {EIFFEL_PARSER} with, hexadecimal

feature

   is_static: BOOLEAN is True

   value_memory: INTEGER_64
	 -- We use here a 64 bit INTEGER to be able to store large values.

   size: INTEGER -- is 8, 16, 32 or 64.

   hexadecimal_flag: BOOLEAN
	 -- When the hexadecimal notation is used.

   underscore_flag: BOOLEAN
	 -- When the original notation use underscore characters.

   to_integer_or_error: INTEGER is
      do
	 Result := value_memory.to_integer
      end

   to_runnable(ct: E_TYPE): like Current is
      do
         Result := Current
	 if result_type = Void then
	    set_result_type
	 end
      end
   
   simplify_1_, simplify_2: like Current is
      do
	 Result := Current
      end
   
   compile_to_c is
      do
	 if hexadecimal_flag then
	    -- To view the corresponding hexadecimal value in the
	    -- generated C code:
	    string_buffer.copy(fz_open_c_comment)
	    append_in(string_buffer)
	    string_buffer.append(fz_close_c_comment)
	    cpp.put_string(string_buffer)	       
	 end
	 -- (Note: Unfortunately, it is not possible to use an inspect 
	 -- statement below because the INTEGER_64 tyepe is not 
	 -- supported by the current language definition.) 
	 if value_memory = Minimum_integer_8  then
	    cpp.put_string(once "INT8_MIN")
	 elseif value_memory = Minimum_integer_16 then
	    cpp.put_string(once "INT16_MIN")
	 elseif value_memory = Minimum_integer_32 then
	    cpp.put_string(once "INT32_MIN")
	 elseif value_memory = Minimum_integer_64 then
	    cpp.put_string(once "INT64_MIN")
	 else
	    cpp.put_string(once "INT")
	    cpp.put_integer(size)
	    cpp.put_string(once "_C(")
	    cpp.put_integer(value_memory)
	    cpp.put_character(')')
	 end
      end

   compile_target_to_jvm, compile_to_jvm is
      do
         code_attribute.opcode_push_integer(value_memory.to_integer)
      end

   jvm_branch_if_false: INTEGER is
      do
      end

   jvm_branch_if_true: INTEGER is
      do
      end

   compile_to_jvm_into(dest: E_TYPE): INTEGER is
      do
         Result := standard_compile_to_jvm_into(dest)
      end

   to_string: STRING is
      local
	 buffer: STRING
      do
	 buffer := string_buffer
	 buffer.clear
	 append_in(buffer)
         Result := buffer.twin
      end

   append_in(buffer: STRING) is
	 -- Append in `buffer' Eiffel prettifyed view of `Current'.
      local
	 i: INTEGER
      do
	 if hexadecimal_flag then
	    buffer.extend('0')
	    buffer.extend('x')
	    inspect
	       size
	    when 8 then
	       value_memory.to_integer_8.to_hexadecimal_in(buffer)
	    when 16 then
	       value_memory.to_integer_16.to_hexadecimal_in(buffer)
	    when 32 then
	       value_memory.to_integer_32.to_hexadecimal_in(buffer)
	    when 64 then
	       value_memory.to_hexadecimal_in(buffer)
	    end
	 else
	    value_memory.append_in(buffer)
	    if underscore_flag then
	       from
		  i := buffer.count - 2
	       until
		  i <= 1
	       loop
		  buffer.insert_character('_', i)
		  i := i - 3
	       end
	    end
	 end
      end
   
feature {TMP_FEATURE}

   to_real_constant: REAL_CONSTANT is
      do
         create Result.make(start_position, value_memory.to_string)
      end

feature {INTEGER_CONSTANT_VISITOR}

   accept(visitor: INTEGER_CONSTANT_VISITOR) is
      do
         visitor.visit_integer_constant(Current)
      end

feature {NONE}
   
   with(uf: like underscore_flag; negative: BOOLEAN
	negative_value: INTEGER_64; sp: like start_position) is
	 -- The `negative_value' has to be negative because negative range 
	 -- is greater than the positive one.  
      require
	 negative_value <= 0
      do
	 underscore_flag := uf
         start_position := sp
	 if negative then
	    value_memory := negative_value
	    if value_memory >= -128 then
	       size := 8
	    elseif value_memory >= -32768 then
	       size := 16
	    elseif value_memory >= -2147483648 then
	       size := 32
	    else
	       size := 64
	    end
	 else
	    value_memory := -negative_value
	    if value_memory <= 127 then
	       size := 8
	    elseif value_memory <= 32767 then
	       size := 16
	    elseif value_memory <= 2147483647 then
	       size := 32
	    else
	       size := 64
	    end
	 end
      ensure
	 underscore_flag = uf
      end

   hexadecimal(negative: BOOLEAN; digit_count: INTEGER_8;
	       a_value: INTEGER_64; sp: like start_position) is
      do
	 hexadecimal_flag := True
	 value_memory := a_value
	 inspect
	    digit_count
	 when 2 then
	    size := 8
	    if negative then
	       value_memory := a_value.to_integer_8 | 0x80
	    end
	 when 4 then
	    size := 16
	    if negative then
	       value_memory := a_value.to_integer_16 | 0x8000
	    end
	 when 8 then
	    size := 32
	    if negative then
	       value_memory := a_value.to_integer_32 | 0x80000000
	    end 
	 when 16 then
	    size := 64
	    if negative then
	       value_memory := a_value | 0x8000000000000000
	    end 
	 end
      end

   make(v: like value_memory; sp: like start_position) is
      do
	 if v >= 0 then
	    with(False, False, -v, sp)
	 else
	    with(False, True, v, sp)
	 end
	 set_result_type
      end
   
   string_buffer: STRING is
      once
	 create Result.make(32)
      end

   set_result_type is
      require
	 result_type = Void
      do
	 inspect
	    size
	 when 8 then
	    create {TYPE_INTEGER}
	    result_type.integer_8(start_position)
	 when 16 then
	    create {TYPE_INTEGER}
	    result_type.integer_16(start_position)
	 when 32 then
	    create {TYPE_INTEGER}
	    result_type.integer_32(start_position)
	 when 64 then
	    create {TYPE_INTEGER}
	    result_type.integer_64(start_position)
	 end
      ensure
	 result_type /= Void
      end
   
invariant

   size = 8 xor size = 16 xor size = 32 xor size = 64

   underscore_flag implies not hexadecimal_flag
   
   hexadecimal_flag implies not underscore_flag 
   
end -- INTEGER_CONSTANT