File ‹CTR_Postprocessing.ML›

(* Title: CTR/CTR_Postprocessing.ML
   Author: Mihails Milehins
   Copyright 2021 (C) Mihails Milehins

Postprocessing of the output of the algorithms associated with the CTR.
*)


signature CTR_POSTPROCESSING =
sig

val postprocess_parametricity : 
  binding -> thm -> local_theory -> ctr_pp_out
val postprocess_relativization : 
  binding -> mixfix -> thm -> Proof.context -> ctr_pp_out
val postprocess_failure : Proof.context -> ctr_pp_out

end;

structure CTR_Postprocessing : CTR_POSTPROCESSING =
struct

(*post-processing of an arbitrary transfer rule*)
fun postprocess_transfer_rule b thm ctxt = 
  let
    val b = ((b |> Binding.path_of |> map fst) @ [Binding.name_of b, "transfer"])
      |> Long_Name.implode
      |> Binding.qualified_name_mandatory
    val lthy = ctxt
      |> Local_Theory.note 
        (
          (b, Transfer.transfer_add |> K |> Attrib.internal  |> single), 
          single thm
        )
      |>> #2
      |>> the_single
      |> #2
    val lthy_print = Bundle.includes ["Transfer.lifting_syntax"] lthy
    val _ =
      Proof_Display.print_theorem (Position.thread_data ()) lthy_print
        (Local_Theory.full_name lthy_print b, [thm])
  in lthy end;

(*post-processing of a parametricity property*)
fun postprocess_parametricity b thm ctxt = 
  PPParametricity (thm, postprocess_transfer_rule b thm ctxt);

(*post-processing of a relativization*)
fun postprocess_relativization b mf thm ctxt = 
  let
    val (_, rhst, _) = thm 
      |> Thm.concl_of
      |> HOLogic.dest_Trueprop 
      |> CTR_Conversions.dest_trt
    val (rhst, ctxt') = rhst
      |> Logic.unvarify_local_term (Local_Theory.begin_nested ctxt |> snd)  
    val (absts, rhst) = rhst
      |> Term.strip_abs_eta (rhst |> strip_abs_vars |> length) 
    val argts = rhst
      |> (fn t => Term.add_frees t [])
      |> rev
      |> subtract op= absts
      |> curry (swap #> op@) absts  
      |> map Free
    val (lhst, lthy) = 
      let 
        fun declare_const_with thy = 
          let val T = map type_of argts ---> type_of rhst 
          in Sign.declare_const_global ((b, T), mf) thy end
      in Local_Theory.raw_theory_result declare_const_with ctxt' end
    val lhst = Term.list_comb (lhst, argts)
    fun make_with_def thy = 
      let
        val b =
          ((b |> Binding.path_of |> map fst) @ [Binding.name_of b ^ "_def"])
          |> Long_Name.implode
          |> Binding.qualified_name_mandatory
      in Global_Theory.add_def (b, Logic.mk_equals (lhst, rhst)) thy end
    val (ow_def_thm, lthy') = Local_Theory.raw_theory_result make_with_def lthy
    val _ =
      Proof_Display.print_theorem (Position.thread_data ()) lthy'
        (Thm.derivation_name ow_def_thm, [ow_def_thm])
    val thm' = thm 
      |> singleton (Proof_Context.export ctxt lthy')
      |> Thm.pure_unfold lthy' (ow_def_thm |> Thm.symmetric |> single)
    val lthy'' = Local_Theory.end_nested lthy'
    val thm'' = singleton (Proof_Context.export lthy' lthy'') thm'
    val ow_def_thm'' = singleton (Proof_Context.export lthy' lthy'') ow_def_thm
  in 
    ((ow_def_thm'', thm''), postprocess_transfer_rule b thm'' lthy'')
    |> PPRelativization 
  end;

(*post-processing of a failed attempt to perform relativization*)
fun postprocess_failure lthy = PPFailure lthy;

end;