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

Diff of /lbbs/src/section_list.c

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

Revision 1.32 by sysadm, Tue Jun 17 13:18:55 2025 UTC Revision 1.50 by sysadm, Mon Nov 3 15:05:02 2025 UTC
# Line 14  Line 14 
14   *                                                                         *   *                                                                         *
15   ***************************************************************************/   ***************************************************************************/
16    
 #include "section_list.h"  
17  #include "log.h"  #include "log.h"
18    #include "section_list.h"
19  #include "trie_dict.h"  #include "trie_dict.h"
20    #include "user_list.h"
21    #include <errno.h>
22    #include <signal.h>
23  #include <stdio.h>  #include <stdio.h>
24    #include <stdlib.h>
25  #include <string.h>  #include <string.h>
 #include <signal.h>  
26  #include <unistd.h>  #include <unistd.h>
27  #include <stdlib.h>  #include <sys/ipc.h>
 #include <errno.h>  
28  #include <sys/param.h>  #include <sys/param.h>
29  #include <sys/sem.h>  #include <sys/sem.h>
30  #include <sys/shm.h>  #include <sys/shm.h>
 #include <sys/ipc.h>  
31    
32  #ifdef _SEM_SEMUN_UNDEFINED  #ifdef _SEM_SEMUN_UNDEFINED
33  union semun  union semun
# Line 42  union semun Line 43  union semun
43  #define SECTION_TRY_LOCK_WAIT_TIME 1 // second  #define SECTION_TRY_LOCK_WAIT_TIME 1 // second
44  #define SECTION_TRY_LOCK_TIMES 10  #define SECTION_TRY_LOCK_TIMES 10
45    
46  #define ARTICLE_BLOCK_PER_SHM 400                 // sizeof(ARTICLE_BLOCK) * ARTICLE_BLOCK_PER_SHM is the size of each shm segment to allocate  #define ARTICLE_BLOCK_PER_SHM 1000               // sizeof(ARTICLE_BLOCK) * ARTICLE_BLOCK_PER_SHM is the size of each shm segment to allocate
47  #define ARTICLE_BLOCK_SHM_COUNT_LIMIT 200 // limited by length (8-bit) of proj_id in ftok(path, proj_id)  #define ARTICLE_BLOCK_SHM_COUNT_LIMIT 80 // limited by length (8-bit) of proj_id in ftok(path, proj_id)
48  #define ARTICLE_BLOCK_PER_POOL (ARTICLE_BLOCK_PER_SHM * ARTICLE_BLOCK_SHM_COUNT_LIMIT)  #define ARTICLE_BLOCK_PER_POOL (ARTICLE_BLOCK_PER_SHM * ARTICLE_BLOCK_SHM_COUNT_LIMIT)
49    
50  #define CALCULATE_PAGE_THRESHOLD 100 // Adjust to tune performance of moving topic between sections  #define CALCULATE_PAGE_THRESHOLD 100 // Adjust to tune performance of moving topic between sections
# Line 75  typedef struct article_block_pool_t ARTI Line 76  typedef struct article_block_pool_t ARTI
76    
77  static ARTICLE_BLOCK_POOL *p_article_block_pool = NULL;  static ARTICLE_BLOCK_POOL *p_article_block_pool = NULL;
78    
79  struct section_list_pool_t  SECTION_LIST_POOL *p_section_list_pool = NULL;
 {  
         int shmid;  
         SECTION_LIST sections[BBS_max_section];  
         int section_count;  
         int semid;  
         TRIE_NODE *p_trie_dict_section_by_name;  
         TRIE_NODE *p_trie_dict_section_by_sid;  
 };  
 typedef struct section_list_pool_t SECTION_LIST_POOL;  
   
 static SECTION_LIST_POOL *p_section_list_pool = NULL;  
