Compare commits

...

2 Commits

Author SHA1 Message Date
8c52e4ba84 优化退出流程,统一注册到on_exit( 2025-09-29 16:07:11 +08:00
afe70e6d17 优化退出信号传递流程 2025-09-28 14:26:09 +08:00
13 changed files with 192 additions and 29 deletions

View File

@ -9,7 +9,8 @@ add_library(Swmem SHARED network/swap.c)
add_library(Interpre SHARED interpreter/interpreter.c tools/pkgmanager/pkginstall.c) add_library(Interpre SHARED interpreter/interpreter.c tools/pkgmanager/pkginstall.c)
add_library(Log SHARED tools/log/log.c) add_library(Log SHARED tools/log/log.c)
add_library(Toml SHARED tools/toml/toml.c) add_library(Toml SHARED tools/toml/toml.c)
add_library(Quit SHARED tools/quit/quit.c)
target_link_libraries(Start_Onebot_back Network Swmem Interpre Log Toml) target_link_libraries(Start_Onebot_back Network Swmem Interpre Log Toml Quit)
include_directories(${PROJECT_SOURCE_DIR}) include_directories(${PROJECT_SOURCE_DIR})

View File

@ -151,6 +151,7 @@ int exce(const int command,ctx *all_ctx)
case QUIT: case QUIT:
printf("shuting down\n"); printf("shuting down\n");
all_ctx->statue = -1; all_ctx->statue = -1;
write(all_ctx->fifofd[1],"q",1);
return 1; return 1;
} }

View File

@ -1,6 +1,8 @@
#define _GNU_SOURCE
#include "tem/ctl.h" #include "tem/ctl.h"
#include "network/network.h" #include "network/network.h"
#include "tools/toml/toml.h" #include "tools/toml/toml.h"
#include "tools/quit/quit.h"
#include <unistd.h> #include <unistd.h>
#include <stdio.h> #include <stdio.h>
#include <stdlib.h> #include <stdlib.h>
@ -45,14 +47,15 @@ int main()
pthread_t network_id; pthread_t network_id;
pthread_create(&network_id,NULL,networkmanager->run_network,(void*)networkmanager); pthread_create(&network_id,NULL,networkmanager->run_network,(void*)networkmanager);
//启动网络监听与线程池,并加载插件 //启动网络监听与线程池,并加载插件
alres *resource = (alres*)malloc(sizeof(alres));
resource->loger = logsmanager;
resource->network = networkmanager;
resource->tem = teml;
on_exit(quit_all,resource);
//注册清理函数
teml->run(teml,fifo); teml->run(teml,fifo);
//启动终端 //启动终端
pthread_join(network_id,NULL); pthread_join(network_id,NULL);
//等待网络管理器进程结束 //等待网络管理器进程结束
free(teml);
free(networkmanager);
free(logsmanager);
//释放内存
return 1; return 1;
} }

View File

