背景:

笔者由于一个项目,这段时间在使用FreeBSD进行内核模块的编程。

之前做过一段时间的Linux下驱动模块编程。对Linux下的模块编程还算熟悉。

如今突然转到FreeBSD下。尽管Linux和FreeBSD两者都是类Unix系统。在实现机制上还是区别非常大的,多少有点不习惯。笔者的项目要在一个插入进内核的模块里。将log信息写入到磁盘的中。在用户层能够用读写文件的形式;在内核模块中,与Linux不同的是。并不推荐用户使用将这样的方法移植到内核模块中。

可是FreeBSD却提供了一种ALQ机制。

ALQ机制依赖于ALQ模块,在sys/conf/files中,能够看到它是可选的。

简单介绍:

ALQ(Asynchronous Logging Queues) 异步日志队列,提供了一种异步的定长和变长的record机制。

它可以记录不论什么vnode。因此能对日志、字符设备及普通文件进行操作。

全部的功能函数接受一个struct alq的參数,struct alq是一种隐含类型,维护一个ALQ的状态信息。这样的logging机制作为一个独立的内核线程来执行的,可以为系统全部的log请求服务。每个ale(asynchronous
log entry)定义成一个struct ale,结构例如以下:

<span style="font-size:12px;"><span style="font-size:10px;">	   struct ale {
intptr_t ae_bytesused; /* # bytes written to ALE. */
char *ae_data; /* Write ptr. */
int ae_pad; /* Unused, compat. */
};</span></span>

一个alq能够以定长或变长的两种模式创建。可变长度的alq适应alq_writen()alq_getn()使用时写的长度。 而定长的alq在队列创建时就已经指定好了。定长模式如今已经被弃用,能够使用变模式来取代它。

函数:

#include <sys/alq.h>

int alq_open_flags(struct alq **app, const char *file, struct ucred *cred, int cmode, int size, int flags);  创建一个可变长的alq。

file为文件名称。

假设文件不存在,alq_open()会试图创建它,而且cmode參数会被传递给alq_open()作为创建文件的属性。 对多数用户来说这个cmdode应被指定为ALQ_DEFAULT_CMODE。

当打开文件和进行I/O操作时,cred參数指定信任证书。size设置队列的大小(单位为字节)。最后一个參数flag通常被设置成ALQ_ORDERED来  指明写线程等待忙碌的alq释放资源的顺序应该被保留。成功返回0,否则返回错误码(open(2))。



int alq_open(struct alq **app,const char *file,struct ucred *cred, int cmode, int size, int count);被弃用的alq_open()封装在alq_open_flags()中。为了兼容老版本号。

全部的         參数都是通过alq_open_flags()间接传递的。

详细的我在此不作介绍,有兴趣的读者能够參考alq

int alq_writen(struct alq *alq, void *data, int len, int flags);从data把len字节的数据写入到设置地可变长模式的alq中。假设alq_write()不能马上写  入这个数据,同一时候flags參    数又设置了ALQ_WAITOK。那么它将使用msleep_spin(9)睡眠等待数据。

一个写操作会自己主动调度队列alq使     之写入磁盘。这样的行为能够通过给flags參数传递
ALQ_NOACTIVATE。来控制它不会被写入磁盘。假设设置ALQ_NOWAIT或队列已满或系统关机,返回EWOULDBLOCK。
int alq_write(struct alq *alq, void *data, int flags);被弃用的alq_write()封闭在alq_writen()中。有兴趣的读者能够參考query=alq&apropos=0&sektion=0&manpath=FreeBSD+10.1-RELEASE&arch=default&format=html">alq
void alq_flush(struct alq *alq);刷新alq到log文件。

假设alq有数据可被刷新。而且没有没有进行在刷新。那么它将会堵塞。去进行IO操作。

否则马上返回。

void alq_close(struct alq *alq);关闭alq。刷新全部的挂起的写请求到log文件。释放申请的全部资源。

struct ale * alq_getn(struct alq *alq, int len, int flags);从alq队列中返回一个ale。它会让alq处于锁定状态直到子alq_post()或alq_post_flags被调用。假设
alq_getn()不能马上获取到len长度字节数据,而且在flags中设置了ALQ_WAITOK,它将会堵塞等待数据。调用者能够通过事先设置struct ale中的 ae_bytesused域,选择返回小于len字节到ale中。假设设置ALQ_NOWAIT或队列已满或系统关机,返回NULL。
struct ale * alq_get(struct alq *alq, int flags); 被弃用。封装于alq_getn()中。
void alq_post_flags(struct alq *alq, struct ale *ale, int flags);调度ale(从alq_getn()或alq_get()中获取)用于写alq。

能够在flags參数中设置ALQ_NOACTIVATE指明队列不会马上被刷新到磁盘。 会对alq进行解锁。

參考文献:

    ALQ