80    
81  int article_block_init(const char *filename, int block_count)  int article_block_init(const char *filename, int block_count)
82  {  {
# Line 353  ARTICLE *article_block_find_by_aid(int32 Line 343  ARTICLE *article_block_find_by_aid(int32
343          }          }
344    
345          left = 0;          left = 0;
346          right = p_article_block_pool->block_count;          right = p_article_block_pool->block_count - 1;
347    
348          // aid in the range [ head aid of blocks[left], tail aid of blocks[right - 1] ]          // aid in the range [ head aid of blocks[left], tail aid of blocks[right] ]
349          while (left < right - 1)          while (left < right)
350          {          {
351                  // get block offset no less than mid value of left and right block offsets                  // get block offset no less than mid value of left and right block offsets
352                  mid = (left + right) / 2 + (right - left) % 2;                  mid = (left + right) / 2 + (left + right) % 2;
   
                 if (mid >= p_article_block_pool->block_count)  
                 {  
                         log_error("block(mid = %d) is out of boundary\n", mid);  
                         return NULL;  
                 }  
353    
354                  if (aid < p_article_block_pool->p_block[mid]->articles[0].aid)                  if (aid < p_article_block_pool->p_block[mid]->articles[0].aid)
355                  {                  {
356                          right = mid;                          right = mid - 1;
357                  }                  }
358                  else                  else // if (aid >= p_article_block_pool->p_block[mid]->articles[0].aid)
359                  {                  {
360                          left = mid;                          left = mid;
361                  }                  }
# Line 630  SECTION_LIST *section_list_create(int32_ Line 614  SECTION_LIST *section_list_create(int32_
614          p_section = p_section_list_pool->sections + p_section_list_pool->section_count;          p_section = p_section_list_pool->sections + p_section_list_pool->section_count;
615    
616          p_section->sid = sid;          p_section->sid = sid;
617            p_section->ex_menu_tm = 0;
618    
619          strncpy(p_section->sname, sname, sizeof(p_section->sname) - 1);          strncpy(p_section->sname, sname, sizeof(p_section->sname) - 1);
620          p_section->sname[sizeof(p_section->sname) - 1] = '\0';          p_section->sname[sizeof(p_section->sname) - 1] = '\0';
# Line 659  SECTION_LIST *section_list_create(int32_ Line 644  SECTION_LIST *section_list_create(int32_
644          return p_section;          return p_section;
645  }  }
646    
647    int section_list_update(SECTION_LIST *p_section, const char *sname, const char *stitle, const char *master_list)
648    {
649            int64_t index;
650    
651            if (p_section == NULL || sname == NULL || stitle == NULL || master_list == NULL)
652            {
653                    log_error("NULL pointer error\n");
654                    return -1;
655            }
656    
657            index = get_section_index(p_section);
658    
659            strncpy(p_section->sname, sname, sizeof(p_section->sname) - 1);
660            p_section->sname[sizeof(p_section->sname) - 1] = '\0';
661    
662            strncpy(p_section->stitle, stitle, sizeof(p_section->stitle) - 1);
663            p_section->stitle[sizeof(p_section->stitle) - 1] = '\0';
664    
665            strncpy(p_section->master_list, master_list, sizeof(p_section->master_list) - 1);
666            p_section->master_list[sizeof(p_section->master_list) - 1] = '\0';
667    
668            if (trie_dict_set(p_section_list_pool->p_trie_dict_section_by_name, sname, index) < 0)
669            {
670                    log_error("trie_dict_set(section, %s, %d) error\n", sname, index);
671                    return -2;
672            }
673    
674            return 0;
675    }
676    
677  void section_list_reset_articles(SECTION_LIST *p_section)  void section_list_reset_articles(SECTION_LIST *p_section)
678  {  {
679          p_section->article_count = 0;          p_section->article_count = 0;
# Line 670  void section_list_reset_articles(SECTION Line 685  void section_list_reset_articles(SECTION
685    
686          p_section->page_count = 0;          p_section->page_count = 0;
687          p_section->last_page_visible_article_count = 0;          p_section->last_page_visible_article_count = 0;
688    
689            p_section->ontop_article_count = 0;
690  }  }
691    
692  SECTION_LIST *section_list_find_by_name(const char *sname)  SECTION_LIST *section_list_find_by_name(const char *sname)
# Line 735  int section_list_append_article(SECTION_ Line 752  int section_list_append_article(SECTION_
752    
753          if (p_section == NULL || p_article_src == NULL)          if (p_section == NULL || p_article_src == NULL)
754          {          {
755                  log_error("section_list_append_article() NULL pointer error\n");                  log_error("NULL pointer error\n");
756                  return -1;                  return -1;
757          }          }
758    
# Line 847  int section_list_append_article(SECTION_ Line 864  int section_list_append_article(SECTION_
864          p_section->p_article_tail = p_article;          p_section->p_article_tail = p_article;
865    
866          // Update page          // Update page
867          if ((p_article->visible && p_section->last_page_visible_article_count % BBS_article_limit_per_page == 0) ||          if ((p_article->visible && p_section->last_page_visible_article_count == BBS_article_limit_per_page) ||
868                  p_section->article_count == 1)                  p_section->page_count == 0)
869          {          {
870                  p_section->p_page_first_article[p_section->page_count] = p_article;                  p_section->p_page_first_article[p_section->page_count] = p_article;
871                  p_section->page_count++;                  p_section->page_count++;
# Line 860  int section_list_append_article(SECTION_ Line 877  int section_list_append_article(SECTION_
877                  p_section->last_page_visible_article_count++;                  p_section->last_page_visible_article_count++;
878          }          }
879    
880            if (p_article->ontop && section_list_update_article_ontop(p_section, p_article) < 0)
881            {
882                    log_error("section_list_update_article_ontop(sid=%d, aid=%d) error\n",
883                                      p_section->sid, p_article->aid);
884                    return -5;
885            }
886    
887          return 0;          return 0;
888  }  }
889    
# Line 871  int section_list_set_article_visible(SEC Line 895  int section_list_set_article_visible(SEC
895    
896          if (p_section == NULL)          if (p_section == NULL)
897          {          {
898                  log_error("section_list_set_article_visible() NULL pointer error\n");                  log_error("NULL pointer error\n");
899                  return -2;                  return -1;
900          }          }
901    
902          p_article = article_block_find_by_aid(aid);          p_article = article_block_find_by_aid(aid);
# Line 883  int section_list_set_article_visible(SEC Line 907  int section_list_set_article_visible(SEC
907    
908          if (p_section->sid != p_article->sid)          if (p_section->sid != p_article->sid)
909          {          {
910                  log_error("section_list_set_article_visible() error: section sid %d != article sid %d\n", p_section->sid, p_article->sid);                  log_error("Inconsistent section sid %d != article sid %d\n", p_section->sid, p_article->sid);
911                  return -2;                  return -2;
912          }          }
913    
# Line 896  int section_list_set_article_visible(SEC Line 920  int section_list_set_article_visible(SEC
920          {          {
921                  p_section->visible_article_count--;                  p_section->visible_article_count--;
922    
923                    if (user_article_cnt_inc(p_article->uid, -1) < 0)
924                    {
925                            log_error("user_article_cnt_inc(uid=%d, -1) error\n", p_article->uid);
926                    }
927    
928                  if (p_article->tid == 0)                  if (p_article->tid == 0)
929                  {                  {
930                          p_section->visible_topic_count--;                          p_section->visible_topic_count--;
# Line 914  int section_list_set_article_visible(SEC Line 943  int section_list_set_article_visible(SEC
943                                          p_reply->visible = 0;                                          p_reply->visible = 0;
944                                          p_section->visible_article_count--;                                          p_section->visible_article_count--;
945                                          affected_count++;                                          affected_count++;
946    
947                                            if (user_article_cnt_inc(p_reply->uid, -1) < 0)
948                                            {
949                                                    log_error("user_article_cnt_inc(uid=%d, -1) error\n", p_reply->uid);
950                                            }
951                                  }                                  }
952                          }                          }
953                  }                  }
# Line 926  int section_list_set_article_visible(SEC Line 960  int section_list_set_article_visible(SEC
960                  {                  {
961                          p_section->visible_topic_count++;                          p_section->visible_topic_count++;
962                  }                  }
963    
964                    if (user_article_cnt_inc(p_article->uid, 1) < 0)
965                    {
966                            log_error("user_article_cnt_inc(uid=%d, 1) error\n", p_article->uid);
967                    }
968          }          }
969    
970          p_article->visible = visible;          p_article->visible = visible;
# Line 934  int section_list_set_article_visible(SEC Line 973  int section_list_set_article_visible(SEC
973          return affected_count;          return affected_count;
974  }  }
975    
976    int section_list_update_article_ontop(SECTION_LIST *p_section, ARTICLE *p_article)
977    {
978            int i;
979    
980            if (p_section == NULL || p_article == NULL)
981            {
982                    log_error("NULL pointer error\n");
983                    return -1;
984            }
985    
986            if (p_section->sid != p_article->sid)
987            {
988                    log_error("Inconsistent section sid %d != article sid %d\n", p_section->sid, p_article->sid);
989                    return -2;
990            }
991    
992            if (p_article->ontop)
993            {
994                    for (i = 0; i < p_section->ontop_article_count; i++)
995                    {
996                            if (p_section->p_ontop_articles[i]->aid == p_article->aid)
997                            {
998                                    log_error("Inconsistent state found: article %d already ontop in section %d\n", p_article->aid, p_section->sid);
999                                    return 0;
1000                            }
1001                            else if (p_section->p_ontop_articles[i]->aid > p_article->aid)
1002                            {
1003                                    break;
1004                            }
1005                    }
1006    
1007                    // Remove the oldest one if the array of ontop articles is full
1008                    if (p_section->ontop_article_count >= BBS_ontop_article_limit_per_section)
1009                    {
1010                            if (i == 0) // p_article is the oldest one
1011                            {
1012                                    return 0;
1013                            }
1014                            memmove((void *)(p_section->p_ontop_articles),
1015                                            (void *)(p_section->p_ontop_articles + 1),
1016                                            sizeof(ARTICLE *) * (size_t)(i - 1));
1017                            p_section->ontop_article_count--;
1018                            i--;
1019                    }
1020                    else
1021                    {
1022                            memmove((void *)(p_section->p_ontop_articles + i + 1),
1023                                            (void *)(p_section->p_ontop_articles + i),
1024                                            sizeof(ARTICLE *) * (size_t)(p_section->ontop_article_count - i));
1025                    }
1026    
1027                    p_section->p_ontop_articles[i] = p_article;
1028                    p_section->ontop_article_count++;
1029    
1030                    // TODO: debug
1031            }
1032            else // ontop == 0
1033            {
1034                    for (i = 0; i < p_section->ontop_article_count; i++)
1035                    {
1036                            if (p_section->p_ontop_articles[i]->aid == p_article->aid)
1037                            {
1038                                    break;
1039                            }
1040                    }
1041                    if (i == p_section->ontop_article_count) // not found
1042                    {
1043                            log_error("Inconsistent state found: article %d not ontop in section %d\n", p_article->aid, p_section->sid);
1044                            return 0;
1045                    }
1046    
1047                    memmove((void *)(p_section->p_ontop_articles + i),
1048                                    (void *)(p_section->p_ontop_articles + i + 1),
1049                                    sizeof(ARTICLE *) * (size_t)(p_section->ontop_article_count - i - 1));
1050                    p_section->ontop_article_count--;
1051            }
1052    
1053            return 0;
1054    }
1055    
1056    int section_list_page_count_with_ontop(SECTION_LIST *p_section)
1057    {
1058            int page_count;
1059    
1060            if (p_section == NULL)
1061            {
1062                    log_error("NULL pointer error\n");
1063                    return -1;
1064            }
1065    
1066            page_count = p_section->page_count - 1 +
1067                                     (p_section->last_page_visible_article_count + p_section->ontop_article_count) / BBS_article_limit_per_page +
1068                                     ((p_section->last_page_visible_article_count + p_section->ontop_article_count) % BBS_article_limit_per_page ? 1 : 0);
1069    
1070            if (page_count < 0)
1071            {
1072                    page_count = 0;
1073            }
1074    
1075            return page_count;
1076    }
1077    
1078    int section_list_page_article_count_with_ontop(SECTION_LIST *p_section, int32_t page_id)
1079    {
1080            if (p_section == NULL)
1081            {
1082                    log_error("NULL pointer error\n");
1083                    return -1;
1084            }
1085    
1086            if (page_id < p_section->page_count - 1)
1087            {
1088                    return BBS_article_limit_per_page;
1089            }
1090            else // if (page_id >= p_section->page_count - 1)
1091            {
1092                    return MIN(MAX(0,
1093                                               (p_section->last_page_visible_article_count + p_section->ontop_article_count -
1094                                                    BBS_article_limit_per_page * (page_id - p_section->page_count + 1))),
1095                                       BBS_article_limit_per_page);
1096            }
1097    }
1098    
1099  ARTICLE *section_list_find_article_with_offset(SECTION_LIST *p_section, int32_t aid, int32_t *p_page, int32_t *p_offset, ARTICLE **pp_next)  ARTICLE *section_list_find_article_with_offset(SECTION_LIST *p_section, int32_t aid, int32_t *p_page, int32_t *p_offset, ARTICLE **pp_next)
1100  {  {
1101          ARTICLE *p_article;          ARTICLE *p_article;
# Line 947  ARTICLE *section_list_find_article_with_ Line 1109  ARTICLE *section_list_find_article_with_
1109    
1110          if (p_section == NULL)          if (p_section == NULL)
1111          {          {
1112                  log_error("section_list_find_article_with_offset() NULL pointer error\n");                  log_error("NULL pointer error\n");
1113                  return NULL;                  return NULL;
1114          }          }
1115    
# Line 959  ARTICLE *section_list_find_article_with_ Line 1121  ARTICLE *section_list_find_article_with_
1121          }          }
1122    
1123          left = 0;          left = 0;
1124          right = p_section->page_count;          right = p_section->page_count - 1;
1125    
1126          // aid in the range [ head aid of pages[left], tail aid of pages[right - 1] ]          // aid in the range [ head aid of pages[left], tail aid of pages[right] ]
1127          while (left < right - 1)          while (left < right)
1128          {          {
1129                  // get page id no less than mid value of left page id and right page id                  // get page id no less than mid value of left page id and right page id
1130                  mid = (left + right) / 2 + (right - left) % 2;                  mid = (left + right) / 2 + (left + right) % 2;
   
                 if (mid >= p_section->page_count)  
                 {  
                         log_error("page id (mid = %d) is out of boundary\n", mid);  
                         return NULL;  
                 }  
1131    
1132                  if (aid < p_section->p_page_first_article[mid]->aid)                  if (aid < p_section->p_page_first_article[mid]->aid)
1133                  {                  {
1134                          right = mid;                          right = mid - 1;
1135                  }                  }
1136                  else                  else // if (aid < p_section->p_page_first_article[mid]->aid)
1137                  {                  {
1138                          left = mid;                          left = mid;
1139                  }                  }
# Line 1039  int section_list_calculate_page(SECTION_ Line 1195  int section_list_calculate_page(SECTION_
1195    
1196          if (p_section == NULL)          if (p_section == NULL)
1197          {          {
1198                  log_error("section_list_calculate_page() NULL pointer error\n");                  log_error("NULL pointer error\n");
1199                  return -1;                  return -1;
1200          }          }
1201    
# Line 1128  int section_list_calculate_page(SECTION_ Line 1284  int section_list_calculate_page(SECTION_
1284          } while (p_article != p_section->p_article_head);          } while (p_article != p_section->p_article_head);
1285    
1286          p_section->page_count = page + (visible_article_count > 0 ? 1 : 0);          p_section->page_count = page + (visible_article_count > 0 ? 1 : 0);
1287          p_section->last_page_visible_article_count = visible_article_count;          p_section->last_page_visible_article_count = (visible_article_count > 0
1288                                                                                                              ? visible_article_count
1289                                                                                                              : (page > 0 ? BBS_article_limit_per_page : 0));
1290    
1291          return 0;          return 0;
1292  }  }
# Line 1198  int section_list_move_topic(SECTION_LIST Line 1356  int section_list_move_topic(SECTION_LIST
1356    
1357          if (p_section_src == NULL || p_section_dest == NULL)          if (p_section_src == NULL || p_section_dest == NULL)
1358          {          {
1359                  log_error("section_list_move_topic() NULL pointer error\n");                  log_error("NULL pointer error\n");
1360                  return -1;                  return -1;
1361          }          }
1362    
# Line 1411  int get_section_index(SECTION_LIST *p_se Line 1569  int get_section_index(SECTION_LIST *p_se
1569          return index;          return index;
1570  }  }
1571    
1572    int get_section_info(SECTION_LIST *p_section, char *sname, char *stitle, char *master_list)
1573    {
1574            if (p_section == NULL)
1575            {
1576                    log_error("NULL pointer error\n");
1577                    return -1;
1578            }
1579    
1580            if (section_list_rd_lock(p_section) < 0)
1581            {
1582                    log_error("section_list_rd_lock(sid=%d) error\n", p_section->sid);
1583                    return -2;
1584            }
1585    
1586            if (sname != NULL)
1587            {
1588                    memcpy(sname, p_section->sname, sizeof(p_section->sname));
1589            }
1590            if (stitle != NULL)
1591            {
1592                    memcpy(stitle, p_section->stitle, sizeof(p_section->stitle));
1593            }
1594            if (master_list != NULL)
1595            {
1596                    memcpy(master_list, p_section->master_list, sizeof(p_section->master_list));
1597            }
1598    
1599            // release lock of section
1600            if (section_list_rd_unlock(p_section) < 0)
1601            {
1602                    log_error("section_list_rd_unlock(sid=%d) error\n", p_section->sid);
1603                    return -2;
1604            }
1605    
1606            return 0;
1607    }
1608    
1609  int section_list_try_rd_lock(SECTION_LIST *p_section, int wait_sec)  int section_list_try_rd_lock(SECTION_LIST *p_section, int wait_sec)
1610  {  {
1611          int index;          int index;
# Line 1570  int section_list_rd_lock(SECTION_LIST *p Line 1765  int section_list_rd_lock(SECTION_LIST *p
1765                          timer++;                          timer++;
1766                          if (timer % SECTION_TRY_LOCK_TIMES == 0)                          if (timer % SECTION_TRY_LOCK_TIMES == 0)
1767                          {                          {
1768                                  log_error("section_list_rd_lock() tried %d times on section %d\n", sid, timer);                                  log_error("section_list_try_rd_lock() tried %d times on section %d\n", timer, sid);
1769                          }                          }
1770                  }                  }
1771                  else // failed                  else // failed
1772                  {                  {
1773                          log_error("section_list_rd_lock() failed on section %d\n", sid);                          log_error("section_list_try_rd_lock() failed on section %d\n", sid);
1774                          break;                          break;
1775                  }                  }
1776          }          }
# Line 1601  int section_list_rw_lock(SECTION_LIST *p Line 1796  int section_list_rw_lock(SECTION_LIST *p
1796                          timer++;                          timer++;
1797                          if (timer % SECTION_TRY_LOCK_TIMES == 0)                          if (timer % SECTION_TRY_LOCK_TIMES == 0)
1798                          {                          {
1799                                  log_error("acquire_section_rw_lock() tried %d times on section %d\n", sid, timer);                                  log_error("section_list_try_rw_lock() tried %d times on section %d\n", timer, sid);
1800                          }                          }
1801                  }                  }
1802                  else // failed                  else // failed
1803                  {                  {
1804                          log_error("acquire_section_rw_lock() failed on section %d\n", sid);                          log_error("section_list_try_rw_lock() failed on section %d\n", sid);
1805                          break;                          break;
1806                  }                  }
1807          }          }


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

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