-
Erwan Jahier authoredErwan Jahier authored
evalConst.ml 16.48 KiB
(** Time-stamp: <modified the 01/07/2008 (at 14:00) by Erwan Jahier> *)
open Printf
open Lxm
open Errors
open SyntaxTree
open SyntaxTreeCore
open CompiledData
open Predef
open PredefEvalConst
open PredefEvalType
(*----------------------------------------------------
EvalArray_error :
- leve par les fonctions ddies aux tableaux
----------------------------------------------------*)
exception EvalArray_error of string
(*----------------------------------------------------
EvalConst_error :
- leve localement dans les sous-fonctions,
- capte dans EvalConst.f et tranforme en Compile_error.
----------------------------------------------------*)
let finish_me msg = print_string ("\n\tXXX evalConst.ml:"^msg^" -> finish me!\n")
let not_evaluable_construct str =
raise (EvalConst_error(
Printf.sprintf "The construct %s is not allowed in static expression"
str))
(*----------------------------------------------------
Utilitaire :
extraire une tranche de tableau
N.B. first_ix last_ix step et width sont supposs
venir de eva et donc tre corrects
N.B. Puisque correct, last_ix est inutile, mais bon ...
-----------------------------------------------------*)
let (make_slice_const :
const_eff array -> type_eff -> slice_info_eff -> const_eff list) =
fun ctab ctype slice ->
let get_res (ix : int) = Array.get ctab (slice.se_first + ix*slice.se_step) in
[Array_const_eff (Array.init slice.se_width get_res, ctype)]
(** Utilitaire : fabriquer si possible une constante tableau *)
let (make_array_const : const_eff list array -> const_eff) =
fun ops ->
let expected_type = ref None in
let treat_arg (op : const_eff list) =
match op with
| [x] -> (
(* non tuple *)
let xtyp = type_of_const_eff x in
match (!expected_type) with
| None -> expected_type := Some xtyp; x
| Some t -> (
if (t = xtyp) then x else
raise (EvalConst_error(
"type error in array, "^
(CompiledDataDump.string_of_type_eff xtyp)^
" mixed with " ^ CompiledDataDump.string_of_type_eff t
))
)
)
| _ -> (* tuple *)
raise (EvalConst_error("array of tuple not allowed"))
in
let res = Array.map treat_arg ops in
match (!expected_type) with
| None -> raise (EvalConst_error("empty array"))
| Some t -> Array_const_eff(res, t)
(** Utilitaire : fabriquer si possible une constante structure
N.B. Par construction on sait que nops n'a pas de doublons
*)
let make_struct_const
(teff : type_eff)
(arg_tab : (Ident.t, Lxm.t * const_eff) Hashtbl.t) =
(
(* on verifie qu'on a bien un type struct *)
match teff with
Struct_type_eff (tnm, flst) -> (
(* on construit la liste dans le BON ordre *)
let make_eff_field ((fn: Ident.t),((ft:type_eff),(fv:const_eff option))) = (
try (
(* on prend en priorit dans arg_tab *)
match (Hashtbl.find arg_tab fn) with
(lxm, v) -> (
(* effet de bord : on vire la valeur de arg_tab *)
Hashtbl.remove arg_tab fn ;
let vt = type_of_const_eff v in
if (vt = ft) then (fn, v) (*ok*)
else raise (Compile_error(
lxm ,
sprintf
"\n*** type error in struct %s, %s instead of %s"
(Ident.string_of_long tnm)
(CompiledDataDump.string_of_type_eff vt)
(CompiledDataDump.string_of_type_eff ft)
))
)
) with Not_found -> (
(* sinon la valeur par dfaut *)
match fv with
Some v -> (fn, v) (* ok : v correcte par construction *)
| None ->
raise (EvalConst_error(
sprintf
"bad struct expression, no value given for field %s"
(Ident.to_string fn)
))
)
) in
(* on mappe flst pour avoir la liste dans le bon ordre *)
let eff_fields = List.map make_eff_field flst in
(* si arg_tab n'est pas vide, erreur sur le premier *)
let raise_error (id : Ident.t) ((lxm : Lxm.t), (veff : const_eff))
= raise(Compile_error(
lxm,
sprintf
"\n*** %s is not a field of struct %s"
(Ident.to_string id)
(CompiledDataDump.string_of_type_eff(teff))
))
in
Hashtbl.iter raise_error arg_tab ;
(* ok : tout s'est bien pass ! *)
Struct_const_eff (eff_fields, teff)
)
| _ -> raise (EvalConst_error(
sprintf
"struct type expected instead of %s"
(CompiledDataDump.string_of_type_eff teff)
))
)
let l2ll l = if l = [] then [] else [l]
(*----------------------------------------------------
Evaluation rcursive des expressions constantes
------------------------------------------------------
f :
- entres : id_solver et val_exp
- sortie : const_eff list
- Effet de bord : Compile_error
Rle :
-> rsoud les rfrences aux idents
-> gre les appels rcursifs (valuation des arguments)
----------------------------------------------------*)
let rec f
(env : id_solver)
(vexp : val_exp)
= (
(*-----------------------------------
fonction rcursive principale
-> capte les nv
-> rcupre les EvalConst_error
-----------------------------------*)
let rec rec_eval_const (vexp : val_exp) = (
match vexp with
| CallByPos ({it=posop; src=lxm}, Oper args) -> (
try eval_by_pos_const posop lxm args
with
| EvalType_error msg ->
raise (Compile_error(lxm, "type error: "^msg))
| EvalConst_error msg ->
raise (Compile_error(lxm, "\n*** can't eval constant: "^msg))
)
| CallByName ({it=nmop; src=lxm}, nmargs ) -> (
try eval_by_name_const nmop lxm nmargs
with EvalConst_error msg ->
raise (Compile_error(lxm, "\n*** can't eval constant: "^msg))
)
)
(*-----------------------------------
fonction rcursive secondaire
eval. exp classique (by pos)
N.B. On distingue les oprations
classiques (avec extention tableau
implicie) des autres. Ici, on traite
toutes les oprations non classiques.
-----------------------------------*)
and eval_by_pos_const
(posop : by_pos_op) (* l'operateur *)
(lxm : Lxm.t) (* source de l'oprateur *)
(args : val_exp list) (* arguments *)
= (
match (posop) with
(* capte les idents de constantes *)
IDENT_n id -> (
(* 2007-07 on interdit les externes *)
match (env.id2const id lxm) with
| Extern_const_eff(_,_, Some const_eff) -> [const_eff]
| Extern_const_eff(_,_,None) ->
raise (EvalConst_error(
sprintf "\n*** cannot access this abstract constant value"))
| x -> [ x ]
)
(* oprateur lazzy *)
| WITH_n(a0,a1,a2) -> (
match (rec_eval_const a0) with
[ Bool_const_eff true] -> rec_eval_const a1
| [ Bool_const_eff false] -> rec_eval_const a2
| x -> type_error_const x "bool"
)
(* mettre plat la liste des args *)
| TUPLE_n -> ( List.flatten (List.map rec_eval_const args))
(* les tableaux de tuples sont interdits *)
| HAT_n -> (
match args with
| [cexp; szexp] -> (
try
let sz = eval_array_size env szexp in
match rec_eval_const cexp with
| [cst] ->
let atab = Array.make sz cst in
[ Array_const_eff (atab, type_of_const_eff cst) ]
| x ->
raise (EvalConst_error("array of tuple not allowed"))
with
EvalArray_error msg -> raise(EvalConst_error msg)
)
| _ -> raise(EvalConst_error
(sprintf "arity error: 2 expected instead of %d"
(List.length args)))
)
| CONCAT_n -> (
let ops = (List.map rec_eval_const args) in
match ops with
| [[Array_const_eff (v0, t0)];
[Array_const_eff (v1, t1)]] -> (
if(t0 = t1) then
[Array_const_eff (Array.append v0 v1, t0)]
else
raise(EvalConst_error(
sprintf
"\n*** type combination error, can't concat %s with %s"
(CompiledDataDump.string_of_type_eff(t0))
(CompiledDataDump.string_of_type_eff(t1))
))
)
| [_;_] ->
raise(EvalConst_error(
"type combination error, array type expected"))
| _ -> raise(EvalConst_error
(sprintf "arity error: 2 expected instead of %d"
(List.length ops)))
)
| ARRAY_n -> (
let ops = (List.map rec_eval_const args) in
[make_array_const (Array.of_list ops)]
)
| ARRAY_ACCES_n ix -> (
let effargs = List.flatten (List.map rec_eval_const args) in
match effargs with
| [Array_const_eff (elts, typelts)] -> (
try
let sz = Array.length elts in
let effix = eval_array_index env ix sz lxm in
[Array.get elts effix ]
with EvalArray_error msg -> raise(EvalConst_error msg)
)
| _ -> type_error_const effargs "some array"
)
| ARRAY_SLICE_n sl -> (
let (elts, typelts) =
match List.flatten (List.map rec_eval_const args) with
| [Array_const_eff (l, t)] -> (l, t)
| x -> type_error_const x "some array"
in
(* on en dduit la taille du tableau *)
let sz = Array.length elts in
(* value la slice *)
try
let sliceff = eval_array_slice env sl sz lxm in
make_slice_const elts typelts sliceff
with
EvalArray_error msg -> raise(EvalConst_error msg)
)
| STRUCT_ACCESS_n fid ->
let ceff_list = List.flatten (List.map rec_eval_const args) in
(match ceff_list with
| [Struct_const_eff (flst, typ)] -> (
try [(List.assoc fid flst)]
with Not_found ->
raise (EvalConst_error
(Printf.sprintf "%s is not a field of struct %s"
(Ident.to_string fid)
(CompiledDataDump.string_of_type_eff(typ))))
)
| [x] -> type_error_const [x] "struct type"
| x -> arity_error_const x "1"
)
| CALL_n _ -> not_evaluable_construct "node call"
| MERGE_n _ -> not_evaluable_construct "merge"
| WHEN_n -> not_evaluable_construct "when"
| FBY_n -> not_evaluable_construct "fby"
| ARROW_n -> not_evaluable_construct "->"
| CURRENT_n -> not_evaluable_construct "current"
| PRE_n -> not_evaluable_construct "pre"
| Predef(op,sargs)
->
if sargs = [] then
let effargs = (List.map rec_eval_const args) in
PredefEvalConst.f op lxm [] effargs
else
(* Well, it migth be possible after all... TODO *)
not_evaluable_construct (op2string op)
) (* FIN DE : eval_by_pos_const *)
(*-------------------------------------*)
(* Fonction rcursive secondaire *)
(*-------------------------------------*)
(* -> Eval. d'une expression spciale *)
(* "par nom" *)
(*-------------------------------------*)
and eval_by_name_const
(namop : by_name_op) (* l'operateur *)
(lxm : Lxm.t) (* source de l'oprateur *)
(namargs : (Ident.t srcflagged * val_exp) list) (* arguments *)
= (
match namop with
| STRUCT_anonymous_n ->
finish_me "anonymous struct";
assert false
| STRUCT_n opid -> (
(* effet de bord : on tabule les param effectif *)
let arg_tab = Hashtbl.create 50 in
let treat_one_arg ((pid:Ident.t srcflagged), (pexp:val_exp)) =
if
(Hashtbl.mem arg_tab pid.it)
then
raise(EvalConst_error(
sprintf
"multiple definition of param %s in %s call"
(Ident.to_string pid.it)
(Ident.string_of_idref opid)))
else
let v = rec_eval_const pexp in
match v with
| [x] -> Hashtbl.add arg_tab pid.it (pid.src, x)
| _ ->
raise(
EvalConst_error(
sprintf
"unexpected tuple value for param %s in %s call"
(Ident.to_string pid.it)
(Ident.string_of_idref opid)
))
in
List.iter treat_one_arg namargs ;
(* pour l'instant, on ne traite que les constructions de struct *)
try let teff = env.id2type opid lxm in
[make_struct_const teff arg_tab]
with _ ->
raise(EvalConst_error(
sprintf "struct type expected instead of %s"
(Ident.string_of_idref opid)))
)
) (* FIN DE : eval_by_name_const *)
(*-------------------------------------*)
(* Corps de la fonction principale *)
(*-------------------------------------*)
in
rec_eval_const vexp
) (* fin de f *)
(*---------------------------------------------------------------------
eval_array_size
-----------------------------------------------------------------------
Rle : calcule une taille de tableau
Entres:
Sorties :
int (strictement positif)
Effets de bord :
EvalArray_error "bad array size, type int expected but get <t>" si t pas int
EvalArray_error "bad array size <n>" si n <= 0
----------------------------------------------------------------------*)
and (eval_array_size: id_solver -> val_exp -> int) =
fun id_solver szexp ->
match (f id_solver szexp) with
| [Int_const_eff sz] ->
if (sz > 0) then sz else
raise(EvalArray_error(sprintf "bad array size %d" sz))
| [x] ->
raise(EvalArray_error(sprintf "bad array size, int expected but get %s"
(CompiledDataDump.string_of_type_eff(type_of_const_eff x))))
| _ ->
raise(EvalArray_error(sprintf "bad array size, int expected, not a tuple"))
(*---------------------------------------------------------------------
eval_array_index
-----------------------------------------------------------------------
Rle :
Entres :
id_solver, val_exp, taille du tableau
Sorties :
int (entre 0 et taille du tableau -1
Effets de bord :
EvalArray_error msg si pas bon
----------------------------------------------------------------------*)
and eval_array_index
(env : id_solver)
(ixexp : val_exp)
(sz : int)
(lxm : Lxm.t)
=
try
(
match (f env ixexp) with
| [Int_const_eff i]
| [Extern_const_eff(_,_, Some (Int_const_eff i))] -> check_int i sz
| [Extern_const_eff(id,_,None)] ->
raise(EvalArray_error("The extern const " ^ (Ident.string_of_long id) ^
" is abstract"))
| [Extern_const_eff(_,_, Some x)]
| [x] -> raise(EvalArray_error(sprintf
"bad array index, int expected but get %s"
(CompiledDataDump.string_of_type_eff(type_of_const_eff x)))
)
| _ -> raise(EvalArray_error(
sprintf "bad array index, int expected but get a tuple"))
)
with
EvalArray_error msg ->
raise (Compile_error(lxm, "\n*** can't eval constant: "^msg))
and check_int i sz =
if ((i >= 0) && (i < sz)) then i
else raise(EvalArray_error(
sprintf "array index %d out of bounds 0..%d" i (sz-1)))
(*---------------------------------------------------------------------
eval_array_slice
-----------------------------------------------------------------------
Rle :
Entres :
id_solver, slice_info, size du tableau,
lxm (source de l'opration slice pour warning)
Sorties :
slice_info_eff, i.e.
(fisrt,last,step,width) tels que step <> 0 et
- si step > 0 alors 0<=first<=last<=sz
- si step < 0 alors 0<=last<=first<=sz
- 1<=width<=sz
Effets de bord :
EvalArray_error msg si pas bon
----------------------------------------------------------------------*)
and eval_array_slice (env : id_solver) (sl : slice_info) (sz : int) (lxm : Lxm.t) =
try
let first_ix = eval_array_index env sl.si_first sz lxm in
let last_ix = eval_array_index env sl.si_last sz lxm in
let step =
match sl.si_step with
| Some stepexp -> (
match (f env stepexp) with
| [Int_const_eff s] -> s (* ok *)
| [x] -> raise(EvalArray_error(
sprintf "bad array step, int expected but get %s"
(CompiledDataDump.string_of_type_eff (type_of_const_eff x))))
| _ -> raise(EvalArray_error(
sprintf "bad array step, int expected but get a tuple"))
)
| None -> if (first_ix <= last_ix) then 1 else -1
in
if
(step = 0)
|| ((step > 0) && (first_ix > last_ix))
|| ((step < 0) && (first_ix < last_ix))
then
let msg = sprintf "bad array slice [%d..%d] step %d" first_ix last_ix step in
raise (EvalArray_error msg)
else
(* index relatif du dernier *)
let last_rel = abs (last_ix-first_ix) in
let abs_step = abs step in
(* le dernier est-il pris dans la tranche ? *)
if ((last_rel mod abs_step) <> 0) then
warning lxm (sprintf "last index out of slice [%d..%d step %d]"
first_ix last_ix step);
let width = 1 + last_rel/abs_step in
(* on force le dernier a tre dans la tranche *)
let real_last_ix = first_ix + (width-1) * step in
(* (first_ix,last_ix,step,width) *)
{
se_first = first_ix;
se_last = real_last_ix;
se_step = step;
se_width = width
}
with
EvalArray_error msg ->
raise (Compile_error(lxm, "\n*** can't eval constant: "^msg))