[BFN] Added the latest version of FPGA driver and modules (#10458)
Signed-off-by: Taras Keryk <tarasx.keryk@intel.com>
This commit is contained in:
parent
c1f2221ae3
commit
593ab45bde
@ -1,7 +1,7 @@
|
||||
/*******************************************************************************
|
||||
Barefoot Networks Switch ASIC Linux driver
|
||||
Copyright(c) 2015 - 2019 Barefoot Networks, Inc.
|
||||
|
||||
|
||||
This program is free software; you can redistribute it and/or modify it
|
||||
under the terms and conditions of the GNU General Public License,
|
||||
version 2, as published by the Free Software Foundation.
|
||||
@ -39,6 +39,7 @@ typedef uint64_t phys_addr_t;
|
||||
|
||||
#define BF_IOC_MAGIC 'b'
|
||||
#define BF_TBUS_MSIX_INDICES_MAX 3
|
||||
#define BF_TBUS_MSIX_INDICES_MIN 1
|
||||
|
||||
typedef struct bf_dma_bus_map_s
|
||||
{
|
||||
|
@ -37,6 +37,7 @@
|
||||
#include <linux/io.h>
|
||||
#include <linux/wait.h>
|
||||
#include <linux/poll.h>
|
||||
#include <linux/version.h>
|
||||
#include <linux/dma-mapping.h>
|
||||
#include "bf_ioctl.h"
|
||||
#include "bf_kdrv.h"
|
||||
@ -60,6 +61,7 @@ extern int bf_kpkt_init(struct pci_dev *pdev,
|
||||
u8 *bar0_vaddr,
|
||||
void **adapter_ptr,
|
||||
int dev_id,
|
||||
int subdev_id,
|
||||
int pci_use_highmem,
|
||||
unsigned long head_room,
|
||||
int kpkt_dr_int_en,
|
||||
@ -72,14 +74,15 @@ extern void bf_kpkt_set_pci_error(void *adapter_ptr, u8 pci_error);
|
||||
/* Keep any global information here that must survive even after the
|
||||
* bf_pci_dev is free-ed up.
|
||||
*/
|
||||
struct bf_global {
|
||||
struct bf_global_s {
|
||||
struct bf_pci_dev *bfdev;
|
||||
struct cdev *bf_cdev;
|
||||
struct fasync_struct *async_queue;
|
||||
/* Pending user space error signal. */
|
||||
bool pending_signal;
|
||||
};
|
||||
|
||||
static int bf_major;
|
||||
static int bf_minor[BF_MAX_DEVICE_CNT] = {0};
|
||||
static int bf_minor[BF_MAX_DEV_SUBDEV_CNT] = {0};
|
||||
static struct class *bf_class = NULL;
|
||||
static char *intr_mode = NULL;
|
||||
static int kpkt_mode = 0;
|
||||
@ -91,7 +94,10 @@ static enum bf_intr_mode bf_intr_mode_default = BF_INTR_MODE_MSI;
|
||||
static spinlock_t bf_nonisr_lock;
|
||||
|
||||
/* dev->minor should index into this array */
|
||||
static struct bf_global bf_global[BF_MAX_DEVICE_CNT];
|
||||
static struct bf_global_s bf_global[BF_MAX_DEV_SUBDEV_CNT];
|
||||
|
||||
/* global tofino3 info to group subdevices to a parent device */
|
||||
static struct bf_tof3_info_s bf_tof3_info[BF_MAX_DEVICE_CNT];
|
||||
|
||||
static void bf_add_listener(struct bf_pci_dev *bfdev,
|
||||
struct bf_listener *listener) {
|
||||
@ -145,7 +151,7 @@ static int bf_get_next_minor_no(int *minor) {
|
||||
int i;
|
||||
|
||||
spin_lock(&bf_nonisr_lock);
|
||||
for (i = 0; i < BF_MAX_DEVICE_CNT; i++) {
|
||||
for (i = 0; i < BF_MAX_DEV_SUBDEV_CNT; i++) {
|
||||
if (bf_minor[i] == 0) {
|
||||
*minor = i;
|
||||
bf_minor[i] = 1; /* mark it as taken */
|
||||
@ -276,7 +282,7 @@ static irqreturn_t bf_pci_irqhandler(int irq, struct bf_pci_dev *bfdev) {
|
||||
bf_kpkt_irqhandler(irq, bfdev->adapter_ptr);
|
||||
}
|
||||
} else if (bfdev->mode == BF_INTR_MODE_MSIX) {
|
||||
if (bfdev->info.tof_type == BF_TOFINO_2 && bf_irq_is_tbus_msix(bfdev,irq)) {
|
||||
if ((bfdev->info.tof_type == BF_TOFINO_2 || bfdev->info.tof_type == BF_TOFINO_3) && bf_irq_is_tbus_msix(bfdev, irq)) {
|
||||
bf_kpkt_irqhandler(irq, bfdev->adapter_ptr);
|
||||
}
|
||||
}
|
||||
@ -470,6 +476,7 @@ static int bf_mmap(struct file *filep, struct vm_area_struct *vma) {
|
||||
|
||||
static int bf_fasync(int fd, struct file *filep, int mode) {
|
||||
int minor;
|
||||
int res;
|
||||
|
||||
if (!filep->private_data) {
|
||||
return (-EINVAL);
|
||||
@ -479,17 +486,32 @@ static int bf_fasync(int fd, struct file *filep, int mode) {
|
||||
return (-EINVAL);
|
||||
}
|
||||
if (mode == 0 && &bf_global[minor].async_queue == NULL) {
|
||||
bf_global[minor].pending_signal = false;
|
||||
return 0; /* nothing to do */
|
||||
}
|
||||
return (fasync_helper(fd, filep, mode, &bf_global[minor].async_queue));
|
||||
res = fasync_helper(fd, filep, mode, &bf_global[minor].async_queue);
|
||||
if (bf_global[minor].pending_signal && bf_global[minor].async_queue) {
|
||||
kill_fasync(&bf_global[minor].async_queue, SIGIO, POLL_ERR);
|
||||
}
|
||||
bf_global[minor].pending_signal = false;
|
||||
return res;
|
||||
}
|
||||
|
||||
static int bf_open(struct inode *inode, struct file *filep) {
|
||||
unsigned id;
|
||||
struct bf_pci_dev *bfdev;
|
||||
struct bf_listener *listener;
|
||||
int i;
|
||||
|
||||
bfdev = bf_global[iminor(inode)].bfdev;
|
||||
id = iminor(inode);
|
||||
if (id >= BF_MAX_DEVICE_CNT) {
|
||||
return (-EINVAL);
|
||||
}
|
||||
|
||||
bfdev = bf_global[id].bfdev;
|
||||
if (bfdev->in_use) {
|
||||
return (-EBUSY);
|
||||
}
|
||||
listener = kmalloc(sizeof(*listener), GFP_KERNEL);
|
||||
if (listener) {
|
||||
listener->bfdev = bfdev;
|
||||
@ -500,6 +522,7 @@ static int bf_open(struct inode *inode, struct file *filep) {
|
||||
listener->event_count[i] = atomic_read(&bfdev->info.event[i]);
|
||||
}
|
||||
filep->private_data = listener;
|
||||
bfdev->in_use = 1;
|
||||
return 0;
|
||||
} else {
|
||||
return (-ENOMEM);
|
||||
@ -512,6 +535,7 @@ static int bf_release(struct inode *inode, struct file *filep) {
|
||||
bf_fasync(-1, filep, 0); /* empty any process id in the notification list */
|
||||
if (listener->bfdev) {
|
||||
bf_remove_listener(listener->bfdev, listener);
|
||||
listener->bfdev->in_use = 0;
|
||||
}
|
||||
kfree(listener);
|
||||
return 0;
|
||||
@ -645,7 +669,15 @@ static long bf_ioctl(struct file *filep, unsigned int cmd, unsigned long arg)
|
||||
}
|
||||
switch(cmd) {
|
||||
case BF_IOCMAPDMAADDR:
|
||||
#if LINUX_VERSION_CODE >= KERNEL_VERSION(5, 0, 0)
|
||||
#if defined(RHEL_RELEASE_CODE)
|
||||
#if defined(RHEL_RELEASE_VERSION)
|
||||
#if RHEL_RELEASE_CODE > RHEL_RELEASE_VERSION(7, 9)
|
||||
if (access_ok(addr, sizeof(bf_dma_bus_map_t))) {
|
||||
#else
|
||||
if (access_ok(VERIFY_WRITE, addr, sizeof(bf_dma_bus_map_t))) {
|
||||
#endif
|
||||
#endif /* RHEL_RELEASE_CODE && RHEL_RELEASE_VERSION */
|
||||
#elif LINUX_VERSION_CODE >= KERNEL_VERSION(5, 0, 0)
|
||||
if (access_ok(addr, sizeof(bf_dma_bus_map_t))) {
|
||||
#else
|
||||
if (access_ok(VERIFY_WRITE, addr, sizeof(bf_dma_bus_map_t))) {
|
||||
@ -669,7 +701,15 @@ static long bf_ioctl(struct file *filep, unsigned int cmd, unsigned long arg)
|
||||
}
|
||||
break;
|
||||
case BF_IOCUNMAPDMAADDR:
|
||||
#if LINUX_VERSION_CODE >= KERNEL_VERSION(5, 0, 0)
|
||||
#if defined(RHEL_RELEASE_CODE)
|
||||
#if defined(RHEL_RELEASE_VERSION)
|
||||
#if RHEL_RELEASE_CODE > RHEL_RELEASE_VERSION(7, 9)
|
||||
if (access_ok(addr, sizeof(bf_dma_bus_map_t))) {
|
||||
#else
|
||||
if (access_ok(VERIFY_WRITE, addr, sizeof(bf_dma_bus_map_t))) {
|
||||
#endif
|
||||
#endif /* RHEL_RELEASE_CODE && RHEL_RELEASE_VERSION */
|
||||
#elif LINUX_VERSION_CODE >= KERNEL_VERSION(5, 0, 0)
|
||||
if (access_ok(addr, sizeof(bf_dma_bus_map_t))) {
|
||||
#else
|
||||
if (access_ok(VERIFY_READ, addr, sizeof(bf_dma_bus_map_t))) {
|
||||
@ -692,21 +732,39 @@ static long bf_ioctl(struct file *filep, unsigned int cmd, unsigned long arg)
|
||||
} else {
|
||||
int i;
|
||||
bf_tbus_msix_indices_t msix_ind;
|
||||
if (copy_from_user(&msix_ind, addr, sizeof(bf_tbus_msix_indices_t))) {
|
||||
return EFAULT;
|
||||
}
|
||||
if (msix_ind.cnt > BF_TBUS_MSIX_INDICES_MAX) {
|
||||
return EINVAL;
|
||||
}
|
||||
for (i = 0; i < msix_ind.cnt; i++) {
|
||||
if (msix_ind.indices[i] >= BF_MSIX_ENTRY_CNT) {
|
||||
#if defined(RHEL_RELEASE_CODE)
|
||||
#if defined(RHEL_RELEASE_VERSION)
|
||||
#if RHEL_RELEASE_CODE > RHEL_RELEASE_VERSION(7, 9)
|
||||
if (access_ok(addr, sizeof(bf_tbus_msix_indices_t))) {
|
||||
#else
|
||||
if (access_ok(VERIFY_WRITE, addr, sizeof(bf_tbus_msix_indices_t))) {
|
||||
#endif
|
||||
#endif /* RHEL_RELEASE_CODE && RHEL_RELEASE_VERSION */
|
||||
#elif LINUX_VERSION_CODE >= KERNEL_VERSION(5, 0, 0)
|
||||
if (access_ok(addr, sizeof(bf_tbus_msix_indices_t))) {
|
||||
#else
|
||||
if (access_ok(VERIFY_READ, addr, sizeof(bf_tbus_msix_indices_t))) {
|
||||
#endif
|
||||
if (copy_from_user(&msix_ind, addr, sizeof(bf_tbus_msix_indices_t))) {
|
||||
return EFAULT;
|
||||
}
|
||||
if (msix_ind.cnt < BF_TBUS_MSIX_INDICES_MIN ||
|
||||
msix_ind.cnt > BF_TBUS_MSIX_INDICES_MAX) {
|
||||
return EINVAL;
|
||||
}
|
||||
for (i = 0; i < msix_ind.cnt; i++) {
|
||||
if (msix_ind.indices[i] < 0 ||
|
||||
msix_ind.indices[i] >= BF_MSIX_ENTRY_CNT) {
|
||||
return EINVAL;
|
||||
}
|
||||
}
|
||||
for (i = 0; i < msix_ind.cnt; i++) {
|
||||
bfdev->info.tbus_msix_ind[i] = msix_ind.indices[i];
|
||||
}
|
||||
bfdev->info.tbus_msix_map_enable = 1;
|
||||
} else {
|
||||
return EFAULT;
|
||||
}
|
||||
for (i = 0; i < msix_ind.cnt; i++) {
|
||||
bfdev->info.tbus_msix_ind[i] = msix_ind.indices[i];
|
||||
}
|
||||
bfdev->info.tbus_msix_map_enable = 1;
|
||||
}
|
||||
break;
|
||||
case BF_GET_INTR_MODE:
|
||||
@ -761,7 +819,7 @@ static int bf_major_init(struct bf_pci_dev *bfdev, int minor) {
|
||||
goto fail_dev_add;
|
||||
}
|
||||
|
||||
bf_major = MAJOR(bf_dev);
|
||||
bfdev->info.major = MAJOR(bf_dev);
|
||||
bf_global[minor].bf_cdev = cdev;
|
||||
return 0;
|
||||
|
||||
@ -771,33 +829,93 @@ fail_dev_add:
|
||||
}
|
||||
|
||||
static void bf_major_cleanup(struct bf_pci_dev *bfdev, int minor) {
|
||||
unregister_chrdev_region(MKDEV(bf_major, 0), BF_MAX_DEVICE_CNT);
|
||||
unregister_chrdev_region(MKDEV(bfdev->info.major, 0), BF_MAX_DEVICE_CNT);
|
||||
cdev_del(bf_global[minor].bf_cdev);
|
||||
}
|
||||
|
||||
static int bf_init_cdev(struct bf_pci_dev *bfdev, int minor) {
|
||||
int ret;
|
||||
ret = bf_major_init(bfdev, minor);
|
||||
if (ret) return ret;
|
||||
|
||||
bf_class = class_create(THIS_MODULE, BF_CLASS_NAME);
|
||||
if (!bf_class) {
|
||||
printk(KERN_ERR "create_class failed for bf_dev\n");
|
||||
ret = -ENODEV;
|
||||
goto err_class_register;
|
||||
}
|
||||
return 0;
|
||||
|
||||
err_class_register:
|
||||
bf_major_cleanup(bfdev, minor);
|
||||
return ret;
|
||||
}
|
||||
|
||||
static void bf_remove_cdev(struct bf_pci_dev *bfdev) {
|
||||
class_destroy(bf_class);
|
||||
bf_major_cleanup(bfdev, bfdev->info.minor);
|
||||
}
|
||||
|
||||
/* return the first unused dev_id based on invalid chip_id */
|
||||
static int bf_multisub_tof_unused_devid_get(void) {
|
||||
int i;
|
||||
for (i = 0; i < BF_MAX_DEVICE_CNT; i++) {
|
||||
if ((bf_tof3_info[i]).minor[0] == -1 &&
|
||||
(bf_tof3_info[i]).minor[1] == -1) {
|
||||
return i;
|
||||
}
|
||||
}
|
||||
return -1;
|
||||
}
|
||||
|
||||
/* special case handling for TOF3. each subdevice creates its own device node
|
||||
* device node is named as /dev/bf<devid>s<subdevid 0-3>
|
||||
*/
|
||||
static int bf_tof3_register_device(struct device *parent,
|
||||
struct bf_pci_dev *bfdev) {
|
||||
struct bf_dev_info *info = &bfdev->info;
|
||||
int minor = 0;
|
||||
u8 *bf_base_addr;
|
||||
volatile u32 *bf_addr;
|
||||
int dev_id = 0, subdev_id = 0, ret = 0;
|
||||
|
||||
if (!info || !info->version) {
|
||||
return -EINVAL;
|
||||
}
|
||||
bf_base_addr = (info->mem[0].internal_addr);
|
||||
if (!bf_base_addr) {
|
||||
return -ENODEV;
|
||||
}
|
||||
/* *** TBD for multi Tofino(with 2 subdevices) systems *** */
|
||||
/* We must be able to identify multiple sub devices as belonging to one
|
||||
physical Tofino(3) device. We have not figured that out yet.
|
||||
until then, we support only one CB device per host CPU */
|
||||
bf_addr = (u32 *)((u8 *)bf_base_addr + TOFINO3_MISC_PAD_STATUS_OFFSET);
|
||||
#if 1 /* USING EMULATOR where subdevice info is not possible to have */
|
||||
bf_multisub_tof_unused_devid_get(); /* keep compiler happy */
|
||||
subdev_id = 0;
|
||||
if (bf_get_next_minor_no(&minor)) {
|
||||
return -EINVAL;
|
||||
}
|
||||
dev_id = minor;
|
||||
#else
|
||||
subdev_id = (int)(*bf_addr & TOFINO3_MISC_PAD_STATUS_DIEID0);
|
||||
if (bf_get_next_minor_no(&minor)) {
|
||||
return -EINVAL;
|
||||
}
|
||||
/* we cannot assume the order in which sub devices are probed */
|
||||
if (subdev_id == 0) {
|
||||
dev_id = bf_multisub_tof_unused_devid_get();
|
||||
bf_tof3_info[dev_id].dev_id = dev_id; /* back reference */
|
||||
(bf_tof3_info[dev_id].minor)[subdev_id] = minor;
|
||||
} else {
|
||||
dev_id = 0; /* TBD : for Tofino with multi sub devices */
|
||||
(bf_tof3_info[dev_id].minor)[subdev_id] = minor;
|
||||
}
|
||||
#endif
|
||||
ret = bf_init_cdev(bfdev, minor);
|
||||
if (ret) {
|
||||
printk(KERN_ERR "BF: device cdev creation failed dev_id %d\n", dev_id);
|
||||
return ret;
|
||||
}
|
||||
info->tof3_info = &(bf_tof3_info[dev_id]);
|
||||
info->dev_id = dev_id;
|
||||
info->subdev_id = subdev_id;
|
||||
printk(KERN_NOTICE "BF: registering dev_id %d subdev_id %d\n",
|
||||
dev_id, subdev_id);
|
||||
info->dev = device_create(bf_class, parent, MKDEV(bfdev->info.major, minor),
|
||||
bfdev, "bf%ds%1d", dev_id, subdev_id);
|
||||
info->minor = minor;
|
||||
return 0;
|
||||
}
|
||||
|
||||
/**
|
||||
* bf_register_device - register a new userspace mem device
|
||||
* @parent: parent device
|
||||
@ -808,7 +926,7 @@ static void bf_remove_cdev(struct bf_pci_dev *bfdev) {
|
||||
int bf_register_device(struct device *parent, struct bf_pci_dev *bfdev) {
|
||||
struct bf_dev_info *info = &bfdev->info;
|
||||
int i, j, ret = 0;
|
||||
int minor;
|
||||
int minor = 0;
|
||||
|
||||
if (!parent || !info || !info->version) {
|
||||
return -EINVAL;
|
||||
@ -820,25 +938,33 @@ int bf_register_device(struct device *parent, struct bf_pci_dev *bfdev) {
|
||||
atomic_set(&info->event[i], 0);
|
||||
}
|
||||
|
||||
if (bf_get_next_minor_no(&minor)) {
|
||||
return -EINVAL;
|
||||
}
|
||||
if (info->tof_type == BF_TOFINO_3) {
|
||||
if ((ret = bf_tof3_register_device(parent, bfdev)) != 0) {
|
||||
printk(KERN_ERR "BF: TOF3 device cdev creation failed %d\n", ret);
|
||||
return ret;
|
||||
}
|
||||
} else {
|
||||
if (bf_get_next_minor_no(&minor)) {
|
||||
return -EINVAL;
|
||||
}
|
||||
|
||||
ret = bf_init_cdev(bfdev, minor);
|
||||
if (ret) {
|
||||
printk(KERN_ERR "BF: device cdev creation failed\n");
|
||||
return ret;
|
||||
}
|
||||
ret = bf_init_cdev(bfdev, minor);
|
||||
if (ret) {
|
||||
printk(KERN_ERR "BF: device cdev creation failed\n");
|
||||
return ret;
|
||||
}
|
||||
|
||||
info->dev = device_create(
|
||||
bf_class, parent, MKDEV(bf_major, minor), bfdev, "bf%d", minor);
|
||||
info->dev = device_create(
|
||||
bf_class, parent, MKDEV(bfdev->info.major, minor), bfdev, "bf%d", minor);
|
||||
info->minor = minor;
|
||||
info->dev_id = minor;
|
||||
info->subdev_id = 0;
|
||||
}
|
||||
if (!info->dev) {
|
||||
printk(KERN_ERR "BF: device creation failed\n");
|
||||
return -ENODEV;
|
||||
}
|
||||
|
||||
info->minor = minor;
|
||||
|
||||
/* bind ISRs and request interrupts */
|
||||
if (info->irq && (bfdev->mode != BF_INTR_MODE_NONE)) {
|
||||
/*
|
||||
@ -906,6 +1032,33 @@ int bf_register_device(struct device *parent, struct bf_pci_dev *bfdev) {
|
||||
return 0;
|
||||
}
|
||||
|
||||
/* special case handling for TOF3. return minor number only after all
|
||||
* sub devices using the minor number are unregistered */
|
||||
static int bf_tof3_unregister_device(struct bf_pci_dev *bfdev) {
|
||||
struct bf_dev_info *info = &bfdev->info;
|
||||
#if 1 //HACK until emulator implements efuse
|
||||
bf_return_minor_no(info->minor);
|
||||
#else
|
||||
int j, dev_id, subdev_id, found;
|
||||
|
||||
if (!info->tof3_info) {
|
||||
printk(KERN_ERR "BF TOF3 bad info in tof3_unregister_device\n");
|
||||
return -1;
|
||||
}
|
||||
dev_id = info->tof3_info->dev_id;
|
||||
subdev_id = info->subdev_id;
|
||||
if (dev_id >= BF_MAX_DEVICE_CNT || subdev_id >= BF_MAX_SUBDEV_CNT) {
|
||||
return -1;
|
||||
}
|
||||
/* update bf_tof3_info structure for the device being unregistered */
|
||||
(bf_tof3_info[dev_id].minor)[subdev_id] = -1;
|
||||
/* return the minor number */
|
||||
bf_return_minor_no(info->minor);
|
||||
info->subdev_id = -1;
|
||||
#endif
|
||||
return 0;
|
||||
}
|
||||
|
||||
/**
|
||||
* bf_unregister_device - register a new userspace mem device
|
||||
* @bfdev: bf pci device
|
||||
@ -914,8 +1067,12 @@ int bf_register_device(struct device *parent, struct bf_pci_dev *bfdev) {
|
||||
*/
|
||||
void bf_unregister_device(struct bf_pci_dev *bfdev) {
|
||||
struct bf_dev_info *info = &bfdev->info;
|
||||
int i;
|
||||
int i, ret;
|
||||
|
||||
if (!info) {
|
||||
printk(KERN_ERR "BF: bad data in device unregister\n");
|
||||
return;
|
||||
}
|
||||
if (info->irq) {
|
||||
if (bfdev->mode == BF_INTR_MODE_LEGACY) {
|
||||
free_irq(info->irq, (void *)&(bfdev->bf_int_vec[0]));
|
||||
@ -929,9 +1086,16 @@ void bf_unregister_device(struct bf_pci_dev *bfdev) {
|
||||
}
|
||||
}
|
||||
}
|
||||
device_destroy(bf_class, MKDEV(bf_major, info->minor));
|
||||
device_destroy(bf_class, MKDEV(info->major, info->minor));
|
||||
bf_remove_cdev(bfdev);
|
||||
bf_return_minor_no(info->minor);
|
||||
if (bfdev->info.tof_type == BF_TOFINO_3) {
|
||||
if ((ret = bf_tof3_unregister_device(bfdev)) != 0) {
|
||||
printk(KERN_ERR "BF: TOF3 device cdev unregister failed %d\n", ret);
|
||||
return;
|
||||
}
|
||||
} else {
|
||||
bf_return_minor_no(info->minor);
|
||||
}
|
||||
return;
|
||||
}
|
||||
|
||||
@ -944,13 +1108,13 @@ static void bf_disable_int_dma(struct bf_pci_dev *bfdev) {
|
||||
u32 *bf_addr;
|
||||
volatile u32 val;
|
||||
|
||||
/* maskinterrupts and DMA */
|
||||
/* mask interrupts and DMA */
|
||||
bf_base_addr = (bfdev->info.mem[0].internal_addr);
|
||||
/* return if called before mmap */
|
||||
if (!bf_base_addr) {
|
||||
return;
|
||||
}
|
||||
/* mask interrupt at shadow level */
|
||||
/* mask interrupts at shadow level */
|
||||
bf_addr = (u32 *)((u8 *)bf_base_addr + 0xc0);
|
||||
for (i = 0; i < 16; i++) {
|
||||
*bf_addr = 0xffffffffUL;
|
||||
@ -968,8 +1132,6 @@ static int bf_pci_probe(struct pci_dev *pdev, const struct pci_device_id *id) {
|
||||
int err, pci_use_highmem;
|
||||
int i, num_irq;
|
||||
|
||||
memset(bf_global, 0, sizeof(bf_global));
|
||||
|
||||
bfdev = kzalloc(sizeof(struct bf_pci_dev), GFP_KERNEL);
|
||||
if (!bfdev) {
|
||||
return -ENOMEM;
|
||||
@ -995,6 +1157,9 @@ static int bf_pci_probe(struct pci_dev *pdev, const struct pci_device_id *id) {
|
||||
case TOFINO2_DEV_ID_B0:
|
||||
bfdev->info.tof_type = BF_TOFINO_2;
|
||||
break;
|
||||
case TOFINO3_DEV_ID_A0:
|
||||
bfdev->info.tof_type = BF_TOFINO_3;
|
||||
break;
|
||||
default:
|
||||
bfdev->info.tof_type = BF_TOFINO_1;
|
||||
break;
|
||||
@ -1005,6 +1170,8 @@ static int bf_pci_probe(struct pci_dev *pdev, const struct pci_device_id *id) {
|
||||
bfdev->info.tbus_msix_ind[i] = BF_TBUS_MSIX_BASE_INDEX_TOF1 + i;
|
||||
} else if (bfdev->info.tof_type == BF_TOFINO_2) {
|
||||
bfdev->info.tbus_msix_ind[i] = BF_TBUS_MSIX_INDEX_INVALID;
|
||||
} else if (bfdev->info.tof_type == BF_TOFINO_3) {
|
||||
bfdev->info.tbus_msix_ind[i] = BF_TBUS_MSIX_INDEX_INVALID;
|
||||
}
|
||||
}
|
||||
/*
|
||||
@ -1068,7 +1235,6 @@ static int bf_pci_probe(struct pci_dev *pdev, const struct pci_device_id *id) {
|
||||
bfdev->info.version = "0.2";
|
||||
bfdev->info.owner = THIS_MODULE;
|
||||
bfdev->pdev = pdev;
|
||||
|
||||
switch (bf_intr_mode_default) {
|
||||
#ifdef CONFIG_PCI_MSI
|
||||
case BF_INTR_MODE_MSIX:
|
||||
@ -1113,6 +1279,7 @@ static int bf_pci_probe(struct pci_dev *pdev, const struct pci_device_id *id) {
|
||||
}
|
||||
#endif /* LINUX_VERSION_CODE */
|
||||
/* ** intentional no-break */
|
||||
/* FALLTHRU */
|
||||
case BF_INTR_MODE_MSI:
|
||||
#if LINUX_VERSION_CODE < KERNEL_VERSION(3, 14, 0)
|
||||
num_irq = pci_enable_msi_block(pdev, BF_MSI_ENTRY_CNT);
|
||||
@ -1150,6 +1317,7 @@ static int bf_pci_probe(struct pci_dev *pdev, const struct pci_device_id *id) {
|
||||
#endif /* LINUX_VERSION_CODE */
|
||||
#endif /* CONFIG_PCI_MSI */
|
||||
/* fall back to Legacy Interrupt, intentional no-break */
|
||||
/* FALLTHRU */
|
||||
|
||||
case BF_INTR_MODE_LEGACY:
|
||||
if (pci_intx_mask_supported(pdev)) {
|
||||
@ -1161,6 +1329,7 @@ static int bf_pci_probe(struct pci_dev *pdev, const struct pci_device_id *id) {
|
||||
}
|
||||
printk(KERN_NOTICE " bf PCI INTx mask not supported\n");
|
||||
/* fall back to no Interrupt, intentional no-break */
|
||||
/* FALLTHRU */
|
||||
case BF_INTR_MODE_NONE:
|
||||
bfdev->info.irq = 0;
|
||||
bfdev->info.num_irq = 0;
|
||||
@ -1183,6 +1352,7 @@ static int bf_pci_probe(struct pci_dev *pdev, const struct pci_device_id *id) {
|
||||
|
||||
bf_global[bfdev->info.minor].async_queue = NULL;
|
||||
bf_global[bfdev->info.minor].bfdev = bfdev;
|
||||
bf_global[bfdev->info.minor].pending_signal = false;
|
||||
|
||||
dev_info(&pdev->dev,
|
||||
"bf device %d registered with irq %ld\n",
|
||||
@ -1194,15 +1364,18 @@ static int bf_pci_probe(struct pci_dev *pdev, const struct pci_device_id *id) {
|
||||
err = bf_kpkt_init(pdev,
|
||||
bfdev->info.mem[0].internal_addr,
|
||||
&bfdev->adapter_ptr,
|
||||
bfdev->info.minor,
|
||||
bfdev->info.dev_id,
|
||||
bfdev->info.subdev_id,
|
||||
pci_use_highmem,
|
||||
kpkt_hd_room,
|
||||
kpkt_dr_int_en,
|
||||
kpkt_rx_count);
|
||||
if (err == 0) {
|
||||
printk(KERN_ALERT "bf_kpkt kernel processing enabled\n");
|
||||
printk(KERN_ALERT "bf_kpkt kernel processing enabled for dev %d subdev_id %d\n",
|
||||
bfdev->info.dev_id, bfdev->info.subdev_id);
|
||||
} else {
|
||||
printk(KERN_ALERT "error starting bf_kpkt kernel processing\n");
|
||||
printk(KERN_ERR "error starting bf_kpkt kernel processing for dev %d subdev_id %d\n",
|
||||
bfdev->info.dev_id, bfdev->info.subdev_id);
|
||||
bfdev->adapter_ptr = NULL;
|
||||
}
|
||||
}
|
||||
@ -1316,16 +1489,29 @@ static pci_ers_result_t bf_pci_mmio_enabled(struct pci_dev *dev) {
|
||||
struct bf_pci_dev *bfdev = pci_get_drvdata(dev);
|
||||
|
||||
printk(KERN_ERR "BF pci_mmio_enabled invoked after pci error\n");
|
||||
#if KERNEL_VERSION(5, 8, 0) <= LINUX_VERSION_CODE
|
||||
pci_aer_clear_nonfatal_status(dev);
|
||||
#else
|
||||
pci_cleanup_aer_uncorrect_error_status(dev);
|
||||
#endif
|
||||
#if defined(RHEL_RELEASE_CODE)
|
||||
#if defined(RHEL_RELEASE_VERSION)
|
||||
#if RHEL_RELEASE_CODE > RHEL_RELEASE_VERSION(7, 9)
|
||||
pci_aer_clear_nonfatal_status(dev);
|
||||
#else
|
||||
pci_cleanup_aer_uncorrect_error_status(dev);
|
||||
#endif
|
||||
#endif /* RHEL_RELEASE_CODE && RHEL_RELEASE_VERSION */
|
||||
#elif LINUX_VERSION_CODE >= KERNEL_VERSION(5, 8, 0)
|
||||
pci_aer_clear_nonfatal_status(dev);
|
||||
#else
|
||||
pci_cleanup_aer_uncorrect_error_status(dev);
|
||||
#endif
|
||||
|
||||
if (bfdev) {
|
||||
/* send a signal to the user space program of the error */
|
||||
int minor = bfdev->info.minor;
|
||||
if (minor < BF_MAX_DEVICE_CNT && bf_global[minor].async_queue) {
|
||||
kill_fasync(&bf_global[minor].async_queue, SIGIO, POLL_ERR);
|
||||
if (minor < BF_MAX_DEVICE_CNT) {
|
||||
if (bf_global[minor].async_queue) {
|
||||
kill_fasync(&bf_global[minor].async_queue, SIGIO, POLL_ERR);
|
||||
} else {
|
||||
bf_global[minor].pending_signal = true;
|
||||
}
|
||||
}
|
||||
}
|
||||
return PCI_ERS_RESULT_RECOVERED;
|
||||
@ -1389,6 +1575,7 @@ static const struct pci_device_id bf_pci_tbl[] = {
|
||||
{PCI_VDEVICE(BF, TOFINO2_DEV_ID_A0), 0},
|
||||
{PCI_VDEVICE(BF, TOFINO2_DEV_ID_A00), 0},
|
||||
{PCI_VDEVICE(BF, TOFINO2_DEV_ID_B0), 0},
|
||||
{PCI_VDEVICE(INTEL, TOFINO3_DEV_ID_A0), 0},
|
||||
/* required last entry */
|
||||
{.device = 0}};
|
||||
|
||||
@ -1409,12 +1596,36 @@ static struct pci_driver bf_pci_driver = {.name = "bf",
|
||||
static int __init bfdrv_init(void) {
|
||||
int ret;
|
||||
|
||||
pr_info("%s: %s - version %s\n", DRV_NAME(kpkt_mode),
|
||||
DRV_DESCRIPTION(kpkt_mode),DRV_VERSION);
|
||||
pr_info("%s: %s\n", DRV_NAME(kpkt_mode),DRV_COPYRIGHT);
|
||||
memset(bf_global, 0, sizeof(bf_global));
|
||||
memset(bf_tof3_info, 0xff, sizeof(bf_tof3_info));
|
||||
bf_class = class_create(THIS_MODULE, BF_CLASS_NAME);
|
||||
if (!bf_class) {
|
||||
printk(KERN_ERR "create_class failed for bf device\n");
|
||||
return -ENODEV;
|
||||
} else {
|
||||
printk(KERN_NOTICE "bf device class created\n");
|
||||
}
|
||||
|
||||
ret = bf_config_intr_mode(intr_mode);
|
||||
|
||||
if (ret < 0) {
|
||||
printk(KERN_ERR "config interrupt mode failed for bf device\n");
|
||||
if (bf_class) {
|
||||
class_destroy(bf_class);
|
||||
bf_class = NULL;
|
||||
}
|
||||
return ret;
|
||||
}
|
||||
|
||||
/* do not enable DR interrupt if not using MSI or not in kpkt mode */
|
||||
if ((bf_intr_mode_default != BF_INTR_MODE_MSI &&
|
||||
bf_intr_mode_default != BF_INTR_MODE_LEGACY) || kpkt_mode == 0) {
|
||||
kpkt_dr_int_en = 0;
|
||||
}
|
||||
|
||||
if (kpkt_mode) {
|
||||
printk(KERN_NOTICE "kpkt_mode %d hd_room %d dr_int_en %d rx_count %d\n",
|
||||
kpkt_mode,
|
||||
@ -1422,15 +1633,17 @@ static int __init bfdrv_init(void) {
|
||||
kpkt_dr_int_en,
|
||||
kpkt_rx_count);
|
||||
}
|
||||
if (ret < 0) {
|
||||
return ret;
|
||||
}
|
||||
|
||||
spin_lock_init(&bf_nonisr_lock);
|
||||
return pci_register_driver(&bf_pci_driver);
|
||||
}
|
||||
|
||||
static void __exit bfdrv_exit(void) {
|
||||
pr_info("%s: module unloading ...\n", DRV_NAME(kpkt_mode));
|
||||
pci_unregister_driver(&bf_pci_driver);
|
||||
class_destroy(bf_class);
|
||||
pr_info("%s: module unloaded successfully\n", DRV_NAME(kpkt_mode));
|
||||
bf_class = NULL;
|
||||
intr_mode = NULL;
|
||||
kpkt_mode = 0;
|
||||
}
|
||||
|
@ -1,7 +1,7 @@
|
||||
/*******************************************************************************
|
||||
Barefoot Networks Switch ASIC Linux driver
|
||||
Copyright(c) 2015 - 2019 Barefoot Networks, Inc.
|
||||
|
||||
|
||||
This program is free software; you can redistribute it and/or modify it
|
||||
under the terms and conditions of the GNU General Public License,
|
||||
version 2, as published by the Free Software Foundation.
|
||||
@ -35,11 +35,13 @@ typedef uint64_t phys_addr_t;
|
||||
#endif
|
||||
|
||||
#define PCI_VENDOR_ID_BF 0x1d1c
|
||||
#define PCI_VENDOR_ID_INTEL 0x8086
|
||||
#define TOFINO_DEV_ID_A0 0x01
|
||||
#define TOFINO_DEV_ID_B0 0x10
|
||||
#define TOFINO2_DEV_ID_A0 0x0100
|
||||
#define TOFINO2_DEV_ID_A00 0x0000
|
||||
#define TOFINO2_DEV_ID_B0 0x0110
|
||||
#define TOFINO3_DEV_ID_A0 0x0DA2
|
||||
|
||||
#ifndef PCI_MSIX_ENTRY_SIZE
|
||||
#define PCI_MSIX_ENTRY_SIZE 16
|
||||
@ -51,7 +53,9 @@ typedef uint64_t phys_addr_t;
|
||||
#endif
|
||||
|
||||
#define BF_CLASS_NAME "bf"
|
||||
#define BF_MAX_DEVICE_CNT 256
|
||||
#define BF_MAX_DEVICE_CNT 8
|
||||
#define BF_MAX_SUBDEV_CNT 2
|
||||
#define BF_MAX_DEV_SUBDEV_CNT (BF_MAX_DEVICE_CNT * BF_MAX_SUBDEV_CNT)
|
||||
#define BF_INTR_MODE_NONE_NAME "none"
|
||||
#define BF_INTR_MODE_LEGACY_NAME "legacy"
|
||||
#define BF_INTR_MODE_MSI_NAME "msi"
|
||||
@ -64,11 +68,22 @@ typedef uint64_t phys_addr_t;
|
||||
#define BF_TBUS_MSIX_INDEX_INVALID (0)
|
||||
#define BF_TBUS_MSIX_BASE_INDEX_TOF1 (32)
|
||||
|
||||
#define TOFINO3_MISC_PAD_STATUS_OFFSET 0x80238UL
|
||||
#define TOFINO3_MISC_PAD_STATUS_DIEID0 (1 << 3)
|
||||
|
||||
#define DRV_NAME(kpkt_mode) (kpkt_mode ? "bf_kpkt" : "bf_kdrv")
|
||||
#define DRV_DESCRIPTION(kpkt_mode) (kpkt_mode ? \
|
||||
"Intel(R) Switch ASIC Linux Packet Driver" : \
|
||||
"Intel(R) Switch ASIC Linux Driver" )
|
||||
#define DRV_COPYRIGHT "Copyright (c) 2015-2022 Intel Corporation."
|
||||
#define DRV_VERSION "1.0"
|
||||
|
||||
/* Tofino generation type */
|
||||
typedef enum {
|
||||
BF_TOFINO_NONE = 0,
|
||||
BF_TOFINO_1,
|
||||
BF_TOFINO_2,
|
||||
BF_TOFINO_3,
|
||||
} bf_tof_type;
|
||||
|
||||
/* device memory */
|
||||
@ -86,10 +101,16 @@ struct bf_listener {
|
||||
struct bf_listener *next;
|
||||
};
|
||||
|
||||
struct bf_tof3_info_s {
|
||||
int dev_id;
|
||||
int minor[BF_MAX_SUBDEV_CNT];
|
||||
};
|
||||
|
||||
/* device information */
|
||||
struct bf_dev_info {
|
||||
struct module *owner;
|
||||
struct device *dev;
|
||||
int major;
|
||||
int minor;
|
||||
atomic_t event[BF_MSIX_ENTRY_CNT];
|
||||
wait_queue_head_t wait;
|
||||
@ -104,6 +125,9 @@ struct bf_dev_info {
|
||||
/* msix index assigned to tbus MSIX for Tofino-2 only */
|
||||
int tbus_msix_ind[BF_TBUS_MSIX_INDICES_MAX];
|
||||
int tbus_msix_map_enable;
|
||||
struct bf_tof3_info_s *tof3_info;
|
||||
int dev_id; /* same as minor number for T1 and T2 */
|
||||
int subdev_id;
|
||||
int pci_error_state; /* was there a pci bus error */
|
||||
};
|
||||
|
||||
@ -126,6 +150,7 @@ struct bf_pci_dev {
|
||||
struct bf_listener *
|
||||
listener_head; /* head of a singly linked list of listeners */
|
||||
void *adapter_ptr; /* pkt processing adapter */
|
||||
int in_use; /* indicates a user space process is using the device */
|
||||
};
|
||||
|
||||
/* TBD: Need to build with CONFIG_PCI_MSI */
|
||||
|
@ -110,9 +110,10 @@ int bf_fpga_ioctl(struct bf_pci_dev *bfdev,
|
||||
/* copy read data to user area */
|
||||
for (i = 0; i < i2c_op.num_i2c; i++) {
|
||||
if (i2c_op.i2c_inst[i].rd_cnt) {
|
||||
if (copy_to_user(&(((bf_fpga_i2c_t *)addr)->i2c_inst[i].rd_buf),
|
||||
&i2c_op.i2c_inst[i].rd_buf,
|
||||
i2c_op.i2c_inst[i].rd_cnt)) {
|
||||
if (copy_to_user(
|
||||
&(((bf_fpga_i2c_t *)addr)->i2c_inst[i].fpga_i2c_buf.rd_buf),
|
||||
&i2c_op.i2c_inst[i].fpga_i2c_buf.rd_buf,
|
||||
i2c_op.i2c_inst[i].rd_cnt)) {
|
||||
return -EFAULT;
|
||||
}
|
||||
}
|
||||
|
@ -93,7 +93,7 @@ typedef struct bf_fpga_i2c_inst_s {
|
||||
unsigned char
|
||||
wr_buf[BF_FPGA_MAX_I2C_WR_DATA]; /* write data source buffer */
|
||||
unsigned char rd_buf[BF_FPGA_MAX_I2C_RD_DATA]; /* read data dest buffer */
|
||||
};
|
||||
} fpga_i2c_buf;
|
||||
unsigned char status;
|
||||
unsigned char retry_cnt; /* if fpga maintains retry count */
|
||||
unsigned char mux; /* if fpga maintains internal MUX */
|
||||
|
@ -120,7 +120,7 @@ static void bf_remove_listener(struct bf_pci_dev *bfdev,
|
||||
|
||||
/* in case of certain error conditions, this function might be called after
|
||||
* bf_pci_remove()
|
||||
*/
|
||||
*/
|
||||
if (!bfdev || !listener) {
|
||||
return;
|
||||
}
|
||||
@ -294,7 +294,12 @@ static int bf_setup_bars(struct pci_dev *dev, struct bf_dev_info *info) {
|
||||
int i, iom, ret;
|
||||
unsigned long flags;
|
||||
static const char *bar_names[BF_MAX_BAR_MAPS] = {
|
||||
"BAR0", "BAR1", "BAR2", "BAR3", "BAR4", "BAR5",
|
||||
"BAR0",
|
||||
"BAR1",
|
||||
"BAR2",
|
||||
"BAR3",
|
||||
"BAR4",
|
||||
"BAR5",
|
||||
};
|
||||
|
||||
iom = 0;
|
||||
@ -849,7 +854,7 @@ static void fpga_print_build_date(u32 build_date) {
|
||||
month = (char)(build_date & 0x0f);
|
||||
build_date >>= 4;
|
||||
day = (char)(build_date & 0x1f);
|
||||
printk(KERN_ALERT "fpga version %02d/%02d/%2d %02d:%02d:%02d",
|
||||
printk(KERN_ALERT "fpga build %02d/%02d/%2d %02d:%02d:%02d",
|
||||
month,
|
||||
day,
|
||||
year,
|
||||
@ -991,6 +996,7 @@ static int bf_pci_probe(struct pci_dev *pdev, const struct pci_device_id *id) {
|
||||
}
|
||||
#endif /* LINUX_VERSION_CODE */
|
||||
/* ** intentional no-break */
|
||||
/* FALLTHRU */
|
||||
case BF_INTR_MODE_MSI:
|
||||
#if LINUX_VERSION_CODE < KERNEL_VERSION(3, 14, 0)
|
||||
num_irq = pci_enable_msi_block(pdev, BF_MSI_ENTRY_CNT);
|
||||
@ -1032,7 +1038,8 @@ static int bf_pci_probe(struct pci_dev *pdev, const struct pci_device_id *id) {
|
||||
}
|
||||
#endif /* LINUX_VERSION_CODE */
|
||||
#endif /* CONFIG_PCI_MSI */
|
||||
/* fall back to Legacy Interrupt, intentional no-break */
|
||||
/* fall back to Legacy Interrupt, intentional no-break */
|
||||
/* FALLTHRU */
|
||||
|
||||
case BF_INTR_MODE_LEGACY:
|
||||
if (pci_intx_mask_supported(pdev)) {
|
||||
@ -1044,6 +1051,7 @@ static int bf_pci_probe(struct pci_dev *pdev, const struct pci_device_id *id) {
|
||||
}
|
||||
printk(KERN_NOTICE "bf_fpga PCI INTx mask not supported\n");
|
||||
/* fall back to no Interrupt, intentional no-break */
|
||||
/* FALLTHRU */
|
||||
case BF_INTR_MODE_NONE:
|
||||
bfdev->info.irq = 0;
|
||||
bfdev->info.num_irq = 0;
|
||||
@ -1084,7 +1092,7 @@ static int bf_pci_probe(struct pci_dev *pdev, const struct pci_device_id *id) {
|
||||
build_date =
|
||||
*((u32 *)(bfdev->info.mem[0].internal_addr) + (BF_FPGA_BUILD_DATE / 4));
|
||||
fpga_print_build_date(build_date);
|
||||
printk(KERN_ALERT "bf_fpga version %hu:%hu probe ok\n",
|
||||
printk(KERN_ALERT "bf_fpga version %hu.%hu probe ok\n",
|
||||
(u16)(build_ver >> 16),
|
||||
(u16)(build_ver));
|
||||
return 0;
|
||||
@ -1239,6 +1247,7 @@ static int bf_config_intr_mode(char *intr_str) {
|
||||
|
||||
static const struct pci_device_id bf_pci_tbl[] = {
|
||||
{PCI_VDEVICE(BF, BF_FPGA_DEV_ID_JBAY_0), 0},
|
||||
{PCI_VDEVICE(BF, BF_FPGA_DEV_ID_CB_0), 0},
|
||||
/* required last entry */
|
||||
{.device = 0}};
|
||||
|
||||
|
@ -28,6 +28,7 @@
|
||||
|
||||
#define PCI_VENDOR_ID_BF 0x1d1c
|
||||
#define BF_FPGA_DEV_ID_JBAY_0 0x01F0
|
||||
#define BF_FPGA_DEV_ID_CB_0 0x01F1
|
||||
|
||||
#ifndef PCI_MSIX_ENTRY_SIZE
|
||||
#define PCI_MSIX_ENTRY_SIZE 16
|
||||
@ -123,8 +124,8 @@ struct bf_pci_dev {
|
||||
u8 instance;
|
||||
char name[16];
|
||||
struct bf_int_vector bf_int_vec[BF_MSIX_ENTRY_CNT];
|
||||
struct bf_listener *
|
||||
listener_head; /* head of a singly linked list of listeners */
|
||||
struct bf_listener
|
||||
*listener_head; /* head of a singly linked list of listeners */
|
||||
struct bf_fpga_sysfs_buff fpga_sysfs_buff[BF_FPGA_SYSFS_CNT];
|
||||
struct bf_fpga_sysfs_buff fpga_sysfs_new_device;
|
||||
struct bf_fpga_sysfs_buff fpga_sysfs_rm_device;
|
||||
|
@ -76,7 +76,7 @@ static ssize_t bf_fpga_sysfs_i2c_get(struct device *dev,
|
||||
i2c_op.i2c_inst[0].status);
|
||||
return -EIO;
|
||||
}
|
||||
memcpy(buf, i2c_op.i2c_inst[0].rd_buf, cur_cnt);
|
||||
memcpy(buf, i2c_op.i2c_inst[0].fpga_i2c_buf.rd_buf, cur_cnt);
|
||||
buf += cur_cnt;
|
||||
size += cur_cnt;
|
||||
cur_size -= cur_cnt;
|
||||
@ -108,7 +108,7 @@ static ssize_t bf_fpga_sysfs_i2c_set(struct device *dev,
|
||||
}
|
||||
i2c_op.i2c_inst[0].wr_cnt = cur_cnt;
|
||||
i2c_op.i2c_inst[0].rd_cnt = 0;
|
||||
memcpy(i2c_op.i2c_inst[0].wr_buf, buf, cur_cnt);
|
||||
memcpy(i2c_op.i2c_inst[0].fpga_i2c_buf.wr_buf, buf, cur_cnt);
|
||||
i2c_op.num_i2c = 1;
|
||||
i2c_op.one_time = 1;
|
||||
i2c_op.inst_hndl.bus_id = sysfs_buf->bus_id;
|
||||
|
@ -146,11 +146,11 @@ static int fpga_i2c_enqueue(int bus_id,
|
||||
wd0 |= (I2C_WR_ADDR_DATA | (delay << I2C_DELAY_SHF));
|
||||
wd1 |= (i2c_inst->i2c_addr << I2C_DEV_ADDR_SHF);
|
||||
/* copy the first byte into register address */
|
||||
wd1 |= ((i2c_inst->wr_buf[0]) << I2C_CMD_OFFSET);
|
||||
wd1 |= ((i2c_inst->fpga_i2c_buf.wr_buf[0]) << I2C_CMD_OFFSET);
|
||||
wd1 |= ((num_wr - 1) << I2C_WR_CNT_SHF);
|
||||
if (num_wr <= 9) {
|
||||
/* copy data into instruction area */
|
||||
memcpy(i2c_data, &i2c_inst->wr_buf[1], (num_wr - 1));
|
||||
memcpy(i2c_data, &i2c_inst->fpga_i2c_buf.wr_buf[1], (num_wr - 1));
|
||||
bf_fpga_i2c_reg_write32(
|
||||
i2c_ctrl, Bf_FPGA_I2C_INST_DATA_LO(inst_id), i2c_data[0]);
|
||||
bf_fpga_i2c_reg_write32(
|
||||
@ -159,7 +159,7 @@ static int fpga_i2c_enqueue(int bus_id,
|
||||
/* copy the data in data area */
|
||||
int len = num_wr - 1;
|
||||
uint32_t addr;
|
||||
uint8_t *val = (uint8_t *)(&i2c_inst->wr_buf[1]);
|
||||
uint8_t *val = (uint8_t *)(&i2c_inst->fpga_i2c_buf.wr_buf[1]);
|
||||
/* store the data pointer Note the indexing required by FPGA specs */
|
||||
i2c_data[0] = BF_FPGA_I2C_DATA_AREA(inst_id);
|
||||
addr = i2c_data[0];
|
||||
@ -195,11 +195,11 @@ static int fpga_i2c_enqueue(int bus_id,
|
||||
wd1 |= (i2c_inst->i2c_addr << I2C_DEV_ADDR_SHF);
|
||||
/* 1st byte of the write buf goes into "register address" field */
|
||||
wd1 |= ((num_wr - 1) << I2C_WR_CNT_SHF);
|
||||
wd1 |= ((i2c_inst->wr_buf[0]) << I2C_CMD_OFFSET);
|
||||
wd1 |= ((i2c_inst->fpga_i2c_buf.wr_buf[0]) << I2C_CMD_OFFSET);
|
||||
wd1 |= ((num_rd) << I2C_RD_CNT_SHF);
|
||||
/* less than 8 bytes data goes to the instruction area */
|
||||
if ((num_wr - 1 + num_rd) <= 8) {
|
||||
memcpy(i2c_data, &i2c_inst->wr_buf[1], (num_wr - 1));
|
||||
memcpy(i2c_data, &i2c_inst->fpga_i2c_buf.wr_buf[1], (num_wr - 1));
|
||||
bf_fpga_i2c_reg_write32(
|
||||
i2c_ctrl, Bf_FPGA_I2C_INST_DATA_LO(inst_id), i2c_data[0]);
|
||||
bf_fpga_i2c_reg_write32(
|
||||
@ -207,7 +207,7 @@ static int fpga_i2c_enqueue(int bus_id,
|
||||
} else {
|
||||
int len = num_wr - 1;
|
||||
uint32_t addr;
|
||||
uint8_t *val = (uint8_t *)(&i2c_inst->wr_buf[1]);
|
||||
uint8_t *val = (uint8_t *)(&i2c_inst->fpga_i2c_buf.wr_buf[1]);
|
||||
/* store the data area pointer */
|
||||
i2c_data[0] = BF_FPGA_I2C_DATA_AREA(inst_id);
|
||||
addr = i2c_data[0];
|
||||
@ -419,7 +419,7 @@ int fpga_i2c_oneshot(bf_fpga_i2c_t *i2c_op) {
|
||||
i,
|
||||
offset,
|
||||
i2c_op->i2c_inst[i].rd_cnt,
|
||||
i2c_op->i2c_inst[i].rd_buf)) {
|
||||
i2c_op->i2c_inst[i].fpga_i2c_buf.rd_buf)) {
|
||||
ret = BF_FPGA_EIO;
|
||||
goto oneshot_error_exit;
|
||||
}
|
||||
|
Reference in New Issue
Block a user