blob: 8e5f5007a2e96cd349b17ebc300b069251147e7e (
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
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
|
open StdLabels
module S = Qsp_syntax.S
module T = Qsp_syntax.T
module Report = Qsp_syntax.Report
module IgnoreCaseString = struct
type t = string
let compare t1 t2 =
String.compare (String.lowercase_ascii t1) (String.lowercase_ascii t2)
let equal t1 t2 =
String.equal (String.lowercase_ascii t1) (String.lowercase_ascii t2)
end
module LocationSet = Set.Make (IgnoreCaseString)
module LocationCalls = Map.Make (IgnoreCaseString)
let identifier = "locations"
let description = "Ensure every call points to an existing location"
let is_global = true
let active = ref true
type t = {
locations : LocationSet.t;
calls : (string * S.pos) list LocationCalls.t;
}
type context = t ref
let initialize () =
ref { locations = LocationSet.empty; calls = LocationCalls.empty }
let finalize : context -> (string * Report.t) list =
fun context ->
LocationCalls.fold
(fun location positions acc ->
let message = Printf.sprintf "The location %s does not exists" location in
List.fold_left ~init:acc (List.rev positions)
~f:(fun acc (loc, position) ->
let report = Report.error position message in
(loc, report) :: acc))
!context.calls []
(** Register a new call to a defined location. *)
let registerCall : S.pos -> string -> t -> t =
fun pos location t ->
let file_name = (fst pos).Lexing.pos_fname in
match
IgnoreCaseString.equal location file_name
|| LocationSet.mem location t.locations
with
| true -> t
| false ->
(* The location is not yet defined, register the call for later *)
let calls =
LocationCalls.update location
(function
| None -> Some [ (file_name, pos) ]
| Some poss ->
Some
(let new_pos = (file_name, pos) in
new_pos :: poss))
t.calls
in
{ t with calls }
(** Add a new location in the list of all the collected elements *)
let registerLocation : string -> t -> t =
fun location t ->
let calls = LocationCalls.remove location t.calls
and locations = LocationSet.add location t.locations in
{ calls; locations }
(** The module Expression is pretty simple, we are only interrested by the
strings ( because only the first argument of [gt …] is read ).
If the string is too much complex, we just ignore it. *)
module Expression = struct
type t = string option
include Default.Expression (struct
type nonrec t = t
let default = None
end)
let v : t -> t' = Fun.id
(* Extract the litteral if this is a simple text *)
let literal : S.pos -> t' T.literal list -> t' =
fun _ ll -> match ll with Text lit :: [] -> Some lit | _ -> None
end
module Instruction = struct
type nonrec t = t -> t
type t' = t
let v : t -> t' = Fun.id
include
Default.Instruction
(Expression)
(struct
type nonrec t = t
let default = Fun.id
let fold : t Seq.t -> t =
fun sequence t -> Seq.fold_left (fun acc t -> t acc) t sequence
end)
(** Keep a track of every gt or gs instruction *)
let call : S.pos -> T.keywords -> Expression.t' list -> t =
fun pos fn args t ->
match (fn, args) with
| T.Goto, Some dest :: _ -> registerCall pos dest t
| T.Gosub, Some dest :: _ -> registerCall pos dest t
| _ -> t
end
module Location = struct
type t = unit
let v : t -> Report.t list = fun () -> []
let location : context -> S.pos -> Instruction.t list -> t =
fun context pos instructions ->
(* Register the location *)
let file_name = (fst pos).Lexing.pos_fname in
let c = registerLocation file_name !context in
(* Then update the list of all the calls to the differents locations *)
context :=
List.fold_left instructions ~init:c ~f:(fun t instruction ->
instruction t)
end
|