FreeBSD内核之中的一个 ALQ机制的使用的更多相关文章

  1. FreeBSD 5.0中强制访问控制机制的使用与源代码分析【转】

    本文主要讲述FreeBSD 5.0操作系统中新增的重要安全机制,即强制访问控制机制(MAC)的使用与源代码分析,主要包括强制访问控制框架及多级安全(MLS)策略两部分内容.这一部分讲述要将MAC框架与 ...

  2. Linux 内核中的 Device Mapper 机制

    本文结合具体代码对 Linux 内核中的 device mapper 映射机制进行了介绍.Device mapper 是 Linux 2.6 内核中提供的一种从逻辑设备到物理设备的映射框架机制,在该机 ...

  3. 【转】深入Windows内核——C++中的消息机制

    上节讲了消息的相关概念,本文将进一步聊聊C++中的消息机制. 从简单例子探析核心原理 在讲之前,我们先看一个简单例子:创建一个窗口和两个按钮,用来控制窗口的背景颜色.其效果 图1.效果图  Win32 ...

  4. Linux 内核的文件 Cache 管理机制介绍

    Linux 内核的文件 Cache 管理机制介绍 http://www.ibm.com/developerworks/cn/linux/l-cache/ 1 前言 自从诞生以来,Linux 就被不断完 ...

  5. [转] Linux 内核中的 Device Mapper 机制

    本文结合具体代码对 Linux 内核中的 device mapper 映射机制进行了介绍.Device mapper 是 Linux 2.6 内核中提供的一种从逻辑设备到物理设备的映射框架机制,在该机 ...

  6. Linux内核分析-构造一个简单的Linux系统MenuOS

    构造一个简单的Linux系统MenuOS linux内核目录结构 arch目录包括了所有和体系结构相关的核心代码.它下面的每一个子目录都代表一种Linux支持的体系结构,例如i386就是Intel C ...

  7. 【IPC进程间通讯之中的一个】邮槽MailSlot

    IPC进程间通信+邮槽MailSlot                IPC(Inter-Process Communication.进程间通信).        现代计算机採用虚拟内存机制,为进程提 ...

  8. 【eclipse插件开发实战】Eclipse插件开发1——eclipse内核结构、扩展点机制

    Eclipse插件开发实战1--eclipse内核结构.扩展点机制 一.前言 本系列总体介绍eclipse插件开发基本理论.插件项目结构及开发步骤,最后再给出两个插件开发实例. 总体安排结构如下: 1 ...

  9. Objective-C(十九、通知-消息发送模式之中的一个)——iOS开发基础

    结合之前的学习笔记以及參考<Objective-C编程全解(第三版)>,对Objective-C知识点进行梳理总结. 知识点一直在变.仅仅是作为參考.以苹果官方文档为准~ 十九.通知-消息 ...

随机推荐

  1. python--11、协程

    协程,又称微线程,纤程.英文名Coroutine. 子程序,或者称为函数,在所有语言中都是层级调用,比如A调用B,B在执行过程中又调用了C,C执行完毕返回,B执行完毕返回,最后是A执行完毕. 所以子程 ...

  2. JS——锚点的运用

    锚点的两种形式: 1.<a href="#a">点击到锚点</a> 2.window.location.hash = "#a"; 最后都 ...

  3. RadioButtonList的兩種實現方式

    一種是修改ItemTemplate,即ListBoxItem裏面的内容 <ListBox ItemsSource="{Binding}"> <ListBox.It ...

  4. Android本地消息推送

    项目介绍:cocos2dx跨平台游戏 项目需求:实现本地消息推送,需求①:定点推送:需求②:根据游戏内逻辑实现推送(比如玩家体力满时,需要计算后到点推送):需求③:清理后台程序或重启后依然能够实现本地 ...

  5. UIWebview 截获html并修改内容。

    需求:混合应用UIWebView打开html后,UIWebView有左右滚动条,要去掉左右滚动效果: 方法:通过js截获UIWebView中的html,然后修改html标签内容:  实例代码:  服务 ...

  6. Random同时生成多个随机数

    贴一个简单示例 public DataTable selectStuInfo() { DataTable dt = new DataTable(); dt.Columns.Add("姓名&q ...

  7. javascript 大数据处理方法

    随着前端的飞速发展,在浏览器端完成复杂的计算,支配并处理大量数据已经屡见不鲜.那么,如何在最小化内存消耗的前提下,高效优雅地完成复杂场景的处理,越来越考验开发者功力,也直接决定了程序的性能. 本文展现 ...

  8. js里的深度克隆

    ES6 数组克隆 let arr = [1,2,3,4,5]; let arr1 = [...a]; arr1 = ["a","b","c" ...

  9. Yin and Yang Stones(思路题)

    Problem Description: A mysterious circular arrangement of black stones and white stones has appeared ...

  10. Spring 注解注入的几种方式(转)

    平常的java开发中,程序员在某个类中需要依赖其它类的方法,则通常是new一个依赖类再调用类实例的方法,这种开发存在的问题是new的类实例不好统一管理,spring提出了依赖注入的思想,即依赖类不由程 ...