@ -5,6 +5,7 @@
#include "http_rel.h" #include "http_rel.h"
#include "cJSON.h" #include "cJSON.h"
#include "tools/log/log.h" #include "tools/log/log.h"
#include "tools/quit/quit.h"
#include <semaphore.h> #include <semaphore.h>
#include <unistd.h> #include <unistd.h>
#include <stdio.h> #include <stdio.h>
@ -82,6 +83,7 @@ int init_network(int port)
addr.sin_port = htons(port); // 端口号必须网络字节序 addr.sin_port = htons(port); // 端口号必须网络字节序
addr.sin_addr.s_addr = htonl(INADDR_ANY); // 0.0.0.0:本机所有网卡 addr.sin_addr.s_addr = htonl(INADDR_ANY); // 0.0.0.0:本机所有网卡
bind(fd, (struct sockaddr *)&addr, sizeof(addr)); bind(fd, (struct sockaddr *)&addr, sizeof(addr));
listen(fd,10);
return fd; return fd;
} }
@ -125,7 +127,7 @@ int iss_work(netm *self,char *command)
void *pth_module(void *args_p) void *pth_module(void *args_p)
{ {
args *argms = (args*)args_p; net_args *argms = (net_args*)args_p;
pth_m *pmd = argms->pth; pth_m *pmd = argms->pth;
log_manager *logger = argms->log; log_manager *logger = argms->log;
//参数解析 //参数解析
@ -171,7 +173,7 @@ int start_pool(netm *self)
//为线程开辟管道 //为线程开辟管道
pipe(self->pool[i].fifo_fd); pipe(self->pool[i].fifo_fd);
//启动线程 //启动线程
args arg; net_args arg;
arg.pth =&self->pool[i]; arg.pth =&self->pool[i];
arg.log = self->logmanager; arg.log = self->logmanager;
self->pool[i].status = 1; self->pool[i].status = 1;
@ -183,8 +185,12 @@ int shutdown_pool(netm *self)
{ {
for(int i = 0;i<MAX_POOL;i++) for(int i = 0;i<MAX_POOL;i++)
{ {
close(self->pool[i].fifo_fd[0]); if(self->pool[i].status == -1)
continue;
self->pool[i].status = -1;
close(self->pool[i].fifo_fd[1]);
} }
return 1;
} }
int server_run(int port,int fifo_fd,netm *self) int server_run(int port,int fifo_fd,netm *self)
@ -194,8 +200,16 @@ int server_run(int port,int fifo_fd,netm *self)
perror("epoll_create1"); perror("epoll_create1");
exit(EXIT_FAILURE); exit(EXIT_FAILURE);
} }
struct epoll_event ev;
//设置epoll同时监听控制管道与http请求
ev.events = EPOLLIN;
ev.data.fd = fifo_fd;
epoll_ctl(epfd, EPOLL_CTL_ADD, fifo_fd, &ev);
char iss_buf[256]; char iss_buf[256];
int http_fd = init_network(port); int http_fd = init_network(port);
ev.data.fd = http_fd;
epoll_ctl(epfd, EPOLL_CTL_ADD, http_fd, &ev);
struct epoll_event events; struct epoll_event events;
for(;;) for(;;)
{ {
@ -206,7 +220,10 @@ int server_run(int port,int fifo_fd,netm *self)
} }
if(events.data.fd ==http_fd) if(events.data.fd ==http_fd)
{ {
sprintf(iss_buf,"s/%d/e",accept4(http_fd,NULL,NULL,SOCK_NONBLOCK | SOCK_CLOEXEC)); int nt_fd = accept4(http_fd,NULL,NULL,SOCK_NONBLOCK | SOCK_CLOEXEC);
if(nt_fd == -1)
continue;
sprintf(iss_buf,"s/%d/e",nt_fd);
self->iss_work(self,iss_buf); self->iss_work(self,iss_buf);
} }
if(events.data.fd == fifo_fd) if(events.data.fd == fifo_fd)
@ -217,9 +234,11 @@ int server_run(int port,int fifo_fd,netm *self)
switch(command){ switch(command){
case 'q': case 'q':
//退出逻辑 //退出逻辑
return 1;
break; break;
case 'u': case 'u':
//插件更新逻辑 //插件更新逻辑
break; break;
} }
} }
@ -231,7 +250,7 @@ void *run_network(void *self_d)
{ {
netm *self = (netm*)self_d; netm *self = (netm*)self_d;
self->start_pool(self); self->start_pool(self);
server_run(self->port,self->fifo_fd[1],self); server_run(self->port,self->fifo_fd[0],self);
self->shutdown_pool(self); self->shutdown_pool(self);
} }

View File

@ -1,7 +1,7 @@
#ifndef NETWORK #ifndef NETWORK
#define NETWORK #define NETWORK
#define MAX_POOL 24 #define MAX_POOL 10
#define MAX_MESSAGE_BUF 10240 #define MAX_MESSAGE_BUF 10240
#include <pthread.h> #include <pthread.h>
#include<tools/log/log.h> #include<tools/log/log.h>
@ -14,11 +14,11 @@ typedef struct pthread_module
atomic_int status; atomic_int status;
}pth_m; }pth_m;
typedef struct args typedef struct net_args
{ {
log_manager *log; log_manager *log;
pth_m *pth; pth_m *pth;
}args; }net_args;
typedef struct network_manager typedef struct network_manager
{ {

View File

@ -5,7 +5,11 @@
#include <stddef.h> #include <stddef.h>
#include <string.h> #include <string.h>
#include <termios.h> #include <termios.h>
#include <pthread.h>
#include <signal.h>
#include "ctl.h" #include "ctl.h"
#include "interpreter/interpreter.h" #include "interpreter/interpreter.h"
#include "tools/log/log.h" #include "tools/log/log.h"
@ -235,6 +239,7 @@ int teml(Ctl *self,int fifo[2])
//创建线程用于定期清理日志 //创建线程用于定期清理日志
pthread_create(&self->logwathcher,NULL,self->logmanager->clear_log,self->logmanager); pthread_create(&self->logwathcher,NULL,self->logmanager->clear_log,self->logmanager);
command->statue = 0; command->statue = 0;
self->command = command;
do do
{ //设置缓冲区,接收用户输入 { //设置缓冲区,接收用户输入
write(STDOUT_FILENO,PROMPT,sizeof(PROMPT)); write(STDOUT_FILENO,PROMPT,sizeof(PROMPT));
@ -248,9 +253,14 @@ int teml(Ctl *self,int fifo[2])
const char fexp[256] = {'\0'}; const char fexp[256] = {'\0'};
memcpy(&input,&fexp,MAX_BUF); memcpy(&input,&fexp,MAX_BUF);
}while(command->statue == 0); }while(command->statue == 0);
log_manager_stop(self->logmanager);
pthread_join(self->logwathcher,NULL);
//关闭log定期清理程序
close(fifo[0]); close(fifo[0]);
close(fifo[1]); close(fifo[1]);
free_history(self); free_history(self);
self->command = NULL;
free(command); free(command);
} }

