xenomai内核解析---内核对象注册表—xnregistry(重要组件)
1. 概述
上篇文章xenomai内核解析--同步互斥机制(一)--优先级倒置讲到,对于所有内核对象:
xnregistry:保存内核对象,提供内核对象存储和快速检索。
xnsynch:资源抽象,提供线程与资源的同步互斥管理机制。
举个应用例子,有两个xenoami任务,使用semaphore做互斥,任务1创建一个名为/test-sem的semaphore,任务2打开这个semaphore,以这个过程为例,带你了解xnregistry。
/*任务1*/
sem_t *dome_sem;
.....
dome_sem = sem_open("/test-sem", O_CREAT | O_EXCL, 0666, 0);
if (dome_sem == SEM_FAILED)
error(1, errno, "sem_open()");
.....
sem_wait(dome_sem);
.....
sem_post(dome_sem);
.....
/*任务2*/
sem_t *dome_sem;
.....
dome_sem = sem_open("/test-sem", 0);
if (dome_sem == SEM_FAILED)
error(1, errno, "sem_open()");
.....
sem_wait(dome_sem);
.....
sem_post(dome_sem);
.....
- 问题1:任务1创建的这个semaphore是如何管理的?
- 问题2:任务2又是如何通过name找到它的?
本片文章解析xenomai内核中的xnregistry。至于xenomai semaphore具体的内核机制及创建流程,以后文章介绍,敬请关注。
2. 命名内核对象管理
xnregistry用于保存xenomai全局内核对象。这些对象分为两种,一种有name,常用于两个及以上进程间,可以通过name来找到同一对象。另一种没有name,常用于同一进程空间。
涉及通过字符串name来查找的内核对象称为命名内核对象,xenomai内核中名内核对象有:有名信号量(sem)、有名消息队列(mq)、进程间通过label相互通讯的xddp/bufp/iddp等。
创建命名对象的时候,
- 先从
register_obj_solts中分配一个xnobject; - 然后将name作为xnobject成员
key,具体对象(cobalt_sem)的地址作为value保存到xnobject的成员objaddr中。 - 向xnregistry中存储该xnobject时,会将name作为key经过hash运算,根据运算得到的hash值
s,选择合适的hash bucket,该bucket在xenomai中为object_index[],然后将xnobject插入选定的object_index[s]链表中。 - 当另一个线程open同一name的对象时,通过name可从
object_index[]中快速得到该内核对象。
name只在创建和通过name查找时使用,一个对象通过name查找或创建后会保存一个xnhandlet,后续操作使用xnhandlet代替,提高xnobjet的访问速度。
创建时注册cobalt_sem流程如下所示。

register_obj_solts用于保存型号量这个xnobject,上面的问题1.object_index用于检索,上面的问题2。
我们接着来看创建后保存的这个xnhandlet,cobalt_sem创建完成后会保存xnhandlet到信号量句柄sem_t中,并拷贝到用户空间,我们可以来看一下libcobalt中的句柄sem_t的形式:
struct cobalt_sem_shadow {
__u32 magic;
__s32 state_offset;
xnhandle_t handle;
} shadow_sem;
从上面图中我们可以看到xnhandlet是一个偏移量,表示这个xnobject基于register_obj_solts的地址偏移,为什么要直接保存到句柄sem_t中呢?sem_wait()/sem_post()操作进入内核的时候就可以直接去获取xnobject做相应的操作了。
另外想一下,一个运行在用户态的实时应用,每次PV操作的时候都需要执行系统调用,对实时系统来说不太友好,毕竟系统调用也是需要花费时间的,xnhandlet只能解决内核里定位xnobject的速度问题,我们能不能不要每次都执行系统调用呢?答案是肯定的,xenomai有相应的机制,请关注后续文章,呵呵~~。
3. 未命名内核对象管理
上面说完了命名内核对象,下面来看未命名内核对象,即非跨进程共享的。
对于没有name的内核对象,通过xnregistry提供的匿名接口来保存。所谓的匿名保存,key为NULL,具体对象(cobalt_sem)的地址作为value到一个分配的结构体xnobject后,不经hash运算,直接计算xnobject基于某个固定地址的偏移量xnhandle_t,通常xnhandle_t会在用户空间的对象结构体中保存一份,比如sem_t、pthread_mutex_t等;之后用户空间对该对象发起系统调用时就可以通过xnhandle_t快速从xnregistry获取该对象,使用匿名的内核对象有:进程间的互斥量mutex、未命名信号量sem、条件变量condition variable、事件event、monitor。
同样以未命名信号量为例,内核对象cobalt_sem注册流程如下。

