-
Notifications
You must be signed in to change notification settings - Fork 9
/
Copy pathinput_parser_env.ml
351 lines (314 loc) · 10.9 KB
/
input_parser_env.ml
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
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
(* ========================================================================== *)
(* FPTaylor: A Tool for Rigorous Estimation of Round-off Errors *)
(* *)
(* Author: Alexey Solovyev, University of Utah *)
(* *)
(* This file is distributed under the terms of the MIT license *)
(* ========================================================================== *)
(* -------------------------------------------------------------------------- *)
(* Data structures for loaded variables, definitions, and expressions *)
(* -------------------------------------------------------------------------- *)
open Expr
open Rounding
open Eval
open Num
open Task
open Interval
type raw_expr =
| Identifier of string
| Numeral of num
| Raw_rounding of rnd_info * raw_expr
| Raw_u_op of string * raw_expr
| Raw_bin_op of string * raw_expr * raw_expr
| Raw_gen_op of string * raw_expr list
type raw_formula =
| Raw_le of raw_expr * raw_expr
| Raw_lt of raw_expr * raw_expr
| Raw_eq of raw_expr * raw_expr
type constant_def = {
const_name : string;
value : Const.t;
}
type var_def = {
var_type : value_type;
var_name : string;
lo_bound : Const.t;
hi_bound : Const.t;
uncertainty : Const.t;
}
type definition = {
def_name : string;
def_expr : expr;
}
type env = {
constants : (string, constant_def) Hashtbl.t;
variables : (string, var_def) Hashtbl.t;
definitions : (string, definition) Hashtbl.t;
mutable constraints : (string * formula) list;
mutable expressions : (string * expr) list;
}
let env = {
constants = Hashtbl.create 10;
variables = Hashtbl.create 10;
definitions = Hashtbl.create 10;
constraints = [];
expressions = [];
}
let reset () =
let clear = Hashtbl.clear in
Log.report `Debug "Resetting input_parser_env";
clear env.constants;
clear env.variables;
clear env.definitions;
env.constraints <- [];
env.expressions <- []
let create_env_from_task (task : task) =
reset ();
env.expressions <- [task.name, task.expression];
env.constraints <- task.constraints;
List.iter (fun (v : Task.var_info) ->
let var = {
var_name = v.var_name;
var_type = v.var_type;
lo_bound = v.lo_bound;
hi_bound = v.hi_bound;
uncertainty = v.uncertainty
} in
Hashtbl.add env.variables v.var_name var)
task.variables
let env_to_tasks () =
let vars = Hashtbl.fold
(fun _ v vs -> {
Task.var_name = v.var_name;
var_type = v.var_type;
lo_bound = v.lo_bound;
hi_bound = v.hi_bound;
uncertainty = v.uncertainty;
} :: vs)
env.variables [] in
let rec loop acc = function
| [] -> acc
| (name, e) :: es ->
let t = {
Task.name = name;
expression = e;
variables = vars;
constraints = env.constraints;
} in
loop (t :: acc) es in
List.rev (loop [] env.expressions)
let find_constant name =
Hashtbl.find env.constants name
let find_variable name =
Hashtbl.find env.variables name
let find_definition name =
Hashtbl.find env.definitions name
let all_variables () =
Hashtbl.fold (fun _ v s -> v :: s) env.variables []
let all_constraints () =
env.constraints
let variable_interval name =
let var = find_variable name in {
low = (Const.to_interval var.lo_bound).low;
high = (Const.to_interval var.hi_bound).high;
}
let get_low_bound name =
let v = find_variable name in
v.lo_bound
let get_high_bound name =
let v = find_variable name in
v.hi_bound
let get_var_type name =
let v = find_variable name in
v.var_type
(* Applies a given rounding operation recursively *)
let rec apply_raw_rounding rnd expr =
match expr with
| Identifier _ -> Raw_rounding (rnd, expr)
| Numeral _ -> Raw_rounding (rnd, expr)
(* Do not apply the rounding inside another rounding and
do not round twice *)
| Raw_rounding _ -> expr
| Raw_u_op (op, arg) ->
let e1 = apply_raw_rounding rnd arg in
Raw_rounding (rnd, Raw_u_op (op, e1))
| Raw_bin_op ("^", arg1, arg2) ->
let e1 = apply_raw_rounding rnd arg1 in
Raw_rounding (rnd, Raw_bin_op ("^", e1, arg2))
| Raw_bin_op (op, arg1, arg2) ->
let e1 = apply_raw_rounding rnd arg1 and
e2 = apply_raw_rounding rnd arg2 in
Raw_rounding (rnd, Raw_bin_op (op, e1, e2))
| Raw_gen_op (op, args) ->
let es = List.map (apply_raw_rounding rnd) args in
Raw_rounding (rnd, Raw_gen_op (op, es))
let apply_raw_rounding_to_formula rnd f =
match f with
| Raw_le (e1, e2) ->
Raw_le (apply_raw_rounding rnd e1, apply_raw_rounding rnd e2)
| Raw_lt (e1, e2) ->
Raw_lt (apply_raw_rounding rnd e1, apply_raw_rounding rnd e2)
| Raw_eq (e1, e2) ->
Raw_eq (apply_raw_rounding rnd e1, apply_raw_rounding rnd e2)
(* Builds an expression from a raw expression *)
let rec transform_raw_expr = function
| Raw_rounding (rnd, arg) ->
let e1 = transform_raw_expr arg in
Rounding (rnd, e1)
| Raw_u_op (str, arg) ->
let e1 = transform_raw_expr arg in
begin
match str with
| "-" -> U_op (Op_neg, e1)
| "abs" -> U_op (Op_abs, e1)
| "inv" -> U_op (Op_inv, e1)
| "sqrt" -> U_op (Op_sqrt, e1)
| "sin" -> U_op (Op_sin, e1)
| "cos" -> U_op (Op_cos, e1)
| "tan" -> U_op (Op_tan, e1)
| "asin" -> U_op (Op_asin, e1)
| "acos" -> U_op (Op_acos, e1)
| "atan" -> U_op (Op_atan, e1)
| "exp" -> U_op (Op_exp, e1)
| "log" -> U_op (Op_log, e1)
| "sinh" -> U_op (Op_sinh, e1)
| "cosh" -> U_op (Op_cosh, e1)
| "tanh" -> U_op (Op_tanh, e1)
| "asinh" -> U_op (Op_asinh, e1)
| "acosh" -> U_op (Op_acosh, e1)
| "atanh" -> U_op (Op_atanh, e1)
| "floor_power2" -> U_op (Op_floor_power2, e1)
| _ -> failwith ("transform_raw_expr: Unknown unary operation: " ^ str)
end
| Raw_bin_op (str, arg1, arg2) ->
let e1 = transform_raw_expr arg1 and
e2 = transform_raw_expr arg2 in
begin
match str with
| "+" -> Bin_op (Op_add, e1, e2)
| "-" -> Bin_op (Op_sub, e1, e2)
| "*" -> Bin_op (Op_mul, e1, e2)
| "/" -> Bin_op (Op_div, e1, e2)
| "max" -> Bin_op (Op_max, e1, e2)
| "min" -> Bin_op (Op_min, e1, e2)
| "^" -> Bin_op (Op_nat_pow, e1, e2)
| "sub2" -> Bin_op (Op_sub2, e1, e2)
| "interval" ->
let lo_bound = Const.to_interval (eval_const_expr e1) and
hi_bound = Const.to_interval (eval_const_expr e2) in
mk_interval_const { low = lo_bound.low; high = hi_bound.high }
| _ -> failwith ("transform_raw_expr: Unknown binary operation: " ^ str)
end
| Raw_gen_op (str, args) ->
let es = List.map transform_raw_expr args in
begin
match str with
| "fma" -> Gen_op (Op_fma, es)
| _ -> failwith ("transform_raw_expr: Unknown operation: " ^ str)
end
| Numeral n -> mk_num_const n
| Identifier name ->
try let def = find_definition name in def.def_expr
with Not_found ->
try let var = find_variable name in Var var.var_name
with Not_found ->
try let c = find_constant name in Const c.value
with Not_found ->
failwith (Printf.sprintf "Unknown identifier: '%s'" name)
(* Builds a formula from a raw formula *)
let transform_raw_formula = function
| Raw_le (r1, r2) -> Le (transform_raw_expr r1, transform_raw_expr r2)
| Raw_lt (r1, r2) -> Lt (transform_raw_expr r1, transform_raw_expr r2)
| Raw_eq (r1, r2) -> Eq (transform_raw_expr r1, transform_raw_expr r2)
(* Adds a constant to the environment *)
let add_constant name raw =
if Hashtbl.mem env.constants name then
failwith ("Constant " ^ name ^ " is already defined")
else
let expr = transform_raw_expr raw in
let c = {
const_name = name;
value = eval_const_expr expr;
} in
Hashtbl.add env.constants name c
let is_same_bounds lo hi =
try
let a = Const.to_num lo and
b = Const.to_num hi in
a =/ b
with _ -> false
(* Adds a variable to the environment *)
let add_variable_with_uncertainty var_type name (lo, hi, uncertainty) =
if Hashtbl.mem env.variables name then
failwith ("Variable " ^ name ^ " is already defined")
else
let lo_expr = transform_raw_expr lo and
hi_expr = transform_raw_expr hi and
u_expr = transform_raw_expr uncertainty in
let lo_bound = eval_const_expr lo_expr and
hi_bound = eval_const_expr hi_expr in
if var_type = real_type && is_same_bounds lo_bound hi_bound then
begin
Log.report `Info "Variable %s is a constant" name;
add_constant name lo
end
else
let v = {
var_type = var_type;
var_name = name;
lo_bound = eval_const_expr lo_expr;
hi_bound = eval_const_expr hi_expr;
uncertainty = eval_const_expr u_expr;
} in
Hashtbl.add env.variables name v
let add_variable var_type name (lo, hi) =
add_variable_with_uncertainty var_type name (lo, hi, Numeral (Int 0))
(* Adds a definition to the environment *)
let add_definition name raw =
if Hashtbl.mem env.definitions name then
failwith ("Definition " ^ name ^ " is already defined")
else
let expr = transform_raw_expr raw in
let d = {
def_name = name;
def_expr = expr;
} in
let _ = Hashtbl.add env.definitions name d in
expr
(* Adds a constraint to the environment *)
let add_constraint name raw =
let c = transform_raw_formula raw in
env.constraints <- env.constraints @ [name, c]
(* Adds a named expression to the environment. Also creates the corresponding definition. *)
let add_expression_with_name name raw =
let expr = add_definition name raw in
env.expressions <- env.expressions @ [(name, expr)]
(* Adds an expression to the environment *)
let add_expression raw =
let name = "Expression " ^ string_of_int (List.length env.expressions + 1) in
add_expression_with_name name raw
(* Prints a raw expression *)
let print_raw_expr fmt =
let p = Format.pp_print_string fmt in
let rec print fmt = function
| Identifier name -> p name
| Numeral n -> p (string_of_num n)
| Raw_rounding (rnd, e) ->
Format.fprintf fmt "%s(%a)" (rounding_to_string rnd) print e;
| Raw_u_op (op, e) ->
Format.fprintf fmt "%s(%a)" op print e;
| Raw_bin_op (op, e1, e2) ->
Format.fprintf fmt "%s(%a,%a)" op print e1 print e2;
| Raw_gen_op (op, es) ->
p op; p "("; Lib.print_list (print fmt) (fun () -> p ",") es; p ")";
in
print fmt
let print_raw_expr_std = print_raw_expr Format.std_formatter
let print_raw_formula fmt = function
| Raw_le (e1, e2) ->
Format.fprintf fmt "%a <= %a\n" print_raw_expr e1 print_raw_expr e2
| Raw_lt (e1, e2) ->
Format.fprintf fmt "%a < %a\n" print_raw_expr e1 print_raw_expr e2
| Raw_eq (e1, e2) ->
Format.fprintf fmt "%a == %a\n" print_raw_expr e1 print_raw_expr e2
let print_raw_formula_std = print_raw_formula Format.std_formatter