Subversion Repositories pentevo

Rev

Blame | Last modification | View Log | Download | RSS feed

  1. /* texutil.c */
  2. /*****************************************************************************/
  3. /* SPDX-License-Identifier: GPL-2.0-only OR GPL-3.0-only                     */
  4. /*                                                                           */
  5. /* AS                                                                        */
  6. /*                                                                           */
  7. /* TeX-->ASCII/HTML Converter: Common Utils/Variables                        */
  8. /*                                                                           */
  9. /*****************************************************************************/
  10.  
  11. #include <stdio.h>
  12. #include <stdlib.h>
  13. #include <string.h>
  14.  
  15. #include "strutil.h"
  16. #include "texutil.h"
  17.  
  18. #define DBG_IF 0
  19.  
  20. /*--------------------------------------------------------------------------*/
  21.  
  22. Boolean DoRepass;
  23.  
  24. tex_infile_t *p_curr_tex_infile = NULL;
  25. char buffer_line[1001] = "", *p_buffer_line_ptr = buffer_line, buffer_last_char = '\0';
  26.  
  27. const char *p_infile_name, *p_outfile_name;
  28. FILE *p_outfile = NULL;
  29.  
  30. typedef struct tex_pushed_token
  31. {
  32.   char token[TOKLEN], sep[TOKLEN];
  33. } tex_pushed_token_t;
  34. static int pushed_token_cnt;
  35. static tex_pushed_token_t pushed_tokens[16];
  36. static char save_sep;
  37. char tex_token_sep_string[TOKLEN],
  38.      tex_backslash_token_sep_string[TOKLEN];
  39. static Boolean did_eof;
  40.  
  41. static tex_counter_t *tex_counters = NULL;
  42. static Word next_counter_ref_num = 0;
  43.  
  44. static tex_newif_t *p_tex_newifs = NULL;
  45.  
  46. static tex_environment_t *tex_environments = NULL;
  47.  
  48. static tex_newcommand_t *tex_newcommands = NULL;
  49.  
  50. tex_env_data_t curr_tex_env_data;
  51. EnvType curr_tex_env;
  52. char *p_curr_tex_user_env_name;
  53. tex_env_save_t *p_env_stack;
  54.  
  55. const char *tex_env_names[EnvCount] =
  56. {
  57.   "___NONE___", "document", "itemize", "enumerate", "description", "table", "tabular",
  58.   "raggedleft", "raggedright", "center", "verbatim", "quote", "tabbing",
  59.   "thebibliography", "___MARGINPAR___", "___CAPTION___", "___HEADING___", "___USER___"
  60. };
  61.  
  62. tex_output_consumer_t *p_current_tex_output_consumer;
  63.  
  64. static tex_if_stack_t *p_if_stack;
  65.  
  66. /*--------------------------------------------------------------------------*/
  67.  
  68. /*!------------------------------------------------------------------------
  69.  * \fn     tex_infile_free(tex_infile_t *p_infile)
  70.  * \brief  free/destroy infile source
  71.  * \param  p_infile source to free
  72.  * ------------------------------------------------------------------------ */
  73.  
  74. static void tex_infile_free(tex_infile_t *p_infile)
  75. {
  76.   if (p_infile->p_name)
  77.   {
  78.     free(p_infile->p_name);
  79.     p_infile->p_name = NULL;
  80.   }
  81.   if (p_infile->p_file)
  82.   {
  83.     fclose(p_infile->p_file);
  84.     p_infile->p_file = NULL;
  85.   }
  86.   if (p_infile->p_buf_line_to_free)
  87.   {
  88.     free(p_infile->p_buf_line_to_free);
  89.     p_infile->p_buf_line_to_free = NULL;
  90.   }
  91.   if (p_infile->p_saved_buffer_line)
  92.   {
  93.     free(p_infile->p_saved_buffer_line);
  94.     p_infile->p_saved_buffer_line = NULL;
  95.   }
  96.   free(p_infile);
  97. }
  98.  
  99. /*!------------------------------------------------------------------------
  100.  * \fn     tex_infile_pop(void)
  101.  * \brief  discard topmost infile source
  102.  * ------------------------------------------------------------------------ */
  103.  
  104. void tex_infile_pop(void)
  105. {
  106.   tex_infile_t *p_old = p_curr_tex_infile;
  107.   if (!p_old)
  108.     return;
  109.   p_curr_tex_infile = p_old->p_next;
  110.   strmaxcpy(buffer_line, p_old->p_saved_buffer_line, sizeof(buffer_line));
  111.   p_buffer_line_ptr = buffer_line + p_old->saved_buffer_line_ptr;
  112.   buffer_last_char = p_old->save_last_char;
  113.   tex_infile_free(p_old);
  114. }
  115.  
  116. /*!------------------------------------------------------------------------
  117.  * \fn     tex_infile_pop_all(void)
  118.  * \brief  discard all infile sources
  119.  * ------------------------------------------------------------------------ */
  120.  
  121. void tex_infile_pop_all(void)
  122. {
  123.   while (p_curr_tex_infile)
  124.     tex_infile_pop();
  125. }
  126.  
  127. /*!------------------------------------------------------------------------
  128.  * \fn     tex_infile_ontop(tex_infile_t *p_new)
  129.  * \brief  put new infile source on top of stack
  130.  * \param  p_new source to become top
  131.  * ------------------------------------------------------------------------ */
  132.  
  133. static void tex_infile_ontop(tex_infile_t *p_new)
  134. {
  135.   p_new->p_saved_buffer_line = as_strdup(buffer_line);
  136.   p_new->saved_buffer_line_ptr = p_buffer_line_ptr - buffer_line;
  137.   p_new->save_last_char = buffer_last_char;
  138.  
  139.   p_new->p_next = p_curr_tex_infile;
  140.   p_curr_tex_infile = p_new;
  141.  
  142.   *buffer_line = '\0'; p_buffer_line_ptr = buffer_line;
  143.   buffer_last_char = '\0';
  144. }
  145.  
  146. /*!------------------------------------------------------------------------
  147.  * \fn     tex_infile_push_file(const char *p_name)
  148.  * \brief  put another source generator on top - data from file
  149.  * \param  p_name name of file to open
  150.  * ------------------------------------------------------------------------ */
  151.  
  152. void tex_infile_push_file(const char *p_name)
  153. {
  154.   tex_infile_t *p_new = (tex_infile_t*)calloc(1, sizeof(*p_new));
  155.  
  156.   p_new->p_name = as_strdup(p_name);
  157.   p_new->p_file = fopen(p_name, "r");
  158.   if (!p_new->p_file)
  159.   {
  160.     char msg[200];
  161.  
  162.     tex_infile_free(p_new);
  163.     as_snprintf(msg, sizeof(msg), "file %s not found", p_name);
  164.     tex_error(msg);
  165.   }
  166.   else
  167.     tex_infile_ontop(p_new);
  168. }
  169.  
  170. /*!------------------------------------------------------------------------
  171.  * \fn     tex_infile_push_line(const char *p_name, const char *p_line, Boolean line_dynamic)
  172.  * \brief  put another source generator on top - data from memory buffer
  173.  * \param  p_name command name this body is from
  174.  * \param  p_line sourc line
  175.  * \param  line_dynamic free line after consumption?
  176.  * ------------------------------------------------------------------------ */
  177.  
  178. void tex_infile_push_line(const char *p_name, const char *p_line, Boolean line_dynamic)
  179. {
  180.   tex_infile_t *p_new = (tex_infile_t*)calloc(1, sizeof(*p_new));
  181.  
  182.   p_new->p_name = as_strdup(p_name);
  183.   p_new->p_file = NULL;
  184.   p_new->p_buf_line = p_line;
  185.   p_new->p_buf_line_to_free = line_dynamic ? (char*)p_line : NULL;
  186.  
  187.   tex_infile_ontop(p_new);
  188. }
  189.  
  190. /*!------------------------------------------------------------------------
  191.  * \fn     tex_infile_gets(char *p_dest, size_t dest_cap, tex_infile_t *p_infile)
  192.  * \brief  read one line of source from infile source
  193.  * \param  p_dest dest buffer
  194.  * \param  dest_cap capacity of dest buffer
  195.  * \param  p_infile source to read from
  196.  * \return * to p_dest or NULL if nothing could be read (EOF)
  197.  * ------------------------------------------------------------------------ */
  198.  
  199. char *tex_infile_gets(char *p_dest, size_t dest_cap, tex_infile_t *p_infile)
  200. {
  201.   if (p_infile->p_file)
  202.     return fgets(p_dest, dest_cap, p_infile->p_file);
  203.   else
  204.   {
  205.     char *p_pos;
  206.     size_t src_len, dest_len;
  207.  
  208.     if (!p_infile->p_buf_line || !*p_infile->p_buf_line)
  209.       return NULL;
  210.     p_pos = strchr(p_infile->p_buf_line, '\n');
  211.     src_len = p_pos ? (size_t)(p_pos - p_infile->p_buf_line) : strlen(p_infile->p_buf_line);
  212.     if (dest_cap <= src_len)
  213.     {
  214.       fprintf(stderr, "warning: increase size of buffer line to at least %u\n", (unsigned)(src_len + 1));
  215.       dest_len = dest_cap - 1;
  216.     }
  217.     else
  218.       dest_len = src_len;
  219.     memcpy(p_dest, p_infile->p_buf_line, dest_len); p_dest[dest_len] = '\0';
  220.     p_infile->p_buf_line += src_len;
  221.     return p_dest;
  222.   }
  223. }
  224.  
  225. /*!------------------------------------------------------------------------
  226.  * \fn     tex_get_char(void)
  227.  * \brief  read next character from current input stream
  228.  * \return character or EOF
  229.  * ------------------------------------------------------------------------ */
  230.  
  231. int tex_get_char(void)
  232. {
  233.   Boolean Comment;
  234.   static Boolean DidPar = False;
  235.   char *Result;
  236.  
  237. again:
  238.   if (buffer_last_char)
  239.   {
  240.     int ret = buffer_last_char;
  241.     buffer_last_char = '\0';
  242.     return ret;
  243.   }
  244.  
  245.   if (*p_buffer_line_ptr == '\0')
  246.   {
  247.     do
  248.     {
  249.       if (!p_curr_tex_infile)
  250.         return EOF;
  251.       do
  252.       {
  253.         Result = tex_infile_gets(buffer_line, sizeof(buffer_line), p_curr_tex_infile);
  254.         if (Result)
  255.         {
  256.           p_curr_tex_infile->curr_line++;
  257.           break;
  258.         }
  259.         tex_infile_pop();
  260.         if (!p_curr_tex_infile)
  261.           return EOF;
  262.         if (buffer_last_char || *p_buffer_line_ptr)
  263.           goto again;
  264.       }
  265.       while (True);
  266.       p_buffer_line_ptr = buffer_line;
  267.       Comment = (strlen(buffer_line) >= 2) && (!strncmp(buffer_line, "%%", 2));
  268.       if ((*buffer_line == '\0') || (*buffer_line == '\n'))
  269.       {
  270.         if ((curr_tex_env == EnvDocument) && (!DidPar))
  271.         {
  272.           strcpy(buffer_line, "\\par\n");
  273.           DidPar = True;
  274.           Comment = False;
  275.         }
  276.       }
  277.       else if (!Comment)
  278.         DidPar = False;
  279.     }
  280.     while (Comment);
  281.   }
  282.   return *(p_buffer_line_ptr++);
  283. }
  284.  
  285. /*!------------------------------------------------------------------------
  286.  * \fn     tex_issep(char inp)
  287.  * \brief  white space delimiter
  288.  * \param  inp character to test
  289.  * \return true if yes
  290.  * ------------------------------------------------------------------------ */
  291.  
  292. Boolean tex_issep(char inp)
  293. {
  294.   return ((inp == ' ') || (inp == '\t') || (inp == '\n'));
  295. }
  296.  
  297. /*!------------------------------------------------------------------------
  298.  * \fn     tex_isalphanum(char inp)
  299.  * \brief  character of identifier?
  300.  * \param  inp character to test
  301.  * \return true if yes
  302.  * ------------------------------------------------------------------------ */
  303.  
  304. Boolean tex_isalphanum(char inp)
  305. {
  306.   return ((inp >= 'A') && (inp <= 'Z'))
  307.       || ((inp >= 'a') && (inp <= 'z'))
  308.       || ((inp >= '0') && (inp <= '9'))
  309.       || (inp == '.');
  310. }
  311.  
  312. /*!------------------------------------------------------------------------
  313.  * \fn     tex_read_token(char *p_dest)
  314.  * \brief  read next token from input stream
  315.  * \param  p_dest where to put (capacity of TOKLEN)
  316.  * \return True if success
  317.  * ------------------------------------------------------------------------ */
  318.  
  319. Boolean tex_read_token(char *p_dest)
  320. {
  321.   int ch, z;
  322.   Boolean good;
  323.   char *p_run;
  324.  
  325.   if (pushed_token_cnt > 0)
  326.   {
  327.     strcpy(p_dest, pushed_tokens[0].token);
  328.     strcpy(tex_token_sep_string, pushed_tokens[0].sep);
  329.     for (z = 0; z < pushed_token_cnt - 1; z++)
  330.       pushed_tokens[z] = pushed_tokens[z + 1];
  331.     pushed_token_cnt--;
  332.     return True;
  333.   }
  334.  
  335.   if (did_eof)
  336.     return FALSE;
  337.  
  338.   p_curr_tex_infile->curr_column = p_buffer_line_ptr - buffer_line + 1;
  339.  
  340.   /* fuehrende Blanks ueberspringen */
  341.  
  342.   *p_dest = '\0';
  343.   *tex_token_sep_string = save_sep;
  344.   p_run = tex_token_sep_string + ((save_sep == '\0') ? 0 : 1);
  345.   do
  346.   {
  347.     ch = tex_get_char();
  348.     if (ch == '\r')
  349.       ch = tex_get_char();
  350.     if (tex_issep(ch))
  351.       *(p_run++) = ' ';
  352.   }
  353.   while ((tex_issep(ch)) && (ch != EOF));
  354.   *p_run = '\0';
  355.   if (ch == EOF)
  356.   {
  357.     did_eof = TRUE;
  358.     return FALSE;
  359.   }
  360.  
  361.   /* jetzt Zeichen kopieren, bis Leerzeichen */
  362.  
  363.   p_run = p_dest;
  364.   save_sep = '\0';
  365.   if (tex_isalphanum(*(p_run++) = ch))
  366.   {
  367.     do
  368.     {
  369.       ch = tex_get_char();
  370.       good = (!tex_issep(ch)) && (tex_isalphanum(ch)) && (ch != EOF);
  371.       if (good)
  372.         *(p_run++) = ch;
  373.     }
  374.     while (good);
  375.  
  376.     /* Dateiende ? */
  377.  
  378.     if (ch == EOF)
  379.       did_eof = TRUE;
  380.  
  381.     /* Zeichen speichern ? */
  382.  
  383.     else if (!tex_issep(ch) && !tex_isalphanum(ch))
  384.       buffer_last_char = ch;
  385.  
  386.     /* Separator speichern ? */
  387.  
  388.     else if (tex_issep(ch))
  389.       save_sep = ' ';
  390.   }
  391.  
  392.   /* Ende */
  393.  
  394.   *p_run = '\0';
  395.   return True;
  396. }
  397.  
  398. /*!------------------------------------------------------------------------
  399.  * \fn     tex_push_back_token(char *p_token)
  400.  * \brief  push back token that was read too much
  401.  * \param  p_token token to put back
  402.  * ------------------------------------------------------------------------ */
  403.  
  404. void tex_push_back_token(char *p_token)
  405. {
  406.   if (pushed_token_cnt >= 16)
  407.     return;
  408.   strcpy(pushed_tokens[pushed_token_cnt].token, p_token);
  409.   strcpy(pushed_tokens[pushed_token_cnt].sep, tex_token_sep_string);
  410.   pushed_token_cnt++;
  411. }
  412.  
  413. /*!------------------------------------------------------------------------
  414.  * \fn     tex_token_reset(void)
  415.  * \brief  reset/initialize token read state machine
  416.  * ------------------------------------------------------------------------ */
  417.  
  418. void tex_token_reset(void)
  419. {
  420.   pushed_token_cnt = 0;
  421.   save_sep = '\0';
  422.   tex_token_sep_string[0] =
  423.   tex_backslash_token_sep_string[0] = '\0';
  424.   did_eof = False;
  425. }
  426.  
  427. /*!------------------------------------------------------------------------
  428.  * \fn     tex_assert_token(const char *p_ref)
  429.  * \brief  assure next token is as given
  430.  * \param  ref token to check for
  431.  * ------------------------------------------------------------------------ */
  432.  
  433. void tex_assert_token(const char *p_ref)
  434. {
  435.   char token[TOKLEN];
  436.  
  437.   tex_read_token(token);
  438.   if (strcmp(p_ref, token))
  439.     tex_error("\"%s\" expected, but got \"%s\"", p_ref, token);
  440. }
  441.  
  442. /*!------------------------------------------------------------------------
  443.  * \fn     tex_collect_token(char *p_dest, const char *p_term)
  444.  * \brief  collect tokens up to terminator
  445.  * \param  p_dest dest buffer of length TOKLEN
  446.  * \param  p_term terminator
  447.  * ------------------------------------------------------------------------ */
  448.  
  449. void tex_collect_token(char *p_dest, const char *p_term)
  450. {
  451.   char comp[TOKLEN];
  452.   Boolean first = True, done;
  453.  
  454.   *p_dest = '\0';
  455.   do
  456.   {
  457.     tex_read_token(comp);
  458.     done = !strcmp(comp, p_term);
  459.     if (!done)
  460.     {
  461.       if (!first)
  462.         strcat(p_dest, tex_token_sep_string);
  463.       strcat(p_dest, comp);
  464.     }
  465.     first = False;
  466.   }
  467.   while (!done);
  468. }
  469.  
  470. /*!------------------------------------------------------------------------
  471.  * \fn     collect_arg_tokens(char *p_dest, size_t dest_cap)
  472.  * \brief  collect tokens up to }
  473.  * \param  p_dest dest buffer
  474.  * \param  dest_cap capacity of dest buffer
  475.  * ------------------------------------------------------------------------ */
  476.  
  477. static void collect_arg_tokens(char *p_dest, size_t dest_cap)
  478. {
  479.   int level = 1;
  480.   char token[TOKLEN];
  481.   Boolean first = True;
  482.  
  483.   *p_dest = '\0';
  484.   do
  485.   {
  486.     tex_read_token(token);
  487.     if (!strcmp(token, "{"))
  488.       level++;
  489.     else if (!strcmp(token, "}"))
  490.       level--;
  491.     if (level != 0)
  492.     {
  493.       if (!first)
  494.         strmaxcat(p_dest, tex_token_sep_string, dest_cap);
  495.       strmaxcat(p_dest, token, dest_cap);
  496.       first = False;
  497.     }
  498.   }
  499.   while (level != 0);
  500. }
  501.  
  502. /*!------------------------------------------------------------------------
  503.  * \fn     tex_warning(const char *p_msg, ...)
  504.  * \brief  print warning
  505.  * \param  p_msg warning message
  506.  * ------------------------------------------------------------------------ */
  507.  
  508. void tex_vwarning(const char *p_msg, va_list ap)
  509. {
  510.   fprintf(stderr, "%s:%d.%d: ",
  511.           p_curr_tex_infile ? p_curr_tex_infile->p_name : "<internal>",
  512.           p_curr_tex_infile ? p_curr_tex_infile->curr_line : 0,
  513.           p_curr_tex_infile ? p_curr_tex_infile->curr_column : 0);
  514.   vfprintf(stderr, p_msg, ap);
  515.   fprintf(stderr, "\n");
  516. }
  517.  
  518. void tex_warning(const char *p_msg, ...)
  519. {
  520.   va_list ap;
  521.  
  522.   va_start(ap, p_msg);
  523.   tex_vwarning(p_msg, ap);
  524.   va_end(ap);
  525. }
  526.  
  527. /*!------------------------------------------------------------------------
  528.  * \fn     tex_error(const char *p_msg, ...)
  529.  * \brief  print warning
  530.  * \param  pMsg error message & abort
  531.  * ------------------------------------------------------------------------ */
  532.  
  533. void tex_verror(const char *p_msg, va_list ap)
  534. {
  535.   tex_vwarning(p_msg, ap);
  536.   tex_infile_pop_all();
  537.   if (p_outfile)
  538.     fclose(p_outfile);
  539.   exit(2);
  540. }
  541.  
  542. void tex_error(const char *p_msg, ...)
  543. {
  544.   va_list ap;
  545.  
  546.   va_start(ap, p_msg);
  547.   tex_verror(p_msg, ap);
  548.   va_end(ap);
  549. }
  550.  
  551. /*!------------------------------------------------------------------------
  552.  * \fn     tex_counter_new(const char *p_name)
  553.  * \brief  create counter structure
  554.  * \param  p_name name of counter
  555.  * \return * to created counter or NULL
  556.  * ------------------------------------------------------------------------ */
  557.  
  558. static tex_counter_t *tex_counter_new(const char *p_name)
  559. {
  560.   tex_counter_t *p_ret = (tex_counter_t*)calloc(1, sizeof(*p_ret));
  561.   p_ret->p_next = NULL;
  562.   p_ret->p_name = as_strdup(p_name);
  563.   p_ret->ref_num = next_counter_ref_num++;
  564.   p_ret->value = 0;
  565.   return p_ret;
  566. }
  567.  
  568. /*!------------------------------------------------------------------------
  569.  * \fn     tex_counter_free(tex_counter_t *p_counter)
  570.  * \brief  destroy/free counter structure
  571.  * \param  p_counter counter to free
  572.  * ------------------------------------------------------------------------ */
  573.  
  574. static void tex_counter_free(tex_counter_t *p_counter)
  575. {
  576.   free(p_counter->p_name);
  577.   free(p_counter);
  578. }
  579.  
  580. /*!------------------------------------------------------------------------
  581.  * \fn     tex_counter_add(const char *p_name)
  582.  * \brief  add counter of given name to list of counters
  583.  * \param  p_name name of counter
  584.  * ------------------------------------------------------------------------ */
  585.  
  586. void tex_counter_add(const char *p_name)
  587. {
  588.   tex_counter_t *p_run, *p_prev, *p_new;
  589.  
  590.   for (p_prev = NULL, p_run = tex_counters; p_run; p_prev = p_run, p_run = p_run->p_next)
  591.     if (!as_strcasecmp(p_name, p_run->p_name))
  592.     {
  593.       tex_warning("counter '%s' re-defined", p_name);
  594.       return;
  595.     }
  596.   p_new = tex_counter_new(p_name);
  597.   if (p_prev)
  598.     p_prev->p_next = p_new;
  599.   else
  600.     tex_counters = p_new;
  601. }
  602.  
  603. /*!------------------------------------------------------------------------
  604.  * \fn     tex_counters_free(void)
  605.  * \brief  destroy list of counters
  606.  * ------------------------------------------------------------------------ */
  607.  
  608. void tex_counters_free(void)
  609. {
  610.   while (tex_counters)
  611.   {
  612.     tex_counter_t *p_old = tex_counters;
  613.  
  614.     tex_counters = p_old->p_next;
  615.     tex_counter_free(p_old);
  616.   }
  617.   next_counter_ref_num = 0;
  618. }
  619.  
  620. /*!------------------------------------------------------------------------
  621.  * \fn     tex_counter_set(const char *p_name, unsigned new_value)
  622.  * \brief  set counter to value
  623.  * \param  p_name counter's name
  624.  * \param  new_value value to set counter to
  625.  * ------------------------------------------------------------------------ */
  626.  
  627. void tex_counter_set(const char *p_name, unsigned new_value)
  628. {
  629.   tex_counter_t *p_run;
  630.  
  631.   for (p_run = tex_counters; p_run; p_run = p_run->p_next)
  632.     if (!as_strcasecmp(p_name, p_run->p_name))
  633.     {
  634.       p_run->value = new_value;
  635.       return;
  636.     }
  637.  
  638.   tex_warning("counter '%s' undefined", p_name);
  639. }
  640.  
  641. /*!------------------------------------------------------------------------
  642.  * \fn     tex_counter_get(const char *p_name)
  643.  * \brief  retrieve value of counter
  644.  * \param  p_name counter's name
  645.  * \return counter's value or 0
  646.  * ------------------------------------------------------------------------ */
  647.  
  648. unsigned tex_counter_get(const char *p_name)
  649. {
  650.   tex_counter_t *p_run;
  651.  
  652.   for (p_run = tex_counters; p_run; p_run = p_run->p_next)
  653.     if (!as_strcasecmp(p_name, p_run->p_name))
  654.       return p_run->value;
  655.  
  656.   tex_warning("counter '%s' undefined", p_name);
  657.   return 0;
  658. }
  659.  
  660. /*!------------------------------------------------------------------------
  661.  * \fn     tex_counter_step(const char *p_name)
  662.  * \brief  increment counter
  663.  * \param  p_name counter to increment
  664.  * ------------------------------------------------------------------------ */
  665.  
  666. void tex_counter_step(const char *p_name)
  667. {
  668.   tex_counter_t *p_run;
  669.  
  670.   for (p_run = tex_counters; p_run; p_run = p_run->p_next)
  671.     if (!as_strcasecmp(p_name, p_run->p_name))
  672.     {
  673.       p_run->value++;
  674.       return;
  675.     }
  676.  
  677.   tex_warning("counter '%s' undefined", p_name);
  678. }
  679.  
  680. /*!------------------------------------------------------------------------
  681.  * \fn     tex_newif_new(const char *p_name)
  682.  * \brief  create newif structure
  683.  * \param  p_name name of newif
  684.  * ------------------------------------------------------------------------ */
  685.  
  686. static tex_newif_t *tex_newif_new(const char *p_name)
  687. {
  688.   tex_newif_t *p_ret = (tex_newif_t*)calloc(1, sizeof(*p_ret));
  689.   p_ret->p_next = NULL;
  690.   p_ret->p_name = as_strdup(p_name);
  691.   p_ret->value = False;
  692.   return p_ret;
  693. }
  694.  
  695. /*!------------------------------------------------------------------------
  696.  * \fn     tex_newif_free(tex_newif_t *p_newif)
  697.  * \brief  destroy newif structure
  698.  * \param  p_newif structure to destroy
  699.  * ------------------------------------------------------------------------ */
  700.  
  701. static void tex_newif_free(tex_newif_t *p_newif)
  702. {
  703.   if (p_newif->p_name) free(p_newif->p_name);
  704.   p_newif->p_name = NULL;
  705.   free(p_newif);
  706. }
  707.  
  708. /*!------------------------------------------------------------------------
  709.  * \fn     tex_newif_add(const char *p_name)
  710.  * \brief  add newif to newif list
  711.  * \param  p_name name of newif
  712.  * ------------------------------------------------------------------------ */
  713.  
  714. void tex_newif_add(const char *p_name)
  715. {
  716.   tex_newif_t *p_new, *p_prev, *p_run;
  717.  
  718.   if (as_strncasecmp(p_name, "if", 2))
  719.   {
  720.     tex_warning("\\newif identifier does not start with \\if");
  721.     return;
  722.   }
  723.   p_name += 2;
  724.  
  725.   for (p_prev = NULL, p_run = p_tex_newifs; p_run; p_prev = p_run, p_run = p_run->p_next)
  726.     if (as_strcasecmp(p_run->p_name, p_name))
  727.     {
  728.       tex_warning("\\newif %s redefined", p_name);
  729.       return;
  730.     }
  731.  
  732.   p_new = tex_newif_new(p_name);
  733.   if (p_prev)
  734.     p_prev->p_next = p_new;
  735.   else
  736.     p_tex_newifs = p_new;
  737. }
  738.  
  739. /*!------------------------------------------------------------------------
  740.  * \fn     tex_newif_lookup(const char *p_name)
  741.  * \brief  match command name against newif commands
  742.  * \param  p_name command name
  743.  * ------------------------------------------------------------------------ */
  744.  
  745. Boolean tex_newif_lookup(const char *p_name)
  746. {
  747.   tex_newif_t *p_run;
  748.   Boolean is_if = !as_strncasecmp(p_name, "if", 2),
  749.           is_true, is_false;
  750.   size_t l = strlen(p_name), l2;
  751.  
  752.   is_true = (l > 4) && !as_strcasecmp(p_name + l - 4, "true");
  753.   is_false = (l > 5) && !as_strcasecmp(p_name + l - 5, "false");
  754.   if (!is_if && !is_true && !is_false)
  755.     return False;
  756.   for (p_run = p_tex_newifs; p_run; p_run = p_run->p_next)
  757.   {
  758.     if (is_if && !as_strcasecmp(p_name + 2, p_run->p_name))
  759.     {
  760.       tex_if_push(p_run->value);
  761.       return True;
  762.     }
  763.     l2 = strlen(p_run->p_name);
  764.     if (is_true && (l == l2 + 4) && !as_strncasecmp(p_name, p_run->p_name, l2))
  765.     {
  766.       p_run->value = True;
  767.       return True;
  768.     }
  769.     else if (is_false && (l == l2 + 5) && !as_strncasecmp(p_name, p_run->p_name, l2))
  770.     {
  771.       p_run->value = False;
  772.       return True;
  773.     }
  774.   }
  775.   return False;
  776. }
  777.  
  778. /*!------------------------------------------------------------------------
  779.  * \fn     tex_newifs_free(void)
  780.  * \brief  free all newifs
  781.  * ------------------------------------------------------------------------ */
  782.  
  783. void tex_newifs_free(void)
  784. {
  785.   while (p_tex_newifs)
  786.   {
  787.     tex_newif_t *p_old = p_tex_newifs;
  788.     p_tex_newifs = p_old->p_next;
  789.     tex_newif_free(p_old);
  790.   }
  791. }
  792.  
  793. /*!------------------------------------------------------------------------
  794.  * \fn     tex_environment_new(const char *p_name)
  795.  * \brief  create user-defined environment structure
  796.  * \param  p_name name of environment
  797.  * \return * to structure or NULL
  798.  * ------------------------------------------------------------------------ */
  799.  
  800. static tex_environment_t* tex_environment_new(const char *p_name)
  801. {
  802.   tex_environment_t *p_ret = (tex_environment_t*)calloc(1, sizeof(*p_ret));
  803.  
  804.   p_ret->p_next = NULL;
  805.   p_ret->p_name = as_strdup(p_name);
  806.   p_ret->p_begin_commands = NULL;
  807.   p_ret->p_end_commands = NULL;
  808.   return p_ret;
  809. }
  810.  
  811. /*!------------------------------------------------------------------------
  812.  * \fn     tex_environment_free(tex_environment_t *p_env)
  813.  * \brief  destroy/free user-defined environment structure
  814.  * \param  p_env environment to free
  815.  * ------------------------------------------------------------------------ */
  816.  
  817. static void tex_environment_free(tex_environment_t *p_env)
  818. {
  819.   free(p_env->p_name);
  820.   p_env->p_name = NULL;
  821.   if (p_env->p_begin_commands)
  822.     free(p_env->p_begin_commands);
  823.   p_env->p_begin_commands = NULL;
  824.   if (p_env->p_end_commands)
  825.     free(p_env->p_end_commands);
  826.   p_env->p_end_commands = NULL;
  827.   free(p_env);
  828. }
  829.  
  830. /*!------------------------------------------------------------------------
  831.  * \fn     tex_environment_add(const char *p_name, const char *p_begin_commands, const char *p_end_commands)
  832.  * \brief  add user-defined environment
  833.  * \param  p_name name of environment
  834.  * \param  p_begin_commands environment entry code
  835.  * \param  p_end_commands environment exit code
  836.  * ------------------------------------------------------------------------ */
  837.  
  838. void tex_environment_add(const char *p_name, const char *p_begin_commands, const char *p_end_commands)
  839. {
  840.   tex_environment_t *p_run, *p_prev, *p_new;
  841.  
  842.   for (p_run = tex_environments, p_prev = NULL; p_run; p_prev = p_run, p_run = p_run->p_next)
  843.     if (!as_strcasecmp(p_name, p_run->p_name))
  844.     {
  845.       tex_warning("environment '%s' re-defined", p_name);
  846.       return;
  847.     }
  848.  
  849.   p_new = tex_environment_new(p_name);
  850.   p_new->p_begin_commands = as_strdup(p_begin_commands);
  851.   p_new->p_end_commands = as_strdup(p_end_commands);
  852.   p_new->p_next = p_run;
  853.   if (p_prev)
  854.     p_prev->p_next = p_new;
  855.   else
  856.     tex_environments = p_new;
  857. }
  858.  
  859. /*!------------------------------------------------------------------------
  860.  * \fn     tex_environments_free(void)
  861.  * \brief  delete all user-defined environments
  862.  * ------------------------------------------------------------------------ */
  863.  
  864. void tex_environments_free(void)
  865. {
  866.   while (tex_environments)
  867.   {
  868.     tex_environment_t *p_old = tex_environments;
  869.  
  870.     tex_environments = p_old->p_next;
  871.     tex_environment_free(p_old);
  872.   }
  873. }
  874.  
  875. /*!------------------------------------------------------------------------
  876.  * \fn     tex_environment_lookup(const char *p_name)
  877.  * \brief  search for user-defined environment
  878.  * \param  p_name name of environment
  879.  * \return * to environment's definition or NULL
  880.  * ------------------------------------------------------------------------ */
  881.  
  882. const tex_environment_t *tex_environment_lookup(const char *p_name)
  883. {
  884.   tex_environment_t *p_run;
  885.  
  886.   for (p_run = tex_environments; p_run; p_run = p_run->p_next)
  887.     if (!as_strcasecmp(p_name, p_run->p_name))
  888.       break;
  889.   return p_run;
  890. }
  891.  
  892.  
  893. /*!------------------------------------------------------------------------
  894.  * \fn     tex_newcommand_new(const char *p_name)
  895.  * \brief  create user-defined command
  896.  * \param  p_name name of command
  897.  * \return * to created structure or NULL
  898.  * ------------------------------------------------------------------------ */
  899.  
  900. static tex_newcommand_t* tex_newcommand_new(const char *p_name)
  901. {
  902.   tex_newcommand_t *p_ret = (tex_newcommand_t*)calloc(1, sizeof(*p_ret));
  903.  
  904.   p_ret->p_next = NULL;
  905.   p_ret->p_name = as_strdup(p_name);
  906.   p_ret->p_body = NULL;
  907.   p_ret->num_args = 0;
  908.   return p_ret;
  909. }
  910.  
  911. /*!------------------------------------------------------------------------
  912.  * \fn     tex_newcommand_free(tex_newcommand_t *p_cmd)
  913.  * \brief  destroy/free user-defined command
  914.  * \param  p_cmd command to free
  915.  * ------------------------------------------------------------------------ */
  916.  
  917. static void tex_newcommand_free(tex_newcommand_t *p_cmd)
  918. {
  919.   free(p_cmd->p_name);
  920.   p_cmd->p_name = NULL;
  921.   if (p_cmd->p_body)
  922.     free(p_cmd->p_body);
  923.   p_cmd->p_body = NULL;
  924.   free(p_cmd);
  925. }
  926.  
  927. /*!------------------------------------------------------------------------
  928.  * \fn     tex_newcommand_add(const char *p_name, const char *p_body, unsigned num_args)
  929.  * \brief  add user-defined command
  930.  * \param  p_name name of command
  931.  * \param  p_body command's body
  932.  * \param  num_args number of arguments
  933.  * ------------------------------------------------------------------------ */
  934.  
  935. void tex_newcommand_add(const char *p_name, const char *p_body, unsigned num_args)
  936. {
  937.   tex_newcommand_t *p_run, *p_prev, *p_new;
  938.  
  939.   for (p_run = tex_newcommands, p_prev = NULL; p_run; p_prev = p_run, p_run = p_run->p_next)
  940.     if (!as_strcasecmp(p_name, p_run->p_name))
  941.     {
  942.       tex_warning("newcommand '%s' re-defined", p_name);
  943.       return;
  944.     }
  945.  
  946.   p_new = tex_newcommand_new(p_name);
  947.   p_new->p_body = as_strdup(p_body);
  948.   p_new->num_args = num_args;
  949.   p_new->p_next = p_run;
  950.   if (p_prev)
  951.     p_prev->p_next = p_new;
  952.   else
  953.     tex_newcommands = p_new;
  954. }
  955.  
  956. /*!------------------------------------------------------------------------
  957.  * \fn     tex_newcommands_free(void)
  958.  * \brief  destroy list of user-defined commands
  959.  * ------------------------------------------------------------------------ */
  960.  
  961. void tex_newcommands_free(void)
  962. {
  963.   while (tex_newcommands)
  964.   {
  965.     tex_newcommand_t *p_old = tex_newcommands;
  966.  
  967.     tex_newcommands = p_old->p_next;
  968.     tex_newcommand_free(p_old);
  969.   }
  970. }
  971.  
  972. /*!------------------------------------------------------------------------
  973.  * \fn     tex_newcommand_lookup(const char *p_name)
  974.  * \brief  lookup user-defined command
  975.  * \param  p_name name of command
  976.  * \return command's definition or NULL
  977.  * ------------------------------------------------------------------------ */
  978.  
  979. const tex_newcommand_t *tex_newcommand_lookup(const char *p_name)
  980. {
  981.   tex_newcommand_t *p_run;
  982.  
  983.   for (p_run = tex_newcommands; p_run; p_run = p_run->p_next)
  984.     if (!as_strcasecmp(p_name, p_run->p_name))
  985.       break;
  986.   return p_run;
  987. }
  988.  
  989. /*!------------------------------------------------------------------------
  990.  * \fn     tex_newcommand_expand_push(const tex_newcommand_t *p_cmd)
  991.  * \brief  expand arguments to command and pushto input stream
  992.  * \param  p_cmd command descriptor
  993.  * ------------------------------------------------------------------------ */
  994.  
  995. void tex_newcommand_expand_push(const tex_newcommand_t *p_cmd)
  996. {
  997.   unsigned z;
  998.   char arg_token[TOKLEN];
  999.   char *p_expanded_line = (char*)p_cmd->p_body;
  1000.   Boolean line_dynamic = False;
  1001.  
  1002.   for (z = 0; z < p_cmd->num_args; z++)
  1003.   {
  1004.     char arg_holder[10], *p_pos;
  1005.     size_t holder_len, arg_len;
  1006.  
  1007.     tex_assert_token("{");
  1008.     collect_arg_tokens(arg_token, sizeof(arg_token));
  1009.     arg_len = strlen(arg_token);
  1010.     holder_len = as_snprintf(arg_holder, sizeof(arg_holder), "#%u", z + 1);
  1011.     while ((p_pos = strstr(p_expanded_line, arg_holder)))
  1012.     {
  1013.       size_t expanded_len = strlen(p_expanded_line),
  1014.              new_expanded_len, pos;
  1015.       ptrdiff_t delta;
  1016.  
  1017.       pos = p_pos - p_expanded_line;
  1018.       /* only generate copy of line if necessary: */
  1019.       if (!line_dynamic)
  1020.       {
  1021.         p_expanded_line = as_strdup(p_expanded_line);
  1022.         line_dynamic = True;
  1023.         p_pos = p_expanded_line + pos;
  1024.       }
  1025.       new_expanded_len = expanded_len - holder_len + arg_len;
  1026.       if (new_expanded_len > expanded_len)
  1027.       {
  1028.         p_expanded_line = (char*)realloc(p_expanded_line, new_expanded_len + 1);
  1029.         p_pos = p_expanded_line + pos;
  1030.       }
  1031.       delta = arg_len - holder_len;
  1032.       memmove(p_pos + holder_len + delta,
  1033.               p_pos + holder_len,
  1034.               expanded_len - pos - holder_len + 1);
  1035.       memcpy(p_pos, arg_token, arg_len);
  1036.     }
  1037.   }
  1038.   tex_infile_push_line(p_cmd->p_name, p_expanded_line, line_dynamic);
  1039. }
  1040.  
  1041. EnvType tex_get_env_type(char *p_name, const tex_environment_t **pp_env)
  1042. {
  1043.   EnvType z;
  1044.  
  1045.   if (!as_strcasecmp(p_name, "longtable"))
  1046.     return EnvTabular;
  1047.   for (z = EnvNone + 1; z < EnvCount; z++)
  1048.     if (!as_strcasecmp(p_name, tex_env_names[z]))
  1049.     {
  1050.       *pp_env = NULL;
  1051.       return z;
  1052.     }
  1053.  
  1054.   *pp_env = tex_environment_lookup(p_name);
  1055.   if (*pp_env)
  1056.     return EnvUser;
  1057.  
  1058.   tex_error("unknown environment '%s'", p_name);
  1059.   return EnvNone;
  1060. }
  1061.  
  1062. void tex_save_env(EnvType new_env, const char *p_user_env_name)
  1063. {
  1064.   tex_env_save_t *p_new_save;
  1065.  
  1066.   p_new_save = (tex_env_save_t*) malloc(sizeof(*p_new_save));
  1067.   p_new_save->p_next = p_env_stack;
  1068.   p_new_save->save_env_data = curr_tex_env_data;
  1069.   p_new_save->save_env = curr_tex_env;
  1070.   p_new_save->p_save_user_env_name = p_curr_tex_user_env_name; p_curr_tex_user_env_name = NULL;
  1071.   p_env_stack = p_new_save;
  1072.   curr_tex_env = new_env;
  1073.   p_curr_tex_user_env_name = (new_env == EnvUser) ? as_strdup(p_user_env_name) : NULL;
  1074.   curr_tex_env_data.FontNest = 0;
  1075. }
  1076.  
  1077. void tex_restore_env(void)
  1078. {
  1079.   tex_env_save_t *p_old_save;
  1080.  
  1081.   p_old_save = p_env_stack;
  1082.   p_env_stack = p_old_save->p_next;
  1083.   curr_tex_env_data = p_old_save->save_env_data;
  1084.   curr_tex_env = p_old_save->save_env;
  1085.   if (p_curr_tex_user_env_name) free(p_curr_tex_user_env_name);
  1086.   p_curr_tex_user_env_name = p_old_save->p_save_user_env_name; p_old_save->p_save_user_env_name = NULL;
  1087.   free(p_old_save);
  1088. }
  1089.  
  1090. /*!------------------------------------------------------------------------
  1091.  * \fn     tex_output_consumer_push(tex_output_consumer_t *p_consumer)
  1092.  * \brief  put a new output consumer at front
  1093.  * \param  p_consumer the new consumer instance
  1094.  * ------------------------------------------------------------------------ */
  1095.  
  1096. void tex_output_consumer_push(tex_output_consumer_t *p_consumer)
  1097. {
  1098.   p_consumer->p_next = p_current_tex_output_consumer;
  1099.   p_current_tex_output_consumer = p_consumer;
  1100. }
  1101.  
  1102. /*!------------------------------------------------------------------------
  1103.  * \fn     tex_output_consumer_pop(void)
  1104.  * \brief  discard innermost output consumer
  1105.  * ------------------------------------------------------------------------ */
  1106.  
  1107. void tex_output_consumer_pop(void)
  1108. {
  1109.   tex_output_consumer_t *p_consumer = p_current_tex_output_consumer;
  1110.   if (p_consumer)
  1111.   {
  1112.     p_current_tex_output_consumer = p_consumer->p_next;
  1113.     if (p_consumer->destroy) p_consumer->destroy(p_consumer);
  1114.     free(p_consumer);
  1115.   }
  1116. }
  1117.  
  1118. /*!------------------------------------------------------------------------
  1119.  * \fn     tex_output_consumer_pop_all(void)
  1120.  * \brief  dissolve output consumer stack
  1121.  * ------------------------------------------------------------------------ */
  1122.  
  1123. void tex_output_consumer_pop_all(void)
  1124. {
  1125.   while (p_current_tex_output_consumer)
  1126.     tex_output_consumer_pop();
  1127. }
  1128.  
  1129. /*!------------------------------------------------------------------------
  1130.  * \fn     TeXIfNum(Word index)
  1131.  * \brief  parse \ifnum command
  1132.  * ------------------------------------------------------------------------ */
  1133.  
  1134. typedef struct tex_ifnum_output_consumer
  1135. {
  1136.   tex_output_consumer_t consumer;
  1137.   int arg_index;
  1138.   int result;
  1139.   char op[10];
  1140.   unsigned num[2];
  1141. } tex_ifnum_output_consumer_t;
  1142.  
  1143. static void if_num_consume(struct tex_output_consumer *p_consumer, const char **pp_str)
  1144. {
  1145.   tex_ifnum_output_consumer_t *p_ctx = (tex_ifnum_output_consumer_t*)p_consumer;
  1146.   int read_rhs = 0;
  1147.  
  1148. #if DBG_IF
  1149.   fprintf(stderr, "if_num_func called with '%s' state %d\n", *pp_str, p_ctx->arg_index);
  1150. #endif
  1151.   for (; **pp_str && !p_ctx->result; (*pp_str)++)
  1152.   {
  1153.     int old_arg_index;
  1154.     do
  1155.     {
  1156. #if DBG_IF
  1157.       fprintf(stderr, " if_num_func char '%c' %d\n", **pp_str, p_ctx->arg_index);
  1158. #endif
  1159.       old_arg_index = p_ctx->arg_index;
  1160.       switch (old_arg_index)
  1161.       {
  1162.         case 0: /* leading spaces before left-hand number */
  1163.           if (isspace(**pp_str));
  1164.           else if (isdigit(**pp_str))
  1165.             p_ctx->arg_index = 1;
  1166.           else
  1167.           {
  1168.             tex_warning("non-numeric character '%c'", **pp_str);
  1169.             p_ctx->result = -1;
  1170.           }
  1171.           break;
  1172.         case 1: /* left-hand number */
  1173.           if (isdigit(**pp_str))
  1174.             p_ctx->num[0] = (p_ctx->num[0] * 10) + (**pp_str - '0');
  1175.           else if (isspace(**pp_str))
  1176.             p_ctx->arg_index = 2;
  1177.           else
  1178.             p_ctx->arg_index = 3;
  1179.           break;
  1180.         case 2: /* leading spaces before operator */
  1181.           if (isspace(**pp_str));
  1182.           else if (isdigit(**pp_str))
  1183.           {
  1184.             tex_warning("numeric character '%c' in operator", **pp_str);
  1185.             p_ctx->result = -1;
  1186.           }
  1187.           else
  1188.             p_ctx->arg_index = 3;
  1189.           break;
  1190.         case 3: /* operator */
  1191.           if (isspace(**pp_str))
  1192.             p_ctx->arg_index = 4;
  1193.           else if (isdigit(**pp_str))
  1194.             p_ctx->arg_index = 5;
  1195.           else
  1196.           {
  1197.             size_t l = strlen(p_ctx->op);
  1198.             if (l < sizeof(p_ctx->op) - 1)
  1199.             {
  1200.               p_ctx->op[l] = **pp_str;
  1201.               p_ctx->op[l + 1] = '\0';
  1202.             }
  1203.           }
  1204.           break;
  1205.         case 4: /* leading spaces before right-hand number */
  1206.           if (isspace(**pp_str));
  1207.           else if (isdigit(**pp_str))
  1208.             p_ctx->arg_index = 5;
  1209.           else
  1210.           {
  1211.             tex_warning("non-numeric character '%c'", **pp_str);
  1212.             p_ctx->result = -1;
  1213.           }
  1214.           break;
  1215.         case 5: /* right-hand number */
  1216.           if (isdigit(**pp_str))
  1217.           {
  1218.             p_ctx->num[1] = (p_ctx->num[1] * 10) + (**pp_str - '0');
  1219.             read_rhs = 1;
  1220.           }
  1221.           else
  1222.             p_ctx->result = 1;
  1223.           break;
  1224.         default:
  1225.           p_ctx->result = -1;
  1226.       }
  1227.     }
  1228.     while (!p_ctx->result && (old_arg_index != p_ctx->arg_index));
  1229.   }
  1230.  
  1231.   /* Last argument is a number and therefore expected to arrive as a single token
  1232.      in the output stream.  So when this token is used up and we read the right-hand
  1233.      number, we may assume we have read all arguments for \ifnum: */
  1234.  
  1235.   if (!**pp_str && read_rhs)
  1236.     p_ctx->result = 1;
  1237.   if (p_ctx->result)
  1238.   {
  1239.     int condition;
  1240.  
  1241. #if DBG_IF
  1242.     fprintf(stderr, "ifnum result %d %u %s %u\n",
  1243.             p_ctx->result,
  1244.             p_ctx->num[0],
  1245.             p_ctx->op,
  1246.             p_ctx->num[1]);
  1247. #endif
  1248.     if (p_ctx->result < 0)
  1249.       condition = True;
  1250.     else if (!as_strcasecmp(p_ctx->op, ">"))
  1251.       condition = (p_ctx->num[0] > p_ctx->num[1]);
  1252.     else if (!as_strcasecmp(p_ctx->op, "<"))
  1253.       condition = (p_ctx->num[0] < p_ctx->num[1]);
  1254.     else
  1255.     {
  1256.       tex_warning("unknown operator: '%s'", p_ctx->op);
  1257.       condition = True;
  1258.     }
  1259.    
  1260.     tex_output_consumer_pop();
  1261.     tex_if_push(condition);
  1262.   }
  1263. }
  1264.  
  1265. void TeXIfNum(Word index)
  1266. {
  1267.   tex_ifnum_output_consumer_t *p_ctx;
  1268.   UNUSED(index);
  1269.  
  1270. #if DBG_IF
  1271.   fprintf(stderr, "insert if\n");
  1272. #endif
  1273.   p_ctx = (tex_ifnum_output_consumer_t*)calloc(1, sizeof(*p_ctx));
  1274.   p_ctx->consumer.consume = if_num_consume;
  1275.   p_ctx->consumer.destroy = NULL;
  1276.   p_ctx->result = 0;
  1277.   p_ctx->op[0] = '\0';
  1278.   p_ctx->num[0] =
  1279.   p_ctx->num[1] = 0;
  1280.   p_ctx->arg_index = 0;
  1281.   tex_output_consumer_push(&p_ctx->consumer);
  1282. }
  1283.  
  1284. /*!------------------------------------------------------------------------
  1285.  * \fn     TeXFi(Word index)
  1286.  * \brief  handle \fi
  1287.  * ------------------------------------------------------------------------ */
  1288.  
  1289. void TeXFi(Word index)
  1290. {
  1291.   UNUSED(index);
  1292.  
  1293.   tex_if_pop();
  1294. }
  1295.  
  1296. /*!------------------------------------------------------------------------
  1297.  * \fn     tex_if_nest(void)
  1298.  * \brief  query depth of if stack
  1299.  * \return depth
  1300.  * ------------------------------------------------------------------------ */
  1301.  
  1302. unsigned tex_if_nest(void)
  1303. {
  1304.   unsigned ret;
  1305.   tex_if_stack_t *p_run;
  1306.  
  1307.   for (ret = 0, p_run = p_if_stack; p_run; ret++, p_run = p_run->p_next);
  1308.   return ret;
  1309. }
  1310.  
  1311. /*!------------------------------------------------------------------------
  1312.  * \fn     tex_if_push(int condition)
  1313.  * \brief  put new if condition on stack
  1314.  * \param  condition evaluated if (true/false)
  1315.  * ------------------------------------------------------------------------ */
  1316.  
  1317. void tex_if_push(int condition)
  1318. {
  1319.   tex_if_stack_t *p_new = (tex_if_stack_t*)calloc(1, sizeof(*p_new));
  1320.  
  1321. #if DBG_IF
  1322.   fprintf(stderr, "push if %u", condition);
  1323. #endif
  1324.  
  1325.   p_new->state = e_tex_if_if;
  1326.   p_new->condition = condition;
  1327.   p_new->p_next = p_if_stack;
  1328.   p_if_stack = p_new;
  1329.  
  1330. #if DBG_IF
  1331.   fprintf(stderr, ", depth %u\n", tex_if_nest());
  1332. #endif
  1333. }
  1334.  
  1335. /*!------------------------------------------------------------------------
  1336.  * \fn     tex_if_toggle(void)
  1337.  * \brief  toggle topmost if condition
  1338.  * ------------------------------------------------------------------------ */
  1339.  
  1340. void tex_if_toggle(void)
  1341. {
  1342.   if (!p_if_stack)
  1343.     tex_warning("\\else without \\if");
  1344.   else if (p_if_stack->state == e_tex_if_else)
  1345.     tex_warning("extra \\else");
  1346.   else
  1347.   {
  1348.     p_if_stack->state = e_tex_if_else;
  1349.     p_if_stack->condition = !p_if_stack->condition;
  1350.   }
  1351. }
  1352.  
  1353. /*!------------------------------------------------------------------------
  1354.  * \fn     tex_if_pop(void)
  1355.  * \brief  remove topmost if condition
  1356.  * ------------------------------------------------------------------------ */
  1357.  
  1358. void tex_if_pop(void)
  1359. {
  1360. #if DBG_IF
  1361.   fprintf(stderr, "pop if");
  1362. #endif
  1363.  
  1364.   if (!p_if_stack)
  1365.     tex_warning("\\fi without \\if");
  1366.   else
  1367.   {
  1368.     tex_if_stack_t *p_if = p_if_stack;
  1369.     p_if_stack = p_if->p_next;
  1370.     free(p_if);
  1371. #if DBG_IF
  1372.     fprintf(stderr, ", depth %u\n", tex_if_nest());
  1373. #endif
  1374.   }
  1375. }
  1376.  
  1377. /*!------------------------------------------------------------------------
  1378.  * \fn     tex_if_pop_all(void)
  1379.  * \brief  clear if stack
  1380.  * ------------------------------------------------------------------------ */
  1381.  
  1382. void tex_if_pop_all(void)
  1383. {
  1384.   if (p_if_stack)
  1385.     tex_warning("\\if without \\fi");
  1386.   while (p_if_stack)
  1387.     tex_if_pop();
  1388. }
  1389.  
  1390. /*!------------------------------------------------------------------------
  1391.  * \fn     tex_if_query(void)
  1392.  * \brief  query current if state
  1393.  * ------------------------------------------------------------------------ */
  1394.  
  1395. int tex_if_query(void)
  1396. {
  1397.   tex_if_stack_t *p_run;
  1398.  
  1399.   for (p_run = p_if_stack; p_run; p_run = p_run->p_next)
  1400.     if (!p_run->condition)
  1401.       return False;
  1402.   return True;
  1403. }
  1404.  
  1405. /*!------------------------------------------------------------------------
  1406.  * \fn     TeXNewCounter(Word index)
  1407.  * \brief  parse \newcounter command
  1408.  * ------------------------------------------------------------------------ */
  1409.  
  1410. void TeXNewCounter(Word index)
  1411. {
  1412.   char token[TOKLEN];
  1413.  
  1414.   UNUSED(index);
  1415.  
  1416.   tex_assert_token("{");
  1417.   tex_read_token(token);
  1418.   tex_assert_token("}");
  1419.   if (tex_if_query())
  1420.     tex_counter_add(token);
  1421. }
  1422.  
  1423. /*!------------------------------------------------------------------------
  1424.  * \fn     TeXStepCounter(Word index)
  1425.  * \brief  parse \stepcounter command
  1426.  * ------------------------------------------------------------------------ */
  1427.  
  1428. void TeXStepCounter(Word index)
  1429. {
  1430.   char token[TOKLEN];
  1431.  
  1432.   UNUSED(index);
  1433.  
  1434.   tex_assert_token("{");
  1435.   tex_read_token(token);
  1436.   tex_assert_token("}");
  1437.   if (tex_if_query())
  1438.     tex_counter_step(token);
  1439. }
  1440.  
  1441. /*!------------------------------------------------------------------------
  1442.  * \fn     TeXSetCounter(Word index)
  1443.  * \brief  parse \setcounter command
  1444.  * ------------------------------------------------------------------------ */
  1445.  
  1446. void TeXSetCounter(Word index)
  1447. {
  1448.   char token[TOKLEN], value_str[TOKLEN], *p_end;
  1449.   unsigned value;
  1450.  
  1451.   UNUSED(index);
  1452.  
  1453.   tex_assert_token("{");
  1454.   tex_read_token(token);
  1455.   tex_assert_token("}");
  1456.   tex_assert_token("{");
  1457.   tex_read_token(value_str);
  1458.   tex_assert_token("}");
  1459.   value = strtol(value_str, &p_end, 10);
  1460.   if (*p_end)
  1461.   {
  1462.     tex_warning("'%s': invalid numeric constant", value_str);
  1463.   }
  1464.   if (tex_if_query())
  1465.     tex_counter_set(token, value);
  1466. }
  1467.  
  1468. /*!------------------------------------------------------------------------
  1469.  * \fn     TeXNewEnvironment(Word Index)
  1470.  * \brief  handle \newenvironment
  1471.  * ------------------------------------------------------------------------ */
  1472.  
  1473. void TeXNewEnvironment(Word Index)
  1474. {
  1475.   char env_name[TOKLEN], env_entry[TOKLEN], env_exit[TOKLEN];
  1476.   UNUSED(Index);
  1477.  
  1478.   tex_assert_token("{");
  1479.   tex_read_token(env_name);
  1480.   tex_assert_token("}");
  1481.   tex_assert_token("{");
  1482.   collect_arg_tokens(env_entry, sizeof(env_entry));
  1483.   tex_assert_token("{");
  1484.   collect_arg_tokens(env_exit, sizeof(env_exit));
  1485.   tex_environment_add(env_name, env_entry, env_exit);
  1486. }
  1487.  
  1488. /*!------------------------------------------------------------------------
  1489.  * \fn     TeXNewCommand(Word Index)
  1490.  * \brief  handle \newcommand
  1491.  * ------------------------------------------------------------------------ */
  1492.  
  1493. void TeXNewCommand(Word Index)
  1494. {
  1495.   char token[TOKLEN], command[TOKLEN], sum_token[TOKLEN], arg_cnt[TOKLEN];
  1496.   unsigned num_args = 0;
  1497.   UNUSED(Index);
  1498.  
  1499.   tex_assert_token("{");
  1500.   tex_assert_token("\\");
  1501.   tex_read_token(command);
  1502.   tex_assert_token("}");
  1503.   tex_read_token(token);
  1504.  
  1505.   if (!strcmp(token, "["))
  1506.   {
  1507.     char *p_end;
  1508.  
  1509.     tex_read_token(arg_cnt);
  1510.     num_args = strtoul(arg_cnt, &p_end, 10);
  1511.     if (*p_end)
  1512.       tex_error("'%s': invalid argument count", arg_cnt);
  1513.     tex_assert_token("]");
  1514.     tex_read_token(token);
  1515.   }
  1516.   if (strcmp(token, "{"))
  1517.     tex_error("\"{\" expected");
  1518.  
  1519.   collect_arg_tokens(sum_token, sizeof(sum_token));
  1520.   if (!as_strcasecmp(command, "cpu")
  1521.    || !as_strcasecmp(command, "asname")
  1522.    || !as_strcasecmp(command, "errentry")
  1523.    || !as_strcasecmp(command, "headid"))
  1524.     tex_newcommand_add(command, sum_token, num_args);
  1525. }
  1526.  
  1527. /*!------------------------------------------------------------------------
  1528.  * \fn     TeXNewIf(Word index)
  1529.  * \brief  handle \newif command
  1530.  * ------------------------------------------------------------------------ */
  1531.  
  1532. void TeXNewIf(Word index)
  1533. {
  1534.   char if_name[TOKLEN];
  1535.   UNUSED(index);
  1536.  
  1537.   tex_assert_token("\\");
  1538.   tex_read_token(if_name);
  1539.   tex_newif_add(if_name);
  1540. }
  1541.