forked from Minki/linux
d82a7aed83
Add a Device Feature List (DFL) feature id [1] for the configurable IOPLL user clock source, which can be used to configure the clock speeds that are used for RTL logic that is programmed into the Partial Reconfiguration (PR) region of an FPGA. The IOPLL user-space driver [2] contains frequency tables [3] with the specific user clock frequencies for an implementation. For each desired frequency, the table values are produced by calling the quartus tool, the same tool that generates the IOPLL RTL logic. The quartus tool allows the RTL designer to select different options which can affect the table values. The table-driven, user-space driver allows for supporting future, modified implementations and provides users the ability to modify the IOPLL implementation. [1] https://github.com/OPAE/dfl-feature-id [2]a494f54a9f/libraries/plugins/xfpga/usrclk/fpga_user_clk.c
[3]a494f54a9f/libraries/plugins/xfpga/usrclk/fpga_user_clk_freq.h
Acked-by: Xu Yilun <yilun.xu@intel.com> Signed-off-by: Russ Weight <russell.h.weight@intel.com> Signed-off-by: Peter Colberg <peter.colberg@intel.com> Link: https://lore.kernel.org/r/20220831204851.4683-1-peter.colberg@intel.com Signed-off-by: Greg Kroah-Hartman <gregkh@linuxfoundation.org>
71 lines
1.7 KiB
C
71 lines
1.7 KiB
C
// SPDX-License-Identifier: GPL-2.0
|
|
/*
|
|
* Generic DFL driver for Userspace I/O devicess
|
|
*
|
|
* Copyright (C) 2021 Intel Corporation, Inc.
|
|
*/
|
|
#include <linux/dfl.h>
|
|
#include <linux/errno.h>
|
|
#include <linux/module.h>
|
|
#include <linux/uio_driver.h>
|
|
|
|
#define DRIVER_NAME "uio_dfl"
|
|
|
|
static int uio_dfl_probe(struct dfl_device *ddev)
|
|
{
|
|
struct resource *r = &ddev->mmio_res;
|
|
struct device *dev = &ddev->dev;
|
|
struct uio_info *uioinfo;
|
|
struct uio_mem *uiomem;
|
|
int ret;
|
|
|
|
uioinfo = devm_kzalloc(dev, sizeof(struct uio_info), GFP_KERNEL);
|
|
if (!uioinfo)
|
|
return -ENOMEM;
|
|
|
|
uioinfo->name = DRIVER_NAME;
|
|
uioinfo->version = "0";
|
|
|
|
uiomem = &uioinfo->mem[0];
|
|
uiomem->memtype = UIO_MEM_PHYS;
|
|
uiomem->addr = r->start & PAGE_MASK;
|
|
uiomem->offs = r->start & ~PAGE_MASK;
|
|
uiomem->size = (uiomem->offs + resource_size(r)
|
|
+ PAGE_SIZE - 1) & PAGE_MASK;
|
|
uiomem->name = r->name;
|
|
|
|
/* Irq is yet to be supported */
|
|
uioinfo->irq = UIO_IRQ_NONE;
|
|
|
|
ret = devm_uio_register_device(dev, uioinfo);
|
|
if (ret)
|
|
dev_err(dev, "unable to register uio device\n");
|
|
|
|
return ret;
|
|
}
|
|
|
|
#define FME_FEATURE_ID_ETH_GROUP 0x10
|
|
#define FME_FEATURE_ID_HSSI_SUBSYS 0x15
|
|
#define PORT_FEATURE_ID_IOPLL_USRCLK 0x14
|
|
|
|
static const struct dfl_device_id uio_dfl_ids[] = {
|
|
{ FME_ID, FME_FEATURE_ID_ETH_GROUP },
|
|
{ FME_ID, FME_FEATURE_ID_HSSI_SUBSYS },
|
|
{ PORT_ID, PORT_FEATURE_ID_IOPLL_USRCLK },
|
|
{ }
|
|
};
|
|
MODULE_DEVICE_TABLE(dfl, uio_dfl_ids);
|
|
|
|
static struct dfl_driver uio_dfl_driver = {
|
|
.drv = {
|
|
.name = DRIVER_NAME,
|
|
},
|
|
.id_table = uio_dfl_ids,
|
|
.probe = uio_dfl_probe,
|
|
};
|
|
module_dfl_driver(uio_dfl_driver);
|
|
|
|
MODULE_DESCRIPTION("Generic DFL driver for Userspace I/O devices");
|
|
MODULE_AUTHOR("Intel Corporation");
|
|
MODULE_LICENSE("GPL v2");
|