diff --git a/CHANGES.md b/CHANGES.md index 98a9cd0ca845..fbddf4f9aafd 100644 --- a/CHANGES.md +++ b/CHANGES.md @@ -17,6 +17,8 @@ unreleased - Add experimental `$ dune init` command. This command is used to create or update project boilerplate. (#1448, fixes #159, @shonfeder) +- Experimental Coq support (1466, @ejgallego) + 1.8.2 (10/03/2019) ------------------ diff --git a/Makefile b/Makefile index 1bafa1802a84..78acd78de07f 100644 --- a/Makefile +++ b/Makefile @@ -28,8 +28,11 @@ test: test-js: $(BIN) build @runtest-js +test-coq: + $(BIN) build @runtest-coq + test-all: - $(BIN) build @runtest @runtest-js + $(BIN) build @runtest @runtest-js @runtest-coq check: $(BIN) build @check diff --git a/doc/coq.rst b/doc/coq.rst new file mode 100644 index 000000000000..92fa68050036 --- /dev/null +++ b/doc/coq.rst @@ -0,0 +1,63 @@ +.. _coq-main: + +****** +Coq +****** + +Dune is also able to build Coq developments. A Coq project is a mix of +Coq ``.v`` files and (optionally) OCaml libraries linking to the Coq +API (in which case we say the project is a *Coq plugin*). To enable +Coq support in a dune project, the language version should be selected +in the ``dune-project`` file. For example: + +.. code:: scheme + + (using coq 0.1) + +This will enable support for the ``coqlib`` stanza in the current project. If the +language version is absent, dune will automatically add this line with the +latest Coq version to the project file once a ``(coqlib ...)`` stanza is used anywhere. + + +Basic Usage +=========== + +The basic form for defining Coq libraries is very similar to the OCaml form: + +.. code:: scheme + + (coqlib + (name ) + (synopsis ) + (modules ) + (flags )) + +The stanza will build all `.v` files on the given directory. +The semantics of fields is: +- ``>`` will be used as the default Coq library prefix + ``-R`` +- the ``modules`` field does allow to constraint the set of modules + included in the library, similarly to its OCaml counterpart +- ```` will be passed to ``coqc``. + +Library Composition and Handling +=================== + +The ``coqlib`` stanza does not yet support composition of Coq +libraries. In the 0.1 version of the language, libraries are located +using Coq's built-in library management, thus Coq will always resort +to the installed version of a particular library. + +This will be fixed in the future. + +Recursive modules +=================== + +Adding: + +.. code:: scheme + (include_subdirs qualified) + +to the ``dune`` file will make Dune to consider all the modules in the +current directory and sub-directories, qualified in the current Coq +style. diff --git a/doc/index.rst b/doc/index.rst index bd04eeecfda3..8cb0e32880db 100644 --- a/doc/index.rst +++ b/doc/index.rst @@ -24,6 +24,7 @@ Welcome to dune's documentation! jsoo variants formatting + coq faq known-issues migration diff --git a/src/coq_module.ml b/src/coq_module.ml new file mode 100644 index 000000000000..2b221f2e6845 --- /dev/null +++ b/src/coq_module.ml @@ -0,0 +1,60 @@ +(* This file is licensed under The MIT License *) +(* (c) MINES ParisTech 2018-2019 *) +(* Written by: Emilio Jesús Gallego Arias *) + +open! Stdune + +module Name = struct + + type t = string + + let make x = x + let compare = String.compare + let pp = Fmt.text + +end + +(* We keep prefix and name separated as the handling of + `From Foo Require Bar.` may benefit from it. *) +type t = + { source: Path.t + ; prefix : string list + ; name : string + } + +let make ~source ~prefix ~name = + { source + ; prefix + ; name + } + +let source x = x.source +let prefix x = x.prefix +let name x = x.name +let obj_file ~obj_dir ~ext x = + let vo_dir = List.fold_left x.prefix ~init:obj_dir ~f:Path.relative in + Path.relative vo_dir (x.name ^ ext) +let pp fmt x = + let open Format in + let pp_sep fmt () = pp_print_string fmt "." in + fprintf fmt "{ prefix = %a; name = %s; source = %a }" + (pp_print_list ~pp_sep pp_print_string) x.prefix x.name Path.pp x.source + +let parse ~dir ~loc s = + let clist = List.rev @@ String.split s ~on:'.' in + match clist with + | [] -> + Errors.fail loc "invalid coq module" + | name :: prefix -> + let prefix = List.rev prefix in + let source = List.fold_left prefix ~init:dir ~f:Path.relative in + let source = Path.relative source (name ^ ".v") in + make ~name ~source ~prefix + +module Value = struct + type nonrec t = t + type key = string + let key x = String.concat ~sep:"." (x.prefix @ [x.name]) +end + +module Eval = Ordered_set_lang.Make(String)(Value) diff --git a/src/coq_module.mli b/src/coq_module.mli new file mode 100644 index 000000000000..b8826130274e --- /dev/null +++ b/src/coq_module.mli @@ -0,0 +1,40 @@ +(* This file is licensed under The MIT License *) +(* (c) MINES ParisTech 2018-2019 *) +(* Written by: Emilio Jesús Gallego Arias *) + +open! Stdune + +module Name : sig + + type t + + val make : string -> t + val compare : t -> t -> Ordering.t + + val pp : t Fmt.t + +end + +type t + +(** A Coq module [a.b.foo] defined in file [a/b/foo.v] *) +val make + : source:Path.t + (** file = .v source file; module name has to be the same so far *) + -> prefix:string list + (** Library-local qualified prefix *) + -> name:Name.t + (** Name of the module *) + -> t + +(** Coq does enforce some invariants wrt module vs file names *) + +val source : t -> Path.t +val prefix : t -> string list +val name : t -> string +val obj_file : obj_dir:Path.t -> ext:string -> t -> Path.t +val pp : t Fmt.t + +(** Parses a form "a.b.c" to a module *) +val parse : dir:Path.t -> loc:Loc.t -> string -> t +module Eval : Ordered_set_lang.S with type value := t diff --git a/src/coq_rules.ml b/src/coq_rules.ml new file mode 100644 index 000000000000..4c3153e3b8c4 --- /dev/null +++ b/src/coq_rules.ml @@ -0,0 +1,113 @@ +(* This file is licensed under The MIT License *) +(* (c) MINES ParisTech 2018-2019 *) +(* Written by: Emilio Jesús Gallego Arias *) + +open! Stdune +open Build.O +module SC = Super_context + +let coq_debug = false + +type coq_context = + { coqdep : Action.program + ; coqc : Action.program + ; coqpp : Action.program + } + +let parse_coqdep ~coq_module (lines : string list) = + if coq_debug then Format.eprintf "Parsing coqdep @\n%!"; + let source = Coq_module.source coq_module in + let invalid p = + Errors.die "coqdep returned invalid output for %s / [phase: %s]" + (Path.to_string source) p in + let line = + match lines with + | [] | _ :: _ :: _ :: _ -> invalid "line" + | [line] -> line + | [l1;_l2] -> + (* .vo is produced before .vio, this is fragile tho *) + l1 + in + match String.lsplit2 line ~on:':' with + | None -> invalid "split" + | Some (basename,deps) -> + let ff = List.hd @@ String.extract_blank_separated_words basename in + let depname, _ = Filename.split_extension ff in + let modname = + Coq_module.(String.concat ~sep:"/" (prefix coq_module @ [name coq_module])) in + if coq_debug + then Format.eprintf "depname / modname: %s / %s@\n%!" depname modname; + if depname <> modname then invalid "basename"; + let deps = String.extract_blank_separated_words deps in + if coq_debug + then Format.eprintf "deps for %a: %a@\n%!" Path.pp source Fmt.(list text) deps; + deps + +let setup_rule ~expander ~dir ~cc ~source_rule ~name ~cflags coq_module = + + if coq_debug + then Format.eprintf "gen_rule coq_module: %a@\n%!" Coq_module.pp coq_module; + let obj_dir = dir in + let source = Coq_module.source coq_module in + let stdout_to = Coq_module.obj_file ~obj_dir ~ext:".v.d" coq_module in + let object_to = Coq_module.obj_file ~obj_dir ~ext:".vo" coq_module in + + let iflags = Arg_spec.As ["-R"; "."; name] in + let cd_arg = Arg_spec.[ iflags; Dep source ] in + + (* coqdep needs the full source to be present :( *) + let coqdep_rule = + source_rule >>> + Build.run ~dir ~stdout_to cc.coqdep cd_arg + in + + (* Process coqdep and generate rules *) + let deps_of = Build.dyn_paths ( + Build.lines_of stdout_to >>^ + parse_coqdep ~coq_module >>^ + List.map ~f:(Path.relative dir) + ) in + let cc_arg = Arg_spec.[ + iflags; + Dep source; + Hidden_targets [object_to] ] + in + [coqdep_rule; + deps_of >>> + Expander.expand_and_eval_set expander cflags ~standard:(Build.return []) >>> + Build.run ~dir cc.coqc (Dyn (fun flags -> As flags) :: cc_arg) + ] + +(* TODO: remove; rgrinberg points out: + - resolve program is actually cached, + - better just to ask for values that we actually use. + *) +let create_ccoq sctx ~dir = + let rr prg = + SC.resolve_program ~dir sctx prg ~loc:None ~hint:"try: opam install coq" in + { coqdep = rr "coqdep" + ; coqc = rr "coqc" + ; coqpp = rr "coqpp" + } + +let setup_rules ~sctx ~dir ~dir_contents (s : Dune_file.Coq.t) = + + if coq_debug then begin + let scope = SC.find_scope_by_dir sctx dir in + Format.eprintf "[gen_rules] @[dir: %a@\nscope: %a@]@\n%!" + Path.pp dir Path.pp (Scope.root scope) + end; + + let cc = create_ccoq sctx ~dir in + let name = snd s.name in + let coq_modules = Dir_contents.coq_modules_of_library dir_contents ~name in + + (* coqdep requires all the files to be in the tree to produce correct + dependencies *) + let source_rule = Build.paths (List.map ~f:Coq_module.source coq_modules) in + let cflags = s.Dune_file.Coq.flags in + let expander = SC.expander sctx ~dir in + let coq_rules = + List.concat_map + ~f:(setup_rule ~expander ~dir ~cc ~source_rule ~name ~cflags) coq_modules in + coq_rules diff --git a/src/coq_rules.mli b/src/coq_rules.mli new file mode 100644 index 000000000000..bcc57d3c8113 --- /dev/null +++ b/src/coq_rules.mli @@ -0,0 +1,14 @@ +(* This file is licensed under The MIT License *) +(* (c) MINES ParisTech 2018-2019 *) +(* Written by: Emilio Jesús Gallego Arias *) + +(* Build rules for Coq's .v -> .vo files *) + +open! Stdune + +val setup_rules + : sctx:Super_context.t + -> dir:Path.t + -> dir_contents:Dir_contents.t + -> Dune_file.Coq.t + -> (unit, Action.t) Build.t list diff --git a/src/dir_contents.ml b/src/dir_contents.ml index f496d872e0f7..e340fd8e740f 100644 --- a/src/dir_contents.ml +++ b/src/dir_contents.ml @@ -180,6 +180,7 @@ type t = ; modules : Modules.t Lazy.t ; c_sources : C_sources.t Lazy.t ; mlds : (Dune_file.Documentation.t * Path.t list) list Lazy.t + ; coq_modules : Coq_module.t list String.Map.t Lazy.t } and kind = @@ -239,6 +240,16 @@ let mlds t (doc : Documentation.t) = (List.map map ~f:(fun (d, _) -> d.Documentation.loc)) ] +let coq_modules_of_library t ~name = + let map = Lazy.force t.coq_modules in + match String.Map.find map name with + | Some x -> x + | None -> + Exn.code_error "Dir_contents.coq_modules_of_library" + [ "name", Sexp.Encoder.string name + ; "available", Sexp.Encoder.(list string) (String.Map.keys map) + ] + (* As a side-effect, setup user rules and copy_files rules. *) let load_text_files sctx ft_dir { Dir_with_dune. @@ -255,6 +266,10 @@ let load_text_files sctx ft_dir let generated_files = List.concat_map stanzas ~f:(fun stanza -> match (stanza : Stanza.t) with + | Coq.T _coq -> + (* Format.eprintf "[coq] generated_files called at sctx: %a@\n%!" Path.pp (File_tree.Dir.path ft_dir); *) + (* FIXME: Need to generate ml files from mlg ? *) + [] | Menhir.T menhir -> Menhir_rules.targets menhir | Rule rule -> @@ -339,6 +354,29 @@ let build_mlds_map (d : _ Dir_with_dune.t) ~files = Some (doc, List.map (String.Map.values mlds) ~f:(Path.relative dir)) | _ -> None) +let coq_modules_of_files ~subdirs = + let filter_v_files (dir, local, files) = + (dir, local, String.Set.filter files ~f:(fun f -> Filename.check_suffix f ".v")) in + let subdirs = List.map subdirs ~f:filter_v_files in + let build_mod_dir (dir, prefix, files) = + String.Set.to_list files |> List.map ~f:(fun file -> + let name, _ = Filename.split_extension file in + let name = Coq_module.Name.make name in + Coq_module.make ~source:(Path.relative dir file) ~prefix ~name) in + let modules = List.concat_map ~f:build_mod_dir subdirs in + modules + +(* TODO: Build reverse map and check duplicates, however, are duplicates harmful? + * In Coq all libs are "wrapped" so including a module twice is not so bad. + *) +let build_coq_modules_map (d : _ Dir_with_dune.t) ~dir ~modules = + List.fold_left d.data ~init:String.Map.empty ~f:(fun map -> function + | Coq.T coq -> + let modules = Coq_module.Eval.eval coq.modules + ~parse:(Coq_module.parse ~dir) ~standard:modules in + String.Map.add map (snd coq.name) modules + | _ -> map) + let cache = Hashtbl.create 32 let clear_cache () = @@ -372,6 +410,9 @@ let rec get sctx ~dir = C_sources.make d ~c_sources:(C_sources.load_sources ~dune_version ~dir:d.ctx_dir ~files)) + ; coq_modules = + lazy (build_coq_modules_map d ~dir:d.ctx_dir + ~modules:(coq_modules_of_files ~subdirs:[dir,[],files])) } | Some (_, None) | None -> @@ -381,6 +422,7 @@ let rec get sctx ~dir = ; modules = lazy Modules.empty ; mlds = lazy [] ; c_sources = lazy C_sources.empty + ; coq_modules = lazy String.Map.empty } in Hashtbl.add cache dir t; @@ -466,6 +508,10 @@ let rec get sctx ~dir = in C_sources.make d ~c_sources ) in + let coq_modules = lazy ( + build_coq_modules_map d ~dir:d.ctx_dir + ~modules:(coq_modules_of_files ~subdirs:((dir,[],files)::subdirs)) + ) in let t = { kind = Group_root (lazy (List.map subdirs ~f:(fun (dir, _, _) -> get sctx ~dir))) @@ -474,10 +520,11 @@ let rec get sctx ~dir = ; modules ; c_sources ; mlds = lazy (build_mlds_map d ~files) + ; coq_modules } in Hashtbl.add cache dir t; - List.iter subdirs ~f:(fun (dir, _, files) -> + List.iter subdirs ~f:(fun (dir, _local, files) -> Hashtbl.add cache dir { kind = Group_part t ; dir @@ -485,5 +532,6 @@ let rec get sctx ~dir = ; modules ; c_sources ; mlds = lazy (build_mlds_map d ~files) + ; coq_modules }); t diff --git a/src/dir_contents.mli b/src/dir_contents.mli index 7c466c7abea2..cabb1782f220 100644 --- a/src/dir_contents.mli +++ b/src/dir_contents.mli @@ -34,6 +34,9 @@ val lookup_module : t -> Module.Name.t -> Dune_file.Buildable.t option (** All mld files attached to this documentation stanza *) val mlds : t -> Dune_file.Documentation.t -> Path.t list +(** Coq modules of library [name] is the Coq library name. *) +val coq_modules_of_library : t -> name:string -> Coq_module.t list + val get : Super_context.t -> dir:Path.t -> t type kind = private diff --git a/src/dune_file.ml b/src/dune_file.ml index 2ce002011342..52bc8ff6fc4b 100644 --- a/src/dune_file.ml +++ b/src/dune_file.ml @@ -1819,6 +1819,60 @@ module Menhir = struct }) end +module Coq = struct + + type t = + (* ; public : Public_lib.t option *\) *) + { name : Loc.t * string + (* TODO: validate name *) + ; synopsis : string option + ; modules : Ordered_set_lang.t + ; flags : Ordered_set_lang.Unexpanded.t + ; libraries : Lib_dep.t list + (** ocaml libraries *) + ; loc : Loc.t + ; enabled_if : Blang.t + } + + let syntax = + Syntax.create + ~name:"coq" + ~desc:"the coq extension (experimental)" + [ 0, 1 ] + + let decode = + record + (* let_map name = field_o "name" Lib_name.Local.decode_loc + * and public = Public_lib.public_name_field *) + (let+ name = field "name" (located string) + and+ loc = loc + and+ synopsis = field_o "synopsis" string + and+ flags = field_oslu "flags" + and+ modules = modules_field "modules" + and+ libraries = field "libraries" Lib_deps.decode ~default:[] + and+ enabled_if = enabled_if + in + (* { name + * ; public *) + { name + ; synopsis + ; modules + ; flags + ; libraries + ; loc + ; enabled_if + }) + + type Stanza.t += T of t + + let () = + Dune_project.Extension.register_simple + syntax + (return [ "coqlib", decode >>| fun x -> [T x] ]) + +end + + module Alias_conf = struct type t = { name : string diff --git a/src/dune_file.mli b/src/dune_file.mli index 7a73b2cf1295..e65b86b9a585 100644 --- a/src/dune_file.mli +++ b/src/dune_file.mli @@ -356,6 +356,23 @@ module Menhir : sig type Stanza.t += T of t end +module Coq : sig + + type t = + (* ; public : Public_lib.t option *\) *) + { name : Loc.t * string + ; synopsis : string option + ; modules : Ordered_set_lang.t + ; flags : Ordered_set_lang.Unexpanded.t + ; libraries : Lib_dep.t list + (** ocaml libraries *) + ; loc : Loc.t + ; enabled_if : Blang.t + } + + type Stanza.t += T of t +end + module Alias_conf : sig type t = { name : string diff --git a/src/gen_rules.ml b/src/gen_rules.ml index 3e08adedfd7e..af9bf6f4a80f 100644 --- a/src/gen_rules.ml +++ b/src/gen_rules.ml @@ -185,6 +185,11 @@ module Gen(P : sig val sctx : Super_context.t end) = struct | Some cctx -> Menhir_rules.gen_rules cctx m ~dir:ctx_dir end + | Coq.T m when Expander.eval_blang expander m.enabled_if -> + (* Format.eprintf "[coq] gen_rules called @\n%!"; *) + let dir = ctx_dir in + let coq_rules = Coq_rules.setup_rules ~sctx ~dir ~dir_contents m in + SC.add_rules ~dir:ctx_dir sctx coq_rules | _ -> ()); let dyn_deps = let pred = diff --git a/test/blackbox-tests/dune.inc b/test/blackbox-tests/dune.inc index e5d00998c2f6..f7b71772accc 100644 --- a/test/blackbox-tests/dune.inc +++ b/test/blackbox-tests/dune.inc @@ -95,6 +95,14 @@ test-cases/copy_files (progn (run %{exe:cram.exe} -test run.t) (diff? run.t run.t.corrected))))) +(alias + (name coq) + (deps (package dune) (source_tree test-cases/coq)) + (action + (chdir + test-cases/coq + (progn (run %{exe:cram.exe} -test run.t) (diff? run.t run.t.corrected))))) + (alias (name cross-compilation) (deps (package dune) (source_tree test-cases/cross-compilation)) @@ -1665,4 +1673,6 @@ (alias (name runtest-disabled) (deps (alias envs-and-contexts))) -(alias (name runtest-js) (deps (alias js_of_ocaml))) \ No newline at end of file +(alias (name runtest-js) (deps (alias js_of_ocaml))) + +(alias (name runtest-coq) (deps (alias coq))) \ No newline at end of file diff --git a/test/blackbox-tests/gen_tests.ml b/test/blackbox-tests/gen_tests.ml index d8aca4fab120..d753339260b4 100644 --- a/test/blackbox-tests/gen_tests.ml +++ b/test/blackbox-tests/gen_tests.ml @@ -61,10 +61,11 @@ module Test = struct ; skip_platforms : Platform.t list ; enabled : bool ; js : bool + ; coq : bool ; external_deps : bool } - let make ?env ?skip_ocaml ?(skip_platforms=[]) ?(enabled=true) ?(js=false) + let make ?env ?skip_ocaml ?(skip_platforms=[]) ?(enabled=true) ?(js=false) ?(coq=false) ?(external_deps=false) name = { name ; env @@ -73,6 +74,7 @@ module Test = struct ; external_deps ; enabled ; js + ; coq } let pp_sexp fmt t = @@ -123,6 +125,7 @@ let exclusions = let odoc = make ~external_deps:true ~skip_ocaml:"4.02.3" in [ make "js_of_ocaml" ~external_deps:true ~js:true ~env:("NODE", Sexp.parse "%{bin:node}") + ; make "coq" ~external_deps:true ~coq:true ; make "github25" ~env:("OCAMLPATH", Dune_lang.atom "./findlib-packages") ; odoc "odoc" ; odoc "odoc-unique-mlds" @@ -179,13 +182,14 @@ let pp_group fmt (name, tests) = let () = let tests = Lazy.force all_tests in - (* The runtest target has a "specoial" definition. It includes all tests - except for js and disabled tests *) + (* The runtest target has a "special" definition. It includes all + tests except for js, coq, and disabled tests *) tests |> List.iter ~f:(fun t -> Format.printf "%a@.@." Test.pp_sexp t); - [ "runtest", (fun (t : Test.t) -> not t.js && t.enabled) + [ "runtest", (fun (t : Test.t) -> not t.js && not t.coq && t.enabled) ; "runtest-no-deps", (fun (t : Test.t) -> not t.external_deps && t.enabled) ; "runtest-disabled", (fun (t : Test.t) -> not t.enabled) - ; "runtest-js", (fun (t : Test.t) -> t.js && t.enabled) ] + ; "runtest-js", (fun (t : Test.t) -> t.js && t.enabled) + ; "runtest-coq", (fun (t : Test.t) -> t.coq && t.enabled) ] |> List.map ~f:(fun (name, predicate) -> (name, List.filter tests ~f:predicate)) |> Format.pp_print_list diff --git a/test/blackbox-tests/test-cases/coq/base/bar.v b/test/blackbox-tests/test-cases/coq/base/bar.v new file mode 100644 index 000000000000..4627b76131c1 --- /dev/null +++ b/test/blackbox-tests/test-cases/coq/base/bar.v @@ -0,0 +1,3 @@ +From basic Require Import foo. + +Definition mynum (i : mynat) := 3. diff --git a/test/blackbox-tests/test-cases/coq/base/dune b/test/blackbox-tests/test-cases/coq/base/dune new file mode 100644 index 000000000000..3a510f40bbc2 --- /dev/null +++ b/test/blackbox-tests/test-cases/coq/base/dune @@ -0,0 +1,5 @@ +(coqlib + (name basic) + ; (public_name dune.test.basic) + (modules :standard) + (synopsis "Test Coq library")) diff --git a/test/blackbox-tests/test-cases/coq/base/dune-project b/test/blackbox-tests/test-cases/coq/base/dune-project new file mode 100644 index 000000000000..412ae241a3bf --- /dev/null +++ b/test/blackbox-tests/test-cases/coq/base/dune-project @@ -0,0 +1,3 @@ +(lang dune 1.8) + +(using coq 0.1) diff --git a/test/blackbox-tests/test-cases/coq/base/foo.v b/test/blackbox-tests/test-cases/coq/base/foo.v new file mode 100644 index 000000000000..53e0ce1b1526 --- /dev/null +++ b/test/blackbox-tests/test-cases/coq/base/foo.v @@ -0,0 +1 @@ +Definition mynat := nat. diff --git a/test/blackbox-tests/test-cases/coq/rec_module/a/bar.v b/test/blackbox-tests/test-cases/coq/rec_module/a/bar.v new file mode 100644 index 000000000000..b1e4f474ced9 --- /dev/null +++ b/test/blackbox-tests/test-cases/coq/rec_module/a/bar.v @@ -0,0 +1,5 @@ +From rec_module Require Import b.foo. +From rec_module Require Import c.ooo. +From rec_module Require c.d.bar. + +Definition mynum (i : mynat) := 3 + ooo_nat + c.d.bar.bar_nat. diff --git a/test/blackbox-tests/test-cases/coq/rec_module/b/foo.v b/test/blackbox-tests/test-cases/coq/rec_module/b/foo.v new file mode 100644 index 000000000000..53e0ce1b1526 --- /dev/null +++ b/test/blackbox-tests/test-cases/coq/rec_module/b/foo.v @@ -0,0 +1 @@ +Definition mynat := nat. diff --git a/test/blackbox-tests/test-cases/coq/rec_module/c/d/bar.v b/test/blackbox-tests/test-cases/coq/rec_module/c/d/bar.v new file mode 100644 index 000000000000..b68bd7244fba --- /dev/null +++ b/test/blackbox-tests/test-cases/coq/rec_module/c/d/bar.v @@ -0,0 +1 @@ +Definition bar_nat : nat := 4. diff --git a/test/blackbox-tests/test-cases/coq/rec_module/c/ooo.v b/test/blackbox-tests/test-cases/coq/rec_module/c/ooo.v new file mode 100644 index 000000000000..07d40e567986 --- /dev/null +++ b/test/blackbox-tests/test-cases/coq/rec_module/c/ooo.v @@ -0,0 +1 @@ +Definition ooo_nat : nat := 10. diff --git a/test/blackbox-tests/test-cases/coq/rec_module/dune b/test/blackbox-tests/test-cases/coq/rec_module/dune new file mode 100644 index 000000000000..71b85ea57f11 --- /dev/null +++ b/test/blackbox-tests/test-cases/coq/rec_module/dune @@ -0,0 +1,7 @@ +(coqlib + (name rec_module) + ; (public_name dune.test.basic) + (modules :standard) + (synopsis "Test Coq library")) + +(include_subdirs qualified) diff --git a/test/blackbox-tests/test-cases/coq/rec_module/dune-project b/test/blackbox-tests/test-cases/coq/rec_module/dune-project new file mode 100644 index 000000000000..412ae241a3bf --- /dev/null +++ b/test/blackbox-tests/test-cases/coq/rec_module/dune-project @@ -0,0 +1,3 @@ +(lang dune 1.8) + +(using coq 0.1) diff --git a/test/blackbox-tests/test-cases/coq/run.t b/test/blackbox-tests/test-cases/coq/run.t new file mode 100644 index 000000000000..866ec5ac4057 --- /dev/null +++ b/test/blackbox-tests/test-cases/coq/run.t @@ -0,0 +1,9 @@ + $ dune build --root base --display short --debug-dependency-path @all + Entering directory 'base' + coqdep bar.v.d + coqdep foo.v.d + coqc foo.vo + coqc bar.vo + + $ dune build --root rec_module --display short --debug-dependency-path @all + Entering directory 'rec_module'