forked from Soonad/Moonad
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathBits.fm
93 lines (82 loc) · 1.99 KB
/
Bits.fm
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
Bits.0: (bs: Bits) -> Bits
(bs) <P> (be, b0, b1) b0(bs)
Bits.1: (bs: Bits) -> Bits
(bs) <P> (be, b0, b1) b1(bs)
Bits.concat: Bits -> Bits -> Bits
(a, b)
a<() Bits>
| b;
| (a.pred) Bits.0(Bits.concat(a.pred)(b));
| (a.pred) Bits.1(Bits.concat(a.pred)(b));
Bits.eql: Bits -> Bits -> Bool
(bs1)
bs1<() Bits -> _>
| (bs2)
case bs2:
| Bool.true;
| () Bool.false;
| () Bool.false;
;
| (bs1.pred) (bs2)
case bs2:
| Bool.false;
| (bs2.pred) Bits.eql(bs1.pred)(bs2.pred);
| () Bool.false;
;
| (bs1.pred) (bs2)
case bs2:
| Bool.false;
| () Bool.false;
| (bs2.pred) Bits.eql(bs1.pred)(bs2.pred);
;
// A sequence of bits.
// T Bits
// | b0(pred: Bits)
// | b1(pred: Bits)
Bits: Type //prim//
bits<P: Bits -> Type> ->
(be: P(Bits.nil)) ->
(b0: (pred: Bits) -> P(Bits.0(pred))) ->
(b1: (pred: Bits) -> P(Bits.1(pred))) ->
P(bits)
Bits.from_string: String -> Bits
(str)
case str:
| Bits.nil;
| (str.head) (str.tail)
case U16.eql(str.head)(Char.parse("1")):
| () Bits.1(Bits.from_string(str.tail));
| () Bits.0(Bits.from_string(str.tail));
| Unit.new;;
Bits.inc: Bits -> Bits
(bits)
case bits:
| Bits.nil;
| (bits.pred) Bits.1(bits.pred);
| (bits.pred) Bits.0(Bits.inc(bits.pred));
Bits.nil: Bits
<P> (be) (b0) (b1) be
Bits.parse_hex: String -> Bits
(str) Bits.parse_hex.go(str)(Bits.nil)
Bits.parse_hex.go: String -> Bits -> Bits //loop//
(str) (res)
case str:
| res;
| (str.head) (str.tail)
Char.hex_value(str.head)<() Bits>
| res;
| (value)
let b0000 = Bits.0(Bits.0(Bits.0(Bits.0(Bits.nil))))
let bhead = Nat.apply<Bits>(value)(Bits.inc)(b0000)
Bits.parse_hex.go(str.tail)(Bits.concat(bhead)(res))
;
;
// Reverses a bistring.
Bits.reverse: Bits -> Bits
(as) Bits.reverse.go(as)(Bits.nil)
Bits.reverse.go: Bits -> Bits -> Bits
(as) (res)
case as:
| res;
| (as.pred) Bits.reverse.go(as.pred)(Bits.0(res));
| (as.pred) Bits.reverse.go(as.pred)(Bits.1(res));