Edit: Of course the module Num is already present in the standard library. I've renamed the module to Arith.
The newly-released OCaml 3.12 includes many extensions to the sub-language of modules. One of the simplest but most practical is the syntax for locally opening modules (let open M in e), and for evaluating expressions in the context of an implicitly-opened module (M.(e), equivalent to the former). The biggest payoff this syntax affords is overloading operators and functions in an expression, in a delimited context denoted by a module used as a dictionary:
let degree n = let pi = 3.1415926535897931 in Arith.F.(2. * pi / 180. * of_int n)
Note that the operators are the usual ones! Another neat example:
let rgba r g b a = Arith.I32.((of_int r lsl 8 lor of_int g) lsl 8 lor of_int b) lsl 8 lor of_int a)
(I've purposefully written the example to showcase the operator precedence, not for clarity). Unfortunately, the standard library doesn't yet include the modules necessary for this to work. Here's my version of the built-in numeric instances, suitable for inclusion in your .ocamlinit file. It is structured as a top-level module Arith, but can be put into a arith.ml file for separate compilation (if you do that, take care to include in the .mli interface file the complete module signature, including externals, so that the compiler can inline the definitions). This module contains sub-modules with definitions for each of the types int, int32, int64, Big_int, float, and Ratio. Every sub-module conforms to the NUM signature (inspired by the type classes in Haskell's Prelude):
module type NUM = sig type t val min_value : t val max_value : t val of_int : int -> t val to_int : t -> int val of_string : string -> t val to_string : t -> string val ( ~- ) : t -> t val ( ~+ ) : t -> t val ( + ) : t -> t -> t val ( - ) : t -> t -> t val ( * ) : t -> t -> t val ( / ) : t -> t -> t val (mod ) : t -> t -> t val abs : t -> t end
so that with the following top-level definitions:
let show (type t) d = let module N = (val d : NUM with type t = t) in N.to_string let read (type t) d = let module N = (val d : NUM with type t = t) in N.of_string
(note the syntax for modules as first-class values) the following code works:
# read (module Arith.I : NUM with type t = int) "123" ;; - : int = 123 # read (module Arith.I32 : NUM with type t = int32) "123" ;; - : int32 = 123l # read (module Arith.I64 : NUM with type t = int64) "123" ;; - : int64 = 123L # read (module Arith.F : NUM with type t = float) "123" ;; - : float = 123.
(the syntax for binding first-class module values is pretty heavy). They also conform to the ORD signature (also borrowed from Haskell):
module type ORD = sig type t val compare : t -> t -> int val ( = ) : t -> t -> bool val ( <> ) : t -> t -> bool val ( < ) : t -> t -> bool val ( <= ) : t -> t -> bool val ( > ) : t -> t -> bool val ( >= ) : t -> t -> bool end
so that the following code is generic on the module implementing it:
let max (type t) d (x : t) (y : t) : t = let module N = (val d : ORD with type t = t) in N.(if x < y then y else x) let min (type t) d (x : t) (y : t) : t = let module N = (val d : ORD with type t = t) in N.(if x < y then x else y)
The sub-modules have short, mnemonic names I, I32, I64, Z, F and Q so that they don't clash with the corresponding standard modules. The first four, the binary integral types, conform to the following BIN signature:
module type BIN = sig type t val succ : t -> t val pred : t -> t val (land) : t -> t -> t val (lor ) : t -> t -> t val (lxor) : t -> t -> t val lnot : t -> t val (lsl ) : t -> int -> t val (lsr ) : t -> int -> t val (asr ) : t -> int -> t end
So, for those of you that can't or won't avail yourselves to extension libraries like OCaml Batteries, here is the complete code for the module Arith:
module Arith = struct
module I = struct
type t = int
let min_value : t = 1 lsl (if 1 lsl 31 = 0 then 30 else 62)
let max_value : t = min_value - 1
external of_int : int -> t = "%identity"
external to_int : t -> int = "%identity"
external of_string : string -> t = "caml_int_of_string"
let to_string : t -> string = Pervasives.string_of_int
external ( ~- ) : t -> t = "%negint"
external ( ~+ ) : t -> t = "%identity"
external ( + ) : t -> t -> t = "%addint"
external ( - ) : t -> t -> t = "%subint"
external ( * ) : t -> t -> t = "%mulint"
external ( / ) : t -> t -> t = "%divint"
external (mod ) : t -> t -> t = "%modint"
let abs (x: t) : t = if x >= 0 then x else -x
let compare : t -> t -> int = Pervasives.compare
let ( = ) : t -> t -> bool = Pervasives.( = )
let ( <> ) : t -> t -> bool = Pervasives.( <> )
let ( < ) : t -> t -> bool = Pervasives.( < )
let ( <= ) : t -> t -> bool = Pervasives.( <= )
let ( > ) : t -> t -> bool = Pervasives.( > )
let ( >= ) : t -> t -> bool = Pervasives.( >= )
external succ : t -> t = "%succint"
external pred : t -> t = "%predint"
external (land) : t -> t -> t = "%andint"
external (lor ) : t -> t -> t = "%orint"
external (lxor) : t -> t -> t = "%xorint"
let lnot (x: t) : t = x lxor (-1)
external (lsl ) : t -> int -> t = "%lslint"
external (lsr ) : t -> int -> t = "%lsrint"
external (asr ) : t -> int -> t = "%asrint"
end
module I32 = struct
type t = int32
let min_value : t = Int32.min_int
let max_value : t = Int32.max_int
external of_int : int -> t = "%int32_of_int"
external to_int : t -> int = "%int32_to_int"
external of_string : string -> t = "caml_int32_of_string"
let to_string : t -> string = Int32.to_string
external ( ~- ) : t -> t = "%int32_neg"
external ( ~+ ) : t -> t = "%identity"
external ( + ) : t -> t -> t = "%int32_add"
external ( - ) : t -> t -> t = "%int32_sub"
external ( * ) : t -> t -> t = "%int32_mul"
external ( / ) : t -> t -> t = "%int32_div"
external (mod ) : t -> t -> t = "%int32_mod"
let abs (x: t) : t = if x >= 0l then x else -x
let compare : t -> t -> int = Pervasives.compare
let ( = ) : t -> t -> bool = Pervasives.( = )
let ( <> ) : t -> t -> bool = Pervasives.( <> )
let ( < ) : t -> t -> bool = Pervasives.( < )
let ( <= ) : t -> t -> bool = Pervasives.( <= )
let ( > ) : t -> t -> bool = Pervasives.( > )
let ( >= ) : t -> t -> bool = Pervasives.( >= )
let succ : t -> t = Int32.succ
let pred : t -> t = Int32.pred
external (land) : t -> t -> t = "%int32_and"
external (lor ) : t -> t -> t = "%int32_or"
external (lxor) : t -> t -> t = "%int32_xor"
let lnot (x: t) : t = x lxor (-1l)
external (lsl ) : t -> int -> t = "%int32_lsl"
external (lsr ) : t -> int -> t = "%int32_asr"
external (asr ) : t -> int -> t = "%int32_lsr"
end
module I64 = struct
type t = int64
let min_value : t = Int64.min_int
let max_value : t = Int64.max_int
external of_int : int -> t = "%int64_of_int"
external to_int : t -> int = "%int64_to_int"
external of_string : string -> t = "caml_int64_of_string"
let to_string : t -> string = Int64.to_string
external ( ~- ) : t -> t = "%int64_neg"
external ( ~+ ) : t -> t = "%identity"
external ( + ) : t -> t -> t = "%int64_add"
external ( - ) : t -> t -> t = "%int64_sub"
external ( * ) : t -> t -> t = "%int64_mul"
external ( / ) : t -> t -> t = "%int64_div"
external (mod ) : t -> t -> t = "%int64_mod"
let abs (x: t) : t = if x >= 0L then x else -x
let compare : t -> t -> int = Pervasives.compare
let ( = ) : t -> t -> bool = Pervasives.( = )
let ( <> ) : t -> t -> bool = Pervasives.( <> )
let ( < ) : t -> t -> bool = Pervasives.( < )
let ( <= ) : t -> t -> bool = Pervasives.( <= )
let ( > ) : t -> t -> bool = Pervasives.( > )
let ( >= ) : t -> t -> bool = Pervasives.( >= )
let succ : t -> t = Int64.succ
let pred : t -> t = Int64.pred
external (land) : t -> t -> t = "%int64_and"
external (lor ) : t -> t -> t = "%int64_or"
external (lxor) : t -> t -> t = "%int64_xor"
let lnot (x: t) : t = x lxor (-1L)
external (lsl ) : t -> int -> t = "%int64_lsl"
external (lsr ) : t -> int -> t = "%int64_asr"
external (asr ) : t -> int -> t = "%int64_lsr"
end
module Z = struct
type t = Big_int.big_int
let min_value : t = Big_int.zero_big_int
let max_value : t = Big_int.zero_big_int
let of_int : int -> t = Big_int.big_int_of_int
let to_int : t -> int = Big_int.int_of_big_int
let of_string : string -> t = Big_int.big_int_of_string
let to_string : t -> string = Big_int.string_of_big_int
let ( ~- ) : t -> t = Big_int.minus_big_int
external ( ~+ ) : t -> t = "%identity"
let ( + ) : t -> t -> t = Big_int.add_big_int
let ( - ) : t -> t -> t = Big_int.sub_big_int
let ( * ) : t -> t -> t = Big_int.mult_big_int
let ( / ) : t -> t -> t = Big_int.div_big_int
let (mod ) : t -> t -> t = Big_int.mod_big_int
let abs : t -> t = Big_int.abs_big_int
let compare : t -> t -> int = Big_int.compare_big_int
let ( = ) : t -> t -> bool = Big_int.eq_big_int
let ( <> ) (x:t) (y:t) = not (x = y)
let ( < ) : t -> t -> bool = Big_int.lt_big_int
let ( <= ) : t -> t -> bool = Big_int.le_big_int
let ( > ) : t -> t -> bool = Big_int.gt_big_int
let ( >= ) : t -> t -> bool = Big_int.ge_big_int
let succ : t -> t = Big_int.succ_big_int
let pred : t -> t = Big_int.pred_big_int
let (land) : t -> t -> t = Big_int.and_big_int
let (lor ) : t -> t -> t = Big_int.or_big_int
let (lxor) : t -> t -> t = Big_int.xor_big_int
let lnot : t -> t = let m1 = of_int (-1) in fun x -> x lxor m1
let (lsl ) : t -> int -> t = Big_int.shift_left_big_int
let (lsr ) : t -> int -> t = Big_int.shift_right_big_int
let (asr ) : t -> int -> t = Big_int.shift_right_towards_zero_big_int
end
module F = struct
type t = float
let min_value : t = Int64.float_of_bits 0xFF_F0_00_00_00_00_00_00L
let max_value : t = Int64.float_of_bits 0x7F_F0_00_00_00_00_00_00L
external of_int : int -> t = "%floatofint"
external to_int : t -> int = "%intoffloat"
external of_string : string -> t = "caml_float_of_string"
let to_string : t -> string = Pervasives.string_of_float
external ( ~- ) : t -> t = "%negfloat"
external ( ~+ ) : t -> t = "%identity"
external ( + ) : t -> t -> t = "%addfloat"
external ( - ) : t -> t -> t = "%subfloat"
external ( * ) : t -> t -> t = "%mulfloat"
external ( / ) : t -> t -> t = "%divfloat"
external (mod ) : t -> t -> t = "caml_modf_float"
let abs (x: t) : t = if x >= 0. then x else -x
external compare : t -> t -> int = "%compare"
external ( = ) : t -> t -> bool = "%equal"
external ( <> ) : t -> t -> bool = "%notequal"
external ( < ) : t -> t -> bool = "%lessthan"
external ( <= ) : t -> t -> bool = "%lessequal"
external ( > ) : t -> t -> bool = "%greaterthan"
external ( >= ) : t -> t -> bool = "%greaterequal"
end
module Q = struct
type t = Ratio.ratio
let min_value : t =
let flag = Arith_status.get_error_when_null_denominator () in
Arith_status.set_error_when_null_denominator false;
let v = Ratio.minus_ratio (Ratio.create_ratio Big_int.unit_big_int Big_int.zero_big_int) in
Arith_status.set_error_when_null_denominator flag;
v
let max_value : t =
let flag = Arith_status.get_error_when_null_denominator () in
Arith_status.set_error_when_null_denominator false;
let v = Ratio.create_ratio Big_int.unit_big_int Big_int.zero_big_int in
Arith_status.set_error_when_null_denominator flag;
v
let of_int : int -> t = Ratio.ratio_of_int
let to_int : t -> int = Ratio.int_of_ratio
let of_string : string -> t = Ratio.ratio_of_string
let to_string : t -> string = Ratio.string_of_ratio
let ( ~- ) : t -> t = Ratio.minus_ratio
external ( ~+ ) : t -> t = "%identity"
let ( + ) : t -> t -> t = Ratio.add_ratio
let ( - ) : t -> t -> t = Ratio.sub_ratio
let ( * ) : t -> t -> t = Ratio.mult_ratio
let ( / ) : t -> t -> t = Ratio.div_ratio
let (mod ) (x:t) (y:t) : t =
Ratio.sub_ratio x
(Ratio.mult_ratio y
(Ratio.ratio_of_big_int
(Ratio.floor_ratio
(Ratio.div_ratio x y))))
let abs : t -> t = Ratio.abs_ratio
let compare : t -> t -> int = Ratio.compare_ratio
let ( = ) : t -> t -> bool = Ratio.eq_ratio
let ( <> ) (x:t) (y:t) = not (x = y)
let ( < ) : t -> t -> bool = Ratio.lt_ratio
let ( <= ) : t -> t -> bool = Ratio.le_ratio
let ( > ) : t -> t -> bool = Ratio.gt_ratio
let ( >= ) : t -> t -> bool = Ratio.ge_ratio
end
end
In the case of Z, there are no meaningful extremal values. I haven't included a module for NativeInt, but you can do so quite easily. Note that, if any of the external functions in the standard library changes, this module must be revised. I hope you find it useful.
7 comments:
That's cool (apart maybe from naming your module Num which already exists) but you cannot do:
# Num.(3**(561-1) mod 561);;
> - : Num.num =
I am working at the moment on interval arithmetic and the ability to overload constants is very handy (think of 0.1).
@Chris, first of all, I've changed the module name to Arith, to avoid confusion. As to the overloading of constants, in the words of X. Leroy, "Taking a leaf from Christophe Troestler's 'delimited overloading' package, but much less powerful". How does pa_do play with 3.12?
Yesterday, I have released a tarball of delimited overloading that compiles with 3.12. There are some more tweaks I'd like to do (especially enabling by default local open so there is no difference with 3.12 when no overloading is defined) but that will have to wait a little longer.
@ChriS, as I'm saddled with Windows 7 I can't compile OMake (precompiled versions don't work with Cygwin 1.7), so I can't compile pa_do, nor Batteries, no Jane St. lib, nor... Not even GODI works.
I'm sure some people are in the same situation I am, struggling with setting up a workable environment in Windows. Maybe for them the new 3.12 can bring a bit of the ease that other libraries bring to a Unix distribution.
There is a Makefile in the latest tarball. It should compiles and install pa_do fine.
@ChriS,
It works! I've bitten the bullet and ported OMake to Cygwin. With that in place, the only adjustments needed were to pa_do's OMakeroot. Most tests from ``File "op_concrete.ml", line 57, characters 55-57:'' on fail, however, with error "The filename, directory name, or volume label syntax is incorrect."
A colleague of mine has windows 7 and I installed OCaml with cygwin. This is an endless trouble with a program regularly failing with an error box but succeeding without a hitch if you run it a second time. I was thinking to move away from cygwin but it requires some time (the program depends on ocamlnet) that I currently do not have...
Post a Comment