struct mm_struct {
..........
#if defined(__GENKSYMS__) || !defined(CONFIG_SPAPR_TCE_IOMMU)
/* We're adding a list_head, so we need to take two reserved
* fields, unfortunately there are no handy RH_KABI macros for
* that case */
RH_KABI_RESERVE(1)
RH_KABI_RESERVE(2)
#else
struct list_head iommu_group_mem_list;
#endif #ifdef CONFIG_X86_INTEL_MPX
RH_KABI_USE(3, void __user *bd_addr)
#else
/* RHEL7: consumed by x86, avoid re-use by other arches */
RH_KABI_RESERVE(3)
#endif
/* This would be in rss_stat[MM_SHMEMPAGES] if not for kABI */
RH_KABI_USE(4, atomic_long_t mm_shmempages) #if IS_ENABLED(CONFIG_HMM)
/* HMM need to track few things per mm */
RH_KABI_USE(5, struct hmm *hmm)
#else
RH_KABI_RESERVE(5)
#endif RH_KABI_RESERVE(6)
RH_KABI_RESERVE(7)
RH_KABI_RESERVE(8)
};

  在struct mm_struct 或者 task_struct 之类的结构中,我们经常能看到 RH_KABI_RESERVE 这种成员,这种成员其实就是为了增加扩展性所留的一手。

这个可以类比tcp协议头,也有tcp_option,原理也是类似的,那么,怎么使用这些成员呢?

比如我想在task里面,记录一些私有的数据,这个数据由于类型只有 unsigned long 类型,所以这个既可以直接用来记录普通数据,也可以用来存放一个指针,然后再指向一个地址,这样存放的内容就多了。

#define _RH_KABI_RESERVE(n)		unsigned long rh_reserved##n
#define _RH_KABI_RESERVE_P(n) void (*rh_reserved##n)(void)
#define RH_KABI_RESERVE(n) _RH_KABI_RESERVE(n);

  举个栗子:

	/* This would be in rss_stat[MM_SHMEMPAGES] if not for kABI */
RH_KABI_USE(4, atomic_long_t mm_shmempages)

  没有用的变量,是 RH_KABI_RESERVE,那么已经使用的,则是 RH_KABI_USE ,我们看到 mm_shmempages 使用的是4号位置。

之后,就可以直接对 mm_shmempages 成员进行赋值了。

比如在 mm_init 中正常使用  mm_shmempages

static struct mm_struct *mm_init(struct mm_struct *mm, struct task_struct *p)
{
atomic_set(&mm->mm_users, 1);
atomic_set(&mm->mm_count, 1);
init_rwsem(&mm->mmap_sem);
INIT_LIST_HEAD(&mm->mmlist);
mm->core_state = NULL;
atomic_long_set(&mm->nr_ptes, 0);
memset(&mm->rss_stat, 0, sizeof(mm->rss_stat));
atomic_long_set(&mm->mm_shmempages, 0);-----------正常当成员使用,因为 RH_KABI_USE 这个宏已经帮我们替换好了。

  

不过需要注意的是,要及时清理,比如我们使用了task_struct里面的RH_KABI_RESERVE,但是在fork的时候忘了清理该成员,导致了一些问题。

RH_KABI_RESERVE的使用的更多相关文章

随机推荐

  1. ha环境下重新格式化hdfs报错

    datanode启动不成功,如下所示,我的136,137.138都是datanode,都启动不了. 查看datanode日志文件发现报错: 一个报错Incompatible clusterIDs in ...

  2. Windows配置多个git用户

    Window配置多个Git账户,SSH连接GitHub.GitLab 最新版本GIt配置对应多个Git仓库(不需要添加多个用户名和邮箱): 在本地git上添加一个用户名和邮箱,生成一对公钥和私钥,把公 ...

  3. U3d学习001-RollBox例子

    1.世界坐标系和局部坐标系(参照物坐标系)——以参照物为父物体节点  2.刚体组件:       获得GetComponent<Rigidbody>();     移动AddForce(n ...

  4. JDBC连接数据库,结合DbUtil数据库连接工具类的使用

    (以Mysql数据库为例) 第一步:在项目里配置数据库驱动 Build Path->configure  Build Path ->Add Exteral JARs   将JDBC驱动包导 ...

  5. 第5章 IP地址和子网划分(4)_超网合并网段

    7. 超网合并网段 7.1 合并网段 (1)子网划分是将一个网络的主机位当网络位,来划分出多个子网.而多个网段合并成一个大网段,合并后的网段称为超网. (2)需求分析 某企业有一个网段,该网段有200 ...

  6. web前端 3大储存 Cookie、localStorge、sessionStorage

    Cookie: //setCookie function setCookie(name,value){ var Days = 30; var exp = new Date(); exp.setTime ...

  7. Android短信收发(二)

    接收SMS类,代码如下 //for receive SMS private SmsReceiver mSmsReceiver; @Override protected void onResume() ...

  8. Cookie快速入门实践

    第一个servlet[比如是CookieDemo01]中的代码如下: import javax.servlet.http.Cookie; //--------省略若干代码----------- pro ...

  9. sqoop导入导出

    sqoop产生背景 什么是sqoop sqoop的优势 sqoop1与sqoop2的比较 为什么选择sqoop1 sqoop在hadoop生态体系中的位置 sqoop基本架构 sqoop import ...

  10. 理解Solr缓存及如何设置缓存大小

    文献地址:http://wangdg.com/understanding-and-tuning-solr-cache/ 理解Solr缓存及如何设置缓存大小 为了得到最好的检索性能,Solr会在内存中缓 ...