/[LeafOK_CVS]/lbbs/src/lml.c
ViewVC logotype

Diff of /lbbs/src/lml.c

Parent Directory Parent Directory | Revision Log Revision Log | View Patch Patch

Revision 1.42 by sysadm, Tue Nov 4 13:49:51 2025 UTC Revision 1.52 by sysadm, Fri Dec 19 06:16:27 2025 UTC
# Line 3  Line 3 
3   * lml   * lml
4   *   - LML render   *   - LML render
5   *   *
6   * Copyright (C) 2004-2025 by Leaflet <leaflet@leafok.com>   * Copyright (C) 2004-2025  Leaflet <leaflet@leafok.com>
7   */   */
8    
9    #ifdef HAVE_CONFIG_H
10    #include "config.h"
11    #endif
12    
13  #include "common.h"  #include "common.h"
14  #include "lml.h"  #include "lml.h"
15  #include "log.h"  #include "log.h"
# Line 15  Line 19 
19  #include <string.h>  #include <string.h>
20  #include <sys/param.h>  #include <sys/param.h>
21    
22  #define LML_TAG_PARAM_BUF_LEN 256  enum _lml_constant_t
23  #define LML_TAG_OUTPUT_BUF_LEN 1024  {
24            LML_TAG_PARAM_BUF_LEN = 256,
25            LML_TAG_OUTPUT_BUF_LEN = 1024,
26            LML_TAG_QUOTE_MAX_LEVEL = 10,
27    };
28    
29  clock_t lml_total_exec_duration = 0;  clock_t lml_total_exec_duration = 0;
30    
# Line 58  static int lml_tag_color_filter(const ch Line 66  static int lml_tag_color_filter(const ch
66          return 0;          return 0;
67  }  }
68    
 #define LML_TAG_QUOTE_MAX_LEVEL 10  
   
69  static const char *lml_tag_quote_color[] = {  static const char *lml_tag_quote_color[] = {
70          "\033[33m", // yellow          "\033[33m", // yellow
71          "\033[32m", // green          "\033[32m", // green
# Line 154  const LML_TAG_DEF lml_tag_def[] = { Line 160  const LML_TAG_DEF lml_tag_def[] = {
160          {"bwf", "\033[1;31m****\033[m", "", "****", NULL},          {"bwf", "\033[1;31m****\033[m", "", "****", NULL},
161  };  };
162    
163  #define LML_TAG_COUNT (sizeof(lml_tag_def) / sizeof(LML_TAG_DEF))  static const int lml_tag_count = sizeof(lml_tag_def) / sizeof(LML_TAG_DEF);
164    static int lml_tag_name_len[sizeof(lml_tag_def) / sizeof(LML_TAG_DEF)];
 static int lml_tag_name_len[LML_TAG_COUNT];  