View File

@ -3,6 +3,7 @@
#include <pthread.h> #include <pthread.h>
#include "tools/log/log.h" #include "tools/log/log.h"
#include "interpreter/interpreter.h"
#define MAX_BUF 256 #define MAX_BUF 256
#define HISTORY_BUF 256 #define HISTORY_BUF 256
@ -17,8 +18,11 @@ typedef struct Ctl
char *history[HISTORY_BUF]; char *history[HISTORY_BUF];
pthread_t logwathcher; pthread_t logwathcher;
log_manager *logmanager; log_manager *logmanager;
ctx *command;//解释器上下文
}Ctl; }Ctl;
Ctl *init_tem(log_manager *logmanager); Ctl *init_tem(log_manager *logmanager);
int free_history(Ctl *self);
#endif #endif

View File

@ -1,8 +1,11 @@
#define _POSIX_C_SOURCE 200112L
#include "log.h" #include "log.h"
#include <stdio.h> #include <stdio.h>
#include <stddef.h> #include <stddef.h>
#include <stdlib.h> #include <stdlib.h>
#include <unistd.h> #include <unistd.h>
#include <time.h>
#include <errno.h>
int in_log(logs *log,log_manager *self) int in_log(logs *log,log_manager *self)
{ {
@ -11,6 +14,7 @@ int in_log(logs *log,log_manager *self)
if(self->log == NULL){ if(self->log == NULL){
self->log = log; self->log = log;
self->rear = log; self->rear = log;
sem_post(&self->log_sem);
return 0; return 0;
} }
self->count++; self->count++;
@ -25,6 +29,11 @@ logs *out_log(log_manager *self)
{ {
sem_wait(&self->log_sem); sem_wait(&self->log_sem);
logs *buf = self->log; logs *buf = self->log;
if(self->log==NULL)
{
sem_post(&self->log_sem);
return NULL;
}
if(self->log->next ==NULL) if(self->log->next ==NULL)
self->log = self->rear = NULL; self->log = self->rear = NULL;
self->count--; self->count--;
@ -33,27 +42,83 @@ logs *out_log(log_manager *self)
return buf; return buf;
} }
int sleep_with_signal(log_manager *self)
{
struct timespec ts;
int rc;
/* 计算绝对超时:当前 + 1000 s */
if (clock_gettime(CLOCK_MONOTONIC, &ts) != 0)
return -1; /* 罕见失败 */
ts.tv_sec += 1000;
/* 纳秒部分无需处理1000 s 整不会溢出 */
pthread_mutex_lock(&self->mtx); /* 进入临界区 */
while (1) {
rc = pthread_cond_timedwait(&self->cond, &self->mtx, &ts);
if (rc == ETIMEDOUT) { /* 1000 s 到点 */
pthread_mutex_unlock(&self->mtx);
return 1; /* 正常超时 */
}
if (rc != 0) { /* 其他错误 */
pthread_mutex_unlock(&self->mtx);
return -1;
}
/* 被 signal / broadcast 提前唤醒,检查 stop */
if (self->stop == 1) {/* 主线程要求退出 */
pthread_mutex_unlock(&self->mtx);
return 0; /* 告诉调用者:该结束了 */
}
}
}
int cleanup(log_manager *self)
{
if(self->log ==NULL)
return 1;
logs *tobeclean,*loc;
sem_wait(&self->log_sem);
loc = self->log;
self->log = NULL;
sem_post(&self->log_sem);
while(loc->next !=NULL)
{
tobeclean = loc;
loc = loc->next;
free(tobeclean);
}
free(loc);
}
void log_manager_stop(log_manager *self)
{
pthread_mutex_lock(&self->mtx);
self->stop = 1; /* 置退出标志 */
pthread_cond_broadcast(&self->cond); /* 唤醒所有等待线程 */
printf("stopping loger\n");
pthread_mutex_unlock(&self->mtx);
printf("done\n");
}
//定期清理函数 //定期清理函数
void *clear_log(void *self_p) void *clear_log(void *self_p)
{ {
log_manager *self = (log_manager*)self_p; log_manager *self = (log_manager*)self_p;
for(;;) for(;;)
{ {
sleep(1000); sleep_with_signal(self);
sem_wait(&self->log_sem); sem_wait(&self->log_sem);
if(self->count<256){ if((self->count<256||self->log==NULL)&&self->stop !=1){
sem_post(&self->log_sem); sem_post(&self->log_sem);
continue; continue;
} }
logs* buf = self->log;
self->log = self->rear =NULL;
sem_post(&self->log_sem); sem_post(&self->log_sem);
logs* tobeclear; cleanup(self);
while(buf->next !=NULL) if(self->stop == 1){
{ return NULL;
tobeclear = buf;
buf = buf->next;
free(tobeclear);
} }
} }
} }
@ -66,8 +131,12 @@ int init_loger(log_manager *self)
return -1; return -1;
} }
sem_init(&self->log_sem, 0,1); sem_init(&self->log_sem, 0,1);
pthread_mutex_init(&self->mtx,NULL);
pthread_cond_init(&self->cond,NULL);
self->in_log = in_log; self->in_log = in_log;
self->out_log = out_log; self->out_log = out_log;
self->clear_log = clear_log; self->clear_log = clear_log;
self->log = NULL; self->log = NULL;
self->stop = 0;
self->cleanup = cleanup;
} }

