/[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.56 by sysadm, Sun May 25 06:53:29 2025 UTC Revision 1.90 by sysadm, Wed Nov 19 15:44: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  Leaflet <leaflet@leafok.com>
7     */
8  /***************************************************************************  
9   *                                                                         *  #ifdef HAVE_CONFIG_H
10   *   This program is free software; you can redistribute it and/or modify  *  #include "config.h"
11   *   it under the terms of the GNU General Public License as published by  *  #endif
  *   the Free Software Foundation; either version 3 of the License, or     *  
  *   (at your option) any later version.                                   *  
  *                                                                         *  
  ***************************************************************************/  
12    
13  #include "bbs.h"  #include "bbs.h"
14  #include "bbs_cmd.h"  #include "bbs_cmd.h"
 #include "user_priv.h"  
15  #include "bbs_cmd.h"  #include "bbs_cmd.h"
16  #include "menu.h"  #include "common.h"
 #include "log.h"  
17  #include "io.h"  #include "io.h"
18    #include "log.h"
19    #include "menu.h"
20  #include "screen.h"  #include "screen.h"
21  #include "common.h"  #include "user_priv.h"
 #include <string.h>  
 #include <stdio.h>  
22  #include <ctype.h>  #include <ctype.h>
 #include <stdlib.h>  
23  #include <errno.h>  #include <errno.h>
24    #include <fcntl.h>
25    #include <stdio.h>
26    #include <stdlib.h>
27    #include <string.h>
28  #include <unistd.h>  #include <unistd.h>
29  #include <sys/shm.h>  #include <sys/mman.h>
30  #include <sys/ipc.h>  #include <sys/stat.h>
31    
32  #define MENU_SCREEN_PATH_PREFIX "var/MENU_SCR_"  enum _menu_constant_t
33  #define MENU_CONF_DELIM_WITH_SPACE " ,\t\r\n"  {
34  #define MENU_CONF_DELIM_WITHOUT_SPACE "\r\n"          MENU_SET_RESERVED_LENGTH = sizeof(int16_t) * 4,
35    };
36    
37  #define MENU_SET_RESERVED_LENGTH (sizeof(int16_t) * 4)  static const char MENU_CONF_DELIM_WITH_SPACE[] = " ,\t\r\n";
38    static const char MENU_CONF_DELIM_WITHOUT_SPACE[] = "\r\n";
39    
40  MENU_SET *p_bbs_menu;  MENU_SET bbs_menu;
41    MENU_SET top10_menu;
42    
43  int load_menu(MENU_SET *p_menu_set, const char *conf_file)  int load_menu(MENU_SET *p_menu_set, const char *conf_file)
44  {  {
45            char filepath[FILE_PATH_LEN];
46            int fd;
47            size_t size;
48            void *p_shm;
49          FILE *fin;          FILE *fin;
50          int fin_line = 0;          int fin_line = 0;
51          char buffer[LINE_BUFFER_LEN];          char buffer[LINE_BUFFER_LEN];
# Line 56  int load_menu(MENU_SET *p_menu_set, cons Line 60  int load_menu(MENU_SET *p_menu_set, cons
60          MENU_ID menu_id;          MENU_ID menu_id;
61          MENU_ITEM_ID menu_item_id;          MENU_ITEM_ID menu_item_id;
62          MENU_SCREEN_ID screen_id;          MENU_SCREEN_ID screen_id;
63          int proj_id;  
64          key_t key;          if (p_menu_set == NULL || conf_file == NULL)
65          size_t size;          {
66                    log_error("NULL pointer error\n");
67                    return -1;
68            }
69    
70            // Initialize the data structure
71            memset(p_menu_set, 0, sizeof(*p_menu_set));
72    
73          // Use trie_dict to search menu_id by menu name          // Use trie_dict to search menu_id by menu name
74          p_menu_set->p_menu_name_dict = trie_dict_create();          p_menu_set->p_menu_name_dict = trie_dict_create();
75          if (p_menu_set->p_menu_name_dict == NULL)          if (p_menu_set->p_menu_name_dict == NULL)
76          {          {
77                  log_error("trie_dict_create() error\n");                  log_error("trie_dict_create() error\n");
78                  return -3;                  return -1;
79          }          }
80    
81          // Use trie_dict to search screen_id by menu screen name          // Use trie_dict to search screen_id by menu screen name
# Line 73  int load_menu(MENU_SET *p_menu_set, cons Line 83  int load_menu(MENU_SET *p_menu_set, cons
83          if (p_menu_set->p_menu_screen_dict == NULL)          if (p_menu_set->p_menu_screen_dict == NULL)
84          {          {
85                  log_error("trie_dict_create() error\n");                  log_error("trie_dict_create() error\n");
86                  return -3;                  return -1;
87          }          }
88    
89          if ((fin = fopen(conf_file, "r")) == NULL)          if ((fin = fopen(conf_file, "r")) == NULL)
90          {          {
91                  log_error("Open %s failed", conf_file);                  log_error("Open %s failed\n", conf_file);
92                  return -2;                  return -2;
93          }          }
94    
95          // 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;  
         }  
   
96          size = MENU_SET_RESERVED_LENGTH +          size = MENU_SET_RESERVED_LENGTH +
97                     sizeof(MENU) * MAX_MENUS +                     sizeof(MENU) * MAX_MENUS +
98                     sizeof(MENU_ITEM) * MAX_MENUITEMS +                     sizeof(MENU_ITEM) * MAX_MENUITEMS +
99                     sizeof(MENU_SCREEN) * MAX_MENUS +                     sizeof(MENU_SCREEN) * MAX_MENUS +
100                     MAX_MENU_SCR_BUF_LENGTH * MAX_MENUS;                     MAX_MENU_SCR_BUF_LENGTH * MAX_MENUS;
101          p_menu_set->shmid = shmget(key, size, IPC_CREAT | IPC_EXCL | 0600);  
102          if (p_menu_set->shmid == -1)          strncpy(filepath, conf_file, sizeof(filepath) - 1);
103            filepath[sizeof(filepath) - 1] = '\0';
104            snprintf(p_menu_set->shm_name, sizeof(p_menu_set->shm_name), "/MENU_SHM_%s", basename(filepath));
105    
106            if (shm_unlink(p_menu_set->shm_name) == -1 && errno != ENOENT)
107          {          {
108                  log_error("shmget(size = %d) error (%d)\n", size, errno);                  log_error("shm_unlink(%s) error (%d)\n", p_menu_set->shm_name, errno);
109                  return -3;                  return -2;
110          }          }
111          p_menu_set->p_reserved = shmat(p_menu_set->shmid, NULL, 0);  
112          if (p_menu_set->p_reserved == (void *)-1)          if ((fd = shm_open(p_menu_set->shm_name, O_CREAT | O_EXCL | O_RDWR, 0600)) == -1)
113          {          {
114                  log_error("shmat() error (%d)\n", errno);                  log_error("shm_open(%s) error (%d)\n", p_menu_set->shm_name, errno);
115                  return -3;                  return -2;
116          }          }
117          p_menu_set->p_menu_pool = p_menu_set->p_reserved + MENU_SET_RESERVED_LENGTH;          if (ftruncate(fd, (off_t)size) == -1)
118          p_menu_set->p_menu_item_pool = p_menu_set->p_menu_pool + sizeof(MENU) * MAX_MENUS;          {
119          p_menu_set->p_menu_screen_pool = p_menu_set->p_menu_item_pool + sizeof(MENU_ITEM) * MAX_MENUITEMS;                  log_error("ftruncate(size=%d) error (%d)\n", size, errno);
120          p_menu_set->p_menu_screen_buf = p_menu_set->p_menu_screen_pool + sizeof(MENU_SCREEN) * MAX_MENUS;                  close(fd);
121                    return -2;
122            }
123    
124            p_shm = mmap(NULL, size, PROT_READ | PROT_WRITE, MAP_SHARED, fd, 0L);
125            if (p_shm == MAP_FAILED)
126            {
127                    log_error("mmap() error (%d)\n", errno);
128                    close(fd);
129                    return -2;
130            }
131    
132            if (close(fd) < 0)
133            {
134                    log_error("close(fd) error (%d)\n", errno);
135                    return -1;
136            }
137    
138            p_menu_set->shm_size = size;
139            p_menu_set->p_reserved = p_shm;
140    
141            p_menu_set->p_menu_pool = (char *)(p_menu_set->p_reserved) + MENU_SET_RESERVED_LENGTH;
142            p_menu_set->p_menu_item_pool = (char *)(p_menu_set->p_menu_pool) + sizeof(MENU) * MAX_MENUS;
143            p_menu_set->p_menu_screen_pool = (char *)(p_menu_set->p_menu_item_pool) + sizeof(MENU_ITEM) * MAX_MENUITEMS;
144            p_menu_set->p_menu_screen_buf = (char *)(p_menu_set->p_menu_screen_pool) + sizeof(MENU_SCREEN) * MAX_MENUS;
145          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;
146    
147          p_menu_set->menu_count = 0;          p_menu_set->menu_count = 0;
# Line 119  int load_menu(MENU_SET *p_menu_set, cons Line 149  int load_menu(MENU_SET *p_menu_set, cons
149          p_menu_set->menu_screen_count = 0;          p_menu_set->menu_screen_count = 0;
150          p_menu_set->choose_step = 0;          p_menu_set->choose_step = 0;
151          p_menu_set->menu_id_path[0] = 0;          p_menu_set->menu_id_path[0] = 0;
152            p_menu_set->menu_item_pos[0] = 0;
153            p_menu_set->allow_exit = 0;
154    
155          while (fgets(buffer, sizeof(buffer), fin))          while (fgets(buffer, sizeof(buffer), fin))
156          {          {
# Line 161  int load_menu(MENU_SET *p_menu_set, cons Line 193  int load_menu(MENU_SET *p_menu_set, cons
193                                  p_menu->title.show = 0;                                  p_menu->title.show = 0;
194                                  p_menu->screen_show = 0;                                  p_menu->screen_show = 0;
195                                  p_menu->page_item_limit = 0;                                  p_menu->page_item_limit = 0;
196                                    p_menu->use_filter = 0;
197                                    p_menu->filter_handler = NULL;
198    
199                                  q = strtok_r(NULL, MENU_CONF_DELIM_WITH_SPACE, &saveptr);                                  q = strtok_r(NULL, MENU_CONF_DELIM_WITH_SPACE, &saveptr);
200                                  if (q == NULL)                                  if (q == NULL)
# Line 169  int load_menu(MENU_SET *p_menu_set, cons Line 203  int load_menu(MENU_SET *p_menu_set, cons
203                                          return -1;                                          return -1;
204                                  }                                  }
205                                  p = q;                                  p = q;
206                                  while (isalnum(*q) || *q == '_' || *q == '-')                                  while (isalnum((int)*q) || *q == '_' || *q == '-')
207                                  {                                  {
208                                          q++;                                          q++;
209                                  }                                  }
# Line 252  int load_menu(MENU_SET *p_menu_set, cons Line 286  int load_menu(MENU_SET *p_menu_set, cons
286                                                  else                                                  else
287                                                  {                                                  {
288                                                          q = p;                                                          q = p;
289                                                          while (isalnum(*q) || *q == '_' || *q == '-')                                                          while (isalnum((int)*q) || *q == '_' || *q == '-')
290                                                          {                                                          {
291                                                                  q++;                                                                  q++;
292                                                          }                                                          }
# Line 279  int load_menu(MENU_SET *p_menu_set, cons Line 313  int load_menu(MENU_SET *p_menu_set, cons
313                                                          return -1;                                                          return -1;
314                                                  }                                                  }
315                                                  p = q;                                                  p = q;
316                                                  while (isdigit(*q))                                                  while (isdigit((int)*q))
317                                                  {                                                  {
318                                                          q++;                                                          q++;
319                                                  }                                                  }
# Line 298  int load_menu(MENU_SET *p_menu_set, cons Line 332  int load_menu(MENU_SET *p_menu_set, cons
332                                                          return -1;                                                          return -1;
333                                                  }                                                  }
334                                                  p = q;                                                  p = q;
335                                                  while (isdigit(*q))                                                  while (isdigit((int)*q))
336                                                  {                                                  {
337                                                          q++;                                                          q++;
338                                                  }                                                  }
# Line 317  int load_menu(MENU_SET *p_menu_set, cons Line 351  int load_menu(MENU_SET *p_menu_set, cons
351                                                          return -1;                                                          return -1;
352                                                  }                                                  }
353                                                  p = q;                                                  p = q;
354                                                  while (isdigit(*q))                                                  while (isdigit((int)*q))
355                                                  {                                                  {
356                                                          q++;                                                          q++;
357                                                  }                                                  }
# Line 336  int load_menu(MENU_SET *p_menu_set, cons Line 370  int load_menu(MENU_SET *p_menu_set, cons
370                                                          return -1;                                                          return -1;
371                                                  }                                                  }
372                                                  p = q;                                                  p = q;
373                                                  while (isdigit(*q))                                                  while (isdigit((int)*q))
374                                                  {                                                  {
375                                                          q++;                                                          q++;
376                                                  }                                                  }
# Line 441  int load_menu(MENU_SET *p_menu_set, cons Line 475  int load_menu(MENU_SET *p_menu_set, cons
475                                                          return -1;                                                          return -1;
476                                                  }                                                  }
477                                                  p = q;                                                  p = q;
478                                                  while (isdigit(*q))                                                  while (isdigit((int)*q))
479                                                  {                                                  {
480                                                          q++;                                                          q++;
481                                                  }                                                  }
# Line 460  int load_menu(MENU_SET *p_menu_set, cons Line 494  int load_menu(MENU_SET *p_menu_set, cons
494                                                          return -1;                                                          return -1;
495                                                  }                                                  }
496                                                  p = q;                                                  p = q;
497                                                  while (isdigit(*q))                                                  while (isdigit((int)*q))
498                                                  {                                                  {
499                                                          q++;                                                          q++;
500                                                  }                                                  }
# Line 528  int load_menu(MENU_SET *p_menu_set, cons Line 562  int load_menu(MENU_SET *p_menu_set, cons
562                                                          return -1;                                                          return -1;
563                                                  }                                                  }
564                                                  p = q;                                                  p = q;
565                                                  while (isdigit(*q))                                                  while (isdigit((int)*q))
566                                                  {                                                  {
567                                                          q++;                                                          q++;
568                                                  }                                                  }
# Line 547  int load_menu(MENU_SET *p_menu_set, cons Line 581  int load_menu(MENU_SET *p_menu_set, cons
581                                                          return -1;                                                          return -1;
582                                                  }                                                  }
583                                                  p = q;                                                  p = q;
584                                                  while (isdigit(*q))                                                  while (isdigit((int)*q))
585                                                  {                                                  {
586                                                          q++;                                                          q++;
587                                                  }                                                  }
# Line 566  int load_menu(MENU_SET *p_menu_set, cons Line 600  int load_menu(MENU_SET *p_menu_set, cons
600                                                          return -1;                                                          return -1;
601                                                  }                                                  }
602                                                  p = q;                                                  p = q;
603                                                  while (isalnum(*q) || *q == '_' || *q == '-')                                                  while (isalnum((int)*q) || *q == '_' || *q == '-')
604                                                  {                                                  {
605                                                          q++;                                                          q++;
606                                                  }                                                  }
# Line 596  int load_menu(MENU_SET *p_menu_set, cons Line 630  int load_menu(MENU_SET *p_menu_set, cons
630                                                          return -1;                                                          return -1;
631                                                  }                                                  }
632                                                  p = q;                                                  p = q;
633                                                  while (isdigit(*q))                                                  while (isdigit((int)*q))
634                                                  {                                                  {
635                                                          q++;                                                          q++;
636                                                  }                                                  }
# Line 615  int load_menu(MENU_SET *p_menu_set, cons Line 649  int load_menu(MENU_SET *p_menu_set, cons
649                                                          return -1;                                                          return -1;
650                                                  }                                                  }
651                                                  p = q;                                                  p = q;
652                                                  while (isdigit(*q))                                                  while (isdigit((int)*q))
653                                                  {                                                  {
654                                                          q++;                                                          q++;
655                                                  }                                                  }
# Line 634  int load_menu(MENU_SET *p_menu_set, cons Line 668  int load_menu(MENU_SET *p_menu_set, cons
668                                                          return -1;                                                          return -1;
669                                                  }                                                  }
670                                                  p = q;                                                  p = q;
671                                                  while (isdigit(*q))                                                  while (isdigit((int)*q))
672                                                  {                                                  {
673                                                          q++;                                                          q++;
674                                                  }                                                  }
# Line 653  int load_menu(MENU_SET *p_menu_set, cons Line 687  int load_menu(MENU_SET *p_menu_set, cons
687                                                          return -1;                                                          return -1;
688                                                  }                                                  }
689                                          }                                          }
690                                            else if (strcmp(p, "use_filter") == 0)
691                                            {
692                                                    p_menu->use_filter = 1;
693    
694                                                    // Check syntax
695                                                    q = strtok_r(NULL, MENU_CONF_DELIM_WITH_SPACE, &saveptr);
696                                                    if (q != NULL)
697                                                    {
698                                                            log_error("Unknown extra content in menu config line %d\n", fin_line);
699                                                            return -1;
700                                                    }
701                                            }
702                                  }                                  }
703                          }                          }
704                          else // BEGIN of menu screen                          else // BEGIN of menu screen
# Line 668  int load_menu(MENU_SET *p_menu_set, cons Line 714  int load_menu(MENU_SET *p_menu_set, cons
714                                  p_screen = get_menu_screen_by_id(p_menu_set, screen_id);                                  p_screen = get_menu_screen_by_id(p_menu_set, screen_id);
715    
716                                  q = p;                                  q = p;
717                                  while (isalnum(*q) || *q == '_' || *q == '-')                                  while (isalnum((int)*q) || *q == '_' || *q == '-')
718                                  {                                  {
719                                          q++;                                          q++;
720                                  }                                  }
# Line 772  int load_menu(MENU_SET *p_menu_set, cons Line 818  int load_menu(MENU_SET *p_menu_set, cons
818                          log_error("Undefined menu screen [%s]\n", p);                          log_error("Undefined menu screen [%s]\n", p);
819                          return -1;                          return -1;
820                  }                  }
821    
822                    // Set menu->filter_handler of each menu pointing to filter
823                    if (p_menu->use_filter == 1)
824                    {
825                            if ((p_menu->filter_handler = get_cmd_handler(p_menu->name)) == NULL)
826                            {
827                                    log_error("Undefined menu filter handler [%s]\n", p_menu->name);
828                                    return -1;
829                            }
830                    }
831          }          }
832    
833          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 807  int load_menu(MENU_SET *p_menu_set, cons Line 863  int load_menu(MENU_SET *p_menu_set, cons
863          return 0;          return 0;
864  }  }
865    
866  static int display_menu_cursor(MENU_SET *p_menu_set, int show)  int display_menu_cursor(MENU_SET *p_menu_set, int show)
867  {  {
868          MENU_ID menu_id;          MENU_ID menu_id;
869          MENU_ITEM_ID menu_item_id;          MENU_ITEM_ID menu_item_id;
# Line 864  static int display_menu_current_page(MEN Line 920  static int display_menu_current_page(MEN
920          {          {
921                  if (p_menu->title.row == 0 && p_menu->title.col == 0)                  if (p_menu->title.row == 0 && p_menu->title.col == 0)
922                  {                  {
923                          show_top(p_menu->title.text);                          show_top(p_menu->title.text, BBS_name, "");
924                  }                  }
925                  else                  else
926                  {                  {
# Line 956  int display_menu(MENU_SET *p_menu_set) Line 1012  int display_menu(MENU_SET *p_menu_set)
1012                          p_menu_set->choose_step--;                          p_menu_set->choose_step--;
1013                          return REDRAW;                          return REDRAW;
1014                  }                  }
1015                  return EXITBBS;                  return EXITMENU;
1016            }
1017    
1018            if (p_menu->item_count <= 0) // empty menu
1019            {
1020                    moveto(p_menu->screen_row, p_menu->screen_col);
1021                    clrtoeol();
1022                    prints("没有可选项");
1023                    press_any_key();
1024                    return -1;
1025          }          }
1026    
1027          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 1030  int display_menu(MENU_SET *p_menu_set)
1030          if (p_menu_item == NULL)          if (p_menu_item == NULL)
1031          {          {
1032                  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);
1033                  menu_item_pos = 0;                  return EXITMENU;
1034          }          }
1035    
1036          if (menu_item_pos > 0 &&          if (menu_item_pos > 0 &&
1037                  checkpriv(&BBS_priv, 0, p_menu_item->priv) != 0 &&                  !(p_menu->use_filter ? (p_menu->filter_handler((void *)p_menu_item) == 0)
1038                  checklevel2(&BBS_priv, p_menu_item->level))                                                           : (checkpriv(&BBS_priv, 0, p_menu_item->priv) == 0 ||
1039                                                                    checklevel2(&BBS_priv, p_menu_item->level) == 0)))
1040          {          {
1041                  menu_selectable = 1;                  menu_selectable = 1;
1042          }          }
# Line 991  int display_menu(MENU_SET *p_menu_set) Line 1057  int display_menu(MENU_SET *p_menu_set)
1057    
1058                  p_menu_set->menu_item_page_id[menu_item_pos] = page_id;                  p_menu_set->menu_item_page_id[menu_item_pos] = page_id;
1059    
1060                  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)
1061                                                               : (checkpriv(&BBS_priv, 0, p_menu_item->priv) == 0 ||
1062                                                                      checklevel2(&BBS_priv, p_menu_item->level) == 0))
1063                  {                  {
1064                          p_menu_set->menu_item_display[menu_item_pos] = 0;                          p_menu_set->menu_item_display[menu_item_pos] = 0;
1065                          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 1093  int display_menu(MENU_SET *p_menu_set)
1093    
1094          if (!menu_selectable)          if (!menu_selectable)
1095          {          {
1096                  log_error("No selectable menu item in current menu (%s)\n", p_menu->name);                  moveto(p_menu->screen_row, p_menu->screen_col);
1097                    clrtoeol();
1098                    prints("没有可选项");
1099                    press_any_key();
1100                  return -1;                  return -1;
1101          }          }
1102    
# Line 1070  int menu_control(MENU_SET *p_menu_set, i Line 1141  int menu_control(MENU_SET *p_menu_set, i
1141    
1142          if (p_menu->item_count == 0)          if (p_menu->item_count == 0)
1143          {          {
1144    #ifdef _DEBUG
1145                  log_error("Empty menu (%s)\n", p_menu->name);                  log_error("Empty menu (%s)\n", p_menu->name);
1146    #endif
1147                  if (p_menu_set->choose_step > 0)                  if (p_menu_set->choose_step > 0)
1148                  {                  {
1149                          p_menu_set->choose_step--;                          p_menu_set->choose_step--;
# Line 1094  int menu_control(MENU_SET *p_menu_set, i Line 1167  int menu_control(MENU_SET *p_menu_set, i
1167          switch (key)          switch (key)
1168          {          {
1169          case CR:          case CR:
                 igetch_reset();  
1170          case KEY_RIGHT:          case KEY_RIGHT:
1171                  if (p_menu_item->submenu)                  if (p_menu_item->submenu)
1172                  {                  {
# Line 1116  int menu_control(MENU_SET *p_menu_set, i Line 1188  int menu_control(MENU_SET *p_menu_set, i
1188                          return ((*(p_menu_item->action_cmd_handler))((void *)(p_menu_item->name)));                          return ((*(p_menu_item->action_cmd_handler))((void *)(p_menu_item->name)));
1189                  }                  }
1190                  break;                  break;
1191            case KEY_ESC:
1192          case KEY_LEFT:          case KEY_LEFT:
1193                  if (p_menu_set->choose_step > 0)                  if (p_menu_set->choose_step > 0)
1194                  {                  {
1195                          p_menu_set->choose_step--;                          p_menu_set->choose_step--;
1196                          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);  
                         }  
1197                  }                  }
1198                  else                  else
1199                  {                  {
1200                            if (p_menu_set->allow_exit)
1201                            {
1202                                    return EXITMENU;
1203                            }
1204    
1205                          display_menu_cursor(p_menu_set, 0);                          display_menu_cursor(p_menu_set, 0);
1206                          menu_item_pos = p_menu->item_count - 1;                          menu_item_pos = p_menu->item_count - 1;
1207                          while (menu_item_pos >= 0)                          while (menu_item_pos >= 0)
# Line 1306  int menu_control(MENU_SET *p_menu_set, i Line 1377  int menu_control(MENU_SET *p_menu_set, i
1377    
1378  int unload_menu(MENU_SET *p_menu_set)  int unload_menu(MENU_SET *p_menu_set)
1379  {  {
1380            if (p_menu_set == NULL)
1381            {
1382                    log_error("NULL pointer error\n");
1383                    return -1;
1384            }
1385    
1386          if (p_menu_set->p_menu_name_dict != NULL)          if (p_menu_set->p_menu_name_dict != NULL)
1387          {          {
1388                  trie_dict_destroy(p_menu_set->p_menu_name_dict);                  trie_dict_destroy(p_menu_set->p_menu_name_dict);
# Line 1320  int unload_menu(MENU_SET *p_menu_set) Line 1397  int unload_menu(MENU_SET *p_menu_set)
1397    
1398          detach_menu_shm(p_menu_set);          detach_menu_shm(p_menu_set);
1399    
1400          if (shmctl(p_menu_set->shmid, IPC_RMID, NULL) == -1)          if (shm_unlink(p_menu_set->shm_name) == -1 && errno != ENOENT)
1401          {          {
1402                  log_error("shmctl(shmid=%d, IPC_RMID) error (%d)\n", p_menu_set->shmid, errno);                  log_error("shm_unlink(%s) error (%d)\n", p_menu_set->shm_name, errno);
1403                  return -1;                  return -2;
1404          }          }
1405    
1406          return 0;          return 0;
1407  }  }
1408    
1409  int set_menu_shm_readonly(MENU_SET *p_menu_set)  int get_menu_shm_readonly(MENU_SET *p_menu_set)
1410  {  {
1411            int fd;
1412          void *p_shm;          void *p_shm;
1413            struct stat sb;
1414            size_t size;
1415    
1416          // Remap shared memory in read-only mode          if (p_menu_set == NULL)
         p_shm = shmat(p_menu_set->shmid, p_menu_set->p_reserved, SHM_RDONLY | SHM_REMAP);  
         if (p_shm == (void *)-1)  
1417          {          {
1418                  log_error("shmat() error (%d)\n", errno);                  log_error("NULL pointer error\n");
1419                  return -1;                  return -1;
1420          }          }
1421            
1422            if ((fd = shm_open(p_menu_set->shm_name, O_RDONLY, 0600)) == -1)
1423            {
1424                    log_error("shm_open(%s) error (%d)\n", p_menu_set->shm_name, errno);
1425                    return -2;
1426            }
1427    
1428            if (fstat(fd, &sb) < 0)
1429            {
1430                    log_error("fstat(fd) error (%d)\n", errno);
1431                    close(fd);
1432                    return -2;
1433            }
1434    
1435            size = (size_t)sb.st_size;
1436    
1437            p_shm = mmap(NULL, size, PROT_READ, MAP_SHARED, fd, 0L);
1438            if (p_shm == MAP_FAILED)
1439            {
1440                    log_error("mmap() error (%d)\n", errno);
1441                    close(fd);
1442                    return -2;
1443            }
1444    
1445            if (close(fd) < 0)
1446            {
1447                    log_error("close(fd) error (%d)\n", errno);
1448                    return -1;
1449            }
1450    
1451            p_menu_set->shm_size = size;
1452          p_menu_set->p_reserved = p_shm;          p_menu_set->p_reserved = p_shm;
1453    
1454            p_menu_set->p_menu_pool = (char *)(p_menu_set->p_reserved) + MENU_SET_RESERVED_LENGTH;
1455            p_menu_set->p_menu_item_pool = (char *)(p_menu_set->p_menu_pool) + sizeof(MENU) * MAX_MENUS;
1456            p_menu_set->p_menu_screen_pool = (char *)(p_menu_set->p_menu_item_pool) + sizeof(MENU_ITEM) * MAX_MENUITEMS;
1457            p_menu_set->p_menu_screen_buf = (char *)(p_menu_set->p_menu_screen_pool) + sizeof(MENU_SCREEN) * MAX_MENUS;
1458            p_menu_set->p_menu_screen_buf_free = p_menu_set->p_menu_screen_buf;
1459    
1460            p_menu_set->choose_step = 0;
1461            p_menu_set->menu_id_path[0] = 0;
1462            p_menu_set->menu_item_pos[0] = 0;
1463    
1464            return 0;
1465    }
1466    
1467    int set_menu_shm_readonly(MENU_SET *p_menu_set)
1468    {
1469            if (p_menu_set == NULL)
1470            {
1471                    log_error("NULL pointer error\n");
1472                    return -1;
1473            }
1474    
1475            if (p_menu_set->p_reserved != NULL && munmap(p_menu_set->p_reserved, p_menu_set->shm_size) < 0)
1476            {
1477                    log_error("munmap() error (%d)\n", errno);
1478                    return -2;
1479            }
1480    
1481            if (get_menu_shm_readonly(p_menu_set) < 0)
1482            {
1483                    log_error("get_menu_shm_readonly() error\n");
1484                    return -3;
1485            }
1486    
1487          return 0;          return 0;
1488  }  }
1489    
1490  int detach_menu_shm(MENU_SET *p_menu_set)  int detach_menu_shm(MENU_SET *p_menu_set)
1491  {  {
1492            if (p_menu_set == NULL)
1493            {
1494                    log_error("NULL pointer error\n");
1495                    return -1;
1496            }
1497    
1498          p_menu_set->menu_count = 0;          p_menu_set->menu_count = 0;
1499          p_menu_set->menu_item_count = 0;          p_menu_set->menu_item_count = 0;
1500          p_menu_set->menu_screen_count = 0;          p_menu_set->menu_screen_count = 0;
# Line 1362  int detach_menu_shm(MENU_SET *p_menu_set Line 1509  int detach_menu_shm(MENU_SET *p_menu_set
1509          p_menu_set->p_menu_name_dict = NULL;          p_menu_set->p_menu_name_dict = NULL;
1510          p_menu_set->p_menu_screen_dict = NULL;          p_menu_set->p_menu_screen_dict = NULL;
1511    
1512          if (p_menu_set->p_reserved != NULL && shmdt(p_menu_set->p_reserved) == -1)          if (p_menu_set->p_reserved != NULL && munmap(p_menu_set->p_reserved, p_menu_set->shm_size) < 0)
1513          {          {
1514                  log_error("shmdt() error (%d)\n", errno);                  log_error("munmap() error (%d)\n", errno);
1515                  return -1;                  return -2;
1516          }          }
1517    
1518          p_menu_set->p_reserved = NULL;          p_menu_set->p_reserved = NULL;


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

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