Extraction从 Coq 中提取 ML
基本的提取方式
现在我们将待提取的定义加载到 Coq 环境中。你可以直接写出定义,
也可以从其它模块中加载。
From Coq Require Import Arith.Arith.
From Coq Require Import Init.Nat.
From Coq Require Import Arith.EqNat.
From LF Require Import ImpCEvalFun.
最后,我们来指定需要提取的定义,以及用于保存提取结果的文件名。
Extraction "imp1.ml" ceval_step.
Coq 在处理此指令时会生成一个名为 imp1.ml 的文件,其中包含了提取后的
ceval_step 以及所有递归依赖的文件。编译本章对应的 .v
文件,然后看看生成的 imp1.ml 吧!
控制提取特定的类型
- 该 Coq 类型应当如何用 OCaml 来表示,以及
- 该类型的构造子应如何转换为目标语言中对应的标识符。
对于非枚举类型(即构造器接受参数的类型),我们需要给出一个 OCaml
表达式来作为该类型元素上的“递归器”。(想想丘奇数。)
(译注:在这一部分,读者可以在为 nat 指定对应的类型后使用
Extraction plus 来得到自然数加法的提取结果,以此加深对“递归器”的理解。)
Extract Inductive nat ⇒ "int"
[ "0" "(fun x -> x + 1)" ]
"(fun zero succ n -> if n=0 then zero () else succ (n-1))".
我们也可以将定义的常量提取为具体的 OCaml 项或者操作符。
Extract Constant plus ⇒ "( + )".
Extract Constant mult ⇒ "( * )".
Extract Constant eqb ⇒ "( = )".
注意:保证提取结果的合理性是'你的责任'。例如,以下指定可能十分自然:
Extract Constant minus ⇒ "( - )".
但是这样做会引起严重的混乱。(思考一下。为什么会这样呢?)
Extract Constant minus ⇒ "( - )".
Extraction "imp2.ml" ceval_step.
检查一下生成的 imp2.ml 文件,留意一下此次的提取结果与 imp1.ml
有何不同。
一个完整的示例
我们还需要翻译另一种布尔值:
提取指令是相同的。
From LF Require Import Imp.
From LF Require Import ImpParser.
From LF Require Import Maps.
Extraction "imp.ml" empty_st ceval_step parse.
现在我们来运行一下生成的 Imp 求值器。首先你应该阅览一下
impdriver.ml(这并非从某个 Coq 源码提取而来,它是手写的。)
然后用下面的指令将求值器与驱动程序一同编译成可执行文件:
ocamlc -w -20 -w -26 -o impdriver imp.mli imp.ml impdriver.ml ./impdriver(编译时所使用的 -w 开关只是为了避免输出一些误报的警告信息。)
讨论
(* 2022-03-14 05:26:59 (UTC+00) *)