一个日志输出系统的设计

本文介绍一个基于linux进程调度策略来调度的信息输出系统,多条信息可以显示多次,每次的显示时间也可以设置:
以下是头文件定义loglet.h:
#include <pthread.h>
#define MAX_PRIO 8
#define MAX_LENGTH 512
static inline void __set_bit(int nr, volatile unsigned long * addr)
{
    ...
}
static inline void clear_bit(int nr, volatile unsigned long * addr)
{
    ...
}
static inline void __clear_bit(int nr, volatile unsigned long * addr)
{
    ...
}
struct list_head {
    ...
};
#define list_entry(ptr, type, member) /
        ...
static inline void INIT_LIST_HEAD(struct list_head *list)
{
    ...
}
static inline void __list_add(struct list_head *new,
                  struct list_head *prev,
                  struct list_head *next)
{
    ...
}
static inline void list_add(struct list_head *new, struct list_head *head)
{
    ...
}
static inline void list_add_tail(struct list_head *new, struct list_head *head)
{
    ...
}
static inline void __list_del(struct list_head * prev, struct list_head * next)
{
    ...
}
static inline void list_del(struct list_head *entry)
{
    ...
}
static inline int list_empty(const struct list_head *head)
{
    ...
}
struct prio_array {
    ...
};
//log_buff定义一条日志任务
struct log_buff {
    char    buf[MAX_LENGTH]; //实际需要展示的数据
    int     prio; //这个日志任务的优先级
    int    times; //这个任务目前为止显示的次数
    int    sed_slice; //持续多久被调度出去
    int    need_go; //是否被抢占
    int    remove_flag; //是否应该被删除
    int    alive_times; //这个任务需要显示的次数
    int    blink; //每次显示持续的时间
    struct list_head tasks;    //全局链表
    struct prio_array *array; //目前属于哪个数组,活跃还是过期?
    struct list_head run_list; //加入链表的锚
};
//log_queue定义一个全局的队列
struct log_queue {
    unsigned long    nr_running; //一共有多少任务
    unsigned long long last_tick; //上次调度的时间
    struct log_buff    *idle; //它的idle任务,没事干的时候显示
    struct prio_array *active;, *expired, arrays[2]; //活跃数组/过期数组
    void   (*handler_init)(void *argv); //输出处理初始化回调函数,比如初始化网络,LCD屏幕等等
    void   (*handler_process)(void *argv, int blink); //输出处理函数,比如写LCD屏幕等
    int   (*prio2times)(int prio); //优先级转化为时间片的函数
    pthread_mutex_t queue_mutex;

};
//以下定义一系列导出函数
//全局初始化
int global_init();
//以下三个设置log_queue的回调函数
void set_handle_process(void   (*handler_process)(void *argv, int blink));
void set_handle_init(void   (*handler_init)(void *argv), void *arg);
void set_prio2times(int   (*prio2times)(int prio));
//设置调度间隔
int set_interval(int val);
void set_idle(const char *info);
int insert_log(const char *info, int prio, int alives, int blink);    
--END--
以下是实现文件代码:
#include "log_let.h"
#include <stdlib.h>
#include <string.h>
#include <sys/time.h>
#include <signal.h>
#include <unistd.h>
#include <stdio.h>

struct log_buff *current = NULL;
struct log_buff *prev = NULL;
unsigned long long tick = 0;
struct log_queue  *global_rq;

//从一个链表中将一个日志退出
static void dequeue_log(struct log_buff *p, struct prio_array *array)
{
    array->nr_active--;
    list_del(&p->run_list);
    if (list_empty(array->queue + p->prio))
        __clear_bit(p->prio, array->bitmap);
}
//将一条日志加入一个链表
static void enqueue_log(struct log_buff *p, struct prio_array *array)
{
    struct list_head *lh = &global_rq->active->queue[p->prio];
    list_add_tail(&p->run_list, &array->queue[p->prio]);
    __set_bit(p->prio, array->bitmap);
    array->nr_active ++;
    p->array = array;

}
//得到默认队列
struct log_queue *get_queue()
{
    return global_rq;
}

