0

PCIe デバイスの MSI 割り込みを処理するカーネル モジュールを作成しようとしています。現在、ドライバーの簡単なスケルトン アウトラインを作成しましたが、'pci_disable_msi(dev)' を呼び出そうとすると、カーネル NULL ポインター逆参照エラーを処理できません。/Documentation/PCI/MSI-HOWTO.txt に記載されているとおりに正確に従っていますが、このエラーが発生することはないように思われます。これはバグですか、それとも私の設定が間違っていますか? 発生した最後の印刷から判断すると、pci_disable_msi() を呼び出したときに fpga_remove() で発生していると確信しています。(明らかに、これはモジュールを削除しているときに発生します)

static struct pci_driver fpga_driver = {
        .name       = "PCIe_test",
        .id_table   = fpga_dev_table,
        .probe      = fpga_probe,
        .remove     = fpga_remove,
        .suspend    = fpga_suspend,
        .resume     = fpga_resume,
};

static irqreturn_t fpga_isr(int irq, struct pci_dev *dev)
{

    printk(KERN_NOTICE "THIS is the ISR\n");

    return IRQ_HANDLED;
}


static int setup_MSI_interrupt(struct pci_dev *dev, int num_msi)
{

    int result;
    result = pci_enable_msi(dev);
    if(result)
    {
        printk(KERN_WARNING "Could not enable MSI\n");
        return result;

    }
    printk(KERN_NOTICE "MSI has been enabled\n");
    printk(KERN_NOTICE "dev->irq line is %d", dev->irq);

    result = request_irq(dev->irq, fpga_isr, IRQF_SHARED, fpga_driver.name, dev);
    printk(KERN_NOTICE "Using IRQ num %d\n", dev->irq);

    if (result) {
        dev_err(&dev->dev, "Failed to allocate irq %d: %d\n", dev->irq, result);
        goto exit1;
    }
        dev_info(&dev->dev, "FPGA using PCIe Interrupt\n");

    return 0;
exit1:
    return -1;
}

static int fpga_probe(struct pci_dev *dev, const struct pci_device_id *id)
{
    printk(KERN_NOTICE "Probing driver\n");
    switch(dev->vendor) {
    case VENDOR://0x1708:
        printk(KERN_NOTICE "Xilinx device found\n");
        break;
    default:
        printk(KERN_NOTICE "Device found that does not match id: id = 0x%04X\n", dev->device);
    };

    int err = pci_enable_device(dev);
    if (err) {
        dev_err(&dev->dev, "Failed to enable FPGA PCI device (%d)\n", err);
        goto exit;
    }

    err = setup_MSI_interrupt(dev, NUM_MSI);

    if(err)
        goto exit;

    return 0;

exit:
    return -1;
}

static void fpga_remove(struct pci_dev *dev)
{
    printk(KERN_NOTICE "REMOVING IRQ # %d\n", dev->irq);

    free_irq(dev->irq, dev);
    printk(KERN_NOTICE "IRQ has been freed\n");
    pci_disable_msi(dev);                                               // This causes a NUll Pointer to be dereferenced but needs to be added
    printk(KERN_NOTICE "MSI has been disabled\n");
}

static int __init fpga_init(void)
{
    printk(KERN_NOTICE "Registering Driver\n");
    return pci_register_driver(&fpga_driver);
    return 0;
}
4

1 に答える 1

1

および関数でstruct pci_devasvoid *dev_idパラメータを使用しないでください。割り込みに適切で同じ一意のパラメーターを使用してそれらを呼び出すことが重要です。これを行わないと、モジュールの削除時にカーネル パニックが発生する可能性があります。request_irq(...)free_irq(...)dev_idpci_disable_msi()

詳細はこちら: request_irq の dev_id パラメータとは?

于 2015-04-14T12:02:34.313 に答える