免费注册 查看新帖 |

Chinaunix

  平台 论坛 博客 文库
12下一页
最近访问板块 发新帖
查看: 6073 | 回复: 12
打印 上一主题 下一主题

内核hash表问题 [复制链接]

论坛徽章:
0
跳转到指定楼层
1 [收藏(0)] [报告]
发表于 2010-12-17 17:34 |只看该作者 |倒序浏览
我需要写一个内核模块,里面要使用hash表,请问有可用的hash函数可以使用吗?没开发过kernel模块,新手望指点!

论坛徽章:
36
IT运维版块每日发帖之星
日期:2016-04-10 06:20:00IT运维版块每日发帖之星
日期:2016-04-16 06:20:0015-16赛季CBA联赛之广东
日期:2016-04-16 19:59:32IT运维版块每日发帖之星
日期:2016-04-18 06:20:00IT运维版块每日发帖之星
日期:2016-04-19 06:20:00每日论坛发贴之星
日期:2016-04-19 06:20:00IT运维版块每日发帖之星
日期:2016-04-25 06:20:00IT运维版块每日发帖之星
日期:2016-05-06 06:20:00IT运维版块每日发帖之星
日期:2016-05-08 06:20:00IT运维版块每日发帖之星
日期:2016-05-13 06:20:00IT运维版块每日发帖之星
日期:2016-05-28 06:20:00每日论坛发贴之星
日期:2016-05-28 06:20:00
2 [报告]
发表于 2010-12-17 17:53 |只看该作者
什么意思,hash 表你自己定义就行了吧,冲突连的话,可以用 list_head 或者 hlist

论坛徽章:
0
3 [报告]
发表于 2010-12-17 18:00 |只看该作者
也就是说,自己写一套hash表的操作函数?我觉得这个kernel完全可以提供一套hash函数给用户使用啊。。

论坛徽章:
36
IT运维版块每日发帖之星
日期:2016-04-10 06:20:00IT运维版块每日发帖之星
日期:2016-04-16 06:20:0015-16赛季CBA联赛之广东
日期:2016-04-16 19:59:32IT运维版块每日发帖之星
日期:2016-04-18 06:20:00IT运维版块每日发帖之星
日期:2016-04-19 06:20:00每日论坛发贴之星
日期:2016-04-19 06:20:00IT运维版块每日发帖之星
日期:2016-04-25 06:20:00IT运维版块每日发帖之星
日期:2016-05-06 06:20:00IT运维版块每日发帖之星
日期:2016-05-08 06:20:00IT运维版块每日发帖之星
日期:2016-05-13 06:20:00IT运维版块每日发帖之星
日期:2016-05-28 06:20:00每日论坛发贴之星
日期:2016-05-28 06:20:00
4 [报告]
发表于 2010-12-17 18:08 |只看该作者
自己写一套hash表的操作函数?

你说一下 hash 操作包括哪些函数

论坛徽章:
0
5 [报告]
发表于 2010-12-18 00:01 |只看该作者
create
insert
find
replace
delete
walk
。。。。。。。

论坛徽章:
36
IT运维版块每日发帖之星
日期:2016-04-10 06:20:00IT运维版块每日发帖之星
日期:2016-04-16 06:20:0015-16赛季CBA联赛之广东
日期:2016-04-16 19:59:32IT运维版块每日发帖之星
日期:2016-04-18 06:20:00IT运维版块每日发帖之星
日期:2016-04-19 06:20:00每日论坛发贴之星
日期:2016-04-19 06:20:00IT运维版块每日发帖之星
日期:2016-04-25 06:20:00IT运维版块每日发帖之星
日期:2016-05-06 06:20:00IT运维版块每日发帖之星
日期:2016-05-08 06:20:00IT运维版块每日发帖之星
日期:2016-05-13 06:20:00IT运维版块每日发帖之星
日期:2016-05-28 06:20:00每日论坛发贴之星
日期:2016-05-28 06:20:00
6 [报告]
发表于 2010-12-18 00:24 |只看该作者
create
insert
find
replace
delete
walk
。。。。。。。
挖土机 发表于 2010-12-18 00:01


我更觉得你的这些操作更多是冲突链的操作,内核态有 list.h 定义了双向链表还有 hlist 的大部分操作。