void remove_log(struct log_buff *buf)
{
    buf->remove_flag = 1;
}
//信号处理函数
void time_tick(void)
{
    unsigned long long now = tick ++;
    struct log_buff *p = current;
    struct log_queue *rq = get_queue();//global_rq;

    rq->last_tick = now;
    //如果是idle日志,不处理
    //如果还没有任何日志,不处理
    //如果和前一次处理的一样,不重复处理
    if (p == rq->idle || !p || p == prev) {
        return;
    }
    prev = p;
    //如果时间片耗尽,则要么加入到“过期数组”,要么被删除
    if (!--p->sed_slice) {
        p->need_go = 1;
        //如果此条日志需要显示的次数到了,那么就要删除它了
        if (p->alive_times > 0 && !--p->alive_times) {
            remove_log(p); //仅仅设置一个删除标志
            return;
        }
        //将一个“过期”的日志加入到一个“过期数组”中,注意需要锁操作
        pthread_mutex_lock(&rq->queue_mutex);    
        dequeue_log(p, rq->active);
        pthread_mutex_unlock(&rq->queue_mutex);    
        //重新设置显示“时间片”信息
        p->sed_slice = rq->prio2times(p->prio);
        pthread_mutex_lock(&rq->queue_mutex);
        enqueue_log(p, rq->expired);
        pthread_mutex_unlock(&rq->queue_mutex);
    }
}
//激活一条日志任务
static void __activate_log(struct log_buff *p, struct log_queue *rq)
{
    struct prio_array *target = rq->active;
    pthread_mutex_lock(&rq->queue_mutex);
    enqueue_log(p, target);
    pthread_mutex_unlock(&rq->queue_mutex);
    rq->nr_running++;
}
//实际的插入动作参数依次是(内容,优先级别,显示次数,每次显示滞留的时间,是否是idle)
int insert_log_real(const char *content, int prio, int alive_times, int blink, int idle)
{
    int ret = 0;
    struct log_queue *rq;
    struct log_buff *lb;
    if ((current == NULL && !idle) || (current && idle)) {
        ret = -1;
        goto last;
    }
    rq = get_queue();
    lb = (struct log_buff *)calloc(1, sizeof(struct log_buff));
    strncpy(lb->buf, content, strlen(content)?:MAX_LENGTH);
    INIT_LIST_HEAD(&lb->run_list);
    lb->prio = prio;
    lb->times = 0;
    lb->alive_times = alive_times;
    lb->blink = blink;
    lb->sed_slice = (rq->prio2times)(prio);//show_style_func(prio);  //可设置一个数学函数
    if (!idle) //idle任务并不加入到任何链表当中,方便操作
        __activate_log(lb, rq);
    else {
        current = lb;
        rq->idle = lb;
    }
    //如果插入了一条信息且当前正在运行idle,则抢占它
    //如果当前的任务的优先级别没有新插入的任务的优先级别高,则抢占它
    if (current && ((current == rq->idle&&!idle) || current->prio>prio))
        current->need_go = 1;
last:
    return ret;
    
}
//插入idle日志任务
void set_idle(const char *idlebuf)
{
    insert_log_real(idlebuf, 7, 0, 1, 1);
}
//插入一般的日志任务
int insert_log(const char *content, int prio, int alive_times, int blink)
{
    return insert_log_real(content, prio, alive_times, blink, 0);
}
//以下这些都是直接从linux内核源代码中copy出来的
static inline unsigned long __ffs(unsigned long word)
{
    ...
}
static inline int sched_find_first_bit(const unsigned long *b)
{
    ...
}
//设置输出初始化回调函数
void set_handle_init(void (*func)(void *argv), void *arg)
{
    struct log_queue* rq = get_queue();
    rq->handler_init = func;
    rq->handler_init(arg);
}
//设置输出处理函数
void set_handle_process(void (*func)(void *argv, int blink))
{
    struct log_queue* rq = get_queue();
    rq->handler_process = func;
}
//设置从优先级到时间片的转换函数
void set_prio2times(int (*func)(int prio))
{
    struct log_queue* rq = get_queue();
    rq->prio2times = func;
}
//全局的初始化
int global_init()
{
    int ret = 0;
    //分配全局队列
    global_rq = (struct log_queue*)calloc(1, sizeof(struct log_queue));
    struct prio_array *active = (struct prio_array*)malloc(sizeof(struct prio_array));
    struct prio_array *expired = (struct prio_array*)malloc(sizeof(struct prio_array));
    memset(active, 0, sizeof(struct prio_array));
    memset(expired, 0, sizeof(struct prio_array));
    global_rq->active = active;
    global_rq->expired = expired;
    ret = pthread_mutex_init(&global_rq->queue_mutex, NULL);
    int i;
    //初始化活动数组
    for (i = 0; i < MAX_PRIO; i++) {
        INIT_LIST_HEAD(&global_rq->active->queue[i]);
    }
    //初始化过期数组
    for (i = 0; i < MAX_PRIO; i++) {
        INIT_LIST_HEAD(&global_rq->expired->queue[i]);
    }
    return ret;
}
//设置调度时间间隔并且开始调度
int set_interval(int second)
{
    static struct itimerval p_realt;
    p_realt.it_interval.tv_sec = second;
    p_realt.it_interval.tv_usec = 0;
    p_realt.it_value.tv_sec = second;
    p_realt.it_value.tv_usec = 0;
        signal(SIGALRM,time_tick);
        if(setitimer(ITIMER_REAL,&p_realt,(struct itimerval *)0) == -1) {
            perror("setitimer error! ");
        return -1;
    }
    return 0;
}
//全局启动
int start()
{
    int ret = 0;
    struct log_buff *previous;
    //以下的逻辑完全就是linux内核O(1)进程调度器的逻辑
    //参见linux内核kernel/sched.c中的schedule函数代码
    while (1) {
        struct log_queue *rq = get_queue();
        struct prio_array *array = rq->active;
        previous = current;
        if (current->need_go) {
redo:
            if (!array->nr_active) {
                rq->active = rq->expired;
                rq->expired = array;
                array = rq->active;
            }
            struct log_buff *lb;
            if (!array->nr_active) {
                lb = rq->idle;
                goto switch_to;                
            }
            int idx = sched_find_first_bit(array->bitmap);
            struct list_head *queue = array->queue + idx;    
            lb = list_entry(queue->next, struct log_buff, run_list);
            if (lb->remove_flag) {
                int redo_flag = 0;
                if (lb == current) {
                    redo_flag = 1;
                }
                pthread_mutex_lock(&rq->queue_mutex);    
                dequeue_log(lb, lb->array);
                pthread_mutex_unlock(&rq->queue_mutex);    
                free(lb);
                lb = NULL;
                if(redo_flag)
                    goto redo;
            }
switch_to:
            lb->need_go = 0;
            current = lb;
        }
        if (current == previous && current == rq->idle) {
            usleep(500);
            continue;
        }
        rq->handler_process(current->buf, current->blink);
    }
    return ret;
}
下面是一个测试程序
#include "lcd_task.h"
#include <stdlib.h>
#include <stdio.h>
#include <malloc.h>
#include <unistd.h>
#include <string.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <termios.h>
#include <errno.h>

