/* See license.txt in the root of this project. */ # ifndef LMT_CONDITIONAL_H # define LMT_CONDITIONAL_H /*tex The next list should be in sync with |if_branch_mapping| at the top of the |c| file with the same name. The next ones also go on the condition stack so we need to retain this order and organization. There is a catch here: the codes of the |if_test_cmd|, |fi_or_else_cmd| and |or_else_cmd| are all in this enumeration. This has to do with the history of not always checking for the cmd code in the fast skipping branches. We could change that but not now. Well, in the end I combined |if_test_cmd|, |fi_or_else_cmd| and |or_else_cmd| because they use the same chr range anyway and it also simplifies some of the testing (especially after some more robust cmd/chr checking was added, and after that the |fi_or_else_cmd| and |or_else_cmd| were combined. The main motivation is that we can have a more consistent \LUA\ token interface end. It is debatable as we divert from the original, but we already did that by introducing more conditionals, |\orelse| and the generic |\ifconditional| that also demandeed all kind of adaptations. Sorry. The comments are mostly the same, including references to the older cmd codes (pre 2.07 there used to be some switch/case statements in places but these were flattened). Btw, the |\unless| prefix is kept out of this because it relates to expansion and prefixes are separate anyway. It would make the code less pretty. One reason for a split in cmd codes is performance but we didn't loose on the change. */ typedef enum if_test_codes { /*tex These are private chr codes: */ no_if_code, /*tex We're not in a condition. */ if_code, /*tex We have a condition. */ /*tex These are public chr codes: */ fi_code, /*tex |\fi| */ else_code, /*tex |\else| */ or_code, /*tex |\or| */ or_else_code, /*tex |\orelse| */ or_unless_code, /*tex |\orunless| */ /*tex Here come the \if... codes. Some are just there to minimize tracing and are not faster, like |\ifzerodim| (we can use |\ifcase| instead but not with |\unless|). For historic reasons we see a mixture if |int| and |num| here but I see no reason the change that now. */ if_char_code, /*tex |\if| */ if_cat_code, /*tex |\ifcat| */ if_int_code, /*tex |\ifnum| */ if_abs_int_code, /*tex |\ifabsnum| */ if_zero_int_code, /*tex |\ifzeronum|*/ if_interval_int_code, if_posit_code, if_abs_posit_code, if_zero_posit_code, if_interval_posit_code, if_dim_code, /*tex |\ifdim| */ if_abs_dim_code, /*tex |\ifabsdim| */ if_zero_dim_code, /*tex |\ifzerodim| */ if_interval_dim_code, if_odd_code, /*tex |\ifodd| */ if_vmode_code, /*tex |\ifvmode| */ if_hmode_code, /*tex |\ifhmode| */ if_mmode_code, /*tex |\ifmmode| */ if_inner_code, /*tex |\ifinner| */ if_void_code, /*tex |\ifvoid| */ if_hbox_code, /*tex |\ifhbox| */ if_vbox_code, /*tex |\ifvbox| */ if_tok_code, /*tex |\iftok| */ if_cstok_code, /*tex |\ifcstok| */ if_x_code, /*tex |\ifx| */ if_true_code, /*tex |\iftrue| */ if_false_code, /*tex |\iffalse| */ if_chk_int_code, /*tex |\ifchknum| */ if_chk_integer_code, /*tex |\ifchknumber| */ if_chk_intexpr_code, /*tex |\ifnumexpr| */ if_val_int_code, /*tex |\ifnumval| */ if_cmp_int_code, /*tex |\ifcmpnum| */ if_chk_dim_code, /*tex |\ifchkdim| */ if_chk_dimension_code, /*tex |\ifchkdimension| */ if_chk_dimexpr_code, /*tex |\ifdimexpr| */ if_val_dim_code, /*tex |\ifdimval| */ if_cmp_dim_code, /*tex |\ifcmpdim| */ if_case_code, /*tex |\ifcase| */ if_defined_code, /*tex |\ifdefined| */ if_csname_code, /*tex |\ifcsname| */ if_in_csname_code, /*tex |\ifincsname| */ if_font_char_code, /*tex |\iffontchar| */ if_condition_code, /*tex |\ifcondition| */ if_flags_code, /*tex |\ifflags| */ if_empty_code, /*tex |\ifempty| */ if_relax_code, /*tex |\ifrelax| */ if_boolean_code, /*tex |\ifboolean| */ if_numexpression_code, /*tex |\ifnumexpression| */ if_dimexpression_code, /*tex |\ifdimexpression| */ if_last_named_cs_code, /*tex |\iflastnamedcs| */ if_math_parameter_code, /*tex |\ifmathparameter| */ if_math_style_code, /*tex |\ifmathstyle| */ if_arguments_code, /*tex |\ifarguments| */ if_parameters_code, /*tex |\ifparameters| */ if_parameter_code, /*tex |\ifparameter| */ if_has_tok_code, /*tex |\ifhastok| */ if_has_toks_code, /*tex |\ifhastoks| */ if_has_xtoks_code, /*tex |\ifhasxtoks| */ if_has_char_code, /*tex |\ifhaschar| */ if_insert_code, /*tex |\ifinsert| */ if_in_alignment_code, /*tex |\ifinalignment| */ if_cramped_code, /*tex |\ifcrampedmathstyle| */ if_list_code, /*tex |\iflist| */ // if_bitwise_and_code, /*tex |\ifbitwiseand| */ } if_test_codes; # define first_if_test_code fi_code # define last_if_test_code if_list_code //define last_if_test_code if_bitwise_and_code # define first_real_if_test_code if_char_code # define last_real_if_test_code if_list_code //define last_real_if_test_code if_bitwise_and_code typedef struct condition_state_info { halfword cond_ptr; /*tex top of the condition stack */ quarterword cur_if; /*tex type of conditional being worked on */ quarterword if_limit; /*tex upper bound on |fi_or_else| codes */ singleword cur_unless; singleword if_unless; singleword if_step; singleword unused; halfword if_line; /*tex line where that conditional began */ halfword if_nesting; halfword skip_line; /*tex skipping began here */ halfword chk_integer; scaled chk_dimension; } condition_state_info ; extern condition_state_info lmt_condition_state; extern void tex_conditional_if (halfword code, int unless); extern void tex_conditional_fi_or_else (void); extern void tex_conditional_unless (void); extern void tex_show_ifs (void); extern void tex_conditional_catch_up (void); /* void tex_quit_fi (void); */ /* void tex_conditional_after_fi (void); */ # endif