ring_log环形日志-6M缓冲区_proc接口
文章目录
- log_tools.c
- log.c
- spin_lock
- seq_puts
- seq_read
- seq_write
- single_open
- makefile
- test.sh
- 测试:
- 运行./test.sh
- 读取日志
- 插入日志
- echo cat测试
- 参考:
log_tools.c
#include <stdlib.h>
#include <stdio.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <string.h>
#include <unistd.h>
#include <getopt.h>#define MYLOG_PROC_PATH "/proc/mylog"
#define TMP_BUF_SIZE 1024
#define BUF_SIEZ 256
#define PER_LOG_SIZE 128#define LOG_LEVEL_EMERG 1
#define LOG_LEVEL_ALERT 2
#define LOG_LEVEL_CRIT 3
#define LOG_LEVEL_ERR 4
#define LOG_LEVEL_WARNING 5
#define LOG_LEVEL_NOTICE 6
#define LOG_LEVEL_INFO 7
#define LOG_LEVEL_DEBUG 8static FILE *fp;static void print_help(void)
{printf("-H --help 显示帮助\n");printf("-h --help 显示帮助\n");printf("-r --readall 读取日志\n");printf("-i --insert 插入日志\n");printf("-i <log_level> <log_message>\n");
}
void print_log_level(void)
{printf("log level must be range 1 to 8\n");printf("LOG_LEVEL_EMERG 1\n");printf("LOG_LEVEL_ALERT 2\n");printf("LOG_LEVEL_CRIT 3\n");printf("LOG_LEVEL_ERR 4\n");printf("LOG_LEVEL_WARNING 5\n");printf("LOG_LEVEL_NOTICE 6\n");printf("LOG_LEVEL_INFO 7\n");printf("LOG_LEVEL_DEBUG 8\n");
}int insert_log(int argc,char const *argv[])
{const char *log_message = NULL;int log_level = -1;char tmpbuf[TMP_BUF_SIZE] = {0};if(argc != 4) {printf("Usage: %s -i <log_level> <log_message>\n",argv[0]);goto ERROR;}log_level = atoi(argv[2]);if (log_level < 1 || log_level > LOG_LEVEL_DEBUG) {printf("error log_level");goto ERROR;}log_message = argv[3];if(strlen(log_message) > PER_LOG_SIZE){goto ERROR;}fseek(fp,0,SEEK_END);sprintf(tmpbuf,"%d %s\n",log_level,log_message);if(fwrite(tmpbuf,strlen(tmpbuf),1,fp) < 0) {perror("fwrite");}printf("insert %d %s\n",log_level,log_message);ERROR:return 0;
}
int read_all_log(void)
{char buf[BUF_SIEZ] = {0};fseek(fp,0,SEEK_SET);while (fgets(buf, sizeof(buf), fp)) {printf("%s", buf);}return 0;
}int main(int argc, char const *argv[])
{int res = -1;int index = -1;struct option argarr[] = {{"read",0,NULL,'r'},{"insert",1,NULL,'i'},{"help",0,NULL,'h'},{"help",0,NULL,'H'},{NULL, 0, NULL, 0}};if(argc < 2) {print_help();goto ERROR;}fp = fopen("/proc/mylog", "r+");if (!fp) {printf("Failed to open /proc/mylog\n");goto ERROR;}res = getopt_long(argc,(char **)argv,"H:i:h:r",argarr,&index);if(res < 0) {goto ERROR;}switch(res){case 'r':if (fp != NULL) {read_all_log();} else {printf("Failed to open /proc/mylog\n");}break;case 'i':if (optarg) {insert_log(argc,argv);} else {printf("Error: no log message specified\n");print_help();}break;case 'h':print_help();break;case 'H':print_help();print_log_level();break;default:printf("Unknown option: %c\n", optopt);print_help();break;}ERROR:if(fp != NULL) {fclose(fp);fp = NULL;}return 0;
}
log.c
#include <linux/init.h>
#include <linux/module.h>
#include <linux/proc_fs.h>
#include <linux/uaccess.h>
#include <linux/list.h>
#include <linux/slab.h>
#include <linux/seq_file.h>#define LOG_PROC_NAME "mylog"
#define LOG_BUF_SIZE (6 * 1024 * 1024)
#define PER_LOG_SIZE 128
#define STR_TIME_LEN 32struct log_entry {struct list_head list;char *msg;size_t len;
};struct log_buffer {struct list_head head;size_t size;size_t used;
};static struct log_buffer buffer;
static spinlock_t log_buf_lock;static ssize_t log_write(struct file *file, const char __user *buf, size_t count, loff_t *pos)
{char *msg = NULL;struct log_entry *entry = NULL;size_t msg_pos = 0;size_t len = 0;size_t i = 0;struct timespec ts = {.tv_sec = 0,.tv_nsec = 0};if(count + STR_TIME_LEN > PER_LOG_SIZE) {// count = PER_LOG_SIZE;goto ERROR;}msg = kmalloc(count + STR_TIME_LEN + 1, GFP_KERNEL);if (!msg) {goto ERROR;}memset(msg,0,count + STR_TIME_LEN + 1);ktime_get_ts(&ts);sprintf(msg, "[%ld.%09ld] ", ts.tv_sec, ts.tv_nsec);msg_pos = 0;msg_pos += strlen(msg);if (copy_from_user(msg + msg_pos, buf, count)) {goto COPY_ERROR;}msg[count + STR_TIME_LEN] = '\0';len = 0;len = strlen(msg) + 1;if (buffer.used + len > buffer.size) {i = buffer.used;while(buffer.used > i/2) {entry = list_first_entry(&buffer.head, struct log_entry, list);buffer.used -= entry->len;list_del(&entry->list);kfree(entry->msg);entry->msg = NULL;kfree(entry);entry = NULL;}}entry = NULL;entry = kmalloc(sizeof(struct log_entry), GFP_KERNEL);if (!entry) {goto COPY_ERROR;}entry->msg = msg;entry->len = len;list_add_tail(&entry->list, &buffer.head);buffer.used += len;return count;COPY_ERROR:if(msg != NULL) {kfree(msg);msg = NULL;}
ERROR:return -1;}
static int log_list_show(struct seq_file *m, void *v)
{struct log_entry *entry = NULL;struct log_entry *n = NULL;spin_lock(&log_buf_lock);list_for_each_entry_safe(entry,n,&buffer.head,list) {seq_printf(m, "%s", entry->msg);}spin_unlock(&log_buf_lock);return 0;
}static int log_list_open(struct inode *inode, struct file *file)
{return single_open(file, log_list_show, NULL);
}static const struct file_operations log_fops = {.owner = THIS_MODULE,.open = log_list_open,.read = seq_read,.write = log_write,
};static int __init log_init(void)
{INIT_LIST_HEAD(&buffer.head);buffer.size = LOG_BUF_SIZE;buffer.used = 0;if (!proc_create(LOG_PROC_NAME, 0666, NULL, &log_fops)) {printk(KERN_ERR "Failed to create /proc/%s\n", LOG_PROC_NAME);return -ENOMEM;}spin_lock_init(&log_buf_lock);printk(KERN_INFO "log module loaded\n");return 0;
}static void __exit log_exit(void)
{struct log_entry *entry = NULL;struct list_head *pos = NULL, *next = NULL;list_for_each_safe(pos, next, &buffer.head) {entry = list_entry(pos, struct log_entry, list);list_del(&entry->list);kfree(entry->msg);kfree(entry);}remove_proc_entry(LOG_PROC_NAME, NULL);printk(KERN_INFO "log module unloaded\n");
}module_init(log_init);
module_exit(log_exit);
MODULE_LICENSE("GPL");
spin_lock
spin_lock是Linux内核中的一种自旋锁,用于保护共享资源的访问。与mutex不同,spin_lock在获取锁时并不会进入睡眠状态,而是使用循环不断尝试获取锁,直到获取到锁才会退出循环,因此也称为自旋锁。
spin_lock的使用方法如下:
-
在需要保护的代码段前调用spin_lock函数获取锁。
-
在代码段结束时调用spin_unlock函数释放锁。
seq相关头文件linux/seq_file.h,seq相关函数的实现在fs/seq_file.c。seq函数最早是在2001年就引入了,但以前内核中一直用得不多,而到了2.6内核后,许多/proc的只读文件中大量使用了seq函数处理。
由于procfs的默认操作函数只使用一页的缓存,在处理较大的proc文件时就有点麻烦,并且在输出一系列结构体中的数据时也比较不灵活,需要自己在read_proc函数中实现迭代,容易出现Bug。所以内核黑客们对一些/proc代码做了研究,抽象出共性,最终形成了seq_file(Sequence file:序列文件)接口。 这个接口提供了一套简单的函数来解决以上proc接口编程时存在的问题,使得编程更加容易,降低了Bug出现的机会。
在需要创建一个由一系列数据顺序组合而成的虚拟文件或一个较大的虚拟文件时,推荐使用seq_file接口。但是我个人认为,并不是只有procfs才可以使用这个seq_file接口,因为其实seq_file是实现的是一个操作函数集,这个函数集并不是与proc绑定的,同样可以用在其他的地方。
seq_puts
是Linux内核中的一个函数,用于将字符串写入序列文件的缓冲区。序列文件是用于顺序访问数据流的特殊文件。seq_puts函数接受两个参数:指向序列文件的指针(m)和要写入文件的字符串。
seq_read
是一个Linux内核函数,用于从序列文件中读取数据并将其放入用户空间缓冲区中
seq_write
是一个Linux内核中的函数,它用于将数据写入到顺序访问的设备中
seq_write 是一个辅助函数,用于在 proc 文件系统中创建一个支持顺序写入的文件。它的原型定义如下:
ssize_t seq_write(struct seq_file *, const char *, size_t);
第一个参数是一个指向 seq_file 的指针,第二个参数是一个指向数据缓冲区的指针,第三个参数是数据的长度。
seq_write 函数的返回值是一个 ssize_t 类型的整数,表示写入的字节数。如果返回值小于 0,则表示写入失败。
在创建一个支持顺序写入的 proc 文件时,可以使用 seq_write 函数来向文件中写入数据。通常需要在文件的 write 函数中调用 seq_write 函数来实现写入操作。
single_open
single_open 是一个辅助函数,用于在 proc 文件系统中创建一个仅支持顺序读取的文件。它的原型定义如下:
int single_open(struct file , int ()(struct seq_file *, void *), void *);
第一个参数是一个指向文件的指针,第二个参数是一个回调函数,用于向 seq_file 中写入数据,第三个参数是一个指向私有数据的指针,可用于传递一些上下文信息给回调函数。
single_open 函数的返回值是一个整数,表示函数执行的状态。如果返回值小于 0,则表示创建文件失败,否则表示创建文件成功。
在创建一个支持顺序读取的 proc 文件时,通常可以使用 single_open 函数来创建文件,然后将回调函数和私有数据传递给它。在回调函数中,可以使用 seq_puts、seq_printf 等函数向 seq_file 中写入数据。最后,使用 single_release 函数来释放资源。
makefile
obj-m := log.o # ubuntu16CC=gccLD=ldKDIR := /usr/src/linux-headers-$(shell uname -r)PWD := $(shell pwd)all:$(MAKE) -C $(KDIR) M=$(PWD) modules CC=$(CC) LD=$(LD)clean:$(MAKE) -C $(KDIR) M=$(PWD) clean
test.sh
#!/bin/shsudo makesudo rmmod log.kosudo insmod log.koecho 1 2 > /proc/mylogcat /proc/mylog./test.sh
测试:
运行./test.sh
读取日志
插入日志
echo cat测试
参考:
linux 在 /proc 里实现文件 - 樊伟胜 - 博客园 (cnblogs.com)
https://www.cnblogs.com/fanweisheng/p/11141527.html
linux内核seq_file接口
https://www.cnblogs.com/embedded-linux/p/9751995.html
相关文章:

ring_log环形日志-6M缓冲区_proc接口
文章目录log_tools.clog.cspin_lockseq_putsseq_readseq_writesingle_openmakefiletest.sh测试:运行./test.sh读取日志插入日志echo cat测试参考:log_tools.c #include <stdlib.h> #include <stdio.h> #include <sys/types.h> #includ…...

Linux内核进程管理几种CPU调度策略
CPU调度我们知道,程序需要获得CPU的资源才能被调度和执行,那么当一个进程由于某种原因放弃CPU然后进入阻塞状态,下一个获得CPU资源去被调度执行的进程会是谁呢?下图中,进程1因为阻塞放弃CPU资源,此时&#…...

SpringBoot整合Flink(施耐德PLC物联网信息采集)
SpringBoot整合Flink(施耐德PLC物联网信息采集)Linux环境安装kafka前情:施耐德PLC设备(TM200C16R)设置好信息采集程序,连接局域网,SpringBoot订阅MQTT主题,消息转至kafka,…...

DFS(深度优先搜索)和BFS(宽度优先搜索)
目录 DFS(深度优先搜索) 全排列的DFS解法 利用DFS递归构建二进制串和递归树的结构剖析 DFS--剪枝 DFS例题--整数划分 BFS(宽度优先搜索) 全排列的BFS解法 DFS(深度优先搜索) 深度优先搜索(Depth First Search&…...

Redis缓存穿透、击穿、雪崩问题及解决方法
系列文章目录 Spring Cache的使用–快速上手篇 分页查询–Java项目实战篇 全局异常处理–Java实战项目篇 完善登录功能–过滤器的使用 上述只是部分文章,对该系列文章感兴趣的可以查看我的主页哦 文章目录系列文章目录前言一、缓存穿透1.1 问题引入1.2 解决方法1.…...

HAL库 STM32 串口通信
一、实验条件将STM32的PA9复用为串口1的TX,PA10复用为串口1的RX。STM32芯片的输出TX和接收RX与CH340的接收RX和发送TX相连(收发交叉且PCB上默认没有相连,所以需要用P3跳线帽进行手动连接),CH340的另一端通过USB口引出与…...

2023-第十四届蓝桥杯冲刺计划!
💬前言 💡本文以目录形式列举大纲,可根据题目点击跳转 🌈冲刺阶段目的:把握高频重点,结合基础算法和常考题型总结,用真题进行模拟练习 根据自己的能力熟练目前已掌握的算法,不会的还可以暴力 ⏳最后三个星期大家一起冲…...

内网渗透基础知识
一、内网概述 内网也指局域网,是指在某一区域内又多台计算机互联成的计算机组。一般是方圆几千米内,局域网可以实现文件管理,应用软件共享,打印机共享,工作组内的历程安排,电子邮件和传真通信服务等功能。…...

鸟哥的Linux私房菜 正则表示法与文件格式化处理
第十一章、正则表示法与文件格式化处理 https://linux.vbird.org/linux_basic/centos7/0330regularex.php 简体版 http://cn.linux.vbird.org/linux_basic/0330regularex.php 11.2.2 grep的一些高级选项 例题一、搜索特定字符串 例题二、利用中括号 [] 来搜寻集合字符 例题四…...
1630.等差子数组
1630. 等差子数组 难度中等 如果一个数列由至少两个元素组成,且每两个连续元素之间的差值都相同,那么这个序列就是 等差数列 。更正式地,数列 s 是等差数列,只需要满足:对于每个有效的 i , s[i1] - s[i] …...

CSS 属性计算过程
CSS 属性计算过程 你是否了解 CSS 的属性计算过程呢? 有的同学可能会讲,CSS属性我倒是知道,例如: p{color : red; }上面的 CSS 代码中,p 是元素选择器,color 就是其中的一个 CSS 属性。 但是要说 CSS 属…...

ThinkPHP02:路由
ThinkPHP02:路由一、路由定义二、变量规则三、路由地址四、路由参数五、路由分组六、MISS七、资源路由八、注解路由九、URL生成一、路由定义 路由默认开启,在 config/app.php 中可以关闭路由。 路由配置在 config/route.php 中,路由定义在 r…...

制作简单进销存管理系统(C#)
实验三:制作简单进销存管理系统 任务要求: 在进销存管理系统中,商品的库存信息有很多种类,比如商品型号、商品名称、商品库存量等。在面向对象编程中,这些商品的信息可以存储到属性中,然后当需要使用这些…...

css总结9(过渡和2D变换)
目录 过渡 2D变换 3D变换 过渡 属性结构图 过渡补充 ### 过渡多个元素样式属性 transition:style1 duration , style2 duration,...; ### 过渡所有属性 transition: all duration; 简单示例 ### 移入时改变长度且加入过渡效果 div { width:100px; height:100px; …...

SpringBoot 结合RabbitMQ与Redis实现商品的并发下单【SpringBoot系列12】
SpringCloud 大型系列课程正在制作中,欢迎大家关注与提意见。 程序员每天的CV 与 板砖,也要知其所以然,本系列课程可以帮助初学者学习 SpringBooot 项目开发 与 SpringCloud 微服务系列项目开发 1 项目准备 SpringBoot 整合 RabbitMQ 消息队…...

【python进阶】序列切片还能这么用?切片的强大比你了解的多太多
📚引言 🙋♂️作者简介:生鱼同学,大数据科学与技术专业硕士在读👨🎓,曾获得华为杯数学建模国家二等奖🏆,MathorCup 数学建模竞赛国家二等奖🏅,…...

[数据结构]直接插入排序、希尔排序
文章目录排序的概念和运用排序的概念排序运用常见的排序算法常见的排序算法直接插入排序希尔排序性能对比排序的概念和运用 排序的概念 排序:所谓排序,就是使一串记录,按照其中的某个或某些关键字的大小,递增或递减的排列起来的操…...

CNN、LeNet、AlexNet、VGG、GoogLeNet、RCNN、Fast RCNN、Faster RCNN、YOLO、YOLOv2、SSD等的关系
卷积神经网络的现状1943年美国数学家提出人工智能1949年心理学家建立神经元模型1957年弗兰克提出 感知器人工神经网络模型1980年建立多层感知器模型1984日本学者提出卷积神经网络原始模型神经感知机1998年提出LeNet-5卷积神经网络,并发展了其在音符和字符上的优势20…...

IO-day1-(fscanf、fprintf.........)
作业一、有一个usr.txt的文件,其中存储着用户的账户和密码,格式如下:zhangsan aaaalisi bbbbb空格前面是账户,空格后面是密码,一行一个账户、密码要求如下:从终端获取一个账户名和密码判断是否能够登录成功…...

C++类和对象(上篇)
目录 1.类的定义 2.类的访问限定符及封装 2.1类的访问限定符 2.2封装 3.类的作用域 4.类的实例化 5.类的大小 6.this 指针 1.类的定义 class className {// 类体:由成员函数和成员变量组成}; // 一定要注意后面的分号 class为定义类的关键字,Clas…...

解决Xshell无法连接Kali Linux 2020.1(2019.3)版本
使用Xshell远程终端工具连接虚拟机的Kali Linux却提示连接不上原因:Kali Linux默认没有打开SSH远程登录,SSH就是一种网络协议,用于加密的远程登录,所以在没有打开SSH协议之前是无法使用Xshell连接Kali Linux的。解决办法ÿ…...

项目文章 | 缓解高胆固醇血症 ,浒苔多糖如何相助?
文章标题:Polysaccharides from Enteromorpha prolifera alleviate hypercholesterolemia via modulating the gut microbiota and bile acid metabolism 发表期刊:Food & Function 影响因子:6.317 作者单位:福建医科大…...

Linux使用宝塔面板搭建网站,并内网穿透实现公网访问
文章目录前言1. 环境安装2. 安装cpolar内网穿透3. 内网穿透4.固定http地址5. 配置二级子域名6.创建一个测试页面前言 宝塔面板作为简单好用的服务器运维管理面板,它支持Linux/Windows系统,我们可用它来一键配置LAMP/LNMP环境、网站、数据库、FTP等&…...

基于深度学习方法与张量方法的图像去噪相关研究
目录 1 研究现状 1.1 基于张量分解的高光谱图像去噪 1.2 基于深度学习的图像去噪算法 1.3 基于深度学习的高光谱去噪 1.4 小结 2 基于深度学习的图像去噪算法 2.1 深度神经网络基本知识 2.2 基于深度学习的图像去噪网络 2.3 稀疏编码 2.3.1 传统稀疏编码 2.3.2 群稀…...

Java基础知识之HashMap的使用
一、HashMap介绍 HashMap是Map接口的一个实现类(HashMap实现了Map的接口),它具有Map的特点。HashMap的底层是哈希表结构。 Map是用于保存具有映射关系的数据集合,它具有双列存储的特点,即一次必须添加两个元素…...

面试--每日一经
操作系统 死锁 死锁:是指两个或两个以上的进程在执行过程中,由于竞争资源或者由于彼此通信而造成的一种阻塞的现象,若无外力作用,它们都将无法推进下去。 死锁的四个必要条件 互斥条件:一个资源每次只能被一个进…...

JavaSE进阶之(十六)枚举
十六、枚举16.1 背景16.2 枚举类型16.3 EnumSet 和 EnumMap01、EnumSet02、EnumMap16.1 背景 在 Java 语言中还没有引入枚举类型之前,表示枚举类型的常用模式是声明一组 int 类型的常量,常常用的就是: public static final int SPRING 1; …...

全同态加密:TFHE
参考文献: Cheon J H, Stehl D. Fully homomophic encryption over the integers revisited[C]//Advances in Cryptology–EUROCRYPT 2015: 34th Annual International Conference on the Theory and Applications of Cryptographic Techniques, Sofia, Bulgaria, …...

【计算机二级】综合题目
计算机二级python真题 文章目录计算机二级python真题一、《大学慕课 两问 》二、综合应用题——价值链三、基本操作题 ——信息输出一、《大学慕课 两问 》 附件中的文件data.txt 是教育部爱课程网中国大学MOOC平台的某个 HTML页面源文件,里面包含了我国参与MOOC建设的一批大学…...

初识Kafka
介绍 Kafka Kafka 是一款基于发布与订阅的消息系统。 用生产者客户端 API 向 Kafka 生产消息,用消费者客户端 API 从 Kafka 读取这些消息。 Kafka 使用 Zookeeper 保存元数据信息。 Kafka 0.9 版本之前,除了 broker 之外, 消费者也会使用…...