blob: 8d63d489ee1c2bc58f1f95e3389cf8033077088d (
plain)
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
|
(*
This file is part of licht.
licht 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 3 of the License, or
(at your option) any later version.
licht 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 licht. If not, see <http://www.gnu.org/licenses/>.
*)
module type COMPARABLE = sig
type t
val eq : t -> t -> bool
val neq : t -> t -> bool
val lt : t -> t -> bool
val le : t -> t -> bool
val gt : t -> t -> bool
val ge : t -> t -> bool
end
module Comparable = struct
let eq = ( = )
let neq = ( <> )
let lt = ( < )
let le = ( <= )
let gt = ( > )
let ge = ( >= )
end
module Num = struct
let rnd () =
let value = Random.bits () in
Q.make (Z.of_int value) (Z.of_int (1 lsl 30))
include Q
let is_integer t = Q.den t == Z.one
let eq = Q.equal
let neq a b = not (Q.equal a b)
let mult = Q.mul
let floor t =
let num = Q.num t and den = Q.den t in
if is_integer t then Q.of_bigint num else Q.of_bigint @@ Z.fdiv num den
let round_down t =
let num = Q.num t and den = Q.den t in
if is_integer t then Q.of_bigint num else Q.of_bigint @@ Z.div num den
let round t =
if is_integer t then t
else
let t' =
match Q.sign t with
| 1 -> Q.add t @@ Q.of_ints 1 2
| -1 -> Q.add t @@ Q.of_ints (-1) 2
| _ -> t
in
let num = Q.num t' and den = Q.den t' in
Q.of_bigint (Z.div num den)
let ge = Q.geq
let ge = Q.geq
let le = Q.leq
let pow t q_factor =
if is_integer q_factor then
let factor = Q.to_int q_factor and num = Q.num t and den = Q.den t in
Q.make (Z.pow num factor) (Z.pow den factor)
else
let factor = Q.to_float q_factor
and num = Z.to_float @@ Q.num t
and den = Z.to_float @@ Q.den t in
Q.div (Q.of_float (num ** factor)) (Q.of_float (den ** factor))
let gcd t1 t2 = Q.of_bigint @@ Z.gcd (Q.to_bigint t1) (Q.to_bigint t2)
let lcm t1 t2 = Q.of_bigint @@ Z.lcm (Q.to_bigint t1) (Q.to_bigint t2)
end
module Bool = struct
type t = bool
include Comparable
let true_ = true
let false_ = false
let or_ = ( || )
let and_ = ( && )
let not = Stdlib.not
end
module String = struct
type t = UTF8.t
include Comparable
end
module Date = Date.Make (Num)
|