int show_style_func(int prio)
{
    return 1;//9-prio;
}
void lcd_init(void *argv)
{
    //打开并设置屏幕
    ... open( ..., O_RDWR | O_NOCTTY | O_NDELAY);
    ...
}

void lcd_process(void *argv, int blink)
{
    //argv就是需要输出的信息,blink就是输出的间隔
    char *buf = (char*)argv;
    ...
    for (i = 0; i < 需要的屏幕数; i++, ...) {
        //换屏处理,每屏幕都显示blink的间隔,下面的处理为了防止被调度timer信号中断sleep
        while(blink = sleep(blink));
        blink = blk;
    }
}
void *test_func(void *arg)
{
    int prio;
    int i, j;
    int total = 20;
    char buf[16];
    while (total--) {
        prio =  random()%8;
        for (i = 0; i < 8; i++) {
            if(prio == i) {
                memset(buf, 0, sizeof(buf));
                for (j = 0; j < 2*i; j++) {
                    strcat(buf, "-");
                }
                if (i == 0)
                    strcat(buf, "0000");
                insert_log(buf, prio, 2, 2);
                break;
            }
        }
        sleep(1);
    }
}

int main(int argc, char **argv)
{
    pthread_t id;
    global_init();
    set_handle_process(lcd_process);
    set_handle_init(lcd_init, NULL);
    set_prio2times(show_style_func);
    set_interval(1);
    set_idle("I'm IDLE");
    insert_log("this is one", 4, 3, 5);
    ...
    pthread_create(&id,NULL,test_func,NULL);
    start();
}

原文链接: https://blog.csdn.net/dog250/article/details/6070074

欢迎关注

微信关注下方公众号,第一时间获取干货硬货;公众号内回复【pdf】免费获取数百本计算机经典书籍;

也有高质量的技术群,里面有嵌入式、搜广推等BAT大佬

    一个日志输出系统的设计

原创文章受到原创版权保护。转载请注明出处:https://www.ccppcoding.com/archives/409556

非原创文章文中已经注明原地址,如有侵权,联系删除

关注公众号【高性能架构探索】,第一时间获取最新文章

转载文章受原作者版权保护。转载请注明原作者出处!

(0)
上一篇 2023年4月26日 上午11:29
下一篇 2023年4月26日 上午11:29

相关推荐