]> matita.cs.unibo.it Git - helm.git/blobdiff - helm/matita/matitaScript.ml
ocaml 3.09 transition
[helm.git] / helm / matita / matitaScript.ml
index 8bd0d60d1f2df270cdfcb582778fcd2b564aef40..34e0408c598f281aa0d6bee4417bc3ad5493e407 100644 (file)
@@ -26,6 +26,8 @@
 open Printf
 open MatitaTypes
 
+module TA = GrafiteAst
+
 let debug = false
 let debug_print = if debug then prerr_endline else ignore
 
@@ -58,7 +60,9 @@ let first_line s =
 let goal_ast n =
   let module A = GrafiteAst in
   let loc = DisambiguateTypes.dummy_floc in
-  A.Executable (loc, A.Tactical (loc, A.Tactic (loc, A.Goal (loc, n))))
+  A.Executable (loc, A.Tactical (loc,
+    A.Tactic (loc, A.Goal (loc, n)),
+    Some (A.Dot loc)))
 
 type guistuff = {
   mathviewer:MatitaTypes.mathViewer;
@@ -69,7 +73,6 @@ type guistuff = {
 }
 
 let eval_with_engine guistuff status user_goal parsed_text st =
-  let module TA = GrafiteAst in
   let module TAPp = GrafiteAstPp in
   let include_ = 
     match guistuff.filenamedata with
@@ -92,15 +95,17 @@ let eval_with_engine guistuff status user_goal parsed_text st =
   (* we add the goal command if needed *)
   let inital_space,new_status,new_status_and_text_list' =
     match status.proof_status with
-      | Incomplete_proof (_, goal) when goal <> user_goal ->
-         let status =
+(*     | Incomplete_proof { stack = stack }
+      when not (List.mem user_goal (Continuationals.head_goals stack)) ->
+        let status =
           MatitaEngine.eval_ast ~include_paths:include_
-            ~do_heavy_checks:true status (goal_ast user_goal) in
-         let initial_space =
-          if initial_space = "" then "\n" else initial_space
-         in
-          "\n", status,
-           [status, initial_space ^ TAPp.pp_tactic (TA.Goal (loc, user_goal))]
+            ~do_heavy_checks:true status (goal_ast user_goal)
+        in
+        let initial_space = if initial_space = "" then "\n" else initial_space
+        in
+        "\n", status,
+        [ status,
+          initial_space ^ TAPp.pp_tactical (TA.Select (loc, [user_goal])) ] *)
       | _ -> initial_space,status,[] in
   let new_status = 
     MatitaEngine.eval_ast 
@@ -110,16 +115,19 @@ let eval_with_engine guistuff status user_goal parsed_text st =
     match ex with
       | TA.Command (_, TA.Alias _)
       | TA.Command (_, TA.Include _)
-      | TA.Command (_, TA.Interpretation _) ->
-          DisambiguateTypes.Environment.empty
+      | TA.Command (_, TA.Interpretation _) -> []
       | _ -> MatitaSync.alias_diff ~from:status new_status
   in
   (* we remove the defined object since we consider them "automatic aliases" *)
+  let dummy_st =
+    TA.Comment (DisambiguateTypes.dummy_floc,
+      TA.Note (DisambiguateTypes.dummy_floc, ""))
+  in
   let initial_space,status,new_status_and_text_list_rev = 
     let module DTE = DisambiguateTypes.Environment in
     let module UM = UriManager in
-    DTE.fold_flatten (
-      fun k ((v,_) as value) (initial_space,status,acc) -> 
+    List.fold_left (
+      fun (initial_space,status,acc) (k,((v,_) as value)) -> 
         let b = 
           try
             let v = UM.strip_xpointer (UM.uri_of_string v) in
@@ -133,16 +141,18 @@ let eval_with_engine guistuff status user_goal parsed_text st =
           let initial_space =
            if initial_space = "" then "\n" else initial_space in
             initial_space ^
-             DisambiguatePp.pp_environment(DTE.cons k value DTE.empty) in
+             DisambiguatePp.pp_environment
+              (DisambiguateTypes.Environment.add k value
+                DisambiguateTypes.Environment.empty) in
          let new_status =
-          {status with aliases = DTE.cons k value status.aliases}
+          MatitaSync.set_proof_aliases status [k,value]
          in
-          "\n",new_status,((new_status, new_text)::acc)
-    ) new_aliases (initial_space,status,[]) in
+          "\n",new_status,((new_status, (new_text, dummy_st))::acc)
+    ) (initial_space,status,[]) new_aliases in
   let parsed_text = initial_space ^ parsed_text in
   let res =
    List.rev new_status_and_text_list_rev @ new_status_and_text_list' @
-    [new_status, parsed_text]
+    [new_status, (parsed_text, st)]
   in
    res,parsed_text_length
 
@@ -179,7 +189,7 @@ let eval_with_engine guistuff status user_goal parsed_text st =
         let title = "Unable to include " ^ what in
         let message = 
          what ^ " is <b>not</b> handled by a development.\n" ^
-         "All dependencies are authomatically solved for a development.\n\n" ^
+         "All dependencies are automatically solved for a development.\n\n" ^
          "<i>Do you want to set up a development?</i>"
         in
         (match guistuff.ask_confirmation ~title ~message with
@@ -201,19 +211,19 @@ let eval_with_engine guistuff status user_goal parsed_text st =
           | Some d -> handle_with_devel d
 ;;
 
-let disambiguate term status =
+let disambiguate_macro_term term status user_goal =
   let module MD = MatitaDisambiguator in
   let dbd = MatitaDb.instance () in
-  let metasenv = MatitaMisc.get_proof_metasenv status in
-  let context = MatitaMisc.get_proof_context status in
-  let aliases = MatitaMisc.get_proof_aliases status in
-  let interps = MD.disambiguate_term dbd context metasenv aliases term in
+  let metasenv = MatitaTypes.get_proof_metasenv status in
+  let context = MatitaTypes.get_proof_context status user_goal in
+  let interps =
+   MD.disambiguate_term ~dbd ~context ~metasenv ~aliases:status.aliases
+    ~universe:(Some status.multi_aliases) term in
   match interps with 
   | [_,_,x,_], _ -> x
   | _ -> assert false
  
-let eval_macro guistuff status unparsed_text parsed_text script mac =
-  let module TA = GrafiteAst in
+let eval_macro guistuff status user_goal unparsed_text parsed_text script mac =
   let module TAPp = GrafiteAstPp in
   let module MQ = MetadataQuery in
   let module MDB = MatitaDb in
@@ -226,17 +236,17 @@ let eval_macro guistuff status unparsed_text parsed_text script mac =
   match mac with
   (* WHELP's stuff *)
   | TA.WMatch (loc, term) -> 
-      let term = disambiguate term status in
+      let term = disambiguate_macro_term term status user_goal in
       let l =  MQ.match_term ~dbd term in
       let query_url =
         MatitaMisc.strip_suffix ~suffix:"."
-          (MatitaMisc.trim_blanks unparsed_text)
+          (HExtlib.trim_blanks unparsed_text)
       in
       let entry = `Whelp (query_url, l) in
       guistuff.mathviewer#show_uri_list ~reuse:true ~entry l;
       [], parsed_text_length
   | TA.WInstance (loc, term) ->
-      let term = disambiguate term status in
+      let term = disambiguate_macro_term term status user_goal in
       let l = MQ.instance ~dbd term in
       let entry = `Whelp (TAPp.pp_macro_cic (TA.WInstance (loc, term)), l) in
       guistuff.mathviewer#show_uri_list ~reuse:true ~entry l;
@@ -247,7 +257,7 @@ let eval_macro guistuff status unparsed_text parsed_text script mac =
       guistuff.mathviewer#show_uri_list ~reuse:true ~entry l;
       [], parsed_text_length
   | TA.WElim (loc, term) ->
-      let term = disambiguate term status in
+      let term = disambiguate_macro_term term status user_goal in
       let uri =
         match term with
         | Cic.MutInd (uri,n,_) -> UriManager.uri_of_uriref uri n None 
@@ -258,7 +268,7 @@ let eval_macro guistuff status unparsed_text parsed_text script mac =
       guistuff.mathviewer#show_uri_list ~reuse:true ~entry l;
       [], parsed_text_length
   | TA.WHint (loc, term) ->
-      let term = disambiguate term status in
+      let term = disambiguate_macro_term term status user_goal in
       let s = ((None,[0,[],term], Cic.Meta (0,[]) ,term),0) in
       let l = List.map fst (MQ.experimental_hint ~dbd s) in
       let entry = `Whelp (TAPp.pp_macro_cic (TA.WHint (loc, term)), l) in
@@ -266,24 +276,26 @@ let eval_macro guistuff status unparsed_text parsed_text script mac =
       [], parsed_text_length
   (* REAL macro *)
   | TA.Hint loc -> 
-      let s = MatitaMisc.get_proof_status status in
-      let l = List.map fst (MQ.experimental_hint ~dbd s) in
+      let proof = MatitaTypes.get_current_proof status in
+      let proof_status = proof, user_goal in
+      let l = List.map fst (MQ.experimental_hint ~dbd proof_status) in
       let selected = guistuff.urichooser l in
       (match selected with
       | [] -> [], parsed_text_length
       | [uri] -> 
-        let ast = 
-         TA.Executable (loc,
-          (TA.Tactical (loc, 
-            TA.Tactic (loc,
-             TA.Apply (loc, CicNotationPt.Uri (UriManager.string_of_uri uri,None))))))
-        in
+          let suri = UriManager.string_of_uri uri in
+          let ast = 
+            TA.Executable (loc, (TA.Tactical (loc,
+              TA.Tactic (loc,
+                TA.Apply (loc, CicNotationPt.Uri (suri, None))),
+                Some (TA.Dot loc))))
+          in
         let new_status = MatitaEngine.eval_ast status ast in
         let extra_text = 
           comment parsed_text ^ 
           "\n" ^ TAPp.pp_statement ast
         in
-        [ new_status , extra_text ], parsed_text_length
+        [ new_status , (extra_text, ast) ], parsed_text_length
       | _ -> 
           MatitaLog.error 
             "The result of the urichooser should be only 1 uri, not:\n";
@@ -292,12 +304,11 @@ let eval_macro guistuff status unparsed_text parsed_text script mac =
           ) selected;
           assert false)
   | TA.Check (_,term) ->
-      let metasenv = MatitaMisc.get_proof_metasenv status in
-      let context = MatitaMisc.get_proof_context status in
-      let aliases = MatitaMisc.get_proof_aliases status in
+      let metasenv = MatitaTypes.get_proof_metasenv status in
+      let context = MatitaTypes.get_proof_context status user_goal in
       let interps = 
-        MatitaDisambiguator.disambiguate_term 
-          dbd context metasenv aliases term 
+        MatitaDisambiguator.disambiguate_term ~dbd ~context ~metasenv
+         ~aliases:status.aliases ~universe:(Some status.multi_aliases) term
       in
       let _, metasenv , term, ugraph =
         match interps with 
@@ -329,18 +340,16 @@ let eval_macro guistuff status unparsed_text parsed_text script mac =
   | TA.Print (_,kind) -> failwith "not implemented"
   | TA.Search_pat (_, search_kind, str) -> failwith "not implemented"
   | TA.Search_term (_, search_kind, term) -> failwith "not implemented"
-
                                 
 let eval_executable guistuff status user_goal unparsed_text parsed_text script
   ex
 =
-  let module TA = GrafiteAst in
   let module TAPp = GrafiteAstPp in
   let module MD = MatitaDisambiguator in
   let module ML = MatitacleanLib in
   let parsed_text_length = String.length parsed_text in
   match ex with
-  | TA.Command (loc, _) | TA.Tactical (loc, _) ->
+  | TA.Command (loc, _) | TA.Tactical (loc, _, _) ->
       (try 
         (match MatitaMisc.baseuri_of_baseuri_decl (TA.Executable (loc,ex)) with
         | None -> ()
@@ -361,41 +370,48 @@ let eval_executable guistuff status user_goal unparsed_text parsed_text script
           guistuff status user_goal parsed_text (TA.Executable (loc, ex))
       with MatitaTypes.Cancel -> [], 0)
   | TA.Macro (_,mac) ->
-      eval_macro guistuff status unparsed_text parsed_text script mac
+      eval_macro guistuff status user_goal unparsed_text parsed_text script mac
+
+let parse_statement baseoffset parsedlen ?error_tag (buffer: GText.buffer) text 
+=
+  try
+    GrafiteParser.parse_statement (Ulexing.from_utf8_string text)
+  with CicNotationParser.Parse_error (floc, err) as exn ->
+    match error_tag with
+    | None -> raise exn
+    | Some error_tag ->
+        let (x, y) = CicNotationPt.loc_of_floc floc in
+        let x = parsedlen + x in
+        let y = parsedlen + y in
+        let x' = baseoffset + x in
+        let y' = baseoffset + y in
+        let x_iter = buffer#get_iter (`OFFSET x') in
+        let y_iter = buffer#get_iter (`OFFSET y') in
+        buffer#apply_tag error_tag ~start:x_iter ~stop:y_iter;
+        let id = ref None in
+        id := Some (buffer#connect#changed ~callback:(fun () ->
+          buffer#remove_tag error_tag ~start:buffer#start_iter
+            ~stop:buffer#end_iter;
+          match !id with
+          | None -> assert false (* a race condition occurred *)
+          | Some id ->
+              (new GObj.gobject_ops buffer#as_buffer)#disconnect id));
+        let flocb,floce = floc in
+        let floc =
+          { flocb with Lexing.pos_cnum = x }, { floce with Lexing.pos_cnum = y }
+        in
+        buffer#place_cursor (buffer#get_iter (`OFFSET x'));
+        raise (CicNotationParser.Parse_error (floc, err))
 
 let rec eval_statement baseoffset parsedlen error_tag (buffer : GText.buffer)
- guistuff status user_goal script unparsed_text
+ guistuff status user_goal script statement
 =
-  if Pcre.pmatch ~rex:only_dust_RE unparsed_text then raise Margin;
-  let st =
-   try
-    GrafiteParser.parse_statement (Stream.of_string unparsed_text)
-   with
-    CicNotationParser.Parse_error (floc,err) as exc ->
-     let (x, y) = CicNotationPt.loc_of_floc floc in
-     let x = parsedlen + x in
-     let y = parsedlen + y in
-     let x' = baseoffset + x in
-     let y' = baseoffset + y in
-     let x_iter = buffer#get_iter (`OFFSET x') in
-     let y_iter = buffer#get_iter (`OFFSET y') in
-      buffer#apply_tag error_tag ~start:x_iter ~stop:y_iter;
-     let id = ref None in
-      id :=
-       Some
-        (buffer#connect#changed
-          ~callback:(
-            fun () ->
-             buffer#remove_tag error_tag ~start:buffer#start_iter
-              ~stop:buffer#end_iter;
-             match !id with
-                None -> assert false (* a race condition occurred *)
-              | Some id ->
-                 (new GObj.gobject_ops buffer#as_buffer)#disconnect id));
-      let flocb,floce = floc in
-      let floc =
-       {flocb with Lexing.pos_cnum = x}, {floce with Lexing.pos_cnum = y } in
-      raise (CicNotationParser.Parse_error (floc,err))
+  let st, unparsed_text =
+    match statement with
+    | `Raw text ->
+        if Pcre.pmatch ~rex:only_dust_RE text then raise Margin;
+        parse_statement baseoffset parsedlen ~error_tag buffer text, text
+    | `Ast (st, text) -> st, text
   in
   let text_of_loc loc =
     let parsed_text_length = snd (CicNotationPt.loc_of_floc loc) in
@@ -409,11 +425,11 @@ let rec eval_statement baseoffset parsedlen error_tag (buffer : GText.buffer)
       let s = String.sub unparsed_text parsed_text_length remain_len in
       let s,len = 
         eval_statement baseoffset (parsedlen + parsed_text_length) error_tag
-         buffer guistuff status user_goal script 
+         buffer guistuff status user_goal script (`Raw s)
       in
       (match s with
-      | (status, text) :: tl ->
-        ((status, parsed_text ^ text)::tl), (parsed_text_length + len)
+      | (status, (text, ast)) :: tl ->
+          ((status, (parsed_text ^ text, ast))::tl), (parsed_text_length + len)
       | [] -> [], 0)
   | GrafiteAst.Executable (loc, ex) ->
       let parsed_text, parsed_text_length = text_of_loc loc in
@@ -446,20 +462,19 @@ object (self)
   
   method private getFilename =
     match guistuff.filenamedata with Some f,_ -> f | _ -> assert false
+
+  method filename = self#getFilename
     
   method private ppFilename =
     match guistuff.filenamedata with 
-    | Some f,_ -> Filename.basename 
+    | Some f,_ -> f 
     | None,_ -> sprintf ".unnamed%d.ma" scriptId
   
   initializer 
-    ignore(GMain.Timeout.add ~ms:300000 
-       ~callback:(fun _ -> self#_saveToBackuptFile ();true));
-    ignore(buffer#connect#modified_changed 
-       (fun _ -> if buffer#modified then 
-          set_star self#ppFilename true 
-        else 
-          set_star self#ppFilename false))
+    ignore (GMain.Timeout.add ~ms:300000 
+       ~callback:(fun _ -> self#_saveToBackupFile ();true));
+    ignore (buffer#connect#modified_changed 
+      (fun _ -> set_star (Filename.basename self#ppFilename) buffer#modified))
 
   val mutable statements = [];    (** executed statements *)
   val mutable history = [ init ];
@@ -485,30 +500,56 @@ object (self)
   method status = match history with hd :: _ -> hd | _ -> assert false
 
   method private _advance ?statement () =
+    let rec aux st =
+      let baseoffset = (buffer#get_iter_at_mark (`MARK locked_mark))#offset in
+      let (entries, parsed_len) = 
+        eval_statement baseoffset 0 error_tag buffer guistuff self#status
+          userGoal self st
+      in
+      let (new_statuses, new_statements, new_asts) =
+        let statuses, statements = List.split entries in
+        let texts, asts = List.split statements in
+        statuses, texts, asts
+      in
+      history <- List.rev new_statuses @ history;
+      statements <- List.rev new_statements @ statements;
+      let start = buffer#get_iter_at_mark (`MARK locked_mark) in
+      let new_text = String.concat "" new_statements in
+      if statement <> None then
+       buffer#insert ~iter:start new_text
+      else
+        let s = match st with `Raw s | `Ast (_, s) -> s in
+        if new_text <> String.sub s 0 parsed_len then
+        begin
+          let stop = start#copy#forward_chars parsed_len in
+          buffer#delete ~start ~stop;
+          buffer#insert ~iter:start new_text;
+        end;
+      self#moveMark (String.length new_text);
+      (*
+      (match List.rev new_asts with (* advance again on punctuation *)
+      | TA.Executable (_, TA.Tactical (_, tac, _)) :: _ ->
+          let baseoffset =
+            (buffer#get_iter_at_mark (`MARK locked_mark))#offset
+          in
+          let text = self#getFuture in
+          (try
+            (match parse_statement baseoffset 0 ?error_tag:None buffer text with
+            | TA.Executable (loc, TA.Tactical (_, tac, None)) as st
+              when GrafiteAst.is_punctuation tac ->
+                let len = snd (CicNotationPt.loc_of_floc loc) in
+                aux (`Ast (st, String.sub text 0 len))
+            | _ -> ())
+          with CicNotationParser.Parse_error _ | End_of_file -> ())
+      | _ -> ())
+      *)
+    in
     let s = match statement with Some s -> s | None -> self#getFuture in
     MatitaLog.debug ("evaluating: " ^ first_line s ^ " ...");
-    let (entries, parsed_len) = 
-     eval_statement (buffer#get_iter_at_mark (`MARK locked_mark))#offset 0
-      error_tag buffer guistuff self#status userGoal self s
-    in
-    let (new_statuses, new_statements) = List.split entries in
-    history <- List.rev new_statuses @ history;
-    statements <- List.rev new_statements @ statements;
-    let start = buffer#get_iter_at_mark (`MARK locked_mark) in
-    let new_text = String.concat "" new_statements in
-    if statement <> None then
-     buffer#insert ~iter:start new_text
-    else
-     if new_text <> String.sub s 0 parsed_len then
-      begin
-       let stop = start#copy#forward_chars parsed_len in
-        buffer#delete ~start ~stop;
-        buffer#insert ~iter:start new_text;
-      end;
-    self#moveMark (String.length new_text)
+    (try aux (`Raw s) with End_of_file -> raise Margin)
 
   method private _retract offset status new_statements new_history =
-   let cur_status = match history with s::_ -> s | [] -> assert false in
+    let cur_status = match history with s::_ -> s | [] -> assert false in
     MatitaSync.time_travel ~present:cur_status ~past:status;
     statements <- new_statements;
     history <- new_history;
@@ -557,11 +598,6 @@ object (self)
     buffer#move_mark mark ~where:new_mark_pos;
     buffer#apply_tag locked_tag ~start:buffer#start_iter ~stop:new_mark_pos;
     buffer#move_mark `INSERT old_insert;
-    begin
-     match self#status.proof_status with
-        Incomplete_proof (_,goal) -> self#setGoal goal
-      | _ -> ()
-    end ;
     let mark_position = buffer#get_iter_at_mark mark in
     if source_view#move_mark_onscreen mark then
      begin
@@ -585,7 +621,7 @@ object (self)
     List.iter (fun o -> o status) observers
 
   method loadFromFile f =
-    buffer#set_text (MatitaMisc.input_file f);
+    buffer#set_text (HExtlib.input_file f);
     self#reset_buffer;
     buffer#set_modified false
     
@@ -601,7 +637,7 @@ object (self)
     close_out oc;
     buffer#set_modified false
   
-  method private _saveToBackuptFile () =
+  method private _saveToBackupFile () =
     if buffer#modified then
       begin
         let f = self#ppFilename ^ "~" in
@@ -631,12 +667,12 @@ object (self)
     buffer#set_modified false
   
   method template () =
-    let template = MatitaMisc.input_file BuildTimeConf.script_template in 
+    let template = HExtlib.input_file BuildTimeConf.script_template in 
     buffer#insert ~iter:(buffer#get_iter `START) template;
     guistuff.filenamedata <- 
       (None,MatitamakeLib.development_for_dir (Unix.getcwd ()));
     buffer#set_modified false;
-    set_star self#ppFilename false
+    set_star (Filename.basename self#ppFilename) false
 
   method goto (pos: [`Top | `Bottom | `Cursor]) () =
     let old_locked_mark =
@@ -723,17 +759,19 @@ object (self)
     | Incomplete_proof _ -> true
     | Intermediate _ -> assert false
 
-  method proofStatus = MatitaMisc.get_proof_status self#status
-  method proofMetasenv = MatitaMisc.get_proof_metasenv self#status
-  method proofContext = MatitaMisc.get_proof_context self#status
-  method proofConclusion = MatitaMisc.get_proof_conclusion self#status
+(*   method proofStatus = MatitaTypes.get_proof_status self#status *)
+  method proofMetasenv = MatitaTypes.get_proof_metasenv self#status
+  method proofContext = MatitaTypes.get_proof_context self#status userGoal
+  method proofConclusion = MatitaTypes.get_proof_conclusion self#status userGoal
+  method stack = MatitaTypes.get_stack self#status
   method setGoal n = userGoal <- n
+  method goal = userGoal
 
   method eos = 
     let s = self#getFuture in
     let rec is_there_and_executable s = 
       if Pcre.pmatch ~rex:only_dust_RE s then raise Margin;
-      let st = GrafiteParser.parse_statement (Stream.of_string s) in
+      let st = GrafiteParser.parse_statement (Ulexing.from_utf8_string s) in
       match st with
       | GrafiteAst.Comment (loc,_)-> 
           let parsed_text_length = snd (CicNotationPt.loc_of_floc loc) in
@@ -746,10 +784,8 @@ object (self)
       is_there_and_executable s
     with 
     | CicNotationParser.Parse_error _ -> false
-    | Margin -> true
-      
-    
-    
+    | Margin | End_of_file -> true
+
   (* debug *)
   method dump () =
     MatitaLog.debug "script status:";
@@ -773,6 +809,5 @@ let script ~source_view ~init ~mathviewer ~urichooser ~develcreator ~ask_confirm
   _script := Some s;
   s
 
-let instance () = match !_script with None -> assert false | Some s -> s
-
+let current () = match !_script with None -> assert false | Some s -> s