4. xnregistry初始化流程
图中resitry_obj_slots[]其大小内核构建时CONFIG_XENO_OPT_REGISTRY_NRSLOTS指定,默认大小512,具体内存在xenomai初始化时调用xnregistry_init()初始化xnregistry时分配。
static int __init xenomai_init(void)
->sys_init()
->xnregistry_init()
xnregistry_init()具体流程如下。
int xnregistry_init(void)
{
int n, ret __maybe_unused;
registry_obj_slots = kmalloc(CONFIG_XENO_OPT_REGISTRY_NRSLOTS *
sizeof(struct xnobject), GFP_KERNEL);
.....
#ifdef CONFIG_XENO_OPT_VFILE
ret = xnvfile_init_dir("registry", ®istry_vfroot, &cobalt_vfroot);
ret = xnvfile_init_regular("usage", &usage_vfile, ®istry_vfroot);
proc_apc = xnapc_alloc("registry_export", ®istry_proc_schedule, NULL);
#endif /* CONFIG_XENO_OPT_VFILE */
next_object_stamp = 0;
for (n = 0; n < CONFIG_XENO_OPT_REGISTRY_NRSLOTS; n++) {
registry_obj_slots[n].objaddr = NULL;
list_add_tail(®istry_obj_slots[n].link,
&free_object_list);
}
/* Slot #0 is reserved/invalid. */
list_get_entry(&free_object_list, struct xnobject, link);
nr_active_objects = 1;
nr_object_entries = xnregistry_hash_size();
object_index = kmalloc(sizeof(*object_index) *
nr_object_entries, GFP_KERNEL);
for (n = 0; n < nr_object_entries; n++)
INIT_HLIST_HEAD(&object_index[n]);
xnsynch_init(®ister_synch, XNSYNCH_FIFO, NULL);
return 0;
}
1.先分配CONFIG_XENO_OPT_REGISTRY_NRSLOTS个xnobject的空间,xenomai运行过程中的xnobject从registry_obj_slots中直接获取,这样就避免频繁的内存申请影响实时性。struct xnobject结构如下:
struct xnobject {
void *objaddr;
const char *key; /* !< Hash key. May be NULL if anonynous. */
unsigned long cstamp; /* !< Creation stamp. */
#ifdef CONFIG_XENO_OPT_VFILE
struct xnpnode *pnode; /* !< v-file information class. */
union {
struct {
struct xnvfile_rev_tag tag;
struct xnvfile_snapshot file;
} vfsnap; /* !< virtual snapshot file. */
struct xnvfile_regular vfreg; /* !< virtual regular file */
struct xnvfile_link link; /* !< virtual link. */
} vfile_u;
struct xnvfile *vfilp;
#endif /* CONFIG_XENO_OPT_VFILE */
struct hlist_node hlink; /* !< Link in h-table */
struct list_head link;
};
objaddr指向具体的内核对象,如cobalt_sem、cobalt_mutex等。
*key 对象的name或label,用户程序可使用name来操作内核对象,具有name的内核对象会保存到一个hash表中,方便通过name查找。如果key为NULL,则不用。
vfilp、vfile_u、pnode 注册到linux虚拟文件系统常用变量。
hlink用于加入hash链表。
link 该对象如果未使用则用于加入空闲链表free_object_list,否则用于加入已使用链表busy_object_list。
- 在linux虚拟文件系统proc目录下创建
registry目录,以及文件usage,注册后可通过/proc/xenomai/registry/usage查看xnobject的使用情况。
$ cat /proc/xenomai/registry/usage
7/512
- 将刚分配的空闲xnobject插入free_object_list链表。
- 将free_object_list的第一个xnobject节点保留,记录xnobject已使用数nr_active_objects。
- 分配散列表object_index[]的空间,并初始化。
- 初始化registry资源同步对象
register_synch,register_synch
5. xnregistry总结
xnregistry:保存内核对象,提供内核对象存储和快速检索。
xnsynch:资源抽象,提供线程与资源的同步互斥管理机制。
xnsynch、xnregistry是xenomai内核机制非常重要的组件,明白他们xenomai的资源管理机制就明白大半了。
xenomai内核解析---内核对象注册表—xnregistry(重要组件)的更多相关文章
- 【原创】xenomai内核解析--xenomai与普通linux进程之间通讯XDDP(二)--实时与非实时关联(bind流程)
版权声明:本文为本文为博主原创文章,转载请注明出处.如有问题,欢迎指正.博客地址:https://www.cnblogs.com/wsg1100/ 1.概述 上篇文章介绍了实时端socket创建和配置 ...
- 【原创】xenomai内核解析--xenomai与普通linux进程之间通讯XDDP(一)--实时端socket创建流程
版权声明:本文为本文为博主原创文章,转载请注明出处.如有问题,欢迎指正.博客地址:https://www.cnblogs.com/wsg1100/ 1.概述 上篇文章xenomai内核解析--实时IP ...
- 《天书夜读:从汇编语言到windows内核编程》八 文件操作与注册表操作
1)Windows运用程序的文件与注册表操作进入R0层之后,都有对应的内核函数实现.在windows内核中,无论打开的是文件.注册表或者设备,都需要使用InitializeObjectAttribut ...
- xenomai内核解析之信号signal(二)---xenomai信号处理机制
xenomai信号 上篇文章讲了linux的信号在内核的发送与处理流程,现在加入了cobalt核,Cobalt内核为xenomai线程提供了信号机制.下面一一解析xenomai内核的信号处理机制. 1 ...
- 【原创】xenomai内核解析--同步互斥机制(一)--优先级倒置
版权声明:本文为本文为博主原创文章,转载请注明出处.如有错误,欢迎指正.博客地址:https://www.cnblogs.com/wsg1100/ 目录 一.xenomai 资源管理简要 二.优先级倒 ...
- 【xenomai内核解析】系列文章大纲
xenomai内核解析 本博客为本人学习linux实时操作系统框架xenomai的一些记录,主要剖析xenomai内核实现,以及与linux相关的知识.方便读者定位具体文章,现列出本博客大纲,后续会陆 ...
- 驱动开发:内核枚举Registry注册表回调
在笔者上一篇文章<驱动开发:内核枚举LoadImage映像回调>中LyShark教大家实现了枚举系统回调中的LoadImage通知消息,本章将实现对Registry注册表通知消息的枚举,与 ...
- Windows进程的内核对象句柄表
当一个进程被初始化时,系统要为它分配一个句柄表.该句柄表只用于内核对象 ,不用于用户对象或GDI对象. 创建内核对象 当进程初次被初始化时,它的句柄表是空的.然后,当进程中的线程调用创建内核对象的函数 ...
- 【原创】xenomai内核解析--双核系统调用(二)--应用如何区分xenomai/linux系统调用或服务
版权声明:本文为本文为博主原创文章,转载请注明出处.如有错误,欢迎指正. 1. 引出问题 上一篇文章xenomai内核解析--双核系统调用(一)以X86处理器为例,分析了xenomai内核调用的流程, ...
随机推荐
- 暑假集训Day0
啊这 跟学长学的要写日记 希望到时候能写省选集训的总结 咳咳 今天上午做了一上午苦力好像让老苏夸了难以接受(年纪两百考到年级两千他居然没有干我) 上午搞卫生搞到了十点半………… 替女生拉包提东西了!! ...
- Srapy 爬取知乎用户信息
今天用scrapy框架爬取一下所有知乎用户的信息.道理很简单,找一个知乎大V(就是粉丝和关注量都很多的那种),找到他的粉丝和他关注的人的信息,然后分别再找这些人的粉丝和关注的人的信息,层层递进,这样下 ...
- java.math.BigDecimal cannot be cast to [Ljava.lang.Object;
从数据库中使用sum函数取出统计值后,放进list中,遍历list的时候强转化成Object是报错. BigDecimal .Integer不是基本类型,是int的包装类,无法把包装当做基本类型来用. ...
- 51单片机入门(补充)1--与C语言的交接
我写完上一个文章,发现我写的还是不够全面,所以,这篇文章将会延续上一个文章的内容,并且再次补充新的东西,如果还有什么地方需要补充,还请各位一一指出,如果你已经学过这些东西,大可以直接跳过,假如说之后有 ...
- 113资讯网:安装程序进入Admin后台出现:SQLSTATE[HY000] [1045] Access denied for user'root'@'localhost' (using password: YES)
各项设置设置正确,就是出现这种原因! 1.config.inc.php解决办法: 修改phpMyAdmin的配置文件里的密码设置,进入phpMyAdmin的安装目录,找到config.inc.php配 ...
- ASP.NET 开源导入导出库Magicodes.IE 多Sheet导入教程
多Sheet导入教程 说明 本教程主要说明如何使用Magicodes.IE.Excel完成多个Sheet数据的Excel导入. 要点 多个相同格式的Sheet数据导入 多个不同格式的Sheet数据导入 ...
- 错误记录-MySql.Data.MySqlClient.MySqlException (0x80004005): Timeout expired.
-- ::25.026 +: [ERR] Connection id "0HLQH64H76UL5", Request id "0HLQH64H76UL5:0000000 ...
- SELinux已经允许,为什么日志显示的仍然是denied?
从日志可以看到,SELinux的Mode已经修改位了permissive = 1,也就是允许模式,但它前面的日志仍然显示的是“denied".本来我还以为是自己哪里没弄好导致的这个问题,但访 ...
- Python3笔记011 - 3.2 选择语句
第3章 流程控制语句 3.2 选择语句 1.if语句 if 表达式: 语句块 执行的流程是:当表达式的布尔值为真时,执行语句块,为假时,离开if语句,程序往下执行. 2.if...else语句 if ...
- mysql根据逗号分割的字符串去关联查询另外一个表的数据
1.说明 在做显示数据的时候,一个字段会存那种逗号分割的字符串,那如何去根据逗号分割字符串去查询另一个表的数据呢? 首先我们查看一下需要显示的数据 select * from company wher ...