module_param_cb(write_queues, &io_queue_count_ops, &write_queues, 0644);
module_param_cb
是 Linux 内核模块中用于定义可通过命令行参数传递给模块的参数的宏。在你提供的代码中,module_param_cb
用于定义一个名为 write_queues
的参数,并使用名为 io_queue_count_ops
的回调函数来管理该参数。
具体来说,module_param_cb
的含义如下:
write_queues
: 参数的名称。&io_queue_count_ops
: 一个回调函数,用于设置和获取参数值的操作。io_queue_count_ops
函数将被调用来设置和获取参数值。&write_queues
: 参数的实际值。参数的实际值将存储在write_queues
变量中。0644
: 参数的访问权限,用于确定参数是否可以读写。
至于 write_queues
的默认值,由于你没有提供参数的默认值,所以默认值将取决于该参数的类型。如果参数是一个整数类型,通常默认值为0。如果需要为参数设置默认值,你可以在 module_param_cb
宏中传递第四个参数,即默认值。例如
module_param_cb(write_queues, &io_queue_count_ops, &write_queues, 0644, int, 8);
上述代码中,参数 write_queues
的默认值被设置为8。请注意,实际默认值的设置可能因特定模块的实现而有所不同
在NVMe pci.c 的驱动中我们可以看到相关的设置和定义
static int io_queue_count_set(const char *val, const struct kernel_param *kp)
{
unsigned int n;
int ret;
ret = kstrtouint(val, 10, &n);
if (ret != 0 || n > num_possible_cpus())
return -EINVAL;
return param_set_uint(val, kp);
}
static const struct kernel_param_ops io_queue_count_ops = {
.set = io_queue_count_set,
.get = param_get_uint,
};
static unsigned int write_queues;
module_param_cb(write_queues, &io_queue_count_ops, &write_queues, 0644);
MODULE_PARM_DESC(write_queues,
"Number of queues to use for writes. If not set, reads and writes "
"will share a queue set.");
static unsigned int poll_queues;
module_param_cb(poll_queues, &io_queue_count_ops, &poll_queues, 0644);
MODULE_PARM_DESC(poll_queues, "Number of queues to use for polled IO.");
-
io_queue_count_set
函数:这是一个回调函数,用于处理字符串类型的参数值,并将其转换为无符号整数。它首先使用kstrtouint
函数将字符串转换为无符号整数。如果转换失败或者转换后的值超过了系统中可能的CPU核心数量,就会返回-EINVAL
表示参数无效。否则,它将使用param_set_uint
函数设置参数值。 -
io_queue_count_ops
结构体:这个结构体包含了操作函数指针,指向参数的设置函数和获取函数。在这里,set
成员指向了io_queue_count_set
函数,用于设置参数值,而get
成员指向标准的param_get_uint
函数,用于获取参数值。 -
module_param_cb
宏:这个宏用于定义内核模块参数。它接受四个参数:参数名、参数操作结构体、参数值的指针以及权限模式。在这个示例中,它定义了两个参数:write_queues
和poll_queues
,将它们的设置和获取操作绑定到了io_queue_count_ops
结构体中定义的函数。 -
MODULE_PARM_DESC
宏:这个宏用于为参数提供描述信息,帮助用户了解参数的作用。它接受两个参数:参数名和描述字符串。在这里,它为两个参数提供了描述信息,说明了它们的作用和含义。从该描述我们知道 write_queue 在驱动加载过程中如果没有设置它将和read_queue 共用一个queue set 这就是read 有多少 write 就有多少