/- Copyright (c) 2016 Microsoft Corporation. All rights reserved. Released under Apache 2.0 license as described in the file LICENSE. Authors: Leonardo de Moura -/ namespace debugger def is_space (c : char) : bool := if c = ' ' ∨ c = char.of_nat 11 ∨ c = '\n' then tt else ff private def split_core : list char → option string → list string | (c::cs) none := if is_space c then split_core cs none else split_core cs (some $ string.singleton c) | (c::cs) (some s) := if is_space c then s :: split_core cs none else split_core cs (s.str c) | [] none := [] | [] (some s) := [s] def split (s : string) : list string := split_core s.to_list none def to_qualified_name_core : list char → name → string → name | [] r s := if s.is_empty then r else r <.> s | (c::cs) r s := if is_space c then to_qualified_name_core cs r s else if c = '.' then if s.is_empty then to_qualified_name_core cs r "" else to_qualified_name_core cs (r <.> s) "" else to_qualified_name_core cs r (s.str c) def to_qualified_name (s : string) : name := to_qualified_name_core s.to_list name.anonymous "" def olean_to_lean (s : string) := s.popn_back 5 ++ "lean" meta def get_file (fn : name) : vm string := do { d ← vm.get_decl fn, some n ← return (vm_decl.olean d) | failure, return (olean_to_lean n) } <|> return "[current file]" meta def pos_info (fn : name) : vm string := do { d ← vm.get_decl fn, some p ← return (vm_decl.pos d) | failure, file ← get_file fn, return sformat!"{file}:{p.line}:{p.column}" } <|> return "" meta def show_fn (header : string) (fn : name) (frame : nat) : vm unit := do pos ← pos_info fn, vm.put_str sformat!"[{frame}] {header}", if header = "" then return () else vm.put_str " ", vm.put_str sformat!"{fn} at {pos}\n" meta def show_curr_fn (header : string) : vm unit := do fn ← vm.curr_fn, sz ← vm.call_stack_size, show_fn header fn (sz-1) meta def is_valid_fn_prefix (p : name) : vm bool := do env ← vm.get_env, return $ env.fold ff (λ d r, r || let n := d.to_name in p.is_prefix_of n) meta def show_frame (frame_idx : nat) : vm unit := do sz ← vm.call_stack_size, fn ← if frame_idx >= sz then vm.curr_fn else vm.call_stack_fn frame_idx, show_fn "" fn frame_idx meta def type_to_string : option expr → nat → vm string | none i := do o ← vm.stack_obj i, match o.kind with | vm_obj_kind.simple := return "[tagged value]" | vm_obj_kind.constructor := return "[constructor]" | vm_obj_kind.closure := return "[closure]" | vm_obj_kind.native_closure := return "[native closure]" | vm_obj_kind.mpz := return "[big num]" | vm_obj_kind.name := return "name" | vm_obj_kind.level := return "level" | vm_obj_kind.expr := return "expr" | vm_obj_kind.declaration := return "declaration" | vm_obj_kind.environment := return "environment" | vm_obj_kind.tactic_state := return "tactic_state" | vm_obj_kind.format := return "format" | vm_obj_kind.options := return "options" | vm_obj_kind.other := return "[other]" end | (some type) i := do fmt ← vm.pp_expr type, opts ← vm.get_options, return $ fmt.to_string opts meta def show_vars_core : nat → nat → nat → vm unit | c i e := if i = e then return () else do (n, type) ← vm.stack_obj_info i, type_str ← type_to_string type i, vm.put_str sformat!"#{c} {n} : {type_str}\n", show_vars_core (c+1) (i+1) e meta def show_vars (frame : nat) : vm unit := do (s, e) ← vm.call_stack_var_range frame, show_vars_core 0 s e meta def show_stack_core : nat → vm unit | 0 := return () | (i+1) := do fn ← vm.call_stack_fn i, show_fn "" fn i, show_stack_core i meta def show_stack : vm unit := do sz ← vm.call_stack_size, show_stack_core sz end debugger