2168ed8e7d
PCI/PCIe have better performance and more devices support, such as NVMe, GPU, Powerful NIC (Like RDMA). PCI/PCIe can access control by IOMMU that the virtualiztion and userspace driver will more safety. PCI/PCIe device could hot plugging, no design modifications SoC required, PCI/PCIe on Embedded SoC is popular now. We make a simple framework to support them. Feature Lists: 1.PCI INTx: the INT[A-D] pin IRQ for legacy PCI, work with platform PIC. 2.MSI/MSI-X: the message write IRQ for PCIe, work with platform's PIC. 3.PME: we only support the D0, D1, D2, D3HOT, D3COLD init by framework. 4.Endpoint: a simple EP framework for PCI FPGA or NTB function. 5.OFW: we only support work on OFW SoC, ACPI support in the future maybe. Host controller: 1. Common PCI host controller on ECAM. 2. Generic PCI host controller on ECAM. Signed-off-by: GuEe-GUI <2991707448@qq.com>
160 lines
4.1 KiB
C
Executable File
160 lines
4.1 KiB
C
Executable File
/*
|
|
* Copyright (c) 2006-2022, RT-Thread Development Team
|
|
*
|
|
* SPDX-License-Identifier: Apache-2.0
|
|
*
|
|
* Change Logs:
|
|
* Date Author Notes
|
|
* 2022-10-24 GuEe-GUI first version
|
|
*/
|
|
|
|
#include <rthw.h>
|
|
#include <rtthread.h>
|
|
|
|
#include <drivers/pci.h>
|
|
|
|
struct rt_spinlock rt_pci_lock = { 0 };
|
|
|
|
#ifdef RT_PCI_LOCKLESS
|
|
#define pci_lock_config(l) do { (void)(l); } while (0)
|
|
#define pci_unlock_config(l) do { (void)(l); } while (0)
|
|
#else
|
|
#define pci_lock_config(l) l = rt_spin_lock_irqsave(&rt_pci_lock)
|
|
#define pci_unlock_config(l) rt_spin_unlock_irqrestore(&rt_pci_lock, l)
|
|
#endif
|
|
|
|
#define PCI_OPS_READ(name, type) \
|
|
rt_err_t rt_pci_bus_read_config_##name(struct rt_pci_bus *bus, rt_uint32_t devfn, int reg, type *value) \
|
|
{ \
|
|
rt_err_t err; \
|
|
rt_ubase_t level; \
|
|
rt_uint32_t data = 0; \
|
|
pci_lock_config(level); \
|
|
err = bus->ops->read(bus, devfn, reg, sizeof(type), &data); \
|
|
*value = err ? (type)(~0) : (type)data; \
|
|
pci_unlock_config(level); \
|
|
return err; \
|
|
}
|
|
|
|
#define PCI_OPS_WRITE(name, type) \
|
|
rt_err_t rt_pci_bus_write_config_##name(struct rt_pci_bus *bus, rt_uint32_t devfn, int reg, type value) \
|
|
{ \
|
|
rt_err_t err; \
|
|
rt_ubase_t level; \
|
|
pci_lock_config(level); \
|
|
err = bus->ops->write(bus, devfn, reg, sizeof(type), value); \
|
|
pci_unlock_config(level); \
|
|
return err; \
|
|
}
|
|
|
|
#define PCI_OPS(name, type) \
|
|
PCI_OPS_READ(name, type) \
|
|
PCI_OPS_WRITE(name, type)
|
|
|
|
PCI_OPS(u8, rt_uint8_t)
|
|
PCI_OPS(u16, rt_uint16_t)
|
|
PCI_OPS(u32, rt_uint32_t)
|
|
|
|
#undef PCI_OP_WRITE
|
|
#undef PCI_OP_READ
|
|
#undef PCI_OPS
|
|
|
|
rt_err_t rt_pci_bus_read_config_uxx(struct rt_pci_bus *bus,
|
|
rt_uint32_t devfn, int reg, int width, rt_uint32_t *value)
|
|
{
|
|
void *base;
|
|
|
|
if ((base = bus->ops->map(bus, devfn, reg)))
|
|
{
|
|
if (width == 1)
|
|
{
|
|
*value = HWREG8(base);
|
|
}
|
|
else if (width == 2)
|
|
{
|
|
*value = HWREG16(base);
|
|
}
|
|
else
|
|
{
|
|
*value = HWREG32(base);
|
|
}
|
|
|
|
return RT_EOK;
|
|
}
|
|
|
|
return -RT_ERROR;
|
|
}
|
|
|
|
rt_err_t rt_pci_bus_write_config_uxx(struct rt_pci_bus *bus,
|
|
rt_uint32_t devfn, int reg, int width, rt_uint32_t value)
|
|
{
|
|
void *base;
|
|
|
|
if ((base = bus->ops->map(bus, devfn, reg)))
|
|
{
|
|
if (width == 1)
|
|
{
|
|
HWREG8(base) = value;
|
|
}
|
|
else if (width == 2)
|
|
{
|
|
HWREG16(base) = value;
|
|
}
|
|
else
|
|
{
|
|
HWREG32(base) = value;
|
|
}
|
|
|
|
return RT_EOK;
|
|
}
|
|
|
|
return -RT_ERROR;
|
|
}
|
|
|
|
rt_err_t rt_pci_bus_read_config_generic_u32(struct rt_pci_bus *bus,
|
|
rt_uint32_t devfn, int reg, int width, rt_uint32_t *value)
|
|
{
|
|
void *base;
|
|
|
|
if ((base = bus->ops->map(bus, devfn, reg)))
|
|
{
|
|
*value = HWREG32(base);
|
|
|
|
if (width <= 2)
|
|
{
|
|
*value = (*value >> (8 * (reg & 3))) & ((1 << (width * 8)) - 1);
|
|
}
|
|
|
|
return RT_EOK;
|
|
}
|
|
|
|
return -RT_ERROR;
|
|
}
|
|
|
|
rt_err_t rt_pci_bus_write_config_generic_u32(struct rt_pci_bus *bus,
|
|
rt_uint32_t devfn, int reg, int width, rt_uint32_t value)
|
|
{
|
|
void *base;
|
|
|
|
if ((base = bus->ops->map(bus, devfn, reg & ~0x3)))
|
|
{
|
|
if (width == 4)
|
|
{
|
|
HWREG32(base) = value;
|
|
}
|
|
else
|
|
{
|
|
rt_uint32_t mask, tmp;
|
|
|
|
mask = ~(((1 << (width * 8)) - 1) << ((reg & 0x3) * 8));
|
|
tmp = HWREG32(base) & mask;
|
|
tmp |= value << ((reg & 0x3) * 8);
|
|
HWREG32(base) = tmp;
|
|
}
|
|
|
|
return RT_EOK;
|
|
}
|
|
|
|
return -RT_ERROR;
|
|
}
|