• linux驱动39:内存池


    内核中有些地方的内存分配是不允许失败的。为了确保成功分配内存,建立了内存池。内存池是某种形式的高速后备缓存。试图始终保存空闲的内存,以便在紧急状态下使用。

    类型:mempool_t,头文件:

    接口:

    创建内存池对象:

    mempool_t *mempool_create(int min_nr, mempool_alloc_t *alloc_fn, mempol_free_t *free_fn, void *pool_data);

    min_nr:内存池始终保持的已分配对象的最少数目

    alloc_fn:原型 typedef void *(mempool_alloc_t )(gfp_t gfp_mask, void *pool_data),通常值为mempool_alloc_slab(内核定义的函数)
    free_fn:原型 typedef  void (mempol_free_t )(void *element, void *pool_data),通常值为mempool_free_slab(内核定义的函数)

    pool_data:被传入alloc_fn和free_fn

    分配内存对象:

    void *mempool_alloc(mempool_t *pool, int gfp_mask);

    首先通过分配函数获得内存对象,如果失败,就会返回预先分配的内存对象

    释放内存对象:

    void mempool_free(void *element, mempool_t *pool);

    如果预先分配的对象数目小于要求的最低数目,就会将该内存对象保留在内存池中,否则返回给系统

    释放内存池:

    void mempool_destroy(mempool_t *pool);

    在释放内存池时,必须将所有分配的内存对象返回到内存池中

    1. #include
    2. #include
    3. #include
    4. #include
    5. #include
    6. #define MEM_SIZE 8
    7. #define MIN_NR 8
    8. static mempool_t *pstPool = NULL;
    9. static struct kmem_cache *pKmemCache = NULL;
    10. void constructor(void *args)
    11. {
    12. //printk("constructor. \n");
    13. memset(args, 0x0, MEM_SIZE);
    14. strcpy(args, "hello");
    15. }
    16. static int __init hello_init(void)
    17. {
    18. printk("hello_init. \n");
    19. pKmemCache = kmem_cache_create("test", MEM_SIZE, 0, SLAB_HWCACHE_ALIGN, constructor);
    20. if (!pKmemCache)
    21. {
    22. printk("kmem_cache_create failed. \n");
    23. return -ENOMEM;
    24. }
    25. pstPool = mempool_create(MIN_NR, mempool_alloc_slab, mempool_free_slab, pKmemCache);
    26. if (!pstPool)
    27. {
    28. printk("mempool_create failed. \n");
    29. return -ENOMEM;
    30. }
    31. void *p = mempool_alloc(pstPool, GFP_KERNEL);
    32. if (!p)
    33. {
    34. printk("mempool_alloc failed. \n");
    35. return -ENOMEM;
    36. }
    37. printk("str:%s\n", p);
    38. mempool_free(p, pstPool);
    39. return 0;
    40. }
    41. static void __exit hello_exit(void)
    42. {
    43. printk("hello_exit. \n");
    44. if (NULL != pstPool)
    45. {
    46. mempool_destroy(pstPool);
    47. pstPool = NULL;
    48. }
    49. }
    50. MODULE_LICENSE("GPL");
    51. module_init(hello_init);
    52. module_exit(hello_exit);

    总结:

    内存池会分配一些内存块,空闲且不会得到真正的使用,容易浪费大量内存。尽量少用内存池。

  • 相关阅读:
    墨者-网络安全
    【树莓派不吃灰】搭建Node-Red可拖拽图形化物联网
    设计模式-day02
    Unity嵌入Android项目开发
    云原生边缘设备解决方案Akri on k3s初体验
    添加一个仅管理员可见的页面
    【移远QuecPython】EC800M物联网开发板的内置GNSS定位的恶性BUG(目前没有完全的解决方案)
    持续集成/持续部署:Git
    【Linux】:Linux环境与版本
    20231012_python练习_服务端与客户端数据交互v2_增加xlsx表格数据批量导入数据库
  • 原文地址:https://blog.csdn.net/dongyoubin/article/details/127816671