ligo/src/passes/1-parser/shared/LexerUnit.ml

111 lines
3.4 KiB
OCaml
Raw Normal View History

2020-01-04 23:32:50 +01:00
(* Functor to build a standalone LIGO lexer *)
module Region = Simple_utils.Region
2020-01-20 10:57:07 +01:00
module type IO =
2020-01-04 23:32:50 +01:00
sig
val ext : string (* LIGO file extension *)
val options : EvalOpt.options (* CLI options *)
end
2020-01-20 10:57:07 +01:00
module Make (IO: IO) (Lexer: Lexer.S) =
2020-01-04 23:32:50 +01:00
struct
open Printf
2020-01-20 10:57:07 +01:00
module SSet = Utils.String.Set
2020-01-04 23:32:50 +01:00
(* Error printing and exception tracing *)
let () = Printexc.record_backtrace true
(* Preprocessing the input source and opening the input channels *)
(* Path for CPP inclusions (#include) *)
let lib_path =
match IO.options#libs with
[] -> ""
| libs -> let mk_I dir path = sprintf " -I %s%s" dir path
in List.fold_right mk_I libs ""
let prefix =
match IO.options#input with
None | Some "-" -> "temp"
2020-01-20 10:57:07 +01:00
| Some file -> Filename.(file |> basename |> remove_extension)
2020-01-04 23:32:50 +01:00
let suffix = ".pp" ^ IO.ext
let pp_input =
if Utils.String.Set.mem "cpp" IO.options#verbose
then prefix ^ suffix
else let pp_input, pp_out = Filename.open_temp_file prefix suffix
in close_out pp_out; pp_input
let cpp_cmd =
match IO.options#input with
None | Some "-" ->
2020-01-20 10:57:07 +01:00
sprintf "cpp -traditional-cpp%s - > %s"
lib_path pp_input
| Some file ->
sprintf "cpp -traditional-cpp%s %s > %s"
lib_path file pp_input
2020-01-04 23:32:50 +01:00
(* Running the lexer on the input file *)
let scan () : (Lexer.token list, string Region.reg) Stdlib.result =
2020-01-20 10:57:07 +01:00
(* Preprocessing the input *)
if SSet.mem "cpp" IO.options#verbose
then eprintf "%s\n%!" cpp_cmd
else ();
if Sys.command cpp_cmd <> 0 then
let msg =
sprintf "External error: the command \"%s\" failed." cpp_cmd
in Stdlib.Error (Region.wrap_ghost msg)
2020-01-20 10:57:07 +01:00
else
match Lexer.open_token_stream (Lexer.File pp_input) with
Ok Lexer.{read; buffer; close; _} ->
let close_all () = close (); close_out stdout in
let rec read_tokens tokens =
match read ~log:(fun _ _ -> ()) buffer with
token ->
if Lexer.Token.is_eof token
then Stdlib.Ok (List.rev tokens)
else read_tokens (token::tokens)
| exception Lexer.Error error ->
let file =
match IO.options#input with
None | Some "-" -> false
| Some _ -> true in
let msg =
Lexer.format_error ~offsets:IO.options#offsets
IO.options#mode ~file error
in Stdlib.Error msg in
let result = read_tokens []
in close_all (); result
| Stdlib.Error (Lexer.File_opening msg) ->
close_out stdout; Stdlib.Error (Region.wrap_ghost msg)
2020-01-20 10:57:07 +01:00
(* Tracing the lexing (effectful) *)
2020-01-04 23:32:50 +01:00
module Log = LexerLog.Make (Lexer)
let trace () : (unit, string Region.reg) Stdlib.result =
2020-01-20 10:57:07 +01:00
(* Preprocessing the input *)
if SSet.mem "cpp" IO.options#verbose
then eprintf "%s\n%!" cpp_cmd
else ();
if Sys.command cpp_cmd <> 0 then
let msg =
sprintf "External error: the command \"%s\" failed." cpp_cmd
in Stdlib.Error (Region.wrap_ghost msg)
2020-01-20 10:57:07 +01:00
else
Log.trace ~offsets:IO.options#offsets
IO.options#mode
(Some pp_input)
IO.options#cmd
2020-01-04 23:32:50 +01:00
end