Source file syntax_table.ml
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
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
open! Core
open! Import
include Value.Make_subtype (struct
let name = "syntax-table"
let here = [%here]
let is_in_subtype = Value.is_syntax_table
end)
let equal = eq
let standard = Funcall.Wrap.("standard-syntax-table" <: nullary @-> return t) ()
let make_syntax_table = Funcall.Wrap.("make-syntax-table" <: nil_or t @-> return t)
let create ?parent () = make_syntax_table parent
let copy = Funcall.Wrap.("copy-syntax-table" <: t @-> return t)
module Class = struct
module T = struct
type t =
| Char_quote
| Close_paren
| Escape
| Expression_prefix
| Generic_string_delimiter
| Inherit_standard
| Open_paren
| Paired
| Punctuation
| String_quote
| Symbol_constitutent
| Whitespace
| Word_constituent
[@@deriving compare, enumerate, hash, sexp_of]
end
include T
include Hashable.Make_plain (T)
let equal = [%compare.equal: t]
let to_string t = [%sexp (t : t)] |> Sexp.to_string
let to_char = function
| Char_quote -> '/'
| Close_paren -> ')'
| Comment_end -> '>'
| Comment_start -> '<'
| Escape -> '\\'
| Expression_prefix -> '\''
| Generic_comment_delimiter -> '!'
| Generic_string_delimiter -> '|'
| Inherit_standard -> '@'
| Open_paren -> '('
| Paired -> '$'
| Punctuation -> '.'
| String_quote -> '"'
| Symbol_constitutent -> '_'
| Whitespace -> ' '
| Word_constituent -> 'w'
;;
let to_char_code t = t |> to_char |> Char_code.of_char_exn
let t_by_char_code =
lazy
(let index t = t |> to_char_code |> Char_code.to_int in
let max_index = List.fold all ~init:0 ~f:(fun ac t -> Int.max ac (index t)) in
let t_by_char_code = Option_array.create ~len:(max_index + 1) in
List.iter all ~f:(fun t -> Option_array.set_some t_by_char_code (index t) t);
t_by_char_code)
;;
let of_char_code_exn char_code =
match Option_array.get (force t_by_char_code) (char_code |> Char_code.to_int) with
| Some t -> t
| None ->
raise_s
[%message
"[Syntax_table.Class.of_char_code_exn] got unknown char code"
(char_code : Char_code.t)]
;;
end
module Flag = struct
type t =
| Commend_end_first_char
| Nested
| Prefix_char
[@@deriving enumerate, sexp_of]
let to_char = function
| Alternative_comment -> 'b'
| Commend_end_first_char -> '3'
| Comment_end_second_char -> '4'
| Comment_start_first_char -> '1'
| Comment_start_second_char -> '2'
| Nested -> 'n'
| Prefix_char -> 'p'
;;
end
module Descriptor = struct
type t = Class.t * Flag.t list [@@deriving sexp_of]
let to_value (class_, flags) =
let s = Bytes.create (1 + List.length flags) in
Bytes.set s 0 (class_ |> Class.to_char);
List.iteri flags ~f:(fun i flag -> Bytes.set s (i + 1) (flag |> Flag.to_char));
s |> Bytes.to_string |> Value.of_utf8_bytes
;;
end
let modify_syntax_entry =
Funcall.Wrap.("modify-syntax-entry" <: Char_code.t @-> value @-> t @-> return nil)
;;
let set t char_code class_ flags =
modify_syntax_entry char_code ((class_, flags) |> Descriptor.to_value) t
;;
let set_char t char class_ flags = set t (char |> Char_code.of_char_exn) class_ flags