165  static int lml_ready = 0;  static int lml_ready = 0;
166    
167  inline static void lml_init(void)  inline static void lml_init(void)
# Line 165  inline static void lml_init(void) Line 170  inline static void lml_init(void)
170    
171          if (!lml_ready)          if (!lml_ready)
172          {          {
173                  for (i = 0; i < LML_TAG_COUNT; i++)                  for (i = 0; i < lml_tag_count; i++)
174                  {                  {
175                          lml_tag_name_len[i] = (int)strlen(lml_tag_def[i].tag_name);                          lml_tag_name_len[i] = (int)strlen(lml_tag_def[i].tag_name);
176                  }                  }
# Line 175  inline static void lml_init(void) Line 180  inline static void lml_init(void)
180  }  }
181    
182  #define CHECK_AND_APPEND_OUTPUT(out_buf, out_buf_len, out_buf_offset, tag_out, tag_out_len, line_width)                             \  #define CHECK_AND_APPEND_OUTPUT(out_buf, out_buf_len, out_buf_offset, tag_out, tag_out_len, line_width)                             \
183            if ((tag_out_len) > 0)                                                                                                          \
184          {                                                                                                                               \          {                                                                                                                               \
185                  if ((out_buf_offset) + (tag_out_len) >= (out_buf_len))                                                                      \                  if ((out_buf_offset) + (tag_out_len) >= (out_buf_len))                                                                      \
186                  {                                                                                                                           \                  {                                                                                                                           \
187                          log_error("Buffer is not longer enough for output string %d >= %d\n", (out_buf_offset) + (tag_out_len), (out_buf_len)); \                          log_error("Buffer is not longer enough for output string %d >= %d", (out_buf_offset) + (tag_out_len), (out_buf_len)); \
188                          out_buf[out_buf_offset] = '\0';                                                                                         \                          out_buf[out_buf_offset] = '\0';                                                                                         \
189                          return (out_buf_offset);                                                                                                \                          return (out_buf_offset);                                                                                                \
190                  }                                                                                                                           \                  }                                                                                                                           \
# Line 208  int lml_render(const char *str_in, char Line 214  int lml_render(const char *str_in, char
214          int fb_quote_level = 0;          int fb_quote_level = 0;
215          int tag_name_found;          int tag_name_found;
216          int line_width = 0;          int line_width = 0;
217            char tab_spaces[TAB_SIZE + 1];
218            int tab_width = 0;
219    
220          clock_begin = clock();          clock_begin = clock();
221    
222    #ifdef _DEBUG
223            size_t str_in_len = strlen(str_in);
224    #endif
225    
226          lml_init();          lml_init();
227    
228          lml_tag_disabled = 0;          lml_tag_disabled = 0;
# Line 223  int lml_render(const char *str_in, char Line 235  int lml_render(const char *str_in, char
235    
236          for (i = 0; str_in[i] != '\0'; i++)          for (i = 0; str_in[i] != '\0'; i++)
237          {          {
238    #ifdef _DEBUG
239                    if (i >= str_in_len)
240                    {
241                            log_error("Bug: i(%d) >= str_in_len(%d)", i, str_in_len);
242                            break;
243                    }
244    #endif
245    
246                  if (!lml_tag_disabled && new_line)                  if (!lml_tag_disabled && new_line)
247                  {                  {
248                          while (str_in[i] == ':' && str_in[i + 1] == ' ') // FB2000 quote leading str                          while (str_in[i] == ':' && str_in[i + 1] == ' ') // FB2000 quote leading str
# Line 249  int lml_render(const char *str_in, char Line 269  int lml_render(const char *str_in, char
269                          CHECK_AND_APPEND_OUTPUT(str_out, buf_len, j, tag_output_buf, tag_output_len, line_width);                          CHECK_AND_APPEND_OUTPUT(str_out, buf_len, j, tag_output_buf, tag_output_len, line_width);
270    
271                          new_line = 0;                          new_line = 0;
272                            i--; // redo at current i
273                            continue;
274                  }                  }
275    
276                  if (lml_tag_disabled && new_line)                  if (lml_tag_disabled && new_line)
# Line 258  int lml_render(const char *str_in, char Line 280  int lml_render(const char *str_in, char
280    
281                  if (!quote_mode && !lml_tag_disabled && str_in[i] == '\033' && str_in[i + 1] == '[') // Escape sequence                  if (!quote_mode && !lml_tag_disabled && str_in[i] == '\033' && str_in[i + 1] == '[') // Escape sequence
282                  {                  {
283                          for (k = i + 2; isdigit(str_in[k]) || str_in[k] == ';' || str_in[k] == '?'; k++)                          for (k = i + 2; isdigit((int)str_in[k]) || str_in[k] == ';' || str_in[k] == '?'; k++)
284                                  ;                                  ;
285    
286                          if (str_in[k] == 'm') // valid -- copy directly                          if (str_in[k] == 'm') // valid -- copy directly
287                          {                          {
288                                  CHECK_AND_APPEND_OUTPUT(str_out, buf_len, j, str_in + i, k - i + 1, line_width);                                  CHECK_AND_APPEND_OUTPUT(str_out, buf_len, j, str_in + i, k - i + 1, line_width);
289                          }                          }
290                          else if (isalpha(str_in[k]))                          else if (isalpha((int)str_in[k]))
291                          {                          {
292                                  // unsupported ANSI CSI command                                  // unsupported ANSI CSI command
293                          }                          }
# Line 282  int lml_render(const char *str_in, char Line 304  int lml_render(const char *str_in, char
304                  {                  {
305                          if (!lml_tag_disabled && tag_start_pos != -1) // tag is not closed                          if (!lml_tag_disabled && tag_start_pos != -1) // tag is not closed
306                          {                          {
307                                  tag_end_pos = i - 1;                                  if (line_width + 1 > width)
                                 tag_output_len = tag_end_pos - tag_start_pos + 1;  
   
                                 if (line_width + tag_output_len > width)  
308                                  {                                  {
309                                          CHECK_AND_APPEND_OUTPUT(str_out, buf_len, j, "\n", 1, line_width);                                          CHECK_AND_APPEND_OUTPUT(str_out, buf_len, j, "\n", 1, line_width);
310                                          new_line = 1;                                          new_line = 1;
311                                          line_width = 0;                                          line_width = 0;
312                                          i--; // redo at current i                                          i--; // redo at current i
313                                            continue;
314                                  }                                  }
315                                  else  
316                                  {                                  CHECK_AND_APPEND_OUTPUT(str_out, buf_len, j, "[", 1, line_width);
317                                          CHECK_AND_APPEND_OUTPUT(str_out, buf_len, j, str_in + tag_start_pos, tag_output_len, line_width);                                  i = tag_start_pos; // restart from tag_start_pos + 1
318                                  }                                  tag_start_pos = -1;
319                                    tag_name_pos = -1;
320                                    continue;
321                          }                          }
322    
323                          if (!lml_tag_disabled && fb_quote_level > 0)                          if (!lml_tag_disabled && fb_quote_level > 0)
# Line 322  int lml_render(const char *str_in, char Line 344  int lml_render(const char *str_in, char
344                  {                  {
345                          continue; // Skip special characters                          continue; // Skip special characters
346                  }                  }
347                    else if (str_in[i] == '\t')
348                    {
349                            tab_width = TAB_SIZE - (line_width % TAB_SIZE);
350                            if (line_width + tab_width > width)
351                            {
352                                    CHECK_AND_APPEND_OUTPUT(str_out, buf_len, j, "\n", 1, line_width);
353                                    new_line = 1;
354                                    line_width = 0;
355                                    // skip current Tab
356                                    continue;
357                            }
358    
359                            for (k = 0; k < tab_width; k++)
360                            {
361                                    tab_spaces[k] = ' ';
362                            }
363                            tab_spaces[tab_width] = '\0';
364                            CHECK_AND_APPEND_OUTPUT(str_out, buf_len, j, tab_spaces, tab_width, line_width);
365                            continue;
366                    }
367    
368                  if (!lml_tag_disabled && str_in[i] == '[')                  if (!lml_tag_disabled && str_in[i] == '[')
369                  {                  {
370                          if (tag_start_pos != -1) // tag is not closed                          if (tag_start_pos != -1) // tag is not closed
371                          {                          {
372                                  tag_end_pos = i - 1;                                  if (line_width + 1 > width)
373                                  tag_output_len = tag_end_pos - tag_start_pos + 1;                                  {
374                                  CHECK_AND_APPEND_OUTPUT(str_out, buf_len, j, str_in + tag_start_pos, tag_output_len, line_width);                                          CHECK_AND_APPEND_OUTPUT(str_out, buf_len, j, "\n", 1, line_width);
375                                            new_line = 1;
376                                            line_width = 0;
377                                            i--; // redo at current i
378                                            continue;
379                                    }
380    
381                                    CHECK_AND_APPEND_OUTPUT(str_out, buf_len, j, "[", 1, line_width);
382                                    i = tag_start_pos; // restart from tag_start_pos + 1
383                                    tag_start_pos = -1;
384                                    tag_name_pos = -1;
385                                    continue;
386                          }                          }
387    
388                          tag_start_pos = i;                          tag_start_pos = i;
# Line 345  int lml_render(const char *str_in, char Line 398  int lml_render(const char *str_in, char
398                                  tag_name_pos++;                                  tag_name_pos++;
399                          }                          }
400    
401                          for (tag_name_found = 0, k = 0; k < LML_TAG_COUNT; k++)                          for (tag_name_found = 0, k = 0; k < lml_tag_count; k++)
402                          {                          {
403                                  if (strncasecmp(lml_tag_def[k].tag_name, str_in + tag_name_pos, (size_t)lml_tag_name_len[k]) == 0)                                  if (strncasecmp(lml_tag_def[k].tag_name, str_in + tag_name_pos, (size_t)lml_tag_name_len[k]) == 0)
404                                  {                                  {
# Line 393  int lml_render(const char *str_in, char Line 446  int lml_render(const char *str_in, char
446                                                                          lml_tag_def[k].tag_name, tag_param_buf, tag_output_buf, LML_TAG_OUTPUT_BUF_LEN, 1);                                                                          lml_tag_def[k].tag_name, tag_param_buf, tag_output_buf, LML_TAG_OUTPUT_BUF_LEN, 1);
447                                                          }                                                          }
448                                                  }                                                  }
449    
450                                                    if (line_width + tag_output_len > width)
451                                                    {
452                                                            CHECK_AND_APPEND_OUTPUT(str_out, buf_len, j, "\n", 1, line_width);
453                                                            new_line = 1;
454                                                            line_width = 0;
455                                                            i--; // redo at current i
456                                                            continue;
457                                                    }
458    
459                                                  CHECK_AND_APPEND_OUTPUT(str_out, buf_len, j, tag_output_buf, tag_output_len, line_width);                                                  CHECK_AND_APPEND_OUTPUT(str_out, buf_len, j, tag_output_buf, tag_output_len, line_width);
460                                                  break;                                                  break;
461                                          default: // tag_name not match                                          default: // tag_name not match


Legend:
Removed lines/characters  
Changed lines/characters
  Added lines/characters

webmaster@leafok.com
ViewVC Help
Powered by ViewVC 1.3.0-beta1