devm_free_irq 与 free_irq 差异
时间: 2023-09-18 22:13:44 浏览: 345
`devm_free_irq`和`free_irq`都是用于释放中断处理程序的函数,但它们有一些差异。
`free_irq`是一个非常常见的函数,它用于释放通过`request_irq`申请的中断。它需要提供中断号和中断处理程序的指针。使用`free_irq`释放中断是一种常规的做法,但是它需要手动调用,在某些情况下可能会出现遗漏或者忘记调用的情况。
相比之下,`devm_free_irq`是一个用于设备驱动程序的函数,它可以自动释放中断,无需手动调用。该函数会在设备被卸载时自动调用,因此更加方便和安全。使用`devm_free_irq`可以避免因为忘记手动调用`free_irq`而导致的中断资源泄漏等问题。
总的来说,如果你的设备驱动程序采用了设备模型(即使用了`devm_`开头的函数),那么建议使用`devm_free_irq`来释放中断。如果没有使用设备模型,则可以使用`free_irq`手动释放中断。
相关问题
devm_request_irq 返回值
devm_request_irq 是 Linux 内核中的一个函数,用于请求分配一个中断,并将中断处理函数与之关联。它的返回值表示请求中断是否成功。
devm_request_irq 的返回值有以下几种可能:
1. 成功:返回值为 0,表示成功地请求分配了中断,并成功地将中断处理函数与之关联。
2. 失败:返回一个负数,表示请求分配中断失败。常见的失败原因包括中断号无效、中断处理函数无效、中断请求失败等。
3. EPROBE_DEFER:返回值为 -EPROBE_DEFER,表示请求被推迟。这通常发生在设备驱动程序初始化过程中,当设备的资源还没有完全准备好时。
需要注意的是,devm_request_irq 是 devm 版本的函数,它会在设备被卸载时自动释放所分配的中断资源,因此不需要手动调用 irq_free 函数来释放中断。
static int phytium_uart_probe(struct amba_device *dev, const struct amba_id *id) { struct phytium_uart_port *pup; struct vendor_data *vendor = id->data; int portnr, ret; portnr = phytium_find_free_port(); if (portnr < 0) return portnr; pup = devm_kzalloc(&dev->dev, sizeof(struct phytium_uart_port), GFP_KERNEL); if(!pup) return -ENOMEM; pup->clk = devm_clk_get(&dev->dev, NULL); if(IS_ERR(pup->clk)) return PTR_ERR(pup->clk); pup->port.irq = dev->irq[0]; pup->port.line = portnr; pup->vendor = vendor; pup->fifosize = 32; pup->port.iotype = pup->vendor->access_32b ? UPIO_MEM32 : UPIO_MEM; pup->port.ops = &phytium_uart_ops; snprintf(pup->type, sizeof(pup->type), "PL011 rev%u", amba_rev(dev)); ret = phytium_setup_port(&dev->dev, pup, &dev->res, portnr); if (ret) return ret; amba_set_drvdata(dev, pup); return phytium_register_port(pup); }在这段linux内核驱动中加入读取acpi表中描述的固定波特率,并设置波特率的操作,给出详细代码
在这段驱动中,由于使用的是AMBA总线,因此无法直接使用ACPI匹配设备。不过,我们可以在驱动中添加代码来读取设备树中的固定波特率信息,从而设置波特率。
下面是一个简单的示例代码,用于在驱动中读取设备树中的固定波特率信息:
```
static int phytium_uart_probe(struct amba_device *dev, const struct amba_id *id)
{
struct phytium_uart_port *pup;
struct vendor_data *vendor = id->data;
int portnr, ret;
u32 baudrate;
// 获取波特率信息
if (of_property_read_u32(dev->dev.of_node, "baudrate", &baudrate)) {
dev_info(&dev->dev, "no baudrate specified, using default\n");
baudrate = 115200;
} else {
dev_info(&dev->dev, "using baudrate %u\n", baudrate);
}
portnr = phytium_find_free_port();
if (portnr < 0)
return portnr;
pup = devm_kzalloc(&dev->dev, sizeof(struct phytium_uart_port),
GFP_KERNEL);
if (!pup)
return -ENOMEM;
pup->clk = devm_clk_get(&dev->dev, NULL);
if (IS_ERR(pup->clk))
return PTR_ERR(pup->clk);
pup->port.irq = dev->irq[0];
pup->port.line = portnr;
pup->vendor = vendor;
pup->fifosize = 32;
pup->port.iotype = pup->vendor->access_32b ? UPIO_MEM32 : UPIO_MEM;
pup->port.ops = &phytium_uart_ops;
snprintf(pup->type, sizeof(pup->type), "PL011 rev%u", amba_rev(dev));
// 设置波特率
pup->port.uartclk = clk_get_rate(pup->clk);
baudrate = uart_get_baud_rate(&pup->port, baudrate, baudrate);
ret = phytium_setup_port(&dev->dev, pup, &dev->res, portnr);
if (ret)
return ret;
amba_set_drvdata(dev, pup);
return phytium_register_port(pup);
}
```
在这个示例代码中,我们使用了`of_property_read_u32()`函数来读取设备树中的`baudrate`属性。如果设备树中没有指定该属性,则默认使用115200作为波特率。接下来,我们使用`uart_get_baud_rate()`函数来获取实际使用的波特率,并将其传递给`phytium_setup_port()`函数来进行端口设置。
阅读全文