论坛徽章:
0
7 [报告]
发表于 2010-12-20 16:06 |只看该作者
是啊,还是有很多东西要考虑,尤其对我这种没有接触过内核开发的人来说,想看看conntrack的hash实现,没看明白,下面是在网上找的一个例子,基本能看清楚,不知道存不存在问题,Godbach帮忙看看:

  1. /*

  2. *    以下代码 在2.6.23 下调试通过

  3. */

  4. //---------------------------------------hash_table.h-----------------------------------------------

  5. #define EXAMPLE_TAB_BITS 16
  6. #define EXAMPLE_TAB_SIZE (1<<EXAMPLE_TAB_BITS)
  7. #define EXAMPLE_TAB_MASK (EXAMPLE_TAB_SIZE-1)
  8. #define EXAMPLE_F_HASHED 0x0001

  9. struct example
  10. {
  11.         struct list_head node;
  12.         char *s;                           /*主体 */         
  13.         spinlock_t lock;              /*自旋锁 */
  14.         atomic_t refcnt;              /*引用计数*/
  15.         volatile __u16 flags;       /*辅助标记 */
  16. };

  17. extern int example_init(void);                  /* 内存空间 初始化*/

  18. extern void example_cleanup(void);       /* 内存清理回收*/

  19. extern int example_hash(struct example *cp);

  20. extern int example_unhash(struct example *cp);

  21. extern struct example *example_get(char *s);     /*查找*/

  22. extern struct example* example_new(char *p);   /*构造新项,并加入哈希表 */

  23. extern int example_destroy(struct example *cp); /* 销毁一个哈希表中的项*/

  24. extern inline void example_put(struct example *cp);        /*引用计数减一*/

  25. //--------------------------------------hash_example.c---------------------------------------------------

  26. #include <linux/module.h>                  
  27. #include <linux/kernel.h>
  28. #include <linux/init.h>

  29. #include "hash_table.h"

  30. static int __init hash_init(void)
  31. {
  32.         int ret;
  33.         struct example *cp=NULL;
  34.         printk(KERN_ALERT "hash mod insert..............\n");         
  35.         ret=example_init();
  36.         if(ret<0)
  37.         {
  38.                 printk(KERN_INFO "can't setup conrol.\n.");
  39.               
  40.         }
  41.         example_new("My girl_1");
  42.         example_new("My girl_2");
  43.         cp=example_get("My girl_2");
  44.         if(NULL!=cp)
  45.                 printk(KERN_ALERT "cp->s:%s\n",cp->s);
  46.         example_put(cp);
  47.         example_destroy(cp);
  48.         cp=example_get("My girl_2");
  49.         if(NULL==cp)
  50.                 printk(KERN_ALERT "break with girl_2!\n");
  51.         return 0;
  52. }
  53. static void __exit hash_exit(void)
  54. {
  55.        printk(KERN_ALERT "hash mod remove..............\n");
  56.        example_cleanup();
  57. }
  58. module_init(hash_init);
  59. module_exit(hash_exit);

  60. //--------------------------------------hash_table.c---------------------------------------------------

  61. #include <linux/module.h>                  

  62. #include <linux/kernel.h>
  63. #include <linux/init.h>

  64. #include <linux/vmalloc.h>
  65. #include <linux/jhash.h>
  66. #include <linux/random.h>
  67. #include <linux/seq_file.h>
  68. #include <linux/sched.h>

  69. #include "hash_table.h"

  70. /*     SLAB 分配器 */
  71. static struct kmem_cache *example_cachep;
  72. /*      hash table     */
  73. static struct list_head *example_tab;
  74. /*     记录 struct example 总数目,以便回收内存空间   */
  75. static atomic_t example_count=ATOMIC_INIT(0);
  76. /*    hash函数使用的随机值    */
  77. static unsigned int example_rnd;

  78. /*
  79. *    一套锁*/
  80. #define CT_LOCKARRAY_BITS 4
  81. #define CT_LOCKARRAY_SIZE    (1<<CT_LOCKARRAY_BITS)
  82. #define CT_LOCKARRAY_MASK    (CT_LOCKARRAY_SIZE-1)

  83. struct tcp_shift_aligned_lock
  84. {
  85.         rwlock_t l;
  86. }__attribute__((__aligned__(SMP_CACHE_BYTES)));

  87. struct tcp_shift_aligned_lock
  88. __example_conntbl_lock_array[CT_LOCKARRAY_SIZE] __cacheline_aligned;

  89. static inline void
  90. ct_read_lock(unsigned key)
  91. {
  92.         read_lock(&__example_conntbl_lock_array[key&CT_LOCKARRAY_MASK].l);
  93. }
  94. static inline void
  95. ct_read_unlock(unsigned key)
  96. {
  97.         read_unlock(&__example_conntbl_lock_array[key&CT_LOCKARRAY_MASK].l);
  98. }
  99. static inline void
  100. ct_write_lock(unsigned key)
  101. {
  102.         write_lock(&__example_conntbl_lock_array[key&CT_LOCKARRAY_MASK].l);
  103. }
  104. static inline void
  105. ct_write_unlock(unsigned key)
  106. {
  107.         write_unlock(&__example_conntbl_lock_array[key&CT_LOCKARRAY_MASK].l);
  108. }
  109. static inline void
  110. ct_read_lock_bh(unsigned key)
  111. {
  112.         read_lock_bh(&__example_conntbl_lock_array[key&CT_LOCKARRAY_MASK].l);
  113. }
  114. static inline void
  115. ct_read_unlock_bh(unsigned key)
  116. {
  117.         read_unlock_bh(&__example_conntbl_lock_array[key&CT_LOCKARRAY_MASK].l);
  118. }
  119. static inline void
  120. ct_write_lock_bh(unsigned key)
  121. {
  122.         write_lock_bh(&__example_conntbl_lock_array[key&CT_LOCKARRAY_MASK].l);
  123. }
  124. static inline void
  125. ct_write_unlock_bh(unsigned key)
  126. {
  127.         write_unlock_bh(&__example_conntbl_lock_array[key&CT_LOCKARRAY_MASK].l);
  128. }

  129. /*    hash 函数 */
  130. static inline unsigned example_hash_key(char *p)
  131. {
  132.         int n;                /*简单示例*/
  133.         n=strlen(p);
  134.         return jhash_2words(n,p[0],example_rnd)&EXAMPLE_TAB_MASK;
  135. }

  136. inline void example_put(struct example* cp)
  137. {
  138.         atomic_dec(&cp->refcnt);
  139. }
  140. int example_hash(struct example *cp)
  141. {
  142.         unsigned hash;
  143.         int ret;
  144.         hash=example_hash_key(cp->s);
  145.         ct_write_lock(hash);
  146.         if(!(cp->flags&EXAMPLE_F_HASHED))
  147.         {
  148.                 list_add(&cp->node,&example_tab[hash]);
  149.                  cp->flags|=EXAMPLE_F_HASHED;
  150.                 atomic_add(1,&cp->refcnt);
  151.                 ret=1;
  152.         }
  153.         else
  154.         {
  155.                 ret=0;
  156.         }
  157.         ct_write_unlock(hash);
  158.         return ret;
  159. }

  160. int example_unhash(struct example *cp)
  161. {
  162.         unsigned hash;
  163.         int ret;
  164.         hash=example_hash_key(cp->s);
  165.         ct_write_lock(hash);
  166.         if(cp->flags&EXAMPLE_F_HASHED)
  167.         {
  168.                 list_del(&cp->node);
  169.                 cp->flags &=~EXAMPLE_F_HASHED;
  170.                 atomic_sub(1,&cp->refcnt);
  171.                 ret=1;
  172.         }
  173.         else
  174.         {
  175.                ret=0;
  176.         }
  177.         ct_write_unlock(hash);
  178.       
  179.         return ret;
  180. }
  181. struct example *example_get(char *s)
  182. {
  183.         struct example *cp;
  184.         unsigned hash;
  185.         hash=example_hash_key(s);
  186.      
  187.         ct_read_lock(hash);
  188.         list_for_each_entry(cp,&example_tab[hash],node)
  189.         {
  190.               spin_lock(&cp->lock);
  191.                 if(strncmp(cp->s,s,strlen(cp->s))==0)
  192.                 {
  193.                spin_unlock(&cp->lock);
  194.                         atomic_inc(&cp->refcnt);
  195.                         ct_read_unlock(hash);
  196.                         return cp;
  197.                 }
  198.                  spin_unlock(&cp->lock);
  199.                              
  200.         }
  201.         ct_read_unlock(hash);
  202.         return NULL;
  203.       
  204. }
  205. struct example* example_new(char *p)
  206. {
  207.         struct example *cp=NULL;
  208.         int n;
  209.        cp=kmem_cache_alloc(example_cachep,GFP_ATOMIC);
  210.         if(cp==NULL)
  211.         {
  212.                 return NULL;
  213.         }
  214.         memset(cp,0,sizeof(*cp));
  215.         INIT_LIST_HEAD(&cp->node);
  216.         spin_lock_init(&cp->lock);
  217.       
  218.         n=strlen(p);
  219.         cp->s=(char*)kmalloc(((n+1)*sizeof(char)),GFP_ATOMIC);
  220.         memcpy(cp->s,p,(n+1)*sizeof(char));
  221.        (cp->s)[n]='\0';      
  222.         atomic_inc(&example_count);
  223.         atomic_set(&cp->refcnt,0);
  224.         example_hash(cp);
  225.         return cp;
  226. }
  227. int example_destroy( struct example* cp)
  228. {
  229.       
  230.         if (cp==NULL) return 0;
  231.         atomic_inc(&cp->refcnt);

  232.    
  233.         if(! example_unhash(cp))
  234.         {
  235.                 example_put(cp);
  236.                 return 0;
  237.         }
  238.       
  239.         if(likely((atomic_read(&cp->refcnt))==1))
  240.         {
  241.                  spin_lock(&cp->lock);
  242.                  if(cp->s)
  243.                         kfree(cp->s);
  244.                  spin_unlock(&cp->lock);
  245.                  atomic_dec(&example_count);
  246.                  kmem_cache_free(example_cachep,cp);
  247.                  return 1;
  248.         }
  249.         example_hash(cp);              //还有人使用此项,hash back!
  250.   
  251.         example_put(cp) ;
  252.         return 0;
  253. }

  254. int example_init(void)
  255. {
  256.         int idx;
  257.         /*    allocte hash table*/
  258.         if(!(example_tab=vmalloc(EXAMPLE_TAB_SIZE*sizeof(struct list_head))))
  259.                 return -ENOMEM;
  260.       
  261.       
  262.        example_cachep=kmem_cache_create("example",
  263.                                                                                 sizeof(struct example),0,
  264.                                                                                 SLAB_HWCACHE_ALIGN,NULL);
  265.         if(!example_cachep)
  266.         {
  267.                 vfree(example_tab);
  268.                 return -ENOMEM;
  269.         }
  270.       
  271.         /*init hash tab*/
  272.         for(idx=0;idx<EXAMPLE_TAB_SIZE;idx++)
  273.         {
  274.                 INIT_LIST_HEAD(&example_tab[idx]);
  275.             
  276.         }
  277.         /*init locking tab*/
  278.         for(idx=0;idx<CT_LOCKARRAY_SIZE;idx++)
  279.         {
  280.                 rwlock_init(&__example_conntbl_lock_array[idx].l);
  281.         }
  282.         /* generate random*/
  283.         get_random_bytes(&example_rnd,sizeof(example_rnd));
  284.         return 0;
  285.       
  286. }

  287. static void example_flush(void)
  288. {
  289.         int idx;
  290.         struct list_head *list=NULL;
  291.         struct example *cp=NULL;
  292.       
  293.         flush_again:
  294.                 for(idx=0;idx<EXAMPLE_TAB_SIZE;idx++)
  295.                 {
  296.                         ct_write_lock(idx);
  297.                         list=&example_tab[idx];
  298.                         while(!list_empty(list))
  299.                         {
  300.                                   cp=list_entry(list->next,struct example,node);
  301.                                   ct_write_unlock(idx);   //      example_destroy 有可能使用了相同的锁,所以先解锁

  302.                                   if(!example_destroy(cp))
  303.                                   {
  304.                                         ct_write_lock(idx);
  305.                                          break;
  306.                                  }
  307.                                 ct_write_lock(idx);
  308.                         }
  309.                        ct_write_unlock(idx);
  310.                 }
  311.       
  312.                 if(atomic_read(&example_count)!=0)
  313.                 {
  314.                         schedule();                 //有些example项,可能正在被使用不能一下被destroy,所以等待下次回收空间
  315.                         goto flush_again;
  316.                 }
  317.       
  318. }

  319. void example_cleanup(void)
  320. {
  321.         example_flush();
  322.         kmem_cache_destroy(example_cachep);
  323.         vfree(example_tab);
  324.       
  325. }
