如何在内核中断上下文中分配内存

判断一个内存分配函数能否用在中断上下文中,主要看这个内存函数最终调用的伙伴系统算法的函数接口page_alloc(gfp_mask,order)的gfp_mask的内容,用在中断上下文中一般是GFP_ATOMIC,否则通常GFP_KERNEL作为内存分配标志。
get_free_pages()也是根据传递的参数gfp_mask,决定这个分配函数是否能睡眠,所以传递的gfp_mask参数决定了是否能用在中断上下文中。

内核中函数分配根据分类方式的不同:
1 根据分配时是否睡眠分为两类:
使用GFP_ATOMIC标志分配内存,告诉分配函数不能睡眠,一般仅仅在中断上下文,或者持有锁的代码中使用。
使用GFP_KERNEL标志分配内存,在分配过程中,可能会阻塞,一般用在进程上下文。
对于常规内存的分配,是用get_free_pages(),是用这个标志作为分配的参数时,可以使用在中断上下文,或者持有锁的代码段中。
对于高端物理内存使用alloc_pages()(用在中断上下文中应该传递GFP_ATOMIC标志)配合持久映射函数kmap()或者临时映射函数kmap_atomic(),其中kmap()用在进程上下文中,而kmap_atomic()用在中断上下文中。

2 根据分配的物理内存的位置可以分为高端物理内存(high zone)和非高端物理内存的分配(dma zone or normal zone)。都有可以睡眠的分配方式和不可睡眠的内存分配方式
上面两种非配方式是交叉,睡眠(或者不可睡眠)的内存分配方式都可以在高端和低端分配

GFP_KERNEL标志的作用:
不管是连续的还是非连续的,可睡眠的还是不可睡眠的分配方式,最终都要使用伙伴系统算法的接口函数,核心就是alloc_pages(),前面提到的GFP_KERNEL以及其他的内存标志最终都会传递给这个函数。所以看一个内存分配函数是否可以用在中断上下文中,只要看这个函数最终使用的伙伴系统的接口函数的flag参数。
看vmalloc()代码就知道,这个函数分配物理内存分配函数page_alloc()传递的参数是GFP_KERNEL|GFP_HIGHMEM,所以自然不能用在中断上下文中。


你可能感兴趣的:(编程,linux,unix,内核)