2014-06-11 00:02:06 +00:00
/*
2015-06-02 00:53:25 +00:00
Copyright ( c ) 2014 - 2015 Microsoft Corporation . All rights reserved .
2014-06-11 00:02:06 +00:00
Released under Apache 2.0 license as described in the file LICENSE .
Author : Leonardo de Moura
*/
2014-06-13 22:13:32 +00:00
# include <algorithm>
2014-10-24 15:19:36 +00:00
# include <string>
2014-06-13 18:24:26 +00:00
# include "util/sstream.h"
2014-06-17 00:27:43 +00:00
# include "util/sexpr/option_declarations.h"
2014-06-13 22:13:32 +00:00
# include "kernel/type_checker.h"
2014-06-26 17:05:00 +00:00
# include "kernel/abstract.h"
2014-07-10 13:07:41 +00:00
# include "kernel/instantiate.h"
2015-07-29 00:06:22 +00:00
# include "kernel/for_each_fn.h"
2014-10-24 21:35:03 +00:00
# include "kernel/inductive/inductive.h"
2015-04-01 04:45:11 +00:00
# include "kernel/quotient/quotient.h"
2015-04-23 22:27:56 +00:00
# include "kernel/hits/hits.h"
2015-02-07 23:19:41 +00:00
# include "kernel/default_converter.h"
2014-06-11 18:52:58 +00:00
# include "library/io_state_stream.h"
2014-06-13 18:24:26 +00:00
# include "library/scoped_ext.h"
# include "library/aliases.h"
2014-09-04 01:37:01 +00:00
# include "library/protected.h"
2014-06-13 22:13:32 +00:00
# include "library/locals.h"
2014-06-26 15:08:39 +00:00
# include "library/coercion.h"
2014-09-19 20:30:08 +00:00
# include "library/reducible.h"
2014-09-27 03:16:03 +00:00
# include "library/normalize.h"
2014-10-27 02:19:45 +00:00
# include "library/print.h"
2015-07-30 00:54:35 +00:00
# include "library/noncomputable.h"
2014-12-10 05:12:39 +00:00
# include "library/class.h"
2014-11-24 16:35:49 +00:00
# include "library/flycheck.h"
2015-05-29 21:37:06 +00:00
# include "library/abbreviation.h"
2014-12-10 19:23:23 +00:00
# include "library/util.h"
2015-05-12 22:08:10 +00:00
# include "library/user_recursors.h"
2014-12-19 23:00:05 +00:00
# include "library/pp_options.h"
2015-06-17 21:42:25 +00:00
# include "library/composition_manager.h"
2015-10-01 01:50:28 +00:00
# include "library/aux_recursors.h"
# include "library/relation_manager.h"
# include "library/projection.h"
# include "library/private.h"
2015-10-15 01:36:41 +00:00
# include "library/decl_stats.h"
2015-11-01 17:59:37 +00:00
# include "library/app_builder.h"
2015-10-15 02:34:53 +00:00
# include "library/meng_paulson.h"
2014-10-29 19:50:52 +00:00
# include "library/definitional/projection.h"
2015-07-22 17:39:30 +00:00
# include "library/simplifier/simp_rule_set.h"
2015-09-12 00:12:32 +00:00
# include "compiler/preprocess_rec.h"
2014-06-18 17:36:21 +00:00
# include "frontends/lean/util.h"
2014-06-11 18:44:16 +00:00
# include "frontends/lean/parser.h"
2014-06-17 20:35:31 +00:00
# include "frontends/lean/calc.h"
2014-06-15 05:13:25 +00:00
# include "frontends/lean/notation_cmd.h"
2014-06-18 20:55:48 +00:00
# include "frontends/lean/inductive_cmd.h"
2014-07-29 02:04:57 +00:00
# include "frontends/lean/structure_cmd.h"
2015-03-15 03:32:39 +00:00
# include "frontends/lean/migrate_cmd.h"
2014-11-24 07:00:59 +00:00
# include "frontends/lean/find_cmd.h"
2014-08-21 17:36:44 +00:00
# include "frontends/lean/begin_end_ext.h"
2014-06-18 17:36:21 +00:00
# include "frontends/lean/decl_cmds.h"
2014-07-08 21:28:33 +00:00
# include "frontends/lean/tactic_hint.h"
2014-09-23 02:22:53 +00:00
# include "frontends/lean/tokens.h"
2014-12-02 20:04:18 +00:00
# include "frontends/lean/parse_table.h"
2014-06-11 00:02:06 +00:00
namespace lean {
2014-10-06 01:50:48 +00:00
static void print_coercions ( parser & p , optional < name > const & C ) {
environment const & env = p . env ( ) ;
options opts = p . regular_stream ( ) . get_options ( ) ;
opts = opts . update ( get_pp_coercions_option_name ( ) , true ) ;
io_state_stream out = p . regular_stream ( ) . update_options ( opts ) ;
char const * arrow = get_pp_unicode ( opts ) ? " ↣ " : " >-> " ;
2015-07-01 23:32:34 +00:00
for_each_coercion_user ( env , [ & ] ( name const & C1 , name const & c , name const & D ) {
2014-10-06 01:50:48 +00:00
if ( ! C | | * C = = C1 )
out < < C1 < < " " < < arrow < < " " < < D < < " : " < < c < < endl ;
} ) ;
2015-07-01 23:32:34 +00:00
for_each_coercion_sort ( env , [ & ] ( name const & C1 , name const & c ) {
2014-10-06 01:50:48 +00:00
if ( ! C | | * C = = C1 )
out < < C1 < < " " < < arrow < < " [sort-class] : " < < c < < endl ;
} ) ;
2015-07-01 23:32:34 +00:00
for_each_coercion_fun ( env , [ & ] ( name const & C1 , name const & c ) {
2014-10-06 01:50:48 +00:00
if ( ! C | | * C = = C1 )
out < < C1 < < " " < < arrow < < " [fun-class] : " < < c < < endl ;
} ) ;
}
2015-07-29 00:06:22 +00:00
struct print_axioms_deps {
environment m_env ;
io_state_stream m_ios ;
name_set m_visited ;
bool m_use_axioms ;
print_axioms_deps ( environment const & env , io_state_stream const & ios ) :
m_env ( env ) , m_ios ( ios ) , m_use_axioms ( false ) { }
void visit ( name const & n ) {
if ( m_visited . contains ( n ) )
return ;
m_visited . insert ( n ) ;
declaration const & d = m_env . get ( n ) ;
if ( ! d . is_definition ( ) & & ! m_env . is_builtin ( n ) ) {
m_use_axioms = true ;
m_ios < < d . get_name ( ) < < " \n " ;
}
visit ( d . get_type ( ) ) ;
if ( d . is_definition ( ) )
visit ( d . get_value ( ) ) ;
}
void visit ( expr const & e ) {
for_each ( e , [ & ] ( expr const & e , unsigned ) {
if ( is_constant ( e ) )
visit ( const_name ( e ) ) ;
return true ;
} ) ;
}
void operator ( ) ( name const & n ) {
visit ( n ) ;
if ( ! m_use_axioms )
m_ios < < " no axioms " < < endl ;
}
} ;
2015-08-12 04:08:45 +00:00
static environment print_axioms ( parser & p ) {
2015-07-29 00:06:22 +00:00
if ( p . curr_is_identifier ( ) ) {
name c = p . check_constant_next ( " invalid 'print axioms', constant expected " ) ;
2015-08-12 04:08:45 +00:00
environment new_env = p . reveal_all_theorems ( ) ;
2015-07-29 00:06:22 +00:00
print_axioms_deps ( p . env ( ) , p . regular_stream ( ) ) ( c ) ;
2015-08-12 04:08:45 +00:00
return new_env ;
2015-07-29 00:06:22 +00:00
} else {
bool has_axioms = false ;
environment const & env = p . env ( ) ;
env . for_each_declaration ( [ & ] ( declaration const & d ) {
name const & n = d . get_name ( ) ;
2015-08-12 04:08:45 +00:00
if ( ! d . is_definition ( ) & & ! env . is_builtin ( n ) & & ! p . in_theorem_queue ( n ) ) {
2015-07-29 00:06:22 +00:00
p . regular_stream ( ) < < n < < " : " < < d . get_type ( ) < < endl ;
has_axioms = true ;
}
} ) ;
if ( ! has_axioms )
p . regular_stream ( ) < < " no axioms " < < endl ;
2015-08-12 04:08:45 +00:00
return p . env ( ) ;
2015-07-29 00:06:22 +00:00
}
2014-10-24 21:35:03 +00:00
}
2014-11-04 16:40:32 +00:00
static void print_prefix ( parser & p ) {
name prefix = p . check_id_next ( " invalid 'print prefix' command, identifier expected " ) ;
environment const & env = p . env ( ) ;
buffer < declaration > to_print ;
env . for_each_declaration ( [ & ] ( declaration const & d ) {
if ( is_prefix_of ( prefix , d . get_name ( ) ) ) {
to_print . push_back ( d ) ;
}
} ) ;
std : : sort ( to_print . begin ( ) , to_print . end ( ) , [ ] ( declaration const & d1 , declaration const & d2 ) { return d1 . get_name ( ) < d2 . get_name ( ) ; } ) ;
for ( declaration const & d : to_print ) {
p . regular_stream ( ) < < d . get_name ( ) < < " : " < < d . get_type ( ) < < endl ;
}
if ( to_print . empty ( ) )
p . regular_stream ( ) < < " no declaration starting with prefix ' " < < prefix < < " ' " < < endl ;
}
2015-07-30 17:18:03 +00:00
static void print_fields ( parser const & p , name const & S , pos_info const & pos ) {
2014-11-05 22:06:54 +00:00
environment const & env = p . env ( ) ;
if ( ! is_structure ( env , S ) )
throw parser_error ( sstream ( ) < < " invalid 'print fields' command, ' " < < S < < " ' is not a structure " , pos ) ;
buffer < name > field_names ;
get_structure_fields ( env , S , field_names ) ;
for ( name const & field_name : field_names ) {
declaration d = env . get ( field_name ) ;
p . regular_stream ( ) < < d . get_name ( ) < < " : " < < d . get_type ( ) < < endl ;
}
}
2014-12-02 20:04:18 +00:00
static bool uses_token ( unsigned num , notation : : transition const * ts , name const & token ) {
for ( unsigned i = 0 ; i < num ; i + + ) {
if ( ts [ i ] . get_token ( ) = = token )
return true ;
}
return false ;
}
static bool uses_some_token ( unsigned num , notation : : transition const * ts , buffer < name > const & tokens ) {
return
tokens . empty ( ) | |
std : : any_of ( tokens . begin ( ) , tokens . end ( ) , [ & ] ( name const & token ) { return uses_token ( num , ts , token ) ; } ) ;
}
2015-07-30 18:05:39 +00:00
static bool print_parse_table ( parser const & p , parse_table const & t , bool nud , buffer < name > const & tokens , bool tactic_table = false ) {
2014-12-02 20:04:18 +00:00
bool found = false ;
io_state ios = p . ios ( ) ;
options os = ios . get_options ( ) ;
2015-01-13 21:02:14 +00:00
os = os . update_if_undef ( get_pp_full_names_option_name ( ) , true ) ;
2014-12-02 20:04:18 +00:00
os = os . update ( get_pp_notation_option_name ( ) , false ) ;
2015-05-19 16:57:13 +00:00
os = os . update ( get_pp_preterm_name ( ) , true ) ;
2014-12-02 20:04:18 +00:00
ios . set_options ( os ) ;
optional < token_table > tt ( get_token_table ( p . env ( ) ) ) ;
2015-08-17 01:21:29 +00:00
t . for_each ( [ & ] ( unsigned num , notation : : transition const * ts , list < notation : : accepting > const & overloads ) {
2014-12-02 20:04:18 +00:00
if ( uses_some_token ( num , ts , tokens ) ) {
io_state_stream out = regular ( p . env ( ) , ios ) ;
2015-07-30 18:05:39 +00:00
if ( tactic_table )
out < < " tactic notation " ;
found = true ;
2014-12-02 20:04:18 +00:00
notation : : display ( out , num , ts , overloads , nud , tt ) ;
}
} ) ;
return found ;
}
static void print_notation ( parser & p ) {
buffer < name > tokens ;
while ( p . curr_is_keyword ( ) ) {
tokens . push_back ( p . get_token_info ( ) . token ( ) ) ;
p . next ( ) ;
}
bool found = false ;
if ( print_parse_table ( p , get_nud_table ( p . env ( ) ) , true , tokens ) )
found = true ;
if ( print_parse_table ( p , get_led_table ( p . env ( ) ) , false , tokens ) )
found = true ;
if ( ! found )
p . regular_stream ( ) < < " no notation " < < endl ;
}
2015-07-30 17:18:03 +00:00
static void print_metaclasses ( parser const & p ) {
2015-02-11 18:12:45 +00:00
buffer < name > c ;
get_metaclasses ( c ) ;
for ( name const & n : c )
p . regular_stream ( ) < < " [ " < < n < < " ] " < < endl ;
}
2015-04-29 22:22:10 +00:00
static void print_definition ( parser const & p , name const & n , pos_info const & pos ) {
environment const & env = p . env ( ) ;
declaration d = env . get ( n ) ;
io_state_stream out = p . regular_stream ( ) ;
options opts = out . get_options ( ) ;
opts = opts . update_if_undef ( get_pp_beta_name ( ) , false ) ;
io_state_stream new_out = out . update_options ( opts ) ;
2015-05-09 01:41:33 +00:00
if ( d . is_axiom ( ) )
throw parser_error ( sstream ( ) < < " invalid 'print definition', theorem ' " < < n
2015-06-13 18:32:59 +00:00
< < " ' is not available (suggestion: use command 'reveal " < < n < < " ') " , pos ) ;
2015-04-29 22:22:10 +00:00
if ( ! d . is_definition ( ) )
throw parser_error ( sstream ( ) < < " invalid 'print definition', ' " < < n < < " ' is not a definition " , pos ) ;
new_out < < d . get_value ( ) < < endl ;
}
2015-07-30 17:18:03 +00:00
static void print_attributes ( parser const & p , name const & n ) {
2015-04-29 23:17:33 +00:00
environment const & env = p . env ( ) ;
io_state_stream out = p . regular_stream ( ) ;
if ( is_coercion ( env , n ) )
out < < " [coercion] " ;
if ( is_class ( env , n ) )
out < < " [class] " ;
if ( is_instance ( env , n ) )
out < < " [instance] " ;
2015-07-22 17:39:30 +00:00
if ( is_simp_rule ( env , n ) )
out < < " [simp] " ;
2015-07-24 01:49:50 +00:00
if ( is_congr_rule ( env , n ) )
out < < " [congr] " ;
2015-04-29 23:17:33 +00:00
switch ( get_reducible_status ( env , n ) ) {
case reducible_status : : Reducible : out < < " [reducible] " ; break ;
case reducible_status : : Irreducible : out < < " [irreducible] " ; break ;
case reducible_status : : Quasireducible : out < < " [quasireducible] " ; break ;
case reducible_status : : Semireducible : break ;
}
}
2015-07-30 17:18:03 +00:00
static void print_inductive ( parser const & p , name const & n , pos_info const & pos ) {
2015-04-29 22:22:10 +00:00
environment const & env = p . env ( ) ;
io_state_stream out = p . regular_stream ( ) ;
if ( auto idecls = inductive : : is_inductive_decl ( env , n ) ) {
level_param_names ls ; unsigned nparams ; list < inductive : : inductive_decl > dlist ;
std : : tie ( ls , nparams , dlist ) = * idecls ;
if ( is_structure ( env , n ) )
out < < " structure " ;
else
out < < " inductive " ;
out < < " " < < n ;
2015-04-29 23:17:33 +00:00
print_attributes ( p , n ) ;
2015-04-29 22:22:10 +00:00
out < < " : " < < env . get ( n ) . get_type ( ) < < " \n " ;
if ( is_structure ( env , n ) ) {
out < < " fields: \n " ;
print_fields ( p , n , pos ) ;
} else {
out < < " constructors: \n " ;
buffer < name > constructors ;
get_intro_rule_names ( env , n , constructors ) ;
for ( name const & c : constructors ) {
out < < c < < " : " < < env . get ( c ) . get_type ( ) < < " \n " ;
}
}
} else {
throw parser_error ( sstream ( ) < < " invalid 'print inductive', ' " < < n < < " ' is not an inductive declaration " , pos ) ;
}
}
2015-05-12 22:08:10 +00:00
static void print_recursor_info ( parser & p ) {
name c = p . check_constant_next ( " invalid 'print [recursor]', constant expected " ) ;
2015-05-18 21:06:29 +00:00
auto out = p . regular_stream ( ) ;
2015-05-12 22:08:10 +00:00
recursor_info info = get_recursor_info ( p . env ( ) , c ) ;
2015-05-18 21:06:29 +00:00
out < < " recursor information \n "
< < " num. parameters: " < < info . get_num_params ( ) < < " \n "
2015-05-22 01:18:56 +00:00
< < " num. indices: " < < info . get_num_indices ( ) < < " \n "
< < " universe param pos.: " ;
for ( unsigned idx : info . get_universe_pos ( ) ) {
if ( idx = = recursor_info : : get_motive_univ_idx ( ) ) {
out < < " [motive univ] " ;
} else {
out < < " " < < idx ;
}
}
out < < " \n " ;
2015-05-18 21:06:29 +00:00
out < < " motive pos.: " < < info . get_motive_pos ( ) + 1 < < " \n "
< < " major premise pos.: " < < info . get_major_pos ( ) + 1 < < " \n "
< < " dep. elimination: " < < info . has_dep_elim ( ) < < " \n " ;
if ( info . get_num_params ( ) > 0 ) {
out < < " parameters pos. at major: " ;
2015-05-19 21:18:36 +00:00
for ( optional < unsigned > const & p : info . get_params_pos ( ) ) {
if ( p )
out < < " " < < * p + 1 ;
else
out < < " [instance] " ;
}
2015-05-18 21:06:29 +00:00
out < < " \n " ;
}
if ( info . get_num_indices ( ) > 0 ) {
out < < " indices pos. at major: " ;
for ( unsigned p : info . get_indices_pos ( ) )
out < < " " < < p + 1 ;
out < < " \n " ;
}
2015-05-12 22:08:10 +00:00
}
2015-07-30 17:18:03 +00:00
static bool print_constant ( parser const & p , char const * kind , declaration const & d , bool is_def = false ) {
2015-04-29 23:17:33 +00:00
io_state_stream out = p . regular_stream ( ) ;
2015-07-30 00:54:35 +00:00
if ( d . is_definition ( ) & & is_marked_noncomputable ( p . env ( ) , d . get_name ( ) ) )
out < < " noncomputable " ;
2015-05-19 00:19:34 +00:00
if ( is_protected ( p . env ( ) , d . get_name ( ) ) )
out < < " protected " ;
2015-04-29 23:17:33 +00:00
out < < kind < < " " < < d . get_name ( ) ;
print_attributes ( p , d . get_name ( ) ) ;
2015-06-13 18:32:59 +00:00
out < < " : " < < d . get_type ( ) ;
if ( is_def )
out < < " := " ;
out < < " \n " ;
2015-04-29 23:17:33 +00:00
return true ;
}
2015-07-30 17:18:03 +00:00
bool print_id_info ( parser const & p , name const & id , bool show_value , pos_info const & pos ) {
// declarations
2015-04-29 23:17:33 +00:00
try {
2015-07-30 17:18:03 +00:00
environment const & env = p . env ( ) ;
io_state_stream out = p . regular_stream ( ) ;
2015-04-29 23:17:33 +00:00
try {
2015-05-19 00:12:16 +00:00
list < name > cs = p . to_constants ( id , " " , pos ) ;
2015-07-30 17:18:03 +00:00
bool first = true ;
2015-05-19 00:12:16 +00:00
for ( name const & c : cs ) {
2015-07-30 17:18:03 +00:00
if ( first ) first = false ; else out < < " \n " ;
2015-05-19 00:12:16 +00:00
declaration const & d = env . get ( c ) ;
if ( d . is_theorem ( ) ) {
2015-07-30 17:18:03 +00:00
print_constant ( p , " theorem " , d , show_value ) ;
if ( show_value )
print_definition ( p , c , pos ) ;
2015-05-19 00:12:16 +00:00
} else if ( d . is_axiom ( ) | | d . is_constant_assumption ( ) ) {
if ( inductive : : is_inductive_decl ( env , c ) ) {
print_inductive ( p , c , pos ) ;
} else if ( inductive : : is_intro_rule ( env , c ) ) {
print_constant ( p , " constructor " , d ) ;
} else if ( inductive : : is_elim_rule ( env , c ) ) {
print_constant ( p , " eliminator " , d ) ;
} else if ( is_quotient_decl ( env , c ) ) {
print_constant ( p , " builtin-quotient-type-constant " , d ) ;
} else if ( is_hits_decl ( env , c ) ) {
print_constant ( p , " builtin-HIT-constant " , d ) ;
} else if ( d . is_axiom ( ) ) {
2015-06-13 19:12:22 +00:00
if ( p . in_theorem_queue ( d . get_name ( ) ) ) {
print_constant ( p , " theorem " , d ) ;
2015-07-30 17:18:03 +00:00
if ( show_value ) {
out < < " ' " < < d . get_name ( ) < < " ' is still in the theorem queue, use command 'reveal "
< < d . get_name ( ) < < " ' to access its definition. \n " ;
}
2015-06-13 19:12:22 +00:00
} else {
print_constant ( p , " axiom " , d ) ;
}
2015-05-19 00:12:16 +00:00
} else {
print_constant ( p , " constant " , d ) ;
}
} else if ( d . is_definition ( ) ) {
2015-07-30 17:18:03 +00:00
print_constant ( p , " definition " , d , show_value ) ;
if ( show_value )
print_definition ( p , c , pos ) ;
2015-04-29 23:17:33 +00:00
}
}
2015-05-19 00:12:16 +00:00
return true ;
2015-04-29 23:17:33 +00:00
} catch ( exception & ex ) { }
// variables and parameters
if ( expr const * type = p . get_local ( id ) ) {
if ( is_local ( * type ) ) {
if ( p . is_local_variable ( * type ) ) {
out < < " variable " < < id < < " : " < < mlocal_type ( * type ) < < " \n " ;
} else {
out < < " parameter " < < id < < " : " < < mlocal_type ( * type ) < < " \n " ;
}
return true ;
}
}
// options
for ( auto odecl : get_option_declarations ( ) ) {
auto opt = odecl . second ;
if ( opt . get_name ( ) = = id | | opt . get_name ( ) = = name ( " lean " ) + id ) {
out < < " option " < < opt . get_name ( ) < < " ( " < < opt . kind ( ) < < " ) "
< < opt . get_description ( ) < < " (default: " < < opt . get_default_value ( ) < < " ) " < < endl ;
return true ;
}
}
} catch ( exception & ) { }
2015-07-30 17:18:03 +00:00
return false ;
}
bool print_token_info ( parser const & p , name const & tk ) {
buffer < name > tokens ;
tokens . push_back ( tk ) ;
bool found = false ;
if ( print_parse_table ( p , get_nud_table ( p . env ( ) ) , true , tokens ) ) {
found = true ;
}
if ( print_parse_table ( p , get_led_table ( p . env ( ) ) , false , tokens ) ) {
found = true ;
}
2015-07-30 18:05:39 +00:00
bool tactic_table = true ;
if ( print_parse_table ( p , get_tactic_nud_table ( p . env ( ) ) , true , tokens , tactic_table ) ) {
found = true ;
}
if ( print_parse_table ( p , get_tactic_led_table ( p . env ( ) ) , false , tokens , tactic_table ) ) {
found = true ;
}
2015-07-30 17:18:03 +00:00
return found ;
}
bool print_polymorphic ( parser & p ) {
auto pos = p . pos ( ) ;
try {
name id = p . check_id_next ( " " ) ;
bool show_value = true ;
if ( print_id_info ( p , id , show_value , pos ) )
return true ;
} catch ( exception & ) { }
2015-04-29 23:17:33 +00:00
// notation
if ( p . curr_is_keyword ( ) ) {
name tk = p . get_token_info ( ) . token ( ) ;
2015-07-30 17:18:03 +00:00
if ( print_token_info ( p , tk ) ) {
2015-04-29 23:17:33 +00:00
p . next ( ) ;
return true ;
2015-07-30 17:18:03 +00:00
}
2015-04-29 23:17:33 +00:00
}
return false ;
}
2015-05-29 23:47:29 +00:00
static void print_reducible_info ( parser & p , reducible_status s1 ) {
buffer < name > r ;
for_each_reducible ( p . env ( ) , [ & ] ( name const & n , reducible_status s2 ) {
if ( s1 = = s2 )
r . push_back ( n ) ;
} ) ;
io_state_stream out = p . regular_stream ( ) ;
std : : sort ( r . begin ( ) , r . end ( ) ) ;
for ( name const & n : r )
out < < n < < " \n " ;
}
2015-07-23 23:11:37 +00:00
static void print_simp_rules ( parser & p ) {
2015-06-02 00:53:25 +00:00
io_state_stream out = p . regular_stream ( ) ;
2015-07-23 01:47:56 +00:00
simp_rule_sets s ;
name ns ;
if ( p . curr_is_identifier ( ) ) {
ns = p . get_name_val ( ) ;
p . next ( ) ;
s = get_simp_rule_sets ( p . env ( ) , ns ) ;
} else {
s = get_simp_rule_sets ( p . env ( ) ) ;
}
2015-07-27 19:57:28 +00:00
format header ;
if ( ! ns . is_anonymous ( ) )
header = format ( " at namespace ' " ) + format ( ns ) + format ( " ' " ) ;
out < < s . pp_simp ( out . get_formatter ( ) , header ) ;
2015-06-02 00:53:25 +00:00
}
2015-07-23 23:11:37 +00:00
static void print_congr_rules ( parser & p ) {
io_state_stream out = p . regular_stream ( ) ;
simp_rule_sets s = get_simp_rule_sets ( p . env ( ) ) ;
2015-07-27 19:57:28 +00:00
out < < s . pp_congr ( out . get_formatter ( ) ) ;
2015-07-23 23:11:37 +00:00
}
2014-06-11 18:52:58 +00:00
environment print_cmd ( parser & p ) {
2014-11-29 21:31:42 +00:00
flycheck_information info ( p . regular_stream ( ) ) ;
if ( info . enabled ( ) ) {
p . display_information_pos ( p . cmd_pos ( ) ) ;
p . regular_stream ( ) < < " print result: \n " ;
}
2014-06-11 18:52:58 +00:00
if ( p . curr ( ) = = scanner : : token_kind : : String ) {
2014-06-17 00:31:28 +00:00
p . regular_stream ( ) < < p . get_str_val ( ) < < endl ;
2014-06-11 18:52:58 +00:00
p . next ( ) ;
2014-09-23 02:22:53 +00:00
} else if ( p . curr_is_token_or_id ( get_raw_tk ( ) ) ) {
2014-06-12 03:56:10 +00:00
p . next ( ) ;
expr e = p . parse_expr ( ) ;
2014-10-19 15:40:56 +00:00
io_state_stream out = p . regular_stream ( ) ;
options opts = out . get_options ( ) ;
opts = opts . update ( get_pp_notation_option_name ( ) , false ) ;
out . update_options ( opts ) < < e < < endl ;
2015-05-29 23:47:29 +00:00
} else if ( p . curr_is_token_or_id ( get_reducible_tk ( ) ) ) {
p . next ( ) ;
print_reducible_info ( p , reducible_status : : Reducible ) ;
} else if ( p . curr_is_token_or_id ( get_quasireducible_tk ( ) ) ) {
p . next ( ) ;
print_reducible_info ( p , reducible_status : : Quasireducible ) ;
} else if ( p . curr_is_token_or_id ( get_irreducible_tk ( ) ) ) {
p . next ( ) ;
print_reducible_info ( p , reducible_status : : Irreducible ) ;
2014-09-23 02:22:53 +00:00
} else if ( p . curr_is_token_or_id ( get_options_tk ( ) ) ) {
2014-06-17 00:31:28 +00:00
p . next ( ) ;
p . regular_stream ( ) < < p . ios ( ) . get_options ( ) < < endl ;
2015-01-22 00:19:46 +00:00
} else if ( p . curr_is_token_or_id ( get_trust_tk ( ) ) ) {
p . next ( ) ;
p . regular_stream ( ) < < " trust level: " < < p . env ( ) . trust_lvl ( ) < < endl ;
2014-10-23 21:52:24 +00:00
} else if ( p . curr_is_token_or_id ( get_definition_tk ( ) ) ) {
p . next ( ) ;
auto pos = p . pos ( ) ;
2015-05-19 00:12:16 +00:00
name id = p . check_id_next ( " invalid 'print definition', constant expected " ) ;
list < name > cs = p . to_constants ( id , " invalid 'print definition', constant expected " , pos ) ;
2015-07-30 17:18:03 +00:00
bool first = true ;
2015-05-19 00:12:16 +00:00
for ( name const & c : cs ) {
2015-07-30 17:18:03 +00:00
if ( first )
first = false ;
else
p . regular_stream ( ) < < " \n " ;
2015-05-19 00:12:16 +00:00
declaration const & d = p . env ( ) . get ( c ) ;
if ( d . is_theorem ( ) ) {
print_constant ( p , " theorem " , d ) ;
print_definition ( p , c , pos ) ;
} else if ( d . is_definition ( ) ) {
print_constant ( p , " definition " , d ) ;
print_definition ( p , c , pos ) ;
} else {
throw parser_error ( sstream ( ) < < " invalid 'print definition', ' " < < c < < " ' is not a definition " , pos ) ;
}
}
2014-10-06 01:50:48 +00:00
} else if ( p . curr_is_token_or_id ( get_instances_tk ( ) ) ) {
p . next ( ) ;
name c = p . check_constant_next ( " invalid 'print instances', constant expected " ) ;
environment const & env = p . env ( ) ;
for ( name const & i : get_class_instances ( env , c ) ) {
p . regular_stream ( ) < < i < < " : " < < env . get ( i ) . get_type ( ) < < endl ;
}
2015-06-29 18:06:59 +00:00
if ( list < name > derived_insts = get_class_derived_trans_instances ( env , c ) ) {
p . regular_stream ( ) < < " Derived transitive instances: \n " ;
for ( name const & i : derived_insts ) {
p . regular_stream ( ) < < i < < " : " < < env . get ( i ) . get_type ( ) < < endl ;
}
}
2014-10-08 00:28:26 +00:00
} else if ( p . curr_is_token_or_id ( get_classes_tk ( ) ) ) {
p . next ( ) ;
environment const & env = p . env ( ) ;
buffer < name > classes ;
get_classes ( env , classes ) ;
std : : sort ( classes . begin ( ) , classes . end ( ) ) ;
for ( name const & c : classes ) {
p . regular_stream ( ) < < c < < " : " < < env . get ( c ) . get_type ( ) < < endl ;
}
2014-11-04 16:40:32 +00:00
} else if ( p . curr_is_token_or_id ( get_prefix_tk ( ) ) ) {
p . next ( ) ;
print_prefix ( p ) ;
2014-10-06 01:50:48 +00:00
} else if ( p . curr_is_token_or_id ( get_coercions_tk ( ) ) ) {
p . next ( ) ;
optional < name > C ;
if ( p . curr_is_identifier ( ) )
C = p . check_constant_next ( " invalid 'print coercions', constant expected " ) ;
print_coercions ( p , C ) ;
2015-02-11 18:12:45 +00:00
} else if ( p . curr_is_token_or_id ( get_metaclasses_tk ( ) ) ) {
p . next ( ) ;
print_metaclasses ( p ) ;
2014-10-24 21:35:03 +00:00
} else if ( p . curr_is_token_or_id ( get_axioms_tk ( ) ) ) {
p . next ( ) ;
2015-08-12 04:08:45 +00:00
return print_axioms ( p ) ;
2014-11-05 22:06:54 +00:00
} else if ( p . curr_is_token_or_id ( get_fields_tk ( ) ) ) {
p . next ( ) ;
2015-04-29 22:22:10 +00:00
auto pos = p . pos ( ) ;
name S = p . check_constant_next ( " invalid 'print fields' command, constant expected " ) ;
print_fields ( p , S , pos ) ;
2014-12-02 20:04:18 +00:00
} else if ( p . curr_is_token_or_id ( get_notation_tk ( ) ) ) {
p . next ( ) ;
print_notation ( p ) ;
2015-04-29 22:22:10 +00:00
} else if ( p . curr_is_token_or_id ( get_inductive_tk ( ) ) ) {
p . next ( ) ;
auto pos = p . pos ( ) ;
name c = p . check_constant_next ( " invalid 'print inductive', constant expected " ) ;
print_inductive ( p , c , pos ) ;
2015-05-12 22:08:10 +00:00
} else if ( p . curr_is_token ( get_recursor_tk ( ) ) ) {
p . next ( ) ;
2015-05-18 21:06:29 +00:00
p . check_token_next ( get_rbracket_tk ( ) , " invalid 'print [recursor]', ']' expected " ) ;
2015-05-12 22:08:10 +00:00
print_recursor_info ( p ) ;
2015-07-22 16:01:42 +00:00
} else if ( p . curr_is_token ( get_simp_attr_tk ( ) ) ) {
2015-06-02 00:53:25 +00:00
p . next ( ) ;
2015-07-23 23:11:37 +00:00
print_simp_rules ( p ) ;
} else if ( p . curr_is_token ( get_congr_attr_tk ( ) ) ) {
p . next ( ) ;
print_congr_rules ( p ) ;
2015-04-29 23:17:33 +00:00
} else if ( print_polymorphic ( p ) ) {
2014-06-11 18:52:58 +00:00
} else {
throw parser_error ( " invalid print command " , p . pos ( ) ) ;
}
return p . env ( ) ;
}
2014-06-13 18:24:26 +00:00
environment section_cmd ( parser & p ) {
2014-08-07 23:59:08 +00:00
name n ;
if ( p . curr_is_identifier ( ) )
n = p . check_atomic_id_next ( " invalid section, atomic identifier expected " ) ;
2014-06-13 18:24:26 +00:00
p . push_local_scope ( ) ;
2014-08-23 22:44:28 +00:00
return push_scope ( p . env ( ) , p . ios ( ) , scope_kind : : Section , n ) ;
}
2014-06-13 18:24:26 +00:00
environment namespace_cmd ( parser & p ) {
2014-07-08 02:15:46 +00:00
auto pos = p . pos ( ) ;
name n = p . check_atomic_id_next ( " invalid namespace declaration, atomic identifier expected " ) ;
if ( is_root_namespace ( n ) )
throw parser_error ( sstream ( ) < < " invalid namespace name, ' " < < n < < " ' is reserved " , pos ) ;
2014-10-10 23:21:30 +00:00
p . push_local_scope ( ) ;
2014-08-23 22:44:28 +00:00
return push_scope ( p . env ( ) , p . ios ( ) , scope_kind : : Namespace , n ) ;
2014-06-13 18:24:26 +00:00
}
2015-04-22 05:40:20 +00:00
static environment redeclare_aliases ( environment env , parser & p ,
list < pair < name , level > > old_level_entries ,
list < pair < name , expr > > old_entries ) {
environment const & old_env = p . env ( ) ;
2015-04-22 18:32:02 +00:00
if ( ! in_section ( old_env ) )
2015-04-22 05:40:20 +00:00
return env ;
2014-10-09 05:21:29 +00:00
list < pair < name , expr > > new_entries = p . get_local_entries ( ) ;
buffer < pair < name , expr > > to_redeclare ;
2015-06-04 04:40:03 +00:00
unsigned new_len = length ( new_entries ) ;
unsigned old_len = length ( old_entries ) ;
lean_assert ( old_len > = new_len ) ;
2014-10-09 05:21:29 +00:00
name_set popped_locals ;
2015-06-04 04:40:03 +00:00
while ( old_len > new_len ) {
2014-10-09 05:21:29 +00:00
pair < name , expr > entry = head ( old_entries ) ;
2014-10-11 22:33:31 +00:00
if ( is_local_ref ( entry . second ) )
2014-10-09 05:21:29 +00:00
to_redeclare . push_back ( entry ) ;
else if ( is_local ( entry . second ) )
popped_locals . insert ( mlocal_name ( entry . second ) ) ;
old_entries = tail ( old_entries ) ;
2015-06-04 04:40:03 +00:00
old_len - - ;
2014-10-09 05:21:29 +00:00
}
name_set popped_levels ;
list < pair < name , level > > new_level_entries = p . get_local_level_entries ( ) ;
while ( ! is_eqp ( old_level_entries , new_level_entries ) ) {
level const & l = head ( old_level_entries ) . second ;
if ( is_param ( l ) )
popped_levels . insert ( param_id ( l ) ) ;
old_level_entries = tail ( old_level_entries ) ;
}
for ( auto const & entry : to_redeclare ) {
2014-10-11 22:33:31 +00:00
expr new_ref = update_local_ref ( entry . second , popped_levels , popped_locals ) ;
2014-10-09 05:21:29 +00:00
if ( ! is_constant ( new_ref ) )
2015-04-22 05:40:20 +00:00
env = p . add_local_ref ( env , entry . first , new_ref ) ;
2014-10-09 05:21:29 +00:00
}
2015-04-22 05:40:20 +00:00
return env ;
2014-10-09 05:21:29 +00:00
}
2014-06-13 18:24:26 +00:00
environment end_scoped_cmd ( parser & p ) {
2014-10-09 05:21:29 +00:00
list < pair < name , level > > level_entries = p . get_local_level_entries ( ) ;
list < pair < name , expr > > entries = p . get_local_entries ( ) ;
2014-10-10 23:21:30 +00:00
p . pop_local_scope ( ) ;
2014-08-07 23:59:08 +00:00
if ( p . curr_is_identifier ( ) ) {
name n = p . check_atomic_id_next ( " invalid end of scope, atomic identifier expected " ) ;
2015-04-22 01:56:28 +00:00
environment env = pop_scope ( p . env ( ) , p . ios ( ) , n ) ;
2015-04-22 05:40:20 +00:00
return redeclare_aliases ( env , p , level_entries , entries ) ;
2014-08-07 23:59:08 +00:00
} else {
2015-04-22 01:56:28 +00:00
environment env = pop_scope ( p . env ( ) , p . ios ( ) ) ;
2015-04-22 05:40:20 +00:00
return redeclare_aliases ( env , p , level_entries , entries ) ;
2014-08-07 23:59:08 +00:00
}
2014-06-13 18:24:26 +00:00
}
2014-09-27 03:16:03 +00:00
environment check_cmd ( parser & p ) {
expr e ; level_param_names ls ;
std : : tie ( e , ls ) = parse_local_expr ( p ) ;
2015-05-29 21:37:06 +00:00
e = expand_abbreviations ( p . env ( ) , e ) ;
2015-05-08 21:36:38 +00:00
auto tc = mk_type_checker ( p . env ( ) , p . mk_ngen ( ) ) ;
2014-09-27 03:16:03 +00:00
expr type = tc - > check ( e , ls ) . first ;
2014-07-10 17:32:00 +00:00
auto reg = p . regular_stream ( ) ;
2014-10-30 20:28:25 +00:00
formatter fmt = reg . get_formatter ( ) ;
2014-07-10 17:32:00 +00:00
options opts = p . ios ( ) . get_options ( ) ;
2015-01-13 21:02:14 +00:00
opts = opts . update_if_undef ( get_pp_metavar_args_name ( ) , true ) ;
2014-10-30 20:28:25 +00:00
fmt = fmt . update_options ( opts ) ;
2014-07-10 17:32:00 +00:00
unsigned indent = get_pp_indent ( opts ) ;
2014-08-24 16:35:25 +00:00
format r = group ( fmt ( e ) + space ( ) + colon ( ) + nest ( indent , line ( ) + fmt ( type ) ) ) ;
2014-11-24 16:35:49 +00:00
flycheck_information info ( p . regular_stream ( ) ) ;
if ( info . enabled ( ) ) {
p . display_information_pos ( p . cmd_pos ( ) ) ;
p . regular_stream ( ) < < " check result: \n " ;
}
2014-07-10 17:32:00 +00:00
reg < < mk_pair ( r , opts ) < < endl ;
2014-06-13 22:13:32 +00:00
return p . env ( ) ;
}
2014-09-27 03:16:03 +00:00
environment eval_cmd ( parser & p ) {
2014-11-10 20:46:04 +00:00
bool whnf = false ;
2014-11-08 04:39:50 +00:00
if ( p . curr_is_token ( get_whnf_tk ( ) ) ) {
p . next ( ) ;
whnf = true ;
}
2014-09-27 03:16:03 +00:00
expr e ; level_param_names ls ;
std : : tie ( e , ls ) = parse_local_expr ( p ) ;
2014-11-08 04:39:50 +00:00
expr r ;
if ( whnf ) {
2015-05-08 21:36:38 +00:00
auto tc = mk_type_checker ( p . env ( ) , p . mk_ngen ( ) ) ;
2014-11-08 04:39:50 +00:00
r = tc - > whnf ( e ) . first ;
} else {
2015-05-09 18:49:55 +00:00
type_checker tc ( p . env ( ) ) ;
2015-05-21 02:09:44 +00:00
bool eta = false ;
bool eval_nested_prop = false ;
r = normalize ( tc , ls , e , eta , eval_nested_prop ) ;
2014-11-08 04:39:50 +00:00
}
2014-11-24 16:35:49 +00:00
flycheck_information info ( p . regular_stream ( ) ) ;
if ( info . enabled ( ) ) {
p . display_information_pos ( p . cmd_pos ( ) ) ;
p . regular_stream ( ) < < " eval result: \n " ;
}
2014-09-27 03:16:03 +00:00
p . regular_stream ( ) < < r < < endl ;
return p . env ( ) ;
}
2015-04-20 21:45:39 +00:00
environment exit_cmd ( parser & p ) {
flycheck_warning wrn ( p . regular_stream ( ) ) ;
p . display_warning_pos ( p . cmd_pos ( ) ) ;
p . regular_stream ( ) < < " using 'exit' to interrupt Lean " < < endl ;
2014-06-16 20:00:35 +00:00
throw interrupt_parser ( ) ;
}
2014-06-17 00:27:43 +00:00
environment set_option_cmd ( parser & p ) {
2015-07-13 23:13:41 +00:00
auto id_kind = parse_option_name ( p , " invalid set option, identifier (i.e., option name) expected " ) ;
name id = id_kind . first ;
option_kind k = id_kind . second ;
2014-06-17 00:27:43 +00:00
if ( k = = BoolOption ) {
2014-09-23 02:22:53 +00:00
if ( p . curr_is_token_or_id ( get_true_tk ( ) ) )
2014-06-17 00:27:43 +00:00
p . set_option ( id , true ) ;
2014-09-23 02:22:53 +00:00
else if ( p . curr_is_token_or_id ( get_false_tk ( ) ) )
2014-06-17 00:27:43 +00:00
p . set_option ( id , false ) ;
else
throw parser_error ( " invalid Boolean option value, 'true' or 'false' expected " , p . pos ( ) ) ;
p . next ( ) ;
} else if ( k = = StringOption ) {
if ( ! p . curr_is_string ( ) )
throw parser_error ( " invalid option value, given option is not a string " , p . pos ( ) ) ;
p . set_option ( id , p . get_str_val ( ) ) ;
p . next ( ) ;
} else if ( k = = DoubleOption ) {
p . set_option ( id , p . parse_double ( ) ) ;
} else if ( k = = UnsignedOption | | k = = IntOption ) {
p . set_option ( id , p . parse_small_nat ( ) ) ;
} else {
throw parser_error ( " invalid option value, 'true', 'false', string, integer or decimal value expected " , p . pos ( ) ) ;
}
p . updt_options ( ) ;
2014-10-30 21:45:35 +00:00
environment env = p . env ( ) ;
return update_fingerprint ( env , p . get_options ( ) . hash ( ) ) ;
2014-06-17 00:27:43 +00:00
}
2015-07-07 23:37:06 +00:00
static bool is_next_metaclass_tk ( parser const & p ) {
return p . curr_is_token ( get_lbracket_tk ( ) ) | | p . curr_is_token ( get_unfold_hints_bracket_tk ( ) ) ;
}
2015-02-11 18:35:04 +00:00
static name parse_metaclass ( parser & p ) {
2014-09-23 02:22:53 +00:00
if ( p . curr_is_token ( get_lbracket_tk ( ) ) ) {
2014-06-17 01:42:39 +00:00
p . next ( ) ;
2015-02-11 18:35:04 +00:00
auto pos = p . pos ( ) ;
2014-06-17 01:42:39 +00:00
name n ;
2015-02-11 18:35:04 +00:00
while ( ! p . curr_is_token ( get_rbracket_tk ( ) ) ) {
if ( p . curr_is_identifier ( ) )
n = n . append_after ( p . get_name_val ( ) . to_string ( ) . c_str ( ) ) ;
else if ( p . curr_is_keyword ( ) | | p . curr_is_command ( ) )
n = n . append_after ( p . get_token_info ( ) . value ( ) . to_string ( ) . c_str ( ) ) ;
else if ( p . curr_is_token ( get_sub_tk ( ) ) )
n = n . append_after ( " - " ) ;
else
throw parser_error ( " invalid 'open' command, identifier or symbol expected " , pos ) ;
p . next ( ) ;
}
2014-09-23 02:22:53 +00:00
p . check_token_next ( get_rbracket_tk ( ) , " invalid 'open' command, ']' expected " ) ;
2015-02-11 18:35:04 +00:00
if ( ! is_metaclass ( n ) & & n ! = get_decls_tk ( ) & & n ! = get_declarations_tk ( ) )
throw parser_error ( sstream ( ) < < " invalid metaclass name '[ " < < n < < " ]' " , pos ) ;
2014-06-17 01:42:39 +00:00
return n ;
2015-07-07 23:37:06 +00:00
} else if ( p . curr_is_token ( get_unfold_hints_bracket_tk ( ) ) ) {
p . next ( ) ;
return get_unfold_hints_tk ( ) ;
2014-06-17 01:42:39 +00:00
} else {
return name ( ) ;
}
}
2015-02-11 19:02:59 +00:00
static void parse_metaclasses ( parser & p , buffer < name > & r ) {
if ( p . curr_is_token ( get_sub_tk ( ) ) ) {
p . next ( ) ;
buffer < name > tmp ;
get_metaclasses ( tmp ) ;
tmp . push_back ( get_decls_tk ( ) ) ;
2015-07-07 23:37:06 +00:00
while ( is_next_metaclass_tk ( p ) ) {
2015-02-11 19:02:59 +00:00
name m = parse_metaclass ( p ) ;
tmp . erase_elem ( m ) ;
2015-05-02 23:01:25 +00:00
if ( m = = get_declarations_tk ( ) )
tmp . erase_elem ( get_decls_tk ( ) ) ;
2015-02-11 19:02:59 +00:00
}
r . append ( tmp ) ;
} else {
2015-07-07 23:37:06 +00:00
while ( is_next_metaclass_tk ( p ) ) {
2015-02-11 19:02:59 +00:00
r . push_back ( parse_metaclass ( p ) ) ;
}
}
}
2014-06-17 01:42:39 +00:00
static void check_identifier ( parser & p , environment const & env , name const & ns , name const & id ) {
name full_id = ns + id ;
if ( ! env . find ( full_id ) )
2014-09-03 23:00:38 +00:00
throw parser_error ( sstream ( ) < < " invalid 'open' command, unknown declaration ' " < < full_id < < " ' " , p . pos ( ) ) ;
2014-06-17 01:42:39 +00:00
}
2014-09-04 01:37:01 +00:00
// add id as an abbreviation for d
2014-09-04 16:30:25 +00:00
static environment add_abbrev ( parser & p , environment const & env , name const & id , name const & d ) {
2014-09-04 01:37:01 +00:00
declaration decl = env . get ( d ) ;
buffer < level > ls ;
for ( name const & l : decl . get_univ_params ( ) )
ls . push_back ( mk_param_univ ( l ) ) ;
expr value = mk_constant ( d , to_list ( ls . begin ( ) , ls . end ( ) ) ) ;
name const & ns = get_namespace ( env ) ;
name full_id = ns + id ;
2014-09-04 16:30:25 +00:00
p . add_abbrev_index ( full_id , d ) ;
2015-01-20 20:36:56 +00:00
environment new_env =
2015-05-08 22:42:42 +00:00
module : : add ( env , check ( env , mk_definition ( env , full_id , decl . get_univ_params ( ) , decl . get_type ( ) , value ) ) ) ;
2014-09-04 01:37:01 +00:00
if ( full_id ! = id )
new_env = add_expr_alias_rec ( new_env , id , full_id ) ;
return new_env ;
}
// open/export [class] id (as id)? (id ...) (renaming id->id id->id) (hiding id ... id)
environment open_export_cmd ( parser & p , bool open ) {
2014-06-17 01:42:39 +00:00
environment env = p . env ( ) ;
2015-05-13 22:33:48 +00:00
unsigned fingerprint = 0 ;
2014-07-07 21:53:06 +00:00
while ( true ) {
2015-02-11 19:02:59 +00:00
buffer < name > metacls ;
parse_metaclasses ( p , metacls ) ;
bool decls = false ;
if ( metacls . empty ( ) | |
std : : find ( metacls . begin ( ) , metacls . end ( ) , get_decls_tk ( ) ) ! = metacls . end ( ) | |
std : : find ( metacls . begin ( ) , metacls . end ( ) , get_declarations_tk ( ) ) ! = metacls . end ( ) )
decls = true ;
2015-05-13 22:33:48 +00:00
for ( name const & n : metacls )
fingerprint = hash ( fingerprint , n . hash ( ) ) ;
2014-07-08 01:17:10 +00:00
auto pos = p . pos ( ) ;
2014-09-04 01:37:01 +00:00
name ns = p . check_id_next ( " invalid 'open/export' command, identifier expected " ) ;
2014-07-08 01:17:10 +00:00
optional < name > real_ns = to_valid_namespace_name ( env , ns ) ;
if ( ! real_ns )
throw parser_error ( sstream ( ) < < " invalid namespace name ' " < < ns < < " ' " , pos ) ;
ns = * real_ns ;
2015-05-20 18:42:16 +00:00
fingerprint = hash ( fingerprint , ns . hash ( ) ) ;
2014-09-04 00:37:02 +00:00
name as ;
2014-09-23 02:22:53 +00:00
if ( p . curr_is_token_or_id ( get_as_tk ( ) ) ) {
2014-09-04 00:37:02 +00:00
p . next ( ) ;
2014-09-04 01:37:01 +00:00
as = p . check_id_next ( " invalid 'open/export' command, identifier expected " ) ;
2014-09-04 00:37:02 +00:00
}
2014-09-04 01:37:01 +00:00
if ( open )
2015-02-11 19:02:59 +00:00
env = using_namespace ( env , p . ios ( ) , ns , metacls ) ;
2014-09-04 01:37:01 +00:00
else
2015-02-11 19:02:59 +00:00
env = export_namespace ( env , p . ios ( ) , ns , metacls ) ;
2014-07-07 21:53:06 +00:00
if ( decls ) {
// Remark: we currently to not allow renaming and hiding of universe levels
buffer < name > exceptions ;
bool found_explicit = false ;
2014-09-23 02:22:53 +00:00
while ( p . curr_is_token ( get_lparen_tk ( ) ) ) {
2014-06-17 01:42:39 +00:00
p . next ( ) ;
2014-09-23 02:22:53 +00:00
if ( p . curr_is_token_or_id ( get_renaming_tk ( ) ) ) {
2014-06-17 01:42:39 +00:00
p . next ( ) ;
2014-07-07 21:53:06 +00:00
while ( p . curr_is_identifier ( ) ) {
name from_id = p . get_name_val ( ) ;
p . next ( ) ;
2014-09-23 02:22:53 +00:00
p . check_token_next ( get_arrow_tk ( ) , " invalid 'open/export' command renaming, '->' expected " ) ;
2014-09-04 01:37:01 +00:00
name to_id = p . check_id_next ( " invalid 'open/export' command renaming, identifier expected " ) ;
2015-05-13 22:33:48 +00:00
fingerprint = hash ( hash ( fingerprint , from_id . hash ( ) ) , to_id . hash ( ) ) ;
2014-07-07 21:53:06 +00:00
check_identifier ( p , env , ns , from_id ) ;
exceptions . push_back ( from_id ) ;
2014-09-04 01:37:01 +00:00
if ( open )
env = add_expr_alias ( env , as + to_id , ns + from_id ) ;
else
2014-09-04 16:30:25 +00:00
env = add_abbrev ( p , env , as + to_id , ns + from_id ) ;
2014-07-07 21:53:06 +00:00
}
2014-09-23 02:22:53 +00:00
} else if ( p . curr_is_token_or_id ( get_hiding_tk ( ) ) ) {
2014-06-17 01:42:39 +00:00
p . next ( ) ;
2014-07-07 21:53:06 +00:00
while ( p . curr_is_identifier ( ) ) {
name id = p . get_name_val ( ) ;
p . next ( ) ;
check_identifier ( p , env , ns , id ) ;
exceptions . push_back ( id ) ;
2015-05-13 22:33:48 +00:00
fingerprint = hash ( fingerprint , id . hash ( ) ) ;
2014-07-07 21:53:06 +00:00
}
} else if ( p . curr_is_identifier ( ) ) {
found_explicit = true ;
while ( p . curr_is_identifier ( ) ) {
name id = p . get_name_val ( ) ;
p . next ( ) ;
2015-05-13 22:33:48 +00:00
fingerprint = hash ( fingerprint , id . hash ( ) ) ;
2014-07-07 21:53:06 +00:00
check_identifier ( p , env , ns , id ) ;
2014-09-04 01:37:01 +00:00
if ( open )
env = add_expr_alias ( env , as + id , ns + id ) ;
else
2014-09-04 16:30:25 +00:00
env = add_abbrev ( p , env , as + id , ns + id ) ;
2014-07-07 21:53:06 +00:00
}
} else {
2014-10-23 17:24:03 +00:00
throw parser_error ( " invalid 'open/export' command option, "
" identifier, 'hiding' or 'renaming' expected " , p . pos ( ) ) ;
2014-06-17 01:42:39 +00:00
}
2014-07-07 21:53:06 +00:00
if ( found_explicit & & ! exceptions . empty ( ) )
2014-10-23 17:24:03 +00:00
throw parser_error ( " invalid 'open/export' command option, "
" mixing explicit and implicit 'open/export' options " , p . pos ( ) ) ;
2014-09-23 02:22:53 +00:00
p . check_token_next ( get_rparen_tk ( ) , " invalid 'open/export' command option, ')' expected " ) ;
2014-09-04 01:37:01 +00:00
}
if ( ! found_explicit ) {
if ( open ) {
env = add_aliases ( env , ns , as , exceptions . size ( ) , exceptions . data ( ) ) ;
} else {
environment new_env = env ;
env . for_each_declaration ( [ & ] ( declaration const & d ) {
if ( ! is_protected ( env , d . get_name ( ) ) & &
is_prefix_of ( ns , d . get_name ( ) ) & &
! is_exception ( d . get_name ( ) , ns , exceptions . size ( ) , exceptions . data ( ) ) ) {
name new_id = d . get_name ( ) . replace_prefix ( ns , as ) ;
if ( ! new_id . is_anonymous ( ) )
2014-09-04 16:30:25 +00:00
new_env = add_abbrev ( p , new_env , new_id , d . get_name ( ) ) ;
2014-09-04 01:37:01 +00:00
}
} ) ;
env = new_env ;
}
2014-06-17 01:42:39 +00:00
}
}
2015-07-07 23:37:06 +00:00
if ( ! is_next_metaclass_tk ( p ) & & ! p . curr_is_identifier ( ) )
2014-07-07 21:53:06 +00:00
break ;
2014-06-17 01:42:39 +00:00
}
2015-05-13 22:33:48 +00:00
return update_fingerprint ( env , fingerprint ) ;
2014-06-17 01:42:39 +00:00
}
2015-05-20 18:42:16 +00:00
static environment open_cmd ( parser & p ) { return open_export_cmd ( p , true ) ; }
static environment export_cmd ( parser & p ) { return open_export_cmd ( p , false ) ; }
static environment override_cmd ( parser & p ) {
environment env = p . env ( ) ;
while ( p . curr_is_identifier ( ) ) {
auto pos = p . pos ( ) ;
name ns = p . check_id_next ( " invalid 'override' command, identifier expected " ) ;
optional < name > real_ns = to_valid_namespace_name ( env , ns ) ;
if ( ! real_ns )
throw parser_error ( sstream ( ) < < " invalid namespace name ' " < < ns < < " ' " , pos ) ;
ns = * real_ns ;
bool persistent = false ;
env = override_notation ( env , ns , persistent ) ;
2015-06-14 22:48:36 +00:00
env = overwrite_aliases ( env , ns , name ( ) ) ;
2015-05-20 18:42:16 +00:00
env = update_fingerprint ( env , ns . hash ( ) ) ;
}
return env ;
}
2014-06-17 01:42:39 +00:00
2015-05-20 18:42:16 +00:00
static environment erase_cache_cmd ( parser & p ) {
2014-08-11 20:55:29 +00:00
name n = p . check_id_next ( " invalid #erase_cache command, identifier expected " ) ;
p . erase_cached_definition ( n ) ;
return p . env ( ) ;
}
2015-05-20 18:42:16 +00:00
static environment projections_cmd ( parser & p ) {
2014-10-29 19:50:52 +00:00
name n = p . check_id_next ( " invalid #projections command, identifier expected " ) ;
if ( p . curr_is_token ( get_dcolon_tk ( ) ) ) {
p . next ( ) ;
buffer < name > proj_names ;
while ( p . curr_is_identifier ( ) ) {
proj_names . push_back ( n + p . get_name_val ( ) ) ;
p . next ( ) ;
}
return mk_projections ( p . env ( ) , n , proj_names ) ;
} else {
return mk_projections ( p . env ( ) , n ) ;
}
}
2015-05-20 18:42:16 +00:00
static environment telescope_eq_cmd ( parser & p ) {
2014-12-08 02:17:15 +00:00
expr e ; level_param_names ls ;
std : : tie ( e , ls ) = parse_local_expr ( p ) ;
buffer < expr > t ;
while ( is_pi ( e ) ) {
expr local = mk_local ( p . mk_fresh_name ( ) , binding_name ( e ) , binding_domain ( e ) , binder_info ( ) ) ;
t . push_back ( local ) ;
e = instantiate ( binding_body ( e ) , local ) ;
}
2015-05-08 21:36:38 +00:00
auto tc = mk_type_checker ( p . env ( ) , p . mk_ngen ( ) ) ;
2014-12-08 02:17:15 +00:00
buffer < expr > eqs ;
mk_telescopic_eq ( * tc , t , eqs ) ;
for ( expr const & eq : eqs ) {
regular ( p . env ( ) , p . ios ( ) ) < < local_pp_name ( eq ) < < " : " < < mlocal_type ( eq ) < < " \n " ;
tc - > check ( mlocal_type ( eq ) , ls ) ;
}
return p . env ( ) ;
}
2015-05-20 18:42:16 +00:00
static environment local_cmd ( parser & p ) {
2015-01-26 19:31:12 +00:00
if ( p . curr_is_token_or_id ( get_attribute_tk ( ) ) ) {
p . next ( ) ;
return local_attribute_cmd ( p ) ;
2015-02-11 01:31:40 +00:00
} else if ( p . curr_is_token ( get_abbreviation_tk ( ) ) ) {
p . next ( ) ;
return local_abbreviation_cmd ( p ) ;
2015-01-26 19:31:12 +00:00
} else {
return local_notation_cmd ( p ) ;
}
}
2015-03-06 21:03:30 +00:00
static environment help_cmd ( parser & p ) {
flycheck_information info ( p . regular_stream ( ) ) ;
if ( info . enabled ( ) ) {
p . display_information_pos ( p . cmd_pos ( ) ) ;
p . regular_stream ( ) < < " help result: \n " ;
}
if ( p . curr_is_token_or_id ( get_options_tk ( ) ) ) {
p . next ( ) ;
for ( auto odecl : get_option_declarations ( ) ) {
auto opt = odecl . second ;
regular ( p . env ( ) , p . ios ( ) )
< < " " < < opt . get_name ( ) < < " ( " < < opt . kind ( ) < < " ) "
< < opt . get_description ( ) < < " (default: " < < opt . get_default_value ( ) < < " ) " < < endl ;
}
} else if ( p . curr_is_token_or_id ( get_commands_tk ( ) ) ) {
p . next ( ) ;
buffer < name > ns ;
cmd_table const & cmds = p . cmds ( ) ;
cmds . for_each ( [ & ] ( name const & n , cmd_info const & ) {
ns . push_back ( n ) ;
} ) ;
std : : sort ( ns . begin ( ) , ns . end ( ) ) ;
for ( name const & n : ns ) {
regular ( p . env ( ) , p . ios ( ) )
< < " " < < n < < " : " < < cmds . find ( n ) - > get_descr ( ) < < endl ;
} ;
} else {
p . regular_stream ( )
< < " help options : describe available options \n "
< < " help commands : describe available commands \n " ;
}
return p . env ( ) ;
}
2015-05-20 18:42:16 +00:00
static environment init_quotient_cmd ( parser & p ) {
2015-04-23 22:27:56 +00:00
if ( ! ( p . env ( ) . prop_proof_irrel ( ) & & p . env ( ) . impredicative ( ) ) )
throw parser_error ( " invalid init_quotient command, this command is only available for kernels containing an impredicative and proof irrelevant Prop " , p . cmd_pos ( ) ) ;
2015-04-01 04:45:11 +00:00
return module : : declare_quotient ( p . env ( ) ) ;
}
2015-05-20 18:42:16 +00:00
static environment init_hits_cmd ( parser & p ) {
2015-04-23 22:27:56 +00:00
if ( p . env ( ) . prop_proof_irrel ( ) | | p . env ( ) . impredicative ( ) )
throw parser_error ( " invalid init_hits command, this command is only available for proof relevant and predicative kernels " , p . cmd_pos ( ) ) ;
return module : : declare_hits ( p . env ( ) ) ;
}
2015-09-11 17:49:07 +00:00
static environment compile_cmd ( parser & p ) {
name n = p . check_constant_next ( " invalid #compile command, constant expected " ) ;
declaration d = p . env ( ) . get ( n ) ;
2015-09-12 00:12:32 +00:00
buffer < name > aux_decls ;
preprocess_rec ( p . env ( ) , d , aux_decls ) ;
2015-09-11 17:49:07 +00:00
return p . env ( ) ;
}
2015-10-01 01:50:28 +00:00
static environment accessible_cmd ( parser & p ) {
environment const & env = p . env ( ) ;
unsigned total = 0 ;
unsigned accessible = 0 ;
unsigned accessible_theorems = 0 ;
env . for_each_declaration ( [ & ] ( declaration const & d ) {
name const & n = d . get_name ( ) ;
total + + ;
if ( ( d . is_theorem ( ) | | d . is_definition ( ) ) & &
! is_instance ( env , n ) & & ! is_simp_rule ( env , n ) & & ! is_congr_rule ( env , n ) & &
! is_user_defined_recursor ( env , n ) & & ! is_aux_recursor ( env , n ) & &
! is_projection ( env , n ) & & ! is_private ( env , n ) & &
! is_user_defined_recursor ( env , n ) & & ! is_aux_recursor ( env , n ) & &
! is_subst_relation ( env , n ) & & ! is_trans_relation ( env , n ) & &
! is_symm_relation ( env , n ) & & ! is_refl_relation ( env , n ) ) {
accessible + + ;
if ( d . is_theorem ( ) )
accessible_theorems + + ;
}
} ) ;
p . regular_stream ( ) < < " total: " < < total < < " , accessible: " < < accessible < < " , accessible theorems: " < < accessible_theorems < < " \n " ;
return env ;
}
2015-10-15 01:36:41 +00:00
static void display_name_set ( parser & p , name const & n , name_set const & s ) {
if ( s . empty ( ) )
return ;
io_state_stream out = p . regular_stream ( ) ;
out < < " " < < n < < " := { " ;
bool first = true ;
s . for_each ( [ & ] ( name const & n2 ) {
if ( is_private ( p . env ( ) , n2 ) )
return ;
if ( first )
first = false ;
else
out < < " , " ;
out < < n2 ;
} ) ;
out < < " } \n " ;
}
static environment decl_stats_cmd ( parser & p ) {
environment const & env = p . env ( ) ;
2015-10-15 02:34:53 +00:00
io_state_stream out = p . regular_stream ( ) ;
out < < " Use sets \n " ;
2015-10-15 01:36:41 +00:00
env . for_each_declaration ( [ & ] ( declaration const & d ) {
if ( ( d . is_theorem ( ) | | d . is_axiom ( ) ) & & ! is_private ( env , d . get_name ( ) ) )
display_name_set ( p , d . get_name ( ) , get_use_set ( env , d . get_name ( ) ) ) ;
} ) ;
2015-10-15 02:34:53 +00:00
out < < " Used-by sets \n " ;
2015-10-15 01:36:41 +00:00
env . for_each_declaration ( [ & ] ( declaration const & d ) {
if ( ! d . is_theorem ( ) & & ! d . is_axiom ( ) & & ! is_private ( env , d . get_name ( ) ) )
display_name_set ( p , d . get_name ( ) , get_used_by_set ( env , d . get_name ( ) ) ) ;
} ) ;
return env ;
}
2015-10-15 02:34:53 +00:00
static environment relevant_thms_cmd ( parser & p ) {
environment const & env = p . env ( ) ;
name_set R ;
while ( p . curr_is_identifier ( ) ) {
R . insert ( p . check_constant_next ( " invalid #relevant_thms command, constant expected " ) ) ;
}
name_set TS = get_relevant_thms ( env , p . get_options ( ) , R ) ;
io_state_stream out = p . regular_stream ( ) ;
TS . for_each ( [ & ] ( name const & T ) {
2015-10-15 03:46:03 +00:00
out < < T < < " \n " ;
2015-10-15 02:34:53 +00:00
} ) ;
return env ;
}
2015-11-01 17:59:37 +00:00
static environment app_builder_cmd ( parser & p ) {
environment const & env = p . env ( ) ;
type_checker tc ( env ) ;
app_builder b ( tc ) ;
name c = p . check_constant_next ( " invalid #app_builder command, constant expected " ) ;
p . check_token_next ( get_lparen_tk ( ) , " invalid #app_builder command, '(' expected " ) ;
buffer < expr > args ;
while ( true ) {
args . push_back ( p . parse_expr ( ) ) ;
if ( ! p . curr_is_token ( get_comma_tk ( ) ) )
break ;
p . next ( ) ;
}
p . check_token_next ( get_rparen_tk ( ) , " invalid #app_builder command, ')' expected " ) ;
if ( auto r = b . mk_app ( c , args . size ( ) , args . data ( ) ) ) {
p . regular_stream ( ) < < * r < < " \n " ;
} else {
p . regular_stream ( ) < < " <failed> \n " ;
}
return env ;
}
2014-09-23 17:00:36 +00:00
void init_cmd_table ( cmd_table & r ) {
2015-10-15 02:34:53 +00:00
add_cmd ( r , cmd_info ( " open " , " create aliases for declarations, and use objects defined in other namespaces " ,
2014-10-29 19:50:52 +00:00
open_cmd ) ) ;
2015-10-15 02:34:53 +00:00
add_cmd ( r , cmd_info ( " export " , " create abbreviations for declarations, "
2014-10-29 19:50:52 +00:00
" and export objects defined in other namespaces " , export_cmd ) ) ;
2015-10-15 02:34:53 +00:00
add_cmd ( r , cmd_info ( " override " , " override notation declarations using the ones defined in the given namespace " ,
2015-05-20 18:42:16 +00:00
override_cmd ) ) ;
2015-10-15 02:34:53 +00:00
add_cmd ( r , cmd_info ( " set_option " , " set configuration option " , set_option_cmd ) ) ;
add_cmd ( r , cmd_info ( " exit " , " exit " , exit_cmd ) ) ;
add_cmd ( r , cmd_info ( " print " , " print a string " , print_cmd ) ) ;
add_cmd ( r , cmd_info ( " section " , " open a new section " , section_cmd ) ) ;
add_cmd ( r , cmd_info ( " namespace " , " open a new namespace " , namespace_cmd ) ) ;
add_cmd ( r , cmd_info ( " end " , " close the current namespace/section " , end_scoped_cmd ) ) ;
add_cmd ( r , cmd_info ( " check " , " type check given expression, and display its type " , check_cmd ) ) ;
add_cmd ( r , cmd_info ( " eval " , " evaluate given expression " , eval_cmd ) ) ;
add_cmd ( r , cmd_info ( " find_decl " , " find definitions and/or theorems " , find_cmd ) ) ;
add_cmd ( r , cmd_info ( " local " , " define local attributes or notation " , local_cmd ) ) ;
add_cmd ( r , cmd_info ( " help " , " brief description of available commands and options " , help_cmd ) ) ;
add_cmd ( r , cmd_info ( " init_quotient " , " initialize quotient type computational rules " , init_quotient_cmd ) ) ;
add_cmd ( r , cmd_info ( " init_hits " , " initialize builtin HITs " , init_hits_cmd ) ) ;
add_cmd ( r , cmd_info ( " #erase_cache " , " erase cached definition (for debugging purposes) " , erase_cache_cmd ) ) ;
add_cmd ( r , cmd_info ( " #projections " , " generate projections for inductive datatype (for debugging purposes) " , projections_cmd ) ) ;
add_cmd ( r , cmd_info ( " #telescope_eq " , " (for debugging purposes) " , telescope_eq_cmd ) ) ;
2015-11-01 17:59:37 +00:00
add_cmd ( r , cmd_info ( " #app_builder " , " (for debugging purposes) " , app_builder_cmd ) ) ;
2015-10-15 02:34:53 +00:00
add_cmd ( r , cmd_info ( " #compile " , " (for debugging purposes) " , compile_cmd ) ) ;
add_cmd ( r , cmd_info ( " #accessible " , " (for debugging purposes) display number of accessible declarations for blast tactic " , accessible_cmd ) ) ;
add_cmd ( r , cmd_info ( " #decl_stats " , " (for debugging purposes) display declaration statistics " , decl_stats_cmd ) ) ;
add_cmd ( r , cmd_info ( " #relevant_thms " , " (for debugging purposes) select relevant theorems using Meng&Paulson heuristic " , relevant_thms_cmd ) ) ;
2015-10-15 01:36:41 +00:00
2014-06-18 17:36:21 +00:00
register_decl_cmds ( r ) ;
2014-06-18 20:55:48 +00:00
register_inductive_cmd ( r ) ;
2014-07-29 02:04:57 +00:00
register_structure_cmd ( r ) ;
2015-03-15 03:32:39 +00:00
register_migrate_cmd ( r ) ;
2014-06-17 20:39:44 +00:00
register_notation_cmds ( r ) ;
2014-08-21 17:36:44 +00:00
register_begin_end_cmds ( r ) ;
2014-07-08 21:28:33 +00:00
register_tactic_hint_cmd ( r ) ;
2014-06-11 18:52:58 +00:00
}
2014-09-23 17:00:36 +00:00
static cmd_table * g_cmds = nullptr ;
2014-06-11 00:02:06 +00:00
cmd_table get_builtin_cmds ( ) {
2014-09-23 17:00:36 +00:00
return * g_cmds ;
}
void initialize_builtin_cmds ( ) {
g_cmds = new cmd_table ( ) ;
init_cmd_table ( * g_cmds ) ;
}
void finalize_builtin_cmds ( ) {
delete g_cmds ;
2014-06-11 00:02:06 +00:00
}
}