blob: eefe37d5d0b622fd05b98e46974c8cbae8f71a3d [file] [log] [blame] [edit]
open Types
(* Values and operators *)
type ('i32, 'i64, 'f32, 'f64) op =
I32 of 'i32 | I64 of 'i64 | F32 of 'f32 | F64 of 'f64
type ('v128) vecop =
V128 of 'v128
type num = (I32.t, I64.t, F32.t, F64.t) op
type vec = (V128.t) vecop
type ref_ = ..
type ref_ += NullRef of ref_type
type value = Num of num | Vec of vec | Ref of ref_
(* Injection & projection *)
let as_num = function
| Num n -> n
| _ -> failwith "as_num"
let as_vec = function
| Vec i -> i
| _ -> failwith "as_vec"
let as_ref = function
| Ref r -> r
| _ -> failwith "as_ref"
exception TypeError of int * num * num_type
module type NumType =
sig
type t
val to_num : t -> num
val of_num : int -> num -> t
end
module I32Num =
struct
type t = I32.t
let to_num i = I32 i
let of_num n = function I32 i -> i | v -> raise (TypeError (n, v, I32Type))
end
module I64Num =
struct
type t = I64.t
let to_num i = I64 i
let of_num n = function I64 i -> i | v -> raise (TypeError (n, v, I64Type))
end
module F32Num =
struct
type t = F32.t
let to_num i = F32 i
let of_num n = function F32 z -> z | v -> raise (TypeError (n, v, F32Type))
end
module F64Num =
struct
type t = F64.t
let to_num i = F64 i
let of_num n = function F64 z -> z | v -> raise (TypeError (n, v, F64Type))
end
module type VecType =
sig
type t
val to_vec : t -> vec
val of_vec : int -> vec -> t
end
module V128Vec =
struct
type t = V128.t
let to_vec i = V128 i
let of_vec n = function V128 z -> z
end
(* Typing *)
let type_of_num = function
| I32 _ -> I32Type
| I64 _ -> I64Type
| F32 _ -> F32Type
| F64 _ -> F64Type
let type_of_vec = function
| V128 _ -> V128Type
let type_of_ref' = ref (function NullRef t -> t | _ -> assert false)
let type_of_ref r = !type_of_ref' r
let type_of_value = function
| Num n -> NumType (type_of_num n)
| Vec i -> VecType (type_of_vec i)
| Ref r -> RefType (type_of_ref r)
(* Comparison *)
let eq_num n1 n2 = n1 = n2
let eq_vec v1 v2 = v1 = v2
let eq_ref' = ref (fun r1 r2 ->
match r1, r2 with
| NullRef _, NullRef _ -> true
| _, _ -> false
)
let eq_ref r1 r2 = !eq_ref' r1 r2
let eq v1 v2 =
match v1, v2 with
| Num n1, Num n2 -> eq_num n1 n2
| Vec v1, Vec v2 -> eq_vec v1 v2
| Ref r1, Ref r2 -> eq_ref r1 r2
| _, _ -> false
(* Defaults *)
let default_num = function
| I32Type -> I32 I32.zero
| I64Type -> I64 I64.zero
| F32Type -> F32 F32.zero
| F64Type -> F64 F64.zero
let default_vec = function
| V128Type -> V128 V128.zero
let default_ref = function
| t -> NullRef t
let default_value = function
| NumType t' -> Num (default_num t')
| VecType t' -> Vec (default_vec t')
| RefType t' -> Ref (default_ref t')
(* Conversion *)
let value_of_bool b = Num (I32 (if b then 1l else 0l))
let string_of_num = function
| I32 i -> I32.to_string_s i
| I64 i -> I64.to_string_s i
| F32 z -> F32.to_string z
| F64 z -> F64.to_string z
let hex_string_of_num = function
| I32 i -> I32.to_hex_string i
| I64 i -> I64.to_hex_string i
| F32 z -> F32.to_hex_string z
| F64 z -> F64.to_hex_string z
let string_of_vec = function
| V128 v -> V128.to_string v
let hex_string_of_vec = function
| V128 v -> V128.to_hex_string v
let string_of_ref' = ref (function NullRef t -> "null" | _ -> "ref")
let string_of_ref r = !string_of_ref' r
let string_of_value = function
| Num n -> string_of_num n
| Vec i -> string_of_vec i
| Ref r -> string_of_ref r
let string_of_values = function
| [v] -> string_of_value v
| vs -> "[" ^ String.concat " " (List.map string_of_value vs) ^ "]"