复制代码

论坛徽章:
36
IT运维版块每日发帖之星
日期:2016-04-10 06:20:00IT运维版块每日发帖之星
日期:2016-04-16 06:20:0015-16赛季CBA联赛之广东
日期:2016-04-16 19:59:32IT运维版块每日发帖之星
日期:2016-04-18 06:20:00IT运维版块每日发帖之星
日期:2016-04-19 06:20:00每日论坛发贴之星
日期:2016-04-19 06:20:00IT运维版块每日发帖之星
日期:2016-04-25 06:20:00IT运维版块每日发帖之星
日期:2016-05-06 06:20:00IT运维版块每日发帖之星
日期:2016-05-08 06:20:00IT运维版块每日发帖之星
日期:2016-05-13 06:20:00IT运维版块每日发帖之星
日期:2016-05-28 06:20:00每日论坛发贴之星
日期:2016-05-28 06:20:00
8 [报告]
发表于 2010-12-20 16:23 |只看该作者
抛开内核态不说,用户态你用过 hash 吧。你通常怎么操作呢?

论坛徽章:
0
9 [报告]
发表于 2010-12-20 17:03 |只看该作者
用户态的hash经常用,上面程序在hash上的原理很简单,无非是hash节点上挂list,具体点儿吧,有个疑问,它创建了8个锁的数组__example_conntbl_lock_array,要保护32个节点的hash表,在操作hash表前使用ct_write_lock(hash),来进行同步,这是每个锁来同步4个hash节点,是这意思吗?

