package codex

  1. Overview
  2. Docs
The Codex library for building static analysers based on abstract interpretation

Install

dune-project
 Dependency

Authors

Maintainers

Sources

1.0-rc4.tar.gz
md5=bc7266a140c6886add673ede90e335d3
sha512=8da42c0ff2c1098c5f9cb2b5b43b306faf7ac93b8f5ae00c176918cee761f249ff45b29309f31a05bbcf6312304f86a0d5a000eb3f1094d3d3c2b9b4c7f5c386

doc/src/codex.fixpoint/regex.ml.html

Source file regex.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
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
(**************************************************************************)
(*  This file is part of the Codex semantics library.                     *)
(*                                                                        *)
(*  Copyright (C) 2013-2025                                               *)
(*    CEA (Commissariat à l'énergie atomique et aux énergies              *)
(*         alternatives)                                                  *)
(*                                                                        *)
(*  you can redistribute it and/or modify it under the terms of the GNU   *)
(*  Lesser General Public License as published by the Free Software       *)
(*  Foundation, version 2.1.                                              *)
(*                                                                        *)
(*  It 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 Lesser General Public License for more details.                   *)
(*                                                                        *)
(*  See the GNU Lesser General Public License version 2.1                 *)
(*  for more details (enclosed in the file LICENSE).                      *)
(*                                                                        *)
(**************************************************************************)

module type LETTER = sig
  type t
  val equal : t -> t -> bool
  val pretty : Format.formatter -> t -> unit
  val hash : t -> int
end

module type S = sig
  type letter

  type t

  val empty : t
  val epsilon : t

  val join : t -> t -> t
  val append : t -> letter -> t
  val append_star : t -> t -> t
  val equal : t -> t -> bool
  val hash : t -> int
  val pretty : Format.formatter -> t -> unit
  val pretty_reverse : Format.formatter -> t -> unit
end

(** Tagged regular expressions.  The tag is used for cons-hashing and share as
    many data structures as possibles. Tags should not be used outside of the
    implementation. *)
type 'l tagged_regex =
  | Empty
  | Epsilon
  | Join of int * 'l tagged_regex * 'l tagged_regex
  | Append of int * 'l tagged_regex * 'l
  | AppendStar of int * 'l tagged_regex * 'l tagged_regex

module Make (L : LETTER) : S
    with type letter = L.t
    and type t = L.t tagged_regex
= struct
  type letter = L.t

  type t = L.t tagged_regex

  let empty = Empty
  let epsilon = Epsilon

  module HashedRegex = struct
    type t = L.t tagged_regex

    let tag = function
    | Empty -> 0
    | Epsilon -> 1
    | Join (x,_,_) -> x
    | Append (x,_,_) -> x
    | AppendStar (x,_,_) -> x

    let shallow_equal x y =
      tag x = tag y

    let equal x y = match x, y with
    | Empty,Empty -> true
    | Epsilon,Epsilon -> true
    | Join (_,x1,y1), Join (_,x2,y2) ->
        shallow_equal x1 x2 && shallow_equal y1 y2
    | Append (_,x1, l1), Append (_,x2, l2) ->
        shallow_equal x1 x2 && L.equal l1 l2
    | AppendStar (_,x1,y1), AppendStar (_,x2,y2) ->
        shallow_equal x1 x2 && shallow_equal y1 y2
    | _ -> false

    let hash x =
      let empty_h = Hashtbl.hash 0 in
      let epsilon_h = Hashtbl.hash 1 in
      match x with
      | Empty -> empty_h
      | Epsilon -> epsilon_h
      | Join (_,x,y) ->
          Hashtbl.hash (0, tag x, tag y)
      | Append (_,r,l) ->
          Hashtbl.hash (1, tag r, L.hash l)
      | AppendStar (_,prefix,body) ->
          Hashtbl.hash (2, tag prefix, tag body)
  end

  module RegexHashSet = Weak.Make(HashedRegex)

  let equal = (==)

  let hash = HashedRegex.tag

  let hash_set_init_size = 2000
  let hash_set =
    RegexHashSet.create hash_set_init_size

  let tag_counter = ref 2

  let maybe_share new_regex =
    let ret = RegexHashSet.merge hash_set new_regex in
    if ret == new_regex then begin
      incr tag_counter
    end;
    ret

  let join x y =
    if x == y then x
    else match x,y with
    | Empty,_ -> y
    | _,Empty -> x
    | _ -> maybe_share (Join (!tag_counter,x,y))

  let append r l =
    match r with
    | Empty -> empty
    | _ ->
        maybe_share (Append (!tag_counter,r,l))

  let append_star r x =
    match r with
    | Empty -> empty
    | _ -> maybe_share (AppendStar (!tag_counter,r,x))

  let pretty fmt r =
    let rec aux precedence fmt r =
      let open Format in
      let pp format = fprintf fmt format in
      match r with
      | Empty -> pp "<empty>"
      | Epsilon -> pp "<epsilon>"
      | Join (_,x,y) ->
          (if precedence > 0
          then fprintf fmt "(%a@ +@ %a)"
          else fprintf fmt "%a@ +@ %a") (aux 0) x (aux 0) y
      | Append (_,r,l) ->
          (if precedence > 1
          then fprintf fmt "@[<v>(%a@,%a)@]"
          else fprintf fmt "@[<v>%a@,%a@]") (pp_if_not_epsilon 1) r L.pretty l
      | AppendStar (_,r,x) ->
          (if precedence > 1
          then fprintf fmt "@[<v>(%a@,%a*)@]"
          else fprintf fmt "@[<v>%a@,%a*@]") (pp_if_not_epsilon 1) r (aux 2) x
    and pp_if_not_epsilon precedence fmt r = match r with
      | Epsilon -> ()
      | _ -> aux precedence fmt r
    in
    aux 0 fmt r

  let pretty_reverse fmt r =
    let rec aux precedence fmt r =
      let open Format in
      let pp format = fprintf fmt format in
      match r with
      | Empty -> pp "<empty>"
      | Epsilon -> pp "<epsilon>"
      | Join (_,x,y) ->
          (if precedence > 0
          then pp "(%a@ +@ %a)"
          else pp "%a@ +@ %a") (aux 0) x (aux 0) y
      | Append (_,r,l) ->
          (if precedence > 1
          then pp "@[<v>(%a@,%a)@]"
          else pp "@[<v>%a@,%a@]") L.pretty l (aux 1) r
      | AppendStar (_,r,x) ->
          (if precedence > 1
          then pp "@[<v>(%a*@,%a)@]"
          else pp "@[<v>%a*@,%a@]") (aux 2) x (aux 1) r
    in
    aux 0 fmt r
end