View File

@ -2,6 +2,7 @@
#define LOG #define LOG
#include <semaphore.h> #include <semaphore.h>
#include <pthread.h>
#define MAX_LOG 256 #define MAX_LOG 256
@ -16,12 +17,17 @@ typedef struct log_manager
int (*in_log)(logs *,struct log_manager*); int (*in_log)(logs *,struct log_manager*);
logs* (*out_log)(struct log_manager*); logs* (*out_log)(struct log_manager*);
void *(*clear_log)(void*); void *(*clear_log)(void*);
int (*cleanup)(struct log_manager*);
sem_t log_sem; sem_t log_sem;
logs *log; logs *log;
logs *rear; logs *rear;
int count; int count;
pthread_mutex_t mtx;
pthread_cond_t cond;
int stop;
}log_manager; }log_manager;
void log_manager_stop(log_manager *self);
int init_loger(log_manager *self); int init_loger(log_manager *self);
#endif #endif

View File

View File

View File

@ -1,7 +1,44 @@
#include <stdio.h> #include <stdio.h>
#include <stdlib.h> #include <stdlib.h>
#include <string.h>
#include "quit.h"
#include "tem/ctl.h"
void *quit_all(void *self_d) void quit_all(int status,void *self_p)
{ {
alres *resouce =(alres*)self_p;
//转换参数
} resouce->network->shutdown_pool(resouce->network);
logs *netlog = (logs*)malloc(sizeof(logs));
netlog->next = NULL;
memcpy(netlog->log,"shuting down networkserver",27);
resouce->loger->in_log(netlog,resouce->loger);
free(resouce->network);
//释放网络资源
if(resouce->tem->command !=NULL){
free_history(resouce->tem);
if(resouce->tem->command->arg != NULL)
{
args* arg = resouce->tem->command->arg;
if(arg->next !=NULL)
{
while(arg->next != NULL){
args* tobefree = arg;
arg = arg->next;
free(tobefree);
}
free(arg);
}
}
free(resouce->tem->command);
}
//释放终端资源
pthread_mutex_destroy(&resouce->loger->mtx);
resouce->loger->cleanup(resouce->loger);
sem_destroy(&resouce->loger->log_sem);
//销毁信号量
free(resouce->loger);
//清理日志
}

View File

@ -1,6 +1,19 @@
#ifndef QUIT #ifndef QUIT_LIB
#define QUIT #define QUIT_LIB
#include "network/network.h"
#include "tem/ctl.h"
#include "tools/log/log.h"
typedef struct all_resources
{
Ctl *tem;
netm *network;
log_manager *loger;
}alres;
void quit_all(int status,void *self_p);
void *quitall(int status,void *arg);
#endif #endif