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

Diff of /lbbs/src/menu.c

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

Revision 1.61 by sysadm, Thu May 29 01:21:44 2025 UTC Revision 1.81 by sysadm, Tue Nov 4 14:15:49 2025 UTC
# Line 1  Line 1 
1  /***************************************************************************  /* SPDX-License-Identifier: GPL-3.0-or-later */
2                                                    menu.c  -  description  /*
3                                                           -------------------   * menu
4          Copyright            : (C) 2004-2025 by Leaflet   *   - configurable user interactive menu feature
5          Email                : leaflet@leafok.com   *
6   ***************************************************************************/   * Copyright (C) 2004-2025 by Leaflet <leaflet@leafok.com>
7     */
 /***************************************************************************  
  *                                                                         *  
  *   This program is free software; you can redistribute it and/or modify  *  
  *   it under the terms of the GNU General Public License as published by  *  
  *   the Free Software Foundation; either version 3 of the License, or     *  
  *   (at your option) any later version.                                   *  
  *                                                                         *  
  ***************************************************************************/  
8    
9  #include "bbs.h"  #include "bbs.h"
10  #include "bbs_cmd.h"  #include "bbs_cmd.h"
 #include "user_priv.h"  
11  #include "bbs_cmd.h"  #include "bbs_cmd.h"
12  #include "menu.h"  #include "common.h"
 #include "log.h"  
13  #include "io.h"  #include "io.h"
14    #include "log.h"
15    #include "menu.h"
16  #include "screen.h"  #include "screen.h"
17  #include "common.h"  #include "user_priv.h"
 #include <string.h>  
 #include <stdio.h>  
18  #include <ctype.h>  #include <ctype.h>
 #include <stdlib.h>  
19  #include <errno.h>  #include <errno.h>
20    #include <stdio.h>
21    #include <stdlib.h>
22    #include <string.h>
23  #include <unistd.h>  #include <unistd.h>
 #include <sys/shm.h>  