论坛徽章:
36
IT运维版块每日发帖之星
日期:2016-04-10 06:20:00IT运维版块每日发帖之星
日期:2016-04-16 06:20:0015-16赛季CBA联赛之广东
日期:2016-04-16 19:59:32IT运维版块每日发帖之星
日期:2016-04-18 06:20:00IT运维版块每日发帖之星
日期:2016-04-19 06:20:00每日论坛发贴之星
日期:2016-04-19 06:20:00IT运维版块每日发帖之星
日期:2016-04-25 06:20:00IT运维版块每日发帖之星
日期:2016-05-06 06:20:00IT运维版块每日发帖之星
日期:2016-05-08 06:20:00IT运维版块每日发帖之星
日期:2016-05-13 06:20:00IT运维版块每日发帖之星
日期:2016-05-28 06:20:00每日论坛发贴之星
日期:2016-05-28 06:20:00
10 [报告]
发表于 2010-12-20 17:20 |只看该作者
用户态的hash经常用,上面程序在hash上的原理很简单,无非是hash节点上挂list,

内核态上也就这样用。
具体点儿吧,有个疑问,它创建了8个锁的数组 __example_conntbl_lock_array,要保护32个节点的hash表,在操作hash表前使用 ct_write_lock(hash),来进行同步,这是每个锁来同步4个hash节点,是这意思吗?

这就是内核态的一些特点。一些临界资源访问时,你采用什么样的机制,来保护这些资源的修改是同步的。最简单的方法,一个 hash 表,用一个全局锁,访问时加锁,完毕解锁。

至于你提到代码中使用了多个锁的问题,那就是和你的需求和实现相关了。

由此是不是可以认为,你的疑问其实是使用 hash 表时,怎么用锁?
您需要登录后才可以回帖 登录 | 注册

本版积分规则 发表回复

  

北京盛拓优讯信息技术有限公司. 版权所有 京ICP备16024965号-6 北京市公安局海淀分局网监中心备案编号:11010802020122 niuxiaotong@pcpop.com 17352615567
未成年举报专区
中国互联网协会会员  联系我们:huangweiwei@itpub.net
感谢所有关心和支持过ChinaUnix的朋友们 转载本站内容请注明原作者名及出处

清除 Cookies - ChinaUnix - Archiver - WAP - TOP