qlcnic: modify PCI and register access routines
Refactor 82xx driver to support new adapter Update PCI and hardware access routines Signed-off-by: Anirban Chakraborty <anirban.chakraborty@qlogic.com> Signed-off-by: Sony Chacko <sony.chacko@qlogic.com> Signed-off-by: David S. Miller <davem@davemloft.net>
This commit is contained in:
parent
797884509d
commit
15087c2b31
@ -1279,7 +1279,7 @@ struct qlcnic_cmd_args {
|
|||||||
int qlcnic_fw_cmd_get_minidump_temp(struct qlcnic_adapter *adapter);
|
int qlcnic_fw_cmd_get_minidump_temp(struct qlcnic_adapter *adapter);
|
||||||
int qlcnic_fw_cmd_set_port(struct qlcnic_adapter *adapter, u32 config);
|
int qlcnic_fw_cmd_set_port(struct qlcnic_adapter *adapter, u32 config);
|
||||||
|
|
||||||
u32 qlcnic_hw_read_wx_2M(struct qlcnic_adapter *adapter, ulong off);
|
int qlcnic_hw_read_wx_2M(struct qlcnic_adapter *adapter, ulong off);
|
||||||
int qlcnic_hw_write_wx_2M(struct qlcnic_adapter *, ulong off, u32 data);
|
int qlcnic_hw_write_wx_2M(struct qlcnic_adapter *, ulong off, u32 data);
|
||||||
int qlcnic_pci_mem_write_2M(struct qlcnic_adapter *, u64 off, u64 data);
|
int qlcnic_pci_mem_write_2M(struct qlcnic_adapter *, u64 off, u64 data);
|
||||||
int qlcnic_pci_mem_read_2M(struct qlcnic_adapter *, u64 off, u64 *data);
|
int qlcnic_pci_mem_read_2M(struct qlcnic_adapter *, u64 off, u64 *data);
|
||||||
@ -1345,7 +1345,7 @@ int qlcnic_rom_fast_read_words(struct qlcnic_adapter *adapter, int addr,
|
|||||||
int qlcnic_alloc_sw_resources(struct qlcnic_adapter *adapter);
|
int qlcnic_alloc_sw_resources(struct qlcnic_adapter *adapter);
|
||||||
void qlcnic_free_sw_resources(struct qlcnic_adapter *adapter);
|
void qlcnic_free_sw_resources(struct qlcnic_adapter *adapter);
|
||||||
|
|
||||||
void __iomem *qlcnic_get_ioaddr(struct qlcnic_adapter *, u32);
|
void __iomem *qlcnic_get_ioaddr(struct qlcnic_hardware_context *, u32);
|
||||||
|
|
||||||
int qlcnic_alloc_hw_resources(struct qlcnic_adapter *adapter);
|
int qlcnic_alloc_hw_resources(struct qlcnic_adapter *adapter);
|
||||||
void qlcnic_free_hw_resources(struct qlcnic_adapter *adapter);
|
void qlcnic_free_hw_resources(struct qlcnic_adapter *adapter);
|
||||||
|
@ -816,55 +816,63 @@ enum {
|
|||||||
#define LSD(x) ((uint32_t)((uint64_t)(x)))
|
#define LSD(x) ((uint32_t)((uint64_t)(x)))
|
||||||
#define MSD(x) ((uint32_t)((((uint64_t)(x)) >> 16) >> 16))
|
#define MSD(x) ((uint32_t)((((uint64_t)(x)) >> 16) >> 16))
|
||||||
|
|
||||||
|
#define QLCNIC_MS_CTRL 0x41000090
|
||||||
|
#define QLCNIC_MS_ADDR_LO 0x41000094
|
||||||
|
#define QLCNIC_MS_ADDR_HI 0x41000098
|
||||||
|
#define QLCNIC_MS_WRTDATA_LO 0x410000A0
|
||||||
|
#define QLCNIC_MS_WRTDATA_HI 0x410000A4
|
||||||
|
#define QLCNIC_MS_WRTDATA_ULO 0x410000B0
|
||||||
|
#define QLCNIC_MS_WRTDATA_UHI 0x410000B4
|
||||||
|
#define QLCNIC_MS_RDDATA_LO 0x410000A8
|
||||||
|
#define QLCNIC_MS_RDDATA_HI 0x410000AC
|
||||||
|
#define QLCNIC_MS_RDDATA_ULO 0x410000B8
|
||||||
|
#define QLCNIC_MS_RDDATA_UHI 0x410000BC
|
||||||
|
|
||||||
|
#define QLCNIC_TA_WRITE_ENABLE (TA_CTL_ENABLE | TA_CTL_WRITE)
|
||||||
|
#define QLCNIC_TA_WRITE_START (TA_CTL_START | TA_CTL_ENABLE | TA_CTL_WRITE)
|
||||||
|
#define QLCNIC_TA_START_ENABLE (TA_CTL_START | TA_CTL_ENABLE)
|
||||||
|
|
||||||
#define QLCNIC_LEGACY_INTR_CONFIG \
|
#define QLCNIC_LEGACY_INTR_CONFIG \
|
||||||
{ \
|
{ \
|
||||||
{ \
|
{ \
|
||||||
.int_vec_bit = PCIX_INT_VECTOR_BIT_F0, \
|
.int_vec_bit = PCIX_INT_VECTOR_BIT_F0, \
|
||||||
.tgt_status_reg = ISR_INT_TARGET_STATUS, \
|
.tgt_status_reg = ISR_INT_TARGET_STATUS, \
|
||||||
.tgt_mask_reg = ISR_INT_TARGET_MASK, \
|
.tgt_mask_reg = ISR_INT_TARGET_MASK, }, \
|
||||||
.pci_int_reg = ISR_MSI_INT_TRIGGER(0) }, \
|
|
||||||
\
|
\
|
||||||
{ \
|
{ \
|
||||||
.int_vec_bit = PCIX_INT_VECTOR_BIT_F1, \
|
.int_vec_bit = PCIX_INT_VECTOR_BIT_F1, \
|
||||||
.tgt_status_reg = ISR_INT_TARGET_STATUS_F1, \
|
.tgt_status_reg = ISR_INT_TARGET_STATUS_F1, \
|
||||||
.tgt_mask_reg = ISR_INT_TARGET_MASK_F1, \
|
.tgt_mask_reg = ISR_INT_TARGET_MASK_F1, }, \
|
||||||
.pci_int_reg = ISR_MSI_INT_TRIGGER(1) }, \
|
|
||||||
\
|
\
|
||||||
{ \
|
{ \
|
||||||
.int_vec_bit = PCIX_INT_VECTOR_BIT_F2, \
|
.int_vec_bit = PCIX_INT_VECTOR_BIT_F2, \
|
||||||
.tgt_status_reg = ISR_INT_TARGET_STATUS_F2, \
|
.tgt_status_reg = ISR_INT_TARGET_STATUS_F2, \
|
||||||
.tgt_mask_reg = ISR_INT_TARGET_MASK_F2, \
|
.tgt_mask_reg = ISR_INT_TARGET_MASK_F2, }, \
|
||||||
.pci_int_reg = ISR_MSI_INT_TRIGGER(2) }, \
|
|
||||||
\
|
\
|
||||||
{ \
|
{ \
|
||||||
.int_vec_bit = PCIX_INT_VECTOR_BIT_F3, \
|
.int_vec_bit = PCIX_INT_VECTOR_BIT_F3, \
|
||||||
.tgt_status_reg = ISR_INT_TARGET_STATUS_F3, \
|
.tgt_status_reg = ISR_INT_TARGET_STATUS_F3, \
|
||||||
.tgt_mask_reg = ISR_INT_TARGET_MASK_F3, \
|
.tgt_mask_reg = ISR_INT_TARGET_MASK_F3, }, \
|
||||||
.pci_int_reg = ISR_MSI_INT_TRIGGER(3) }, \
|
|
||||||
\
|
\
|
||||||
{ \
|
{ \
|
||||||
.int_vec_bit = PCIX_INT_VECTOR_BIT_F4, \
|
.int_vec_bit = PCIX_INT_VECTOR_BIT_F4, \
|
||||||
.tgt_status_reg = ISR_INT_TARGET_STATUS_F4, \
|
.tgt_status_reg = ISR_INT_TARGET_STATUS_F4, \
|
||||||
.tgt_mask_reg = ISR_INT_TARGET_MASK_F4, \
|
.tgt_mask_reg = ISR_INT_TARGET_MASK_F4, }, \
|
||||||
.pci_int_reg = ISR_MSI_INT_TRIGGER(4) }, \
|
|
||||||
\
|
\
|
||||||
{ \
|
{ \
|
||||||
.int_vec_bit = PCIX_INT_VECTOR_BIT_F5, \
|
.int_vec_bit = PCIX_INT_VECTOR_BIT_F5, \
|
||||||
.tgt_status_reg = ISR_INT_TARGET_STATUS_F5, \
|
.tgt_status_reg = ISR_INT_TARGET_STATUS_F5, \
|
||||||
.tgt_mask_reg = ISR_INT_TARGET_MASK_F5, \
|
.tgt_mask_reg = ISR_INT_TARGET_MASK_F5, }, \
|
||||||
.pci_int_reg = ISR_MSI_INT_TRIGGER(5) }, \
|
|
||||||
\
|
\
|
||||||
{ \
|
{ \
|
||||||
.int_vec_bit = PCIX_INT_VECTOR_BIT_F6, \
|
.int_vec_bit = PCIX_INT_VECTOR_BIT_F6, \
|
||||||
.tgt_status_reg = ISR_INT_TARGET_STATUS_F6, \
|
.tgt_status_reg = ISR_INT_TARGET_STATUS_F6, \
|
||||||
.tgt_mask_reg = ISR_INT_TARGET_MASK_F6, \
|
.tgt_mask_reg = ISR_INT_TARGET_MASK_F6, }, \
|
||||||
.pci_int_reg = ISR_MSI_INT_TRIGGER(6) }, \
|
|
||||||
\
|
\
|
||||||
{ \
|
{ \
|
||||||
.int_vec_bit = PCIX_INT_VECTOR_BIT_F7, \
|
.int_vec_bit = PCIX_INT_VECTOR_BIT_F7, \
|
||||||
.tgt_status_reg = ISR_INT_TARGET_STATUS_F7, \
|
.tgt_status_reg = ISR_INT_TARGET_STATUS_F7, \
|
||||||
.tgt_mask_reg = ISR_INT_TARGET_MASK_F7, \
|
.tgt_mask_reg = ISR_INT_TARGET_MASK_F7, }, \
|
||||||
.pci_int_reg = ISR_MSI_INT_TRIGGER(7) }, \
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/* NIU REGS */
|
/* NIU REGS */
|
||||||
|
@ -6,6 +6,7 @@
|
|||||||
*/
|
*/
|
||||||
|
|
||||||
#include "qlcnic.h"
|
#include "qlcnic.h"
|
||||||
|
#include "qlcnic_hdr.h"
|
||||||
|
|
||||||
#include <linux/slab.h>
|
#include <linux/slab.h>
|
||||||
#include <net/ip.h>
|
#include <net/ip.h>
|
||||||
@ -22,6 +23,15 @@
|
|||||||
#define CRB_HI(off) ((crb_hub_agt[CRB_BLK(off)] << 20) | ((off) & 0xf0000))
|
#define CRB_HI(off) ((crb_hub_agt[CRB_BLK(off)] << 20) | ((off) & 0xf0000))
|
||||||
#define CRB_INDIRECT_2M (0x1e0000UL)
|
#define CRB_INDIRECT_2M (0x1e0000UL)
|
||||||
|
|
||||||
|
struct qlcnic_ms_reg_ctrl {
|
||||||
|
u32 ocm_window;
|
||||||
|
u32 control;
|
||||||
|
u32 hi;
|
||||||
|
u32 low;
|
||||||
|
u32 rd[4];
|
||||||
|
u32 wd[4];
|
||||||
|
u64 off;
|
||||||
|
};
|
||||||
|
|
||||||
#ifndef readq
|
#ifndef readq
|
||||||
static inline u64 readq(void __iomem *addr)
|
static inline u64 readq(void __iomem *addr)
|
||||||
@ -266,10 +276,44 @@ static const unsigned crb_hub_agt[64] = {
|
|||||||
0,
|
0,
|
||||||
};
|
};
|
||||||
|
|
||||||
|
static const u32 msi_tgt_status[8] = {
|
||||||
|
ISR_INT_TARGET_STATUS, ISR_INT_TARGET_STATUS_F1,
|
||||||
|
ISR_INT_TARGET_STATUS_F2, ISR_INT_TARGET_STATUS_F3,
|
||||||
|
ISR_INT_TARGET_STATUS_F4, ISR_INT_TARGET_STATUS_F5,
|
||||||
|
ISR_INT_TARGET_STATUS_F6, ISR_INT_TARGET_STATUS_F7
|
||||||
|
};
|
||||||
|
|
||||||
/* PCI Windowing for DDR regions. */
|
/* PCI Windowing for DDR regions. */
|
||||||
|
|
||||||
#define QLCNIC_PCIE_SEM_TIMEOUT 10000
|
#define QLCNIC_PCIE_SEM_TIMEOUT 10000
|
||||||
|
|
||||||
|
static void qlcnic_read_window_reg(u32 addr, void __iomem *bar0, u32 *data)
|
||||||
|
{
|
||||||
|
u32 dest;
|
||||||
|
void __iomem *val;
|
||||||
|
|
||||||
|
dest = addr & 0xFFFF0000;
|
||||||
|
val = bar0 + QLCNIC_FW_DUMP_REG1;
|
||||||
|
writel(dest, val);
|
||||||
|
readl(val);
|
||||||
|
val = bar0 + QLCNIC_FW_DUMP_REG2 + LSW(addr);
|
||||||
|
*data = readl(val);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void qlcnic_write_window_reg(u32 addr, void __iomem *bar0, u32 data)
|
||||||
|
{
|
||||||
|
u32 dest;
|
||||||
|
void __iomem *val;
|
||||||
|
|
||||||
|
dest = addr & 0xFFFF0000;
|
||||||
|
val = bar0 + QLCNIC_FW_DUMP_REG1;
|
||||||
|
writel(dest, val);
|
||||||
|
readl(val);
|
||||||
|
val = bar0 + QLCNIC_FW_DUMP_REG2 + LSW(addr);
|
||||||
|
writel(data, val);
|
||||||
|
readl(val);
|
||||||
|
}
|
||||||
|
|
||||||
int
|
int
|
||||||
qlcnic_pcie_sem_lock(struct qlcnic_adapter *adapter, int sem, u32 id_reg)
|
qlcnic_pcie_sem_lock(struct qlcnic_adapter *adapter, int sem, u32 id_reg)
|
||||||
{
|
{
|
||||||
@ -300,6 +344,23 @@ qlcnic_pcie_sem_unlock(struct qlcnic_adapter *adapter, int sem)
|
|||||||
QLCRD32(adapter, QLCNIC_PCIE_REG(PCIE_SEM_UNLOCK(sem)));
|
QLCRD32(adapter, QLCNIC_PCIE_REG(PCIE_SEM_UNLOCK(sem)));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static int qlcnic_ind_rd(struct qlcnic_adapter *adapter, u32 addr)
|
||||||
|
{
|
||||||
|
u32 data;
|
||||||
|
|
||||||
|
if (qlcnic_82xx_check(adapter))
|
||||||
|
qlcnic_read_window_reg(addr, adapter->ahw->pci_base0, &data);
|
||||||
|
else
|
||||||
|
return -EIO;
|
||||||
|
return data;
|
||||||
|
}
|
||||||
|
|
||||||
|
static void qlcnic_ind_wr(struct qlcnic_adapter *adapter, u32 addr, u32 data)
|
||||||
|
{
|
||||||
|
if (qlcnic_82xx_check(adapter))
|
||||||
|
qlcnic_write_window_reg(addr, adapter->ahw->pci_base0, data);
|
||||||
|
}
|
||||||
|
|
||||||
static int
|
static int
|
||||||
qlcnic_send_cmd_descs(struct qlcnic_adapter *adapter,
|
qlcnic_send_cmd_descs(struct qlcnic_adapter *adapter,
|
||||||
struct cmd_desc_type0 *cmd_desc_arr, int nr_desc)
|
struct cmd_desc_type0 *cmd_desc_arr, int nr_desc)
|
||||||
@ -863,8 +924,7 @@ int qlcnic_set_features(struct net_device *netdev, netdev_features_t features)
|
|||||||
* 0 if no window access is needed. 'off' is set to 2M addr
|
* 0 if no window access is needed. 'off' is set to 2M addr
|
||||||
* In: 'off' is offset from base in 128M pci map
|
* In: 'off' is offset from base in 128M pci map
|
||||||
*/
|
*/
|
||||||
static int
|
static int qlcnic_pci_get_crb_addr_2M(struct qlcnic_hardware_context *ahw,
|
||||||
qlcnic_pci_get_crb_addr_2M(struct qlcnic_adapter *adapter,
|
|
||||||
ulong off, void __iomem **addr)
|
ulong off, void __iomem **addr)
|
||||||
{
|
{
|
||||||
const struct crb_128M_2M_sub_block_map *m;
|
const struct crb_128M_2M_sub_block_map *m;
|
||||||
@ -880,7 +940,7 @@ qlcnic_pci_get_crb_addr_2M(struct qlcnic_adapter *adapter,
|
|||||||
m = &crb_128M_2M_map[CRB_BLK(off)].sub_block[CRB_SUBBLK(off)];
|
m = &crb_128M_2M_map[CRB_BLK(off)].sub_block[CRB_SUBBLK(off)];
|
||||||
|
|
||||||
if (m->valid && (m->start_128M <= off) && (m->end_128M > off)) {
|
if (m->valid && (m->start_128M <= off) && (m->end_128M > off)) {
|
||||||
*addr = adapter->ahw->pci_base0 + m->start_2M +
|
*addr = ahw->pci_base0 + m->start_2M +
|
||||||
(off - m->start_128M);
|
(off - m->start_128M);
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
@ -888,7 +948,7 @@ qlcnic_pci_get_crb_addr_2M(struct qlcnic_adapter *adapter,
|
|||||||
/*
|
/*
|
||||||
* Not in direct map, use crb window
|
* Not in direct map, use crb window
|
||||||
*/
|
*/
|
||||||
*addr = adapter->ahw->pci_base0 + CRB_INDIRECT_2M + (off & MASK(16));
|
*addr = ahw->pci_base0 + CRB_INDIRECT_2M + (off & MASK(16));
|
||||||
return 1;
|
return 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -929,7 +989,7 @@ qlcnic_hw_write_wx_2M(struct qlcnic_adapter *adapter, ulong off, u32 data)
|
|||||||
int rv;
|
int rv;
|
||||||
void __iomem *addr = NULL;
|
void __iomem *addr = NULL;
|
||||||
|
|
||||||
rv = qlcnic_pci_get_crb_addr_2M(adapter, off, &addr);
|
rv = qlcnic_pci_get_crb_addr_2M(adapter->ahw, off, &addr);
|
||||||
|
|
||||||
if (rv == 0) {
|
if (rv == 0) {
|
||||||
writel(data, addr);
|
writel(data, addr);
|
||||||
@ -954,15 +1014,14 @@ qlcnic_hw_write_wx_2M(struct qlcnic_adapter *adapter, ulong off, u32 data)
|
|||||||
return -EIO;
|
return -EIO;
|
||||||
}
|
}
|
||||||
|
|
||||||
u32
|
int qlcnic_hw_read_wx_2M(struct qlcnic_adapter *adapter, ulong off)
|
||||||
qlcnic_hw_read_wx_2M(struct qlcnic_adapter *adapter, ulong off)
|
|
||||||
{
|
{
|
||||||
unsigned long flags;
|
unsigned long flags;
|
||||||
int rv;
|
int rv;
|
||||||
u32 data = -1;
|
u32 data = -1;
|
||||||
void __iomem *addr = NULL;
|
void __iomem *addr = NULL;
|
||||||
|
|
||||||
rv = qlcnic_pci_get_crb_addr_2M(adapter, off, &addr);
|
rv = qlcnic_pci_get_crb_addr_2M(adapter->ahw, off, &addr);
|
||||||
|
|
||||||
if (rv == 0)
|
if (rv == 0)
|
||||||
return readl(addr);
|
return readl(addr);
|
||||||
@ -985,46 +1044,28 @@ qlcnic_hw_read_wx_2M(struct qlcnic_adapter *adapter, ulong off)
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
void __iomem *
|
void __iomem *qlcnic_get_ioaddr(struct qlcnic_hardware_context *ahw,
|
||||||
qlcnic_get_ioaddr(struct qlcnic_adapter *adapter, u32 offset)
|
u32 offset)
|
||||||
{
|
{
|
||||||
void __iomem *addr = NULL;
|
void __iomem *addr = NULL;
|
||||||
|
|
||||||
WARN_ON(qlcnic_pci_get_crb_addr_2M(adapter, offset, &addr));
|
WARN_ON(qlcnic_pci_get_crb_addr_2M(ahw, offset, &addr));
|
||||||
|
|
||||||
return addr;
|
return addr;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static int qlcnic_pci_mem_access_direct(struct qlcnic_adapter *adapter,
|
||||||
static int
|
u32 window, u64 off, u64 *data, int op)
|
||||||
qlcnic_pci_set_window_2M(struct qlcnic_adapter *adapter,
|
|
||||||
u64 addr, u32 *start)
|
|
||||||
{
|
|
||||||
u32 window;
|
|
||||||
|
|
||||||
window = OCM_WIN_P3P(addr);
|
|
||||||
|
|
||||||
writel(window, adapter->ahw->ocm_win_crb);
|
|
||||||
/* read back to flush */
|
|
||||||
readl(adapter->ahw->ocm_win_crb);
|
|
||||||
|
|
||||||
*start = QLCNIC_PCI_OCM0_2M + GET_MEM_OFFS_2M(addr);
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
static int
|
|
||||||
qlcnic_pci_mem_access_direct(struct qlcnic_adapter *adapter, u64 off,
|
|
||||||
u64 *data, int op)
|
|
||||||
{
|
{
|
||||||
void __iomem *addr;
|
void __iomem *addr;
|
||||||
int ret;
|
|
||||||
u32 start;
|
u32 start;
|
||||||
|
|
||||||
mutex_lock(&adapter->ahw->mem_lock);
|
mutex_lock(&adapter->ahw->mem_lock);
|
||||||
|
|
||||||
ret = qlcnic_pci_set_window_2M(adapter, off, &start);
|
writel(window, adapter->ahw->ocm_win_crb);
|
||||||
if (ret != 0)
|
/* read back to flush */
|
||||||
goto unlock;
|
readl(adapter->ahw->ocm_win_crb);
|
||||||
|
start = QLCNIC_PCI_OCM0_2M + off;
|
||||||
|
|
||||||
addr = adapter->ahw->pci_base0 + start;
|
addr = adapter->ahw->pci_base0 + start;
|
||||||
|
|
||||||
@ -1033,10 +1074,12 @@ qlcnic_pci_mem_access_direct(struct qlcnic_adapter *adapter, u64 off,
|
|||||||
else /* write */
|
else /* write */
|
||||||
writeq(*data, addr);
|
writeq(*data, addr);
|
||||||
|
|
||||||
unlock:
|
/* Set window to 0 */
|
||||||
mutex_unlock(&adapter->ahw->mem_lock);
|
writel(0, adapter->ahw->ocm_win_crb);
|
||||||
|
readl(adapter->ahw->ocm_win_crb);
|
||||||
|
|
||||||
return ret;
|
mutex_unlock(&adapter->ahw->mem_lock);
|
||||||
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
void
|
void
|
||||||
@ -1061,52 +1104,74 @@ qlcnic_pci_camqm_write_2M(struct qlcnic_adapter *adapter, u64 off, u64 data)
|
|||||||
mutex_unlock(&adapter->ahw->mem_lock);
|
mutex_unlock(&adapter->ahw->mem_lock);
|
||||||
}
|
}
|
||||||
|
|
||||||
int
|
|
||||||
qlcnic_pci_mem_write_2M(struct qlcnic_adapter *adapter,
|
|
||||||
u64 off, u64 data)
|
/* Set MS memory control data for different adapters */
|
||||||
|
static void qlcnic_set_ms_controls(struct qlcnic_adapter *adapter, u64 off,
|
||||||
|
struct qlcnic_ms_reg_ctrl *ms)
|
||||||
{
|
{
|
||||||
int i, j, ret;
|
ms->control = QLCNIC_MS_CTRL;
|
||||||
|
ms->low = QLCNIC_MS_ADDR_LO;
|
||||||
|
ms->hi = QLCNIC_MS_ADDR_HI;
|
||||||
|
if (off & 0xf) {
|
||||||
|
ms->wd[0] = QLCNIC_MS_WRTDATA_LO;
|
||||||
|
ms->rd[0] = QLCNIC_MS_RDDATA_LO;
|
||||||
|
ms->wd[1] = QLCNIC_MS_WRTDATA_HI;
|
||||||
|
ms->rd[1] = QLCNIC_MS_RDDATA_HI;
|
||||||
|
ms->wd[2] = QLCNIC_MS_WRTDATA_ULO;
|
||||||
|
ms->wd[3] = QLCNIC_MS_WRTDATA_UHI;
|
||||||
|
ms->rd[2] = QLCNIC_MS_RDDATA_ULO;
|
||||||
|
ms->rd[3] = QLCNIC_MS_RDDATA_UHI;
|
||||||
|
} else {
|
||||||
|
ms->wd[0] = QLCNIC_MS_WRTDATA_ULO;
|
||||||
|
ms->rd[0] = QLCNIC_MS_RDDATA_ULO;
|
||||||
|
ms->wd[1] = QLCNIC_MS_WRTDATA_UHI;
|
||||||
|
ms->rd[1] = QLCNIC_MS_RDDATA_UHI;
|
||||||
|
ms->wd[2] = QLCNIC_MS_WRTDATA_LO;
|
||||||
|
ms->wd[3] = QLCNIC_MS_WRTDATA_HI;
|
||||||
|
ms->rd[2] = QLCNIC_MS_RDDATA_LO;
|
||||||
|
ms->rd[3] = QLCNIC_MS_RDDATA_HI;
|
||||||
|
}
|
||||||
|
|
||||||
|
ms->ocm_window = OCM_WIN_P3P(off);
|
||||||
|
ms->off = GET_MEM_OFFS_2M(off);
|
||||||
|
}
|
||||||
|
|
||||||
|
int qlcnic_pci_mem_write_2M(struct qlcnic_adapter *adapter, u64 off, u64 data)
|
||||||
|
{
|
||||||
|
int j, ret = 0;
|
||||||
u32 temp, off8;
|
u32 temp, off8;
|
||||||
void __iomem *mem_crb;
|
struct qlcnic_ms_reg_ctrl ms;
|
||||||
|
|
||||||
/* Only 64-bit aligned access */
|
/* Only 64-bit aligned access */
|
||||||
if (off & 7)
|
if (off & 7)
|
||||||
return -EIO;
|
return -EIO;
|
||||||
|
|
||||||
/* P3 onward, test agent base for MIU and SIU is same */
|
memset(&ms, 0, sizeof(struct qlcnic_ms_reg_ctrl));
|
||||||
if (ADDR_IN_RANGE(off, QLCNIC_ADDR_QDR_NET,
|
if (!(ADDR_IN_RANGE(off, QLCNIC_ADDR_QDR_NET,
|
||||||
QLCNIC_ADDR_QDR_NET_MAX)) {
|
QLCNIC_ADDR_QDR_NET_MAX) ||
|
||||||
mem_crb = qlcnic_get_ioaddr(adapter,
|
ADDR_IN_RANGE(off, QLCNIC_ADDR_DDR_NET,
|
||||||
QLCNIC_CRB_QDR_NET+MIU_TEST_AGT_BASE);
|
QLCNIC_ADDR_DDR_NET_MAX)))
|
||||||
goto correct;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (ADDR_IN_RANGE(off, QLCNIC_ADDR_DDR_NET, QLCNIC_ADDR_DDR_NET_MAX)) {
|
|
||||||
mem_crb = qlcnic_get_ioaddr(adapter,
|
|
||||||
QLCNIC_CRB_DDR_NET+MIU_TEST_AGT_BASE);
|
|
||||||
goto correct;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (ADDR_IN_RANGE(off, QLCNIC_ADDR_OCM0, QLCNIC_ADDR_OCM0_MAX))
|
|
||||||
return qlcnic_pci_mem_access_direct(adapter, off, &data, 1);
|
|
||||||
|
|
||||||
return -EIO;
|
return -EIO;
|
||||||
|
|
||||||
correct:
|
qlcnic_set_ms_controls(adapter, off, &ms);
|
||||||
|
|
||||||
|
if (ADDR_IN_RANGE(off, QLCNIC_ADDR_OCM0, QLCNIC_ADDR_OCM0_MAX))
|
||||||
|
return qlcnic_pci_mem_access_direct(adapter, ms.ocm_window,
|
||||||
|
ms.off, &data, 1);
|
||||||
|
|
||||||
off8 = off & ~0xf;
|
off8 = off & ~0xf;
|
||||||
|
|
||||||
mutex_lock(&adapter->ahw->mem_lock);
|
mutex_lock(&adapter->ahw->mem_lock);
|
||||||
|
|
||||||
writel(off8, (mem_crb + MIU_TEST_AGT_ADDR_LO));
|
qlcnic_ind_wr(adapter, ms.low, off8);
|
||||||
writel(0, (mem_crb + MIU_TEST_AGT_ADDR_HI));
|
qlcnic_ind_wr(adapter, ms.hi, 0);
|
||||||
|
|
||||||
i = 0;
|
qlcnic_ind_wr(adapter, ms.control, TA_CTL_ENABLE);
|
||||||
writel(TA_CTL_ENABLE, (mem_crb + TEST_AGT_CTRL));
|
qlcnic_ind_wr(adapter, ms.control, QLCNIC_TA_START_ENABLE);
|
||||||
writel((TA_CTL_START | TA_CTL_ENABLE),
|
|
||||||
(mem_crb + TEST_AGT_CTRL));
|
|
||||||
|
|
||||||
for (j = 0; j < MAX_CTL_CHECK; j++) {
|
for (j = 0; j < MAX_CTL_CHECK; j++) {
|
||||||
temp = readl(mem_crb + TEST_AGT_CTRL);
|
temp = qlcnic_ind_rd(adapter, ms.control);
|
||||||
if ((temp & TA_CTL_BUSY) == 0)
|
if ((temp & TA_CTL_BUSY) == 0)
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
@ -1116,24 +1181,18 @@ correct:
|
|||||||
goto done;
|
goto done;
|
||||||
}
|
}
|
||||||
|
|
||||||
i = (off & 0xf) ? 0 : 2;
|
/* This is the modify part of read-modify-write */
|
||||||
writel(readl(mem_crb + MIU_TEST_AGT_RDDATA(i)),
|
qlcnic_ind_wr(adapter, ms.wd[0], qlcnic_ind_rd(adapter, ms.rd[0]));
|
||||||
mem_crb + MIU_TEST_AGT_WRDATA(i));
|
qlcnic_ind_wr(adapter, ms.wd[1], qlcnic_ind_rd(adapter, ms.rd[1]));
|
||||||
writel(readl(mem_crb + MIU_TEST_AGT_RDDATA(i+1)),
|
/* This is the write part of read-modify-write */
|
||||||
mem_crb + MIU_TEST_AGT_WRDATA(i+1));
|
qlcnic_ind_wr(adapter, ms.wd[2], data & 0xffffffff);
|
||||||
i = (off & 0xf) ? 2 : 0;
|
qlcnic_ind_wr(adapter, ms.wd[3], (data >> 32) & 0xffffffff);
|
||||||
|
|
||||||
writel(data & 0xffffffff,
|
qlcnic_ind_wr(adapter, ms.control, QLCNIC_TA_WRITE_ENABLE);
|
||||||
mem_crb + MIU_TEST_AGT_WRDATA(i));
|
qlcnic_ind_wr(adapter, ms.control, QLCNIC_TA_WRITE_START);
|
||||||
writel((data >> 32) & 0xffffffff,
|
|
||||||
mem_crb + MIU_TEST_AGT_WRDATA(i+1));
|
|
||||||
|
|
||||||
writel((TA_CTL_ENABLE | TA_CTL_WRITE), (mem_crb + TEST_AGT_CTRL));
|
|
||||||
writel((TA_CTL_START | TA_CTL_ENABLE | TA_CTL_WRITE),
|
|
||||||
(mem_crb + TEST_AGT_CTRL));
|
|
||||||
|
|
||||||
for (j = 0; j < MAX_CTL_CHECK; j++) {
|
for (j = 0; j < MAX_CTL_CHECK; j++) {
|
||||||
temp = readl(mem_crb + TEST_AGT_CTRL);
|
temp = qlcnic_ind_rd(adapter, ms.control);
|
||||||
if ((temp & TA_CTL_BUSY) == 0)
|
if ((temp & TA_CTL_BUSY) == 0)
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
@ -1152,52 +1211,41 @@ done:
|
|||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
int
|
int qlcnic_pci_mem_read_2M(struct qlcnic_adapter *adapter, u64 off, u64 *data)
|
||||||
qlcnic_pci_mem_read_2M(struct qlcnic_adapter *adapter,
|
|
||||||
u64 off, u64 *data)
|
|
||||||
{
|
{
|
||||||
int j, ret;
|
int j, ret;
|
||||||
u32 temp, off8;
|
u32 temp, off8;
|
||||||
u64 val;
|
u64 val;
|
||||||
void __iomem *mem_crb;
|
struct qlcnic_ms_reg_ctrl ms;
|
||||||
|
|
||||||
/* Only 64-bit aligned access */
|
/* Only 64-bit aligned access */
|
||||||
if (off & 7)
|
if (off & 7)
|
||||||
return -EIO;
|
return -EIO;
|
||||||
|
if (!(ADDR_IN_RANGE(off, QLCNIC_ADDR_QDR_NET,
|
||||||
/* P3 onward, test agent base for MIU and SIU is same */
|
QLCNIC_ADDR_QDR_NET_MAX) ||
|
||||||
if (ADDR_IN_RANGE(off, QLCNIC_ADDR_QDR_NET,
|
ADDR_IN_RANGE(off, QLCNIC_ADDR_DDR_NET,
|
||||||
QLCNIC_ADDR_QDR_NET_MAX)) {
|
QLCNIC_ADDR_DDR_NET_MAX)))
|
||||||
mem_crb = qlcnic_get_ioaddr(adapter,
|
|
||||||
QLCNIC_CRB_QDR_NET+MIU_TEST_AGT_BASE);
|
|
||||||
goto correct;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (ADDR_IN_RANGE(off, QLCNIC_ADDR_DDR_NET, QLCNIC_ADDR_DDR_NET_MAX)) {
|
|
||||||
mem_crb = qlcnic_get_ioaddr(adapter,
|
|
||||||
QLCNIC_CRB_DDR_NET+MIU_TEST_AGT_BASE);
|
|
||||||
goto correct;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (ADDR_IN_RANGE(off, QLCNIC_ADDR_OCM0, QLCNIC_ADDR_OCM0_MAX)) {
|
|
||||||
return qlcnic_pci_mem_access_direct(adapter,
|
|
||||||
off, data, 0);
|
|
||||||
}
|
|
||||||
|
|
||||||
return -EIO;
|
return -EIO;
|
||||||
|
|
||||||
correct:
|
memset(&ms, 0, sizeof(struct qlcnic_ms_reg_ctrl));
|
||||||
off8 = off & ~0xf;
|
qlcnic_set_ms_controls(adapter, off, &ms);
|
||||||
|
|
||||||
|
if (ADDR_IN_RANGE(off, QLCNIC_ADDR_OCM0, QLCNIC_ADDR_OCM0_MAX))
|
||||||
|
return qlcnic_pci_mem_access_direct(adapter, ms.ocm_window,
|
||||||
|
ms.off, data, 0);
|
||||||
|
|
||||||
mutex_lock(&adapter->ahw->mem_lock);
|
mutex_lock(&adapter->ahw->mem_lock);
|
||||||
|
|
||||||
writel(off8, (mem_crb + MIU_TEST_AGT_ADDR_LO));
|
off8 = off & ~0xf;
|
||||||
writel(0, (mem_crb + MIU_TEST_AGT_ADDR_HI));
|
|
||||||
writel(TA_CTL_ENABLE, (mem_crb + TEST_AGT_CTRL));
|
qlcnic_ind_wr(adapter, ms.low, off8);
|
||||||
writel((TA_CTL_START | TA_CTL_ENABLE), (mem_crb + TEST_AGT_CTRL));
|
qlcnic_ind_wr(adapter, ms.hi, 0);
|
||||||
|
|
||||||
|
qlcnic_ind_wr(adapter, ms.control, TA_CTL_ENABLE);
|
||||||
|
qlcnic_ind_wr(adapter, ms.control, QLCNIC_TA_START_ENABLE);
|
||||||
|
|
||||||
for (j = 0; j < MAX_CTL_CHECK; j++) {
|
for (j = 0; j < MAX_CTL_CHECK; j++) {
|
||||||
temp = readl(mem_crb + TEST_AGT_CTRL);
|
temp = qlcnic_ind_rd(adapter, ms.control);
|
||||||
if ((temp & TA_CTL_BUSY) == 0)
|
if ((temp & TA_CTL_BUSY) == 0)
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
@ -1208,13 +1256,10 @@ correct:
|
|||||||
"failed to read through agent\n");
|
"failed to read through agent\n");
|
||||||
ret = -EIO;
|
ret = -EIO;
|
||||||
} else {
|
} else {
|
||||||
off8 = MIU_TEST_AGT_RDDATA_LO;
|
|
||||||
if (off & 0xf)
|
|
||||||
off8 = MIU_TEST_AGT_RDDATA_UPPER_LO;
|
|
||||||
|
|
||||||
temp = readl(mem_crb + off8 + 4);
|
temp = qlcnic_ind_rd(adapter, ms.rd[3]);
|
||||||
val = (u64)temp << 32;
|
val = (u64)temp << 32;
|
||||||
val |= readl(mem_crb + off8);
|
val |= qlcnic_ind_rd(adapter, ms.rd[2]);
|
||||||
*data = val;
|
*data = val;
|
||||||
ret = 0;
|
ret = 0;
|
||||||
}
|
}
|
||||||
|
@ -283,32 +283,31 @@ static int qlcnic_enable_msix(struct qlcnic_adapter *adapter, u32 num_msix)
|
|||||||
return err;
|
return err;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
static void qlcnic_enable_msi_legacy(struct qlcnic_adapter *adapter)
|
static void qlcnic_enable_msi_legacy(struct qlcnic_adapter *adapter)
|
||||||
{
|
{
|
||||||
|
u32 offset, mask_reg;
|
||||||
const struct qlcnic_legacy_intr_set *legacy_intrp;
|
const struct qlcnic_legacy_intr_set *legacy_intrp;
|
||||||
|
struct qlcnic_hardware_context *ahw = adapter->ahw;
|
||||||
struct pci_dev *pdev = adapter->pdev;
|
struct pci_dev *pdev = adapter->pdev;
|
||||||
|
|
||||||
if (use_msi && !pci_enable_msi(pdev)) {
|
if (use_msi && !pci_enable_msi(pdev)) {
|
||||||
adapter->flags |= QLCNIC_MSI_ENABLED;
|
adapter->flags |= QLCNIC_MSI_ENABLED;
|
||||||
adapter->tgt_status_reg = qlcnic_get_ioaddr(adapter,
|
offset = msi_tgt_status[adapter->ahw->pci_func];
|
||||||
msi_tgt_status[adapter->ahw->pci_func]);
|
adapter->tgt_status_reg = qlcnic_get_ioaddr(adapter->ahw,
|
||||||
|
offset);
|
||||||
dev_info(&pdev->dev, "using msi interrupts\n");
|
dev_info(&pdev->dev, "using msi interrupts\n");
|
||||||
adapter->msix_entries[0].vector = pdev->irq;
|
adapter->msix_entries[0].vector = pdev->irq;
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
legacy_intrp = &legacy_intr[adapter->ahw->pci_func];
|
legacy_intrp = &legacy_intr[adapter->ahw->pci_func];
|
||||||
|
|
||||||
adapter->ahw->int_vec_bit = legacy_intrp->int_vec_bit;
|
adapter->ahw->int_vec_bit = legacy_intrp->int_vec_bit;
|
||||||
adapter->tgt_status_reg = qlcnic_get_ioaddr(adapter,
|
offset = legacy_intrp->tgt_status_reg;
|
||||||
legacy_intrp->tgt_status_reg);
|
adapter->tgt_status_reg = qlcnic_get_ioaddr(ahw, offset);
|
||||||
adapter->tgt_mask_reg = qlcnic_get_ioaddr(adapter,
|
mask_reg = legacy_intrp->tgt_mask_reg;
|
||||||
legacy_intrp->tgt_mask_reg);
|
adapter->tgt_mask_reg = qlcnic_get_ioaddr(ahw, mask_reg);
|
||||||
adapter->isr_int_vec = qlcnic_get_ioaddr(adapter, ISR_INT_VECTOR);
|
adapter->isr_int_vec = qlcnic_get_ioaddr(ahw, ISR_INT_VECTOR);
|
||||||
|
adapter->crb_int_state_reg = qlcnic_get_ioaddr(ahw, ISR_INT_STATE_REG);
|
||||||
adapter->crb_int_state_reg = qlcnic_get_ioaddr(adapter,
|
|
||||||
ISR_INT_STATE_REG);
|
|
||||||
dev_info(&pdev->dev, "using legacy interrupts\n");
|
dev_info(&pdev->dev, "using legacy interrupts\n");
|
||||||
adapter->msix_entries[0].vector = pdev->irq;
|
adapter->msix_entries[0].vector = pdev->irq;
|
||||||
}
|
}
|
||||||
@ -480,20 +479,32 @@ qlcnic_check_vf(struct qlcnic_adapter *adapter)
|
|||||||
adapter->nic_ops = &qlcnic_ops;
|
adapter->nic_ops = &qlcnic_ops;
|
||||||
}
|
}
|
||||||
|
|
||||||
static int
|
#define QLCNIC_82XX_BAR0_LENGTH 0x00200000UL
|
||||||
qlcnic_setup_pci_map(struct qlcnic_adapter *adapter)
|
static void qlcnic_get_bar_length(u32 dev_id, ulong *bar)
|
||||||
{
|
{
|
||||||
|
switch (dev_id) {
|
||||||
|
case PCI_DEVICE_ID_QLOGIC_QLE824X:
|
||||||
|
*bar = QLCNIC_82XX_BAR0_LENGTH;
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
*bar = 0;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static int qlcnic_setup_pci_map(struct pci_dev *pdev,
|
||||||
|
struct qlcnic_hardware_context *ahw)
|
||||||
|
{
|
||||||
|
u32 offset;
|
||||||
void __iomem *mem_ptr0 = NULL;
|
void __iomem *mem_ptr0 = NULL;
|
||||||
resource_size_t mem_base;
|
resource_size_t mem_base;
|
||||||
unsigned long mem_len, pci_len0 = 0;
|
unsigned long mem_len, pci_len0 = 0, bar0_len;
|
||||||
|
|
||||||
struct pci_dev *pdev = adapter->pdev;
|
|
||||||
|
|
||||||
/* remap phys address */
|
/* remap phys address */
|
||||||
mem_base = pci_resource_start(pdev, 0); /* 0 is for BAR 0 */
|
mem_base = pci_resource_start(pdev, 0); /* 0 is for BAR 0 */
|
||||||
mem_len = pci_resource_len(pdev, 0);
|
mem_len = pci_resource_len(pdev, 0);
|
||||||
|
|
||||||
if (mem_len == QLCNIC_PCI_2MB_SIZE) {
|
qlcnic_get_bar_length(pdev->device, &bar0_len);
|
||||||
|
if (mem_len >= bar0_len) {
|
||||||
|
|
||||||
mem_ptr0 = pci_ioremap_bar(pdev, 0);
|
mem_ptr0 = pci_ioremap_bar(pdev, 0);
|
||||||
if (mem_ptr0 == NULL) {
|
if (mem_ptr0 == NULL) {
|
||||||
@ -506,15 +517,10 @@ qlcnic_setup_pci_map(struct qlcnic_adapter *adapter)
|
|||||||
}
|
}
|
||||||
|
|
||||||
dev_info(&pdev->dev, "%dMB memory map\n", (int)(mem_len>>20));
|
dev_info(&pdev->dev, "%dMB memory map\n", (int)(mem_len>>20));
|
||||||
|
ahw->pci_base0 = mem_ptr0;
|
||||||
adapter->ahw->pci_base0 = mem_ptr0;
|
ahw->pci_len0 = pci_len0;
|
||||||
adapter->ahw->pci_len0 = pci_len0;
|
offset = QLCNIC_PCIX_PS_REG(PCIX_OCM_WINDOW_REG(ahw->pci_func));
|
||||||
|
qlcnic_get_ioaddr(ahw, offset);
|
||||||
qlcnic_check_vf(adapter);
|
|
||||||
|
|
||||||
adapter->ahw->ocm_win_crb = qlcnic_get_ioaddr(adapter,
|
|
||||||
QLCNIC_PCIX_PS_REG(PCIX_OCM_WINDOW_REG(
|
|
||||||
adapter->ahw->pci_func)));
|
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
@ -1510,9 +1516,10 @@ qlcnic_probe(struct pci_dev *pdev, const struct pci_device_id *ent)
|
|||||||
spin_lock_init(&adapter->tx_clean_lock);
|
spin_lock_init(&adapter->tx_clean_lock);
|
||||||
INIT_LIST_HEAD(&adapter->mac_list);
|
INIT_LIST_HEAD(&adapter->mac_list);
|
||||||
|
|
||||||
err = qlcnic_setup_pci_map(adapter);
|
err = qlcnic_setup_pci_map(pdev, adapter->ahw);
|
||||||
if (err)
|
if (err)
|
||||||
goto err_out_free_hw;
|
goto err_out_free_hw;
|
||||||
|
qlcnic_check_vf(adapter);
|
||||||
|
|
||||||
/* This will be reset for mezz cards */
|
/* This will be reset for mezz cards */
|
||||||
adapter->portnum = adapter->ahw->pci_func;
|
adapter->portnum = adapter->ahw->pci_func;
|
||||||
|
Loading…
Reference in New Issue
Block a user