24  #include <sys/ipc.h>  #include <sys/ipc.h>
25    #include <sys/shm.h>
26    
27  #define MENU_SCREEN_PATH_PREFIX "var/MENU_SCR_"  #define MENU_SCREEN_PATH_PREFIX "var/MENU_SCR_"
28  #define MENU_CONF_DELIM_WITH_SPACE " ,\t\r\n"  #define MENU_CONF_DELIM_WITH_SPACE " ,\t\r\n"
# Line 38  Line 30 
30    
31  #define MENU_SET_RESERVED_LENGTH (sizeof(int16_t) * 4)  #define MENU_SET_RESERVED_LENGTH (sizeof(int16_t) * 4)
32    
33  MENU_SET *p_bbs_menu;  #define MENU_SHMGET_RETRY_LIMIT 10
34    
35    MENU_SET bbs_menu;
36    MENU_SET top10_menu;
37    
38  int load_menu(MENU_SET *p_menu_set, const char *conf_file)  int load_menu(MENU_SET *p_menu_set, const char *conf_file)
39  {  {
# Line 59  int load_menu(MENU_SET *p_menu_set, cons Line 54  int load_menu(MENU_SET *p_menu_set, cons
54          int proj_id;          int proj_id;
55          key_t key;          key_t key;
56          size_t size;          size_t size;
57            int retry_cnt;
58    
59            // Initialize the data structure
60            memset(p_menu_set, 0, sizeof(*p_menu_set));
61    
62          // Use trie_dict to search menu_id by menu name          // Use trie_dict to search menu_id by menu name
63          p_menu_set->p_menu_name_dict = trie_dict_create();          p_menu_set->p_menu_name_dict = trie_dict_create();
# Line 78  int load_menu(MENU_SET *p_menu_set, cons Line 77  int load_menu(MENU_SET *p_menu_set, cons
77    
78          if ((fin = fopen(conf_file, "r")) == NULL)          if ((fin = fopen(conf_file, "r")) == NULL)
79          {          {
80                  log_error("Open %s failed", conf_file);                  log_error("Open %s failed\n", conf_file);
81                  return -2;                  return -2;
82          }          }
83    
84          // Allocate shared memory          // Allocate shared memory
         proj_id = (int)(time(NULL) % getpid());  
         key = ftok(conf_file, proj_id);  
         if (key == -1)  
         {  
                 log_error("ftok(%s %d) error (%d)\n", conf_file, proj_id, errno);  
                 return -2;  
         }  
   
85          size = MENU_SET_RESERVED_LENGTH +          size = MENU_SET_RESERVED_LENGTH +
86                     sizeof(MENU) * MAX_MENUS +                     sizeof(MENU) * MAX_MENUS +
87                     sizeof(MENU_ITEM) * MAX_MENUITEMS +                     sizeof(MENU_ITEM) * MAX_MENUITEMS +
88                     sizeof(MENU_SCREEN) * MAX_MENUS +                     sizeof(MENU_SCREEN) * MAX_MENUS +
89                     MAX_MENU_SCR_BUF_LENGTH * MAX_MENUS;                     MAX_MENU_SCR_BUF_LENGTH * MAX_MENUS;
90          p_menu_set->shmid = shmget(key, size, IPC_CREAT | IPC_EXCL | 0600);  
91          if (p_menu_set->shmid == -1)          proj_id = (int)(time(NULL) % getpid());
92            retry_cnt = 0;
93    
94            do
95          {          {
96                  log_error("shmget(size = %d) error (%d)\n", size, errno);                  key = ftok(conf_file, proj_id + retry_cnt);
97                  return -3;                  if (key == -1)
98          }                  {
99                            log_error("ftok(%s %d) error (%d)\n", conf_file, proj_id + retry_cnt, errno);
100                            return -3;
101                    }
102    
103                    p_menu_set->shmid = shmget(key, size, IPC_CREAT | IPC_EXCL | 0600);
104                    if (p_menu_set->shmid == -1)
105                    {
106                            if (errno != EEXIST || retry_cnt + 1 >= MENU_SHMGET_RETRY_LIMIT)
107                            {
108                                    log_error("shmget(conf_file=%s, size=%d) error (%d) %d times\n",
109                                                      conf_file, size, errno, retry_cnt + 1);
110                                    return -3;
111                            }
112                            log_error("shmget(conf_file=%s, proj_id=%d, key=0x%x, size=%d) error (%d), retry ...\n",
113                                              conf_file, proj_id + retry_cnt, key, size, errno);
114                            retry_cnt++;
115                    }
116            } while (p_menu_set->shmid == -1);
117    
118          p_menu_set->p_reserved = shmat(p_menu_set->shmid, NULL, 0);          p_menu_set->p_reserved = shmat(p_menu_set->shmid, NULL, 0);
119          if (p_menu_set->p_reserved == (void *)-1)          if (p_menu_set->p_reserved == (void *)-1)
120          {          {
121                  log_error("shmat() error (%d)\n", errno);                  log_error("shmat() error (%d)\n", errno);
122                  return -3;                  return -3;
123          }          }
124          p_menu_set->p_menu_pool = p_menu_set->p_reserved + MENU_SET_RESERVED_LENGTH;  
125          p_menu_set->p_menu_item_pool = p_menu_set->p_menu_pool + sizeof(MENU) * MAX_MENUS;          p_menu_set->p_menu_pool = (char *)(p_menu_set->p_reserved) + MENU_SET_RESERVED_LENGTH;
126          p_menu_set->p_menu_screen_pool = p_menu_set->p_menu_item_pool + sizeof(MENU_ITEM) * MAX_MENUITEMS;          p_menu_set->p_menu_item_pool = (char *)(p_menu_set->p_menu_pool) + sizeof(MENU) * MAX_MENUS;
127          p_menu_set->p_menu_screen_buf = p_menu_set->p_menu_screen_pool + sizeof(MENU_SCREEN) * MAX_MENUS;          p_menu_set->p_menu_screen_pool = (char *)(p_menu_set->p_menu_item_pool) + sizeof(MENU_ITEM) * MAX_MENUITEMS;
128            p_menu_set->p_menu_screen_buf = (char *)(p_menu_set->p_menu_screen_pool) + sizeof(MENU_SCREEN) * MAX_MENUS;
129          p_menu_set->p_menu_screen_buf_free = p_menu_set->p_menu_screen_buf;          p_menu_set->p_menu_screen_buf_free = p_menu_set->p_menu_screen_buf;
130    
131          p_menu_set->menu_count = 0;          p_menu_set->menu_count = 0;
# Line 119  int load_menu(MENU_SET *p_menu_set, cons Line 133  int load_menu(MENU_SET *p_menu_set, cons
133          p_menu_set->menu_screen_count = 0;          p_menu_set->menu_screen_count = 0;
134          p_menu_set->choose_step = 0;          p_menu_set->choose_step = 0;
135          p_menu_set->menu_id_path[0] = 0;          p_menu_set->menu_id_path[0] = 0;
136            p_menu_set->menu_item_pos[0] = 0;
137            p_menu_set->allow_exit = 0;
138    
139          while (fgets(buffer, sizeof(buffer), fin))          while (fgets(buffer, sizeof(buffer), fin))
140          {          {
# Line 161  int load_menu(MENU_SET *p_menu_set, cons Line 177  int load_menu(MENU_SET *p_menu_set, cons
177                                  p_menu->title.show = 0;                                  p_menu->title.show = 0;
178                                  p_menu->screen_show = 0;                                  p_menu->screen_show = 0;
179                                  p_menu->page_item_limit = 0;                                  p_menu->page_item_limit = 0;
180                                    p_menu->use_filter = 0;
181                                    p_menu->filter_handler = NULL;
182    
183                                  q = strtok_r(NULL, MENU_CONF_DELIM_WITH_SPACE, &saveptr);                                  q = strtok_r(NULL, MENU_CONF_DELIM_WITH_SPACE, &saveptr);
184                                  if (q == NULL)                                  if (q == NULL)
# Line 653  int load_menu(MENU_SET *p_menu_set, cons Line 671  int load_menu(MENU_SET *p_menu_set, cons
671                                                          return -1;                                                          return -1;
672                                                  }                                                  }
673                                          }                                          }
674                                            else if (strcmp(p, "use_filter") == 0)
675                                            {
676                                                    p_menu->use_filter = 1;
677    
678                                                    // Check syntax
679                                                    q = strtok_r(NULL, MENU_CONF_DELIM_WITH_SPACE, &saveptr);
680                                                    if (q != NULL)
681                                                    {
682                                                            log_error("Unknown extra content in menu config line %d\n", fin_line);
683                                                            return -1;
684                                                    }
685                                            }
686                                  }                                  }
687                          }                          }
688                          else // BEGIN of menu screen                          else // BEGIN of menu screen
# Line 772  int load_menu(MENU_SET *p_menu_set, cons Line 802  int load_menu(MENU_SET *p_menu_set, cons
802                          log_error("Undefined menu screen [%s]\n", p);                          log_error("Undefined menu screen [%s]\n", p);
803                          return -1;                          return -1;
804                  }                  }
805    
806                    // Set menu->filter_handler of each menu pointing to filter
807                    if (p_menu->use_filter == 1)
808                    {
809                            if ((p_menu->filter_handler = get_cmd_handler(p_menu->name)) == NULL)
810                            {
811                                    log_error("Undefined menu filter handler [%s]\n", p_menu->name);
812                                    return -1;
813                            }
814                    }
815          }          }
816    
817          for (menu_item_id = 0; menu_item_id < p_menu_set->menu_item_count; menu_item_id++)          for (menu_item_id = 0; menu_item_id < p_menu_set->menu_item_count; menu_item_id++)
# Line 956  int display_menu(MENU_SET *p_menu_set) Line 996  int display_menu(MENU_SET *p_menu_set)
996                          p_menu_set->choose_step--;                          p_menu_set->choose_step--;
997                          return REDRAW;                          return REDRAW;
998                  }                  }
999                  return EXITBBS;                  return EXITMENU;
1000            }
1001    
1002            if (p_menu->item_count <= 0) // empty menu
1003            {
1004                    moveto(p_menu->screen_row, p_menu->screen_col);
1005                    clrtoeol();
1006                    prints("没有可选项");
1007                    press_any_key();
1008                    return -1;
1009          }          }
1010    
1011          menu_item_pos = p_menu_set->menu_item_pos[p_menu_set->choose_step];          menu_item_pos = p_menu_set->menu_item_pos[p_menu_set->choose_step];
# Line 965  int display_menu(MENU_SET *p_menu_set) Line 1014  int display_menu(MENU_SET *p_menu_set)
1014          if (p_menu_item == NULL)          if (p_menu_item == NULL)
1015          {          {
1016                  log_error("get_menu_item_by_id(%d) return NULL pointer\n", menu_item_id);                  log_error("get_menu_item_by_id(%d) return NULL pointer\n", menu_item_id);
1017                  menu_item_pos = 0;                  return EXITMENU;
1018          }          }
1019    
1020          if (menu_item_pos > 0 &&          if (menu_item_pos > 0 &&
1021                  checkpriv(&BBS_priv, 0, p_menu_item->priv) != 0 &&                  !(p_menu->use_filter ? (p_menu->filter_handler((void *)p_menu_item) == 0)
1022                  checklevel2(&BBS_priv, p_menu_item->level))                                                           : (checkpriv(&BBS_priv, 0, p_menu_item->priv) == 0 ||
1023                                                                    checklevel2(&BBS_priv, p_menu_item->level) == 0)))
1024          {          {
1025                  menu_selectable = 1;                  menu_selectable = 1;
1026          }          }
# Line 991  int display_menu(MENU_SET *p_menu_set) Line 1041  int display_menu(MENU_SET *p_menu_set)
1041    
1042                  p_menu_set->menu_item_page_id[menu_item_pos] = page_id;                  p_menu_set->menu_item_page_id[menu_item_pos] = page_id;
1043    
1044                  if (checkpriv(&BBS_priv, 0, p_menu_item->priv) == 0 || checklevel2(&BBS_priv, p_menu_item->level) == 0)                  if (p_menu->use_filter ? (p_menu->filter_handler((void *)p_menu_item) == 0)
1045                                                               : (checkpriv(&BBS_priv, 0, p_menu_item->priv) == 0 ||
1046                                                                      checklevel2(&BBS_priv, p_menu_item->level) == 0))
1047                  {                  {
1048                          p_menu_set->menu_item_display[menu_item_pos] = 0;                          p_menu_set->menu_item_display[menu_item_pos] = 0;
1049                          p_menu_set->menu_item_r_row[menu_item_pos] = 0;                          p_menu_set->menu_item_r_row[menu_item_pos] = 0;
# Line 1025  int display_menu(MENU_SET *p_menu_set) Line 1077  int display_menu(MENU_SET *p_menu_set)
1077    
1078          if (!menu_selectable)          if (!menu_selectable)
1079          {          {
1080                  log_error("No selectable menu item in current menu (%s)\n", p_menu->name);                  moveto(p_menu->screen_row, p_menu->screen_col);
1081                    clrtoeol();
1082                    prints("没有可选项");
1083                    press_any_key();
1084                  return -1;                  return -1;
1085          }          }
1086    
# Line 1070  int menu_control(MENU_SET *p_menu_set, i Line 1125  int menu_control(MENU_SET *p_menu_set, i
1125    
1126          if (p_menu->item_count == 0)          if (p_menu->item_count == 0)
1127          {          {
1128    #ifdef _DEBUG
1129                  log_error("Empty menu (%s)\n", p_menu->name);                  log_error("Empty menu (%s)\n", p_menu->name);
1130    #endif
1131                  if (p_menu_set->choose_step > 0)                  if (p_menu_set->choose_step > 0)
1132                  {                  {
1133                          p_menu_set->choose_step--;                          p_menu_set->choose_step--;
# Line 1094  int menu_control(MENU_SET *p_menu_set, i Line 1151  int menu_control(MENU_SET *p_menu_set, i
1151          switch (key)          switch (key)
1152          {          {
1153          case CR:          case CR:
                 igetch_reset();  
1154          case KEY_RIGHT:          case KEY_RIGHT:
1155                  if (p_menu_item->submenu)                  if (p_menu_item->submenu)
1156                  {                  {
# Line 1121  int menu_control(MENU_SET *p_menu_set, i Line 1177  int menu_control(MENU_SET *p_menu_set, i
1177                  if (p_menu_set->choose_step > 0)                  if (p_menu_set->choose_step > 0)
1178                  {                  {
1179                          p_menu_set->choose_step--;                          p_menu_set->choose_step--;
1180                          if (p_menu_set->choose_step == 0)                          return REDRAW;
                         {  
                                 return REDRAW;  
                         }  
                         if (display_menu(p_menu_set) != 0)  
                         {  
                                 return menu_control(p_menu_set, KEY_LEFT);  
                         }  
1181                  }                  }
1182                  else                  else
1183                  {                  {
1184                            if (p_menu_set->allow_exit)
1185                            {
1186                                    return EXITMENU;
1187                            }
1188    
1189                          display_menu_cursor(p_menu_set, 0);                          display_menu_cursor(p_menu_set, 0);
1190                          menu_item_pos = p_menu->item_count - 1;                          menu_item_pos = p_menu->item_count - 1;
1191                          while (menu_item_pos >= 0)                          while (menu_item_pos >= 0)
# Line 1339  int unload_menu(MENU_SET *p_menu_set) Line 1393  int unload_menu(MENU_SET *p_menu_set)
1393          return 0;          return 0;
1394  }  }
1395    
1396    int get_menu_shm_readonly(MENU_SET *p_menu_set)
1397    {
1398            void *p_shm;
1399    
1400            p_shm = shmat(p_menu_set->shmid, NULL, SHM_RDONLY);
1401            if (p_shm == (void *)-1)
1402            {
1403                    log_error("shmat(menu_shm shmid = %d) error (%d)\n", p_menu_set->shmid, errno);
1404                    return -1;
1405            }
1406    
1407            p_menu_set->p_reserved = p_shm;
1408            p_menu_set->p_menu_pool = (char *)(p_menu_set->p_reserved) + MENU_SET_RESERVED_LENGTH;
1409            p_menu_set->p_menu_item_pool = (char *)(p_menu_set->p_menu_pool) + sizeof(MENU) * MAX_MENUS;
1410            p_menu_set->p_menu_screen_pool = (char *)(p_menu_set->p_menu_item_pool) + sizeof(MENU_ITEM) * MAX_MENUITEMS;
1411            p_menu_set->p_menu_screen_buf = (char *)(p_menu_set->p_menu_screen_pool) + sizeof(MENU_SCREEN) * MAX_MENUS;
1412            p_menu_set->p_menu_screen_buf_free = p_menu_set->p_menu_screen_buf;
1413    
1414            p_menu_set->choose_step = 0;
1415            p_menu_set->menu_id_path[0] = 0;
1416            p_menu_set->menu_item_pos[0] = 0;
1417    
1418            return 0;
1419    }
1420    
1421  int set_menu_shm_readonly(MENU_SET *p_menu_set)  int set_menu_shm_readonly(MENU_SET *p_menu_set)
1422  {  {
1423          void *p_shm;          void *p_shm;
# Line 1350  int set_menu_shm_readonly(MENU_SET *p_me Line 1429  int set_menu_shm_readonly(MENU_SET *p_me
1429                  log_error("shmat(menu_shm shmid = %d) error (%d)\n", p_menu_set->shmid, errno);                  log_error("shmat(menu_shm shmid = %d) error (%d)\n", p_menu_set->shmid, errno);
1430                  return -1;                  return -1;
1431          }          }
1432            
1433          p_menu_set->p_reserved = p_shm;          p_menu_set->p_reserved = p_shm;
1434    
1435          return 0;          return 0;


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

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