bnx2x_main.c 318.6 KB
Newer Older
1
/* bnx2x_main.c: Broadcom Everest network driver.
E
Eliezer Tamir 已提交
2
 *
3
 * Copyright (c) 2007-2012 Broadcom Corporation
E
Eliezer Tamir 已提交
4 5 6 7 8
 *
 * This program is free software; you can redistribute it and/or modify
 * it under the terms of the GNU General Public License as published by
 * the Free Software Foundation.
 *
9 10
 * Maintained by: Eilon Greenstein <eilong@broadcom.com>
 * Written by: Eliezer Tamir
E
Eliezer Tamir 已提交
11 12
 * Based on code from Michael Chan's bnx2 driver
 * UDP CSUM errata workaround by Arik Gendelman
E
Eilon Greenstein 已提交
13
 * Slowpath and fastpath rework by Vladislav Zolotarov
E
Eliezer Tamir 已提交
14
 * Statistics and Link management by Yitchak Gertner
E
Eliezer Tamir 已提交
15 16 17
 *
 */

18 19
#define pr_fmt(fmt) KBUILD_MODNAME ": " fmt

E
Eliezer Tamir 已提交
20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41
#include <linux/module.h>
#include <linux/moduleparam.h>
#include <linux/kernel.h>
#include <linux/device.h>  /* for dev_info() */
#include <linux/timer.h>
#include <linux/errno.h>
#include <linux/ioport.h>
#include <linux/slab.h>
#include <linux/interrupt.h>
#include <linux/pci.h>
#include <linux/init.h>
#include <linux/netdevice.h>
#include <linux/etherdevice.h>
#include <linux/skbuff.h>
#include <linux/dma-mapping.h>
#include <linux/bitops.h>
#include <linux/irq.h>
#include <linux/delay.h>
#include <asm/byteorder.h>
#include <linux/time.h>
#include <linux/ethtool.h>
#include <linux/mii.h>
42
#include <linux/if.h>
43
#include <linux/if_vlan.h>
E
Eliezer Tamir 已提交
44
#include <net/ip.h>
45
#include <net/ipv6.h>
E
Eliezer Tamir 已提交
46 47
#include <net/tcp.h>
#include <net/checksum.h>
48
#include <net/ip6_checksum.h>
E
Eliezer Tamir 已提交
49 50
#include <linux/workqueue.h>
#include <linux/crc32.h>
51
#include <linux/crc32c.h>
E
Eliezer Tamir 已提交
52 53 54
#include <linux/prefetch.h>
#include <linux/zlib.h>
#include <linux/io.h>
B
Ben Hutchings 已提交
55
#include <linux/stringify.h>
56
#include <linux/vmalloc.h>
E
Eliezer Tamir 已提交
57 58 59

#include "bnx2x.h"
#include "bnx2x_init.h"
60
#include "bnx2x_init_ops.h"
D
Dmitry Kravkov 已提交
61
#include "bnx2x_cmn.h"
V
Vladislav Zolotarov 已提交
62
#include "bnx2x_dcb.h"
V
Vladislav Zolotarov 已提交
63
#include "bnx2x_sp.h"
E
Eliezer Tamir 已提交
64

65 66 67
#include <linux/firmware.h>
#include "bnx2x_fw_file_hdr.h"
/* FW files */
B
Ben Hutchings 已提交
68 69 70 71 72
#define FW_FILE_VERSION					\
	__stringify(BCM_5710_FW_MAJOR_VERSION) "."	\
	__stringify(BCM_5710_FW_MINOR_VERSION) "."	\
	__stringify(BCM_5710_FW_REVISION_VERSION) "."	\
	__stringify(BCM_5710_FW_ENGINEERING_VERSION)
73 74
#define FW_FILE_NAME_E1		"bnx2x/bnx2x-e1-" FW_FILE_VERSION ".fw"
#define FW_FILE_NAME_E1H	"bnx2x/bnx2x-e1h-" FW_FILE_VERSION ".fw"
D
Dmitry Kravkov 已提交
75
#define FW_FILE_NAME_E2		"bnx2x/bnx2x-e2-" FW_FILE_VERSION ".fw"
76

77 78
/* Time in jiffies before concluding the transmitter is hung */
#define TX_TIMEOUT		(5*HZ)
E
Eliezer Tamir 已提交
79

A
Andrew Morton 已提交
80
static char version[] __devinitdata =
81
	"Broadcom NetXtreme II 5771x/578xx 10/20-Gigabit Ethernet Driver "
E
Eliezer Tamir 已提交
82 83
	DRV_MODULE_NAME " " DRV_MODULE_VERSION " (" DRV_MODULE_RELDATE ")\n";

84
MODULE_AUTHOR("Eliezer Tamir");
D
Dmitry Kravkov 已提交
85
MODULE_DESCRIPTION("Broadcom NetXtreme II "
86 87 88
		   "BCM57710/57711/57711E/"
		   "57712/57712_MF/57800/57800_MF/57810/57810_MF/"
		   "57840/57840_MF Driver");
E
Eliezer Tamir 已提交
89 90
MODULE_LICENSE("GPL");
MODULE_VERSION(DRV_MODULE_VERSION);
B
Ben Hutchings 已提交
91 92
MODULE_FIRMWARE(FW_FILE_NAME_E1);
MODULE_FIRMWARE(FW_FILE_NAME_E1H);
D
Dmitry Kravkov 已提交
93
MODULE_FIRMWARE(FW_FILE_NAME_E2);
E
Eliezer Tamir 已提交
94

E
Eilon Greenstein 已提交
95 96
static int multi_mode = 1;
module_param(multi_mode, int, 0);
E
Eilon Greenstein 已提交
97 98 99
MODULE_PARM_DESC(multi_mode, " Multi queue mode "
			     "(0 Disable; 1 Enable (default))");

100
int num_queues;
101 102 103
module_param(num_queues, int, 0);
MODULE_PARM_DESC(num_queues, " Number of queues for multi_mode=1"
				" (default is as a number of CPUs)");
E
Eilon Greenstein 已提交
104

105 106
static int disable_tpa;
module_param(disable_tpa, int, 0);
107
MODULE_PARM_DESC(disable_tpa, " Disable the TPA (LRO) feature");
E
Eilon Greenstein 已提交
108

D
Dmitry Kravkov 已提交
109 110
#define INT_MODE_INTx			1
#define INT_MODE_MSI			2
E
Eilon Greenstein 已提交
111 112
static int int_mode;
module_param(int_mode, int, 0);
113
MODULE_PARM_DESC(int_mode, " Force interrupt mode other than MSI-X "
V
Vladislav Zolotarov 已提交
114
				"(1 INT#x; 2 MSI)");
E
Eilon Greenstein 已提交
115

116 117 118 119
static int dropless_fc;
module_param(dropless_fc, int, 0);
MODULE_PARM_DESC(dropless_fc, " Pause on exhausted host ring");

120 121 122 123
static int mrrs = -1;
module_param(mrrs, int, 0);
MODULE_PARM_DESC(mrrs, " Force Max Read Req Size (0..3) (for debug)");

124
static int debug;
E
Eliezer Tamir 已提交
125
module_param(debug, int, 0);
126 127
MODULE_PARM_DESC(debug, " Default debug msglevel");

E
Eliezer Tamir 已提交
128

129 130

struct workqueue_struct *bnx2x_wq;
V
Vladislav Zolotarov 已提交
131

E
Eliezer Tamir 已提交
132 133
enum bnx2x_board_type {
	BCM57710 = 0,
134 135 136 137 138 139 140 141 142 143
	BCM57711,
	BCM57711E,
	BCM57712,
	BCM57712_MF,
	BCM57800,
	BCM57800_MF,
	BCM57810,
	BCM57810_MF,
	BCM57840,
	BCM57840_MF
E
Eliezer Tamir 已提交
144 145
};

146
/* indexed by board_type, above */
A
Andrew Morton 已提交
147
static struct {
E
Eliezer Tamir 已提交
148 149
	char *name;
} board_info[] __devinitdata = {
150 151 152 153 154 155 156 157 158 159 160 161
	{ "Broadcom NetXtreme II BCM57710 10 Gigabit PCIe [Everest]" },
	{ "Broadcom NetXtreme II BCM57711 10 Gigabit PCIe" },
	{ "Broadcom NetXtreme II BCM57711E 10 Gigabit PCIe" },
	{ "Broadcom NetXtreme II BCM57712 10 Gigabit Ethernet" },
	{ "Broadcom NetXtreme II BCM57712 10 Gigabit Ethernet Multi Function" },
	{ "Broadcom NetXtreme II BCM57800 10 Gigabit Ethernet" },
	{ "Broadcom NetXtreme II BCM57800 10 Gigabit Ethernet Multi Function" },
	{ "Broadcom NetXtreme II BCM57810 10 Gigabit Ethernet" },
	{ "Broadcom NetXtreme II BCM57810 10 Gigabit Ethernet Multi Function" },
	{ "Broadcom NetXtreme II BCM57840 10/20 Gigabit Ethernet" },
	{ "Broadcom NetXtreme II BCM57840 10/20 Gigabit "
						"Ethernet Multi Function"}
E
Eliezer Tamir 已提交
162 163
};

164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196
#ifndef PCI_DEVICE_ID_NX2_57710
#define PCI_DEVICE_ID_NX2_57710		CHIP_NUM_57710
#endif
#ifndef PCI_DEVICE_ID_NX2_57711
#define PCI_DEVICE_ID_NX2_57711		CHIP_NUM_57711
#endif
#ifndef PCI_DEVICE_ID_NX2_57711E
#define PCI_DEVICE_ID_NX2_57711E	CHIP_NUM_57711E
#endif
#ifndef PCI_DEVICE_ID_NX2_57712
#define PCI_DEVICE_ID_NX2_57712		CHIP_NUM_57712
#endif
#ifndef PCI_DEVICE_ID_NX2_57712_MF
#define PCI_DEVICE_ID_NX2_57712_MF	CHIP_NUM_57712_MF
#endif
#ifndef PCI_DEVICE_ID_NX2_57800
#define PCI_DEVICE_ID_NX2_57800		CHIP_NUM_57800
#endif
#ifndef PCI_DEVICE_ID_NX2_57800_MF
#define PCI_DEVICE_ID_NX2_57800_MF	CHIP_NUM_57800_MF
#endif
#ifndef PCI_DEVICE_ID_NX2_57810
#define PCI_DEVICE_ID_NX2_57810		CHIP_NUM_57810
#endif
#ifndef PCI_DEVICE_ID_NX2_57810_MF
#define PCI_DEVICE_ID_NX2_57810_MF	CHIP_NUM_57810_MF
#endif
#ifndef PCI_DEVICE_ID_NX2_57840
#define PCI_DEVICE_ID_NX2_57840		CHIP_NUM_57840
#endif
#ifndef PCI_DEVICE_ID_NX2_57840_MF
#define PCI_DEVICE_ID_NX2_57840_MF	CHIP_NUM_57840_MF
#endif
197
static DEFINE_PCI_DEVICE_TABLE(bnx2x_pci_tbl) = {
E
Eilon Greenstein 已提交
198 199 200
	{ PCI_VDEVICE(BROADCOM, PCI_DEVICE_ID_NX2_57710), BCM57710 },
	{ PCI_VDEVICE(BROADCOM, PCI_DEVICE_ID_NX2_57711), BCM57711 },
	{ PCI_VDEVICE(BROADCOM, PCI_DEVICE_ID_NX2_57711E), BCM57711E },
D
Dmitry Kravkov 已提交
201
	{ PCI_VDEVICE(BROADCOM, PCI_DEVICE_ID_NX2_57712), BCM57712 },
202 203 204 205 206 207 208
	{ PCI_VDEVICE(BROADCOM, PCI_DEVICE_ID_NX2_57712_MF), BCM57712_MF },
	{ PCI_VDEVICE(BROADCOM, PCI_DEVICE_ID_NX2_57800), BCM57800 },
	{ PCI_VDEVICE(BROADCOM, PCI_DEVICE_ID_NX2_57800_MF), BCM57800_MF },
	{ PCI_VDEVICE(BROADCOM, PCI_DEVICE_ID_NX2_57810), BCM57810 },
	{ PCI_VDEVICE(BROADCOM, PCI_DEVICE_ID_NX2_57810_MF), BCM57810_MF },
	{ PCI_VDEVICE(BROADCOM, PCI_DEVICE_ID_NX2_57840), BCM57840 },
	{ PCI_VDEVICE(BROADCOM, PCI_DEVICE_ID_NX2_57840_MF), BCM57840_MF },
E
Eliezer Tamir 已提交
209 210 211 212 213 214 215 216 217
	{ 0 }
};

MODULE_DEVICE_TABLE(pci, bnx2x_pci_tbl);

/****************************************************************************
* General service functions
****************************************************************************/

218 219 220 221 222 223 224 225 226 227 228 229 230 231 232 233 234 235
static inline void __storm_memset_dma_mapping(struct bnx2x *bp,
				       u32 addr, dma_addr_t mapping)
{
	REG_WR(bp,  addr, U64_LO(mapping));
	REG_WR(bp,  addr + 4, U64_HI(mapping));
}

static inline void storm_memset_spq_addr(struct bnx2x *bp,
					 dma_addr_t mapping, u16 abs_fid)
{
	u32 addr = XSEM_REG_FAST_MEMORY +
			XSTORM_SPQ_PAGE_BASE_OFFSET(abs_fid);

	__storm_memset_dma_mapping(bp, addr, mapping);
}

static inline void storm_memset_vf_to_pf(struct bnx2x *bp, u16 abs_fid,
					 u16 pf_id)
236
{
237 238 239 240 241 242 243 244
	REG_WR8(bp, BAR_XSTRORM_INTMEM + XSTORM_VF_TO_PF_OFFSET(abs_fid),
		pf_id);
	REG_WR8(bp, BAR_CSTRORM_INTMEM + CSTORM_VF_TO_PF_OFFSET(abs_fid),
		pf_id);
	REG_WR8(bp, BAR_TSTRORM_INTMEM + TSTORM_VF_TO_PF_OFFSET(abs_fid),
		pf_id);
	REG_WR8(bp, BAR_USTRORM_INTMEM + USTORM_VF_TO_PF_OFFSET(abs_fid),
		pf_id);
245 246
}

247 248 249 250 251 252 253 254 255 256 257 258
static inline void storm_memset_func_en(struct bnx2x *bp, u16 abs_fid,
					u8 enable)
{
	REG_WR8(bp, BAR_XSTRORM_INTMEM + XSTORM_FUNC_EN_OFFSET(abs_fid),
		enable);
	REG_WR8(bp, BAR_CSTRORM_INTMEM + CSTORM_FUNC_EN_OFFSET(abs_fid),
		enable);
	REG_WR8(bp, BAR_TSTRORM_INTMEM + TSTORM_FUNC_EN_OFFSET(abs_fid),
		enable);
	REG_WR8(bp, BAR_USTRORM_INTMEM + USTORM_FUNC_EN_OFFSET(abs_fid),
		enable);
}
259 260 261 262 263 264 265 266 267 268 269 270 271 272 273 274 275 276 277

static inline void storm_memset_eq_data(struct bnx2x *bp,
				struct event_ring_data *eq_data,
				u16 pfid)
{
	size_t size = sizeof(struct event_ring_data);

	u32 addr = BAR_CSTRORM_INTMEM + CSTORM_EVENT_RING_DATA_OFFSET(pfid);

	__storm_memset_struct(bp, addr, size, (u32 *)eq_data);
}

static inline void storm_memset_eq_prod(struct bnx2x *bp, u16 eq_prod,
					u16 pfid)
{
	u32 addr = BAR_CSTRORM_INTMEM + CSTORM_EVENT_RING_PROD_OFFSET(pfid);
	REG_WR16(bp, addr, eq_prod);
}

E
Eliezer Tamir 已提交
278 279 280
/* used only at init
 * locking is done by mcp
 */
281
static void bnx2x_reg_wr_ind(struct bnx2x *bp, u32 addr, u32 val)
E
Eliezer Tamir 已提交
282 283 284 285 286 287 288 289 290 291 292 293 294 295 296 297 298 299 300
{
	pci_write_config_dword(bp->pdev, PCICFG_GRC_ADDRESS, addr);
	pci_write_config_dword(bp->pdev, PCICFG_GRC_DATA, val);
	pci_write_config_dword(bp->pdev, PCICFG_GRC_ADDRESS,
			       PCICFG_VENDOR_ID_OFFSET);
}

static u32 bnx2x_reg_rd_ind(struct bnx2x *bp, u32 addr)
{
	u32 val;

	pci_write_config_dword(bp->pdev, PCICFG_GRC_ADDRESS, addr);
	pci_read_config_dword(bp->pdev, PCICFG_GRC_DATA, &val);
	pci_write_config_dword(bp->pdev, PCICFG_GRC_ADDRESS,
			       PCICFG_VENDOR_ID_OFFSET);

	return val;
}

D
Dmitry Kravkov 已提交
301 302 303 304 305 306
#define DMAE_DP_SRC_GRC		"grc src_addr [%08x]"
#define DMAE_DP_SRC_PCI		"pci src_addr [%x:%08x]"
#define DMAE_DP_DST_GRC		"grc dst_addr [%08x]"
#define DMAE_DP_DST_PCI		"pci dst_addr [%x:%08x]"
#define DMAE_DP_DST_NONE	"dst_addr [none]"

307 308
static void bnx2x_dp_dmae(struct bnx2x *bp, struct dmae_command *dmae,
			  int msglvl)
D
Dmitry Kravkov 已提交
309 310 311 312 313 314 315 316 317 318 319 320 321 322 323 324 325 326 327 328 329 330 331 332 333 334 335 336 337 338 339 340 341 342 343 344 345 346 347 348 349 350 351
{
	u32 src_type = dmae->opcode & DMAE_COMMAND_SRC;

	switch (dmae->opcode & DMAE_COMMAND_DST) {
	case DMAE_CMD_DST_PCI:
		if (src_type == DMAE_CMD_SRC_PCI)
			DP(msglvl, "DMAE: opcode 0x%08x\n"
			   "src [%x:%08x], len [%d*4], dst [%x:%08x]\n"
			   "comp_addr [%x:%08x], comp_val 0x%08x\n",
			   dmae->opcode, dmae->src_addr_hi, dmae->src_addr_lo,
			   dmae->len, dmae->dst_addr_hi, dmae->dst_addr_lo,
			   dmae->comp_addr_hi, dmae->comp_addr_lo,
			   dmae->comp_val);
		else
			DP(msglvl, "DMAE: opcode 0x%08x\n"
			   "src [%08x], len [%d*4], dst [%x:%08x]\n"
			   "comp_addr [%x:%08x], comp_val 0x%08x\n",
			   dmae->opcode, dmae->src_addr_lo >> 2,
			   dmae->len, dmae->dst_addr_hi, dmae->dst_addr_lo,
			   dmae->comp_addr_hi, dmae->comp_addr_lo,
			   dmae->comp_val);
		break;
	case DMAE_CMD_DST_GRC:
		if (src_type == DMAE_CMD_SRC_PCI)
			DP(msglvl, "DMAE: opcode 0x%08x\n"
			   "src [%x:%08x], len [%d*4], dst_addr [%08x]\n"
			   "comp_addr [%x:%08x], comp_val 0x%08x\n",
			   dmae->opcode, dmae->src_addr_hi, dmae->src_addr_lo,
			   dmae->len, dmae->dst_addr_lo >> 2,
			   dmae->comp_addr_hi, dmae->comp_addr_lo,
			   dmae->comp_val);
		else
			DP(msglvl, "DMAE: opcode 0x%08x\n"
			   "src [%08x], len [%d*4], dst [%08x]\n"
			   "comp_addr [%x:%08x], comp_val 0x%08x\n",
			   dmae->opcode, dmae->src_addr_lo >> 2,
			   dmae->len, dmae->dst_addr_lo >> 2,
			   dmae->comp_addr_hi, dmae->comp_addr_lo,
			   dmae->comp_val);
		break;
	default:
		if (src_type == DMAE_CMD_SRC_PCI)
			DP(msglvl, "DMAE: opcode 0x%08x\n"
352 353
			   "src_addr [%x:%08x]  len [%d * 4]  dst_addr [none]\n"
			   "comp_addr [%x:%08x]  comp_val 0x%08x\n",
D
Dmitry Kravkov 已提交
354 355 356 357 358
			   dmae->opcode, dmae->src_addr_hi, dmae->src_addr_lo,
			   dmae->len, dmae->comp_addr_hi, dmae->comp_addr_lo,
			   dmae->comp_val);
		else
			DP(msglvl, "DMAE: opcode 0x%08x\n"
359 360
			   "src_addr [%08x]  len [%d * 4]  dst_addr [none]\n"
			   "comp_addr [%x:%08x]  comp_val 0x%08x\n",
D
Dmitry Kravkov 已提交
361 362 363 364 365 366 367 368
			   dmae->opcode, dmae->src_addr_lo >> 2,
			   dmae->len, dmae->comp_addr_hi, dmae->comp_addr_lo,
			   dmae->comp_val);
		break;
	}

}

E
Eliezer Tamir 已提交
369
/* copy command into DMAE command memory and set DMAE command go */
370
void bnx2x_post_dmae(struct bnx2x *bp, struct dmae_command *dmae, int idx)
E
Eliezer Tamir 已提交
371 372 373 374 375 376 377 378 379 380 381
{
	u32 cmd_offset;
	int i;

	cmd_offset = (DMAE_REG_CMD_MEM + sizeof(struct dmae_command) * idx);
	for (i = 0; i < (sizeof(struct dmae_command)/4); i++) {
		REG_WR(bp, cmd_offset + i*4, *(((u32 *)dmae) + i));
	}
	REG_WR(bp, dmae_reg_go_c[idx], 1);
}

D
Dmitry Kravkov 已提交
382
u32 bnx2x_dmae_opcode_add_comp(u32 opcode, u8 comp_type)
E
Eliezer Tamir 已提交
383
{
D
Dmitry Kravkov 已提交
384 385 386
	return opcode | ((comp_type << DMAE_COMMAND_C_DST_SHIFT) |
			   DMAE_CMD_C_ENABLE);
}
387

D
Dmitry Kravkov 已提交
388 389 390 391
u32 bnx2x_dmae_opcode_clr_src_reset(u32 opcode)
{
	return opcode & ~DMAE_CMD_SRC_RESET;
}
392

D
Dmitry Kravkov 已提交
393 394 395 396 397 398 399
u32 bnx2x_dmae_opcode(struct bnx2x *bp, u8 src_type, u8 dst_type,
			     bool with_comp, u8 comp_type)
{
	u32 opcode = 0;

	opcode |= ((src_type << DMAE_COMMAND_SRC_SHIFT) |
		   (dst_type << DMAE_COMMAND_DST_SHIFT));
400

D
Dmitry Kravkov 已提交
401 402 403
	opcode |= (DMAE_CMD_SRC_RESET | DMAE_CMD_DST_RESET);

	opcode |= (BP_PORT(bp) ? DMAE_CMD_PORT_1 : DMAE_CMD_PORT_0);
404 405
	opcode |= ((BP_VN(bp) << DMAE_CMD_E1HVN_SHIFT) |
		   (BP_VN(bp) << DMAE_COMMAND_DST_VN_SHIFT));
D
Dmitry Kravkov 已提交
406
	opcode |= (DMAE_COM_SET_ERR << DMAE_COMMAND_ERR_POLICY_SHIFT);
E
Eliezer Tamir 已提交
407 408

#ifdef __BIG_ENDIAN
D
Dmitry Kravkov 已提交
409
	opcode |= DMAE_CMD_ENDIANITY_B_DW_SWAP;
E
Eliezer Tamir 已提交
410
#else
D
Dmitry Kravkov 已提交
411
	opcode |= DMAE_CMD_ENDIANITY_DW_SWAP;
E
Eliezer Tamir 已提交
412
#endif
D
Dmitry Kravkov 已提交
413 414 415 416 417
	if (with_comp)
		opcode = bnx2x_dmae_opcode_add_comp(opcode, comp_type);
	return opcode;
}

418 419 420
static void bnx2x_prep_dmae_with_comp(struct bnx2x *bp,
				      struct dmae_command *dmae,
				      u8 src_type, u8 dst_type)
D
Dmitry Kravkov 已提交
421 422 423 424 425 426 427 428 429 430 431 432 433 434
{
	memset(dmae, 0, sizeof(struct dmae_command));

	/* set the opcode */
	dmae->opcode = bnx2x_dmae_opcode(bp, src_type, dst_type,
					 true, DMAE_COMP_PCI);

	/* fill in the completion parameters */
	dmae->comp_addr_lo = U64_LO(bnx2x_sp_mapping(bp, wb_comp));
	dmae->comp_addr_hi = U64_HI(bnx2x_sp_mapping(bp, wb_comp));
	dmae->comp_val = DMAE_COMP_VAL;
}

/* issue a dmae command over the init-channel and wailt for completion */
435 436
static int bnx2x_issue_dmae_with_comp(struct bnx2x *bp,
				      struct dmae_command *dmae)
D
Dmitry Kravkov 已提交
437 438
{
	u32 *wb_comp = bnx2x_sp(bp, wb_comp);
439
	int cnt = CHIP_REV_IS_SLOW(bp) ? (400000) : 4000;
D
Dmitry Kravkov 已提交
440 441
	int rc = 0;

442 443 444 445 446
	/*
	 * Lock the dmae channel. Disable BHs to prevent a dead-lock
	 * as long as this code is called both from syscall context and
	 * from ndo_set_rx_mode() flow that may be called from BH.
	 */
447
	spin_lock_bh(&bp->dmae_lock);
448

D
Dmitry Kravkov 已提交
449
	/* reset completion */
E
Eliezer Tamir 已提交
450 451
	*wb_comp = 0;

D
Dmitry Kravkov 已提交
452 453
	/* post the command on the channel used for initializations */
	bnx2x_post_dmae(bp, dmae, INIT_DMAE_C(bp));
E
Eliezer Tamir 已提交
454

D
Dmitry Kravkov 已提交
455
	/* wait for completion */
E
Eliezer Tamir 已提交
456
	udelay(5);
D
Dmitry Kravkov 已提交
457
	while ((*wb_comp & ~DMAE_PCI_ERR_FLAG) != DMAE_COMP_VAL) {
458

A
Ariel Elior 已提交
459 460 461
		if (!cnt ||
		    (bp->recovery_state != BNX2X_RECOVERY_DONE &&
		     bp->recovery_state != BNX2X_RECOVERY_NIC_LOADING)) {
E
Eilon Greenstein 已提交
462
			BNX2X_ERR("DMAE timeout!\n");
D
Dmitry Kravkov 已提交
463 464
			rc = DMAE_TIMEOUT;
			goto unlock;
E
Eliezer Tamir 已提交
465
		}
466
		cnt--;
D
Dmitry Kravkov 已提交
467
		udelay(50);
E
Eliezer Tamir 已提交
468
	}
D
Dmitry Kravkov 已提交
469 470 471 472 473 474
	if (*wb_comp & DMAE_PCI_ERR_FLAG) {
		BNX2X_ERR("DMAE PCI error!\n");
		rc = DMAE_PCI_ERROR;
	}

unlock:
475
	spin_unlock_bh(&bp->dmae_lock);
D
Dmitry Kravkov 已提交
476 477 478 479 480 481 482 483 484 485 486
	return rc;
}

void bnx2x_write_dmae(struct bnx2x *bp, dma_addr_t dma_addr, u32 dst_addr,
		      u32 len32)
{
	struct dmae_command dmae;

	if (!bp->dmae_ready) {
		u32 *data = bnx2x_sp(bp, wb_data[0]);

487 488 489 490
		if (CHIP_IS_E1(bp))
			bnx2x_init_ind_wr(bp, dst_addr, data, len32);
		else
			bnx2x_init_str_wr(bp, dst_addr, data, len32);
D
Dmitry Kravkov 已提交
491 492 493 494 495 496 497 498 499 500 501 502 503 504 505 506 507
		return;
	}

	/* set opcode and fixed command fields */
	bnx2x_prep_dmae_with_comp(bp, &dmae, DMAE_SRC_PCI, DMAE_DST_GRC);

	/* fill in addresses and len */
	dmae.src_addr_lo = U64_LO(dma_addr);
	dmae.src_addr_hi = U64_HI(dma_addr);
	dmae.dst_addr_lo = dst_addr >> 2;
	dmae.dst_addr_hi = 0;
	dmae.len = len32;

	bnx2x_dp_dmae(bp, &dmae, BNX2X_MSG_OFF);

	/* issue the command and wait for completion */
	bnx2x_issue_dmae_with_comp(bp, &dmae);
E
Eliezer Tamir 已提交
508 509
}

Y
Yaniv Rosner 已提交
510
void bnx2x_read_dmae(struct bnx2x *bp, u32 src_addr, u32 len32)
E
Eliezer Tamir 已提交
511
{
512
	struct dmae_command dmae;
513 514 515 516 517

	if (!bp->dmae_ready) {
		u32 *data = bnx2x_sp(bp, wb_data[0]);
		int i;

M
Merav Sicron 已提交
518
		if (CHIP_IS_E1(bp))
519 520
			for (i = 0; i < len32; i++)
				data[i] = bnx2x_reg_rd_ind(bp, src_addr + i*4);
M
Merav Sicron 已提交
521
		else
522 523 524
			for (i = 0; i < len32; i++)
				data[i] = REG_RD(bp, src_addr + i*4);

525 526 527
		return;
	}

D
Dmitry Kravkov 已提交
528 529
	/* set opcode and fixed command fields */
	bnx2x_prep_dmae_with_comp(bp, &dmae, DMAE_SRC_GRC, DMAE_DST_PCI);
E
Eliezer Tamir 已提交
530

D
Dmitry Kravkov 已提交
531
	/* fill in addresses and len */
532 533 534 535 536
	dmae.src_addr_lo = src_addr >> 2;
	dmae.src_addr_hi = 0;
	dmae.dst_addr_lo = U64_LO(bnx2x_sp_mapping(bp, wb_data));
	dmae.dst_addr_hi = U64_HI(bnx2x_sp_mapping(bp, wb_data));
	dmae.len = len32;
537

D
Dmitry Kravkov 已提交
538
	bnx2x_dp_dmae(bp, &dmae, BNX2X_MSG_OFF);
539

D
Dmitry Kravkov 已提交
540 541
	/* issue the command and wait for completion */
	bnx2x_issue_dmae_with_comp(bp, &dmae);
542 543
}

544 545
static void bnx2x_write_dmae_phys_len(struct bnx2x *bp, dma_addr_t phys_addr,
				      u32 addr, u32 len)
546
{
547
	int dmae_wr_max = DMAE_LEN32_WR_MAX(bp);
548 549
	int offset = 0;

550
	while (len > dmae_wr_max) {
551
		bnx2x_write_dmae(bp, phys_addr + offset,
552 553 554
				 addr + offset, dmae_wr_max);
		offset += dmae_wr_max * 4;
		len -= dmae_wr_max;
555 556 557 558 559
	}

	bnx2x_write_dmae(bp, phys_addr + offset, addr + offset, len);
}

560 561 562 563 564 565 566 567
/* used only for slowpath so not inlined */
static void bnx2x_wb_wr(struct bnx2x *bp, int reg, u32 val_hi, u32 val_lo)
{
	u32 wb_write[2];

	wb_write[0] = val_hi;
	wb_write[1] = val_lo;
	REG_WR_DMAE(bp, reg, wb_write, 2);
E
Eliezer Tamir 已提交
568 569
}

570 571 572 573 574 575 576 577 578 579 580
#ifdef USE_WB_RD
static u64 bnx2x_wb_rd(struct bnx2x *bp, int reg)
{
	u32 wb_data[2];

	REG_RD_DMAE(bp, reg, wb_data, 2);

	return HILO_U64(wb_data[0], wb_data[1]);
}
#endif

E
Eliezer Tamir 已提交
581 582 583
static int bnx2x_mc_assert(struct bnx2x *bp)
{
	char last_idx;
584 585 586 587 588 589 590 591 592 593 594 595 596 597 598 599 600 601 602 603 604 605
	int i, rc = 0;
	u32 row0, row1, row2, row3;

	/* XSTORM */
	last_idx = REG_RD8(bp, BAR_XSTRORM_INTMEM +
			   XSTORM_ASSERT_LIST_INDEX_OFFSET);
	if (last_idx)
		BNX2X_ERR("XSTORM_ASSERT_LIST_INDEX 0x%x\n", last_idx);

	/* print the asserts */
	for (i = 0; i < STROM_ASSERT_ARRAY_SIZE; i++) {

		row0 = REG_RD(bp, BAR_XSTRORM_INTMEM +
			      XSTORM_ASSERT_LIST_OFFSET(i));
		row1 = REG_RD(bp, BAR_XSTRORM_INTMEM +
			      XSTORM_ASSERT_LIST_OFFSET(i) + 4);
		row2 = REG_RD(bp, BAR_XSTRORM_INTMEM +
			      XSTORM_ASSERT_LIST_OFFSET(i) + 8);
		row3 = REG_RD(bp, BAR_XSTRORM_INTMEM +
			      XSTORM_ASSERT_LIST_OFFSET(i) + 12);

		if (row0 != COMMON_ASM_INVALID_ASSERT_OPCODE) {
M
Merav Sicron 已提交
606
			BNX2X_ERR("XSTORM_ASSERT_INDEX 0x%x = 0x%08x 0x%08x 0x%08x 0x%08x\n",
607 608 609 610 611 612 613 614 615 616 617 618 619 620 621 622 623 624 625 626 627 628 629 630 631 632
				  i, row3, row2, row1, row0);
			rc++;
		} else {
			break;
		}
	}

	/* TSTORM */
	last_idx = REG_RD8(bp, BAR_TSTRORM_INTMEM +
			   TSTORM_ASSERT_LIST_INDEX_OFFSET);
	if (last_idx)
		BNX2X_ERR("TSTORM_ASSERT_LIST_INDEX 0x%x\n", last_idx);

	/* print the asserts */
	for (i = 0; i < STROM_ASSERT_ARRAY_SIZE; i++) {

		row0 = REG_RD(bp, BAR_TSTRORM_INTMEM +
			      TSTORM_ASSERT_LIST_OFFSET(i));
		row1 = REG_RD(bp, BAR_TSTRORM_INTMEM +
			      TSTORM_ASSERT_LIST_OFFSET(i) + 4);
		row2 = REG_RD(bp, BAR_TSTRORM_INTMEM +
			      TSTORM_ASSERT_LIST_OFFSET(i) + 8);
		row3 = REG_RD(bp, BAR_TSTRORM_INTMEM +
			      TSTORM_ASSERT_LIST_OFFSET(i) + 12);

		if (row0 != COMMON_ASM_INVALID_ASSERT_OPCODE) {
M
Merav Sicron 已提交
633
			BNX2X_ERR("TSTORM_ASSERT_INDEX 0x%x = 0x%08x 0x%08x 0x%08x 0x%08x\n",
634 635 636 637 638 639 640 641 642 643 644 645 646 647 648 649 650 651 652 653 654 655 656 657 658 659
				  i, row3, row2, row1, row0);
			rc++;
		} else {
			break;
		}
	}

	/* CSTORM */
	last_idx = REG_RD8(bp, BAR_CSTRORM_INTMEM +
			   CSTORM_ASSERT_LIST_INDEX_OFFSET);
	if (last_idx)
		BNX2X_ERR("CSTORM_ASSERT_LIST_INDEX 0x%x\n", last_idx);

	/* print the asserts */
	for (i = 0; i < STROM_ASSERT_ARRAY_SIZE; i++) {

		row0 = REG_RD(bp, BAR_CSTRORM_INTMEM +
			      CSTORM_ASSERT_LIST_OFFSET(i));
		row1 = REG_RD(bp, BAR_CSTRORM_INTMEM +
			      CSTORM_ASSERT_LIST_OFFSET(i) + 4);
		row2 = REG_RD(bp, BAR_CSTRORM_INTMEM +
			      CSTORM_ASSERT_LIST_OFFSET(i) + 8);
		row3 = REG_RD(bp, BAR_CSTRORM_INTMEM +
			      CSTORM_ASSERT_LIST_OFFSET(i) + 12);

		if (row0 != COMMON_ASM_INVALID_ASSERT_OPCODE) {
M
Merav Sicron 已提交
660
			BNX2X_ERR("CSTORM_ASSERT_INDEX 0x%x = 0x%08x 0x%08x 0x%08x 0x%08x\n",
661 662 663 664 665 666 667 668 669 670 671 672 673 674 675 676 677 678 679 680 681 682 683 684 685 686
				  i, row3, row2, row1, row0);
			rc++;
		} else {
			break;
		}
	}

	/* USTORM */
	last_idx = REG_RD8(bp, BAR_USTRORM_INTMEM +
			   USTORM_ASSERT_LIST_INDEX_OFFSET);
	if (last_idx)
		BNX2X_ERR("USTORM_ASSERT_LIST_INDEX 0x%x\n", last_idx);

	/* print the asserts */
	for (i = 0; i < STROM_ASSERT_ARRAY_SIZE; i++) {

		row0 = REG_RD(bp, BAR_USTRORM_INTMEM +
			      USTORM_ASSERT_LIST_OFFSET(i));
		row1 = REG_RD(bp, BAR_USTRORM_INTMEM +
			      USTORM_ASSERT_LIST_OFFSET(i) + 4);
		row2 = REG_RD(bp, BAR_USTRORM_INTMEM +
			      USTORM_ASSERT_LIST_OFFSET(i) + 8);
		row3 = REG_RD(bp, BAR_USTRORM_INTMEM +
			      USTORM_ASSERT_LIST_OFFSET(i) + 12);

		if (row0 != COMMON_ASM_INVALID_ASSERT_OPCODE) {
M
Merav Sicron 已提交
687
			BNX2X_ERR("USTORM_ASSERT_INDEX 0x%x = 0x%08x 0x%08x 0x%08x 0x%08x\n",
688 689 690 691
				  i, row3, row2, row1, row0);
			rc++;
		} else {
			break;
E
Eliezer Tamir 已提交
692 693
		}
	}
694

E
Eliezer Tamir 已提交
695 696
	return rc;
}
E
Eliezer Tamir 已提交
697

698
void bnx2x_fw_dump_lvl(struct bnx2x *bp, const char *lvl)
E
Eliezer Tamir 已提交
699
{
700
	u32 addr, val;
E
Eliezer Tamir 已提交
701
	u32 mark, offset;
702
	__be32 data[9];
E
Eliezer Tamir 已提交
703
	int word;
D
Dmitry Kravkov 已提交
704
	u32 trace_shmem_base;
705 706 707 708
	if (BP_NOMCP(bp)) {
		BNX2X_ERR("NO MCP - can not dump\n");
		return;
	}
709 710 711 712 713 714 715
	netdev_printk(lvl, bp->dev, "bc %d.%d.%d\n",
		(bp->common.bc_ver & 0xff0000) >> 16,
		(bp->common.bc_ver & 0xff00) >> 8,
		(bp->common.bc_ver & 0xff));

	val = REG_RD(bp, MCP_REG_MCPR_CPU_PROGRAM_COUNTER);
	if (val == REG_RD(bp, MCP_REG_MCPR_CPU_PROGRAM_COUNTER))
M
Merav Sicron 已提交
716
		BNX2X_ERR("%s" "MCP PC at 0x%x\n", lvl, val);
V
Vladislav Zolotarov 已提交
717

D
Dmitry Kravkov 已提交
718 719 720 721 722
	if (BP_PATH(bp) == 0)
		trace_shmem_base = bp->common.shmem_base;
	else
		trace_shmem_base = SHMEM2_RD(bp, other_shmem_base_addr);
	addr = trace_shmem_base - 0x0800 + 4;
V
Vladislav Zolotarov 已提交
723
	mark = REG_RD(bp, addr);
D
Dmitry Kravkov 已提交
724 725
	mark = (CHIP_IS_E1x(bp) ? MCP_REG_MCPR_SCRATCH : MCP_A_REG_MCPR_SCRATCH)
			+ ((mark + 0x3) & ~0x3) - 0x08000000;
726
	printk("%s" "begin fw dump (mark 0x%x)\n", lvl, mark);
E
Eliezer Tamir 已提交
727

728
	printk("%s", lvl);
D
Dmitry Kravkov 已提交
729
	for (offset = mark; offset <= trace_shmem_base; offset += 0x8*4) {
E
Eliezer Tamir 已提交
730
		for (word = 0; word < 8; word++)
V
Vladislav Zolotarov 已提交
731
			data[word] = htonl(REG_RD(bp, offset + 4*word));
E
Eliezer Tamir 已提交
732
		data[8] = 0x0;
733
		pr_cont("%s", (char *)data);
E
Eliezer Tamir 已提交
734
	}
V
Vladislav Zolotarov 已提交
735
	for (offset = addr + 4; offset <= mark; offset += 0x8*4) {
E
Eliezer Tamir 已提交
736
		for (word = 0; word < 8; word++)
V
Vladislav Zolotarov 已提交
737
			data[word] = htonl(REG_RD(bp, offset + 4*word));
E
Eliezer Tamir 已提交
738
		data[8] = 0x0;
739
		pr_cont("%s", (char *)data);
E
Eliezer Tamir 已提交
740
	}
741 742 743 744 745 746
	printk("%s" "end of fw dump\n", lvl);
}

static inline void bnx2x_fw_dump(struct bnx2x *bp)
{
	bnx2x_fw_dump_lvl(bp, KERN_ERR);
E
Eliezer Tamir 已提交
747 748
}

749
void bnx2x_panic_dump(struct bnx2x *bp)
E
Eliezer Tamir 已提交
750 751
{
	int i;
752 753 754 755 756
	u16 j;
	struct hc_sp_status_block_data sp_sb_data;
	int func = BP_FUNC(bp);
#ifdef BNX2X_STOP_ON_ERROR
	u16 start = 0, end = 0;
757
	u8 cos;
758
#endif
E
Eliezer Tamir 已提交
759

Y
Yitchak Gertner 已提交
760
	bp->stats_state = STATS_STATE_DISABLED;
761
	bp->eth_stats.unrecoverable_error++;
Y
Yitchak Gertner 已提交
762 763
	DP(BNX2X_MSG_STATS, "stats_state - DISABLED\n");

E
Eliezer Tamir 已提交
764 765
	BNX2X_ERR("begin crash dump -----------------\n");

E
Eilon Greenstein 已提交
766 767
	/* Indices */
	/* Common */
M
Merav Sicron 已提交
768
	BNX2X_ERR("def_idx(0x%x)  def_att_idx(0x%x)  attn_state(0x%x)  spq_prod_idx(0x%x) next_stats_cnt(0x%x)\n",
769 770
		  bp->def_idx, bp->def_att_idx, bp->attn_state,
		  bp->spq_prod_idx, bp->stats_counter);
771 772 773 774 775 776 777 778
	BNX2X_ERR("DSB: attn bits(0x%x)  ack(0x%x)  id(0x%x)  idx(0x%x)\n",
		  bp->def_status_blk->atten_status_block.attn_bits,
		  bp->def_status_blk->atten_status_block.attn_bits_ack,
		  bp->def_status_blk->atten_status_block.status_block_id,
		  bp->def_status_blk->atten_status_block.attn_bits_index);
	BNX2X_ERR("     def (");
	for (i = 0; i < HC_SP_SB_MAX_INDICES; i++)
		pr_cont("0x%x%s",
779 780
			bp->def_status_blk->sp_sb.index_values[i],
			(i == HC_SP_SB_MAX_INDICES - 1) ? ")  " : " ");
781 782 783 784 785 786

	for (i = 0; i < sizeof(struct hc_sp_status_block_data)/sizeof(u32); i++)
		*((u32 *)&sp_sb_data + i) = REG_RD(bp, BAR_CSTRORM_INTMEM +
			CSTORM_SP_STATUS_BLOCK_DATA_OFFSET(func) +
			i*sizeof(u32));

787
	pr_cont("igu_sb_id(0x%x)  igu_seg_id(0x%x) pf_id(0x%x)  vnic_id(0x%x)  vf_id(0x%x)  vf_valid (0x%x) state(0x%x)\n",
788 789 790 791 792
	       sp_sb_data.igu_sb_id,
	       sp_sb_data.igu_seg_id,
	       sp_sb_data.p_func.pf_id,
	       sp_sb_data.p_func.vnic_id,
	       sp_sb_data.p_func.vf_id,
793 794
	       sp_sb_data.p_func.vf_valid,
	       sp_sb_data.state);
795

E
Eilon Greenstein 已提交
796

V
Vladislav Zolotarov 已提交
797
	for_each_eth_queue(bp, i) {
E
Eliezer Tamir 已提交
798
		struct bnx2x_fastpath *fp = &bp->fp[i];
799
		int loop;
D
Dmitry Kravkov 已提交
800
		struct hc_status_block_data_e2 sb_data_e2;
801 802
		struct hc_status_block_data_e1x sb_data_e1x;
		struct hc_status_block_sm  *hc_sm_p =
803 804 805
			CHIP_IS_E1x(bp) ?
			sb_data_e1x.common.state_machine :
			sb_data_e2.common.state_machine;
806
		struct hc_index_data *hc_index_p =
807 808 809
			CHIP_IS_E1x(bp) ?
			sb_data_e1x.index_data :
			sb_data_e2.index_data;
810
		u8 data_size, cos;
811
		u32 *sb_data_p;
812
		struct bnx2x_fp_txdata txdata;
813 814

		/* Rx */
M
Merav Sicron 已提交
815
		BNX2X_ERR("fp%d: rx_bd_prod(0x%x)  rx_bd_cons(0x%x)  rx_comp_prod(0x%x)  rx_comp_cons(0x%x)  *rx_cons_sb(0x%x)\n",
E
Eilon Greenstein 已提交
816
			  i, fp->rx_bd_prod, fp->rx_bd_cons,
817
			  fp->rx_comp_prod,
Y
Yitchak Gertner 已提交
818
			  fp->rx_comp_cons, le16_to_cpu(*fp->rx_cons_sb));
M
Merav Sicron 已提交
819
		BNX2X_ERR("     rx_sge_prod(0x%x)  last_max_sge(0x%x)  fp_hc_idx(0x%x)\n",
E
Eilon Greenstein 已提交
820
			  fp->rx_sge_prod, fp->last_max_sge,
821
			  le16_to_cpu(fp->fp_hc_idx));
E
Eliezer Tamir 已提交
822

823
		/* Tx */
824 825 826
		for_each_cos_in_tx_queue(fp, cos)
		{
			txdata = fp->txdata[cos];
M
Merav Sicron 已提交
827
			BNX2X_ERR("fp%d: tx_pkt_prod(0x%x)  tx_pkt_cons(0x%x)  tx_bd_prod(0x%x)  tx_bd_cons(0x%x)  *tx_cons_sb(0x%x)\n",
828 829 830 831 832
				  i, txdata.tx_pkt_prod,
				  txdata.tx_pkt_cons, txdata.tx_bd_prod,
				  txdata.tx_bd_cons,
				  le16_to_cpu(*txdata.tx_cons_sb));
		}
833

834 835
		loop = CHIP_IS_E1x(bp) ?
			HC_SB_MAX_INDICES_E1X : HC_SB_MAX_INDICES_E2;
836 837 838

		/* host sb data */

V
Vladislav Zolotarov 已提交
839 840 841 842
#ifdef BCM_CNIC
		if (IS_FCOE_FP(fp))
			continue;
#endif
843 844 845 846 847 848 849 850 851 852 853 854
		BNX2X_ERR("     run indexes (");
		for (j = 0; j < HC_SB_MAX_SM; j++)
			pr_cont("0x%x%s",
			       fp->sb_running_index[j],
			       (j == HC_SB_MAX_SM - 1) ? ")" : " ");

		BNX2X_ERR("     indexes (");
		for (j = 0; j < loop; j++)
			pr_cont("0x%x%s",
			       fp->sb_index_values[j],
			       (j == loop - 1) ? ")" : " ");
		/* fw sb data */
855 856 857
		data_size = CHIP_IS_E1x(bp) ?
			sizeof(struct hc_status_block_data_e1x) :
			sizeof(struct hc_status_block_data_e2);
858
		data_size /= sizeof(u32);
859 860 861
		sb_data_p = CHIP_IS_E1x(bp) ?
			(u32 *)&sb_data_e1x :
			(u32 *)&sb_data_e2;
862 863 864 865 866 867
		/* copy sb data in here */
		for (j = 0; j < data_size; j++)
			*(sb_data_p + j) = REG_RD(bp, BAR_CSTRORM_INTMEM +
				CSTORM_STATUS_BLOCK_DATA_OFFSET(fp->fw_sb_id) +
				j * sizeof(u32));

868
		if (!CHIP_IS_E1x(bp)) {
M
Merav Sicron 已提交
869
			pr_cont("pf_id(0x%x)  vf_id(0x%x)  vf_valid(0x%x) vnic_id(0x%x)  same_igu_sb_1b(0x%x) state(0x%x)\n",
D
Dmitry Kravkov 已提交
870 871 872 873
				sb_data_e2.common.p_func.pf_id,
				sb_data_e2.common.p_func.vf_id,
				sb_data_e2.common.p_func.vf_valid,
				sb_data_e2.common.p_func.vnic_id,
874 875
				sb_data_e2.common.same_igu_sb_1b,
				sb_data_e2.common.state);
D
Dmitry Kravkov 已提交
876
		} else {
M
Merav Sicron 已提交
877
			pr_cont("pf_id(0x%x)  vf_id(0x%x)  vf_valid(0x%x) vnic_id(0x%x)  same_igu_sb_1b(0x%x) state(0x%x)\n",
D
Dmitry Kravkov 已提交
878 879 880 881
				sb_data_e1x.common.p_func.pf_id,
				sb_data_e1x.common.p_func.vf_id,
				sb_data_e1x.common.p_func.vf_valid,
				sb_data_e1x.common.p_func.vnic_id,
882 883
				sb_data_e1x.common.same_igu_sb_1b,
				sb_data_e1x.common.state);
D
Dmitry Kravkov 已提交
884
		}
885 886 887

		/* SB_SMs data */
		for (j = 0; j < HC_SB_MAX_SM; j++) {
M
Merav Sicron 已提交
888 889 890 891 892 893
			pr_cont("SM[%d] __flags (0x%x) igu_sb_id (0x%x)  igu_seg_id(0x%x) time_to_expire (0x%x) timer_value(0x%x)\n",
				j, hc_sm_p[j].__flags,
				hc_sm_p[j].igu_sb_id,
				hc_sm_p[j].igu_seg_id,
				hc_sm_p[j].time_to_expire,
				hc_sm_p[j].timer_value);
894 895 896 897
		}

		/* Indecies data */
		for (j = 0; j < loop; j++) {
M
Merav Sicron 已提交
898
			pr_cont("INDEX[%d] flags (0x%x) timeout (0x%x)\n", j,
899 900 901
			       hc_index_p[j].flags,
			       hc_index_p[j].timeout);
		}
E
Eilon Greenstein 已提交
902
	}
E
Eliezer Tamir 已提交
903

904
#ifdef BNX2X_STOP_ON_ERROR
E
Eilon Greenstein 已提交
905 906
	/* Rings */
	/* Rx */
V
Vladislav Zolotarov 已提交
907
	for_each_rx_queue(bp, i) {
E
Eilon Greenstein 已提交
908
		struct bnx2x_fastpath *fp = &bp->fp[i];
E
Eliezer Tamir 已提交
909 910 911

		start = RX_BD(le16_to_cpu(*fp->rx_cons_sb) - 10);
		end = RX_BD(le16_to_cpu(*fp->rx_cons_sb) + 503);
E
Eilon Greenstein 已提交
912
		for (j = start; j != end; j = RX_BD(j + 1)) {
E
Eliezer Tamir 已提交
913 914 915
			u32 *rx_bd = (u32 *)&fp->rx_desc_ring[j];
			struct sw_rx_bd *sw_bd = &fp->rx_buf_ring[j];

E
Eilon Greenstein 已提交
916
			BNX2X_ERR("fp%d: rx_bd[%x]=[%x:%x]  sw_bd=[%p]\n",
917
				  i, j, rx_bd[1], rx_bd[0], sw_bd->data);
E
Eliezer Tamir 已提交
918 919
		}

920 921
		start = RX_SGE(fp->rx_sge_prod);
		end = RX_SGE(fp->last_max_sge);
E
Eilon Greenstein 已提交
922
		for (j = start; j != end; j = RX_SGE(j + 1)) {
923 924 925
			u32 *rx_sge = (u32 *)&fp->rx_sge_ring[j];
			struct sw_rx_page *sw_page = &fp->rx_page_ring[j];

E
Eilon Greenstein 已提交
926 927
			BNX2X_ERR("fp%d: rx_sge[%x]=[%x:%x]  sw_page=[%p]\n",
				  i, j, rx_sge[1], rx_sge[0], sw_page->page);
928 929
		}

E
Eliezer Tamir 已提交
930 931
		start = RCQ_BD(fp->rx_comp_cons - 10);
		end = RCQ_BD(fp->rx_comp_cons + 503);
E
Eilon Greenstein 已提交
932
		for (j = start; j != end; j = RCQ_BD(j + 1)) {
E
Eliezer Tamir 已提交
933 934
			u32 *cqe = (u32 *)&fp->rx_comp_ring[j];

E
Eilon Greenstein 已提交
935 936
			BNX2X_ERR("fp%d: cqe[%x]=[%x:%x:%x:%x]\n",
				  i, j, cqe[0], cqe[1], cqe[2], cqe[3]);
E
Eliezer Tamir 已提交
937 938 939
		}
	}

E
Eilon Greenstein 已提交
940
	/* Tx */
V
Vladislav Zolotarov 已提交
941
	for_each_tx_queue(bp, i) {
E
Eilon Greenstein 已提交
942
		struct bnx2x_fastpath *fp = &bp->fp[i];
943 944 945 946 947 948 949 950 951
		for_each_cos_in_tx_queue(fp, cos) {
			struct bnx2x_fp_txdata *txdata = &fp->txdata[cos];

			start = TX_BD(le16_to_cpu(*txdata->tx_cons_sb) - 10);
			end = TX_BD(le16_to_cpu(*txdata->tx_cons_sb) + 245);
			for (j = start; j != end; j = TX_BD(j + 1)) {
				struct sw_tx_bd *sw_bd =
					&txdata->tx_buf_ring[j];

M
Merav Sicron 已提交
952
				BNX2X_ERR("fp%d: txdata %d, packet[%x]=[%p,%x]\n",
953 954 955
					  i, cos, j, sw_bd->skb,
					  sw_bd->first_bd);
			}
E
Eilon Greenstein 已提交
956

957 958 959 960
			start = TX_BD(txdata->tx_bd_cons - 10);
			end = TX_BD(txdata->tx_bd_cons + 254);
			for (j = start; j != end; j = TX_BD(j + 1)) {
				u32 *tx_bd = (u32 *)&txdata->tx_desc_ring[j];
E
Eilon Greenstein 已提交
961

M
Merav Sicron 已提交
962
				BNX2X_ERR("fp%d: txdata %d, tx_bd[%x]=[%x:%x:%x:%x]\n",
963 964 965
					  i, cos, j, tx_bd[0], tx_bd[1],
					  tx_bd[2], tx_bd[3]);
			}
E
Eilon Greenstein 已提交
966 967
		}
	}
968
#endif
969
	bnx2x_fw_dump(bp);
E
Eliezer Tamir 已提交
970 971 972 973
	bnx2x_mc_assert(bp);
	BNX2X_ERR("end crash dump -----------------\n");
}

974 975 976 977 978 979 980
/*
 * FLR Support for E2
 *
 * bnx2x_pf_flr_clnup() is called during nic_load in the per function HW
 * initialization.
 */
#define FLR_WAIT_USEC		10000	/* 10 miliseconds */
981 982
#define FLR_WAIT_INTERVAL	50	/* usec */
#define	FLR_POLL_CNT		(FLR_WAIT_USEC/FLR_WAIT_INTERVAL) /* 200 */
983 984 985 986 987 988 989 990 991 992 993 994 995 996 997 998 999 1000 1001 1002 1003 1004 1005 1006 1007 1008 1009 1010 1011 1012 1013 1014

struct pbf_pN_buf_regs {
	int pN;
	u32 init_crd;
	u32 crd;
	u32 crd_freed;
};

struct pbf_pN_cmd_regs {
	int pN;
	u32 lines_occup;
	u32 lines_freed;
};

static void bnx2x_pbf_pN_buf_flushed(struct bnx2x *bp,
				     struct pbf_pN_buf_regs *regs,
				     u32 poll_count)
{
	u32 init_crd, crd, crd_start, crd_freed, crd_freed_start;
	u32 cur_cnt = poll_count;

	crd_freed = crd_freed_start = REG_RD(bp, regs->crd_freed);
	crd = crd_start = REG_RD(bp, regs->crd);
	init_crd = REG_RD(bp, regs->init_crd);

	DP(BNX2X_MSG_SP, "INIT CREDIT[%d] : %x\n", regs->pN, init_crd);
	DP(BNX2X_MSG_SP, "CREDIT[%d]      : s:%x\n", regs->pN, crd);
	DP(BNX2X_MSG_SP, "CREDIT_FREED[%d]: s:%x\n", regs->pN, crd_freed);

	while ((crd != init_crd) && ((u32)SUB_S32(crd_freed, crd_freed_start) <
	       (init_crd - crd_start))) {
		if (cur_cnt--) {
1015
			udelay(FLR_WAIT_INTERVAL);
1016 1017 1018 1019 1020 1021 1022 1023 1024 1025 1026 1027 1028
			crd = REG_RD(bp, regs->crd);
			crd_freed = REG_RD(bp, regs->crd_freed);
		} else {
			DP(BNX2X_MSG_SP, "PBF tx buffer[%d] timed out\n",
			   regs->pN);
			DP(BNX2X_MSG_SP, "CREDIT[%d]      : c:%x\n",
			   regs->pN, crd);
			DP(BNX2X_MSG_SP, "CREDIT_FREED[%d]: c:%x\n",
			   regs->pN, crd_freed);
			break;
		}
	}
	DP(BNX2X_MSG_SP, "Waited %d*%d usec for PBF tx buffer[%d]\n",
1029
	   poll_count-cur_cnt, FLR_WAIT_INTERVAL, regs->pN);
1030 1031 1032 1033 1034 1035 1036 1037 1038 1039 1040 1041 1042 1043 1044 1045 1046
}

static void bnx2x_pbf_pN_cmd_flushed(struct bnx2x *bp,
				     struct pbf_pN_cmd_regs *regs,
				     u32 poll_count)
{
	u32 occup, to_free, freed, freed_start;
	u32 cur_cnt = poll_count;

	occup = to_free = REG_RD(bp, regs->lines_occup);
	freed = freed_start = REG_RD(bp, regs->lines_freed);

	DP(BNX2X_MSG_SP, "OCCUPANCY[%d]   : s:%x\n", regs->pN, occup);
	DP(BNX2X_MSG_SP, "LINES_FREED[%d] : s:%x\n", regs->pN, freed);

	while (occup && ((u32)SUB_S32(freed, freed_start) < to_free)) {
		if (cur_cnt--) {
1047
			udelay(FLR_WAIT_INTERVAL);
1048 1049 1050 1051 1052 1053 1054 1055 1056 1057 1058 1059 1060
			occup = REG_RD(bp, regs->lines_occup);
			freed = REG_RD(bp, regs->lines_freed);
		} else {
			DP(BNX2X_MSG_SP, "PBF cmd queue[%d] timed out\n",
			   regs->pN);
			DP(BNX2X_MSG_SP, "OCCUPANCY[%d]   : s:%x\n",
			   regs->pN, occup);
			DP(BNX2X_MSG_SP, "LINES_FREED[%d] : s:%x\n",
			   regs->pN, freed);
			break;
		}
	}
	DP(BNX2X_MSG_SP, "Waited %d*%d usec for PBF cmd queue[%d]\n",
1061
	   poll_count-cur_cnt, FLR_WAIT_INTERVAL, regs->pN);
1062 1063 1064 1065 1066 1067 1068 1069 1070
}

static inline u32 bnx2x_flr_clnup_reg_poll(struct bnx2x *bp, u32 reg,
				     u32 expected, u32 poll_count)
{
	u32 cur_cnt = poll_count;
	u32 val;

	while ((val = REG_RD(bp, reg)) != expected && cur_cnt--)
1071
		udelay(FLR_WAIT_INTERVAL);
1072 1073 1074 1075 1076 1077 1078 1079 1080 1081 1082 1083 1084 1085 1086 1087 1088 1089 1090 1091 1092 1093 1094 1095 1096 1097 1098 1099 1100 1101 1102 1103 1104 1105 1106 1107 1108 1109 1110 1111 1112 1113 1114 1115 1116 1117 1118 1119 1120 1121 1122 1123 1124 1125 1126 1127 1128 1129 1130 1131 1132 1133 1134 1135 1136 1137 1138 1139 1140 1141 1142 1143 1144 1145 1146 1147 1148 1149 1150 1151 1152 1153 1154 1155 1156 1157 1158 1159 1160 1161 1162 1163 1164 1165 1166 1167 1168 1169 1170 1171 1172 1173 1174 1175 1176 1177 1178 1179 1180 1181 1182 1183

	return val;
}

static inline int bnx2x_flr_clnup_poll_hw_counter(struct bnx2x *bp, u32 reg,
						  char *msg, u32 poll_cnt)
{
	u32 val = bnx2x_flr_clnup_reg_poll(bp, reg, 0, poll_cnt);
	if (val != 0) {
		BNX2X_ERR("%s usage count=%d\n", msg, val);
		return 1;
	}
	return 0;
}

static u32 bnx2x_flr_clnup_poll_count(struct bnx2x *bp)
{
	/* adjust polling timeout */
	if (CHIP_REV_IS_EMUL(bp))
		return FLR_POLL_CNT * 2000;

	if (CHIP_REV_IS_FPGA(bp))
		return FLR_POLL_CNT * 120;

	return FLR_POLL_CNT;
}

static void bnx2x_tx_hw_flushed(struct bnx2x *bp, u32 poll_count)
{
	struct pbf_pN_cmd_regs cmd_regs[] = {
		{0, (CHIP_IS_E3B0(bp)) ?
			PBF_REG_TQ_OCCUPANCY_Q0 :
			PBF_REG_P0_TQ_OCCUPANCY,
		    (CHIP_IS_E3B0(bp)) ?
			PBF_REG_TQ_LINES_FREED_CNT_Q0 :
			PBF_REG_P0_TQ_LINES_FREED_CNT},
		{1, (CHIP_IS_E3B0(bp)) ?
			PBF_REG_TQ_OCCUPANCY_Q1 :
			PBF_REG_P1_TQ_OCCUPANCY,
		    (CHIP_IS_E3B0(bp)) ?
			PBF_REG_TQ_LINES_FREED_CNT_Q1 :
			PBF_REG_P1_TQ_LINES_FREED_CNT},
		{4, (CHIP_IS_E3B0(bp)) ?
			PBF_REG_TQ_OCCUPANCY_LB_Q :
			PBF_REG_P4_TQ_OCCUPANCY,
		    (CHIP_IS_E3B0(bp)) ?
			PBF_REG_TQ_LINES_FREED_CNT_LB_Q :
			PBF_REG_P4_TQ_LINES_FREED_CNT}
	};

	struct pbf_pN_buf_regs buf_regs[] = {
		{0, (CHIP_IS_E3B0(bp)) ?
			PBF_REG_INIT_CRD_Q0 :
			PBF_REG_P0_INIT_CRD ,
		    (CHIP_IS_E3B0(bp)) ?
			PBF_REG_CREDIT_Q0 :
			PBF_REG_P0_CREDIT,
		    (CHIP_IS_E3B0(bp)) ?
			PBF_REG_INTERNAL_CRD_FREED_CNT_Q0 :
			PBF_REG_P0_INTERNAL_CRD_FREED_CNT},
		{1, (CHIP_IS_E3B0(bp)) ?
			PBF_REG_INIT_CRD_Q1 :
			PBF_REG_P1_INIT_CRD,
		    (CHIP_IS_E3B0(bp)) ?
			PBF_REG_CREDIT_Q1 :
			PBF_REG_P1_CREDIT,
		    (CHIP_IS_E3B0(bp)) ?
			PBF_REG_INTERNAL_CRD_FREED_CNT_Q1 :
			PBF_REG_P1_INTERNAL_CRD_FREED_CNT},
		{4, (CHIP_IS_E3B0(bp)) ?
			PBF_REG_INIT_CRD_LB_Q :
			PBF_REG_P4_INIT_CRD,
		    (CHIP_IS_E3B0(bp)) ?
			PBF_REG_CREDIT_LB_Q :
			PBF_REG_P4_CREDIT,
		    (CHIP_IS_E3B0(bp)) ?
			PBF_REG_INTERNAL_CRD_FREED_CNT_LB_Q :
			PBF_REG_P4_INTERNAL_CRD_FREED_CNT},
	};

	int i;

	/* Verify the command queues are flushed P0, P1, P4 */
	for (i = 0; i < ARRAY_SIZE(cmd_regs); i++)
		bnx2x_pbf_pN_cmd_flushed(bp, &cmd_regs[i], poll_count);


	/* Verify the transmission buffers are flushed P0, P1, P4 */
	for (i = 0; i < ARRAY_SIZE(buf_regs); i++)
		bnx2x_pbf_pN_buf_flushed(bp, &buf_regs[i], poll_count);
}

#define OP_GEN_PARAM(param) \
	(((param) << SDM_OP_GEN_COMP_PARAM_SHIFT) & SDM_OP_GEN_COMP_PARAM)

#define OP_GEN_TYPE(type) \
	(((type) << SDM_OP_GEN_COMP_TYPE_SHIFT) & SDM_OP_GEN_COMP_TYPE)

#define OP_GEN_AGG_VECT(index) \
	(((index) << SDM_OP_GEN_AGG_VECT_IDX_SHIFT) & SDM_OP_GEN_AGG_VECT_IDX)


static inline int bnx2x_send_final_clnup(struct bnx2x *bp, u8 clnup_func,
					 u32 poll_cnt)
{
	struct sdm_op_gen op_gen = {0};

	u32 comp_addr = BAR_CSTRORM_INTMEM +
			CSTORM_FINAL_CLEANUP_COMPLETE_OFFSET(clnup_func);
	int ret = 0;

	if (REG_RD(bp, comp_addr)) {
1184
		BNX2X_ERR("Cleanup complete was not 0 before sending\n");
1185 1186 1187 1188 1189 1190 1191 1192
		return 1;
	}

	op_gen.command |= OP_GEN_PARAM(XSTORM_AGG_INT_FINAL_CLEANUP_INDEX);
	op_gen.command |= OP_GEN_TYPE(XSTORM_AGG_INT_FINAL_CLEANUP_COMP_TYPE);
	op_gen.command |= OP_GEN_AGG_VECT(clnup_func);
	op_gen.command |= 1 << SDM_OP_GEN_AGG_VECT_IDX_VALID_SHIFT;

1193
	DP(BNX2X_MSG_SP, "sending FW Final cleanup\n");
1194 1195 1196 1197
	REG_WR(bp, XSDM_REG_OPERATION_GEN, op_gen.command);

	if (bnx2x_flr_clnup_reg_poll(bp, comp_addr, 1, poll_cnt) != 1) {
		BNX2X_ERR("FW final cleanup did not succeed\n");
M
Merav Sicron 已提交
1198 1199
		DP(BNX2X_MSG_SP, "At timeout completion address contained %x\n",
		   (REG_RD(bp, comp_addr)));
1200 1201 1202 1203 1204 1205 1206 1207 1208 1209 1210 1211 1212
		ret = 1;
	}
	/* Zero completion for nxt FLR */
	REG_WR(bp, comp_addr, 0);

	return ret;
}

static inline u8 bnx2x_is_pcie_pending(struct pci_dev *dev)
{
	int pos;
	u16 status;

1213
	pos = pci_pcie_cap(dev);
1214 1215 1216 1217 1218 1219 1220 1221 1222 1223 1224 1225 1226 1227 1228 1229 1230 1231 1232 1233 1234 1235 1236 1237 1238 1239 1240 1241 1242 1243 1244 1245 1246 1247 1248 1249 1250 1251 1252 1253 1254 1255 1256 1257 1258 1259 1260 1261 1262 1263 1264 1265 1266 1267 1268 1269 1270 1271 1272 1273 1274 1275 1276 1277 1278 1279 1280 1281 1282 1283 1284 1285 1286 1287 1288 1289 1290 1291 1292 1293 1294 1295 1296 1297 1298 1299 1300 1301 1302 1303 1304 1305 1306 1307 1308 1309
	if (!pos)
		return false;

	pci_read_config_word(dev, pos + PCI_EXP_DEVSTA, &status);
	return status & PCI_EXP_DEVSTA_TRPND;
}

/* PF FLR specific routines
*/
static int bnx2x_poll_hw_usage_counters(struct bnx2x *bp, u32 poll_cnt)
{

	/* wait for CFC PF usage-counter to zero (includes all the VFs) */
	if (bnx2x_flr_clnup_poll_hw_counter(bp,
			CFC_REG_NUM_LCIDS_INSIDE_PF,
			"CFC PF usage counter timed out",
			poll_cnt))
		return 1;


	/* Wait for DQ PF usage-counter to zero (until DQ cleanup) */
	if (bnx2x_flr_clnup_poll_hw_counter(bp,
			DORQ_REG_PF_USAGE_CNT,
			"DQ PF usage counter timed out",
			poll_cnt))
		return 1;

	/* Wait for QM PF usage-counter to zero (until DQ cleanup) */
	if (bnx2x_flr_clnup_poll_hw_counter(bp,
			QM_REG_PF_USG_CNT_0 + 4*BP_FUNC(bp),
			"QM PF usage counter timed out",
			poll_cnt))
		return 1;

	/* Wait for Timer PF usage-counters to zero (until DQ cleanup) */
	if (bnx2x_flr_clnup_poll_hw_counter(bp,
			TM_REG_LIN0_VNIC_UC + 4*BP_PORT(bp),
			"Timers VNIC usage counter timed out",
			poll_cnt))
		return 1;
	if (bnx2x_flr_clnup_poll_hw_counter(bp,
			TM_REG_LIN0_NUM_SCANS + 4*BP_PORT(bp),
			"Timers NUM_SCANS usage counter timed out",
			poll_cnt))
		return 1;

	/* Wait DMAE PF usage counter to zero */
	if (bnx2x_flr_clnup_poll_hw_counter(bp,
			dmae_reg_go_c[INIT_DMAE_C(bp)],
			"DMAE dommand register timed out",
			poll_cnt))
		return 1;

	return 0;
}

static void bnx2x_hw_enable_status(struct bnx2x *bp)
{
	u32 val;

	val = REG_RD(bp, CFC_REG_WEAK_ENABLE_PF);
	DP(BNX2X_MSG_SP, "CFC_REG_WEAK_ENABLE_PF is 0x%x\n", val);

	val = REG_RD(bp, PBF_REG_DISABLE_PF);
	DP(BNX2X_MSG_SP, "PBF_REG_DISABLE_PF is 0x%x\n", val);

	val = REG_RD(bp, IGU_REG_PCI_PF_MSI_EN);
	DP(BNX2X_MSG_SP, "IGU_REG_PCI_PF_MSI_EN is 0x%x\n", val);

	val = REG_RD(bp, IGU_REG_PCI_PF_MSIX_EN);
	DP(BNX2X_MSG_SP, "IGU_REG_PCI_PF_MSIX_EN is 0x%x\n", val);

	val = REG_RD(bp, IGU_REG_PCI_PF_MSIX_FUNC_MASK);
	DP(BNX2X_MSG_SP, "IGU_REG_PCI_PF_MSIX_FUNC_MASK is 0x%x\n", val);

	val = REG_RD(bp, PGLUE_B_REG_SHADOW_BME_PF_7_0_CLR);
	DP(BNX2X_MSG_SP, "PGLUE_B_REG_SHADOW_BME_PF_7_0_CLR is 0x%x\n", val);

	val = REG_RD(bp, PGLUE_B_REG_FLR_REQUEST_PF_7_0_CLR);
	DP(BNX2X_MSG_SP, "PGLUE_B_REG_FLR_REQUEST_PF_7_0_CLR is 0x%x\n", val);

	val = REG_RD(bp, PGLUE_B_REG_INTERNAL_PFID_ENABLE_MASTER);
	DP(BNX2X_MSG_SP, "PGLUE_B_REG_INTERNAL_PFID_ENABLE_MASTER is 0x%x\n",
	   val);
}

static int bnx2x_pf_flr_clnup(struct bnx2x *bp)
{
	u32 poll_cnt = bnx2x_flr_clnup_poll_count(bp);

	DP(BNX2X_MSG_SP, "Cleanup after FLR PF[%d]\n", BP_ABS_FUNC(bp));

	/* Re-enable PF target read access */
	REG_WR(bp, PGLUE_B_REG_INTERNAL_PFID_ENABLE_TARGET_READ, 1);

	/* Poll HW usage counters */
1310
	DP(BNX2X_MSG_SP, "Polling usage counters\n");
1311 1312 1313 1314 1315 1316 1317 1318 1319 1320 1321 1322 1323 1324 1325 1326 1327 1328 1329 1330 1331 1332 1333 1334 1335 1336 1337 1338 1339 1340 1341 1342 1343
	if (bnx2x_poll_hw_usage_counters(bp, poll_cnt))
		return -EBUSY;

	/* Zero the igu 'trailing edge' and 'leading edge' */

	/* Send the FW cleanup command */
	if (bnx2x_send_final_clnup(bp, (u8)BP_FUNC(bp), poll_cnt))
		return -EBUSY;

	/* ATC cleanup */

	/* Verify TX hw is flushed */
	bnx2x_tx_hw_flushed(bp, poll_cnt);

	/* Wait 100ms (not adjusted according to platform) */
	msleep(100);

	/* Verify no pending pci transactions */
	if (bnx2x_is_pcie_pending(bp->pdev))
		BNX2X_ERR("PCIE Transactions still pending\n");

	/* Debug */
	bnx2x_hw_enable_status(bp);

	/*
	 * Master enable - Due to WB DMAE writes performed before this
	 * register is re-initialized as part of the regular function init
	 */
	REG_WR(bp, PGLUE_B_REG_INTERNAL_PFID_ENABLE_MASTER, 1);

	return 0;
}

D
Dmitry Kravkov 已提交
1344
static void bnx2x_hc_int_enable(struct bnx2x *bp)
E
Eliezer Tamir 已提交
1345
{
1346
	int port = BP_PORT(bp);
E
Eliezer Tamir 已提交
1347 1348 1349
	u32 addr = port ? HC_REG_CONFIG_1 : HC_REG_CONFIG_0;
	u32 val = REG_RD(bp, addr);
	int msix = (bp->flags & USING_MSIX_FLAG) ? 1 : 0;
E
Eilon Greenstein 已提交
1350
	int msi = (bp->flags & USING_MSI_FLAG) ? 1 : 0;
E
Eliezer Tamir 已提交
1351 1352

	if (msix) {
E
Eilon Greenstein 已提交
1353 1354
		val &= ~(HC_CONFIG_0_REG_SINGLE_ISR_EN_0 |
			 HC_CONFIG_0_REG_INT_LINE_EN_0);
E
Eliezer Tamir 已提交
1355 1356
		val |= (HC_CONFIG_0_REG_MSI_MSIX_INT_EN_0 |
			HC_CONFIG_0_REG_ATTN_BIT_EN_0);
E
Eilon Greenstein 已提交
1357 1358 1359 1360 1361
	} else if (msi) {
		val &= ~HC_CONFIG_0_REG_INT_LINE_EN_0;
		val |= (HC_CONFIG_0_REG_SINGLE_ISR_EN_0 |
			HC_CONFIG_0_REG_MSI_MSIX_INT_EN_0 |
			HC_CONFIG_0_REG_ATTN_BIT_EN_0);
E
Eliezer Tamir 已提交
1362 1363
	} else {
		val |= (HC_CONFIG_0_REG_SINGLE_ISR_EN_0 |
E
Eliezer Tamir 已提交
1364
			HC_CONFIG_0_REG_MSI_MSIX_INT_EN_0 |
E
Eliezer Tamir 已提交
1365 1366
			HC_CONFIG_0_REG_INT_LINE_EN_0 |
			HC_CONFIG_0_REG_ATTN_BIT_EN_0);
E
Eliezer Tamir 已提交
1367

1368
		if (!CHIP_IS_E1(bp)) {
M
Merav Sicron 已提交
1369 1370
			DP(NETIF_MSG_IFUP,
			   "write %x to HC %d (addr 0x%x)\n", val, port, addr);
E
Eliezer Tamir 已提交
1371

1372
			REG_WR(bp, addr, val);
E
Eliezer Tamir 已提交
1373

1374 1375
			val &= ~HC_CONFIG_0_REG_MSI_MSIX_INT_EN_0;
		}
E
Eliezer Tamir 已提交
1376 1377
	}

1378 1379 1380
	if (CHIP_IS_E1(bp))
		REG_WR(bp, HC_REG_INT_MASK + port*4, 0x1FFFF);

M
Merav Sicron 已提交
1381 1382 1383
	DP(NETIF_MSG_IFUP,
	   "write %x to HC %d (addr 0x%x) mode %s\n", val, port, addr,
	   (msix ? "MSI-X" : (msi ? "MSI" : "INTx")));
E
Eliezer Tamir 已提交
1384 1385

	REG_WR(bp, addr, val);
E
Eilon Greenstein 已提交
1386 1387 1388 1389 1390
	/*
	 * Ensure that HC_CONFIG is written before leading/trailing edge config
	 */
	mmiowb();
	barrier();
1391

D
Dmitry Kravkov 已提交
1392
	if (!CHIP_IS_E1(bp)) {
1393
		/* init leading/trailing edge */
D
Dmitry Kravkov 已提交
1394
		if (IS_MF(bp)) {
1395
			val = (0xee0f | (1 << (BP_VN(bp) + 4)));
1396
			if (bp->port.pmf)
E
Eilon Greenstein 已提交
1397 1398
				/* enable nig and gpio3 attention */
				val |= 0x1100;
1399 1400 1401 1402 1403 1404
		} else
			val = 0xffff;

		REG_WR(bp, HC_REG_TRAILING_EDGE_0 + port*8, val);
		REG_WR(bp, HC_REG_LEADING_EDGE_0 + port*8, val);
	}
E
Eilon Greenstein 已提交
1405 1406 1407

	/* Make sure that interrupts are indeed enabled from here on */
	mmiowb();
E
Eliezer Tamir 已提交
1408 1409
}

D
Dmitry Kravkov 已提交
1410 1411 1412 1413 1414 1415 1416 1417 1418 1419 1420 1421 1422 1423 1424 1425 1426 1427 1428 1429 1430 1431 1432 1433 1434 1435 1436 1437
static void bnx2x_igu_int_enable(struct bnx2x *bp)
{
	u32 val;
	int msix = (bp->flags & USING_MSIX_FLAG) ? 1 : 0;
	int msi = (bp->flags & USING_MSI_FLAG) ? 1 : 0;

	val = REG_RD(bp, IGU_REG_PF_CONFIGURATION);

	if (msix) {
		val &= ~(IGU_PF_CONF_INT_LINE_EN |
			 IGU_PF_CONF_SINGLE_ISR_EN);
		val |= (IGU_PF_CONF_FUNC_EN |
			IGU_PF_CONF_MSI_MSIX_EN |
			IGU_PF_CONF_ATTN_BIT_EN);
	} else if (msi) {
		val &= ~IGU_PF_CONF_INT_LINE_EN;
		val |= (IGU_PF_CONF_FUNC_EN |
			IGU_PF_CONF_MSI_MSIX_EN |
			IGU_PF_CONF_ATTN_BIT_EN |
			IGU_PF_CONF_SINGLE_ISR_EN);
	} else {
		val &= ~IGU_PF_CONF_MSI_MSIX_EN;
		val |= (IGU_PF_CONF_FUNC_EN |
			IGU_PF_CONF_INT_LINE_EN |
			IGU_PF_CONF_ATTN_BIT_EN |
			IGU_PF_CONF_SINGLE_ISR_EN);
	}

M
Merav Sicron 已提交
1438
	DP(NETIF_MSG_IFUP, "write 0x%x to IGU  mode %s\n",
D
Dmitry Kravkov 已提交
1439 1440 1441 1442 1443 1444 1445 1446
	   val, (msix ? "MSI-X" : (msi ? "MSI" : "INTx")));

	REG_WR(bp, IGU_REG_PF_CONFIGURATION, val);

	barrier();

	/* init leading/trailing edge */
	if (IS_MF(bp)) {
1447
		val = (0xee0f | (1 << (BP_VN(bp) + 4)));
D
Dmitry Kravkov 已提交
1448 1449 1450 1451 1452 1453 1454 1455 1456 1457 1458 1459 1460 1461 1462 1463 1464 1465 1466 1467 1468 1469
		if (bp->port.pmf)
			/* enable nig and gpio3 attention */
			val |= 0x1100;
	} else
		val = 0xffff;

	REG_WR(bp, IGU_REG_TRAILING_EDGE_LATCH, val);
	REG_WR(bp, IGU_REG_LEADING_EDGE_LATCH, val);

	/* Make sure that interrupts are indeed enabled from here on */
	mmiowb();
}

void bnx2x_int_enable(struct bnx2x *bp)
{
	if (bp->common.int_block == INT_BLOCK_HC)
		bnx2x_hc_int_enable(bp);
	else
		bnx2x_igu_int_enable(bp);
}

static void bnx2x_hc_int_disable(struct bnx2x *bp)
E
Eliezer Tamir 已提交
1470
{
1471
	int port = BP_PORT(bp);
E
Eliezer Tamir 已提交
1472 1473 1474
	u32 addr = port ? HC_REG_CONFIG_1 : HC_REG_CONFIG_0;
	u32 val = REG_RD(bp, addr);

1475 1476 1477 1478 1479 1480 1481 1482 1483 1484 1485 1486 1487 1488 1489 1490 1491 1492 1493 1494
	/*
	 * in E1 we must use only PCI configuration space to disable
	 * MSI/MSIX capablility
	 * It's forbitten to disable IGU_PF_CONF_MSI_MSIX_EN in HC block
	 */
	if (CHIP_IS_E1(bp)) {
		/*  Since IGU_PF_CONF_MSI_MSIX_EN still always on
		 *  Use mask register to prevent from HC sending interrupts
		 *  after we exit the function
		 */
		REG_WR(bp, HC_REG_INT_MASK + port*4, 0);

		val &= ~(HC_CONFIG_0_REG_SINGLE_ISR_EN_0 |
			 HC_CONFIG_0_REG_INT_LINE_EN_0 |
			 HC_CONFIG_0_REG_ATTN_BIT_EN_0);
	} else
		val &= ~(HC_CONFIG_0_REG_SINGLE_ISR_EN_0 |
			 HC_CONFIG_0_REG_MSI_MSIX_INT_EN_0 |
			 HC_CONFIG_0_REG_INT_LINE_EN_0 |
			 HC_CONFIG_0_REG_ATTN_BIT_EN_0);
E
Eliezer Tamir 已提交
1495

M
Merav Sicron 已提交
1496 1497
	DP(NETIF_MSG_IFDOWN,
	   "write %x to HC %d (addr 0x%x)\n",
E
Eliezer Tamir 已提交
1498 1499
	   val, port, addr);

E
Eilon Greenstein 已提交
1500 1501 1502
	/* flush all outstanding writes */
	mmiowb();

E
Eliezer Tamir 已提交
1503 1504 1505 1506 1507
	REG_WR(bp, addr, val);
	if (REG_RD(bp, addr) != val)
		BNX2X_ERR("BUG! proper val not read from IGU!\n");
}

D
Dmitry Kravkov 已提交
1508 1509 1510 1511 1512 1513 1514 1515
static void bnx2x_igu_int_disable(struct bnx2x *bp)
{
	u32 val = REG_RD(bp, IGU_REG_PF_CONFIGURATION);

	val &= ~(IGU_PF_CONF_MSI_MSIX_EN |
		 IGU_PF_CONF_INT_LINE_EN |
		 IGU_PF_CONF_ATTN_BIT_EN);

M
Merav Sicron 已提交
1516
	DP(NETIF_MSG_IFDOWN, "write %x to IGU\n", val);
D
Dmitry Kravkov 已提交
1517 1518 1519 1520 1521 1522 1523 1524 1525

	/* flush all outstanding writes */
	mmiowb();

	REG_WR(bp, IGU_REG_PF_CONFIGURATION, val);
	if (REG_RD(bp, IGU_REG_PF_CONFIGURATION) != val)
		BNX2X_ERR("BUG! proper val not read from IGU!\n");
}

1526
void bnx2x_int_disable(struct bnx2x *bp)
D
Dmitry Kravkov 已提交
1527 1528 1529 1530 1531 1532 1533
{
	if (bp->common.int_block == INT_BLOCK_HC)
		bnx2x_hc_int_disable(bp);
	else
		bnx2x_igu_int_disable(bp);
}

D
Dmitry Kravkov 已提交
1534
void bnx2x_int_disable_sync(struct bnx2x *bp, int disable_hw)
E
Eliezer Tamir 已提交
1535 1536
{
	int msix = (bp->flags & USING_MSIX_FLAG) ? 1 : 0;
E
Eilon Greenstein 已提交
1537
	int i, offset;
E
Eliezer Tamir 已提交
1538

Y
Yitchak Gertner 已提交
1539 1540 1541
	if (disable_hw)
		/* prevent the HW from sending interrupts */
		bnx2x_int_disable(bp);
E
Eliezer Tamir 已提交
1542 1543 1544

	/* make sure all ISRs are done */
	if (msix) {
E
Eilon Greenstein 已提交
1545 1546
		synchronize_irq(bp->msix_table[0].vector);
		offset = 1;
1547 1548 1549
#ifdef BCM_CNIC
		offset++;
#endif
V
Vladislav Zolotarov 已提交
1550
		for_each_eth_queue(bp, i)
D
Dmitry Kravkov 已提交
1551
			synchronize_irq(bp->msix_table[offset++].vector);
E
Eliezer Tamir 已提交
1552 1553 1554 1555
	} else
		synchronize_irq(bp->pdev->irq);

	/* make sure sp_task is not running */
1556
	cancel_delayed_work(&bp->sp_task);
1557
	cancel_delayed_work(&bp->period_task);
1558
	flush_workqueue(bnx2x_wq);
E
Eliezer Tamir 已提交
1559 1560
}

1561
/* fast path */
E
Eliezer Tamir 已提交
1562 1563

/*
1564
 * General service functions
E
Eliezer Tamir 已提交
1565 1566
 */

1567 1568 1569 1570 1571 1572 1573 1574
/* Return true if succeeded to acquire the lock */
static bool bnx2x_trylock_hw_lock(struct bnx2x *bp, u32 resource)
{
	u32 lock_status;
	u32 resource_bit = (1 << resource);
	int func = BP_FUNC(bp);
	u32 hw_lock_control_reg;

M
Merav Sicron 已提交
1575 1576
	DP(NETIF_MSG_HW | NETIF_MSG_IFUP,
	   "Trying to take a lock on resource %d\n", resource);
1577 1578 1579

	/* Validating that the resource is within range */
	if (resource > HW_LOCK_MAX_RESOURCE_VALUE) {
M
Merav Sicron 已提交
1580
		DP(NETIF_MSG_HW | NETIF_MSG_IFUP,
1581 1582
		   "resource(0x%x) > HW_LOCK_MAX_RESOURCE_VALUE(0x%x)\n",
		   resource, HW_LOCK_MAX_RESOURCE_VALUE);
1583
		return false;
1584 1585 1586 1587 1588 1589 1590 1591 1592 1593 1594 1595 1596 1597
	}

	if (func <= 5)
		hw_lock_control_reg = (MISC_REG_DRIVER_CONTROL_1 + func*8);
	else
		hw_lock_control_reg =
				(MISC_REG_DRIVER_CONTROL_7 + (func - 6)*8);

	/* Try to acquire the lock */
	REG_WR(bp, hw_lock_control_reg + 4, resource_bit);
	lock_status = REG_RD(bp, hw_lock_control_reg);
	if (lock_status & resource_bit)
		return true;

M
Merav Sicron 已提交
1598 1599
	DP(NETIF_MSG_HW | NETIF_MSG_IFUP,
	   "Failed to get a lock on resource %d\n", resource);
1600 1601 1602
	return false;
}

1603 1604 1605 1606 1607 1608 1609 1610 1611 1612 1613 1614 1615 1616 1617 1618 1619 1620 1621 1622 1623 1624 1625 1626 1627 1628 1629 1630
/**
 * bnx2x_get_leader_lock_resource - get the recovery leader resource id
 *
 * @bp:	driver handle
 *
 * Returns the recovery leader resource id according to the engine this function
 * belongs to. Currently only only 2 engines is supported.
 */
static inline int bnx2x_get_leader_lock_resource(struct bnx2x *bp)
{
	if (BP_PATH(bp))
		return HW_LOCK_RESOURCE_RECOVERY_LEADER_1;
	else
		return HW_LOCK_RESOURCE_RECOVERY_LEADER_0;
}

/**
 * bnx2x_trylock_leader_lock- try to aquire a leader lock.
 *
 * @bp: driver handle
 *
 * Tries to aquire a leader lock for cuurent engine.
 */
static inline bool bnx2x_trylock_leader_lock(struct bnx2x *bp)
{
	return bnx2x_trylock_hw_lock(bp, bnx2x_get_leader_lock_resource(bp));
}

1631
#ifdef BCM_CNIC
1632
static void bnx2x_cnic_cfc_comp(struct bnx2x *bp, int cid, u8 err);
1633
#endif
1634

1635
void bnx2x_sp_event(struct bnx2x_fastpath *fp, union eth_rx_cqe *rr_cqe)
E
Eliezer Tamir 已提交
1636 1637 1638 1639
{
	struct bnx2x *bp = fp->bp;
	int cid = SW_CID(rr_cqe->ramrod_cqe.conn_and_cmd_data);
	int command = CQE_CMD(rr_cqe->ramrod_cqe.conn_and_cmd_data);
1640 1641
	enum bnx2x_queue_cmd drv_cmd = BNX2X_Q_CMD_MAX;
	struct bnx2x_queue_sp_obj *q_obj = &fp->q_obj;
E
Eliezer Tamir 已提交
1642

1643
	DP(BNX2X_MSG_SP,
E
Eliezer Tamir 已提交
1644
	   "fp %d  cid %d  got ramrod #%d  state is %x  type is %d\n",
1645
	   fp->index, cid, command, bp->state,
1646
	   rr_cqe->ramrod_cqe.ramrod_type);
E
Eliezer Tamir 已提交
1647

1648 1649
	switch (command) {
	case (RAMROD_CMD_ID_ETH_CLIENT_UPDATE):
1650
		DP(BNX2X_MSG_SP, "got UPDATE ramrod. CID %d\n", cid);
1651 1652
		drv_cmd = BNX2X_Q_CMD_UPDATE;
		break;
1653

1654
	case (RAMROD_CMD_ID_ETH_CLIENT_SETUP):
1655
		DP(BNX2X_MSG_SP, "got MULTI[%d] setup ramrod\n", cid);
1656
		drv_cmd = BNX2X_Q_CMD_SETUP;
E
Eliezer Tamir 已提交
1657 1658
		break;

1659
	case (RAMROD_CMD_ID_ETH_TX_QUEUE_SETUP):
M
Merav Sicron 已提交
1660
		DP(BNX2X_MSG_SP, "got MULTI[%d] tx-only setup ramrod\n", cid);
1661 1662 1663
		drv_cmd = BNX2X_Q_CMD_SETUP_TX_ONLY;
		break;

1664
	case (RAMROD_CMD_ID_ETH_HALT):
1665
		DP(BNX2X_MSG_SP, "got MULTI[%d] halt ramrod\n", cid);
1666
		drv_cmd = BNX2X_Q_CMD_HALT;
E
Eliezer Tamir 已提交
1667 1668
		break;

1669
	case (RAMROD_CMD_ID_ETH_TERMINATE):
1670
		DP(BNX2X_MSG_SP, "got MULTI[%d] teminate ramrod\n", cid);
1671
		drv_cmd = BNX2X_Q_CMD_TERMINATE;
E
Eliezer Tamir 已提交
1672 1673
		break;

1674
	case (RAMROD_CMD_ID_ETH_EMPTY):
1675
		DP(BNX2X_MSG_SP, "got MULTI[%d] empty ramrod\n", cid);
1676
		drv_cmd = BNX2X_Q_CMD_EMPTY;
1677
		break;
1678 1679 1680 1681 1682

	default:
		BNX2X_ERR("unexpected MC reply (%d) on fp[%d]\n",
			  command, fp->index);
		return;
1683
	}
1684

1685 1686 1687 1688 1689 1690 1691 1692 1693 1694 1695 1696 1697 1698 1699
	if ((drv_cmd != BNX2X_Q_CMD_MAX) &&
	    q_obj->complete_cmd(bp, q_obj, drv_cmd))
		/* q_obj->complete_cmd() failure means that this was
		 * an unexpected completion.
		 *
		 * In this case we don't want to increase the bp->spq_left
		 * because apparently we haven't sent this command the first
		 * place.
		 */
#ifdef BNX2X_STOP_ON_ERROR
		bnx2x_panic();
#else
		return;
#endif

1700
	smp_mb__before_atomic_inc();
1701
	atomic_inc(&bp->cq_spq_left);
1702 1703
	/* push the change in bp->spq_left and towards the memory */
	smp_mb__after_atomic_inc();
1704

1705 1706
	DP(BNX2X_MSG_SP, "bp->cq_spq_left %x\n", atomic_read(&bp->cq_spq_left));

1707
	return;
E
Eliezer Tamir 已提交
1708 1709
}

1710 1711 1712 1713 1714 1715 1716 1717 1718
void bnx2x_update_rx_prod(struct bnx2x *bp, struct bnx2x_fastpath *fp,
			u16 bd_prod, u16 rx_comp_prod, u16 rx_sge_prod)
{
	u32 start = BAR_USTRORM_INTMEM + fp->ustorm_rx_prods_offset;

	bnx2x_update_rx_prod_gen(bp, fp, bd_prod, rx_comp_prod, rx_sge_prod,
				 start);
}

D
Dmitry Kravkov 已提交
1719
irqreturn_t bnx2x_interrupt(int irq, void *dev_instance)
E
Eliezer Tamir 已提交
1720
{
E
Eilon Greenstein 已提交
1721
	struct bnx2x *bp = netdev_priv(dev_instance);
E
Eliezer Tamir 已提交
1722
	u16 status = bnx2x_ack_int(bp);
1723
	u16 mask;
E
Eilon Greenstein 已提交
1724
	int i;
1725
	u8 cos;
E
Eliezer Tamir 已提交
1726

1727
	/* Return here if interrupt is shared and it's not for us */
E
Eliezer Tamir 已提交
1728 1729 1730 1731
	if (unlikely(status == 0)) {
		DP(NETIF_MSG_INTR, "not our interrupt!\n");
		return IRQ_NONE;
	}
E
Eilon Greenstein 已提交
1732
	DP(NETIF_MSG_INTR, "got an interrupt  status 0x%x\n", status);
E
Eliezer Tamir 已提交
1733

1734 1735 1736 1737 1738
#ifdef BNX2X_STOP_ON_ERROR
	if (unlikely(bp->panic))
		return IRQ_HANDLED;
#endif

V
Vladislav Zolotarov 已提交
1739
	for_each_eth_queue(bp, i) {
E
Eilon Greenstein 已提交
1740
		struct bnx2x_fastpath *fp = &bp->fp[i];
E
Eliezer Tamir 已提交
1741

1742
		mask = 0x2 << (fp->index + CNIC_PRESENT);
E
Eilon Greenstein 已提交
1743
		if (status & mask) {
1744
			/* Handle Rx or Tx according to SB id */
1745
			prefetch(fp->rx_cons_sb);
1746 1747
			for_each_cos_in_tx_queue(fp, cos)
				prefetch(fp->txdata[cos].tx_cons_sb);
1748
			prefetch(&fp->sb_running_index[SM_RX_ID]);
1749
			napi_schedule(&bnx2x_fp(bp, fp->index, napi));
E
Eilon Greenstein 已提交
1750 1751
			status &= ~mask;
		}
E
Eliezer Tamir 已提交
1752 1753
	}

1754
#ifdef BCM_CNIC
1755
	mask = 0x2;
1756 1757 1758
	if (status & (mask | 0x1)) {
		struct cnic_ops *c_ops = NULL;

1759 1760 1761 1762 1763 1764 1765
		if (likely(bp->state == BNX2X_STATE_OPEN)) {
			rcu_read_lock();
			c_ops = rcu_dereference(bp->cnic_ops);
			if (c_ops)
				c_ops->cnic_handler(bp->cnic_data, NULL);
			rcu_read_unlock();
		}
1766 1767 1768 1769

		status &= ~mask;
	}
#endif
E
Eliezer Tamir 已提交
1770

1771
	if (unlikely(status & 0x1)) {
1772
		queue_delayed_work(bnx2x_wq, &bp->sp_task, 0);
E
Eliezer Tamir 已提交
1773 1774 1775 1776 1777 1778

		status &= ~0x1;
		if (!status)
			return IRQ_HANDLED;
	}

V
Vladislav Zolotarov 已提交
1779 1780
	if (unlikely(status))
		DP(NETIF_MSG_INTR, "got an unknown interrupt! (status 0x%x)\n",
1781
		   status);
E
Eliezer Tamir 已提交
1782

Y
Yaniv Rosner 已提交
1783
	return IRQ_HANDLED;
E
Eliezer Tamir 已提交
1784 1785
}

Y
Yaniv Rosner 已提交
1786 1787 1788 1789 1790
/* Link */

/*
 * General service functions
 */
E
Eliezer Tamir 已提交
1791

D
Dmitry Kravkov 已提交
1792
int bnx2x_acquire_hw_lock(struct bnx2x *bp, u32 resource)
Y
Yaniv Rosner 已提交
1793 1794 1795
{
	u32 lock_status;
	u32 resource_bit = (1 << resource);
Y
Yitchak Gertner 已提交
1796 1797
	int func = BP_FUNC(bp);
	u32 hw_lock_control_reg;
Y
Yaniv Rosner 已提交
1798
	int cnt;
E
Eliezer Tamir 已提交
1799

Y
Yaniv Rosner 已提交
1800 1801
	/* Validating that the resource is within range */
	if (resource > HW_LOCK_MAX_RESOURCE_VALUE) {
M
Merav Sicron 已提交
1802
		BNX2X_ERR("resource(0x%x) > HW_LOCK_MAX_RESOURCE_VALUE(0x%x)\n",
Y
Yaniv Rosner 已提交
1803 1804 1805
		   resource, HW_LOCK_MAX_RESOURCE_VALUE);
		return -EINVAL;
	}
E
Eliezer Tamir 已提交
1806

Y
Yitchak Gertner 已提交
1807 1808 1809 1810 1811 1812 1813
	if (func <= 5) {
		hw_lock_control_reg = (MISC_REG_DRIVER_CONTROL_1 + func*8);
	} else {
		hw_lock_control_reg =
				(MISC_REG_DRIVER_CONTROL_7 + (func - 6)*8);
	}

Y
Yaniv Rosner 已提交
1814
	/* Validating that the resource is not already taken */
Y
Yitchak Gertner 已提交
1815
	lock_status = REG_RD(bp, hw_lock_control_reg);
Y
Yaniv Rosner 已提交
1816
	if (lock_status & resource_bit) {
M
Merav Sicron 已提交
1817
		BNX2X_ERR("lock_status 0x%x  resource_bit 0x%x\n",
Y
Yaniv Rosner 已提交
1818 1819 1820
		   lock_status, resource_bit);
		return -EEXIST;
	}
E
Eliezer Tamir 已提交
1821

E
Eilon Greenstein 已提交
1822 1823
	/* Try for 5 second every 5ms */
	for (cnt = 0; cnt < 1000; cnt++) {
Y
Yaniv Rosner 已提交
1824
		/* Try to acquire the lock */
Y
Yitchak Gertner 已提交
1825 1826
		REG_WR(bp, hw_lock_control_reg + 4, resource_bit);
		lock_status = REG_RD(bp, hw_lock_control_reg);
Y
Yaniv Rosner 已提交
1827 1828
		if (lock_status & resource_bit)
			return 0;
E
Eliezer Tamir 已提交
1829

Y
Yaniv Rosner 已提交
1830
		msleep(5);
E
Eliezer Tamir 已提交
1831
	}
M
Merav Sicron 已提交
1832
	BNX2X_ERR("Timeout\n");
Y
Yaniv Rosner 已提交
1833 1834
	return -EAGAIN;
}
E
Eliezer Tamir 已提交
1835

1836 1837 1838 1839 1840
int bnx2x_release_leader_lock(struct bnx2x *bp)
{
	return bnx2x_release_hw_lock(bp, bnx2x_get_leader_lock_resource(bp));
}

D
Dmitry Kravkov 已提交
1841
int bnx2x_release_hw_lock(struct bnx2x *bp, u32 resource)
Y
Yaniv Rosner 已提交
1842 1843 1844
{
	u32 lock_status;
	u32 resource_bit = (1 << resource);
Y
Yitchak Gertner 已提交
1845 1846
	int func = BP_FUNC(bp);
	u32 hw_lock_control_reg;
E
Eliezer Tamir 已提交
1847

Y
Yaniv Rosner 已提交
1848 1849
	/* Validating that the resource is within range */
	if (resource > HW_LOCK_MAX_RESOURCE_VALUE) {
M
Merav Sicron 已提交
1850
		BNX2X_ERR("resource(0x%x) > HW_LOCK_MAX_RESOURCE_VALUE(0x%x)\n",
Y
Yaniv Rosner 已提交
1851 1852 1853 1854
		   resource, HW_LOCK_MAX_RESOURCE_VALUE);
		return -EINVAL;
	}

Y
Yitchak Gertner 已提交
1855 1856 1857 1858 1859 1860 1861
	if (func <= 5) {
		hw_lock_control_reg = (MISC_REG_DRIVER_CONTROL_1 + func*8);
	} else {
		hw_lock_control_reg =
				(MISC_REG_DRIVER_CONTROL_7 + (func - 6)*8);
	}

Y
Yaniv Rosner 已提交
1862
	/* Validating that the resource is currently taken */
Y
Yitchak Gertner 已提交
1863
	lock_status = REG_RD(bp, hw_lock_control_reg);
Y
Yaniv Rosner 已提交
1864
	if (!(lock_status & resource_bit)) {
M
Merav Sicron 已提交
1865
		BNX2X_ERR("lock_status 0x%x resource_bit 0x%x. unlock was called but lock wasn't taken!\n",
Y
Yaniv Rosner 已提交
1866 1867
		   lock_status, resource_bit);
		return -EFAULT;
E
Eliezer Tamir 已提交
1868 1869
	}

D
Dmitry Kravkov 已提交
1870 1871
	REG_WR(bp, hw_lock_control_reg, resource_bit);
	return 0;
Y
Yaniv Rosner 已提交
1872
}
E
Eliezer Tamir 已提交
1873

D
Dmitry Kravkov 已提交
1874

E
Eilon Greenstein 已提交
1875 1876 1877 1878 1879 1880 1881 1882 1883 1884 1885 1886 1887 1888 1889 1890 1891 1892 1893 1894 1895 1896 1897 1898 1899 1900 1901 1902 1903 1904
int bnx2x_get_gpio(struct bnx2x *bp, int gpio_num, u8 port)
{
	/* The GPIO should be swapped if swap register is set and active */
	int gpio_port = (REG_RD(bp, NIG_REG_PORT_SWAP) &&
			 REG_RD(bp, NIG_REG_STRAP_OVERRIDE)) ^ port;
	int gpio_shift = gpio_num +
			(gpio_port ? MISC_REGISTERS_GPIO_PORT_SHIFT : 0);
	u32 gpio_mask = (1 << gpio_shift);
	u32 gpio_reg;
	int value;

	if (gpio_num > MISC_REGISTERS_GPIO_3) {
		BNX2X_ERR("Invalid GPIO %d\n", gpio_num);
		return -EINVAL;
	}

	/* read GPIO value */
	gpio_reg = REG_RD(bp, MISC_REG_GPIO);

	/* get the requested pin value */
	if ((gpio_reg & gpio_mask) == gpio_mask)
		value = 1;
	else
		value = 0;

	DP(NETIF_MSG_LINK, "pin %d  value 0x%x\n", gpio_num, value);

	return value;
}

1905
int bnx2x_set_gpio(struct bnx2x *bp, int gpio_num, u32 mode, u8 port)
Y
Yaniv Rosner 已提交
1906 1907 1908
{
	/* The GPIO should be swapped if swap register is set and active */
	int gpio_port = (REG_RD(bp, NIG_REG_PORT_SWAP) &&
1909
			 REG_RD(bp, NIG_REG_STRAP_OVERRIDE)) ^ port;
Y
Yaniv Rosner 已提交
1910 1911 1912 1913
	int gpio_shift = gpio_num +
			(gpio_port ? MISC_REGISTERS_GPIO_PORT_SHIFT : 0);
	u32 gpio_mask = (1 << gpio_shift);
	u32 gpio_reg;
E
Eliezer Tamir 已提交
1914

Y
Yaniv Rosner 已提交
1915 1916 1917 1918
	if (gpio_num > MISC_REGISTERS_GPIO_3) {
		BNX2X_ERR("Invalid GPIO %d\n", gpio_num);
		return -EINVAL;
	}
E
Eliezer Tamir 已提交
1919

Y
Yitchak Gertner 已提交
1920
	bnx2x_acquire_hw_lock(bp, HW_LOCK_RESOURCE_GPIO);
Y
Yaniv Rosner 已提交
1921 1922
	/* read GPIO and mask except the float bits */
	gpio_reg = (REG_RD(bp, MISC_REG_GPIO) & MISC_REGISTERS_GPIO_FLOAT);
E
Eliezer Tamir 已提交
1923

Y
Yaniv Rosner 已提交
1924 1925
	switch (mode) {
	case MISC_REGISTERS_GPIO_OUTPUT_LOW:
M
Merav Sicron 已提交
1926 1927
		DP(NETIF_MSG_LINK,
		   "Set GPIO %d (shift %d) -> output low\n",
Y
Yaniv Rosner 已提交
1928 1929 1930 1931 1932
		   gpio_num, gpio_shift);
		/* clear FLOAT and set CLR */
		gpio_reg &= ~(gpio_mask << MISC_REGISTERS_GPIO_FLOAT_POS);
		gpio_reg |=  (gpio_mask << MISC_REGISTERS_GPIO_CLR_POS);
		break;
E
Eliezer Tamir 已提交
1933

Y
Yaniv Rosner 已提交
1934
	case MISC_REGISTERS_GPIO_OUTPUT_HIGH:
M
Merav Sicron 已提交
1935 1936
		DP(NETIF_MSG_LINK,
		   "Set GPIO %d (shift %d) -> output high\n",
Y
Yaniv Rosner 已提交
1937 1938 1939 1940 1941
		   gpio_num, gpio_shift);
		/* clear FLOAT and set SET */
		gpio_reg &= ~(gpio_mask << MISC_REGISTERS_GPIO_FLOAT_POS);
		gpio_reg |=  (gpio_mask << MISC_REGISTERS_GPIO_SET_POS);
		break;
E
Eliezer Tamir 已提交
1942

1943
	case MISC_REGISTERS_GPIO_INPUT_HI_Z:
M
Merav Sicron 已提交
1944 1945
		DP(NETIF_MSG_LINK,
		   "Set GPIO %d (shift %d) -> input\n",
Y
Yaniv Rosner 已提交
1946 1947 1948 1949
		   gpio_num, gpio_shift);
		/* set FLOAT */
		gpio_reg |= (gpio_mask << MISC_REGISTERS_GPIO_FLOAT_POS);
		break;
E
Eliezer Tamir 已提交
1950

Y
Yaniv Rosner 已提交
1951 1952
	default:
		break;
E
Eliezer Tamir 已提交
1953 1954
	}

Y
Yaniv Rosner 已提交
1955
	REG_WR(bp, MISC_REG_GPIO, gpio_reg);
Y
Yitchak Gertner 已提交
1956
	bnx2x_release_hw_lock(bp, HW_LOCK_RESOURCE_GPIO);
E
Eliezer Tamir 已提交
1957

Y
Yaniv Rosner 已提交
1958
	return 0;
E
Eliezer Tamir 已提交
1959 1960
}

1961 1962 1963 1964 1965 1966 1967 1968 1969 1970 1971 1972 1973 1974 1975 1976 1977 1978 1979 1980 1981 1982 1983 1984 1985 1986 1987 1988 1989 1990 1991 1992 1993 1994 1995 1996 1997 1998 1999 2000 2001 2002 2003 2004 2005 2006 2007
int bnx2x_set_mult_gpio(struct bnx2x *bp, u8 pins, u32 mode)
{
	u32 gpio_reg = 0;
	int rc = 0;

	/* Any port swapping should be handled by caller. */

	bnx2x_acquire_hw_lock(bp, HW_LOCK_RESOURCE_GPIO);
	/* read GPIO and mask except the float bits */
	gpio_reg = REG_RD(bp, MISC_REG_GPIO);
	gpio_reg &= ~(pins << MISC_REGISTERS_GPIO_FLOAT_POS);
	gpio_reg &= ~(pins << MISC_REGISTERS_GPIO_CLR_POS);
	gpio_reg &= ~(pins << MISC_REGISTERS_GPIO_SET_POS);

	switch (mode) {
	case MISC_REGISTERS_GPIO_OUTPUT_LOW:
		DP(NETIF_MSG_LINK, "Set GPIO 0x%x -> output low\n", pins);
		/* set CLR */
		gpio_reg |= (pins << MISC_REGISTERS_GPIO_CLR_POS);
		break;

	case MISC_REGISTERS_GPIO_OUTPUT_HIGH:
		DP(NETIF_MSG_LINK, "Set GPIO 0x%x -> output high\n", pins);
		/* set SET */
		gpio_reg |= (pins << MISC_REGISTERS_GPIO_SET_POS);
		break;

	case MISC_REGISTERS_GPIO_INPUT_HI_Z:
		DP(NETIF_MSG_LINK, "Set GPIO 0x%x -> input\n", pins);
		/* set FLOAT */
		gpio_reg |= (pins << MISC_REGISTERS_GPIO_FLOAT_POS);
		break;

	default:
		BNX2X_ERR("Invalid GPIO mode assignment %d\n", mode);
		rc = -EINVAL;
		break;
	}

	if (rc == 0)
		REG_WR(bp, MISC_REG_GPIO, gpio_reg);

	bnx2x_release_hw_lock(bp, HW_LOCK_RESOURCE_GPIO);

	return rc;
}

E
Eilon Greenstein 已提交
2008 2009 2010 2011 2012 2013 2014 2015 2016 2017 2018 2019 2020 2021 2022 2023 2024 2025 2026 2027 2028
int bnx2x_set_gpio_int(struct bnx2x *bp, int gpio_num, u32 mode, u8 port)
{
	/* The GPIO should be swapped if swap register is set and active */
	int gpio_port = (REG_RD(bp, NIG_REG_PORT_SWAP) &&
			 REG_RD(bp, NIG_REG_STRAP_OVERRIDE)) ^ port;
	int gpio_shift = gpio_num +
			(gpio_port ? MISC_REGISTERS_GPIO_PORT_SHIFT : 0);
	u32 gpio_mask = (1 << gpio_shift);
	u32 gpio_reg;

	if (gpio_num > MISC_REGISTERS_GPIO_3) {
		BNX2X_ERR("Invalid GPIO %d\n", gpio_num);
		return -EINVAL;
	}

	bnx2x_acquire_hw_lock(bp, HW_LOCK_RESOURCE_GPIO);
	/* read GPIO int */
	gpio_reg = REG_RD(bp, MISC_REG_GPIO_INT);

	switch (mode) {
	case MISC_REGISTERS_GPIO_INT_OUTPUT_CLR:
M
Merav Sicron 已提交
2029 2030 2031
		DP(NETIF_MSG_LINK,
		   "Clear GPIO INT %d (shift %d) -> output low\n",
		   gpio_num, gpio_shift);
E
Eilon Greenstein 已提交
2032 2033 2034 2035 2036 2037
		/* clear SET and set CLR */
		gpio_reg &= ~(gpio_mask << MISC_REGISTERS_GPIO_INT_SET_POS);
		gpio_reg |=  (gpio_mask << MISC_REGISTERS_GPIO_INT_CLR_POS);
		break;

	case MISC_REGISTERS_GPIO_INT_OUTPUT_SET:
M
Merav Sicron 已提交
2038 2039 2040
		DP(NETIF_MSG_LINK,
		   "Set GPIO INT %d (shift %d) -> output high\n",
		   gpio_num, gpio_shift);
E
Eilon Greenstein 已提交
2041 2042 2043 2044 2045 2046 2047 2048 2049 2050 2051 2052 2053 2054 2055
		/* clear CLR and set SET */
		gpio_reg &= ~(gpio_mask << MISC_REGISTERS_GPIO_INT_CLR_POS);
		gpio_reg |=  (gpio_mask << MISC_REGISTERS_GPIO_INT_SET_POS);
		break;

	default:
		break;
	}

	REG_WR(bp, MISC_REG_GPIO_INT, gpio_reg);
	bnx2x_release_hw_lock(bp, HW_LOCK_RESOURCE_GPIO);

	return 0;
}

Y
Yaniv Rosner 已提交
2056
static int bnx2x_set_spio(struct bnx2x *bp, int spio_num, u32 mode)
E
Eliezer Tamir 已提交
2057
{
Y
Yaniv Rosner 已提交
2058 2059
	u32 spio_mask = (1 << spio_num);
	u32 spio_reg;
E
Eliezer Tamir 已提交
2060

Y
Yaniv Rosner 已提交
2061 2062 2063 2064
	if ((spio_num < MISC_REGISTERS_SPIO_4) ||
	    (spio_num > MISC_REGISTERS_SPIO_7)) {
		BNX2X_ERR("Invalid SPIO %d\n", spio_num);
		return -EINVAL;
E
Eliezer Tamir 已提交
2065 2066
	}

Y
Yitchak Gertner 已提交
2067
	bnx2x_acquire_hw_lock(bp, HW_LOCK_RESOURCE_SPIO);
Y
Yaniv Rosner 已提交
2068 2069
	/* read SPIO and mask except the float bits */
	spio_reg = (REG_RD(bp, MISC_REG_SPIO) & MISC_REGISTERS_SPIO_FLOAT);
E
Eliezer Tamir 已提交
2070

Y
Yaniv Rosner 已提交
2071
	switch (mode) {
E
Eilon Greenstein 已提交
2072
	case MISC_REGISTERS_SPIO_OUTPUT_LOW:
M
Merav Sicron 已提交
2073
		DP(NETIF_MSG_HW, "Set SPIO %d -> output low\n", spio_num);
Y
Yaniv Rosner 已提交
2074 2075 2076 2077
		/* clear FLOAT and set CLR */
		spio_reg &= ~(spio_mask << MISC_REGISTERS_SPIO_FLOAT_POS);
		spio_reg |=  (spio_mask << MISC_REGISTERS_SPIO_CLR_POS);
		break;
E
Eliezer Tamir 已提交
2078

E
Eilon Greenstein 已提交
2079
	case MISC_REGISTERS_SPIO_OUTPUT_HIGH:
M
Merav Sicron 已提交
2080
		DP(NETIF_MSG_HW, "Set SPIO %d -> output high\n", spio_num);
Y
Yaniv Rosner 已提交
2081 2082 2083 2084
		/* clear FLOAT and set SET */
		spio_reg &= ~(spio_mask << MISC_REGISTERS_SPIO_FLOAT_POS);
		spio_reg |=  (spio_mask << MISC_REGISTERS_SPIO_SET_POS);
		break;
E
Eliezer Tamir 已提交
2085

Y
Yaniv Rosner 已提交
2086
	case MISC_REGISTERS_SPIO_INPUT_HI_Z:
M
Merav Sicron 已提交
2087
		DP(NETIF_MSG_HW, "Set SPIO %d -> input\n", spio_num);
Y
Yaniv Rosner 已提交
2088 2089 2090
		/* set FLOAT */
		spio_reg |= (spio_mask << MISC_REGISTERS_SPIO_FLOAT_POS);
		break;
E
Eliezer Tamir 已提交
2091

Y
Yaniv Rosner 已提交
2092 2093
	default:
		break;
E
Eliezer Tamir 已提交
2094 2095
	}

Y
Yaniv Rosner 已提交
2096
	REG_WR(bp, MISC_REG_SPIO, spio_reg);
Y
Yitchak Gertner 已提交
2097
	bnx2x_release_hw_lock(bp, HW_LOCK_RESOURCE_SPIO);
Y
Yaniv Rosner 已提交
2098

E
Eliezer Tamir 已提交
2099 2100 2101
	return 0;
}

D
Dmitry Kravkov 已提交
2102
void bnx2x_calc_fc_adv(struct bnx2x *bp)
E
Eliezer Tamir 已提交
2103
{
Y
Yaniv Rosner 已提交
2104
	u8 cfg_idx = bnx2x_get_link_cfg_idx(bp);
2105 2106
	switch (bp->link_vars.ieee_fc &
		MDIO_COMBO_IEEE0_AUTO_NEG_ADV_PAUSE_MASK) {
Y
Yaniv Rosner 已提交
2107
	case MDIO_COMBO_IEEE0_AUTO_NEG_ADV_PAUSE_NONE:
Y
Yaniv Rosner 已提交
2108
		bp->port.advertising[cfg_idx] &= ~(ADVERTISED_Asym_Pause |
D
Dmitry Kravkov 已提交
2109
						   ADVERTISED_Pause);
Y
Yaniv Rosner 已提交
2110
		break;
E
Eilon Greenstein 已提交
2111

Y
Yaniv Rosner 已提交
2112
	case MDIO_COMBO_IEEE0_AUTO_NEG_ADV_PAUSE_BOTH:
Y
Yaniv Rosner 已提交
2113
		bp->port.advertising[cfg_idx] |= (ADVERTISED_Asym_Pause |
D
Dmitry Kravkov 已提交
2114
						  ADVERTISED_Pause);
Y
Yaniv Rosner 已提交
2115
		break;
E
Eilon Greenstein 已提交
2116

Y
Yaniv Rosner 已提交
2117
	case MDIO_COMBO_IEEE0_AUTO_NEG_ADV_PAUSE_ASYMMETRIC:
Y
Yaniv Rosner 已提交
2118
		bp->port.advertising[cfg_idx] |= ADVERTISED_Asym_Pause;
Y
Yaniv Rosner 已提交
2119
		break;
E
Eilon Greenstein 已提交
2120

Y
Yaniv Rosner 已提交
2121
	default:
Y
Yaniv Rosner 已提交
2122
		bp->port.advertising[cfg_idx] &= ~(ADVERTISED_Asym_Pause |
D
Dmitry Kravkov 已提交
2123
						   ADVERTISED_Pause);
Y
Yaniv Rosner 已提交
2124 2125 2126
		break;
	}
}
E
Eliezer Tamir 已提交
2127

D
Dmitry Kravkov 已提交
2128
u8 bnx2x_initial_phy_init(struct bnx2x *bp, int load_mode)
Y
Yaniv Rosner 已提交
2129
{
2130 2131
	if (!BP_NOMCP(bp)) {
		u8 rc;
Y
Yaniv Rosner 已提交
2132 2133
		int cfx_idx = bnx2x_get_link_cfg_idx(bp);
		u16 req_line_speed = bp->link_params.req_line_speed[cfx_idx];
2134 2135 2136 2137 2138 2139
		/*
		 * Initialize link parameters structure variables
		 * It is recommended to turn off RX FC for jumbo frames
		 * for better performance
		 */
		if (CHIP_IS_E1x(bp) && (bp->dev->mtu > 5000))
2140
			bp->link_params.req_fc_auto_adv = BNX2X_FLOW_CTRL_TX;
Y
Yaniv Rosner 已提交
2141
		else
2142
			bp->link_params.req_fc_auto_adv = BNX2X_FLOW_CTRL_BOTH;
E
Eliezer Tamir 已提交
2143

Y
Yitchak Gertner 已提交
2144
		bnx2x_acquire_phy_lock(bp);
E
Eilon Greenstein 已提交
2145

Y
Yaniv Rosner 已提交
2146
		if (load_mode == LOAD_DIAG) {
2147 2148 2149 2150 2151 2152 2153 2154 2155 2156 2157 2158
			struct link_params *lp = &bp->link_params;
			lp->loopback_mode = LOOPBACK_XGXS;
			/* do PHY loopback at 10G speed, if possible */
			if (lp->req_line_speed[cfx_idx] < SPEED_10000) {
				if (lp->speed_cap_mask[cfx_idx] &
				    PORT_HW_CFG_SPEED_CAPABILITY_D0_10G)
					lp->req_line_speed[cfx_idx] =
					SPEED_10000;
				else
					lp->req_line_speed[cfx_idx] =
					SPEED_1000;
			}
Y
Yaniv Rosner 已提交
2159
		}
E
Eilon Greenstein 已提交
2160

2161
		rc = bnx2x_phy_init(&bp->link_params, &bp->link_vars);
E
Eilon Greenstein 已提交
2162

Y
Yitchak Gertner 已提交
2163
		bnx2x_release_phy_lock(bp);
E
Eliezer Tamir 已提交
2164

2165 2166
		bnx2x_calc_fc_adv(bp);

E
Eilon Greenstein 已提交
2167 2168
		if (CHIP_REV_IS_SLOW(bp) && bp->link_vars.link_up) {
			bnx2x_stats_handle(bp, STATS_EVENT_LINK_UP);
2169
			bnx2x_link_report(bp);
2170 2171
		} else
			queue_delayed_work(bnx2x_wq, &bp->period_task, 0);
Y
Yaniv Rosner 已提交
2172
		bp->link_params.req_line_speed[cfx_idx] = req_line_speed;
2173 2174
		return rc;
	}
E
Eilon Greenstein 已提交
2175
	BNX2X_ERR("Bootcode is missing - can not initialize link\n");
2176
	return -EINVAL;
E
Eliezer Tamir 已提交
2177 2178
}

D
Dmitry Kravkov 已提交
2179
void bnx2x_link_set(struct bnx2x *bp)
E
Eliezer Tamir 已提交
2180
{
2181
	if (!BP_NOMCP(bp)) {
Y
Yitchak Gertner 已提交
2182
		bnx2x_acquire_phy_lock(bp);
2183
		bnx2x_link_reset(&bp->link_params, &bp->link_vars, 1);
2184
		bnx2x_phy_init(&bp->link_params, &bp->link_vars);
Y
Yitchak Gertner 已提交
2185
		bnx2x_release_phy_lock(bp);
E
Eliezer Tamir 已提交
2186

2187 2188
		bnx2x_calc_fc_adv(bp);
	} else
E
Eilon Greenstein 已提交
2189
		BNX2X_ERR("Bootcode is missing - can not set link\n");
Y
Yaniv Rosner 已提交
2190
}
E
Eliezer Tamir 已提交
2191

Y
Yaniv Rosner 已提交
2192 2193
static void bnx2x__link_reset(struct bnx2x *bp)
{
2194
	if (!BP_NOMCP(bp)) {
Y
Yitchak Gertner 已提交
2195
		bnx2x_acquire_phy_lock(bp);
E
Eilon Greenstein 已提交
2196
		bnx2x_link_reset(&bp->link_params, &bp->link_vars, 1);
Y
Yitchak Gertner 已提交
2197
		bnx2x_release_phy_lock(bp);
2198
	} else
E
Eilon Greenstein 已提交
2199
		BNX2X_ERR("Bootcode is missing - can not reset link\n");
Y
Yaniv Rosner 已提交
2200
}
E
Eliezer Tamir 已提交
2201

Y
Yaniv Rosner 已提交
2202
u8 bnx2x_link_test(struct bnx2x *bp, u8 is_serdes)
Y
Yaniv Rosner 已提交
2203
{
2204
	u8 rc = 0;
E
Eliezer Tamir 已提交
2205

2206 2207
	if (!BP_NOMCP(bp)) {
		bnx2x_acquire_phy_lock(bp);
Y
Yaniv Rosner 已提交
2208 2209
		rc = bnx2x_test_link(&bp->link_params, &bp->link_vars,
				     is_serdes);
2210 2211 2212
		bnx2x_release_phy_lock(bp);
	} else
		BNX2X_ERR("Bootcode is missing - can not test link\n");
E
Eliezer Tamir 已提交
2213

Y
Yaniv Rosner 已提交
2214 2215
	return rc;
}
E
Eliezer Tamir 已提交
2216

E
Eilon Greenstein 已提交
2217
static void bnx2x_init_port_minmax(struct bnx2x *bp)
2218
{
E
Eilon Greenstein 已提交
2219 2220 2221
	u32 r_param = bp->link_vars.line_speed / 8;
	u32 fair_periodic_timeout_usec;
	u32 t_fair;
2222

E
Eilon Greenstein 已提交
2223 2224 2225
	memset(&(bp->cmng.rs_vars), 0,
	       sizeof(struct rate_shaping_vars_per_port));
	memset(&(bp->cmng.fair_vars), 0, sizeof(struct fairness_vars_per_port));
2226

E
Eilon Greenstein 已提交
2227 2228
	/* 100 usec in SDM ticks = 25 since each tick is 4 usec */
	bp->cmng.rs_vars.rs_periodic_timeout = RS_PERIODIC_TIMEOUT_USEC / 4;
2229

E
Eilon Greenstein 已提交
2230 2231 2232 2233
	/* this is the threshold below which no timer arming will occur
	   1.25 coefficient is for the threshold to be a little bigger
	   than the real time, to compensate for timer in-accuracy */
	bp->cmng.rs_vars.rs_threshold =
2234 2235
				(RS_PERIODIC_TIMEOUT_USEC * r_param * 5) / 4;

E
Eilon Greenstein 已提交
2236 2237 2238 2239
	/* resolution of fairness timer */
	fair_periodic_timeout_usec = QM_ARB_BYTES / r_param;
	/* for 10G it is 1000usec. for 1G it is 10000usec. */
	t_fair = T_FAIR_COEF / bp->link_vars.line_speed;
2240

E
Eilon Greenstein 已提交
2241 2242
	/* this is the threshold below which we won't arm the timer anymore */
	bp->cmng.fair_vars.fair_threshold = QM_ARB_BYTES;
2243

E
Eilon Greenstein 已提交
2244 2245 2246 2247 2248 2249
	/* we multiply by 1e3/8 to get bytes/msec.
	   We don't want the credits to pass a credit
	   of the t_fair*FAIR_MEM (algorithm resolution) */
	bp->cmng.fair_vars.upper_bound = r_param * t_fair * FAIR_MEM;
	/* since each tick is 4 usec */
	bp->cmng.fair_vars.fairness_timeout = fair_periodic_timeout_usec / 4;
2250 2251
}

2252 2253 2254 2255 2256 2257 2258 2259 2260 2261 2262 2263 2264 2265 2266
/* Calculates the sum of vn_min_rates.
   It's needed for further normalizing of the min_rates.
   Returns:
     sum of vn_min_rates.
       or
     0 - if all the min_rates are 0.
     In the later case fainess algorithm should be deactivated.
     If not all min_rates are zero then those that are zeroes will be set to 1.
 */
static void bnx2x_calc_vn_weight_sum(struct bnx2x *bp)
{
	int all_zero = 1;
	int vn;

	bp->vn_weight_sum = 0;
2267
	for (vn = VN_0; vn < BP_MAX_VN_NUM(bp); vn++) {
D
Dmitry Kravkov 已提交
2268
		u32 vn_cfg = bp->mf_config[vn];
2269 2270 2271 2272 2273 2274 2275 2276 2277 2278 2279 2280 2281 2282 2283 2284
		u32 vn_min_rate = ((vn_cfg & FUNC_MF_CFG_MIN_BW_MASK) >>
				   FUNC_MF_CFG_MIN_BW_SHIFT) * 100;

		/* Skip hidden vns */
		if (vn_cfg & FUNC_MF_CFG_FUNC_HIDE)
			continue;

		/* If min rate is zero - set it to 1 */
		if (!vn_min_rate)
			vn_min_rate = DEF_MIN_RATE;
		else
			all_zero = 0;

		bp->vn_weight_sum += vn_min_rate;
	}

2285 2286 2287 2288 2289 2290
	/* if ETS or all min rates are zeros - disable fairness */
	if (BNX2X_IS_ETS_ENABLED(bp)) {
		bp->cmng.flags.cmng_enables &=
					~CMNG_FLAGS_PER_PORT_FAIRNESS_VN;
		DP(NETIF_MSG_IFUP, "Fairness will be disabled due to ETS\n");
	} else if (all_zero) {
2291 2292 2293 2294 2295 2296 2297
		bp->cmng.flags.cmng_enables &=
					~CMNG_FLAGS_PER_PORT_FAIRNESS_VN;
		DP(NETIF_MSG_IFUP, "All MIN values are zeroes"
		   "  fairness will be disabled\n");
	} else
		bp->cmng.flags.cmng_enables |=
					CMNG_FLAGS_PER_PORT_FAIRNESS_VN;
2298 2299
}

D
Dmitry Kravkov 已提交
2300
static void bnx2x_init_vn_minmax(struct bnx2x *bp, int vn)
2301 2302 2303
{
	struct rate_shaping_vars_per_vn m_rs_vn;
	struct fairness_vars_per_vn m_fair_vn;
D
Dmitry Kravkov 已提交
2304
	u32 vn_cfg = bp->mf_config[vn];
2305
	int func = func_by_vn(bp, vn);
2306 2307 2308 2309 2310 2311 2312 2313 2314
	u16 vn_min_rate, vn_max_rate;
	int i;

	/* If function is hidden - set min and max to zeroes */
	if (vn_cfg & FUNC_MF_CFG_FUNC_HIDE) {
		vn_min_rate = 0;
		vn_max_rate = 0;

	} else {
2315 2316
		u32 maxCfg = bnx2x_extract_max_cfg(bp, vn_cfg);

2317 2318
		vn_min_rate = ((vn_cfg & FUNC_MF_CFG_MIN_BW_MASK) >>
				FUNC_MF_CFG_MIN_BW_SHIFT) * 100;
2319 2320 2321
		/* If fairness is enabled (not all min rates are zeroes) and
		   if current min rate is zero - set it to 1.
		   This is a requirement of the algorithm. */
D
Dmitry Kravkov 已提交
2322
		if (bp->vn_weight_sum && (vn_min_rate == 0))
2323
			vn_min_rate = DEF_MIN_RATE;
2324 2325 2326 2327 2328 2329 2330

		if (IS_MF_SI(bp))
			/* maxCfg in percents of linkspeed */
			vn_max_rate = (bp->link_vars.line_speed * maxCfg) / 100;
		else
			/* maxCfg is absolute in 100Mb units */
			vn_max_rate = maxCfg * 100;
2331
	}
D
Dmitry Kravkov 已提交
2332

E
Eilon Greenstein 已提交
2333
	DP(NETIF_MSG_IFUP,
2334
	   "func %d: vn_min_rate %d  vn_max_rate %d  vn_weight_sum %d\n",
E
Eilon Greenstein 已提交
2335
	   func, vn_min_rate, vn_max_rate, bp->vn_weight_sum);
2336 2337 2338 2339 2340 2341 2342 2343 2344 2345 2346

	memset(&m_rs_vn, 0, sizeof(struct rate_shaping_vars_per_vn));
	memset(&m_fair_vn, 0, sizeof(struct fairness_vars_per_vn));

	/* global vn counter - maximal Mbps for this vn */
	m_rs_vn.vn_counter.rate = vn_max_rate;

	/* quota - number of bytes transmitted in this period */
	m_rs_vn.vn_counter.quota =
				(vn_max_rate * RS_PERIODIC_TIMEOUT_USEC) / 8;

E
Eilon Greenstein 已提交
2347
	if (bp->vn_weight_sum) {
2348 2349
		/* credit for each period of the fairness algorithm:
		   number of bytes in T_FAIR (the vn share the port rate).
E
Eilon Greenstein 已提交
2350 2351 2352
		   vn_weight_sum should not be larger than 10000, thus
		   T_FAIR_COEF / (8 * vn_weight_sum) will always be greater
		   than zero */
2353
		m_fair_vn.vn_credit_delta =
V
Vladislav Zolotarov 已提交
2354 2355
			max_t(u32, (vn_min_rate * (T_FAIR_COEF /
						   (8 * bp->vn_weight_sum))),
2356 2357
			      (bp->cmng.fair_vars.fair_threshold +
							MIN_ABOVE_THRESH));
V
Vladislav Zolotarov 已提交
2358
		DP(NETIF_MSG_IFUP, "m_fair_vn.vn_credit_delta %d\n",
2359 2360 2361 2362 2363 2364 2365 2366 2367 2368 2369 2370 2371 2372
		   m_fair_vn.vn_credit_delta);
	}

	/* Store it to internal memory */
	for (i = 0; i < sizeof(struct rate_shaping_vars_per_vn)/4; i++)
		REG_WR(bp, BAR_XSTRORM_INTMEM +
		       XSTORM_RATE_SHAPING_PER_VN_VARS_OFFSET(func) + i * 4,
		       ((u32 *)(&m_rs_vn))[i]);

	for (i = 0; i < sizeof(struct fairness_vars_per_vn)/4; i++)
		REG_WR(bp, BAR_XSTRORM_INTMEM +
		       XSTORM_FAIRNESS_PER_VN_VARS_OFFSET(func) + i * 4,
		       ((u32 *)(&m_fair_vn))[i]);
}
D
Dmitry Kravkov 已提交
2373

2374 2375 2376 2377
static int bnx2x_get_cmng_fns_mode(struct bnx2x *bp)
{
	if (CHIP_REV_IS_SLOW(bp))
		return CMNG_FNS_NONE;
D
Dmitry Kravkov 已提交
2378
	if (IS_MF(bp))
2379 2380 2381 2382 2383
		return CMNG_FNS_MINMAX;

	return CMNG_FNS_NONE;
}

2384
void bnx2x_read_mf_cfg(struct bnx2x *bp)
2385
{
2386
	int vn, n = (CHIP_MODE_IS_4_PORT(bp) ? 2 : 1);
2387 2388 2389 2390

	if (BP_NOMCP(bp))
		return; /* what should be the default bvalue in this case */

2391 2392 2393 2394 2395 2396 2397 2398 2399 2400 2401
	/* For 2 port configuration the absolute function number formula
	 * is:
	 *      abs_func = 2 * vn + BP_PORT + BP_PATH
	 *
	 *      and there are 4 functions per port
	 *
	 * For 4 port configuration it is
	 *      abs_func = 4 * vn + 2 * BP_PORT + BP_PATH
	 *
	 *      and there are 2 functions per port
	 */
2402
	for (vn = VN_0; vn < BP_MAX_VN_NUM(bp); vn++) {
2403 2404 2405 2406 2407
		int /*abs*/func = n * (2 * vn + BP_PORT(bp)) + BP_PATH(bp);

		if (func >= E1H_FUNC_MAX)
			break;

D
Dmitry Kravkov 已提交
2408
		bp->mf_config[vn] =
2409 2410 2411 2412 2413 2414 2415 2416 2417 2418 2419 2420 2421 2422 2423 2424 2425 2426 2427 2428 2429 2430 2431 2432
			MF_CFG_RD(bp, func_mf_config[func].config);
	}
}

static void bnx2x_cmng_fns_init(struct bnx2x *bp, u8 read_cfg, u8 cmng_type)
{

	if (cmng_type == CMNG_FNS_MINMAX) {
		int vn;

		/* clear cmng_enables */
		bp->cmng.flags.cmng_enables = 0;

		/* read mf conf from shmem */
		if (read_cfg)
			bnx2x_read_mf_cfg(bp);

		/* Init rate shaping and fairness contexts */
		bnx2x_init_port_minmax(bp);

		/* vn_weight_sum and enable fairness if not 0 */
		bnx2x_calc_vn_weight_sum(bp);

		/* calculate and set min-max rate for each vn */
2433
		if (bp->port.pmf)
2434
			for (vn = VN_0; vn < BP_MAX_VN_NUM(bp); vn++)
2435
				bnx2x_init_vn_minmax(bp, vn);
2436 2437 2438 2439 2440 2441 2442 2443 2444 2445 2446 2447 2448 2449

		/* always enable rate shaping and fairness */
		bp->cmng.flags.cmng_enables |=
					CMNG_FLAGS_PER_PORT_RATE_SHAPING_VN;
		if (!bp->vn_weight_sum)
			DP(NETIF_MSG_IFUP, "All MIN values are zeroes"
				   "  fairness will be disabled\n");
		return;
	}

	/* rate shaping and fairness are disabled */
	DP(NETIF_MSG_IFUP,
	   "rate shaping and fairness are disabled\n");
}
2450

Y
Yaniv Rosner 已提交
2451 2452 2453
/* This function is called upon link interrupt */
static void bnx2x_link_attn(struct bnx2x *bp)
{
Y
Yitchak Gertner 已提交
2454 2455 2456
	/* Make sure that we are synced with the current statistics */
	bnx2x_stats_handle(bp, STATS_EVENT_STOP);

Y
Yaniv Rosner 已提交
2457
	bnx2x_link_update(&bp->link_params, &bp->link_vars);
E
Eliezer Tamir 已提交
2458

Y
Yitchak Gertner 已提交
2459 2460
	if (bp->link_vars.link_up) {

2461
		/* dropless flow control */
D
Dmitry Kravkov 已提交
2462
		if (!CHIP_IS_E1(bp) && bp->dropless_fc) {
2463 2464 2465 2466 2467 2468 2469
			int port = BP_PORT(bp);
			u32 pause_enabled = 0;

			if (bp->link_vars.flow_ctrl & BNX2X_FLOW_CTRL_TX)
				pause_enabled = 1;

			REG_WR(bp, BAR_USTRORM_INTMEM +
E
Eilon Greenstein 已提交
2470
			       USTORM_ETH_PAUSE_ENABLED_OFFSET(port),
2471 2472 2473
			       pause_enabled);
		}

2474
		if (bp->link_vars.mac_type != MAC_TYPE_EMAC) {
Y
Yitchak Gertner 已提交
2475 2476 2477
			struct host_port_stats *pstats;

			pstats = bnx2x_sp(bp, port_stats);
2478
			/* reset old mac stats */
Y
Yitchak Gertner 已提交
2479 2480 2481
			memset(&(pstats->mac_stx[0]), 0,
			       sizeof(struct mac_stx));
		}
2482
		if (bp->state == BNX2X_STATE_OPEN)
Y
Yitchak Gertner 已提交
2483 2484 2485
			bnx2x_stats_handle(bp, STATS_EVENT_LINK_UP);
	}

D
Dmitry Kravkov 已提交
2486 2487
	if (bp->link_vars.link_up && bp->link_vars.line_speed) {
		int cmng_fns = bnx2x_get_cmng_fns_mode(bp);
E
Eilon Greenstein 已提交
2488

D
Dmitry Kravkov 已提交
2489 2490 2491 2492 2493 2494 2495
		if (cmng_fns != CMNG_FNS_NONE) {
			bnx2x_cmng_fns_init(bp, false, cmng_fns);
			storm_memset_cmng(bp, &bp->cmng, BP_PORT(bp));
		} else
			/* rate shaping and fairness are disabled */
			DP(NETIF_MSG_IFUP,
			   "single function mode without fairness\n");
2496
	}
D
Dmitry Kravkov 已提交
2497

2498 2499
	__bnx2x_link_report(bp);

D
Dmitry Kravkov 已提交
2500 2501
	if (IS_MF(bp))
		bnx2x_link_sync_notify(bp);
Y
Yaniv Rosner 已提交
2502
}
E
Eliezer Tamir 已提交
2503

D
Dmitry Kravkov 已提交
2504
void bnx2x__link_status_update(struct bnx2x *bp)
Y
Yaniv Rosner 已提交
2505
{
2506
	if (bp->state != BNX2X_STATE_OPEN)
Y
Yaniv Rosner 已提交
2507
		return;
E
Eliezer Tamir 已提交
2508

D
Dmitry Kravkov 已提交
2509 2510 2511
	/* read updated dcb configuration */
	bnx2x_dcbx_pmf_update(bp);

Y
Yaniv Rosner 已提交
2512
	bnx2x_link_status_update(&bp->link_params, &bp->link_vars);
E
Eliezer Tamir 已提交
2513

Y
Yitchak Gertner 已提交
2514 2515 2516 2517 2518
	if (bp->link_vars.link_up)
		bnx2x_stats_handle(bp, STATS_EVENT_LINK_UP);
	else
		bnx2x_stats_handle(bp, STATS_EVENT_STOP);

Y
Yaniv Rosner 已提交
2519 2520
	/* indicate link status */
	bnx2x_link_report(bp);
E
Eliezer Tamir 已提交
2521 2522
}

2523 2524 2525 2526 2527 2528
static void bnx2x_pmf_update(struct bnx2x *bp)
{
	int port = BP_PORT(bp);
	u32 val;

	bp->port.pmf = 1;
M
Merav Sicron 已提交
2529
	DP(BNX2X_MSG_MCP, "pmf %d\n", bp->port.pmf);
2530

2531 2532 2533 2534 2535 2536 2537 2538 2539
	/*
	 * We need the mb() to ensure the ordering between the writing to
	 * bp->port.pmf here and reading it from the bnx2x_periodic_task().
	 */
	smp_mb();

	/* queue a periodic task */
	queue_delayed_work(bnx2x_wq, &bp->period_task, 0);

2540 2541
	bnx2x_dcbx_pmf_update(bp);

2542
	/* enable nig attention */
2543
	val = (0xff0f | (1 << (BP_VN(bp) + 4)));
D
Dmitry Kravkov 已提交
2544 2545 2546
	if (bp->common.int_block == INT_BLOCK_HC) {
		REG_WR(bp, HC_REG_TRAILING_EDGE_0 + port*8, val);
		REG_WR(bp, HC_REG_LEADING_EDGE_0 + port*8, val);
2547
	} else if (!CHIP_IS_E1x(bp)) {
D
Dmitry Kravkov 已提交
2548 2549 2550
		REG_WR(bp, IGU_REG_TRAILING_EDGE_LATCH, val);
		REG_WR(bp, IGU_REG_LEADING_EDGE_LATCH, val);
	}
Y
Yitchak Gertner 已提交
2551 2552

	bnx2x_stats_handle(bp, STATS_EVENT_PMF);
2553 2554
}

Y
Yaniv Rosner 已提交
2555
/* end of Link */
E
Eliezer Tamir 已提交
2556 2557 2558 2559 2560 2561 2562

/* slow path */

/*
 * General service functions
 */

2563
/* send the MCP a request, block until there is a reply */
Y
Yaniv Rosner 已提交
2564
u32 bnx2x_fw_command(struct bnx2x *bp, u32 command, u32 param)
2565
{
D
Dmitry Kravkov 已提交
2566
	int mb_idx = BP_FW_MB_IDX(bp);
2567
	u32 seq;
2568 2569 2570 2571
	u32 rc = 0;
	u32 cnt = 1;
	u8 delay = CHIP_REV_IS_SLOW(bp) ? 100 : 10;

E
Eilon Greenstein 已提交
2572
	mutex_lock(&bp->fw_mb_mutex);
2573
	seq = ++bp->fw_seq;
D
Dmitry Kravkov 已提交
2574 2575 2576
	SHMEM_WR(bp, func_mb[mb_idx].drv_mb_param, param);
	SHMEM_WR(bp, func_mb[mb_idx].drv_mb_header, (command | seq));

D
Dmitry Kravkov 已提交
2577 2578
	DP(BNX2X_MSG_MCP, "wrote command (%x) to FW MB param 0x%08x\n",
			(command | seq), param);
2579 2580 2581 2582 2583

	do {
		/* let the FW do it's magic ... */
		msleep(delay);

D
Dmitry Kravkov 已提交
2584
		rc = SHMEM_RD(bp, func_mb[mb_idx].fw_mb_header);
2585

E
Eilon Greenstein 已提交
2586 2587
		/* Give the FW up to 5 second (500*10ms) */
	} while ((seq != (rc & FW_MSG_SEQ_NUMBER_MASK)) && (cnt++ < 500));
2588 2589 2590 2591 2592 2593 2594 2595 2596 2597 2598 2599 2600

	DP(BNX2X_MSG_MCP, "[after %d ms] read (%x) seq is (%x) from FW MB\n",
	   cnt*delay, rc, seq);

	/* is this a reply to our command? */
	if (seq == (rc & FW_MSG_SEQ_NUMBER_MASK))
		rc &= FW_MSG_CODE_MASK;
	else {
		/* FW BUG! */
		BNX2X_ERR("FW failed to respond!\n");
		bnx2x_fw_dump(bp);
		rc = 0;
	}
E
Eilon Greenstein 已提交
2601
	mutex_unlock(&bp->fw_mb_mutex);
2602 2603 2604 2605

	return rc;
}

V
Vladislav Zolotarov 已提交
2606

2607 2608 2609 2610 2611 2612 2613 2614 2615 2616 2617 2618 2619 2620 2621 2622 2623 2624 2625 2626
void bnx2x_func_init(struct bnx2x *bp, struct bnx2x_func_init_params *p)
{
	if (CHIP_IS_E1x(bp)) {
		struct tstorm_eth_function_common_config tcfg = {0};

		storm_memset_func_cfg(bp, &tcfg, p->func_id);
	}

	/* Enable the function in the FW */
	storm_memset_vf_to_pf(bp, p->func_id, p->pf_id);
	storm_memset_func_en(bp, p->func_id, 1);

	/* spq */
	if (p->func_flgs & FUNC_FLG_SPQ) {
		storm_memset_spq_addr(bp, p->spq_map, p->func_id);
		REG_WR(bp, XSEM_REG_FAST_MEMORY +
		       XSTORM_SPQ_PROD_OFFSET(p->func_id), p->spq_prod);
	}
}

2627 2628 2629 2630 2631 2632 2633 2634 2635 2636 2637 2638
/**
 * bnx2x_get_tx_only_flags - Return common flags
 *
 * @bp		device handle
 * @fp		queue handle
 * @zero_stats	TRUE if statistics zeroing is needed
 *
 * Return the flags that are common for the Tx-only and not normal connections.
 */
static inline unsigned long bnx2x_get_common_flags(struct bnx2x *bp,
						   struct bnx2x_fastpath *fp,
						   bool zero_stats)
M
Michael Chan 已提交
2639
{
2640 2641 2642 2643
	unsigned long flags = 0;

	/* PF driver will always initialize the Queue to an ACTIVE state */
	__set_bit(BNX2X_Q_FLG_ACTIVE, &flags);
M
Michael Chan 已提交
2644

2645 2646 2647 2648
	/* tx only connections collect statistics (on the same index as the
	 *  parent connection). The statistics are zeroed when the parent
	 *  connection is initialized.
	 */
B
Barak Witkowski 已提交
2649 2650 2651 2652 2653

	__set_bit(BNX2X_Q_FLG_STATS, &flags);
	if (zero_stats)
		__set_bit(BNX2X_Q_FLG_ZERO_STATS, &flags);

2654 2655 2656 2657 2658 2659 2660 2661 2662 2663

	return flags;
}

static inline unsigned long bnx2x_get_q_flags(struct bnx2x *bp,
					      struct bnx2x_fastpath *fp,
					      bool leading)
{
	unsigned long flags = 0;

2664 2665 2666
	/* calculate other queue flags */
	if (IS_MF_SD(bp))
		__set_bit(BNX2X_Q_FLG_OV, &flags);
M
Michael Chan 已提交
2667

2668 2669
	if (IS_FCOE_FP(fp))
		__set_bit(BNX2X_Q_FLG_FCOE, &flags);
2670

2671
	if (!fp->disable_tpa) {
2672
		__set_bit(BNX2X_Q_FLG_TPA, &flags);
2673
		__set_bit(BNX2X_Q_FLG_TPA_IPV6, &flags);
D
Dmitry Kravkov 已提交
2674 2675
		if (fp->mode == TPA_MODE_GRO)
			__set_bit(BNX2X_Q_FLG_TPA_GRO, &flags);
2676
	}
2677 2678 2679 2680 2681

	if (leading) {
		__set_bit(BNX2X_Q_FLG_LEADING_RSS, &flags);
		__set_bit(BNX2X_Q_FLG_MCAST, &flags);
	}
2682

2683 2684
	/* Always set HW VLAN stripping */
	__set_bit(BNX2X_Q_FLG_VLAN, &flags);
2685

2686 2687

	return flags | bnx2x_get_common_flags(bp, fp, true);
2688 2689
}

2690
static void bnx2x_pf_q_prep_general(struct bnx2x *bp,
2691 2692
	struct bnx2x_fastpath *fp, struct bnx2x_general_setup_params *gen_init,
	u8 cos)
2693 2694 2695 2696 2697 2698 2699 2700 2701
{
	gen_init->stat_id = bnx2x_stats_id(fp);
	gen_init->spcl_id = fp->cl_id;

	/* Always use mini-jumbo MTU for FCoE L2 ring */
	if (IS_FCOE_FP(fp))
		gen_init->mtu = BNX2X_FCOE_MINI_JUMBO_MTU;
	else
		gen_init->mtu = bp->dev->mtu;
2702 2703

	gen_init->cos = cos;
2704 2705 2706
}

static void bnx2x_pf_rx_q_prep(struct bnx2x *bp,
2707
	struct bnx2x_fastpath *fp, struct rxq_pause_params *pause,
2708
	struct bnx2x_rxq_setup_params *rxq_init)
2709
{
2710
	u8 max_sge = 0;
2711 2712 2713 2714
	u16 sge_sz = 0;
	u16 tpa_agg_size = 0;

	if (!fp->disable_tpa) {
2715 2716 2717 2718 2719 2720 2721 2722
		pause->sge_th_lo = SGE_TH_LO(bp);
		pause->sge_th_hi = SGE_TH_HI(bp);

		/* validate SGE ring has enough to cross high threshold */
		WARN_ON(bp->dropless_fc &&
				pause->sge_th_hi + FW_PREFETCH_CNT >
				MAX_RX_SGE_CNT * NUM_RX_SGE_PAGES);

2723 2724 2725 2726 2727 2728 2729 2730 2731 2732 2733 2734 2735
		tpa_agg_size = min_t(u32,
			(min_t(u32, 8, MAX_SKB_FRAGS) *
			SGE_PAGE_SIZE * PAGES_PER_SGE), 0xffff);
		max_sge = SGE_PAGE_ALIGN(bp->dev->mtu) >>
			SGE_PAGE_SHIFT;
		max_sge = ((max_sge + PAGES_PER_SGE - 1) &
			  (~(PAGES_PER_SGE-1))) >> PAGES_PER_SGE_SHIFT;
		sge_sz = (u16)min_t(u32, SGE_PAGE_SIZE * PAGES_PER_SGE,
				    0xffff);
	}

	/* pause - not for e1 */
	if (!CHIP_IS_E1(bp)) {
2736 2737 2738 2739 2740 2741 2742 2743 2744 2745 2746 2747 2748 2749 2750
		pause->bd_th_lo = BD_TH_LO(bp);
		pause->bd_th_hi = BD_TH_HI(bp);

		pause->rcq_th_lo = RCQ_TH_LO(bp);
		pause->rcq_th_hi = RCQ_TH_HI(bp);
		/*
		 * validate that rings have enough entries to cross
		 * high thresholds
		 */
		WARN_ON(bp->dropless_fc &&
				pause->bd_th_hi + FW_PREFETCH_CNT >
				bp->rx_ring_size);
		WARN_ON(bp->dropless_fc &&
				pause->rcq_th_hi + FW_PREFETCH_CNT >
				NUM_RCQ_RINGS * MAX_RCQ_DESC_CNT);
2751

2752 2753 2754 2755 2756 2757 2758 2759
		pause->pri_map = 1;
	}

	/* rxq setup */
	rxq_init->dscr_map = fp->rx_desc_mapping;
	rxq_init->sge_map = fp->rx_sge_mapping;
	rxq_init->rcq_map = fp->rx_comp_mapping;
	rxq_init->rcq_np_map = fp->rx_comp_mapping + BCM_PAGE_SIZE;
2760

2761 2762 2763
	/* This should be a maximum number of data bytes that may be
	 * placed on the BD (not including paddings).
	 */
2764 2765
	rxq_init->buf_sz = fp->rx_buf_size - BNX2X_FW_RX_ALIGN_START -
		BNX2X_FW_RX_ALIGN_END -	IP_HEADER_ALIGNMENT_PADDING;
2766

2767 2768 2769 2770
	rxq_init->cl_qzone_id = fp->cl_qzone_id;
	rxq_init->tpa_agg_sz = tpa_agg_size;
	rxq_init->sge_buf_sz = sge_sz;
	rxq_init->max_sges_pkt = max_sge;
2771
	rxq_init->rss_engine_id = BP_FUNC(bp);
2772
	rxq_init->mcast_engine_id = BP_FUNC(bp);
2773 2774 2775 2776 2777 2778

	/* Maximum number or simultaneous TPA aggregation for this Queue.
	 *
	 * For PF Clients it should be the maximum avaliable number.
	 * VF driver(s) may want to define it to a smaller value.
	 */
2779
	rxq_init->max_tpa_queues = MAX_AGG_QS(bp);
2780

2781 2782 2783
	rxq_init->cache_line_log = BNX2X_RX_ALIGN_SHIFT;
	rxq_init->fw_sb_id = fp->fw_sb_id;

V
Vladislav Zolotarov 已提交
2784 2785 2786
	if (IS_FCOE_FP(fp))
		rxq_init->sb_cq_index = HC_SP_INDEX_ETH_FCOE_RX_CQ_CONS;
	else
2787
		rxq_init->sb_cq_index = HC_INDEX_ETH_RX_CQ_CONS;
2788 2789
}

2790
static void bnx2x_pf_tx_q_prep(struct bnx2x *bp,
2791 2792
	struct bnx2x_fastpath *fp, struct bnx2x_txq_setup_params *txq_init,
	u8 cos)
2793
{
2794 2795
	txq_init->dscr_map = fp->txdata[cos].tx_desc_mapping;
	txq_init->sb_cq_index = HC_INDEX_ETH_FIRST_TX_CQ_CONS + cos;
2796 2797
	txq_init->traffic_type = LLFC_TRAFFIC_TYPE_NW;
	txq_init->fw_sb_id = fp->fw_sb_id;
V
Vladislav Zolotarov 已提交
2798

2799 2800 2801 2802 2803 2804
	/*
	 * set the tss leading client id for TX classfication ==
	 * leading RSS client id
	 */
	txq_init->tss_leading_cl_id = bnx2x_fp(bp, 0, cl_id);

V
Vladislav Zolotarov 已提交
2805 2806 2807 2808
	if (IS_FCOE_FP(fp)) {
		txq_init->sb_cq_index = HC_SP_INDEX_ETH_FCOE_TX_CQ_CONS;
		txq_init->traffic_type = LLFC_TRAFFIC_TYPE_FCOE;
	}
2809 2810
}

2811
static void bnx2x_pf_init(struct bnx2x *bp)
2812 2813 2814 2815 2816
{
	struct bnx2x_func_init_params func_init = {0};
	struct event_ring_data eq_data = { {0} };
	u16 flags;

2817
	if (!CHIP_IS_E1x(bp)) {
D
Dmitry Kravkov 已提交
2818 2819 2820 2821 2822 2823 2824 2825 2826 2827 2828 2829 2830 2831
		/* reset IGU PF statistics: MSIX + ATTN */
		/* PF */
		REG_WR(bp, IGU_REG_STATISTIC_NUM_MESSAGE_SENT +
			   BNX2X_IGU_STAS_MSG_VF_CNT*4 +
			   (CHIP_MODE_IS_4_PORT(bp) ?
				BP_FUNC(bp) : BP_VN(bp))*4, 0);
		/* ATTN */
		REG_WR(bp, IGU_REG_STATISTIC_NUM_MESSAGE_SENT +
			   BNX2X_IGU_STAS_MSG_VF_CNT*4 +
			   BNX2X_IGU_STAS_MSG_PF_CNT*4 +
			   (CHIP_MODE_IS_4_PORT(bp) ?
				BP_FUNC(bp) : BP_VN(bp))*4, 0);
	}

2832 2833 2834
	/* function setup flags */
	flags = (FUNC_FLG_STATS | FUNC_FLG_LEADING | FUNC_FLG_SPQ);

2835 2836
	/* This flag is relevant for E1x only.
	 * E2 doesn't have a TPA configuration in a function level.
2837
	 */
2838
	flags |= (bp->flags & TPA_ENABLE_FLAG) ? FUNC_FLG_TPA : 0;
2839 2840 2841 2842 2843 2844 2845 2846 2847 2848 2849 2850

	func_init.func_flgs = flags;
	func_init.pf_id = BP_FUNC(bp);
	func_init.func_id = BP_FUNC(bp);
	func_init.spq_map = bp->spq_mapping;
	func_init.spq_prod = bp->spq_prod_idx;

	bnx2x_func_init(bp, &func_init);

	memset(&(bp->cmng), 0, sizeof(struct cmng_struct_per_port));

	/*
2851 2852 2853 2854 2855
	 * Congestion management values depend on the link rate
	 * There is no active link so initial link rate is set to 10 Gbps.
	 * When the link comes up The congestion management values are
	 * re-calculated according to the actual link rate.
	 */
2856 2857 2858 2859 2860 2861 2862 2863 2864 2865 2866 2867 2868 2869 2870 2871 2872 2873 2874 2875 2876
	bp->link_vars.line_speed = SPEED_10000;
	bnx2x_cmng_fns_init(bp, true, bnx2x_get_cmng_fns_mode(bp));

	/* Only the PMF sets the HW */
	if (bp->port.pmf)
		storm_memset_cmng(bp, &bp->cmng, BP_PORT(bp));

	/* init Event Queue */
	eq_data.base_addr.hi = U64_HI(bp->eq_mapping);
	eq_data.base_addr.lo = U64_LO(bp->eq_mapping);
	eq_data.producer = bp->eq_prod;
	eq_data.index_id = HC_SP_INDEX_EQ_CONS;
	eq_data.sb_id = DEF_SB_ID;
	storm_memset_eq_data(bp, &eq_data, BP_FUNC(bp));
}


static void bnx2x_e1h_disable(struct bnx2x *bp)
{
	int port = BP_PORT(bp);

2877
	bnx2x_tx_disable(bp);
2878 2879 2880 2881 2882 2883 2884 2885 2886 2887 2888 2889 2890 2891 2892 2893 2894 2895 2896

	REG_WR(bp, NIG_REG_LLH0_FUNC_EN + port*8, 0);
}

static void bnx2x_e1h_enable(struct bnx2x *bp)
{
	int port = BP_PORT(bp);

	REG_WR(bp, NIG_REG_LLH0_FUNC_EN + port*8, 1);

	/* Tx queue should be only reenabled */
	netif_tx_wake_all_queues(bp->dev);

	/*
	 * Should not call netif_carrier_on since it will be called if the link
	 * is up when checking for link state
	 */
}

2897 2898 2899 2900 2901 2902 2903 2904 2905 2906 2907 2908 2909 2910 2911 2912 2913 2914 2915 2916 2917 2918 2919 2920 2921 2922 2923 2924 2925 2926 2927
#define DRV_INFO_ETH_STAT_NUM_MACS_REQUIRED 3

static void bnx2x_drv_info_ether_stat(struct bnx2x *bp)
{
	struct eth_stats_info *ether_stat =
		&bp->slowpath->drv_info_to_mcp.ether_stat;

	/* leave last char as NULL */
	memcpy(ether_stat->version, DRV_MODULE_VERSION,
	       ETH_STAT_INFO_VERSION_LEN - 1);

	bp->fp[0].mac_obj.get_n_elements(bp, &bp->fp[0].mac_obj,
					 DRV_INFO_ETH_STAT_NUM_MACS_REQUIRED,
					 ether_stat->mac_local);

	ether_stat->mtu_size = bp->dev->mtu;

	if (bp->dev->features & NETIF_F_RXCSUM)
		ether_stat->feature_flags |= FEATURE_ETH_CHKSUM_OFFLOAD_MASK;
	if (bp->dev->features & NETIF_F_TSO)
		ether_stat->feature_flags |= FEATURE_ETH_LSO_MASK;
	ether_stat->feature_flags |= bp->common.boot_mode;

	ether_stat->promiscuous_mode = (bp->dev->flags & IFF_PROMISC) ? 1 : 0;

	ether_stat->txq_size = bp->tx_ring_size;
	ether_stat->rxq_size = bp->rx_ring_size;
}

static void bnx2x_drv_info_fcoe_stat(struct bnx2x *bp)
{
2928
#ifdef BCM_CNIC
2929 2930 2931 2932 2933 2934 2935 2936 2937 2938 2939 2940 2941 2942 2943 2944 2945 2946 2947 2948 2949 2950 2951 2952 2953 2954 2955 2956 2957 2958 2959 2960 2961 2962 2963 2964 2965 2966 2967 2968 2969 2970 2971 2972 2973 2974 2975 2976 2977 2978
	struct bnx2x_dcbx_app_params *app = &bp->dcbx_port_params.app;
	struct fcoe_stats_info *fcoe_stat =
		&bp->slowpath->drv_info_to_mcp.fcoe_stat;

	memcpy(fcoe_stat->mac_local, bp->fip_mac, ETH_ALEN);

	fcoe_stat->qos_priority =
		app->traffic_type_priority[LLFC_TRAFFIC_TYPE_FCOE];

	/* insert FCoE stats from ramrod response */
	if (!NO_FCOE(bp)) {
		struct tstorm_per_queue_stats *fcoe_q_tstorm_stats =
			&bp->fw_stats_data->queue_stats[FCOE_IDX].
			tstorm_queue_statistics;

		struct xstorm_per_queue_stats *fcoe_q_xstorm_stats =
			&bp->fw_stats_data->queue_stats[FCOE_IDX].
			xstorm_queue_statistics;

		struct fcoe_statistics_params *fw_fcoe_stat =
			&bp->fw_stats_data->fcoe;

		ADD_64(fcoe_stat->rx_bytes_hi, 0, fcoe_stat->rx_bytes_lo,
		       fw_fcoe_stat->rx_stat0.fcoe_rx_byte_cnt);

		ADD_64(fcoe_stat->rx_bytes_hi,
		       fcoe_q_tstorm_stats->rcv_ucast_bytes.hi,
		       fcoe_stat->rx_bytes_lo,
		       fcoe_q_tstorm_stats->rcv_ucast_bytes.lo);

		ADD_64(fcoe_stat->rx_bytes_hi,
		       fcoe_q_tstorm_stats->rcv_bcast_bytes.hi,
		       fcoe_stat->rx_bytes_lo,
		       fcoe_q_tstorm_stats->rcv_bcast_bytes.lo);

		ADD_64(fcoe_stat->rx_bytes_hi,
		       fcoe_q_tstorm_stats->rcv_mcast_bytes.hi,
		       fcoe_stat->rx_bytes_lo,
		       fcoe_q_tstorm_stats->rcv_mcast_bytes.lo);

		ADD_64(fcoe_stat->rx_frames_hi, 0, fcoe_stat->rx_frames_lo,
		       fw_fcoe_stat->rx_stat0.fcoe_rx_pkt_cnt);

		ADD_64(fcoe_stat->rx_frames_hi, 0, fcoe_stat->rx_frames_lo,
		       fcoe_q_tstorm_stats->rcv_ucast_pkts);

		ADD_64(fcoe_stat->rx_frames_hi, 0, fcoe_stat->rx_frames_lo,
		       fcoe_q_tstorm_stats->rcv_bcast_pkts);

		ADD_64(fcoe_stat->rx_frames_hi, 0, fcoe_stat->rx_frames_lo,
2979
		       fcoe_q_tstorm_stats->rcv_mcast_pkts);
2980 2981 2982 2983 2984 2985 2986 2987 2988 2989 2990 2991 2992 2993 2994 2995 2996 2997 2998 2999 3000 3001 3002 3003 3004 3005 3006 3007 3008 3009 3010 3011 3012 3013 3014 3015 3016 3017 3018

		ADD_64(fcoe_stat->tx_bytes_hi, 0, fcoe_stat->tx_bytes_lo,
		       fw_fcoe_stat->tx_stat.fcoe_tx_byte_cnt);

		ADD_64(fcoe_stat->tx_bytes_hi,
		       fcoe_q_xstorm_stats->ucast_bytes_sent.hi,
		       fcoe_stat->tx_bytes_lo,
		       fcoe_q_xstorm_stats->ucast_bytes_sent.lo);

		ADD_64(fcoe_stat->tx_bytes_hi,
		       fcoe_q_xstorm_stats->bcast_bytes_sent.hi,
		       fcoe_stat->tx_bytes_lo,
		       fcoe_q_xstorm_stats->bcast_bytes_sent.lo);

		ADD_64(fcoe_stat->tx_bytes_hi,
		       fcoe_q_xstorm_stats->mcast_bytes_sent.hi,
		       fcoe_stat->tx_bytes_lo,
		       fcoe_q_xstorm_stats->mcast_bytes_sent.lo);

		ADD_64(fcoe_stat->tx_frames_hi, 0, fcoe_stat->tx_frames_lo,
		       fw_fcoe_stat->tx_stat.fcoe_tx_pkt_cnt);

		ADD_64(fcoe_stat->tx_frames_hi, 0, fcoe_stat->tx_frames_lo,
		       fcoe_q_xstorm_stats->ucast_pkts_sent);

		ADD_64(fcoe_stat->tx_frames_hi, 0, fcoe_stat->tx_frames_lo,
		       fcoe_q_xstorm_stats->bcast_pkts_sent);

		ADD_64(fcoe_stat->tx_frames_hi, 0, fcoe_stat->tx_frames_lo,
		       fcoe_q_xstorm_stats->mcast_pkts_sent);
	}

	/* ask L5 driver to add data to the struct */
	bnx2x_cnic_notify(bp, CNIC_CTL_FCOE_STATS_GET_CMD);
#endif
}

static void bnx2x_drv_info_iscsi_stat(struct bnx2x *bp)
{
3019
#ifdef BCM_CNIC
3020 3021 3022 3023 3024 3025 3026 3027 3028 3029 3030 3031 3032 3033
	struct bnx2x_dcbx_app_params *app = &bp->dcbx_port_params.app;
	struct iscsi_stats_info *iscsi_stat =
		&bp->slowpath->drv_info_to_mcp.iscsi_stat;

	memcpy(iscsi_stat->mac_local, bp->cnic_eth_dev.iscsi_mac, ETH_ALEN);

	iscsi_stat->qos_priority =
		app->traffic_type_priority[LLFC_TRAFFIC_TYPE_ISCSI];

	/* ask L5 driver to add data to the struct */
	bnx2x_cnic_notify(bp, CNIC_CTL_ISCSI_STATS_GET_CMD);
#endif
}

3034 3035 3036 3037 3038 3039 3040 3041 3042 3043 3044 3045 3046 3047 3048 3049 3050 3051 3052 3053
/* called due to MCP event (on pmf):
 *	reread new bandwidth configuration
 *	configure FW
 *	notify others function about the change
 */
static inline void bnx2x_config_mf_bw(struct bnx2x *bp)
{
	if (bp->link_vars.link_up) {
		bnx2x_cmng_fns_init(bp, true, CMNG_FNS_MINMAX);
		bnx2x_link_sync_notify(bp);
	}
	storm_memset_cmng(bp, &bp->cmng, BP_PORT(bp));
}

static inline void bnx2x_set_mf_bw(struct bnx2x *bp)
{
	bnx2x_config_mf_bw(bp);
	bnx2x_fw_command(bp, DRV_MSG_CODE_SET_MF_BW_ACK, 0);
}

3054 3055 3056 3057 3058 3059 3060 3061 3062 3063 3064 3065 3066 3067 3068 3069 3070 3071 3072 3073 3074 3075 3076 3077 3078 3079 3080 3081 3082 3083 3084 3085 3086 3087 3088 3089 3090 3091 3092 3093 3094 3095 3096 3097
static void bnx2x_handle_drv_info_req(struct bnx2x *bp)
{
	enum drv_info_opcode op_code;
	u32 drv_info_ctl = SHMEM2_RD(bp, drv_info_control);

	/* if drv_info version supported by MFW doesn't match - send NACK */
	if ((drv_info_ctl & DRV_INFO_CONTROL_VER_MASK) != DRV_INFO_CUR_VER) {
		bnx2x_fw_command(bp, DRV_MSG_CODE_DRV_INFO_NACK, 0);
		return;
	}

	op_code = (drv_info_ctl & DRV_INFO_CONTROL_OP_CODE_MASK) >>
		  DRV_INFO_CONTROL_OP_CODE_SHIFT;

	memset(&bp->slowpath->drv_info_to_mcp, 0,
	       sizeof(union drv_info_to_mcp));

	switch (op_code) {
	case ETH_STATS_OPCODE:
		bnx2x_drv_info_ether_stat(bp);
		break;
	case FCOE_STATS_OPCODE:
		bnx2x_drv_info_fcoe_stat(bp);
		break;
	case ISCSI_STATS_OPCODE:
		bnx2x_drv_info_iscsi_stat(bp);
		break;
	default:
		/* if op code isn't supported - send NACK */
		bnx2x_fw_command(bp, DRV_MSG_CODE_DRV_INFO_NACK, 0);
		return;
	}

	/* if we got drv_info attn from MFW then these fields are defined in
	 * shmem2 for sure
	 */
	SHMEM2_WR(bp, drv_info_host_addr_lo,
		U64_LO(bnx2x_sp_mapping(bp, drv_info_to_mcp)));
	SHMEM2_WR(bp, drv_info_host_addr_hi,
		U64_HI(bnx2x_sp_mapping(bp, drv_info_to_mcp)));

	bnx2x_fw_command(bp, DRV_MSG_CODE_DRV_INFO_ACK, 0);
}

3098 3099 3100 3101 3102 3103 3104 3105 3106 3107 3108
static void bnx2x_dcc_event(struct bnx2x *bp, u32 dcc_event)
{
	DP(BNX2X_MSG_MCP, "dcc_event 0x%x\n", dcc_event);

	if (dcc_event & DRV_STATUS_DCC_DISABLE_ENABLE_PF) {

		/*
		 * This is the only place besides the function initialization
		 * where the bp->flags can change so it is done without any
		 * locks
		 */
D
Dmitry Kravkov 已提交
3109
		if (bp->mf_config[BP_VN(bp)] & FUNC_MF_CFG_FUNC_DISABLED) {
M
Merav Sicron 已提交
3110
			DP(BNX2X_MSG_MCP, "mf_cfg function disabled\n");
3111 3112 3113 3114
			bp->flags |= MF_FUNC_DIS;

			bnx2x_e1h_disable(bp);
		} else {
M
Merav Sicron 已提交
3115
			DP(BNX2X_MSG_MCP, "mf_cfg function enabled\n");
3116 3117 3118 3119 3120 3121 3122
			bp->flags &= ~MF_FUNC_DIS;

			bnx2x_e1h_enable(bp);
		}
		dcc_event &= ~DRV_STATUS_DCC_DISABLE_ENABLE_PF;
	}
	if (dcc_event & DRV_STATUS_DCC_BANDWIDTH_ALLOCATION) {
3123
		bnx2x_config_mf_bw(bp);
3124 3125 3126 3127 3128 3129 3130 3131 3132 3133 3134 3135 3136 3137 3138 3139 3140 3141
		dcc_event &= ~DRV_STATUS_DCC_BANDWIDTH_ALLOCATION;
	}

	/* Report results to MCP */
	if (dcc_event)
		bnx2x_fw_command(bp, DRV_MSG_CODE_DCC_FAILURE, 0);
	else
		bnx2x_fw_command(bp, DRV_MSG_CODE_DCC_OK, 0);
}

/* must be called under the spq lock */
static inline struct eth_spe *bnx2x_sp_get_next(struct bnx2x *bp)
{
	struct eth_spe *next_spe = bp->spq_prod_bd;

	if (bp->spq_prod_bd == bp->spq_last_bd) {
		bp->spq_prod_bd = bp->spq;
		bp->spq_prod_idx = 0;
M
Merav Sicron 已提交
3142
		DP(BNX2X_MSG_SP, "end of spq\n");
3143 3144 3145 3146 3147 3148 3149 3150
	} else {
		bp->spq_prod_bd++;
		bp->spq_prod_idx++;
	}
	return next_spe;
}

/* must be called under the spq lock */
M
Michael Chan 已提交
3151 3152 3153 3154
static inline void bnx2x_sp_prod_update(struct bnx2x *bp)
{
	int func = BP_FUNC(bp);

V
Vladislav Zolotarov 已提交
3155 3156 3157 3158 3159 3160
	/*
	 * Make sure that BD data is updated before writing the producer:
	 * BD data is written to the memory, the producer is read from the
	 * memory, thus we need a full memory barrier to ensure the ordering.
	 */
	mb();
M
Michael Chan 已提交
3161

3162
	REG_WR16(bp, BAR_XSTRORM_INTMEM + XSTORM_SPQ_PROD_OFFSET(func),
D
Dmitry Kravkov 已提交
3163
		 bp->spq_prod_idx);
M
Michael Chan 已提交
3164 3165 3166
	mmiowb();
}

3167 3168 3169 3170 3171 3172 3173 3174 3175
/**
 * bnx2x_is_contextless_ramrod - check if the current command ends on EQ
 *
 * @cmd:	command to check
 * @cmd_type:	command type
 */
static inline bool bnx2x_is_contextless_ramrod(int cmd, int cmd_type)
{
	if ((cmd_type == NONE_CONNECTION_TYPE) ||
3176
	    (cmd == RAMROD_CMD_ID_ETH_FORWARD_SETUP) ||
3177 3178 3179 3180 3181 3182 3183 3184 3185 3186 3187 3188 3189 3190 3191 3192 3193 3194 3195 3196 3197 3198 3199 3200 3201 3202
	    (cmd == RAMROD_CMD_ID_ETH_CLASSIFICATION_RULES) ||
	    (cmd == RAMROD_CMD_ID_ETH_FILTER_RULES) ||
	    (cmd == RAMROD_CMD_ID_ETH_MULTICAST_RULES) ||
	    (cmd == RAMROD_CMD_ID_ETH_SET_MAC) ||
	    (cmd == RAMROD_CMD_ID_ETH_RSS_UPDATE))
		return true;
	else
		return false;

}


/**
 * bnx2x_sp_post - place a single command on an SP ring
 *
 * @bp:		driver handle
 * @command:	command to place (e.g. SETUP, FILTER_RULES, etc.)
 * @cid:	SW CID the command is related to
 * @data_hi:	command private data address (high 32 bits)
 * @data_lo:	command private data address (low 32 bits)
 * @cmd_type:	command type (e.g. NONE, ETH)
 *
 * SP data is handled as if it's always an address pair, thus data fields are
 * not swapped to little endian in upper functions. Instead this function swaps
 * data as if it's two u32 fields.
 */
D
Dmitry Kravkov 已提交
3203
int bnx2x_sp_post(struct bnx2x *bp, int command, int cid,
3204
		  u32 data_hi, u32 data_lo, int cmd_type)
E
Eliezer Tamir 已提交
3205
{
M
Michael Chan 已提交
3206
	struct eth_spe *spe;
3207
	u16 type;
3208
	bool common = bnx2x_is_contextless_ramrod(command, cmd_type);
E
Eliezer Tamir 已提交
3209 3210

#ifdef BNX2X_STOP_ON_ERROR
M
Merav Sicron 已提交
3211 3212
	if (unlikely(bp->panic)) {
		BNX2X_ERR("Can't post SP when there is panic\n");
E
Eliezer Tamir 已提交
3213
		return -EIO;
M
Merav Sicron 已提交
3214
	}
E
Eliezer Tamir 已提交
3215 3216
#endif

3217
	spin_lock_bh(&bp->spq_lock);
E
Eliezer Tamir 已提交
3218

3219 3220 3221 3222 3223 3224 3225 3226 3227 3228 3229 3230
	if (common) {
		if (!atomic_read(&bp->eq_spq_left)) {
			BNX2X_ERR("BUG! EQ ring full!\n");
			spin_unlock_bh(&bp->spq_lock);
			bnx2x_panic();
			return -EBUSY;
		}
	} else if (!atomic_read(&bp->cq_spq_left)) {
			BNX2X_ERR("BUG! SPQ ring full!\n");
			spin_unlock_bh(&bp->spq_lock);
			bnx2x_panic();
			return -EBUSY;
E
Eliezer Tamir 已提交
3231
	}
E
Eliezer Tamir 已提交
3232

M
Michael Chan 已提交
3233 3234
	spe = bnx2x_sp_get_next(bp);

E
Eliezer Tamir 已提交
3235
	/* CID needs port number to be encoded int it */
M
Michael Chan 已提交
3236
	spe->hdr.conn_and_cmd_data =
V
Vladislav Zolotarov 已提交
3237 3238
			cpu_to_le32((command << SPE_HDR_CMD_ID_SHIFT) |
				    HW_CID(bp, cid));
3239

3240
	type = (cmd_type << SPE_HDR_CONN_TYPE_SHIFT) & SPE_HDR_CONN_TYPE;
E
Eliezer Tamir 已提交
3241

3242 3243
	type |= ((BP_FUNC(bp) << SPE_HDR_FUNCTION_ID_SHIFT) &
		 SPE_HDR_FUNCTION_ID);
E
Eliezer Tamir 已提交
3244

3245 3246 3247 3248 3249
	spe->hdr.type = cpu_to_le16(type);

	spe->data.update_data_addr.hi = cpu_to_le32(data_hi);
	spe->data.update_data_addr.lo = cpu_to_le32(data_lo);

3250 3251 3252 3253 3254 3255 3256 3257 3258
	/*
	 * It's ok if the actual decrement is issued towards the memory
	 * somewhere between the spin_lock and spin_unlock. Thus no
	 * more explict memory barrier is needed.
	 */
	if (common)
		atomic_dec(&bp->eq_spq_left);
	else
		atomic_dec(&bp->cq_spq_left);
3259

E
Eliezer Tamir 已提交
3260

M
Merav Sicron 已提交
3261 3262
	DP(BNX2X_MSG_SP,
	   "SPQE[%x] (%x:%x)  (cmd, common?) (%d,%d)  hw_cid %x  data (%x:%x) type(0x%x) left (CQ, EQ) (%x,%x)\n",
V
Vladislav Zolotarov 已提交
3263 3264
	   bp->spq_prod_idx, (u32)U64_HI(bp->spq_mapping),
	   (u32)(U64_LO(bp->spq_mapping) +
3265
	   (void *)bp->spq_prod_bd - (void *)bp->spq), command, common,
3266 3267
	   HW_CID(bp, cid), data_hi, data_lo, type,
	   atomic_read(&bp->cq_spq_left), atomic_read(&bp->eq_spq_left));
V
Vladislav Zolotarov 已提交
3268

M
Michael Chan 已提交
3269
	bnx2x_sp_prod_update(bp);
3270
	spin_unlock_bh(&bp->spq_lock);
E
Eliezer Tamir 已提交
3271 3272 3273 3274
	return 0;
}

/* acquire split MCP access lock register */
Y
Yitchak Gertner 已提交
3275
static int bnx2x_acquire_alr(struct bnx2x *bp)
E
Eliezer Tamir 已提交
3276
{
3277
	u32 j, val;
3278
	int rc = 0;
E
Eliezer Tamir 已提交
3279 3280

	might_sleep();
3281
	for (j = 0; j < 1000; j++) {
E
Eliezer Tamir 已提交
3282 3283 3284 3285 3286 3287 3288 3289 3290
		val = (1UL << 31);
		REG_WR(bp, GRCBASE_MCP + 0x9c, val);
		val = REG_RD(bp, GRCBASE_MCP + 0x9c);
		if (val & (1L << 31))
			break;

		msleep(5);
	}
	if (!(val & (1L << 31))) {
3291
		BNX2X_ERR("Cannot acquire MCP access lock register\n");
E
Eliezer Tamir 已提交
3292 3293 3294 3295 3296 3297
		rc = -EBUSY;
	}

	return rc;
}

Y
Yitchak Gertner 已提交
3298 3299
/* release split MCP access lock register */
static void bnx2x_release_alr(struct bnx2x *bp)
E
Eliezer Tamir 已提交
3300
{
3301
	REG_WR(bp, GRCBASE_MCP + 0x9c, 0);
E
Eliezer Tamir 已提交
3302 3303
}

3304 3305 3306
#define BNX2X_DEF_SB_ATT_IDX	0x0001
#define BNX2X_DEF_SB_IDX	0x0002

E
Eliezer Tamir 已提交
3307 3308
static inline u16 bnx2x_update_dsb_idx(struct bnx2x *bp)
{
3309
	struct host_sp_status_block *def_sb = bp->def_status_blk;
E
Eliezer Tamir 已提交
3310 3311 3312 3313 3314
	u16 rc = 0;

	barrier(); /* status block is written to by the chip */
	if (bp->def_att_idx != def_sb->atten_status_block.attn_bits_index) {
		bp->def_att_idx = def_sb->atten_status_block.attn_bits_index;
3315
		rc |= BNX2X_DEF_SB_ATT_IDX;
E
Eliezer Tamir 已提交
3316
	}
3317 3318 3319 3320

	if (bp->def_idx != def_sb->sp_sb.running_index) {
		bp->def_idx = def_sb->sp_sb.running_index;
		rc |= BNX2X_DEF_SB_IDX;
E
Eliezer Tamir 已提交
3321
	}
3322 3323 3324

	/* Do not reorder: indecies reading should complete before handling */
	barrier();
E
Eliezer Tamir 已提交
3325 3326 3327 3328 3329 3330 3331 3332 3333
	return rc;
}

/*
 * slow path service functions
 */

static void bnx2x_attn_int_asserted(struct bnx2x *bp, u32 asserted)
{
3334
	int port = BP_PORT(bp);
E
Eliezer Tamir 已提交
3335 3336
	u32 aeu_addr = port ? MISC_REG_AEU_MASK_ATTN_FUNC_1 :
			      MISC_REG_AEU_MASK_ATTN_FUNC_0;
3337 3338
	u32 nig_int_mask_addr = port ? NIG_REG_MASK_INTERRUPT_PORT1 :
				       NIG_REG_MASK_INTERRUPT_PORT0;
E
Eilon Greenstein 已提交
3339
	u32 aeu_mask;
3340
	u32 nig_mask = 0;
D
Dmitry Kravkov 已提交
3341
	u32 reg_addr;
E
Eliezer Tamir 已提交
3342 3343 3344 3345

	if (bp->attn_state & asserted)
		BNX2X_ERR("IGU ERROR\n");

E
Eilon Greenstein 已提交
3346 3347 3348
	bnx2x_acquire_hw_lock(bp, HW_LOCK_RESOURCE_PORT0_ATT_MASK + port);
	aeu_mask = REG_RD(bp, aeu_addr);

E
Eliezer Tamir 已提交
3349
	DP(NETIF_MSG_HW, "aeu_mask %x  newly asserted %x\n",
E
Eilon Greenstein 已提交
3350
	   aeu_mask, asserted);
3351
	aeu_mask &= ~(asserted & 0x3ff);
E
Eilon Greenstein 已提交
3352
	DP(NETIF_MSG_HW, "new mask %x\n", aeu_mask);
E
Eliezer Tamir 已提交
3353

E
Eilon Greenstein 已提交
3354 3355
	REG_WR(bp, aeu_addr, aeu_mask);
	bnx2x_release_hw_lock(bp, HW_LOCK_RESOURCE_PORT0_ATT_MASK + port);
E
Eliezer Tamir 已提交
3356

E
Eilon Greenstein 已提交
3357
	DP(NETIF_MSG_HW, "attn_state %x\n", bp->attn_state);
E
Eliezer Tamir 已提交
3358
	bp->attn_state |= asserted;
E
Eilon Greenstein 已提交
3359
	DP(NETIF_MSG_HW, "new state %x\n", bp->attn_state);
E
Eliezer Tamir 已提交
3360 3361 3362 3363

	if (asserted & ATTN_HARD_WIRED_MASK) {
		if (asserted & ATTN_NIG_FOR_FUNC) {

3364 3365
			bnx2x_acquire_phy_lock(bp);

3366
			/* save nig interrupt mask */
3367
			nig_mask = REG_RD(bp, nig_int_mask_addr);
E
Eliezer Tamir 已提交
3368

3369 3370 3371 3372 3373 3374 3375 3376
			/* If nig_mask is not set, no need to call the update
			 * function.
			 */
			if (nig_mask) {
				REG_WR(bp, nig_int_mask_addr, 0);

				bnx2x_link_attn(bp);
			}
E
Eliezer Tamir 已提交
3377 3378 3379 3380 3381 3382 3383 3384 3385 3386 3387 3388 3389 3390 3391 3392 3393 3394 3395 3396 3397 3398 3399 3400 3401 3402 3403 3404 3405 3406 3407 3408 3409 3410 3411 3412 3413 3414 3415 3416 3417 3418 3419 3420 3421

			/* handle unicore attn? */
		}
		if (asserted & ATTN_SW_TIMER_4_FUNC)
			DP(NETIF_MSG_HW, "ATTN_SW_TIMER_4_FUNC!\n");

		if (asserted & GPIO_2_FUNC)
			DP(NETIF_MSG_HW, "GPIO_2_FUNC!\n");

		if (asserted & GPIO_3_FUNC)
			DP(NETIF_MSG_HW, "GPIO_3_FUNC!\n");

		if (asserted & GPIO_4_FUNC)
			DP(NETIF_MSG_HW, "GPIO_4_FUNC!\n");

		if (port == 0) {
			if (asserted & ATTN_GENERAL_ATTN_1) {
				DP(NETIF_MSG_HW, "ATTN_GENERAL_ATTN_1!\n");
				REG_WR(bp, MISC_REG_AEU_GENERAL_ATTN_1, 0x0);
			}
			if (asserted & ATTN_GENERAL_ATTN_2) {
				DP(NETIF_MSG_HW, "ATTN_GENERAL_ATTN_2!\n");
				REG_WR(bp, MISC_REG_AEU_GENERAL_ATTN_2, 0x0);
			}
			if (asserted & ATTN_GENERAL_ATTN_3) {
				DP(NETIF_MSG_HW, "ATTN_GENERAL_ATTN_3!\n");
				REG_WR(bp, MISC_REG_AEU_GENERAL_ATTN_3, 0x0);
			}
		} else {
			if (asserted & ATTN_GENERAL_ATTN_4) {
				DP(NETIF_MSG_HW, "ATTN_GENERAL_ATTN_4!\n");
				REG_WR(bp, MISC_REG_AEU_GENERAL_ATTN_4, 0x0);
			}
			if (asserted & ATTN_GENERAL_ATTN_5) {
				DP(NETIF_MSG_HW, "ATTN_GENERAL_ATTN_5!\n");
				REG_WR(bp, MISC_REG_AEU_GENERAL_ATTN_5, 0x0);
			}
			if (asserted & ATTN_GENERAL_ATTN_6) {
				DP(NETIF_MSG_HW, "ATTN_GENERAL_ATTN_6!\n");
				REG_WR(bp, MISC_REG_AEU_GENERAL_ATTN_6, 0x0);
			}
		}

	} /* if hardwired */

D
Dmitry Kravkov 已提交
3422 3423 3424 3425 3426 3427 3428 3429 3430
	if (bp->common.int_block == INT_BLOCK_HC)
		reg_addr = (HC_REG_COMMAND_REG + port*32 +
			    COMMAND_REG_ATTN_BITS_SET);
	else
		reg_addr = (BAR_IGU_INTMEM + IGU_CMD_ATTN_BIT_SET_UPPER*8);

	DP(NETIF_MSG_HW, "about to mask 0x%08x at %s addr 0x%x\n", asserted,
	   (bp->common.int_block == INT_BLOCK_HC) ? "HC" : "IGU", reg_addr);
	REG_WR(bp, reg_addr, asserted);
E
Eliezer Tamir 已提交
3431 3432

	/* now set back the mask */
3433
	if (asserted & ATTN_NIG_FOR_FUNC) {
3434
		REG_WR(bp, nig_int_mask_addr, nig_mask);
3435 3436
		bnx2x_release_phy_lock(bp);
	}
E
Eliezer Tamir 已提交
3437 3438
}

E
Eilon Greenstein 已提交
3439 3440 3441
static inline void bnx2x_fan_failure(struct bnx2x *bp)
{
	int port = BP_PORT(bp);
Y
Yaniv Rosner 已提交
3442
	u32 ext_phy_config;
E
Eilon Greenstein 已提交
3443
	/* mark the failure */
Y
Yaniv Rosner 已提交
3444 3445 3446 3447 3448 3449
	ext_phy_config =
		SHMEM_RD(bp,
			 dev_info.port_hw_config[port].external_phy_config);

	ext_phy_config &= ~PORT_HW_CFG_XGXS_EXT_PHY_TYPE_MASK;
	ext_phy_config |= PORT_HW_CFG_XGXS_EXT_PHY_TYPE_FAILURE;
E
Eilon Greenstein 已提交
3450
	SHMEM_WR(bp, dev_info.port_hw_config[port].external_phy_config,
Y
Yaniv Rosner 已提交
3451
		 ext_phy_config);
E
Eilon Greenstein 已提交
3452 3453

	/* log the failure */
M
Merav Sicron 已提交
3454 3455
	netdev_err(bp->dev, "Fan Failure on Network Controller has caused the driver to shutdown the card to prevent permanent damage.\n"
			    "Please contact OEM Support for assistance\n");
3456 3457 3458 3459 3460 3461 3462 3463 3464 3465 3466

	/*
	 * Scheudle device reset (unload)
	 * This is due to some boards consuming sufficient power when driver is
	 * up to overheat if fan fails.
	 */
	smp_mb__before_clear_bit();
	set_bit(BNX2X_SP_RTNL_FAN_FAILURE, &bp->sp_rtnl_state);
	smp_mb__after_clear_bit();
	schedule_delayed_work(&bp->sp_rtnl_task, 0);

E
Eilon Greenstein 已提交
3467
}
3468

3469
static inline void bnx2x_attn_int_deasserted0(struct bnx2x *bp, u32 attn)
E
Eliezer Tamir 已提交
3470
{
3471
	int port = BP_PORT(bp);
3472
	int reg_offset;
3473
	u32 val;
3474

3475 3476
	reg_offset = (port ? MISC_REG_AEU_ENABLE1_FUNC_1_OUT_0 :
			     MISC_REG_AEU_ENABLE1_FUNC_0_OUT_0);
3477

3478
	if (attn & AEU_INPUTS_ATTN_BITS_SPIO5) {
3479 3480 3481 3482 3483 3484 3485

		val = REG_RD(bp, reg_offset);
		val &= ~AEU_INPUTS_ATTN_BITS_SPIO5;
		REG_WR(bp, reg_offset, val);

		BNX2X_ERR("SPIO5 hw attention\n");

E
Eilon Greenstein 已提交
3486
		/* Fan failure attention */
3487
		bnx2x_hw_reset_phy(&bp->link_params);
E
Eilon Greenstein 已提交
3488
		bnx2x_fan_failure(bp);
3489
	}
3490

3491
	if ((attn & bp->link_vars.aeu_int_mask) && bp->port.pmf) {
E
Eilon Greenstein 已提交
3492 3493 3494 3495 3496
		bnx2x_acquire_phy_lock(bp);
		bnx2x_handle_module_detect_int(&bp->link_params);
		bnx2x_release_phy_lock(bp);
	}

3497 3498 3499 3500 3501 3502 3503
	if (attn & HW_INTERRUT_ASSERT_SET_0) {

		val = REG_RD(bp, reg_offset);
		val &= ~(attn & HW_INTERRUT_ASSERT_SET_0);
		REG_WR(bp, reg_offset, val);

		BNX2X_ERR("FATAL HW block attention set0 0x%x\n",
3504
			  (u32)(attn & HW_INTERRUT_ASSERT_SET_0));
3505 3506
		bnx2x_panic();
	}
3507 3508 3509 3510 3511 3512
}

static inline void bnx2x_attn_int_deasserted1(struct bnx2x *bp, u32 attn)
{
	u32 val;

3513
	if (attn & AEU_INPUTS_ATTN_BITS_DOORBELLQ_HW_INTERRUPT) {
3514 3515 3516 3517 3518 3519 3520

		val = REG_RD(bp, DORQ_REG_DORQ_INT_STS_CLR);
		BNX2X_ERR("DB hw attention 0x%x\n", val);
		/* DORQ discard attention */
		if (val & 0x2)
			BNX2X_ERR("FATAL error from DORQ\n");
	}
3521 3522 3523 3524 3525 3526 3527 3528 3529 3530 3531 3532 3533 3534

	if (attn & HW_INTERRUT_ASSERT_SET_1) {

		int port = BP_PORT(bp);
		int reg_offset;

		reg_offset = (port ? MISC_REG_AEU_ENABLE1_FUNC_1_OUT_1 :
				     MISC_REG_AEU_ENABLE1_FUNC_0_OUT_1);

		val = REG_RD(bp, reg_offset);
		val &= ~(attn & HW_INTERRUT_ASSERT_SET_1);
		REG_WR(bp, reg_offset, val);

		BNX2X_ERR("FATAL HW block attention set1 0x%x\n",
3535
			  (u32)(attn & HW_INTERRUT_ASSERT_SET_1));
3536 3537
		bnx2x_panic();
	}
3538 3539 3540 3541 3542 3543 3544 3545 3546 3547 3548 3549 3550 3551 3552 3553 3554
}

static inline void bnx2x_attn_int_deasserted2(struct bnx2x *bp, u32 attn)
{
	u32 val;

	if (attn & AEU_INPUTS_ATTN_BITS_CFC_HW_INTERRUPT) {

		val = REG_RD(bp, CFC_REG_CFC_INT_STS_CLR);
		BNX2X_ERR("CFC hw attention 0x%x\n", val);
		/* CFC error attention */
		if (val & 0x2)
			BNX2X_ERR("FATAL error from CFC\n");
	}

	if (attn & AEU_INPUTS_ATTN_BITS_PXP_HW_INTERRUPT) {
		val = REG_RD(bp, PXP_REG_PXP_INT_STS_CLR_0);
3555
		BNX2X_ERR("PXP hw attention-0 0x%x\n", val);
3556 3557 3558
		/* RQ_USDMDP_FIFO_OVERFLOW */
		if (val & 0x18000)
			BNX2X_ERR("FATAL error from PXP\n");
3559 3560

		if (!CHIP_IS_E1x(bp)) {
D
Dmitry Kravkov 已提交
3561 3562 3563
			val = REG_RD(bp, PXP_REG_PXP_INT_STS_CLR_1);
			BNX2X_ERR("PXP hw attention-1 0x%x\n", val);
		}
3564
	}
3565 3566 3567 3568 3569 3570 3571 3572 3573 3574 3575 3576 3577 3578

	if (attn & HW_INTERRUT_ASSERT_SET_2) {

		int port = BP_PORT(bp);
		int reg_offset;

		reg_offset = (port ? MISC_REG_AEU_ENABLE1_FUNC_1_OUT_2 :
				     MISC_REG_AEU_ENABLE1_FUNC_0_OUT_2);

		val = REG_RD(bp, reg_offset);
		val &= ~(attn & HW_INTERRUT_ASSERT_SET_2);
		REG_WR(bp, reg_offset, val);

		BNX2X_ERR("FATAL HW block attention set2 0x%x\n",
3579
			  (u32)(attn & HW_INTERRUT_ASSERT_SET_2));
3580 3581
		bnx2x_panic();
	}
3582 3583 3584 3585
}

static inline void bnx2x_attn_int_deasserted3(struct bnx2x *bp, u32 attn)
{
3586 3587
	u32 val;

3588 3589
	if (attn & EVEREST_GEN_ATTN_IN_USE_MASK) {

3590 3591 3592 3593
		if (attn & BNX2X_PMF_LINK_ASSERT) {
			int func = BP_FUNC(bp);

			REG_WR(bp, MISC_REG_AEU_GENERAL_ATTN_12 + func*4, 0);
D
Dmitry Kravkov 已提交
3594 3595 3596 3597
			bp->mf_config[BP_VN(bp)] = MF_CFG_RD(bp,
					func_mf_config[BP_ABS_FUNC(bp)].config);
			val = SHMEM_RD(bp,
				       func_mb[BP_FW_MB_IDX(bp)].drv_status);
3598 3599 3600
			if (val & DRV_STATUS_DCC_EVENT_MASK)
				bnx2x_dcc_event(bp,
					    (val & DRV_STATUS_DCC_EVENT_MASK));
3601 3602 3603 3604

			if (val & DRV_STATUS_SET_MF_BW)
				bnx2x_set_mf_bw(bp);

3605 3606
			if (val & DRV_STATUS_DRV_INFO_REQ)
				bnx2x_handle_drv_info_req(bp);
3607
			if ((bp->port.pmf == 0) && (val & DRV_STATUS_PMF))
3608 3609
				bnx2x_pmf_update(bp);

V
Vladislav Zolotarov 已提交
3610
			if (bp->port.pmf &&
S
Shmulik Ravid 已提交
3611 3612
			    (val & DRV_STATUS_DCBX_NEGOTIATION_RESULTS) &&
				bp->dcbx_enabled > 0)
V
Vladislav Zolotarov 已提交
3613 3614 3615
				/* start dcbx state machine */
				bnx2x_dcbx_set_params(bp,
					BNX2X_DCBX_STATE_NEG_RECEIVED);
3616 3617 3618 3619 3620 3621 3622 3623 3624 3625 3626 3627 3628 3629 3630
			if (bp->link_vars.periodic_flags &
			    PERIODIC_FLAGS_LINK_EVENT) {
				/*  sync with link */
				bnx2x_acquire_phy_lock(bp);
				bp->link_vars.periodic_flags &=
					~PERIODIC_FLAGS_LINK_EVENT;
				bnx2x_release_phy_lock(bp);
				if (IS_MF(bp))
					bnx2x_link_sync_notify(bp);
				bnx2x_link_report(bp);
			}
			/* Always call it here: bnx2x_link_report() will
			 * prevent the link indication duplication.
			 */
			bnx2x__link_status_update(bp);
3631
		} else if (attn & BNX2X_MC_ASSERT_BITS) {
3632 3633

			BNX2X_ERR("MC assert!\n");
3634
			bnx2x_mc_assert(bp);
3635 3636 3637 3638 3639 3640 3641 3642 3643 3644
			REG_WR(bp, MISC_REG_AEU_GENERAL_ATTN_10, 0);
			REG_WR(bp, MISC_REG_AEU_GENERAL_ATTN_9, 0);
			REG_WR(bp, MISC_REG_AEU_GENERAL_ATTN_8, 0);
			REG_WR(bp, MISC_REG_AEU_GENERAL_ATTN_7, 0);
			bnx2x_panic();

		} else if (attn & BNX2X_MCP_ASSERT) {

			BNX2X_ERR("MCP assert!\n");
			REG_WR(bp, MISC_REG_AEU_GENERAL_ATTN_11, 0);
3645
			bnx2x_fw_dump(bp);
3646 3647 3648 3649 3650 3651

		} else
			BNX2X_ERR("Unknown HW assert! (attn 0x%x)\n", attn);
	}

	if (attn & EVEREST_LATCHED_ATTN_IN_USE_MASK) {
3652 3653
		BNX2X_ERR("LATCHED attention 0x%08x (masked)\n", attn);
		if (attn & BNX2X_GRC_TIMEOUT) {
D
Dmitry Kravkov 已提交
3654 3655
			val = CHIP_IS_E1(bp) ? 0 :
					REG_RD(bp, MISC_REG_GRC_TIMEOUT_ATTN);
3656 3657 3658
			BNX2X_ERR("GRC time-out 0x%08x\n", val);
		}
		if (attn & BNX2X_GRC_RSV) {
D
Dmitry Kravkov 已提交
3659 3660
			val = CHIP_IS_E1(bp) ? 0 :
					REG_RD(bp, MISC_REG_GRC_RSV_ATTN);
3661 3662
			BNX2X_ERR("GRC reserved 0x%08x\n", val);
		}
3663 3664 3665 3666
		REG_WR(bp, MISC_REG_AEU_CLR_LATCH_SIGNAL, 0x7ff);
	}
}

3667 3668 3669 3670 3671 3672 3673 3674 3675 3676 3677 3678 3679 3680 3681 3682 3683 3684 3685 3686 3687 3688 3689 3690 3691 3692 3693 3694 3695 3696 3697
/*
 * Bits map:
 * 0-7   - Engine0 load counter.
 * 8-15  - Engine1 load counter.
 * 16    - Engine0 RESET_IN_PROGRESS bit.
 * 17    - Engine1 RESET_IN_PROGRESS bit.
 * 18    - Engine0 ONE_IS_LOADED. Set when there is at least one active function
 *         on the engine
 * 19    - Engine1 ONE_IS_LOADED.
 * 20    - Chip reset flow bit. When set none-leader must wait for both engines
 *         leader to complete (check for both RESET_IN_PROGRESS bits and not for
 *         just the one belonging to its engine).
 *
 */
#define BNX2X_RECOVERY_GLOB_REG		MISC_REG_GENERIC_POR_1

#define BNX2X_PATH0_LOAD_CNT_MASK	0x000000ff
#define BNX2X_PATH0_LOAD_CNT_SHIFT	0
#define BNX2X_PATH1_LOAD_CNT_MASK	0x0000ff00
#define BNX2X_PATH1_LOAD_CNT_SHIFT	8
#define BNX2X_PATH0_RST_IN_PROG_BIT	0x00010000
#define BNX2X_PATH1_RST_IN_PROG_BIT	0x00020000
#define BNX2X_GLOBAL_RESET_BIT		0x00040000

/*
 * Set the GLOBAL_RESET bit.
 *
 * Should be run under rtnl lock
 */
void bnx2x_set_reset_global(struct bnx2x *bp)
{
A
Ariel Elior 已提交
3698 3699 3700
	u32 val;
	bnx2x_acquire_hw_lock(bp, HW_LOCK_RESOURCE_RECOVERY_REG);
	val = REG_RD(bp, BNX2X_RECOVERY_GLOB_REG);
3701
	REG_WR(bp, BNX2X_RECOVERY_GLOB_REG, val | BNX2X_GLOBAL_RESET_BIT);
A
Ariel Elior 已提交
3702
	bnx2x_release_hw_lock(bp, HW_LOCK_RESOURCE_RECOVERY_REG);
3703 3704 3705 3706 3707 3708 3709 3710 3711
}

/*
 * Clear the GLOBAL_RESET bit.
 *
 * Should be run under rtnl lock
 */
static inline void bnx2x_clear_reset_global(struct bnx2x *bp)
{
A
Ariel Elior 已提交
3712 3713 3714
	u32 val;
	bnx2x_acquire_hw_lock(bp, HW_LOCK_RESOURCE_RECOVERY_REG);
	val = REG_RD(bp, BNX2X_RECOVERY_GLOB_REG);
3715
	REG_WR(bp, BNX2X_RECOVERY_GLOB_REG, val & (~BNX2X_GLOBAL_RESET_BIT));
A
Ariel Elior 已提交
3716
	bnx2x_release_hw_lock(bp, HW_LOCK_RESOURCE_RECOVERY_REG);
3717
}
D
Dmitry Kravkov 已提交
3718

3719
/*
3720 3721
 * Checks the GLOBAL_RESET bit.
 *
3722 3723
 * should be run under rtnl lock
 */
3724 3725 3726 3727 3728 3729 3730 3731 3732 3733 3734 3735 3736
static inline bool bnx2x_reset_is_global(struct bnx2x *bp)
{
	u32 val	= REG_RD(bp, BNX2X_RECOVERY_GLOB_REG);

	DP(NETIF_MSG_HW, "GEN_REG_VAL=0x%08x\n", val);
	return (val & BNX2X_GLOBAL_RESET_BIT) ? true : false;
}

/*
 * Clear RESET_IN_PROGRESS bit for the current engine.
 *
 * Should be run under rtnl lock
 */
3737 3738
static inline void bnx2x_set_reset_done(struct bnx2x *bp)
{
A
Ariel Elior 已提交
3739
	u32 val;
3740 3741
	u32 bit = BP_PATH(bp) ?
		BNX2X_PATH1_RST_IN_PROG_BIT : BNX2X_PATH0_RST_IN_PROG_BIT;
A
Ariel Elior 已提交
3742 3743
	bnx2x_acquire_hw_lock(bp, HW_LOCK_RESOURCE_RECOVERY_REG);
	val = REG_RD(bp, BNX2X_RECOVERY_GLOB_REG);
3744 3745 3746 3747

	/* Clear the bit */
	val &= ~bit;
	REG_WR(bp, BNX2X_RECOVERY_GLOB_REG, val);
A
Ariel Elior 已提交
3748 3749

	bnx2x_release_hw_lock(bp, HW_LOCK_RESOURCE_RECOVERY_REG);
3750 3751 3752
}

/*
3753 3754
 * Set RESET_IN_PROGRESS for the current engine.
 *
3755 3756
 * should be run under rtnl lock
 */
3757
void bnx2x_set_reset_in_progress(struct bnx2x *bp)
3758
{
A
Ariel Elior 已提交
3759
	u32 val;
3760 3761
	u32 bit = BP_PATH(bp) ?
		BNX2X_PATH1_RST_IN_PROG_BIT : BNX2X_PATH0_RST_IN_PROG_BIT;
A
Ariel Elior 已提交
3762 3763
	bnx2x_acquire_hw_lock(bp, HW_LOCK_RESOURCE_RECOVERY_REG);
	val = REG_RD(bp, BNX2X_RECOVERY_GLOB_REG);
3764 3765 3766 3767

	/* Set the bit */
	val |= bit;
	REG_WR(bp, BNX2X_RECOVERY_GLOB_REG, val);
A
Ariel Elior 已提交
3768
	bnx2x_release_hw_lock(bp, HW_LOCK_RESOURCE_RECOVERY_REG);
3769 3770 3771
}

/*
3772
 * Checks the RESET_IN_PROGRESS bit for the given engine.
3773 3774
 * should be run under rtnl lock
 */
3775
bool bnx2x_reset_is_done(struct bnx2x *bp, int engine)
3776
{
3777 3778 3779 3780 3781 3782
	u32 val	= REG_RD(bp, BNX2X_RECOVERY_GLOB_REG);
	u32 bit = engine ?
		BNX2X_PATH1_RST_IN_PROG_BIT : BNX2X_PATH0_RST_IN_PROG_BIT;

	/* return false if bit is set */
	return (val & bit) ? false : true;
3783 3784 3785
}

/*
3786
 * set pf load for the current pf.
3787
 *
3788 3789
 * should be run under rtnl lock
 */
3790
void bnx2x_set_pf_load(struct bnx2x *bp)
3791
{
A
Ariel Elior 已提交
3792
	u32 val1, val;
3793 3794 3795 3796
	u32 mask = BP_PATH(bp) ? BNX2X_PATH1_LOAD_CNT_MASK :
			     BNX2X_PATH0_LOAD_CNT_MASK;
	u32 shift = BP_PATH(bp) ? BNX2X_PATH1_LOAD_CNT_SHIFT :
			     BNX2X_PATH0_LOAD_CNT_SHIFT;
3797

A
Ariel Elior 已提交
3798 3799 3800
	bnx2x_acquire_hw_lock(bp, HW_LOCK_RESOURCE_RECOVERY_REG);
	val = REG_RD(bp, BNX2X_RECOVERY_GLOB_REG);

M
Merav Sicron 已提交
3801
	DP(NETIF_MSG_IFUP, "Old GEN_REG_VAL=0x%08x\n", val);
3802

3803 3804 3805
	/* get the current counter value */
	val1 = (val & mask) >> shift;

3806 3807
	/* set bit of that PF */
	val1 |= (1 << bp->pf_num);
3808 3809 3810 3811 3812 3813 3814 3815

	/* clear the old value */
	val &= ~mask;

	/* set the new one */
	val |= ((val1 << shift) & mask);

	REG_WR(bp, BNX2X_RECOVERY_GLOB_REG, val);
A
Ariel Elior 已提交
3816
	bnx2x_release_hw_lock(bp, HW_LOCK_RESOURCE_RECOVERY_REG);
3817 3818
}

3819
/**
3820
 * bnx2x_clear_pf_load - clear pf load mark
3821 3822 3823 3824 3825
 *
 * @bp:		driver handle
 *
 * Should be run under rtnl lock.
 * Decrements the load counter for the current engine. Returns
3826
 * whether other functions are still loaded
3827
 */
3828
bool bnx2x_clear_pf_load(struct bnx2x *bp)
3829
{
A
Ariel Elior 已提交
3830
	u32 val1, val;
3831 3832 3833 3834
	u32 mask = BP_PATH(bp) ? BNX2X_PATH1_LOAD_CNT_MASK :
			     BNX2X_PATH0_LOAD_CNT_MASK;
	u32 shift = BP_PATH(bp) ? BNX2X_PATH1_LOAD_CNT_SHIFT :
			     BNX2X_PATH0_LOAD_CNT_SHIFT;
3835

A
Ariel Elior 已提交
3836 3837
	bnx2x_acquire_hw_lock(bp, HW_LOCK_RESOURCE_RECOVERY_REG);
	val = REG_RD(bp, BNX2X_RECOVERY_GLOB_REG);
M
Merav Sicron 已提交
3838
	DP(NETIF_MSG_IFDOWN, "Old GEN_REG_VAL=0x%08x\n", val);
3839

3840 3841 3842
	/* get the current counter value */
	val1 = (val & mask) >> shift;

3843 3844
	/* clear bit of that PF */
	val1 &= ~(1 << bp->pf_num);
3845 3846 3847 3848 3849 3850 3851 3852

	/* clear the old value */
	val &= ~mask;

	/* set the new one */
	val |= ((val1 << shift) & mask);

	REG_WR(bp, BNX2X_RECOVERY_GLOB_REG, val);
A
Ariel Elior 已提交
3853 3854
	bnx2x_release_hw_lock(bp, HW_LOCK_RESOURCE_RECOVERY_REG);
	return val1 != 0;
3855 3856 3857
}

/*
3858
 * Read the load status for the current engine.
3859
 *
3860 3861
 * should be run under rtnl lock
 */
3862
static inline bool bnx2x_get_load_status(struct bnx2x *bp, int engine)
3863
{
3864 3865 3866 3867 3868 3869
	u32 mask = (engine ? BNX2X_PATH1_LOAD_CNT_MASK :
			     BNX2X_PATH0_LOAD_CNT_MASK);
	u32 shift = (engine ? BNX2X_PATH1_LOAD_CNT_SHIFT :
			     BNX2X_PATH0_LOAD_CNT_SHIFT);
	u32 val = REG_RD(bp, BNX2X_RECOVERY_GLOB_REG);

M
Merav Sicron 已提交
3870
	DP(NETIF_MSG_HW | NETIF_MSG_IFUP, "GLOB_REG=0x%08x\n", val);
3871 3872 3873

	val = (val & mask) >> shift;

M
Merav Sicron 已提交
3874 3875
	DP(NETIF_MSG_HW | NETIF_MSG_IFUP, "load mask for engine %d = 0x%x\n",
	   engine, val);
3876

3877
	return val != 0;
3878 3879
}

3880
/*
3881
 * Reset the load status for the current engine.
3882
 */
3883
static inline void bnx2x_clear_load_status(struct bnx2x *bp)
3884
{
A
Ariel Elior 已提交
3885
	u32 val;
3886
	u32 mask = (BP_PATH(bp) ? BNX2X_PATH1_LOAD_CNT_MASK :
A
Ariel Elior 已提交
3887 3888 3889
		    BNX2X_PATH0_LOAD_CNT_MASK);
	bnx2x_acquire_hw_lock(bp, HW_LOCK_RESOURCE_RECOVERY_REG);
	val = REG_RD(bp, BNX2X_RECOVERY_GLOB_REG);
3890
	REG_WR(bp, BNX2X_RECOVERY_GLOB_REG, val & (~mask));
A
Ariel Elior 已提交
3891
	bnx2x_release_hw_lock(bp, HW_LOCK_RESOURCE_RECOVERY_REG);
3892 3893 3894 3895
}

static inline void _print_next_block(int idx, const char *blk)
{
3896
	pr_cont("%s%s", idx ? ", " : "", blk);
3897 3898
}

3899 3900
static inline int bnx2x_check_blocks_with_parity0(u32 sig, int par_num,
						  bool print)
3901 3902 3903 3904 3905 3906 3907 3908
{
	int i = 0;
	u32 cur_bit = 0;
	for (i = 0; sig; i++) {
		cur_bit = ((u32)0x1 << i);
		if (sig & cur_bit) {
			switch (cur_bit) {
			case AEU_INPUTS_ATTN_BITS_BRB_PARITY_ERROR:
3909 3910
				if (print)
					_print_next_block(par_num++, "BRB");
3911 3912
				break;
			case AEU_INPUTS_ATTN_BITS_PARSER_PARITY_ERROR:
3913 3914
				if (print)
					_print_next_block(par_num++, "PARSER");
3915 3916
				break;
			case AEU_INPUTS_ATTN_BITS_TSDM_PARITY_ERROR:
3917 3918
				if (print)
					_print_next_block(par_num++, "TSDM");
3919 3920
				break;
			case AEU_INPUTS_ATTN_BITS_SEARCHER_PARITY_ERROR:
3921 3922 3923 3924 3925 3926 3927
				if (print)
					_print_next_block(par_num++,
							  "SEARCHER");
				break;
			case AEU_INPUTS_ATTN_BITS_TCM_PARITY_ERROR:
				if (print)
					_print_next_block(par_num++, "TCM");
3928 3929
				break;
			case AEU_INPUTS_ATTN_BITS_TSEMI_PARITY_ERROR:
3930 3931 3932 3933 3934 3935
				if (print)
					_print_next_block(par_num++, "TSEMI");
				break;
			case AEU_INPUTS_ATTN_BITS_PBCLIENT_PARITY_ERROR:
				if (print)
					_print_next_block(par_num++, "XPB");
3936 3937 3938 3939 3940 3941 3942 3943 3944 3945 3946
				break;
			}

			/* Clear the bit */
			sig &= ~cur_bit;
		}
	}

	return par_num;
}

3947 3948
static inline int bnx2x_check_blocks_with_parity1(u32 sig, int par_num,
						  bool *global, bool print)
3949 3950 3951 3952 3953 3954 3955
{
	int i = 0;
	u32 cur_bit = 0;
	for (i = 0; sig; i++) {
		cur_bit = ((u32)0x1 << i);
		if (sig & cur_bit) {
			switch (cur_bit) {
3956 3957 3958
			case AEU_INPUTS_ATTN_BITS_PBF_PARITY_ERROR:
				if (print)
					_print_next_block(par_num++, "PBF");
3959 3960
				break;
			case AEU_INPUTS_ATTN_BITS_QM_PARITY_ERROR:
3961 3962 3963 3964 3965 3966
				if (print)
					_print_next_block(par_num++, "QM");
				break;
			case AEU_INPUTS_ATTN_BITS_TIMERS_PARITY_ERROR:
				if (print)
					_print_next_block(par_num++, "TM");
3967 3968
				break;
			case AEU_INPUTS_ATTN_BITS_XSDM_PARITY_ERROR:
3969 3970 3971 3972 3973 3974
				if (print)
					_print_next_block(par_num++, "XSDM");
				break;
			case AEU_INPUTS_ATTN_BITS_XCM_PARITY_ERROR:
				if (print)
					_print_next_block(par_num++, "XCM");
3975 3976
				break;
			case AEU_INPUTS_ATTN_BITS_XSEMI_PARITY_ERROR:
3977 3978
				if (print)
					_print_next_block(par_num++, "XSEMI");
3979 3980
				break;
			case AEU_INPUTS_ATTN_BITS_DOORBELLQ_PARITY_ERROR:
3981 3982 3983 3984 3985 3986 3987
				if (print)
					_print_next_block(par_num++,
							  "DOORBELLQ");
				break;
			case AEU_INPUTS_ATTN_BITS_NIG_PARITY_ERROR:
				if (print)
					_print_next_block(par_num++, "NIG");
3988 3989
				break;
			case AEU_INPUTS_ATTN_BITS_VAUX_PCI_CORE_PARITY_ERROR:
3990 3991 3992 3993
				if (print)
					_print_next_block(par_num++,
							  "VAUX PCI CORE");
				*global = true;
3994 3995
				break;
			case AEU_INPUTS_ATTN_BITS_DEBUG_PARITY_ERROR:
3996 3997
				if (print)
					_print_next_block(par_num++, "DEBUG");
3998 3999
				break;
			case AEU_INPUTS_ATTN_BITS_USDM_PARITY_ERROR:
4000 4001
				if (print)
					_print_next_block(par_num++, "USDM");
4002
				break;
4003 4004 4005 4006
			case AEU_INPUTS_ATTN_BITS_UCM_PARITY_ERROR:
				if (print)
					_print_next_block(par_num++, "UCM");
				break;
4007
			case AEU_INPUTS_ATTN_BITS_USEMI_PARITY_ERROR:
4008 4009
				if (print)
					_print_next_block(par_num++, "USEMI");
4010 4011
				break;
			case AEU_INPUTS_ATTN_BITS_UPB_PARITY_ERROR:
4012 4013
				if (print)
					_print_next_block(par_num++, "UPB");
4014 4015
				break;
			case AEU_INPUTS_ATTN_BITS_CSDM_PARITY_ERROR:
4016 4017
				if (print)
					_print_next_block(par_num++, "CSDM");
4018
				break;
4019 4020 4021 4022
			case AEU_INPUTS_ATTN_BITS_CCM_PARITY_ERROR:
				if (print)
					_print_next_block(par_num++, "CCM");
				break;
4023 4024 4025 4026 4027 4028 4029 4030 4031 4032
			}

			/* Clear the bit */
			sig &= ~cur_bit;
		}
	}

	return par_num;
}

4033 4034
static inline int bnx2x_check_blocks_with_parity2(u32 sig, int par_num,
						  bool print)
4035 4036 4037 4038 4039 4040 4041 4042
{
	int i = 0;
	u32 cur_bit = 0;
	for (i = 0; sig; i++) {
		cur_bit = ((u32)0x1 << i);
		if (sig & cur_bit) {
			switch (cur_bit) {
			case AEU_INPUTS_ATTN_BITS_CSEMI_PARITY_ERROR:
4043 4044
				if (print)
					_print_next_block(par_num++, "CSEMI");
4045 4046
				break;
			case AEU_INPUTS_ATTN_BITS_PXP_PARITY_ERROR:
4047 4048
				if (print)
					_print_next_block(par_num++, "PXP");
4049 4050
				break;
			case AEU_IN_ATTN_BITS_PXPPCICLOCKCLIENT_PARITY_ERROR:
4051 4052
				if (print)
					_print_next_block(par_num++,
4053 4054 4055
					"PXPPCICLOCKCLIENT");
				break;
			case AEU_INPUTS_ATTN_BITS_CFC_PARITY_ERROR:
4056 4057
				if (print)
					_print_next_block(par_num++, "CFC");
4058 4059
				break;
			case AEU_INPUTS_ATTN_BITS_CDU_PARITY_ERROR:
4060 4061 4062 4063 4064 4065
				if (print)
					_print_next_block(par_num++, "CDU");
				break;
			case AEU_INPUTS_ATTN_BITS_DMAE_PARITY_ERROR:
				if (print)
					_print_next_block(par_num++, "DMAE");
4066 4067
				break;
			case AEU_INPUTS_ATTN_BITS_IGU_PARITY_ERROR:
4068 4069
				if (print)
					_print_next_block(par_num++, "IGU");
4070 4071
				break;
			case AEU_INPUTS_ATTN_BITS_MISC_PARITY_ERROR:
4072 4073
				if (print)
					_print_next_block(par_num++, "MISC");
4074 4075 4076 4077 4078 4079 4080 4081 4082 4083 4084
				break;
			}

			/* Clear the bit */
			sig &= ~cur_bit;
		}
	}

	return par_num;
}

4085 4086
static inline int bnx2x_check_blocks_with_parity3(u32 sig, int par_num,
						  bool *global, bool print)
4087 4088 4089 4090 4091 4092 4093 4094
{
	int i = 0;
	u32 cur_bit = 0;
	for (i = 0; sig; i++) {
		cur_bit = ((u32)0x1 << i);
		if (sig & cur_bit) {
			switch (cur_bit) {
			case AEU_INPUTS_ATTN_BITS_MCP_LATCHED_ROM_PARITY:
4095 4096 4097
				if (print)
					_print_next_block(par_num++, "MCP ROM");
				*global = true;
4098 4099
				break;
			case AEU_INPUTS_ATTN_BITS_MCP_LATCHED_UMP_RX_PARITY:
4100 4101 4102 4103
				if (print)
					_print_next_block(par_num++,
							  "MCP UMP RX");
				*global = true;
4104 4105
				break;
			case AEU_INPUTS_ATTN_BITS_MCP_LATCHED_UMP_TX_PARITY:
4106 4107 4108 4109
				if (print)
					_print_next_block(par_num++,
							  "MCP UMP TX");
				*global = true;
4110 4111
				break;
			case AEU_INPUTS_ATTN_BITS_MCP_LATCHED_SCPAD_PARITY:
4112 4113 4114 4115
				if (print)
					_print_next_block(par_num++,
							  "MCP SCPAD");
				*global = true;
4116 4117 4118 4119 4120 4121 4122 4123 4124 4125 4126
				break;
			}

			/* Clear the bit */
			sig &= ~cur_bit;
		}
	}

	return par_num;
}

4127 4128 4129 4130 4131 4132 4133 4134 4135 4136 4137 4138 4139 4140 4141 4142 4143 4144 4145 4146 4147 4148 4149 4150 4151 4152 4153
static inline int bnx2x_check_blocks_with_parity4(u32 sig, int par_num,
						  bool print)
{
	int i = 0;
	u32 cur_bit = 0;
	for (i = 0; sig; i++) {
		cur_bit = ((u32)0x1 << i);
		if (sig & cur_bit) {
			switch (cur_bit) {
			case AEU_INPUTS_ATTN_BITS_PGLUE_PARITY_ERROR:
				if (print)
					_print_next_block(par_num++, "PGLUE_B");
				break;
			case AEU_INPUTS_ATTN_BITS_ATC_PARITY_ERROR:
				if (print)
					_print_next_block(par_num++, "ATC");
				break;
			}

			/* Clear the bit */
			sig &= ~cur_bit;
		}
	}

	return par_num;
}

4154
static inline bool bnx2x_parity_attn(struct bnx2x *bp, bool *global, bool print,
4155
				     u32 *sig)
4156
{
4157 4158 4159 4160 4161
	if ((sig[0] & HW_PRTY_ASSERT_SET_0) ||
	    (sig[1] & HW_PRTY_ASSERT_SET_1) ||
	    (sig[2] & HW_PRTY_ASSERT_SET_2) ||
	    (sig[3] & HW_PRTY_ASSERT_SET_3) ||
	    (sig[4] & HW_PRTY_ASSERT_SET_4)) {
4162
		int par_num = 0;
M
Merav Sicron 已提交
4163 4164
		DP(NETIF_MSG_HW, "Was parity error: HW block parity attention:\n"
				 "[0]:0x%08x [1]:0x%08x [2]:0x%08x [3]:0x%08x [4]:0x%08x\n",
4165 4166 4167 4168 4169
			  sig[0] & HW_PRTY_ASSERT_SET_0,
			  sig[1] & HW_PRTY_ASSERT_SET_1,
			  sig[2] & HW_PRTY_ASSERT_SET_2,
			  sig[3] & HW_PRTY_ASSERT_SET_3,
			  sig[4] & HW_PRTY_ASSERT_SET_4);
4170 4171 4172 4173
		if (print)
			netdev_err(bp->dev,
				   "Parity errors detected in blocks: ");
		par_num = bnx2x_check_blocks_with_parity0(
4174
			sig[0] & HW_PRTY_ASSERT_SET_0, par_num, print);
4175
		par_num = bnx2x_check_blocks_with_parity1(
4176
			sig[1] & HW_PRTY_ASSERT_SET_1, par_num, global, print);
4177
		par_num = bnx2x_check_blocks_with_parity2(
4178
			sig[2] & HW_PRTY_ASSERT_SET_2, par_num, print);
4179
		par_num = bnx2x_check_blocks_with_parity3(
4180 4181 4182 4183
			sig[3] & HW_PRTY_ASSERT_SET_3, par_num, global, print);
		par_num = bnx2x_check_blocks_with_parity4(
			sig[4] & HW_PRTY_ASSERT_SET_4, par_num, print);

4184 4185
		if (print)
			pr_cont("\n");
4186

4187 4188 4189 4190 4191
		return true;
	} else
		return false;
}

4192 4193 4194 4195 4196 4197 4198 4199
/**
 * bnx2x_chk_parity_attn - checks for parity attentions.
 *
 * @bp:		driver handle
 * @global:	true if there was a global attention
 * @print:	show parity attention in syslog
 */
bool bnx2x_chk_parity_attn(struct bnx2x *bp, bool *global, bool print)
4200
{
4201
	struct attn_route attn = { {0} };
4202 4203 4204 4205 4206 4207 4208 4209 4210 4211 4212 4213 4214 4215 4216
	int port = BP_PORT(bp);

	attn.sig[0] = REG_RD(bp,
		MISC_REG_AEU_AFTER_INVERT_1_FUNC_0 +
			     port*4);
	attn.sig[1] = REG_RD(bp,
		MISC_REG_AEU_AFTER_INVERT_2_FUNC_0 +
			     port*4);
	attn.sig[2] = REG_RD(bp,
		MISC_REG_AEU_AFTER_INVERT_3_FUNC_0 +
			     port*4);
	attn.sig[3] = REG_RD(bp,
		MISC_REG_AEU_AFTER_INVERT_4_FUNC_0 +
			     port*4);

4217 4218 4219 4220 4221 4222
	if (!CHIP_IS_E1x(bp))
		attn.sig[4] = REG_RD(bp,
			MISC_REG_AEU_AFTER_INVERT_5_FUNC_0 +
				     port*4);

	return bnx2x_parity_attn(bp, global, print, attn.sig);
4223 4224
}

D
Dmitry Kravkov 已提交
4225 4226 4227 4228 4229 4230 4231 4232 4233

static inline void bnx2x_attn_int_deasserted4(struct bnx2x *bp, u32 attn)
{
	u32 val;
	if (attn & AEU_INPUTS_ATTN_BITS_PGLUE_HW_INTERRUPT) {

		val = REG_RD(bp, PGLUE_B_REG_PGLUE_B_INT_STS_CLR);
		BNX2X_ERR("PGLUE hw attention 0x%x\n", val);
		if (val & PGLUE_B_PGLUE_B_INT_STS_REG_ADDRESS_ERROR)
M
Merav Sicron 已提交
4234
			BNX2X_ERR("PGLUE_B_PGLUE_B_INT_STS_REG_ADDRESS_ERROR\n");
D
Dmitry Kravkov 已提交
4235
		if (val & PGLUE_B_PGLUE_B_INT_STS_REG_INCORRECT_RCV_BEHAVIOR)
M
Merav Sicron 已提交
4236
			BNX2X_ERR("PGLUE_B_PGLUE_B_INT_STS_REG_INCORRECT_RCV_BEHAVIOR\n");
D
Dmitry Kravkov 已提交
4237
		if (val & PGLUE_B_PGLUE_B_INT_STS_REG_WAS_ERROR_ATTN)
M
Merav Sicron 已提交
4238
			BNX2X_ERR("PGLUE_B_PGLUE_B_INT_STS_REG_WAS_ERROR_ATTN\n");
D
Dmitry Kravkov 已提交
4239
		if (val & PGLUE_B_PGLUE_B_INT_STS_REG_VF_LENGTH_VIOLATION_ATTN)
M
Merav Sicron 已提交
4240
			BNX2X_ERR("PGLUE_B_PGLUE_B_INT_STS_REG_VF_LENGTH_VIOLATION_ATTN\n");
D
Dmitry Kravkov 已提交
4241 4242
		if (val &
		    PGLUE_B_PGLUE_B_INT_STS_REG_VF_GRC_SPACE_VIOLATION_ATTN)
M
Merav Sicron 已提交
4243
			BNX2X_ERR("PGLUE_B_PGLUE_B_INT_STS_REG_VF_GRC_SPACE_VIOLATION_ATTN\n");
D
Dmitry Kravkov 已提交
4244 4245
		if (val &
		    PGLUE_B_PGLUE_B_INT_STS_REG_VF_MSIX_BAR_VIOLATION_ATTN)
M
Merav Sicron 已提交
4246
			BNX2X_ERR("PGLUE_B_PGLUE_B_INT_STS_REG_VF_MSIX_BAR_VIOLATION_ATTN\n");
D
Dmitry Kravkov 已提交
4247
		if (val & PGLUE_B_PGLUE_B_INT_STS_REG_TCPL_ERROR_ATTN)
M
Merav Sicron 已提交
4248
			BNX2X_ERR("PGLUE_B_PGLUE_B_INT_STS_REG_TCPL_ERROR_ATTN\n");
D
Dmitry Kravkov 已提交
4249
		if (val & PGLUE_B_PGLUE_B_INT_STS_REG_TCPL_IN_TWO_RCBS_ATTN)
M
Merav Sicron 已提交
4250
			BNX2X_ERR("PGLUE_B_PGLUE_B_INT_STS_REG_TCPL_IN_TWO_RCBS_ATTN\n");
D
Dmitry Kravkov 已提交
4251
		if (val & PGLUE_B_PGLUE_B_INT_STS_REG_CSSNOOP_FIFO_OVERFLOW)
M
Merav Sicron 已提交
4252
			BNX2X_ERR("PGLUE_B_PGLUE_B_INT_STS_REG_CSSNOOP_FIFO_OVERFLOW\n");
D
Dmitry Kravkov 已提交
4253 4254 4255 4256 4257 4258 4259
	}
	if (attn & AEU_INPUTS_ATTN_BITS_ATC_HW_INTERRUPT) {
		val = REG_RD(bp, ATC_REG_ATC_INT_STS_CLR);
		BNX2X_ERR("ATC hw attention 0x%x\n", val);
		if (val & ATC_ATC_INT_STS_REG_ADDRESS_ERROR)
			BNX2X_ERR("ATC_ATC_INT_STS_REG_ADDRESS_ERROR\n");
		if (val & ATC_ATC_INT_STS_REG_ATC_TCPL_TO_NOT_PEND)
M
Merav Sicron 已提交
4260
			BNX2X_ERR("ATC_ATC_INT_STS_REG_ATC_TCPL_TO_NOT_PEND\n");
D
Dmitry Kravkov 已提交
4261
		if (val & ATC_ATC_INT_STS_REG_ATC_GPA_MULTIPLE_HITS)
M
Merav Sicron 已提交
4262
			BNX2X_ERR("ATC_ATC_INT_STS_REG_ATC_GPA_MULTIPLE_HITS\n");
D
Dmitry Kravkov 已提交
4263
		if (val & ATC_ATC_INT_STS_REG_ATC_RCPL_TO_EMPTY_CNT)
M
Merav Sicron 已提交
4264
			BNX2X_ERR("ATC_ATC_INT_STS_REG_ATC_RCPL_TO_EMPTY_CNT\n");
D
Dmitry Kravkov 已提交
4265 4266 4267
		if (val & ATC_ATC_INT_STS_REG_ATC_TCPL_ERROR)
			BNX2X_ERR("ATC_ATC_INT_STS_REG_ATC_TCPL_ERROR\n");
		if (val & ATC_ATC_INT_STS_REG_ATC_IREQ_LESS_THAN_STU)
M
Merav Sicron 已提交
4268
			BNX2X_ERR("ATC_ATC_INT_STS_REG_ATC_IREQ_LESS_THAN_STU\n");
D
Dmitry Kravkov 已提交
4269 4270 4271 4272 4273 4274 4275 4276 4277 4278 4279
	}

	if (attn & (AEU_INPUTS_ATTN_BITS_PGLUE_PARITY_ERROR |
		    AEU_INPUTS_ATTN_BITS_ATC_PARITY_ERROR)) {
		BNX2X_ERR("FATAL parity attention set4 0x%x\n",
		(u32)(attn & (AEU_INPUTS_ATTN_BITS_PGLUE_PARITY_ERROR |
		    AEU_INPUTS_ATTN_BITS_ATC_PARITY_ERROR)));
	}

}

4280 4281 4282
static void bnx2x_attn_int_deasserted(struct bnx2x *bp, u32 deasserted)
{
	struct attn_route attn, *group_mask;
4283
	int port = BP_PORT(bp);
4284
	int index;
E
Eliezer Tamir 已提交
4285 4286
	u32 reg_addr;
	u32 val;
E
Eilon Greenstein 已提交
4287
	u32 aeu_mask;
4288
	bool global = false;
E
Eliezer Tamir 已提交
4289 4290 4291

	/* need to take HW lock because MCP or other port might also
	   try to handle this event */
Y
Yitchak Gertner 已提交
4292
	bnx2x_acquire_alr(bp);
E
Eliezer Tamir 已提交
4293

4294 4295
	if (bnx2x_chk_parity_attn(bp, &global, true)) {
#ifndef BNX2X_STOP_ON_ERROR
4296
		bp->recovery_state = BNX2X_RECOVERY_INIT;
4297
		schedule_delayed_work(&bp->sp_rtnl_task, 0);
4298 4299 4300 4301 4302
		/* Disable HW interrupts */
		bnx2x_int_disable(bp);
		/* In case of parity errors don't handle attentions so that
		 * other function would "see" parity errors.
		 */
4303 4304 4305 4306
#else
		bnx2x_panic();
#endif
		bnx2x_release_alr(bp);
4307 4308 4309
		return;
	}

E
Eliezer Tamir 已提交
4310 4311 4312 4313
	attn.sig[0] = REG_RD(bp, MISC_REG_AEU_AFTER_INVERT_1_FUNC_0 + port*4);
	attn.sig[1] = REG_RD(bp, MISC_REG_AEU_AFTER_INVERT_2_FUNC_0 + port*4);
	attn.sig[2] = REG_RD(bp, MISC_REG_AEU_AFTER_INVERT_3_FUNC_0 + port*4);
	attn.sig[3] = REG_RD(bp, MISC_REG_AEU_AFTER_INVERT_4_FUNC_0 + port*4);
4314
	if (!CHIP_IS_E1x(bp))
D
Dmitry Kravkov 已提交
4315 4316 4317 4318 4319 4320 4321
		attn.sig[4] =
		      REG_RD(bp, MISC_REG_AEU_AFTER_INVERT_5_FUNC_0 + port*4);
	else
		attn.sig[4] = 0;

	DP(NETIF_MSG_HW, "attn: %08x %08x %08x %08x %08x\n",
	   attn.sig[0], attn.sig[1], attn.sig[2], attn.sig[3], attn.sig[4]);
E
Eliezer Tamir 已提交
4322 4323 4324

	for (index = 0; index < MAX_DYNAMIC_ATTN_GRPS; index++) {
		if (deasserted & (1 << index)) {
4325
			group_mask = &bp->attn_group[index];
E
Eliezer Tamir 已提交
4326

M
Merav Sicron 已提交
4327
			DP(NETIF_MSG_HW, "group[%d]: %08x %08x %08x %08x %08x\n",
D
Dmitry Kravkov 已提交
4328 4329 4330 4331
			   index,
			   group_mask->sig[0], group_mask->sig[1],
			   group_mask->sig[2], group_mask->sig[3],
			   group_mask->sig[4]);
E
Eliezer Tamir 已提交
4332

D
Dmitry Kravkov 已提交
4333 4334
			bnx2x_attn_int_deasserted4(bp,
					attn.sig[4] & group_mask->sig[4]);
4335
			bnx2x_attn_int_deasserted3(bp,
4336
					attn.sig[3] & group_mask->sig[3]);
4337
			bnx2x_attn_int_deasserted1(bp,
4338
					attn.sig[1] & group_mask->sig[1]);
4339
			bnx2x_attn_int_deasserted2(bp,
4340
					attn.sig[2] & group_mask->sig[2]);
4341
			bnx2x_attn_int_deasserted0(bp,
4342
					attn.sig[0] & group_mask->sig[0]);
E
Eliezer Tamir 已提交
4343 4344 4345
		}
	}

Y
Yitchak Gertner 已提交
4346
	bnx2x_release_alr(bp);
E
Eliezer Tamir 已提交
4347

D
Dmitry Kravkov 已提交
4348 4349 4350 4351 4352
	if (bp->common.int_block == INT_BLOCK_HC)
		reg_addr = (HC_REG_COMMAND_REG + port*32 +
			    COMMAND_REG_ATTN_BITS_CLR);
	else
		reg_addr = (BAR_IGU_INTMEM + IGU_CMD_ATTN_BIT_CLR_UPPER*8);
E
Eliezer Tamir 已提交
4353 4354

	val = ~deasserted;
D
Dmitry Kravkov 已提交
4355 4356
	DP(NETIF_MSG_HW, "about to mask 0x%08x at %s addr 0x%x\n", val,
	   (bp->common.int_block == INT_BLOCK_HC) ? "HC" : "IGU", reg_addr);
4357
	REG_WR(bp, reg_addr, val);
E
Eliezer Tamir 已提交
4358 4359

	if (~bp->attn_state & deasserted)
E
Eilon Greenstein 已提交
4360
		BNX2X_ERR("IGU ERROR\n");
E
Eliezer Tamir 已提交
4361 4362 4363 4364

	reg_addr = port ? MISC_REG_AEU_MASK_ATTN_FUNC_1 :
			  MISC_REG_AEU_MASK_ATTN_FUNC_0;

E
Eilon Greenstein 已提交
4365 4366 4367 4368 4369
	bnx2x_acquire_hw_lock(bp, HW_LOCK_RESOURCE_PORT0_ATT_MASK + port);
	aeu_mask = REG_RD(bp, reg_addr);

	DP(NETIF_MSG_HW, "aeu_mask %x  newly deasserted %x\n",
	   aeu_mask, deasserted);
4370
	aeu_mask |= (deasserted & 0x3ff);
E
Eilon Greenstein 已提交
4371
	DP(NETIF_MSG_HW, "new mask %x\n", aeu_mask);
E
Eliezer Tamir 已提交
4372

E
Eilon Greenstein 已提交
4373 4374
	REG_WR(bp, reg_addr, aeu_mask);
	bnx2x_release_hw_lock(bp, HW_LOCK_RESOURCE_PORT0_ATT_MASK + port);
E
Eliezer Tamir 已提交
4375 4376 4377 4378 4379 4380 4381 4382 4383

	DP(NETIF_MSG_HW, "attn_state %x\n", bp->attn_state);
	bp->attn_state &= ~deasserted;
	DP(NETIF_MSG_HW, "new state %x\n", bp->attn_state);
}

static void bnx2x_attn_int(struct bnx2x *bp)
{
	/* read local copy of bits */
E
Eilon Greenstein 已提交
4384 4385 4386 4387
	u32 attn_bits = le32_to_cpu(bp->def_status_blk->atten_status_block.
								attn_bits);
	u32 attn_ack = le32_to_cpu(bp->def_status_blk->atten_status_block.
								attn_bits_ack);
E
Eliezer Tamir 已提交
4388 4389 4390 4391 4392 4393 4394 4395 4396 4397 4398
	u32 attn_state = bp->attn_state;

	/* look for changed bits */
	u32 asserted   =  attn_bits & ~attn_ack & ~attn_state;
	u32 deasserted = ~attn_bits &  attn_ack &  attn_state;

	DP(NETIF_MSG_HW,
	   "attn_bits %x  attn_ack %x  asserted %x  deasserted %x\n",
	   attn_bits, attn_ack, asserted, deasserted);

	if (~(attn_bits ^ attn_ack) & (attn_bits ^ attn_state))
4399
		BNX2X_ERR("BAD attention state\n");
E
Eliezer Tamir 已提交
4400 4401 4402 4403 4404 4405 4406 4407 4408

	/* handle bits that were raised */
	if (asserted)
		bnx2x_attn_int_asserted(bp, asserted);

	if (deasserted)
		bnx2x_attn_int_deasserted(bp, deasserted);
}

4409 4410 4411 4412 4413 4414 4415 4416 4417
void bnx2x_igu_ack_sb(struct bnx2x *bp, u8 igu_sb_id, u8 segment,
		      u16 index, u8 op, u8 update)
{
	u32 igu_addr = BAR_IGU_INTMEM + (IGU_CMD_INT_ACK_BASE + igu_sb_id)*8;

	bnx2x_igu_ack_sb_gen(bp, igu_sb_id, segment, index, op, update,
			     igu_addr);
}

4418 4419 4420 4421 4422 4423 4424 4425 4426 4427 4428
static inline void bnx2x_update_eq_prod(struct bnx2x *bp, u16 prod)
{
	/* No memory barriers */
	storm_memset_eq_prod(bp, prod, BP_FUNC(bp));
	mmiowb(); /* keep prod updates ordered */
}

#ifdef BCM_CNIC
static int  bnx2x_cnic_handle_cfc_del(struct bnx2x *bp, u32 cid,
				      union event_ring_elem *elem)
{
4429 4430
	u8 err = elem->message.error;

4431
	if (!bp->cnic_eth_dev.starting_cid  ||
4432 4433
	    (cid < bp->cnic_eth_dev.starting_cid &&
	    cid != bp->cnic_eth_dev.iscsi_l2_cid))
4434 4435 4436 4437
		return 1;

	DP(BNX2X_MSG_SP, "got delete ramrod for CNIC CID %d\n", cid);

4438 4439
	if (unlikely(err)) {

4440 4441 4442 4443
		BNX2X_ERR("got delete ramrod for CNIC CID %d with error!\n",
			  cid);
		bnx2x_panic_dump(bp);
	}
4444
	bnx2x_cnic_cfc_comp(bp, cid, err);
4445 4446 4447 4448
	return 0;
}
#endif

4449 4450 4451 4452 4453 4454 4455 4456 4457 4458 4459 4460 4461 4462 4463 4464 4465 4466 4467 4468 4469 4470 4471 4472 4473 4474 4475 4476 4477 4478 4479 4480 4481 4482 4483 4484 4485 4486
static inline void bnx2x_handle_mcast_eqe(struct bnx2x *bp)
{
	struct bnx2x_mcast_ramrod_params rparam;
	int rc;

	memset(&rparam, 0, sizeof(rparam));

	rparam.mcast_obj = &bp->mcast_obj;

	netif_addr_lock_bh(bp->dev);

	/* Clear pending state for the last command */
	bp->mcast_obj.raw.clear_pending(&bp->mcast_obj.raw);

	/* If there are pending mcast commands - send them */
	if (bp->mcast_obj.check_pending(&bp->mcast_obj)) {
		rc = bnx2x_config_mcast(bp, &rparam, BNX2X_MCAST_CMD_CONT);
		if (rc < 0)
			BNX2X_ERR("Failed to send pending mcast commands: %d\n",
				  rc);
	}

	netif_addr_unlock_bh(bp->dev);
}

static inline void bnx2x_handle_classification_eqe(struct bnx2x *bp,
						   union event_ring_elem *elem)
{
	unsigned long ramrod_flags = 0;
	int rc = 0;
	u32 cid = elem->message.data.eth_event.echo & BNX2X_SWCID_MASK;
	struct bnx2x_vlan_mac_obj *vlan_mac_obj;

	/* Always push next commands out, don't wait here */
	__set_bit(RAMROD_CONT, &ramrod_flags);

	switch (elem->message.data.eth_event.echo >> BNX2X_SWCID_SHIFT) {
	case BNX2X_FILTER_MAC_PENDING:
M
Merav Sicron 已提交
4487
		DP(BNX2X_MSG_SP, "Got SETUP_MAC completions\n");
4488 4489 4490 4491 4492 4493 4494 4495 4496
#ifdef BCM_CNIC
		if (cid == BNX2X_ISCSI_ETH_CID)
			vlan_mac_obj = &bp->iscsi_l2_mac_obj;
		else
#endif
			vlan_mac_obj = &bp->fp[cid].mac_obj;

		break;
	case BNX2X_FILTER_MCAST_PENDING:
M
Merav Sicron 已提交
4497
		DP(BNX2X_MSG_SP, "Got SETUP_MCAST completions\n");
4498 4499 4500 4501 4502 4503 4504 4505 4506 4507 4508 4509 4510 4511 4512 4513 4514 4515 4516 4517 4518 4519 4520 4521 4522 4523 4524 4525 4526 4527 4528 4529 4530 4531 4532 4533 4534 4535 4536 4537 4538 4539 4540 4541 4542 4543 4544 4545
		/* This is only relevant for 57710 where multicast MACs are
		 * configured as unicast MACs using the same ramrod.
		 */
		bnx2x_handle_mcast_eqe(bp);
		return;
	default:
		BNX2X_ERR("Unsupported classification command: %d\n",
			  elem->message.data.eth_event.echo);
		return;
	}

	rc = vlan_mac_obj->complete(bp, vlan_mac_obj, elem, &ramrod_flags);

	if (rc < 0)
		BNX2X_ERR("Failed to schedule new commands: %d\n", rc);
	else if (rc > 0)
		DP(BNX2X_MSG_SP, "Scheduled next pending commands...\n");

}

#ifdef BCM_CNIC
static void bnx2x_set_iscsi_eth_rx_mode(struct bnx2x *bp, bool start);
#endif

static inline void bnx2x_handle_rx_mode_eqe(struct bnx2x *bp)
{
	netif_addr_lock_bh(bp->dev);

	clear_bit(BNX2X_FILTER_RX_MODE_PENDING, &bp->sp_state);

	/* Send rx_mode command again if was requested */
	if (test_and_clear_bit(BNX2X_FILTER_RX_MODE_SCHED, &bp->sp_state))
		bnx2x_set_storm_rx_mode(bp);
#ifdef BCM_CNIC
	else if (test_and_clear_bit(BNX2X_FILTER_ISCSI_ETH_START_SCHED,
				    &bp->sp_state))
		bnx2x_set_iscsi_eth_rx_mode(bp, true);
	else if (test_and_clear_bit(BNX2X_FILTER_ISCSI_ETH_STOP_SCHED,
				    &bp->sp_state))
		bnx2x_set_iscsi_eth_rx_mode(bp, false);
#endif

	netif_addr_unlock_bh(bp->dev);
}

static inline struct bnx2x_queue_sp_obj *bnx2x_cid_to_q_obj(
	struct bnx2x *bp, u32 cid)
{
4546
	DP(BNX2X_MSG_SP, "retrieving fp from cid %d\n", cid);
4547 4548 4549 4550 4551
#ifdef BCM_CNIC
	if (cid == BNX2X_FCOE_ETH_CID)
		return &bnx2x_fcoe(bp, q_obj);
	else
#endif
4552
		return &bnx2x_fp(bp, CID_TO_FP(cid), q_obj);
4553 4554
}

4555 4556 4557 4558 4559 4560 4561
static void bnx2x_eq_int(struct bnx2x *bp)
{
	u16 hw_cons, sw_cons, sw_prod;
	union event_ring_elem *elem;
	u32 cid;
	u8 opcode;
	int spqe_cnt = 0;
4562 4563 4564
	struct bnx2x_queue_sp_obj *q_obj;
	struct bnx2x_func_sp_obj *f_obj = &bp->func_obj;
	struct bnx2x_raw_obj *rss_raw = &bp->rss_conf_obj.raw;
4565 4566 4567 4568 4569 4570 4571 4572 4573 4574 4575

	hw_cons = le16_to_cpu(*bp->eq_cons_sb);

	/* The hw_cos range is 1-255, 257 - the sw_cons range is 0-254, 256.
	 * when we get the the next-page we nned to adjust so the loop
	 * condition below will be met. The next element is the size of a
	 * regular element and hence incrementing by 1
	 */
	if ((hw_cons & EQ_DESC_MAX_PAGE) == EQ_DESC_MAX_PAGE)
		hw_cons++;

L
Lucas De Marchi 已提交
4576
	/* This function may never run in parallel with itself for a
4577 4578 4579 4580 4581 4582
	 * specific bp, thus there is no need in "paired" read memory
	 * barrier here.
	 */
	sw_cons = bp->eq_cons;
	sw_prod = bp->eq_prod;

4583
	DP(BNX2X_MSG_SP, "EQ:  hw_cons %u  sw_cons %u bp->eq_spq_left %x\n",
4584
			hw_cons, sw_cons, atomic_read(&bp->eq_spq_left));
4585 4586 4587 4588 4589 4590 4591 4592 4593 4594 4595 4596 4597 4598

	for (; sw_cons != hw_cons;
	      sw_prod = NEXT_EQ_IDX(sw_prod), sw_cons = NEXT_EQ_IDX(sw_cons)) {


		elem = &bp->eq_ring[EQ_DESC(sw_cons)];

		cid = SW_CID(elem->message.data.cfc_del_event.cid);
		opcode = elem->message.opcode;


		/* handle eq element */
		switch (opcode) {
		case EVENT_RING_OPCODE_STAT_QUERY:
M
Merav Sicron 已提交
4599 4600
			DP(BNX2X_MSG_SP | BNX2X_MSG_STATS,
			   "got statistics comp event %d\n",
4601
			   bp->stats_comp++);
4602
			/* nothing to do with stats comp */
4603
			goto next_spqe;
4604 4605 4606 4607 4608 4609 4610

		case EVENT_RING_OPCODE_CFC_DEL:
			/* handle according to cid range */
			/*
			 * we may want to verify here that the bp state is
			 * HALTING
			 */
4611
			DP(BNX2X_MSG_SP,
4612 4613 4614 4615 4616
			   "got delete ramrod for MULTI[%d]\n", cid);
#ifdef BCM_CNIC
			if (!bnx2x_cnic_handle_cfc_del(bp, cid, elem))
				goto next_spqe;
#endif
4617 4618 4619 4620 4621 4622
			q_obj = bnx2x_cid_to_q_obj(bp, cid);

			if (q_obj->complete_cmd(bp, q_obj, BNX2X_Q_CMD_CFC_DEL))
				break;


4623 4624

			goto next_spqe;
V
Vladislav Zolotarov 已提交
4625 4626

		case EVENT_RING_OPCODE_STOP_TRAFFIC:
M
Merav Sicron 已提交
4627
			DP(BNX2X_MSG_SP | BNX2X_MSG_DCB, "got STOP TRAFFIC\n");
D
Dmitry Kravkov 已提交
4628 4629 4630
			if (f_obj->complete_cmd(bp, f_obj,
						BNX2X_F_CMD_TX_STOP))
				break;
V
Vladislav Zolotarov 已提交
4631 4632
			bnx2x_dcbx_set_params(bp, BNX2X_DCBX_STATE_TX_PAUSED);
			goto next_spqe;
4633

V
Vladislav Zolotarov 已提交
4634
		case EVENT_RING_OPCODE_START_TRAFFIC:
M
Merav Sicron 已提交
4635
			DP(BNX2X_MSG_SP | BNX2X_MSG_DCB, "got START TRAFFIC\n");
D
Dmitry Kravkov 已提交
4636 4637 4638
			if (f_obj->complete_cmd(bp, f_obj,
						BNX2X_F_CMD_TX_START))
				break;
V
Vladislav Zolotarov 已提交
4639 4640
			bnx2x_dcbx_set_params(bp, BNX2X_DCBX_STATE_TX_RELEASED);
			goto next_spqe;
4641
		case EVENT_RING_OPCODE_FUNCTION_START:
M
Merav Sicron 已提交
4642 4643
			DP(BNX2X_MSG_SP | NETIF_MSG_IFUP,
			   "got FUNC_START ramrod\n");
4644 4645 4646 4647 4648 4649
			if (f_obj->complete_cmd(bp, f_obj, BNX2X_F_CMD_START))
				break;

			goto next_spqe;

		case EVENT_RING_OPCODE_FUNCTION_STOP:
M
Merav Sicron 已提交
4650 4651
			DP(BNX2X_MSG_SP | NETIF_MSG_IFUP,
			   "got FUNC_STOP ramrod\n");
4652 4653 4654 4655
			if (f_obj->complete_cmd(bp, f_obj, BNX2X_F_CMD_STOP))
				break;

			goto next_spqe;
4656 4657 4658
		}

		switch (opcode | bp->state) {
4659 4660 4661
		case (EVENT_RING_OPCODE_RSS_UPDATE_RULES |
		      BNX2X_STATE_OPEN):
		case (EVENT_RING_OPCODE_RSS_UPDATE_RULES |
4662
		      BNX2X_STATE_OPENING_WAIT4_PORT):
4663 4664
			cid = elem->message.data.eth_event.echo &
				BNX2X_SWCID_MASK;
4665
			DP(BNX2X_MSG_SP, "got RSS_UPDATE ramrod. CID %d\n",
4666 4667
			   cid);
			rss_raw->clear_pending(rss_raw);
4668 4669
			break;

4670 4671 4672
		case (EVENT_RING_OPCODE_SET_MAC | BNX2X_STATE_OPEN):
		case (EVENT_RING_OPCODE_SET_MAC | BNX2X_STATE_DIAG):
		case (EVENT_RING_OPCODE_SET_MAC |
4673
		      BNX2X_STATE_CLOSING_WAIT4_HALT):
4674 4675 4676 4677 4678 4679
		case (EVENT_RING_OPCODE_CLASSIFICATION_RULES |
		      BNX2X_STATE_OPEN):
		case (EVENT_RING_OPCODE_CLASSIFICATION_RULES |
		      BNX2X_STATE_DIAG):
		case (EVENT_RING_OPCODE_CLASSIFICATION_RULES |
		      BNX2X_STATE_CLOSING_WAIT4_HALT):
4680
			DP(BNX2X_MSG_SP, "got (un)set mac ramrod\n");
4681
			bnx2x_handle_classification_eqe(bp, elem);
4682 4683
			break;

4684 4685 4686 4687 4688 4689
		case (EVENT_RING_OPCODE_MULTICAST_RULES |
		      BNX2X_STATE_OPEN):
		case (EVENT_RING_OPCODE_MULTICAST_RULES |
		      BNX2X_STATE_DIAG):
		case (EVENT_RING_OPCODE_MULTICAST_RULES |
		      BNX2X_STATE_CLOSING_WAIT4_HALT):
4690
			DP(BNX2X_MSG_SP, "got mcast ramrod\n");
4691
			bnx2x_handle_mcast_eqe(bp);
4692 4693
			break;

4694 4695 4696 4697 4698
		case (EVENT_RING_OPCODE_FILTERS_RULES |
		      BNX2X_STATE_OPEN):
		case (EVENT_RING_OPCODE_FILTERS_RULES |
		      BNX2X_STATE_DIAG):
		case (EVENT_RING_OPCODE_FILTERS_RULES |
4699
		      BNX2X_STATE_CLOSING_WAIT4_HALT):
4700
			DP(BNX2X_MSG_SP, "got rx_mode ramrod\n");
4701
			bnx2x_handle_rx_mode_eqe(bp);
4702 4703 4704
			break;
		default:
			/* unknown event log error and continue */
4705 4706
			BNX2X_ERR("Unknown EQ event %d, bp->state 0x%x\n",
				  elem->message.opcode, bp->state);
4707 4708 4709 4710 4711
		}
next_spqe:
		spqe_cnt++;
	} /* for */

4712
	smp_mb__before_atomic_inc();
4713
	atomic_add(spqe_cnt, &bp->eq_spq_left);
4714 4715 4716 4717 4718 4719 4720 4721 4722 4723

	bp->eq_cons = sw_cons;
	bp->eq_prod = sw_prod;
	/* Make sure that above mem writes were issued towards the memory */
	smp_wmb();

	/* update producer */
	bnx2x_update_eq_prod(bp, bp->eq_prod);
}

E
Eliezer Tamir 已提交
4724 4725
static void bnx2x_sp_task(struct work_struct *work)
{
4726
	struct bnx2x *bp = container_of(work, struct bnx2x, sp_task.work);
E
Eliezer Tamir 已提交
4727 4728 4729
	u16 status;

	status = bnx2x_update_dsb_idx(bp);
4730 4731
/*	if (status == 0)				     */
/*		BNX2X_ERR("spurious slowpath interrupt!\n"); */
E
Eliezer Tamir 已提交
4732

M
Merav Sicron 已提交
4733
	DP(BNX2X_MSG_SP, "got a slowpath interrupt (status 0x%x)\n", status);
E
Eliezer Tamir 已提交
4734

4735
	/* HW attentions */
4736
	if (status & BNX2X_DEF_SB_ATT_IDX) {
E
Eliezer Tamir 已提交
4737
		bnx2x_attn_int(bp);
4738
		status &= ~BNX2X_DEF_SB_ATT_IDX;
V
Vladislav Zolotarov 已提交
4739 4740
	}

4741 4742
	/* SP events: STAT_QUERY and others */
	if (status & BNX2X_DEF_SB_IDX) {
V
Vladislav Zolotarov 已提交
4743 4744
#ifdef BCM_CNIC
		struct bnx2x_fastpath *fp = bnx2x_fcoe_fp(bp);
4745

V
Vladislav Zolotarov 已提交
4746
		if ((!NO_FCOE(bp)) &&
4747 4748 4749 4750 4751 4752
			(bnx2x_has_rx_work(fp) || bnx2x_has_tx_work(fp))) {
			/*
			 * Prevent local bottom-halves from running as
			 * we are going to change the local NAPI list.
			 */
			local_bh_disable();
V
Vladislav Zolotarov 已提交
4753
			napi_schedule(&bnx2x_fcoe(bp, napi));
4754 4755
			local_bh_enable();
		}
V
Vladislav Zolotarov 已提交
4756
#endif
4757 4758 4759 4760 4761 4762 4763
		/* Handle EQ completions */
		bnx2x_eq_int(bp);

		bnx2x_ack_sb(bp, bp->igu_dsb_id, USTORM_ID,
			le16_to_cpu(bp->def_idx), IGU_INT_NOP, 1);

		status &= ~BNX2X_DEF_SB_IDX;
V
Vladislav Zolotarov 已提交
4764 4765 4766
	}

	if (unlikely(status))
M
Merav Sicron 已提交
4767
		DP(BNX2X_MSG_SP, "got an unknown interrupt! (status 0x%x)\n",
V
Vladislav Zolotarov 已提交
4768
		   status);
E
Eliezer Tamir 已提交
4769

4770 4771
	bnx2x_ack_sb(bp, bp->igu_dsb_id, ATTENTION_ID,
	     le16_to_cpu(bp->def_att_idx), IGU_INT_ENABLE, 1);
E
Eliezer Tamir 已提交
4772 4773
}

D
Dmitry Kravkov 已提交
4774
irqreturn_t bnx2x_msix_sp_int(int irq, void *dev_instance)
E
Eliezer Tamir 已提交
4775 4776 4777 4778
{
	struct net_device *dev = dev_instance;
	struct bnx2x *bp = netdev_priv(dev);

4779 4780
	bnx2x_ack_sb(bp, bp->igu_dsb_id, USTORM_ID, 0,
		     IGU_INT_DISABLE, 0);
E
Eliezer Tamir 已提交
4781 4782 4783 4784 4785 4786

#ifdef BNX2X_STOP_ON_ERROR
	if (unlikely(bp->panic))
		return IRQ_HANDLED;
#endif

4787 4788 4789 4790 4791 4792 4793 4794 4795 4796 4797
#ifdef BCM_CNIC
	{
		struct cnic_ops *c_ops;

		rcu_read_lock();
		c_ops = rcu_dereference(bp->cnic_ops);
		if (c_ops)
			c_ops->cnic_handler(bp->cnic_data, NULL);
		rcu_read_unlock();
	}
#endif
4798
	queue_delayed_work(bnx2x_wq, &bp->sp_task, 0);
E
Eliezer Tamir 已提交
4799 4800 4801 4802 4803 4804

	return IRQ_HANDLED;
}

/* end of slow path */

4805 4806 4807 4808 4809 4810 4811 4812

void bnx2x_drv_pulse(struct bnx2x *bp)
{
	SHMEM_WR(bp, func_mb[BP_FW_MB_IDX(bp)].drv_pulse_mb,
		 bp->fw_drv_pulse_wr_seq);
}


E
Eliezer Tamir 已提交
4813 4814 4815 4816 4817 4818 4819
static void bnx2x_timer(unsigned long data)
{
	struct bnx2x *bp = (struct bnx2x *) data;

	if (!netif_running(bp->dev))
		return;

4820
	if (!BP_NOMCP(bp)) {
D
Dmitry Kravkov 已提交
4821
		int mb_idx = BP_FW_MB_IDX(bp);
E
Eliezer Tamir 已提交
4822 4823 4824 4825 4826 4827 4828
		u32 drv_pulse;
		u32 mcp_pulse;

		++bp->fw_drv_pulse_wr_seq;
		bp->fw_drv_pulse_wr_seq &= DRV_PULSE_SEQ_MASK;
		/* TBD - add SYSTEM_TIME */
		drv_pulse = bp->fw_drv_pulse_wr_seq;
4829
		bnx2x_drv_pulse(bp);
E
Eliezer Tamir 已提交
4830

D
Dmitry Kravkov 已提交
4831
		mcp_pulse = (SHMEM_RD(bp, func_mb[mb_idx].mcp_pulse_mb) &
E
Eliezer Tamir 已提交
4832 4833 4834 4835 4836 4837 4838 4839 4840 4841 4842 4843
			     MCP_PULSE_SEQ_MASK);
		/* The delta between driver pulse and mcp response
		 * should be 1 (before mcp response) or 0 (after mcp response)
		 */
		if ((drv_pulse != mcp_pulse) &&
		    (drv_pulse != ((mcp_pulse + 1) & MCP_PULSE_SEQ_MASK))) {
			/* someone lost a heartbeat... */
			BNX2X_ERR("drv_pulse (0x%x) != mcp_pulse (0x%x)\n",
				  drv_pulse, mcp_pulse);
		}
	}

4844
	if (bp->state == BNX2X_STATE_OPEN)
Y
Yitchak Gertner 已提交
4845
		bnx2x_stats_handle(bp, STATS_EVENT_UPDATE);
E
Eliezer Tamir 已提交
4846 4847 4848 4849 4850 4851 4852 4853 4854 4855 4856 4857

	mod_timer(&bp->timer, jiffies + bp->current_interval);
}

/* end of Statistics */

/* nic init */

/*
 * nic init service functions
 */

4858
static inline void bnx2x_fill(struct bnx2x *bp, u32 addr, int fill, u32 len)
E
Eliezer Tamir 已提交
4859
{
4860 4861 4862 4863 4864 4865 4866
	u32 i;
	if (!(len%4) && !(addr%4))
		for (i = 0; i < len; i += 4)
			REG_WR(bp, addr + i, fill);
	else
		for (i = 0; i < len; i++)
			REG_WR8(bp, addr + i, fill);
4867 4868 4869

}

4870 4871 4872 4873 4874
/* helper: writes FP SP data to FW - data_size in dwords */
static inline void bnx2x_wr_fp_sb_data(struct bnx2x *bp,
				       int fw_sb_id,
				       u32 *sb_data_p,
				       u32 data_size)
4875
{
E
Eliezer Tamir 已提交
4876
	int index;
4877 4878 4879 4880 4881 4882
	for (index = 0; index < data_size; index++)
		REG_WR(bp, BAR_CSTRORM_INTMEM +
			CSTORM_STATUS_BLOCK_DATA_OFFSET(fw_sb_id) +
			sizeof(u32)*index,
			*(sb_data_p + index));
}
E
Eliezer Tamir 已提交
4883

4884 4885 4886 4887
static inline void bnx2x_zero_fp_sb(struct bnx2x *bp, int fw_sb_id)
{
	u32 *sb_data_p;
	u32 data_size = 0;
D
Dmitry Kravkov 已提交
4888
	struct hc_status_block_data_e2 sb_data_e2;
4889
	struct hc_status_block_data_e1x sb_data_e1x;
E
Eliezer Tamir 已提交
4890

4891
	/* disable the function first */
4892
	if (!CHIP_IS_E1x(bp)) {
D
Dmitry Kravkov 已提交
4893
		memset(&sb_data_e2, 0, sizeof(struct hc_status_block_data_e2));
4894
		sb_data_e2.common.state = SB_DISABLED;
D
Dmitry Kravkov 已提交
4895 4896 4897 4898 4899 4900
		sb_data_e2.common.p_func.vf_valid = false;
		sb_data_p = (u32 *)&sb_data_e2;
		data_size = sizeof(struct hc_status_block_data_e2)/sizeof(u32);
	} else {
		memset(&sb_data_e1x, 0,
		       sizeof(struct hc_status_block_data_e1x));
4901
		sb_data_e1x.common.state = SB_DISABLED;
D
Dmitry Kravkov 已提交
4902 4903 4904 4905
		sb_data_e1x.common.p_func.vf_valid = false;
		sb_data_p = (u32 *)&sb_data_e1x;
		data_size = sizeof(struct hc_status_block_data_e1x)/sizeof(u32);
	}
4906
	bnx2x_wr_fp_sb_data(bp, fw_sb_id, sb_data_p, data_size);
E
Eliezer Tamir 已提交
4907

4908 4909 4910 4911 4912 4913 4914
	bnx2x_fill(bp, BAR_CSTRORM_INTMEM +
			CSTORM_STATUS_BLOCK_OFFSET(fw_sb_id), 0,
			CSTORM_STATUS_BLOCK_SIZE);
	bnx2x_fill(bp, BAR_CSTRORM_INTMEM +
			CSTORM_SYNC_BLOCK_OFFSET(fw_sb_id), 0,
			CSTORM_SYNC_BLOCK_SIZE);
}
4915

4916 4917 4918 4919 4920 4921 4922 4923 4924 4925 4926
/* helper:  writes SP SB data to FW */
static inline void bnx2x_wr_sp_sb_data(struct bnx2x *bp,
		struct hc_sp_status_block_data *sp_sb_data)
{
	int func = BP_FUNC(bp);
	int i;
	for (i = 0; i < sizeof(struct hc_sp_status_block_data)/sizeof(u32); i++)
		REG_WR(bp, BAR_CSTRORM_INTMEM +
			CSTORM_SP_STATUS_BLOCK_DATA_OFFSET(func) +
			i*sizeof(u32),
			*((u32 *)sp_sb_data + i));
4927 4928
}

4929
static inline void bnx2x_zero_sp_sb(struct bnx2x *bp)
4930 4931
{
	int func = BP_FUNC(bp);
4932 4933
	struct hc_sp_status_block_data sp_sb_data;
	memset(&sp_sb_data, 0, sizeof(struct hc_sp_status_block_data));
E
Eliezer Tamir 已提交
4934

4935
	sp_sb_data.state = SB_DISABLED;
4936 4937 4938 4939 4940 4941 4942 4943 4944 4945 4946 4947 4948 4949 4950 4951 4952 4953 4954 4955 4956 4957
	sp_sb_data.p_func.vf_valid = false;

	bnx2x_wr_sp_sb_data(bp, &sp_sb_data);

	bnx2x_fill(bp, BAR_CSTRORM_INTMEM +
			CSTORM_SP_STATUS_BLOCK_OFFSET(func), 0,
			CSTORM_SP_STATUS_BLOCK_SIZE);
	bnx2x_fill(bp, BAR_CSTRORM_INTMEM +
			CSTORM_SP_SYNC_BLOCK_OFFSET(func), 0,
			CSTORM_SP_SYNC_BLOCK_SIZE);

}


static inline
void bnx2x_setup_ndsb_state_machine(struct hc_status_block_sm *hc_sm,
					   int igu_sb_id, int igu_seg_id)
{
	hc_sm->igu_sb_id = igu_sb_id;
	hc_sm->igu_seg_id = igu_seg_id;
	hc_sm->timer_value = 0xFF;
	hc_sm->time_to_expire = 0xFFFFFFFF;
E
Eliezer Tamir 已提交
4958 4959
}

4960 4961 4962 4963 4964 4965 4966 4967 4968 4969 4970 4971 4972 4973 4974 4975 4976 4977 4978 4979 4980 4981 4982 4983 4984 4985 4986 4987 4988 4989 4990

/* allocates state machine ids. */
static inline
void bnx2x_map_sb_state_machines(struct hc_index_data *index_data)
{
	/* zero out state machine indices */
	/* rx indices */
	index_data[HC_INDEX_ETH_RX_CQ_CONS].flags &= ~HC_INDEX_DATA_SM_ID;

	/* tx indices */
	index_data[HC_INDEX_OOO_TX_CQ_CONS].flags &= ~HC_INDEX_DATA_SM_ID;
	index_data[HC_INDEX_ETH_TX_CQ_CONS_COS0].flags &= ~HC_INDEX_DATA_SM_ID;
	index_data[HC_INDEX_ETH_TX_CQ_CONS_COS1].flags &= ~HC_INDEX_DATA_SM_ID;
	index_data[HC_INDEX_ETH_TX_CQ_CONS_COS2].flags &= ~HC_INDEX_DATA_SM_ID;

	/* map indices */
	/* rx indices */
	index_data[HC_INDEX_ETH_RX_CQ_CONS].flags |=
		SM_RX_ID << HC_INDEX_DATA_SM_ID_SHIFT;

	/* tx indices */
	index_data[HC_INDEX_OOO_TX_CQ_CONS].flags |=
		SM_TX_ID << HC_INDEX_DATA_SM_ID_SHIFT;
	index_data[HC_INDEX_ETH_TX_CQ_CONS_COS0].flags |=
		SM_TX_ID << HC_INDEX_DATA_SM_ID_SHIFT;
	index_data[HC_INDEX_ETH_TX_CQ_CONS_COS1].flags |=
		SM_TX_ID << HC_INDEX_DATA_SM_ID_SHIFT;
	index_data[HC_INDEX_ETH_TX_CQ_CONS_COS2].flags |=
		SM_TX_ID << HC_INDEX_DATA_SM_ID_SHIFT;
}

4991
static void bnx2x_init_sb(struct bnx2x *bp, dma_addr_t mapping, int vfid,
4992
			  u8 vf_valid, int fw_sb_id, int igu_sb_id)
E
Eliezer Tamir 已提交
4993
{
4994 4995
	int igu_seg_id;

D
Dmitry Kravkov 已提交
4996
	struct hc_status_block_data_e2 sb_data_e2;
4997 4998 4999 5000 5001
	struct hc_status_block_data_e1x sb_data_e1x;
	struct hc_status_block_sm  *hc_sm_p;
	int data_size;
	u32 *sb_data_p;

D
Dmitry Kravkov 已提交
5002 5003 5004 5005
	if (CHIP_INT_MODE_IS_BC(bp))
		igu_seg_id = HC_SEG_ACCESS_NORM;
	else
		igu_seg_id = IGU_SEG_ACCESS_NORM;
5006 5007 5008

	bnx2x_zero_fp_sb(bp, fw_sb_id);

5009
	if (!CHIP_IS_E1x(bp)) {
D
Dmitry Kravkov 已提交
5010
		memset(&sb_data_e2, 0, sizeof(struct hc_status_block_data_e2));
5011
		sb_data_e2.common.state = SB_ENABLED;
D
Dmitry Kravkov 已提交
5012 5013 5014 5015 5016 5017 5018 5019 5020 5021
		sb_data_e2.common.p_func.pf_id = BP_FUNC(bp);
		sb_data_e2.common.p_func.vf_id = vfid;
		sb_data_e2.common.p_func.vf_valid = vf_valid;
		sb_data_e2.common.p_func.vnic_id = BP_VN(bp);
		sb_data_e2.common.same_igu_sb_1b = true;
		sb_data_e2.common.host_sb_addr.hi = U64_HI(mapping);
		sb_data_e2.common.host_sb_addr.lo = U64_LO(mapping);
		hc_sm_p = sb_data_e2.common.state_machine;
		sb_data_p = (u32 *)&sb_data_e2;
		data_size = sizeof(struct hc_status_block_data_e2)/sizeof(u32);
5022
		bnx2x_map_sb_state_machines(sb_data_e2.index_data);
D
Dmitry Kravkov 已提交
5023 5024 5025
	} else {
		memset(&sb_data_e1x, 0,
		       sizeof(struct hc_status_block_data_e1x));
5026
		sb_data_e1x.common.state = SB_ENABLED;
D
Dmitry Kravkov 已提交
5027 5028 5029 5030 5031 5032 5033 5034 5035 5036
		sb_data_e1x.common.p_func.pf_id = BP_FUNC(bp);
		sb_data_e1x.common.p_func.vf_id = 0xff;
		sb_data_e1x.common.p_func.vf_valid = false;
		sb_data_e1x.common.p_func.vnic_id = BP_VN(bp);
		sb_data_e1x.common.same_igu_sb_1b = true;
		sb_data_e1x.common.host_sb_addr.hi = U64_HI(mapping);
		sb_data_e1x.common.host_sb_addr.lo = U64_LO(mapping);
		hc_sm_p = sb_data_e1x.common.state_machine;
		sb_data_p = (u32 *)&sb_data_e1x;
		data_size = sizeof(struct hc_status_block_data_e1x)/sizeof(u32);
5037
		bnx2x_map_sb_state_machines(sb_data_e1x.index_data);
D
Dmitry Kravkov 已提交
5038
	}
5039 5040 5041 5042 5043 5044

	bnx2x_setup_ndsb_state_machine(&hc_sm_p[SM_RX_ID],
				       igu_sb_id, igu_seg_id);
	bnx2x_setup_ndsb_state_machine(&hc_sm_p[SM_TX_ID],
				       igu_sb_id, igu_seg_id);

M
Merav Sicron 已提交
5045
	DP(NETIF_MSG_IFUP, "Init FW SB %d\n", fw_sb_id);
5046 5047 5048 5049 5050

	/* write indecies to HW */
	bnx2x_wr_fp_sb_data(bp, fw_sb_id, sb_data_p, data_size);
}

5051
static void bnx2x_update_coalesce_sb(struct bnx2x *bp, u8 fw_sb_id,
5052 5053
				     u16 tx_usec, u16 rx_usec)
{
5054
	bnx2x_update_coalesce_sb_index(bp, fw_sb_id, HC_INDEX_ETH_RX_CQ_CONS,
5055
				    false, rx_usec);
5056 5057 5058 5059 5060 5061 5062 5063 5064
	bnx2x_update_coalesce_sb_index(bp, fw_sb_id,
				       HC_INDEX_ETH_TX_CQ_CONS_COS0, false,
				       tx_usec);
	bnx2x_update_coalesce_sb_index(bp, fw_sb_id,
				       HC_INDEX_ETH_TX_CQ_CONS_COS1, false,
				       tx_usec);
	bnx2x_update_coalesce_sb_index(bp, fw_sb_id,
				       HC_INDEX_ETH_TX_CQ_CONS_COS2, false,
				       tx_usec);
5065
}
D
Dmitry Kravkov 已提交
5066

5067 5068 5069 5070 5071 5072
static void bnx2x_init_def_sb(struct bnx2x *bp)
{
	struct host_sp_status_block *def_sb = bp->def_status_blk;
	dma_addr_t mapping = bp->def_status_blk_mapping;
	int igu_sp_sb_index;
	int igu_seg_id;
5073 5074
	int port = BP_PORT(bp);
	int func = BP_FUNC(bp);
5075
	int reg_offset, reg_offset_en5;
E
Eliezer Tamir 已提交
5076
	u64 section;
5077 5078 5079 5080
	int index;
	struct hc_sp_status_block_data sp_sb_data;
	memset(&sp_sb_data, 0, sizeof(struct hc_sp_status_block_data));

D
Dmitry Kravkov 已提交
5081 5082 5083 5084 5085 5086 5087
	if (CHIP_INT_MODE_IS_BC(bp)) {
		igu_sp_sb_index = DEF_SB_IGU_ID;
		igu_seg_id = HC_SEG_ACCESS_DEF;
	} else {
		igu_sp_sb_index = bp->igu_dsb_id;
		igu_seg_id = IGU_SEG_ACCESS_DEF;
	}
E
Eliezer Tamir 已提交
5088 5089

	/* ATTN */
5090
	section = ((u64)mapping) + offsetof(struct host_sp_status_block,
E
Eliezer Tamir 已提交
5091
					    atten_status_block);
5092
	def_sb->atten_status_block.status_block_id = igu_sp_sb_index;
E
Eliezer Tamir 已提交
5093

5094 5095
	bp->attn_state = 0;

E
Eliezer Tamir 已提交
5096 5097
	reg_offset = (port ? MISC_REG_AEU_ENABLE1_FUNC_1_OUT_0 :
			     MISC_REG_AEU_ENABLE1_FUNC_0_OUT_0);
5098 5099
	reg_offset_en5 = (port ? MISC_REG_AEU_ENABLE5_FUNC_1_OUT_0 :
				 MISC_REG_AEU_ENABLE5_FUNC_0_OUT_0);
5100
	for (index = 0; index < MAX_DYNAMIC_ATTN_GRPS; index++) {
5101 5102 5103 5104 5105
		int sindex;
		/* take care of sig[0]..sig[4] */
		for (sindex = 0; sindex < 4; sindex++)
			bp->attn_group[index].sig[sindex] =
			   REG_RD(bp, reg_offset + sindex*0x4 + 0x10*index);
D
Dmitry Kravkov 已提交
5106

5107
		if (!CHIP_IS_E1x(bp))
D
Dmitry Kravkov 已提交
5108 5109 5110 5111 5112 5113
			/*
			 * enable5 is separate from the rest of the registers,
			 * and therefore the address skip is 4
			 * and not 16 between the different groups
			 */
			bp->attn_group[index].sig[4] = REG_RD(bp,
5114
					reg_offset_en5 + 0x4*index);
D
Dmitry Kravkov 已提交
5115 5116
		else
			bp->attn_group[index].sig[4] = 0;
E
Eliezer Tamir 已提交
5117 5118
	}

D
Dmitry Kravkov 已提交
5119 5120 5121 5122 5123 5124
	if (bp->common.int_block == INT_BLOCK_HC) {
		reg_offset = (port ? HC_REG_ATTN_MSG1_ADDR_L :
				     HC_REG_ATTN_MSG0_ADDR_L);

		REG_WR(bp, reg_offset, U64_LO(section));
		REG_WR(bp, reg_offset + 4, U64_HI(section));
5125
	} else if (!CHIP_IS_E1x(bp)) {
D
Dmitry Kravkov 已提交
5126 5127 5128
		REG_WR(bp, IGU_REG_ATTN_MSG_ADDR_L, U64_LO(section));
		REG_WR(bp, IGU_REG_ATTN_MSG_ADDR_H, U64_HI(section));
	}
E
Eliezer Tamir 已提交
5129

5130 5131
	section = ((u64)mapping) + offsetof(struct host_sp_status_block,
					    sp_sb);
E
Eliezer Tamir 已提交
5132

5133
	bnx2x_zero_sp_sb(bp);
E
Eliezer Tamir 已提交
5134

5135
	sp_sb_data.state		= SB_ENABLED;
5136 5137 5138 5139 5140
	sp_sb_data.host_sb_addr.lo	= U64_LO(section);
	sp_sb_data.host_sb_addr.hi	= U64_HI(section);
	sp_sb_data.igu_sb_id		= igu_sp_sb_index;
	sp_sb_data.igu_seg_id		= igu_seg_id;
	sp_sb_data.p_func.pf_id		= func;
D
Dmitry Kravkov 已提交
5141
	sp_sb_data.p_func.vnic_id	= BP_VN(bp);
5142
	sp_sb_data.p_func.vf_id		= 0xff;
E
Eliezer Tamir 已提交
5143

5144
	bnx2x_wr_sp_sb_data(bp, &sp_sb_data);
5145

5146
	bnx2x_ack_sb(bp, bp->igu_dsb_id, USTORM_ID, 0, IGU_INT_ENABLE, 0);
E
Eliezer Tamir 已提交
5147 5148
}

D
Dmitry Kravkov 已提交
5149
void bnx2x_update_coalesce(struct bnx2x *bp)
E
Eliezer Tamir 已提交
5150 5151 5152
{
	int i;

V
Vladislav Zolotarov 已提交
5153
	for_each_eth_queue(bp, i)
5154
		bnx2x_update_coalesce_sb(bp, bp->fp[i].fw_sb_id,
5155
					 bp->tx_ticks, bp->rx_ticks);
E
Eliezer Tamir 已提交
5156 5157 5158 5159 5160
}

static void bnx2x_init_sp_ring(struct bnx2x *bp)
{
	spin_lock_init(&bp->spq_lock);
5161
	atomic_set(&bp->cq_spq_left, MAX_SPQ_PENDING);
E
Eliezer Tamir 已提交
5162 5163 5164 5165 5166 5167 5168

	bp->spq_prod_idx = 0;
	bp->dsb_sp_prod = BNX2X_SP_DSB_INDEX;
	bp->spq_prod_bd = bp->spq;
	bp->spq_last_bd = bp->spq_prod_bd + MAX_SP_DESC_CNT;
}

5169
static void bnx2x_init_eq_ring(struct bnx2x *bp)
E
Eliezer Tamir 已提交
5170 5171
{
	int i;
5172 5173 5174
	for (i = 1; i <= NUM_EQ_PAGES; i++) {
		union event_ring_elem *elem =
			&bp->eq_ring[EQ_DESC_CNT_PAGE * i - 1];
E
Eliezer Tamir 已提交
5175

5176 5177 5178 5179 5180 5181
		elem->next_page.addr.hi =
			cpu_to_le32(U64_HI(bp->eq_mapping +
				   BCM_PAGE_SIZE * (i % NUM_EQ_PAGES)));
		elem->next_page.addr.lo =
			cpu_to_le32(U64_LO(bp->eq_mapping +
				   BCM_PAGE_SIZE*(i % NUM_EQ_PAGES)));
E
Eliezer Tamir 已提交
5182
	}
5183 5184 5185
	bp->eq_cons = 0;
	bp->eq_prod = NUM_EQ_DESC;
	bp->eq_cons_sb = BNX2X_EQ_INDEX;
5186 5187 5188
	/* we want a warning message before it gets rought... */
	atomic_set(&bp->eq_spq_left,
		min_t(int, MAX_SP_DESC_CNT - MAX_SPQ_PENDING, NUM_EQ_DESC) - 1);
E
Eliezer Tamir 已提交
5189 5190
}

5191 5192 5193 5194 5195 5196 5197

/* called with netif_addr_lock_bh() */
void bnx2x_set_q_rx_mode(struct bnx2x *bp, u8 cl_id,
			 unsigned long rx_mode_flags,
			 unsigned long rx_accept_flags,
			 unsigned long tx_accept_flags,
			 unsigned long ramrod_flags)
5198
{
5199 5200 5201 5202 5203 5204 5205 5206 5207 5208
	struct bnx2x_rx_mode_ramrod_params ramrod_param;
	int rc;

	memset(&ramrod_param, 0, sizeof(ramrod_param));

	/* Prepare ramrod parameters */
	ramrod_param.cid = 0;
	ramrod_param.cl_id = cl_id;
	ramrod_param.rx_mode_obj = &bp->rx_mode_obj;
	ramrod_param.func_id = BP_FUNC(bp);
5209

5210 5211
	ramrod_param.pstate = &bp->sp_state;
	ramrod_param.state = BNX2X_FILTER_RX_MODE_PENDING;
5212

5213 5214 5215 5216 5217 5218 5219 5220 5221 5222 5223 5224 5225 5226 5227 5228
	ramrod_param.rdata = bnx2x_sp(bp, rx_mode_rdata);
	ramrod_param.rdata_mapping = bnx2x_sp_mapping(bp, rx_mode_rdata);

	set_bit(BNX2X_FILTER_RX_MODE_PENDING, &bp->sp_state);

	ramrod_param.ramrod_flags = ramrod_flags;
	ramrod_param.rx_mode_flags = rx_mode_flags;

	ramrod_param.rx_accept_flags = rx_accept_flags;
	ramrod_param.tx_accept_flags = tx_accept_flags;

	rc = bnx2x_config_rx_mode(bp, &ramrod_param);
	if (rc < 0) {
		BNX2X_ERR("Set rx_mode %d failed\n", bp->rx_mode);
		return;
	}
E
Eliezer Tamir 已提交
5229 5230
}

5231 5232
/* called with netif_addr_lock_bh() */
void bnx2x_set_storm_rx_mode(struct bnx2x *bp)
5233
{
5234 5235
	unsigned long rx_mode_flags = 0, ramrod_flags = 0;
	unsigned long rx_accept_flags = 0, tx_accept_flags = 0;
5236

5237 5238 5239 5240 5241 5242 5243 5244 5245 5246 5247 5248 5249 5250 5251 5252 5253 5254 5255 5256 5257 5258 5259 5260 5261 5262 5263 5264 5265 5266 5267 5268 5269 5270 5271 5272 5273 5274 5275 5276 5277 5278 5279 5280 5281 5282 5283 5284 5285 5286 5287 5288 5289 5290 5291 5292 5293 5294 5295 5296
#ifdef BCM_CNIC
	if (!NO_FCOE(bp))

		/* Configure rx_mode of FCoE Queue */
		__set_bit(BNX2X_RX_MODE_FCOE_ETH, &rx_mode_flags);
#endif

	switch (bp->rx_mode) {
	case BNX2X_RX_MODE_NONE:
		/*
		 * 'drop all' supersedes any accept flags that may have been
		 * passed to the function.
		 */
		break;
	case BNX2X_RX_MODE_NORMAL:
		__set_bit(BNX2X_ACCEPT_UNICAST, &rx_accept_flags);
		__set_bit(BNX2X_ACCEPT_MULTICAST, &rx_accept_flags);
		__set_bit(BNX2X_ACCEPT_BROADCAST, &rx_accept_flags);

		/* internal switching mode */
		__set_bit(BNX2X_ACCEPT_UNICAST, &tx_accept_flags);
		__set_bit(BNX2X_ACCEPT_MULTICAST, &tx_accept_flags);
		__set_bit(BNX2X_ACCEPT_BROADCAST, &tx_accept_flags);

		break;
	case BNX2X_RX_MODE_ALLMULTI:
		__set_bit(BNX2X_ACCEPT_UNICAST, &rx_accept_flags);
		__set_bit(BNX2X_ACCEPT_ALL_MULTICAST, &rx_accept_flags);
		__set_bit(BNX2X_ACCEPT_BROADCAST, &rx_accept_flags);

		/* internal switching mode */
		__set_bit(BNX2X_ACCEPT_UNICAST, &tx_accept_flags);
		__set_bit(BNX2X_ACCEPT_ALL_MULTICAST, &tx_accept_flags);
		__set_bit(BNX2X_ACCEPT_BROADCAST, &tx_accept_flags);

		break;
	case BNX2X_RX_MODE_PROMISC:
		/* According to deffinition of SI mode, iface in promisc mode
		 * should receive matched and unmatched (in resolution of port)
		 * unicast packets.
		 */
		__set_bit(BNX2X_ACCEPT_UNMATCHED, &rx_accept_flags);
		__set_bit(BNX2X_ACCEPT_UNICAST, &rx_accept_flags);
		__set_bit(BNX2X_ACCEPT_ALL_MULTICAST, &rx_accept_flags);
		__set_bit(BNX2X_ACCEPT_BROADCAST, &rx_accept_flags);

		/* internal switching mode */
		__set_bit(BNX2X_ACCEPT_ALL_MULTICAST, &tx_accept_flags);
		__set_bit(BNX2X_ACCEPT_BROADCAST, &tx_accept_flags);

		if (IS_MF_SI(bp))
			__set_bit(BNX2X_ACCEPT_ALL_UNICAST, &tx_accept_flags);
		else
			__set_bit(BNX2X_ACCEPT_UNICAST, &tx_accept_flags);

		break;
	default:
		BNX2X_ERR("Unknown rx_mode: %d\n", bp->rx_mode);
		return;
	}
E
Eilon Greenstein 已提交
5297

5298 5299 5300
	if (bp->rx_mode != BNX2X_RX_MODE_NONE) {
		__set_bit(BNX2X_ACCEPT_ANY_VLAN, &rx_accept_flags);
		__set_bit(BNX2X_ACCEPT_ANY_VLAN, &tx_accept_flags);
5301 5302
	}

5303 5304 5305 5306 5307 5308 5309 5310 5311 5312 5313
	__set_bit(RAMROD_RX, &ramrod_flags);
	__set_bit(RAMROD_TX, &ramrod_flags);

	bnx2x_set_q_rx_mode(bp, bp->fp->cl_id, rx_mode_flags, rx_accept_flags,
			    tx_accept_flags, ramrod_flags);
}

static void bnx2x_init_internal_common(struct bnx2x *bp)
{
	int i;

5314 5315 5316 5317 5318 5319 5320 5321
	if (IS_MF_SI(bp))
		/*
		 * In switch independent mode, the TSTORM needs to accept
		 * packets that failed classification, since approximate match
		 * mac addresses aren't written to NIG LLH
		 */
		REG_WR8(bp, BAR_TSTRORM_INTMEM +
			    TSTORM_ACCEPT_CLASSIFY_FAILED_OFFSET, 2);
5322 5323 5324
	else if (!CHIP_IS_E1(bp)) /* 57710 doesn't support MF */
		REG_WR8(bp, BAR_TSTRORM_INTMEM +
			    TSTORM_ACCEPT_CLASSIFY_FAILED_OFFSET, 0);
5325

5326 5327 5328
	/* Zero this manually as its initialization is
	   currently missing in the initTool */
	for (i = 0; i < (USTORM_AGG_DATA_SIZE >> 2); i++)
E
Eilon Greenstein 已提交
5329
		REG_WR(bp, BAR_USTRORM_INTMEM +
5330
		       USTORM_AGG_DATA_OFFSET + i * 4, 0);
5331
	if (!CHIP_IS_E1x(bp)) {
D
Dmitry Kravkov 已提交
5332 5333 5334 5335
		REG_WR8(bp, BAR_CSTRORM_INTMEM + CSTORM_IGU_MODE_OFFSET,
			CHIP_INT_MODE_IS_BC(bp) ?
			HC_IGU_BC_MODE : HC_IGU_NBC_MODE);
	}
5336
}
E
Eilon Greenstein 已提交
5337

5338 5339 5340 5341
static void bnx2x_init_internal(struct bnx2x *bp, u32 load_code)
{
	switch (load_code) {
	case FW_MSG_CODE_DRV_LOAD_COMMON:
D
Dmitry Kravkov 已提交
5342
	case FW_MSG_CODE_DRV_LOAD_COMMON_CHIP:
5343 5344 5345 5346
		bnx2x_init_internal_common(bp);
		/* no break */

	case FW_MSG_CODE_DRV_LOAD_PORT:
5347
		/* nothing to do */
5348 5349 5350
		/* no break */

	case FW_MSG_CODE_DRV_LOAD_FUNCTION:
5351 5352
		/* internal memory per function is
		   initialized inside bnx2x_pf_init */
5353 5354 5355 5356 5357 5358 5359 5360
		break;

	default:
		BNX2X_ERR("Unknown load_code (0x%x) from MCP\n", load_code);
		break;
	}
}

5361
static inline u8 bnx2x_fp_igu_sb_id(struct bnx2x_fastpath *fp)
5362
{
5363
	return fp->bp->igu_base_sb + fp->index + CNIC_PRESENT;
5364
}
5365

5366 5367
static inline u8 bnx2x_fp_fw_sb_id(struct bnx2x_fastpath *fp)
{
5368
	return fp->bp->base_fw_ndsb + fp->index + CNIC_PRESENT;
5369 5370 5371 5372 5373 5374 5375 5376 5377 5378
}

static inline u8 bnx2x_fp_cl_id(struct bnx2x_fastpath *fp)
{
	if (CHIP_IS_E1x(fp->bp))
		return BP_L_ID(fp->bp) + fp->index;
	else	/* We want Client ID to be the same as IGU SB ID for 57712 */
		return bnx2x_fp_igu_sb_id(fp);
}

5379
static void bnx2x_init_eth_fp(struct bnx2x *bp, int fp_idx)
5380 5381
{
	struct bnx2x_fastpath *fp = &bp->fp[fp_idx];
5382
	u8 cos;
5383
	unsigned long q_type = 0;
5384
	u32 cids[BNX2X_MULTI_TX_COS] = { 0 };
5385
	fp->rx_queue = fp_idx;
5386
	fp->cid = fp_idx;
5387 5388 5389
	fp->cl_id = bnx2x_fp_cl_id(fp);
	fp->fw_sb_id = bnx2x_fp_fw_sb_id(fp);
	fp->igu_sb_id = bnx2x_fp_igu_sb_id(fp);
5390
	/* qZone id equals to FW (per path) client id */
5391 5392
	fp->cl_qzone_id  = bnx2x_fp_qzone_id(fp);

5393
	/* init shortcut */
5394
	fp->ustorm_rx_prods_offset = bnx2x_rx_ustorm_prods_offset(fp);
5395

5396 5397 5398
	/* Setup SB indicies */
	fp->rx_cons_sb = BNX2X_RX_SB_INDEX;

5399 5400 5401
	/* Configure Queue State object */
	__set_bit(BNX2X_Q_TYPE_HAS_RX, &q_type);
	__set_bit(BNX2X_Q_TYPE_HAS_TX, &q_type);
5402 5403 5404 5405 5406 5407 5408 5409 5410 5411 5412 5413 5414 5415 5416

	BUG_ON(fp->max_cos > BNX2X_MULTI_TX_COS);

	/* init tx data */
	for_each_cos_in_tx_queue(fp, cos) {
		bnx2x_init_txdata(bp, &fp->txdata[cos],
				  CID_COS_TO_TX_ONLY_CID(fp->cid, cos),
				  FP_COS_TO_TXQ(fp, cos),
				  BNX2X_TX_SB_INDEX_BASE + cos);
		cids[cos] = fp->txdata[cos].cid;
	}

	bnx2x_init_queue_obj(bp, &fp->q_obj, fp->cl_id, cids, fp->max_cos,
			     BP_FUNC(bp), bnx2x_sp(bp, q_rdata),
			     bnx2x_sp_mapping(bp, q_rdata), q_type);
5417 5418 5419 5420 5421 5422

	/**
	 * Configure classification DBs: Always enable Tx switching
	 */
	bnx2x_init_vlan_mac_fp_objs(fp, BNX2X_OBJ_TYPE_RX_TX);

M
Merav Sicron 已提交
5423
	DP(NETIF_MSG_IFUP, "queue[%d]:  bnx2x_init_sb(%p,%p)  cl_id %d  fw_sb %d  igu_sb %d\n",
5424
		   fp_idx, bp, fp->status_blk.e2_sb, fp->cl_id, fp->fw_sb_id,
5425 5426 5427 5428 5429 5430 5431
		   fp->igu_sb_id);
	bnx2x_init_sb(bp, fp->status_blk_mapping, BNX2X_VF_ID_INVALID, false,
		      fp->fw_sb_id, fp->igu_sb_id);

	bnx2x_update_fpsb_idx(fp);
}

D
Dmitry Kravkov 已提交
5432
void bnx2x_nic_init(struct bnx2x *bp, u32 load_code)
E
Eliezer Tamir 已提交
5433 5434 5435
{
	int i;

V
Vladislav Zolotarov 已提交
5436
	for_each_eth_queue(bp, i)
5437
		bnx2x_init_eth_fp(bp, i);
5438
#ifdef BCM_CNIC
V
Vladislav Zolotarov 已提交
5439 5440
	if (!NO_FCOE(bp))
		bnx2x_init_fcoe_fp(bp);
5441 5442 5443

	bnx2x_init_sb(bp, bp->cnic_sb_mapping,
		      BNX2X_VF_ID_INVALID, false,
5444
		      bnx2x_cnic_fw_sb_id(bp), bnx2x_cnic_igu_sb_id(bp));
5445

5446
#endif
E
Eliezer Tamir 已提交
5447

5448 5449 5450 5451
	/* Initialize MOD_ABS interrupts */
	bnx2x_init_mod_abs_int(bp, &bp->link_vars, bp->common.chip_id,
			       bp->common.shmem_base, bp->common.shmem2_base,
			       BP_PORT(bp));
5452 5453 5454
	/* ensure status block indices were read */
	rmb();

5455
	bnx2x_init_def_sb(bp);
5456
	bnx2x_update_dsb_idx(bp);
E
Eliezer Tamir 已提交
5457
	bnx2x_init_rx_rings(bp);
5458
	bnx2x_init_tx_rings(bp);
E
Eliezer Tamir 已提交
5459
	bnx2x_init_sp_ring(bp);
5460
	bnx2x_init_eq_ring(bp);
5461
	bnx2x_init_internal(bp, load_code);
5462
	bnx2x_pf_init(bp);
5463 5464 5465 5466 5467 5468
	bnx2x_stats_init(bp);

	/* flush all before enabling interrupts */
	mb();
	mmiowb();

E
Eliezer Tamir 已提交
5469
	bnx2x_int_enable(bp);
5470 5471 5472 5473 5474

	/* Check for SPIO5 */
	bnx2x_attn_int_deasserted0(bp,
		REG_RD(bp, MISC_REG_AEU_AFTER_INVERT_1_FUNC_0 + BP_PORT(bp)*4) &
				   AEU_INPUTS_ATTN_BITS_SPIO5);
E
Eliezer Tamir 已提交
5475 5476 5477 5478 5479 5480 5481 5482 5483 5484
}

/* end of nic init */

/*
 * gzip service functions
 */

static int bnx2x_gunzip_init(struct bnx2x *bp)
{
5485 5486
	bp->gunzip_buf = dma_alloc_coherent(&bp->pdev->dev, FW_BUF_SIZE,
					    &bp->gunzip_mapping, GFP_KERNEL);
E
Eliezer Tamir 已提交
5487 5488 5489 5490 5491 5492 5493
	if (bp->gunzip_buf  == NULL)
		goto gunzip_nomem1;

	bp->strm = kmalloc(sizeof(*bp->strm), GFP_KERNEL);
	if (bp->strm  == NULL)
		goto gunzip_nomem2;

5494
	bp->strm->workspace = vmalloc(zlib_inflate_workspacesize());
E
Eliezer Tamir 已提交
5495 5496 5497 5498 5499 5500 5501 5502 5503 5504
	if (bp->strm->workspace == NULL)
		goto gunzip_nomem3;

	return 0;

gunzip_nomem3:
	kfree(bp->strm);
	bp->strm = NULL;

gunzip_nomem2:
5505 5506
	dma_free_coherent(&bp->pdev->dev, FW_BUF_SIZE, bp->gunzip_buf,
			  bp->gunzip_mapping);
E
Eliezer Tamir 已提交
5507 5508 5509
	bp->gunzip_buf = NULL;

gunzip_nomem1:
M
Merav Sicron 已提交
5510
	BNX2X_ERR("Cannot allocate firmware buffer for un-compression\n");
E
Eliezer Tamir 已提交
5511 5512 5513 5514 5515
	return -ENOMEM;
}

static void bnx2x_gunzip_end(struct bnx2x *bp)
{
5516
	if (bp->strm) {
5517
		vfree(bp->strm->workspace);
5518 5519 5520
		kfree(bp->strm);
		bp->strm = NULL;
	}
E
Eliezer Tamir 已提交
5521 5522

	if (bp->gunzip_buf) {
5523 5524
		dma_free_coherent(&bp->pdev->dev, FW_BUF_SIZE, bp->gunzip_buf,
				  bp->gunzip_mapping);
E
Eliezer Tamir 已提交
5525 5526 5527 5528
		bp->gunzip_buf = NULL;
	}
}

5529
static int bnx2x_gunzip(struct bnx2x *bp, const u8 *zbuf, int len)
E
Eliezer Tamir 已提交
5530 5531 5532 5533
{
	int n, rc;

	/* check gzip header */
5534 5535
	if ((zbuf[0] != 0x1f) || (zbuf[1] != 0x8b) || (zbuf[2] != Z_DEFLATED)) {
		BNX2X_ERR("Bad gzip header\n");
E
Eliezer Tamir 已提交
5536
		return -EINVAL;
5537
	}
E
Eliezer Tamir 已提交
5538 5539 5540

	n = 10;

5541
#define FNAME				0x8
E
Eliezer Tamir 已提交
5542 5543 5544 5545

	if (zbuf[3] & FNAME)
		while ((zbuf[n++] != 0) && (n < len));

5546
	bp->strm->next_in = (typeof(bp->strm->next_in))zbuf + n;
E
Eliezer Tamir 已提交
5547 5548 5549 5550 5551 5552 5553 5554 5555 5556
	bp->strm->avail_in = len - n;
	bp->strm->next_out = bp->gunzip_buf;
	bp->strm->avail_out = FW_BUF_SIZE;

	rc = zlib_inflateInit2(bp->strm, -MAX_WBITS);
	if (rc != Z_OK)
		return rc;

	rc = zlib_inflate(bp->strm, Z_FINISH);
	if ((rc != Z_OK) && (rc != Z_STREAM_END))
5557 5558
		netdev_err(bp->dev, "Firmware decompression error: %s\n",
			   bp->strm->msg);
E
Eliezer Tamir 已提交
5559 5560 5561

	bp->gunzip_outlen = (FW_BUF_SIZE - bp->strm->avail_out);
	if (bp->gunzip_outlen & 0x3)
M
Merav Sicron 已提交
5562 5563
		netdev_err(bp->dev,
			   "Firmware decompression error: gunzip_outlen (%d) not aligned\n",
V
Vladislav Zolotarov 已提交
5564
				bp->gunzip_outlen);
E
Eliezer Tamir 已提交
5565 5566 5567 5568 5569 5570 5571 5572 5573 5574 5575 5576 5577
	bp->gunzip_outlen >>= 2;

	zlib_inflateEnd(bp->strm);

	if (rc == Z_STREAM_END)
		return 0;

	return rc;
}

/* nic load/unload */

/*
5578
 * General service functions
E
Eliezer Tamir 已提交
5579 5580 5581 5582 5583 5584 5585 5586 5587 5588
 */

/* send a NIG loopback debug packet */
static void bnx2x_lb_pckt(struct bnx2x *bp)
{
	u32 wb_write[3];

	/* Ethernet source and destination addresses */
	wb_write[0] = 0x55555555;
	wb_write[1] = 0x55555555;
5589
	wb_write[2] = 0x20;		/* SOP */
E
Eliezer Tamir 已提交
5590 5591 5592 5593 5594
	REG_WR_DMAE(bp, NIG_REG_DEBUG_PACKET_LB, wb_write, 3);

	/* NON-IP protocol */
	wb_write[0] = 0x09000000;
	wb_write[1] = 0x55555555;
5595
	wb_write[2] = 0x10;		/* EOP, eop_bvalid = 0 */
E
Eliezer Tamir 已提交
5596 5597 5598 5599 5600 5601 5602 5603 5604 5605 5606 5607 5608
	REG_WR_DMAE(bp, NIG_REG_DEBUG_PACKET_LB, wb_write, 3);
}

/* some of the internal memories
 * are not directly readable from the driver
 * to test them we send debug packets
 */
static int bnx2x_int_mem_test(struct bnx2x *bp)
{
	int factor;
	int count, i;
	u32 val = 0;

5609
	if (CHIP_REV_IS_FPGA(bp))
E
Eliezer Tamir 已提交
5610
		factor = 120;
5611 5612 5613
	else if (CHIP_REV_IS_EMUL(bp))
		factor = 200;
	else
E
Eliezer Tamir 已提交
5614 5615 5616 5617 5618 5619
		factor = 1;

	/* Disable inputs of parser neighbor blocks */
	REG_WR(bp, TSDM_REG_ENABLE_IN1, 0x0);
	REG_WR(bp, TCM_REG_PRS_IFEN, 0x0);
	REG_WR(bp, CFC_REG_DEBUG0, 0x1);
5620
	REG_WR(bp, NIG_REG_PRS_REQ_IN_EN, 0x0);
E
Eliezer Tamir 已提交
5621 5622 5623 5624 5625 5626 5627 5628 5629 5630 5631

	/*  Write 0 to parser credits for CFC search request */
	REG_WR(bp, PRS_REG_CFC_SEARCH_INITIAL_CREDIT, 0x0);

	/* send Ethernet packet */
	bnx2x_lb_pckt(bp);

	/* TODO do i reset NIG statistic? */
	/* Wait until NIG register shows 1 packet of size 0x10 */
	count = 1000 * factor;
	while (count) {
5632

E
Eliezer Tamir 已提交
5633 5634 5635 5636 5637 5638 5639 5640 5641 5642 5643 5644 5645 5646 5647 5648 5649 5650 5651 5652 5653 5654 5655 5656 5657 5658 5659 5660 5661
		bnx2x_read_dmae(bp, NIG_REG_STAT2_BRB_OCTET, 2);
		val = *bnx2x_sp(bp, wb_data[0]);
		if (val == 0x10)
			break;

		msleep(10);
		count--;
	}
	if (val != 0x10) {
		BNX2X_ERR("NIG timeout  val = 0x%x\n", val);
		return -1;
	}

	/* Wait until PRS register shows 1 packet */
	count = 1000 * factor;
	while (count) {
		val = REG_RD(bp, PRS_REG_NUM_OF_PACKETS);
		if (val == 1)
			break;

		msleep(10);
		count--;
	}
	if (val != 0x1) {
		BNX2X_ERR("PRS timeout val = 0x%x\n", val);
		return -2;
	}

	/* Reset and init BRB, PRS */
5662
	REG_WR(bp, GRCBASE_MISC + MISC_REGISTERS_RESET_REG_1_CLEAR, 0x03);
E
Eliezer Tamir 已提交
5663
	msleep(50);
5664
	REG_WR(bp, GRCBASE_MISC + MISC_REGISTERS_RESET_REG_1_SET, 0x03);
E
Eliezer Tamir 已提交
5665
	msleep(50);
5666 5667
	bnx2x_init_block(bp, BLOCK_BRB1, PHASE_COMMON);
	bnx2x_init_block(bp, BLOCK_PRS, PHASE_COMMON);
E
Eliezer Tamir 已提交
5668 5669 5670 5671 5672 5673 5674

	DP(NETIF_MSG_HW, "part2\n");

	/* Disable inputs of parser neighbor blocks */
	REG_WR(bp, TSDM_REG_ENABLE_IN1, 0x0);
	REG_WR(bp, TCM_REG_PRS_IFEN, 0x0);
	REG_WR(bp, CFC_REG_DEBUG0, 0x1);
5675
	REG_WR(bp, NIG_REG_PRS_REQ_IN_EN, 0x0);
E
Eliezer Tamir 已提交
5676 5677 5678 5679 5680 5681 5682 5683 5684 5685 5686 5687

	/* Write 0 to parser credits for CFC search request */
	REG_WR(bp, PRS_REG_CFC_SEARCH_INITIAL_CREDIT, 0x0);

	/* send 10 Ethernet packets */
	for (i = 0; i < 10; i++)
		bnx2x_lb_pckt(bp);

	/* Wait until NIG register shows 10 + 1
	   packets of size 11*0x10 = 0xb0 */
	count = 1000 * factor;
	while (count) {
5688

E
Eliezer Tamir 已提交
5689 5690 5691 5692 5693 5694 5695 5696 5697 5698 5699 5700 5701 5702 5703 5704 5705 5706 5707 5708 5709 5710 5711 5712 5713 5714 5715 5716 5717 5718 5719 5720 5721 5722 5723 5724 5725 5726 5727 5728 5729 5730
		bnx2x_read_dmae(bp, NIG_REG_STAT2_BRB_OCTET, 2);
		val = *bnx2x_sp(bp, wb_data[0]);
		if (val == 0xb0)
			break;

		msleep(10);
		count--;
	}
	if (val != 0xb0) {
		BNX2X_ERR("NIG timeout  val = 0x%x\n", val);
		return -3;
	}

	/* Wait until PRS register shows 2 packets */
	val = REG_RD(bp, PRS_REG_NUM_OF_PACKETS);
	if (val != 2)
		BNX2X_ERR("PRS timeout  val = 0x%x\n", val);

	/* Write 1 to parser credits for CFC search request */
	REG_WR(bp, PRS_REG_CFC_SEARCH_INITIAL_CREDIT, 0x1);

	/* Wait until PRS register shows 3 packets */
	msleep(10 * factor);
	/* Wait until NIG register shows 1 packet of size 0x10 */
	val = REG_RD(bp, PRS_REG_NUM_OF_PACKETS);
	if (val != 3)
		BNX2X_ERR("PRS timeout  val = 0x%x\n", val);

	/* clear NIG EOP FIFO */
	for (i = 0; i < 11; i++)
		REG_RD(bp, NIG_REG_INGRESS_EOP_LB_FIFO);
	val = REG_RD(bp, NIG_REG_INGRESS_EOP_LB_EMPTY);
	if (val != 1) {
		BNX2X_ERR("clear of NIG failed\n");
		return -4;
	}

	/* Reset and init BRB, PRS, NIG */
	REG_WR(bp, GRCBASE_MISC + MISC_REGISTERS_RESET_REG_1_CLEAR, 0x03);
	msleep(50);
	REG_WR(bp, GRCBASE_MISC + MISC_REGISTERS_RESET_REG_1_SET, 0x03);
	msleep(50);
5731 5732
	bnx2x_init_block(bp, BLOCK_BRB1, PHASE_COMMON);
	bnx2x_init_block(bp, BLOCK_PRS, PHASE_COMMON);
5733
#ifndef BCM_CNIC
E
Eliezer Tamir 已提交
5734 5735 5736 5737 5738 5739 5740 5741
	/* set NIC mode */
	REG_WR(bp, PRS_REG_NIC_MODE, 1);
#endif

	/* Enable inputs of parser neighbor blocks */
	REG_WR(bp, TSDM_REG_ENABLE_IN1, 0x7fffffff);
	REG_WR(bp, TCM_REG_PRS_IFEN, 0x1);
	REG_WR(bp, CFC_REG_DEBUG0, 0x0);
5742
	REG_WR(bp, NIG_REG_PRS_REQ_IN_EN, 0x1);
E
Eliezer Tamir 已提交
5743 5744 5745 5746 5747 5748

	DP(NETIF_MSG_HW, "done\n");

	return 0; /* OK */
}

5749
static void bnx2x_enable_blocks_attention(struct bnx2x *bp)
E
Eliezer Tamir 已提交
5750 5751
{
	REG_WR(bp, PXP_REG_PXP_INT_MASK_0, 0);
5752
	if (!CHIP_IS_E1x(bp))
D
Dmitry Kravkov 已提交
5753 5754 5755
		REG_WR(bp, PXP_REG_PXP_INT_MASK_1, 0x40);
	else
		REG_WR(bp, PXP_REG_PXP_INT_MASK_1, 0);
E
Eliezer Tamir 已提交
5756 5757
	REG_WR(bp, DORQ_REG_DORQ_INT_MASK, 0);
	REG_WR(bp, CFC_REG_CFC_INT_MASK, 0);
D
Dmitry Kravkov 已提交
5758 5759 5760 5761 5762 5763 5764
	/*
	 * mask read length error interrupts in brb for parser
	 * (parsing unit and 'checksum and crc' unit)
	 * these errors are legal (PU reads fixed length and CAC can cause
	 * read length error on truncated packets)
	 */
	REG_WR(bp, BRB1_REG_BRB1_INT_MASK, 0xFC00);
E
Eliezer Tamir 已提交
5765 5766 5767 5768 5769
	REG_WR(bp, QM_REG_QM_INT_MASK, 0);
	REG_WR(bp, TM_REG_TM_INT_MASK, 0);
	REG_WR(bp, XSDM_REG_XSDM_INT_MASK_0, 0);
	REG_WR(bp, XSDM_REG_XSDM_INT_MASK_1, 0);
	REG_WR(bp, XCM_REG_XCM_INT_MASK, 0);
5770 5771
/*	REG_WR(bp, XSEM_REG_XSEM_INT_MASK_0, 0); */
/*	REG_WR(bp, XSEM_REG_XSEM_INT_MASK_1, 0); */
E
Eliezer Tamir 已提交
5772 5773 5774
	REG_WR(bp, USDM_REG_USDM_INT_MASK_0, 0);
	REG_WR(bp, USDM_REG_USDM_INT_MASK_1, 0);
	REG_WR(bp, UCM_REG_UCM_INT_MASK, 0);
5775 5776
/*	REG_WR(bp, USEM_REG_USEM_INT_MASK_0, 0); */
/*	REG_WR(bp, USEM_REG_USEM_INT_MASK_1, 0); */
E
Eliezer Tamir 已提交
5777 5778 5779 5780
	REG_WR(bp, GRCBASE_UPB + PB_REG_PB_INT_MASK, 0);
	REG_WR(bp, CSDM_REG_CSDM_INT_MASK_0, 0);
	REG_WR(bp, CSDM_REG_CSDM_INT_MASK_1, 0);
	REG_WR(bp, CCM_REG_CCM_INT_MASK, 0);
5781 5782
/*	REG_WR(bp, CSEM_REG_CSEM_INT_MASK_0, 0); */
/*	REG_WR(bp, CSEM_REG_CSEM_INT_MASK_1, 0); */
D
Dmitry Kravkov 已提交
5783

5784 5785
	if (CHIP_REV_IS_FPGA(bp))
		REG_WR(bp, PXP2_REG_PXP2_INT_MASK_0, 0x580000);
5786
	else if (!CHIP_IS_E1x(bp))
D
Dmitry Kravkov 已提交
5787 5788 5789 5790 5791 5792
		REG_WR(bp, PXP2_REG_PXP2_INT_MASK_0,
			   (PXP2_PXP2_INT_MASK_0_REG_PGL_CPL_OF
				| PXP2_PXP2_INT_MASK_0_REG_PGL_CPL_AFT
				| PXP2_PXP2_INT_MASK_0_REG_PGL_PCIE_ATTN
				| PXP2_PXP2_INT_MASK_0_REG_PGL_READ_BLOCKED
				| PXP2_PXP2_INT_MASK_0_REG_PGL_WRITE_BLOCKED));
5793 5794
	else
		REG_WR(bp, PXP2_REG_PXP2_INT_MASK_0, 0x480000);
E
Eliezer Tamir 已提交
5795 5796 5797
	REG_WR(bp, TSDM_REG_TSDM_INT_MASK_0, 0);
	REG_WR(bp, TSDM_REG_TSDM_INT_MASK_1, 0);
	REG_WR(bp, TCM_REG_TCM_INT_MASK, 0);
5798
/*	REG_WR(bp, TSEM_REG_TSEM_INT_MASK_0, 0); */
5799 5800 5801 5802 5803

	if (!CHIP_IS_E1x(bp))
		/* enable VFC attentions: bits 11 and 12, bits 31:13 reserved */
		REG_WR(bp, TSEM_REG_TSEM_INT_MASK_1, 0x07ff);

E
Eliezer Tamir 已提交
5804 5805
	REG_WR(bp, CDU_REG_CDU_INT_MASK, 0);
	REG_WR(bp, DMAE_REG_DMAE_INT_MASK, 0);
5806
/*	REG_WR(bp, MISC_REG_MISC_INT_MASK, 0); */
5807
	REG_WR(bp, PBF_REG_PBF_INT_MASK, 0x18);		/* bit 3,4 masked */
E
Eliezer Tamir 已提交
5808 5809
}

E
Eilon Greenstein 已提交
5810 5811
static void bnx2x_reset_common(struct bnx2x *bp)
{
5812 5813
	u32 val = 0x1400;

E
Eilon Greenstein 已提交
5814 5815 5816
	/* reset_common */
	REG_WR(bp, GRCBASE_MISC + MISC_REGISTERS_RESET_REG_1_CLEAR,
	       0xd3ffff7f);
5817 5818 5819 5820 5821 5822 5823 5824 5825 5826 5827 5828 5829

	if (CHIP_IS_E3(bp)) {
		val |= MISC_REGISTERS_RESET_REG_2_MSTAT0;
		val |= MISC_REGISTERS_RESET_REG_2_MSTAT1;
	}

	REG_WR(bp, GRCBASE_MISC + MISC_REGISTERS_RESET_REG_2_CLEAR, val);
}

static void bnx2x_setup_dmae(struct bnx2x *bp)
{
	bp->dmae_ready = 0;
	spin_lock_init(&bp->dmae_lock);
E
Eilon Greenstein 已提交
5830 5831
}

5832 5833 5834 5835 5836 5837
static void bnx2x_init_pxp(struct bnx2x *bp)
{
	u16 devctl;
	int r_order, w_order;

	pci_read_config_word(bp->pdev,
V
Vladislav Zolotarov 已提交
5838
			     pci_pcie_cap(bp->pdev) + PCI_EXP_DEVCTL, &devctl);
5839 5840 5841 5842 5843 5844 5845 5846 5847 5848 5849
	DP(NETIF_MSG_HW, "read 0x%x from devctl\n", devctl);
	w_order = ((devctl & PCI_EXP_DEVCTL_PAYLOAD) >> 5);
	if (bp->mrrs == -1)
		r_order = ((devctl & PCI_EXP_DEVCTL_READRQ) >> 12);
	else {
		DP(NETIF_MSG_HW, "force read order to %d\n", bp->mrrs);
		r_order = bp->mrrs;
	}

	bnx2x_init_pxp_arb(bp, r_order, w_order);
}
E
Eilon Greenstein 已提交
5850 5851 5852

static void bnx2x_setup_fan_failure_detection(struct bnx2x *bp)
{
5853
	int is_required;
E
Eilon Greenstein 已提交
5854
	u32 val;
5855
	int port;
E
Eilon Greenstein 已提交
5856

5857 5858 5859 5860
	if (BP_NOMCP(bp))
		return;

	is_required = 0;
E
Eilon Greenstein 已提交
5861 5862 5863 5864 5865 5866 5867 5868 5869 5870 5871 5872 5873 5874
	val = SHMEM_RD(bp, dev_info.shared_hw_config.config2) &
	      SHARED_HW_CFG_FAN_FAILURE_MASK;

	if (val == SHARED_HW_CFG_FAN_FAILURE_ENABLED)
		is_required = 1;

	/*
	 * The fan failure mechanism is usually related to the PHY type since
	 * the power consumption of the board is affected by the PHY. Currently,
	 * fan is required for most designs with SFX7101, BCM8727 and BCM8481.
	 */
	else if (val == SHARED_HW_CFG_FAN_FAILURE_PHY_TYPE)
		for (port = PORT_0; port < PORT_MAX; port++) {
			is_required |=
5875 5876 5877
				bnx2x_fan_failure_det_req(
					bp,
					bp->common.shmem_base,
Y
Yaniv Rosner 已提交
5878
					bp->common.shmem2_base,
5879
					port);
E
Eilon Greenstein 已提交
5880 5881 5882 5883 5884 5885 5886 5887 5888 5889 5890 5891 5892 5893
		}

	DP(NETIF_MSG_HW, "fan detection setting: %d\n", is_required);

	if (is_required == 0)
		return;

	/* Fan failure is indicated by SPIO 5 */
	bnx2x_set_spio(bp, MISC_REGISTERS_SPIO_5,
		       MISC_REGISTERS_SPIO_INPUT_HI_Z);

	/* set to active low mode */
	val = REG_RD(bp, MISC_REG_SPIO_INT);
	val |= ((1 << MISC_REGISTERS_SPIO_5) <<
V
Vladislav Zolotarov 已提交
5894
					MISC_REGISTERS_SPIO_INT_OLD_SET_POS);
E
Eilon Greenstein 已提交
5895 5896 5897 5898 5899 5900 5901 5902
	REG_WR(bp, MISC_REG_SPIO_INT, val);

	/* enable interrupt to signal the IGU */
	val = REG_RD(bp, MISC_REG_SPIO_EVENT_EN);
	val |= (1 << MISC_REGISTERS_SPIO_5);
	REG_WR(bp, MISC_REG_SPIO_EVENT_EN, val);
}

D
Dmitry Kravkov 已提交
5903 5904 5905 5906 5907 5908 5909 5910 5911 5912 5913 5914 5915 5916 5917 5918 5919 5920 5921 5922 5923 5924 5925 5926 5927 5928 5929 5930 5931 5932 5933 5934 5935 5936 5937 5938 5939 5940 5941 5942 5943 5944 5945
static void bnx2x_pretend_func(struct bnx2x *bp, u8 pretend_func_num)
{
	u32 offset = 0;

	if (CHIP_IS_E1(bp))
		return;
	if (CHIP_IS_E1H(bp) && (pretend_func_num >= E1H_FUNC_MAX))
		return;

	switch (BP_ABS_FUNC(bp)) {
	case 0:
		offset = PXP2_REG_PGL_PRETEND_FUNC_F0;
		break;
	case 1:
		offset = PXP2_REG_PGL_PRETEND_FUNC_F1;
		break;
	case 2:
		offset = PXP2_REG_PGL_PRETEND_FUNC_F2;
		break;
	case 3:
		offset = PXP2_REG_PGL_PRETEND_FUNC_F3;
		break;
	case 4:
		offset = PXP2_REG_PGL_PRETEND_FUNC_F4;
		break;
	case 5:
		offset = PXP2_REG_PGL_PRETEND_FUNC_F5;
		break;
	case 6:
		offset = PXP2_REG_PGL_PRETEND_FUNC_F6;
		break;
	case 7:
		offset = PXP2_REG_PGL_PRETEND_FUNC_F7;
		break;
	default:
		return;
	}

	REG_WR(bp, offset, pretend_func_num);
	REG_RD(bp, offset);
	DP(NETIF_MSG_HW, "Pretending to func %d\n", pretend_func_num);
}

5946
void bnx2x_pf_disable(struct bnx2x *bp)
D
Dmitry Kravkov 已提交
5947 5948 5949 5950 5951 5952 5953 5954 5955
{
	u32 val = REG_RD(bp, IGU_REG_PF_CONFIGURATION);
	val &= ~IGU_PF_CONF_FUNC_EN;

	REG_WR(bp, IGU_REG_PF_CONFIGURATION, val);
	REG_WR(bp, PGLUE_B_REG_INTERNAL_PFID_ENABLE_MASTER, 0);
	REG_WR(bp, CFC_REG_WEAK_ENABLE_PF, 0);
}

5956 5957 5958 5959 5960 5961 5962 5963 5964 5965 5966 5967 5968 5969 5970 5971 5972 5973 5974 5975 5976 5977 5978
static inline void bnx2x__common_init_phy(struct bnx2x *bp)
{
	u32 shmem_base[2], shmem2_base[2];
	shmem_base[0] =  bp->common.shmem_base;
	shmem2_base[0] = bp->common.shmem2_base;
	if (!CHIP_IS_E1x(bp)) {
		shmem_base[1] =
			SHMEM2_RD(bp, other_shmem_base_addr);
		shmem2_base[1] =
			SHMEM2_RD(bp, other_shmem2_base_addr);
	}
	bnx2x_acquire_phy_lock(bp);
	bnx2x_common_init_phy(bp, shmem_base, shmem2_base,
			      bp->common.chip_id);
	bnx2x_release_phy_lock(bp);
}

/**
 * bnx2x_init_hw_common - initialize the HW at the COMMON phase.
 *
 * @bp:		driver handle
 */
static int bnx2x_init_hw_common(struct bnx2x *bp)
E
Eliezer Tamir 已提交
5979
{
5980
	u32 val;
E
Eliezer Tamir 已提交
5981

M
Merav Sicron 已提交
5982
	DP(NETIF_MSG_HW, "starting common init  func %d\n", BP_ABS_FUNC(bp));
E
Eliezer Tamir 已提交
5983

5984 5985 5986 5987
	/*
	 * take the UNDI lock to protect undi_unload flow from accessing
	 * registers while we're resetting the chip
	 */
5988
	bnx2x_acquire_hw_lock(bp, HW_LOCK_RESOURCE_RESET);
5989

E
Eilon Greenstein 已提交
5990
	bnx2x_reset_common(bp);
5991
	REG_WR(bp, GRCBASE_MISC + MISC_REGISTERS_RESET_REG_1_SET, 0xffffffff);
E
Eliezer Tamir 已提交
5992

5993 5994 5995 5996 5997 5998 5999
	val = 0xfffc;
	if (CHIP_IS_E3(bp)) {
		val |= MISC_REGISTERS_RESET_REG_2_MSTAT0;
		val |= MISC_REGISTERS_RESET_REG_2_MSTAT1;
	}
	REG_WR(bp, GRCBASE_MISC + MISC_REGISTERS_RESET_REG_2_SET, val);

6000
	bnx2x_release_hw_lock(bp, HW_LOCK_RESOURCE_RESET);
6001

6002
	bnx2x_init_block(bp, BLOCK_MISC, PHASE_COMMON);
E
Eliezer Tamir 已提交
6003

6004 6005
	if (!CHIP_IS_E1x(bp)) {
		u8 abs_func_id;
D
Dmitry Kravkov 已提交
6006 6007 6008 6009 6010 6011 6012 6013

		/**
		 * 4-port mode or 2-port mode we need to turn of master-enable
		 * for everyone, after that, turn it back on for self.
		 * so, we disregard multi-function or not, and always disable
		 * for all functions on the given path, this means 0,2,4,6 for
		 * path 0 and 1,3,5,7 for path 1
		 */
6014 6015 6016
		for (abs_func_id = BP_PATH(bp);
		     abs_func_id < E2_FUNC_MAX*2; abs_func_id += 2) {
			if (abs_func_id == BP_ABS_FUNC(bp)) {
D
Dmitry Kravkov 已提交
6017 6018 6019 6020 6021 6022
				REG_WR(bp,
				    PGLUE_B_REG_INTERNAL_PFID_ENABLE_MASTER,
				    1);
				continue;
			}

6023
			bnx2x_pretend_func(bp, abs_func_id);
D
Dmitry Kravkov 已提交
6024 6025 6026 6027 6028
			/* clear pf enable */
			bnx2x_pf_disable(bp);
			bnx2x_pretend_func(bp, BP_ABS_FUNC(bp));
		}
	}
E
Eliezer Tamir 已提交
6029

6030
	bnx2x_init_block(bp, BLOCK_PXP, PHASE_COMMON);
6031 6032 6033 6034 6035
	if (CHIP_IS_E1(bp)) {
		/* enable HW interrupt from PXP on USDM overflow
		   bit 16 on INT_MASK_0 */
		REG_WR(bp, PXP_REG_PXP_INT_MASK_0, 0);
	}
E
Eliezer Tamir 已提交
6036

6037
	bnx2x_init_block(bp, BLOCK_PXP2, PHASE_COMMON);
6038
	bnx2x_init_pxp(bp);
E
Eliezer Tamir 已提交
6039 6040

#ifdef __BIG_ENDIAN
6041 6042 6043 6044 6045
	REG_WR(bp, PXP2_REG_RQ_QM_ENDIAN_M, 1);
	REG_WR(bp, PXP2_REG_RQ_TM_ENDIAN_M, 1);
	REG_WR(bp, PXP2_REG_RQ_SRC_ENDIAN_M, 1);
	REG_WR(bp, PXP2_REG_RQ_CDU_ENDIAN_M, 1);
	REG_WR(bp, PXP2_REG_RQ_DBG_ENDIAN_M, 1);
E
Eilon Greenstein 已提交
6046 6047
	/* make sure this value is 0 */
	REG_WR(bp, PXP2_REG_RQ_HC_ENDIAN_M, 0);
6048 6049 6050 6051 6052 6053

/*	REG_WR(bp, PXP2_REG_RD_PBF_SWAP_MODE, 1); */
	REG_WR(bp, PXP2_REG_RD_QM_SWAP_MODE, 1);
	REG_WR(bp, PXP2_REG_RD_TM_SWAP_MODE, 1);
	REG_WR(bp, PXP2_REG_RD_SRC_SWAP_MODE, 1);
	REG_WR(bp, PXP2_REG_RD_CDURD_SWAP_MODE, 1);
E
Eliezer Tamir 已提交
6054 6055
#endif

6056 6057
	bnx2x_ilt_init_page_size(bp, INITOP_SET);

6058 6059
	if (CHIP_REV_IS_FPGA(bp) && CHIP_IS_E1H(bp))
		REG_WR(bp, PXP2_REG_PGL_TAGS_LIMIT, 0x1);
E
Eliezer Tamir 已提交
6060

6061 6062 6063 6064 6065 6066 6067 6068 6069 6070 6071 6072 6073
	/* let the HW do it's magic ... */
	msleep(100);
	/* finish PXP init */
	val = REG_RD(bp, PXP2_REG_RQ_CFG_DONE);
	if (val != 1) {
		BNX2X_ERR("PXP2 CFG failed\n");
		return -EBUSY;
	}
	val = REG_RD(bp, PXP2_REG_RD_INIT_DONE);
	if (val != 1) {
		BNX2X_ERR("PXP2 RD_INIT failed\n");
		return -EBUSY;
	}
E
Eliezer Tamir 已提交
6074

D
Dmitry Kravkov 已提交
6075 6076 6077 6078 6079
	/* Timers bug workaround E2 only. We need to set the entire ILT to
	 * have entries with value "0" and valid bit on.
	 * This needs to be done by the first PF that is loaded in a path
	 * (i.e. common phase)
	 */
6080 6081 6082 6083 6084 6085 6086 6087 6088 6089 6090 6091 6092 6093 6094 6095 6096 6097 6098 6099 6100 6101 6102 6103 6104 6105 6106 6107 6108 6109 6110 6111 6112 6113 6114 6115 6116 6117 6118 6119 6120 6121 6122 6123 6124 6125 6126 6127 6128 6129 6130 6131 6132 6133 6134 6135 6136 6137 6138 6139 6140 6141 6142
	if (!CHIP_IS_E1x(bp)) {
/* In E2 there is a bug in the timers block that can cause function 6 / 7
 * (i.e. vnic3) to start even if it is marked as "scan-off".
 * This occurs when a different function (func2,3) is being marked
 * as "scan-off". Real-life scenario for example: if a driver is being
 * load-unloaded while func6,7 are down. This will cause the timer to access
 * the ilt, translate to a logical address and send a request to read/write.
 * Since the ilt for the function that is down is not valid, this will cause
 * a translation error which is unrecoverable.
 * The Workaround is intended to make sure that when this happens nothing fatal
 * will occur. The workaround:
 *	1.  First PF driver which loads on a path will:
 *		a.  After taking the chip out of reset, by using pretend,
 *		    it will write "0" to the following registers of
 *		    the other vnics.
 *		    REG_WR(pdev, PGLUE_B_REG_INTERNAL_PFID_ENABLE_MASTER, 0);
 *		    REG_WR(pdev, CFC_REG_WEAK_ENABLE_PF,0);
 *		    REG_WR(pdev, CFC_REG_STRONG_ENABLE_PF,0);
 *		    And for itself it will write '1' to
 *		    PGLUE_B_REG_INTERNAL_PFID_ENABLE_MASTER to enable
 *		    dmae-operations (writing to pram for example.)
 *		    note: can be done for only function 6,7 but cleaner this
 *			  way.
 *		b.  Write zero+valid to the entire ILT.
 *		c.  Init the first_timers_ilt_entry, last_timers_ilt_entry of
 *		    VNIC3 (of that port). The range allocated will be the
 *		    entire ILT. This is needed to prevent  ILT range error.
 *	2.  Any PF driver load flow:
 *		a.  ILT update with the physical addresses of the allocated
 *		    logical pages.
 *		b.  Wait 20msec. - note that this timeout is needed to make
 *		    sure there are no requests in one of the PXP internal
 *		    queues with "old" ILT addresses.
 *		c.  PF enable in the PGLC.
 *		d.  Clear the was_error of the PF in the PGLC. (could have
 *		    occured while driver was down)
 *		e.  PF enable in the CFC (WEAK + STRONG)
 *		f.  Timers scan enable
 *	3.  PF driver unload flow:
 *		a.  Clear the Timers scan_en.
 *		b.  Polling for scan_on=0 for that PF.
 *		c.  Clear the PF enable bit in the PXP.
 *		d.  Clear the PF enable in the CFC (WEAK + STRONG)
 *		e.  Write zero+valid to all ILT entries (The valid bit must
 *		    stay set)
 *		f.  If this is VNIC 3 of a port then also init
 *		    first_timers_ilt_entry to zero and last_timers_ilt_entry
 *		    to the last enrty in the ILT.
 *
 *	Notes:
 *	Currently the PF error in the PGLC is non recoverable.
 *	In the future the there will be a recovery routine for this error.
 *	Currently attention is masked.
 *	Having an MCP lock on the load/unload process does not guarantee that
 *	there is no Timer disable during Func6/7 enable. This is because the
 *	Timers scan is currently being cleared by the MCP on FLR.
 *	Step 2.d can be done only for PF6/7 and the driver can also check if
 *	there is error before clearing it. But the flow above is simpler and
 *	more general.
 *	All ILT entries are written by zero+valid and not just PF6/7
 *	ILT entries since in the future the ILT entries allocation for
 *	PF-s might be dynamic.
 */
D
Dmitry Kravkov 已提交
6143 6144 6145 6146 6147
		struct ilt_client_info ilt_cli;
		struct bnx2x_ilt ilt;
		memset(&ilt_cli, 0, sizeof(struct ilt_client_info));
		memset(&ilt, 0, sizeof(struct bnx2x_ilt));

6148
		/* initialize dummy TM client */
D
Dmitry Kravkov 已提交
6149 6150 6151 6152 6153 6154 6155
		ilt_cli.start = 0;
		ilt_cli.end = ILT_NUM_PAGE_ENTRIES - 1;
		ilt_cli.client_num = ILT_CLIENT_TM;

		/* Step 1: set zeroes to all ilt page entries with valid bit on
		 * Step 2: set the timers first/last ilt entry to point
		 * to the entire range to prevent ILT range error for 3rd/4th
6156
		 * vnic	(this code assumes existance of the vnic)
D
Dmitry Kravkov 已提交
6157 6158 6159 6160 6161 6162 6163 6164 6165 6166 6167 6168 6169 6170 6171 6172 6173
		 *
		 * both steps performed by call to bnx2x_ilt_client_init_op()
		 * with dummy TM client
		 *
		 * we must use pretend since PXP2_REG_RQ_##blk##_FIRST_ILT
		 * and his brother are split registers
		 */
		bnx2x_pretend_func(bp, (BP_PATH(bp) + 6));
		bnx2x_ilt_client_init_op_ilt(bp, &ilt, &ilt_cli, INITOP_CLEAR);
		bnx2x_pretend_func(bp, BP_ABS_FUNC(bp));

		REG_WR(bp, PXP2_REG_RQ_DRAM_ALIGN, BNX2X_PXP_DRAM_ALIGN);
		REG_WR(bp, PXP2_REG_RQ_DRAM_ALIGN_RD, BNX2X_PXP_DRAM_ALIGN);
		REG_WR(bp, PXP2_REG_RQ_DRAM_ALIGN_SEL, 1);
	}


6174 6175
	REG_WR(bp, PXP2_REG_RQ_DISABLE_INPUTS, 0);
	REG_WR(bp, PXP2_REG_RD_DISABLE_INPUTS, 0);
E
Eliezer Tamir 已提交
6176

6177
	if (!CHIP_IS_E1x(bp)) {
D
Dmitry Kravkov 已提交
6178 6179
		int factor = CHIP_REV_IS_EMUL(bp) ? 1000 :
				(CHIP_REV_IS_FPGA(bp) ? 400 : 0);
6180
		bnx2x_init_block(bp, BLOCK_PGLUE_B, PHASE_COMMON);
D
Dmitry Kravkov 已提交
6181

6182
		bnx2x_init_block(bp, BLOCK_ATC, PHASE_COMMON);
D
Dmitry Kravkov 已提交
6183 6184 6185 6186 6187 6188 6189 6190 6191 6192 6193 6194 6195

		/* let the HW do it's magic ... */
		do {
			msleep(200);
			val = REG_RD(bp, ATC_REG_ATC_INIT_DONE);
		} while (factor-- && (val != 1));

		if (val != 1) {
			BNX2X_ERR("ATC_INIT failed\n");
			return -EBUSY;
		}
	}

6196
	bnx2x_init_block(bp, BLOCK_DMAE, PHASE_COMMON);
E
Eliezer Tamir 已提交
6197

6198 6199
	/* clean the DMAE memory */
	bp->dmae_ready = 1;
6200 6201 6202 6203 6204 6205 6206
	bnx2x_init_fill(bp, TSEM_REG_PRAM, 0, 8, 1);

	bnx2x_init_block(bp, BLOCK_TCM, PHASE_COMMON);

	bnx2x_init_block(bp, BLOCK_UCM, PHASE_COMMON);

	bnx2x_init_block(bp, BLOCK_CCM, PHASE_COMMON);
E
Eliezer Tamir 已提交
6207

6208
	bnx2x_init_block(bp, BLOCK_XCM, PHASE_COMMON);
E
Eliezer Tamir 已提交
6209

6210 6211 6212 6213 6214
	bnx2x_read_dmae(bp, XSEM_REG_PASSIVE_BUFFER, 3);
	bnx2x_read_dmae(bp, CSEM_REG_PASSIVE_BUFFER, 3);
	bnx2x_read_dmae(bp, TSEM_REG_PASSIVE_BUFFER, 3);
	bnx2x_read_dmae(bp, USEM_REG_PASSIVE_BUFFER, 3);

6215
	bnx2x_init_block(bp, BLOCK_QM, PHASE_COMMON);
6216

D
Dmitry Kravkov 已提交
6217

6218 6219 6220
	/* QM queues pointers table */
	bnx2x_qm_init_ptr_table(bp, bp->qm_cid_count, INITOP_SET);

6221 6222 6223
	/* soft reset pulse */
	REG_WR(bp, QM_REG_SOFT_RESET, 1);
	REG_WR(bp, QM_REG_SOFT_RESET, 0);
E
Eliezer Tamir 已提交
6224

6225
#ifdef BCM_CNIC
6226
	bnx2x_init_block(bp, BLOCK_TM, PHASE_COMMON);
E
Eliezer Tamir 已提交
6227 6228
#endif

6229
	bnx2x_init_block(bp, BLOCK_DORQ, PHASE_COMMON);
6230
	REG_WR(bp, DORQ_REG_DPM_CID_OFST, BNX2X_DB_SHIFT);
6231
	if (!CHIP_REV_IS_SLOW(bp))
6232 6233
		/* enable hw interrupt from doorbell Q */
		REG_WR(bp, DORQ_REG_DORQ_INT_MASK, 0);
E
Eliezer Tamir 已提交
6234

6235
	bnx2x_init_block(bp, BLOCK_BRB1, PHASE_COMMON);
D
Dmitry Kravkov 已提交
6236

6237
	bnx2x_init_block(bp, BLOCK_PRS, PHASE_COMMON);
6238
	REG_WR(bp, PRS_REG_A_PRSU_20, 0xf);
6239

D
Dmitry Kravkov 已提交
6240
	if (!CHIP_IS_E1(bp))
6241
		REG_WR(bp, PRS_REG_E1HOV_MODE, bp->path_has_ovlan);
D
Dmitry Kravkov 已提交
6242

6243 6244 6245 6246 6247 6248
	if (!CHIP_IS_E1x(bp) && !CHIP_IS_E3B0(bp))
		/* Bit-map indicating which L2 hdrs may appear
		 * after the basic Ethernet header
		 */
		REG_WR(bp, PRS_REG_HDRS_AFTER_BASIC,
		       bp->path_has_ovlan ? 7 : 6);
E
Eliezer Tamir 已提交
6249

6250 6251 6252 6253
	bnx2x_init_block(bp, BLOCK_TSDM, PHASE_COMMON);
	bnx2x_init_block(bp, BLOCK_CSDM, PHASE_COMMON);
	bnx2x_init_block(bp, BLOCK_USDM, PHASE_COMMON);
	bnx2x_init_block(bp, BLOCK_XSDM, PHASE_COMMON);
E
Eliezer Tamir 已提交
6254

6255 6256 6257 6258 6259 6260 6261 6262
	if (!CHIP_IS_E1x(bp)) {
		/* reset VFC memories */
		REG_WR(bp, TSEM_REG_FAST_MEMORY + VFC_REG_MEMORIES_RST,
			   VFC_MEMORIES_RST_REG_CAM_RST |
			   VFC_MEMORIES_RST_REG_RAM_RST);
		REG_WR(bp, XSEM_REG_FAST_MEMORY + VFC_REG_MEMORIES_RST,
			   VFC_MEMORIES_RST_REG_CAM_RST |
			   VFC_MEMORIES_RST_REG_RAM_RST);
E
Eliezer Tamir 已提交
6263

6264 6265
		msleep(20);
	}
E
Eliezer Tamir 已提交
6266

6267 6268 6269 6270
	bnx2x_init_block(bp, BLOCK_TSEM, PHASE_COMMON);
	bnx2x_init_block(bp, BLOCK_USEM, PHASE_COMMON);
	bnx2x_init_block(bp, BLOCK_CSEM, PHASE_COMMON);
	bnx2x_init_block(bp, BLOCK_XSEM, PHASE_COMMON);
D
Dmitry Kravkov 已提交
6271

6272 6273 6274 6275 6276
	/* sync semi rtc */
	REG_WR(bp, GRCBASE_MISC + MISC_REGISTERS_RESET_REG_1_CLEAR,
	       0x80000000);
	REG_WR(bp, GRCBASE_MISC + MISC_REGISTERS_RESET_REG_1_SET,
	       0x80000000);
E
Eliezer Tamir 已提交
6277

6278 6279 6280
	bnx2x_init_block(bp, BLOCK_UPB, PHASE_COMMON);
	bnx2x_init_block(bp, BLOCK_XPB, PHASE_COMMON);
	bnx2x_init_block(bp, BLOCK_PBF, PHASE_COMMON);
E
Eliezer Tamir 已提交
6281

6282 6283 6284
	if (!CHIP_IS_E1x(bp))
		REG_WR(bp, PBF_REG_HDRS_AFTER_BASIC,
		       bp->path_has_ovlan ? 7 : 6);
D
Dmitry Kravkov 已提交
6285

6286
	REG_WR(bp, SRC_REG_SOFT_RST, 1);
D
Dmitry Kravkov 已提交
6287

6288 6289
	bnx2x_init_block(bp, BLOCK_SRC, PHASE_COMMON);

6290 6291 6292 6293 6294 6295 6296 6297 6298 6299 6300 6301
#ifdef BCM_CNIC
	REG_WR(bp, SRC_REG_KEYSEARCH_0, 0x63285672);
	REG_WR(bp, SRC_REG_KEYSEARCH_1, 0x24b8f2cc);
	REG_WR(bp, SRC_REG_KEYSEARCH_2, 0x223aef9b);
	REG_WR(bp, SRC_REG_KEYSEARCH_3, 0x26001e3a);
	REG_WR(bp, SRC_REG_KEYSEARCH_4, 0x7ae91116);
	REG_WR(bp, SRC_REG_KEYSEARCH_5, 0x5ce5230b);
	REG_WR(bp, SRC_REG_KEYSEARCH_6, 0x298d8adf);
	REG_WR(bp, SRC_REG_KEYSEARCH_7, 0x6eb0ff09);
	REG_WR(bp, SRC_REG_KEYSEARCH_8, 0x1830f82f);
	REG_WR(bp, SRC_REG_KEYSEARCH_9, 0x01e46be7);
#endif
6302
	REG_WR(bp, SRC_REG_SOFT_RST, 0);
E
Eliezer Tamir 已提交
6303

6304 6305
	if (sizeof(union cdu_context) != 1024)
		/* we currently assume that a context is 1024 bytes */
M
Merav Sicron 已提交
6306 6307 6308
		dev_alert(&bp->pdev->dev,
			  "please adjust the size of cdu_context(%ld)\n",
			  (long)sizeof(union cdu_context));
E
Eliezer Tamir 已提交
6309

6310
	bnx2x_init_block(bp, BLOCK_CDU, PHASE_COMMON);
6311 6312
	val = (4 << 24) + (0 << 12) + 1024;
	REG_WR(bp, CDU_REG_CDU_GLOBAL_PARAMS, val);
E
Eliezer Tamir 已提交
6313

6314
	bnx2x_init_block(bp, BLOCK_CFC, PHASE_COMMON);
6315
	REG_WR(bp, CFC_REG_INIT_REG, 0x7FF);
E
Eilon Greenstein 已提交
6316 6317 6318 6319 6320
	/* enable context validation interrupt from CFC */
	REG_WR(bp, CFC_REG_CFC_INT_MASK, 0);

	/* set the thresholds to prevent CFC/CDU race */
	REG_WR(bp, CFC_REG_DEBUG0, 0x20020000);
E
Eliezer Tamir 已提交
6321

6322
	bnx2x_init_block(bp, BLOCK_HC, PHASE_COMMON);
D
Dmitry Kravkov 已提交
6323

6324
	if (!CHIP_IS_E1x(bp) && BP_NOMCP(bp))
D
Dmitry Kravkov 已提交
6325 6326
		REG_WR(bp, IGU_REG_RESET_MEMORIES, 0x36);

6327 6328
	bnx2x_init_block(bp, BLOCK_IGU, PHASE_COMMON);
	bnx2x_init_block(bp, BLOCK_MISC_AEU, PHASE_COMMON);
E
Eliezer Tamir 已提交
6329

6330 6331 6332
	/* Reset PCIE errors for debug */
	REG_WR(bp, 0x2814, 0xffffffff);
	REG_WR(bp, 0x3820, 0xffffffff);
E
Eliezer Tamir 已提交
6333

6334
	if (!CHIP_IS_E1x(bp)) {
D
Dmitry Kravkov 已提交
6335 6336 6337 6338 6339 6340 6341 6342 6343 6344 6345 6346 6347
		REG_WR(bp, PCICFG_OFFSET + PXPCS_TL_CONTROL_5,
			   (PXPCS_TL_CONTROL_5_ERR_UNSPPORT1 |
				PXPCS_TL_CONTROL_5_ERR_UNSPPORT));
		REG_WR(bp, PCICFG_OFFSET + PXPCS_TL_FUNC345_STAT,
			   (PXPCS_TL_FUNC345_STAT_ERR_UNSPPORT4 |
				PXPCS_TL_FUNC345_STAT_ERR_UNSPPORT3 |
				PXPCS_TL_FUNC345_STAT_ERR_UNSPPORT2));
		REG_WR(bp, PCICFG_OFFSET + PXPCS_TL_FUNC678_STAT,
			   (PXPCS_TL_FUNC678_STAT_ERR_UNSPPORT7 |
				PXPCS_TL_FUNC678_STAT_ERR_UNSPPORT6 |
				PXPCS_TL_FUNC678_STAT_ERR_UNSPPORT5));
	}

6348
	bnx2x_init_block(bp, BLOCK_NIG, PHASE_COMMON);
D
Dmitry Kravkov 已提交
6349
	if (!CHIP_IS_E1(bp)) {
6350 6351 6352
		/* in E3 this done in per-port section */
		if (!CHIP_IS_E3(bp))
			REG_WR(bp, NIG_REG_LLH_MF_MODE, IS_MF(bp));
D
Dmitry Kravkov 已提交
6353
	}
6354 6355 6356
	if (CHIP_IS_E1H(bp))
		/* not applicable for E2 (and above ...) */
		REG_WR(bp, NIG_REG_LLH_E1HOV_MODE, IS_MF_SD(bp));
6357 6358 6359 6360 6361 6362 6363 6364 6365 6366 6367 6368 6369 6370 6371 6372 6373 6374 6375 6376 6377

	if (CHIP_REV_IS_SLOW(bp))
		msleep(200);

	/* finish CFC init */
	val = reg_poll(bp, CFC_REG_LL_INIT_DONE, 1, 100, 10);
	if (val != 1) {
		BNX2X_ERR("CFC LL_INIT failed\n");
		return -EBUSY;
	}
	val = reg_poll(bp, CFC_REG_AC_INIT_DONE, 1, 100, 10);
	if (val != 1) {
		BNX2X_ERR("CFC AC_INIT failed\n");
		return -EBUSY;
	}
	val = reg_poll(bp, CFC_REG_CAM_INIT_DONE, 1, 100, 10);
	if (val != 1) {
		BNX2X_ERR("CFC CAM_INIT failed\n");
		return -EBUSY;
	}
	REG_WR(bp, CFC_REG_DEBUG0, 0);
E
Eliezer Tamir 已提交
6378

D
Dmitry Kravkov 已提交
6379 6380 6381 6382 6383
	if (CHIP_IS_E1(bp)) {
		/* read NIG statistic
		   to see if this is our first up since powerup */
		bnx2x_read_dmae(bp, NIG_REG_STAT2_BRB_OCTET, 2);
		val = *bnx2x_sp(bp, wb_data[0]);
6384

D
Dmitry Kravkov 已提交
6385 6386 6387 6388 6389
		/* do internal memory self test */
		if ((val == 0) && bnx2x_int_mem_test(bp)) {
			BNX2X_ERR("internal mem self test failed\n");
			return -EBUSY;
		}
6390 6391
	}

E
Eilon Greenstein 已提交
6392 6393
	bnx2x_setup_fan_failure_detection(bp);

6394 6395
	/* clear PXP2 attentions */
	REG_RD(bp, PXP2_REG_PXP2_INT_STS_CLR_0);
E
Eliezer Tamir 已提交
6396

6397
	bnx2x_enable_blocks_attention(bp);
6398
	bnx2x_enable_blocks_parity(bp);
E
Eliezer Tamir 已提交
6399

Y
Yaniv Rosner 已提交
6400
	if (!BP_NOMCP(bp)) {
6401 6402
		if (CHIP_IS_E1x(bp))
			bnx2x__common_init_phy(bp);
Y
Yaniv Rosner 已提交
6403 6404 6405
	} else
		BNX2X_ERR("Bootcode is missing - can not initialize link\n");

6406 6407
	return 0;
}
E
Eliezer Tamir 已提交
6408

6409 6410 6411 6412 6413 6414 6415 6416 6417 6418 6419 6420 6421 6422 6423 6424 6425 6426 6427
/**
 * bnx2x_init_hw_common_chip - init HW at the COMMON_CHIP phase.
 *
 * @bp:		driver handle
 */
static int bnx2x_init_hw_common_chip(struct bnx2x *bp)
{
	int rc = bnx2x_init_hw_common(bp);

	if (rc)
		return rc;

	/* In E2 2-PORT mode, same ext phy is used for the two paths */
	if (!BP_NOMCP(bp))
		bnx2x__common_init_phy(bp);

	return 0;
}

6428
static int bnx2x_init_hw_port(struct bnx2x *bp)
6429 6430
{
	int port = BP_PORT(bp);
6431
	int init_phase = port ? PHASE_PORT1 : PHASE_PORT0;
6432
	u32 low, high;
6433
	u32 val;
E
Eliezer Tamir 已提交
6434

6435 6436
	bnx2x__link_reset(bp);

M
Merav Sicron 已提交
6437
	DP(NETIF_MSG_HW, "starting port init  port %d\n", port);
6438 6439

	REG_WR(bp, NIG_REG_MASK_INTERRUPT_PORT0 + port*4, 0);
E
Eliezer Tamir 已提交
6440

6441 6442 6443
	bnx2x_init_block(bp, BLOCK_MISC, init_phase);
	bnx2x_init_block(bp, BLOCK_PXP, init_phase);
	bnx2x_init_block(bp, BLOCK_PXP2, init_phase);
E
Eilon Greenstein 已提交
6444

D
Dmitry Kravkov 已提交
6445 6446 6447 6448 6449
	/* Timers bug workaround: disables the pf_master bit in pglue at
	 * common phase, we need to enable it here before any dmae access are
	 * attempted. Therefore we manually added the enable-master to the
	 * port phase (it also happens in the function phase)
	 */
6450
	if (!CHIP_IS_E1x(bp))
D
Dmitry Kravkov 已提交
6451 6452
		REG_WR(bp, PGLUE_B_REG_INTERNAL_PFID_ENABLE_MASTER, 1);

6453 6454 6455 6456 6457 6458 6459 6460 6461
	bnx2x_init_block(bp, BLOCK_ATC, init_phase);
	bnx2x_init_block(bp, BLOCK_DMAE, init_phase);
	bnx2x_init_block(bp, BLOCK_PGLUE_B, init_phase);
	bnx2x_init_block(bp, BLOCK_QM, init_phase);

	bnx2x_init_block(bp, BLOCK_TCM, init_phase);
	bnx2x_init_block(bp, BLOCK_UCM, init_phase);
	bnx2x_init_block(bp, BLOCK_CCM, init_phase);
	bnx2x_init_block(bp, BLOCK_XCM, init_phase);
E
Eliezer Tamir 已提交
6462

6463 6464
	/* QM cid (connection) count */
	bnx2x_qm_init_cid_count(bp, bp->qm_cid_count, INITOP_SET);
E
Eliezer Tamir 已提交
6465

6466
#ifdef BCM_CNIC
6467
	bnx2x_init_block(bp, BLOCK_TM, init_phase);
6468 6469
	REG_WR(bp, TM_REG_LIN0_SCAN_TIME + port*4, 20);
	REG_WR(bp, TM_REG_LIN0_MAX_ACTIVE_CID + port*4, 31);
E
Eliezer Tamir 已提交
6470
#endif
V
Vladislav Zolotarov 已提交
6471

6472
	bnx2x_init_block(bp, BLOCK_DORQ, init_phase);
D
Dmitry Kravkov 已提交
6473 6474

	if (CHIP_IS_E1(bp) || CHIP_IS_E1H(bp)) {
6475 6476 6477 6478 6479 6480 6481 6482 6483 6484 6485 6486 6487 6488 6489 6490
		bnx2x_init_block(bp, BLOCK_BRB1, init_phase);

		if (IS_MF(bp))
			low = ((bp->flags & ONE_PORT_FLAG) ? 160 : 246);
		else if (bp->dev->mtu > 4096) {
			if (bp->flags & ONE_PORT_FLAG)
				low = 160;
			else {
				val = bp->dev->mtu;
				/* (24*1024 + val*4)/256 */
				low = 96 + (val/64) +
						((val % 64) ? 1 : 0);
			}
		} else
			low = ((bp->flags & ONE_PORT_FLAG) ? 80 : 160);
		high = low + 56;	/* 14*1024/256 */
D
Dmitry Kravkov 已提交
6491 6492
		REG_WR(bp, BRB1_REG_PAUSE_LOW_THRESHOLD_0 + port*4, low);
		REG_WR(bp, BRB1_REG_PAUSE_HIGH_THRESHOLD_0 + port*4, high);
6493 6494
	}

6495 6496 6497 6498
	if (CHIP_MODE_IS_4_PORT(bp))
		REG_WR(bp, (BP_PORT(bp) ?
			    BRB1_REG_MAC_GUARANTIED_1 :
			    BRB1_REG_MAC_GUARANTIED_0), 40);
6499

E
Eilon Greenstein 已提交
6500

6501 6502 6503 6504 6505 6506 6507 6508 6509 6510
	bnx2x_init_block(bp, BLOCK_PRS, init_phase);
	if (CHIP_IS_E3B0(bp))
		/* Ovlan exists only if we are in multi-function +
		 * switch-dependent mode, in switch-independent there
		 * is no ovlan headers
		 */
		REG_WR(bp, BP_PORT(bp) ?
		       PRS_REG_HDRS_AFTER_BASIC_PORT_1 :
		       PRS_REG_HDRS_AFTER_BASIC_PORT_0,
		       (bp->path_has_ovlan ? 7 : 6));
E
Eilon Greenstein 已提交
6511

6512 6513 6514 6515
	bnx2x_init_block(bp, BLOCK_TSDM, init_phase);
	bnx2x_init_block(bp, BLOCK_CSDM, init_phase);
	bnx2x_init_block(bp, BLOCK_USDM, init_phase);
	bnx2x_init_block(bp, BLOCK_XSDM, init_phase);
E
Eilon Greenstein 已提交
6516

6517 6518 6519 6520
	bnx2x_init_block(bp, BLOCK_TSEM, init_phase);
	bnx2x_init_block(bp, BLOCK_USEM, init_phase);
	bnx2x_init_block(bp, BLOCK_CSEM, init_phase);
	bnx2x_init_block(bp, BLOCK_XSEM, init_phase);
6521

6522 6523
	bnx2x_init_block(bp, BLOCK_UPB, init_phase);
	bnx2x_init_block(bp, BLOCK_XPB, init_phase);
E
Eliezer Tamir 已提交
6524

6525 6526 6527
	bnx2x_init_block(bp, BLOCK_PBF, init_phase);

	if (CHIP_IS_E1x(bp)) {
D
Dmitry Kravkov 已提交
6528 6529
		/* configure PBF to work without PAUSE mtu 9000 */
		REG_WR(bp, PBF_REG_P0_PAUSE_ENABLE + port*4, 0);
E
Eliezer Tamir 已提交
6530

D
Dmitry Kravkov 已提交
6531 6532 6533 6534
		/* update threshold */
		REG_WR(bp, PBF_REG_P0_ARB_THRSH + port*4, (9040/16));
		/* update init credit */
		REG_WR(bp, PBF_REG_P0_INIT_CRD + port*4, (9040/16) + 553 - 22);
E
Eliezer Tamir 已提交
6535

D
Dmitry Kravkov 已提交
6536 6537 6538 6539 6540
		/* probe changes */
		REG_WR(bp, PBF_REG_INIT_P0 + port*4, 1);
		udelay(50);
		REG_WR(bp, PBF_REG_INIT_P0 + port*4, 0);
	}
E
Eliezer Tamir 已提交
6541

6542
#ifdef BCM_CNIC
6543
	bnx2x_init_block(bp, BLOCK_SRC, init_phase);
E
Eliezer Tamir 已提交
6544
#endif
6545 6546
	bnx2x_init_block(bp, BLOCK_CDU, init_phase);
	bnx2x_init_block(bp, BLOCK_CFC, init_phase);
6547 6548 6549 6550 6551

	if (CHIP_IS_E1(bp)) {
		REG_WR(bp, HC_REG_LEADING_EDGE_0 + port*8, 0);
		REG_WR(bp, HC_REG_TRAILING_EDGE_0 + port*8, 0);
	}
6552
	bnx2x_init_block(bp, BLOCK_HC, init_phase);
6553

6554
	bnx2x_init_block(bp, BLOCK_IGU, init_phase);
D
Dmitry Kravkov 已提交
6555

6556
	bnx2x_init_block(bp, BLOCK_MISC_AEU, init_phase);
6557 6558 6559 6560
	/* init aeu_mask_attn_func_0/1:
	 *  - SF mode: bits 3-7 are masked. only bits 0-2 are in use
	 *  - MF mode: bit 3 is masked. bits 0-2 are in use as in SF
	 *             bits 4-7 are used for "per vn group attention" */
V
Vladislav Zolotarov 已提交
6561 6562 6563 6564
	val = IS_MF(bp) ? 0xF7 : 0x7;
	/* Enable DCBX attention for all but E1 */
	val |= CHIP_IS_E1(bp) ? 0 : 0x10;
	REG_WR(bp, MISC_REG_AEU_MASK_ATTN_FUNC_0 + port*4, val);
6565

6566 6567 6568 6569 6570 6571 6572 6573 6574 6575 6576 6577 6578 6579 6580 6581 6582 6583
	bnx2x_init_block(bp, BLOCK_NIG, init_phase);

	if (!CHIP_IS_E1x(bp)) {
		/* Bit-map indicating which L2 hdrs may appear after the
		 * basic Ethernet header
		 */
		REG_WR(bp, BP_PORT(bp) ?
			   NIG_REG_P1_HDRS_AFTER_BASIC :
			   NIG_REG_P0_HDRS_AFTER_BASIC,
			   IS_MF_SD(bp) ? 7 : 6);

		if (CHIP_IS_E3(bp))
			REG_WR(bp, BP_PORT(bp) ?
				   NIG_REG_LLH1_MF_MODE :
				   NIG_REG_LLH_MF_MODE, IS_MF(bp));
	}
	if (!CHIP_IS_E3(bp))
		REG_WR(bp, NIG_REG_XGXS_SERDES0_MODE_SEL + port*4, 1);
6584

D
Dmitry Kravkov 已提交
6585
	if (!CHIP_IS_E1(bp)) {
D
Dmitry Kravkov 已提交
6586
		/* 0x2 disable mf_ov, 0x1 enable */
6587
		REG_WR(bp, NIG_REG_LLH0_BRB1_DRV_MASK_MF + port*4,
6588
		       (IS_MF_SD(bp) ? 0x1 : 0x2));
6589

6590
		if (!CHIP_IS_E1x(bp)) {
D
Dmitry Kravkov 已提交
6591 6592 6593 6594 6595 6596 6597 6598 6599 6600 6601 6602 6603
			val = 0;
			switch (bp->mf_mode) {
			case MULTI_FUNCTION_SD:
				val = 1;
				break;
			case MULTI_FUNCTION_SI:
				val = 2;
				break;
			}

			REG_WR(bp, (BP_PORT(bp) ? NIG_REG_LLH1_CLS_TYPE :
						  NIG_REG_LLH0_CLS_TYPE), val);
		}
6604 6605 6606 6607 6608
		{
			REG_WR(bp, NIG_REG_LLFC_ENABLE_0 + port*4, 0);
			REG_WR(bp, NIG_REG_LLFC_OUT_EN_0 + port*4, 0);
			REG_WR(bp, NIG_REG_PAUSE_ENABLE_0 + port*4, 1);
		}
6609 6610
	}

6611 6612 6613 6614

	/* If SPIO5 is set to generate interrupts, enable it for this port */
	val = REG_RD(bp, MISC_REG_SPIO_EVENT_EN);
	if (val & (1 << MISC_REGISTERS_SPIO_5)) {
E
Eilon Greenstein 已提交
6615 6616 6617
		u32 reg_addr = (port ? MISC_REG_AEU_ENABLE1_FUNC_1_OUT_0 :
				       MISC_REG_AEU_ENABLE1_FUNC_0_OUT_0);
		val = REG_RD(bp, reg_addr);
E
Eliezer Tamir 已提交
6618
		val |= AEU_INPUTS_ATTN_BITS_SPIO5;
E
Eilon Greenstein 已提交
6619
		REG_WR(bp, reg_addr, val);
E
Eliezer Tamir 已提交
6620
	}
E
Eliezer Tamir 已提交
6621

6622 6623 6624 6625 6626 6627 6628
	return 0;
}

static void bnx2x_ilt_wr(struct bnx2x *bp, u32 index, dma_addr_t addr)
{
	int reg;

D
Dmitry Kravkov 已提交
6629
	if (CHIP_IS_E1(bp))
6630
		reg = PXP2_REG_RQ_ONCHIP_AT + index*8;
D
Dmitry Kravkov 已提交
6631 6632
	else
		reg = PXP2_REG_RQ_ONCHIP_AT_B0 + index*8;
6633 6634 6635 6636

	bnx2x_wb_wr(bp, reg, ONCHIP_ADDR1(addr), ONCHIP_ADDR2(addr));
}

D
Dmitry Kravkov 已提交
6637 6638
static inline void bnx2x_igu_clear_sb(struct bnx2x *bp, u8 idu_sb_id)
{
6639
	bnx2x_igu_clear_sb_gen(bp, BP_FUNC(bp), idu_sb_id, true /*PF*/);
D
Dmitry Kravkov 已提交
6640 6641 6642 6643 6644 6645 6646 6647 6648
}

static inline void bnx2x_clear_func_ilt(struct bnx2x *bp, u32 func)
{
	u32 i, base = FUNC_ILT_BASE(func);
	for (i = base; i < base + ILT_PER_FUNC; i++)
		bnx2x_ilt_wr(bp, i, 0);
}

6649
static int bnx2x_init_hw_func(struct bnx2x *bp)
6650 6651 6652
{
	int port = BP_PORT(bp);
	int func = BP_FUNC(bp);
6653
	int init_phase = PHASE_PF0 + func;
6654 6655
	struct bnx2x_ilt *ilt = BP_ILT(bp);
	u16 cdu_ilt_start;
E
Eilon Greenstein 已提交
6656
	u32 addr, val;
6657
	u32 main_mem_base, main_mem_size, main_mem_prty_clr;
6658
	int i, main_mem_width, rc;
6659

M
Merav Sicron 已提交
6660
	DP(NETIF_MSG_HW, "starting func init  func %d\n", func);
6661

6662
	/* FLR cleanup - hmmm */
6663 6664 6665 6666 6667
	if (!CHIP_IS_E1x(bp)) {
		rc = bnx2x_pf_flr_clnup(bp);
		if (rc)
			return rc;
	}
6668

E
Eilon Greenstein 已提交
6669
	/* set MSI reconfigure capability */
D
Dmitry Kravkov 已提交
6670 6671 6672 6673 6674 6675
	if (bp->common.int_block == INT_BLOCK_HC) {
		addr = (port ? HC_REG_CONFIG_1 : HC_REG_CONFIG_0);
		val = REG_RD(bp, addr);
		val |= HC_CONFIG_0_REG_MSI_ATTN_EN_0;
		REG_WR(bp, addr, val);
	}
E
Eilon Greenstein 已提交
6676

6677 6678 6679
	bnx2x_init_block(bp, BLOCK_PXP, init_phase);
	bnx2x_init_block(bp, BLOCK_PXP2, init_phase);

6680 6681
	ilt = BP_ILT(bp);
	cdu_ilt_start = ilt->clients[ILT_CLIENT_CDU].start;
6682

6683 6684 6685 6686 6687 6688 6689
	for (i = 0; i < L2_ILT_LINES(bp); i++) {
		ilt->lines[cdu_ilt_start + i].page =
			bp->context.vcxt + (ILT_PAGE_CIDS * i);
		ilt->lines[cdu_ilt_start + i].page_mapping =
			bp->context.cxt_mapping + (CDU_ILT_PAGE_SZ * i);
		/* cdu ilt pages are allocated manually so there's no need to
		set the size */
6690
	}
6691
	bnx2x_ilt_init_op(bp, INITOP_SET);
D
Dmitry Kravkov 已提交
6692

6693 6694
#ifdef BCM_CNIC
	bnx2x_src_init_t2(bp, bp->t2, bp->t2_mapping, SRC_CONN_NUM);
6695

6696 6697 6698
	/* T1 hash bits value determines the T1 number of entries */
	REG_WR(bp, SRC_REG_NUMBER_HASH_BITS0 + port*4, SRC_HASH_BITS);
#endif
6699

6700 6701 6702 6703
#ifndef BCM_CNIC
	/* set NIC mode */
	REG_WR(bp, PRS_REG_NIC_MODE, 1);
#endif  /* BCM_CNIC */
6704

6705
	if (!CHIP_IS_E1x(bp)) {
D
Dmitry Kravkov 已提交
6706 6707 6708 6709 6710 6711 6712 6713 6714 6715 6716 6717 6718 6719 6720 6721 6722 6723 6724 6725 6726 6727 6728 6729
		u32 pf_conf = IGU_PF_CONF_FUNC_EN;

		/* Turn on a single ISR mode in IGU if driver is going to use
		 * INT#x or MSI
		 */
		if (!(bp->flags & USING_MSIX_FLAG))
			pf_conf |= IGU_PF_CONF_SINGLE_ISR_EN;
		/*
		 * Timers workaround bug: function init part.
		 * Need to wait 20msec after initializing ILT,
		 * needed to make sure there are no requests in
		 * one of the PXP internal queues with "old" ILT addresses
		 */
		msleep(20);
		/*
		 * Master enable - Due to WB DMAE writes performed before this
		 * register is re-initialized as part of the regular function
		 * init
		 */
		REG_WR(bp, PGLUE_B_REG_INTERNAL_PFID_ENABLE_MASTER, 1);
		/* Enable the function in IGU */
		REG_WR(bp, IGU_REG_PF_CONFIGURATION, pf_conf);
	}

6730
	bp->dmae_ready = 1;
6731

6732
	bnx2x_init_block(bp, BLOCK_PGLUE_B, init_phase);
6733

6734
	if (!CHIP_IS_E1x(bp))
D
Dmitry Kravkov 已提交
6735 6736
		REG_WR(bp, PGLUE_B_REG_WAS_ERROR_PF_7_0_CLR, func);

6737 6738 6739 6740 6741 6742 6743 6744 6745 6746 6747 6748 6749 6750 6751
	bnx2x_init_block(bp, BLOCK_ATC, init_phase);
	bnx2x_init_block(bp, BLOCK_DMAE, init_phase);
	bnx2x_init_block(bp, BLOCK_NIG, init_phase);
	bnx2x_init_block(bp, BLOCK_SRC, init_phase);
	bnx2x_init_block(bp, BLOCK_MISC, init_phase);
	bnx2x_init_block(bp, BLOCK_TCM, init_phase);
	bnx2x_init_block(bp, BLOCK_UCM, init_phase);
	bnx2x_init_block(bp, BLOCK_CCM, init_phase);
	bnx2x_init_block(bp, BLOCK_XCM, init_phase);
	bnx2x_init_block(bp, BLOCK_TSEM, init_phase);
	bnx2x_init_block(bp, BLOCK_USEM, init_phase);
	bnx2x_init_block(bp, BLOCK_CSEM, init_phase);
	bnx2x_init_block(bp, BLOCK_XSEM, init_phase);

	if (!CHIP_IS_E1x(bp))
D
Dmitry Kravkov 已提交
6752 6753
		REG_WR(bp, QM_REG_PF_EN, 1);

6754 6755 6756 6757 6758 6759 6760 6761 6762 6763 6764 6765 6766 6767 6768 6769 6770 6771 6772 6773
	if (!CHIP_IS_E1x(bp)) {
		REG_WR(bp, TSEM_REG_VFPF_ERR_NUM, BNX2X_MAX_NUM_OF_VFS + func);
		REG_WR(bp, USEM_REG_VFPF_ERR_NUM, BNX2X_MAX_NUM_OF_VFS + func);
		REG_WR(bp, CSEM_REG_VFPF_ERR_NUM, BNX2X_MAX_NUM_OF_VFS + func);
		REG_WR(bp, XSEM_REG_VFPF_ERR_NUM, BNX2X_MAX_NUM_OF_VFS + func);
	}
	bnx2x_init_block(bp, BLOCK_QM, init_phase);

	bnx2x_init_block(bp, BLOCK_TM, init_phase);
	bnx2x_init_block(bp, BLOCK_DORQ, init_phase);
	bnx2x_init_block(bp, BLOCK_BRB1, init_phase);
	bnx2x_init_block(bp, BLOCK_PRS, init_phase);
	bnx2x_init_block(bp, BLOCK_TSDM, init_phase);
	bnx2x_init_block(bp, BLOCK_CSDM, init_phase);
	bnx2x_init_block(bp, BLOCK_USDM, init_phase);
	bnx2x_init_block(bp, BLOCK_XSDM, init_phase);
	bnx2x_init_block(bp, BLOCK_UPB, init_phase);
	bnx2x_init_block(bp, BLOCK_XPB, init_phase);
	bnx2x_init_block(bp, BLOCK_PBF, init_phase);
	if (!CHIP_IS_E1x(bp))
D
Dmitry Kravkov 已提交
6774 6775
		REG_WR(bp, PBF_REG_DISABLE_PF, 0);

6776
	bnx2x_init_block(bp, BLOCK_CDU, init_phase);
6777

6778
	bnx2x_init_block(bp, BLOCK_CFC, init_phase);
6779

6780
	if (!CHIP_IS_E1x(bp))
D
Dmitry Kravkov 已提交
6781 6782
		REG_WR(bp, CFC_REG_WEAK_ENABLE_PF, 1);

D
Dmitry Kravkov 已提交
6783
	if (IS_MF(bp)) {
6784
		REG_WR(bp, NIG_REG_LLH0_FUNC_EN + port*8, 1);
D
Dmitry Kravkov 已提交
6785
		REG_WR(bp, NIG_REG_LLH0_FUNC_VLAN_ID + port*8, bp->mf_ov);
6786 6787
	}

6788
	bnx2x_init_block(bp, BLOCK_MISC_AEU, init_phase);
6789

6790
	/* HC init per function */
D
Dmitry Kravkov 已提交
6791 6792 6793 6794 6795 6796 6797
	if (bp->common.int_block == INT_BLOCK_HC) {
		if (CHIP_IS_E1H(bp)) {
			REG_WR(bp, MISC_REG_AEU_GENERAL_ATTN_12 + func*4, 0);

			REG_WR(bp, HC_REG_LEADING_EDGE_0 + port*8, 0);
			REG_WR(bp, HC_REG_TRAILING_EDGE_0 + port*8, 0);
		}
6798
		bnx2x_init_block(bp, BLOCK_HC, init_phase);
D
Dmitry Kravkov 已提交
6799 6800 6801 6802

	} else {
		int num_segs, sb_idx, prod_offset;

6803 6804
		REG_WR(bp, MISC_REG_AEU_GENERAL_ATTN_12 + func*4, 0);

6805
		if (!CHIP_IS_E1x(bp)) {
D
Dmitry Kravkov 已提交
6806 6807 6808 6809
			REG_WR(bp, IGU_REG_LEADING_EDGE_LATCH, 0);
			REG_WR(bp, IGU_REG_TRAILING_EDGE_LATCH, 0);
		}

6810
		bnx2x_init_block(bp, BLOCK_IGU, init_phase);
D
Dmitry Kravkov 已提交
6811

6812
		if (!CHIP_IS_E1x(bp)) {
D
Dmitry Kravkov 已提交
6813 6814 6815 6816 6817 6818 6819 6820 6821 6822 6823 6824 6825 6826 6827 6828 6829 6830 6831 6832 6833 6834 6835 6836 6837 6838 6839 6840 6841 6842 6843 6844 6845 6846 6847 6848 6849 6850 6851 6852 6853 6854 6855 6856 6857 6858 6859
			int dsb_idx = 0;
			/**
			 * Producer memory:
			 * E2 mode: address 0-135 match to the mapping memory;
			 * 136 - PF0 default prod; 137 - PF1 default prod;
			 * 138 - PF2 default prod; 139 - PF3 default prod;
			 * 140 - PF0 attn prod;    141 - PF1 attn prod;
			 * 142 - PF2 attn prod;    143 - PF3 attn prod;
			 * 144-147 reserved.
			 *
			 * E1.5 mode - In backward compatible mode;
			 * for non default SB; each even line in the memory
			 * holds the U producer and each odd line hold
			 * the C producer. The first 128 producers are for
			 * NDSB (PF0 - 0-31; PF1 - 32-63 and so on). The last 20
			 * producers are for the DSB for each PF.
			 * Each PF has five segments: (the order inside each
			 * segment is PF0; PF1; PF2; PF3) - 128-131 U prods;
			 * 132-135 C prods; 136-139 X prods; 140-143 T prods;
			 * 144-147 attn prods;
			 */
			/* non-default-status-blocks */
			num_segs = CHIP_INT_MODE_IS_BC(bp) ?
				IGU_BC_NDSB_NUM_SEGS : IGU_NORM_NDSB_NUM_SEGS;
			for (sb_idx = 0; sb_idx < bp->igu_sb_cnt; sb_idx++) {
				prod_offset = (bp->igu_base_sb + sb_idx) *
					num_segs;

				for (i = 0; i < num_segs; i++) {
					addr = IGU_REG_PROD_CONS_MEMORY +
							(prod_offset + i) * 4;
					REG_WR(bp, addr, 0);
				}
				/* send consumer update with value 0 */
				bnx2x_ack_sb(bp, bp->igu_base_sb + sb_idx,
					     USTORM_ID, 0, IGU_INT_NOP, 1);
				bnx2x_igu_clear_sb(bp,
						   bp->igu_base_sb + sb_idx);
			}

			/* default-status-blocks */
			num_segs = CHIP_INT_MODE_IS_BC(bp) ?
				IGU_BC_DSB_NUM_SEGS : IGU_NORM_DSB_NUM_SEGS;

			if (CHIP_MODE_IS_4_PORT(bp))
				dsb_idx = BP_FUNC(bp);
			else
6860
				dsb_idx = BP_VN(bp);
D
Dmitry Kravkov 已提交
6861 6862 6863 6864 6865

			prod_offset = (CHIP_INT_MODE_IS_BC(bp) ?
				       IGU_BC_BASE_DSB_PROD + dsb_idx :
				       IGU_NORM_BASE_DSB_PROD + dsb_idx);

6866 6867 6868 6869
			/*
			 * igu prods come in chunks of E1HVN_MAX (4) -
			 * does not matters what is the current chip mode
			 */
D
Dmitry Kravkov 已提交
6870 6871 6872 6873 6874 6875 6876 6877 6878 6879 6880 6881 6882 6883 6884 6885 6886 6887 6888 6889 6890 6891 6892 6893 6894 6895 6896 6897 6898 6899 6900 6901 6902 6903 6904
			for (i = 0; i < (num_segs * E1HVN_MAX);
			     i += E1HVN_MAX) {
				addr = IGU_REG_PROD_CONS_MEMORY +
							(prod_offset + i)*4;
				REG_WR(bp, addr, 0);
			}
			/* send consumer update with 0 */
			if (CHIP_INT_MODE_IS_BC(bp)) {
				bnx2x_ack_sb(bp, bp->igu_dsb_id,
					     USTORM_ID, 0, IGU_INT_NOP, 1);
				bnx2x_ack_sb(bp, bp->igu_dsb_id,
					     CSTORM_ID, 0, IGU_INT_NOP, 1);
				bnx2x_ack_sb(bp, bp->igu_dsb_id,
					     XSTORM_ID, 0, IGU_INT_NOP, 1);
				bnx2x_ack_sb(bp, bp->igu_dsb_id,
					     TSTORM_ID, 0, IGU_INT_NOP, 1);
				bnx2x_ack_sb(bp, bp->igu_dsb_id,
					     ATTENTION_ID, 0, IGU_INT_NOP, 1);
			} else {
				bnx2x_ack_sb(bp, bp->igu_dsb_id,
					     USTORM_ID, 0, IGU_INT_NOP, 1);
				bnx2x_ack_sb(bp, bp->igu_dsb_id,
					     ATTENTION_ID, 0, IGU_INT_NOP, 1);
			}
			bnx2x_igu_clear_sb(bp, bp->igu_dsb_id);

			/* !!! these should become driver const once
			   rf-tool supports split-68 const */
			REG_WR(bp, IGU_REG_SB_INT_BEFORE_MASK_LSB, 0);
			REG_WR(bp, IGU_REG_SB_INT_BEFORE_MASK_MSB, 0);
			REG_WR(bp, IGU_REG_SB_MASK_LSB, 0);
			REG_WR(bp, IGU_REG_SB_MASK_MSB, 0);
			REG_WR(bp, IGU_REG_PBA_STATUS_LSB, 0);
			REG_WR(bp, IGU_REG_PBA_STATUS_MSB, 0);
		}
6905 6906
	}

E
Eliezer Tamir 已提交
6907
	/* Reset PCIE errors for debug */
E
Eliezer Tamir 已提交
6908 6909
	REG_WR(bp, 0x2114, 0xffffffff);
	REG_WR(bp, 0x2120, 0xffffffff);
6910

6911 6912 6913 6914 6915 6916 6917 6918 6919
	if (CHIP_IS_E1x(bp)) {
		main_mem_size = HC_REG_MAIN_MEMORY_SIZE / 2; /*dwords*/
		main_mem_base = HC_REG_MAIN_MEMORY +
				BP_PORT(bp) * (main_mem_size * 4);
		main_mem_prty_clr = HC_REG_HC_PRTY_STS_CLR;
		main_mem_width = 8;

		val = REG_RD(bp, main_mem_prty_clr);
		if (val)
M
Merav Sicron 已提交
6920 6921 6922
			DP(NETIF_MSG_HW,
			   "Hmmm... Parity errors in HC block during function init (0x%x)!\n",
			   val);
6923 6924 6925 6926 6927 6928 6929 6930 6931 6932 6933 6934 6935

		/* Clear "false" parity errors in MSI-X table */
		for (i = main_mem_base;
		     i < main_mem_base + main_mem_size * 4;
		     i += main_mem_width) {
			bnx2x_read_dmae(bp, i, main_mem_width / 4);
			bnx2x_write_dmae(bp, bnx2x_sp_mapping(bp, wb_data),
					 i, main_mem_width / 4);
		}
		/* Clear HC parity attention */
		REG_RD(bp, main_mem_prty_clr);
	}

6936 6937 6938 6939 6940 6941 6942 6943 6944 6945 6946 6947
#ifdef BNX2X_STOP_ON_ERROR
	/* Enable STORMs SP logging */
	REG_WR8(bp, BAR_USTRORM_INTMEM +
	       USTORM_RECORD_SLOW_PATH_OFFSET(BP_FUNC(bp)), 1);
	REG_WR8(bp, BAR_TSTRORM_INTMEM +
	       TSTORM_RECORD_SLOW_PATH_OFFSET(BP_FUNC(bp)), 1);
	REG_WR8(bp, BAR_CSTRORM_INTMEM +
	       CSTORM_RECORD_SLOW_PATH_OFFSET(BP_FUNC(bp)), 1);
	REG_WR8(bp, BAR_XSTRORM_INTMEM +
	       XSTORM_RECORD_SLOW_PATH_OFFSET(BP_FUNC(bp)), 1);
#endif

Y
Yaniv Rosner 已提交
6948
	bnx2x_phy_probe(&bp->link_params);
D
Dmitry Kravkov 已提交
6949

6950 6951 6952
	return 0;
}

E
Eliezer Tamir 已提交
6953

D
Dmitry Kravkov 已提交
6954
void bnx2x_free_mem(struct bnx2x *bp)
E
Eliezer Tamir 已提交
6955 6956
{
	/* fastpath */
6957
	bnx2x_free_fp_mem(bp);
E
Eliezer Tamir 已提交
6958 6959 6960
	/* end of fastpath */

	BNX2X_PCI_FREE(bp->def_status_blk, bp->def_status_blk_mapping,
6961
		       sizeof(struct host_sp_status_block));
E
Eliezer Tamir 已提交
6962

6963 6964 6965
	BNX2X_PCI_FREE(bp->fw_stats, bp->fw_stats_mapping,
		       bp->fw_stats_data_sz + bp->fw_stats_req_sz);

E
Eliezer Tamir 已提交
6966
	BNX2X_PCI_FREE(bp->slowpath, bp->slowpath_mapping,
6967
		       sizeof(struct bnx2x_slowpath));
E
Eliezer Tamir 已提交
6968

6969 6970 6971 6972 6973 6974
	BNX2X_PCI_FREE(bp->context.vcxt, bp->context.cxt_mapping,
		       bp->context.size);

	bnx2x_ilt_mem_op(bp, ILT_MEMOP_FREE);

	BNX2X_FREE(bp->ilt->lines);
D
Dmitry Kravkov 已提交
6975

6976
#ifdef BCM_CNIC
6977
	if (!CHIP_IS_E1x(bp))
D
Dmitry Kravkov 已提交
6978 6979 6980 6981 6982
		BNX2X_PCI_FREE(bp->cnic_sb.e2_sb, bp->cnic_sb_mapping,
			       sizeof(struct host_hc_status_block_e2));
	else
		BNX2X_PCI_FREE(bp->cnic_sb.e1x_sb, bp->cnic_sb_mapping,
			       sizeof(struct host_hc_status_block_e1x));
D
Dmitry Kravkov 已提交
6983

6984
	BNX2X_PCI_FREE(bp->t2, bp->t2_mapping, SRC_T2_SZ);
E
Eliezer Tamir 已提交
6985
#endif
D
Dmitry Kravkov 已提交
6986

6987
	BNX2X_PCI_FREE(bp->spq, bp->spq_mapping, BCM_PAGE_SIZE);
E
Eliezer Tamir 已提交
6988

6989 6990
	BNX2X_PCI_FREE(bp->eq_ring, bp->eq_mapping,
		       BCM_PAGE_SIZE * NUM_EQ_PAGES);
6991 6992 6993 6994 6995
}

static inline int bnx2x_alloc_fw_stats_mem(struct bnx2x *bp)
{
	int num_groups;
B
Barak Witkowski 已提交
6996
	int is_fcoe_stats = NO_FCOE(bp) ? 0 : 1;
6997

B
Barak Witkowski 已提交
6998 6999
	/* number of queues for statistics is number of eth queues + FCoE */
	u8 num_queue_stats = BNX2X_NUM_ETH_QUEUES(bp) + is_fcoe_stats;
7000 7001

	/* Total number of FW statistics requests =
B
Barak Witkowski 已提交
7002 7003 7004 7005
	 * 1 for port stats + 1 for PF stats + potential 1 for FCoE stats +
	 * num of queues
	 */
	bp->fw_stats_num = 2 + is_fcoe_stats + num_queue_stats;
7006

7007 7008 7009 7010 7011 7012

	/* Request is built from stats_query_header and an array of
	 * stats_query_cmd_group each of which contains
	 * STATS_QUERY_CMD_COUNT rules. The real number or requests is
	 * configured in the stats_query_header.
	 */
B
Barak Witkowski 已提交
7013 7014
	num_groups = ((bp->fw_stats_num) / STATS_QUERY_CMD_COUNT) +
		     (((bp->fw_stats_num) % STATS_QUERY_CMD_COUNT) ? 1 : 0);
7015 7016 7017 7018 7019 7020 7021 7022

	bp->fw_stats_req_sz = sizeof(struct stats_query_header) +
			num_groups * sizeof(struct stats_query_cmd_group);

	/* Data for statistics requests + stats_conter
	 *
	 * stats_counter holds per-STORM counters that are incremented
	 * when STORM has finished with the current request.
B
Barak Witkowski 已提交
7023 7024 7025
	 *
	 * memory for FCoE offloaded statistics are counted anyway,
	 * even if they will not be sent.
7026 7027 7028
	 */
	bp->fw_stats_data_sz = sizeof(struct per_port_stats) +
		sizeof(struct per_pf_stats) +
B
Barak Witkowski 已提交
7029
		sizeof(struct fcoe_statistics_params) +
7030 7031 7032 7033 7034 7035 7036 7037 7038 7039 7040 7041 7042 7043 7044 7045 7046 7047 7048 7049
		sizeof(struct per_queue_stats) * num_queue_stats +
		sizeof(struct stats_counter);

	BNX2X_PCI_ALLOC(bp->fw_stats, &bp->fw_stats_mapping,
			bp->fw_stats_data_sz + bp->fw_stats_req_sz);

	/* Set shortcuts */
	bp->fw_stats_req = (struct bnx2x_fw_stats_req *)bp->fw_stats;
	bp->fw_stats_req_mapping = bp->fw_stats_mapping;

	bp->fw_stats_data = (struct bnx2x_fw_stats_data *)
		((u8 *)bp->fw_stats + bp->fw_stats_req_sz);

	bp->fw_stats_data_mapping = bp->fw_stats_mapping +
				   bp->fw_stats_req_sz;
	return 0;

alloc_mem_err:
	BNX2X_PCI_FREE(bp->fw_stats, bp->fw_stats_mapping,
		       bp->fw_stats_data_sz + bp->fw_stats_req_sz);
M
Merav Sicron 已提交
7050
	BNX2X_ERR("Can't allocate memory\n");
7051
	return -ENOMEM;
E
Eliezer Tamir 已提交
7052 7053
}

D
Dmitry Kravkov 已提交
7054

D
Dmitry Kravkov 已提交
7055
int bnx2x_alloc_mem(struct bnx2x *bp)
E
Eliezer Tamir 已提交
7056
{
7057
#ifdef BCM_CNIC
7058 7059
	if (!CHIP_IS_E1x(bp))
		/* size = the status block + ramrod buffers */
D
Dmitry Kravkov 已提交
7060 7061 7062 7063 7064
		BNX2X_PCI_ALLOC(bp->cnic_sb.e2_sb, &bp->cnic_sb_mapping,
				sizeof(struct host_hc_status_block_e2));
	else
		BNX2X_PCI_ALLOC(bp->cnic_sb.e1x_sb, &bp->cnic_sb_mapping,
				sizeof(struct host_hc_status_block_e1x));
E
Eilon Greenstein 已提交
7065

7066 7067 7068
	/* allocate searcher T2 table */
	BNX2X_PCI_ALLOC(bp->t2, &bp->t2_mapping, SRC_T2_SZ);
#endif
E
Eliezer Tamir 已提交
7069

E
Eilon Greenstein 已提交
7070

7071 7072
	BNX2X_PCI_ALLOC(bp->def_status_blk, &bp->def_status_blk_mapping,
			sizeof(struct host_sp_status_block));
E
Eliezer Tamir 已提交
7073

7074 7075
	BNX2X_PCI_ALLOC(bp->slowpath, &bp->slowpath_mapping,
			sizeof(struct bnx2x_slowpath));
E
Eliezer Tamir 已提交
7076

7077 7078 7079 7080 7081
#ifdef BCM_CNIC
	/* write address to which L5 should insert its values */
	bp->cnic_eth_dev.addr_drv_info_to_mcp = &bp->slowpath->drv_info_to_mcp;
#endif

7082 7083 7084 7085
	/* Allocated memory for FW statistics  */
	if (bnx2x_alloc_fw_stats_mem(bp))
		goto alloc_mem_err;

7086
	bp->context.size = sizeof(union cdu_context) * BNX2X_L2_CID_COUNT(bp);
D
Dmitry Kravkov 已提交
7087

7088 7089
	BNX2X_PCI_ALLOC(bp->context.vcxt, &bp->context.cxt_mapping,
			bp->context.size);
7090

7091
	BNX2X_ALLOC(bp->ilt->lines, sizeof(struct ilt_line) * ILT_MAX_LINES);
7092

7093 7094
	if (bnx2x_ilt_mem_op(bp, ILT_MEMOP_ALLOC))
		goto alloc_mem_err;
7095

D
Dmitry Kravkov 已提交
7096 7097
	/* Slow path ring */
	BNX2X_PCI_ALLOC(bp->spq, &bp->spq_mapping, BCM_PAGE_SIZE);
7098

7099 7100 7101
	/* EQ */
	BNX2X_PCI_ALLOC(bp->eq_ring, &bp->eq_mapping,
			BCM_PAGE_SIZE * NUM_EQ_PAGES);
7102

7103 7104 7105 7106 7107 7108 7109

	/* fastpath */
	/* need to be done at the end, since it's self adjusting to amount
	 * of memory available for RSS queues
	 */
	if (bnx2x_alloc_fp_mem(bp))
		goto alloc_mem_err;
D
Dmitry Kravkov 已提交
7110
	return 0;
E
Eilon Greenstein 已提交
7111

D
Dmitry Kravkov 已提交
7112 7113
alloc_mem_err:
	bnx2x_free_mem(bp);
M
Merav Sicron 已提交
7114
	BNX2X_ERR("Can't allocate memory\n");
D
Dmitry Kravkov 已提交
7115
	return -ENOMEM;
7116 7117
}

E
Eliezer Tamir 已提交
7118 7119 7120 7121
/*
 * Init service functions
 */

7122 7123 7124
int bnx2x_set_mac_one(struct bnx2x *bp, u8 *mac,
		      struct bnx2x_vlan_mac_obj *obj, bool set,
		      int mac_type, unsigned long *ramrod_flags)
E
Eliezer Tamir 已提交
7125
{
7126 7127
	int rc;
	struct bnx2x_vlan_mac_ramrod_params ramrod_param;
E
Eliezer Tamir 已提交
7128

7129
	memset(&ramrod_param, 0, sizeof(ramrod_param));
E
Eliezer Tamir 已提交
7130

7131 7132 7133
	/* Fill general parameters */
	ramrod_param.vlan_mac_obj = obj;
	ramrod_param.ramrod_flags = *ramrod_flags;
E
Eliezer Tamir 已提交
7134

7135 7136 7137
	/* Fill a user request section if needed */
	if (!test_bit(RAMROD_CONT, ramrod_flags)) {
		memcpy(ramrod_param.user_req.u.mac.mac, mac, ETH_ALEN);
E
Eliezer Tamir 已提交
7138

7139
		__set_bit(mac_type, &ramrod_param.user_req.vlan_mac_flags);
7140

7141 7142 7143 7144 7145
		/* Set the command: ADD or DEL */
		if (set)
			ramrod_param.user_req.cmd = BNX2X_VLAN_MAC_ADD;
		else
			ramrod_param.user_req.cmd = BNX2X_VLAN_MAC_DEL;
E
Eliezer Tamir 已提交
7146 7147
	}

7148 7149 7150 7151
	rc = bnx2x_config_vlan_mac(bp, &ramrod_param);
	if (rc < 0)
		BNX2X_ERR("%s MAC failed\n", (set ? "Set" : "Del"));
	return rc;
E
Eliezer Tamir 已提交
7152 7153
}

7154 7155 7156
int bnx2x_del_all_macs(struct bnx2x *bp,
		       struct bnx2x_vlan_mac_obj *mac_obj,
		       int mac_type, bool wait_for_comp)
7157
{
7158 7159
	int rc;
	unsigned long ramrod_flags = 0, vlan_mac_flags = 0;
7160

7161 7162 7163
	/* Wait for completion of requested */
	if (wait_for_comp)
		__set_bit(RAMROD_COMP_WAIT, &ramrod_flags);
7164

7165 7166
	/* Set the mac type of addresses we want to clear */
	__set_bit(mac_type, &vlan_mac_flags);
7167

7168 7169 7170
	rc = mac_obj->delete_all(bp, mac_obj, &vlan_mac_flags, &ramrod_flags);
	if (rc < 0)
		BNX2X_ERR("Failed to delete MACs: %d\n", rc);
7171

7172
	return rc;
7173 7174
}

7175
int bnx2x_set_eth_mac(struct bnx2x *bp, bool set)
7176
{
7177
	unsigned long ramrod_flags = 0;
7178

D
Dmitry Kravkov 已提交
7179 7180
#ifdef BCM_CNIC
	if (is_zero_ether_addr(bp->dev->dev_addr) && IS_MF_ISCSI_SD(bp)) {
M
Merav Sicron 已提交
7181 7182
		DP(NETIF_MSG_IFUP | NETIF_MSG_IFDOWN,
		   "Ignoring Zero MAC for STORAGE SD mode\n");
D
Dmitry Kravkov 已提交
7183 7184 7185 7186
		return 0;
	}
#endif

7187
	DP(NETIF_MSG_IFUP, "Adding Eth MAC\n");
7188

7189 7190 7191 7192
	__set_bit(RAMROD_COMP_WAIT, &ramrod_flags);
	/* Eth MAC is set on RSS leading client (fp[0]) */
	return bnx2x_set_mac_one(bp, bp->dev->dev_addr, &bp->fp->mac_obj, set,
				 BNX2X_ETH_MAC, &ramrod_flags);
7193
}
7194

7195
int bnx2x_setup_leading(struct bnx2x *bp)
V
Vladislav Zolotarov 已提交
7196
{
7197
	return bnx2x_setup_queue(bp, &bp->fp[0], 1);
7198
}
E
Eliezer Tamir 已提交
7199

7200
/**
7201
 * bnx2x_set_int_mode - configure interrupt mode
7202
 *
7203
 * @bp:		driver handle
7204
 *
7205
 * In case of MSI-X it will also try to enable MSI-X.
7206
 */
D
Dmitry Kravkov 已提交
7207
static void __devinit bnx2x_set_int_mode(struct bnx2x *bp)
E
Eilon Greenstein 已提交
7208
{
D
Dmitry Kravkov 已提交
7209
	switch (int_mode) {
7210 7211 7212 7213
	case INT_MODE_MSI:
		bnx2x_enable_msi(bp);
		/* falling through... */
	case INT_MODE_INTx:
7214
		bp->num_queues = 1 + NON_ETH_CONTEXT_USE;
M
Merav Sicron 已提交
7215
		BNX2X_DEV_INFO("set number of queues to 1\n");
E
Eilon Greenstein 已提交
7216
		break;
7217 7218 7219
	default:
		/* Set number of queues according to bp->multi_mode value */
		bnx2x_set_num_queues(bp);
E
Eilon Greenstein 已提交
7220

M
Merav Sicron 已提交
7221
		BNX2X_DEV_INFO("set number of queues to %d\n", bp->num_queues);
E
Eilon Greenstein 已提交
7222

7223 7224 7225 7226
		/* if we can't use MSI-X we only need one fp,
		 * so try to enable MSI-X with the requested number of fp's
		 * and fallback to MSI or legacy INTx with one fp
		 */
D
Dmitry Kravkov 已提交
7227
		if (bnx2x_enable_msix(bp)) {
7228
			/* failed to enable MSI-X */
M
Merav Sicron 已提交
7229 7230 7231
			BNX2X_DEV_INFO("Failed to enable MSI-X (%d), set number of queues to %d\n",
				       bp->num_queues, 1 + NON_ETH_CONTEXT_USE);

7232
			bp->num_queues = 1 + NON_ETH_CONTEXT_USE;
7233

D
Dmitry Kravkov 已提交
7234
			/* Try to enable MSI */
7235 7236 7237
			if (!(bp->flags & DISABLE_MSI_FLAG))
				bnx2x_enable_msi(bp);
		}
D
Dmitry Kravkov 已提交
7238 7239
		break;
	}
E
Eliezer Tamir 已提交
7240 7241
}

7242 7243 7244 7245 7246 7247
/* must be called prioir to any HW initializations */
static inline u16 bnx2x_cid_ilt_lines(struct bnx2x *bp)
{
	return L2_ILT_LINES(bp);
}

7248 7249 7250 7251 7252 7253 7254 7255 7256 7257 7258 7259 7260 7261 7262
void bnx2x_ilt_set_info(struct bnx2x *bp)
{
	struct ilt_client_info *ilt_client;
	struct bnx2x_ilt *ilt = BP_ILT(bp);
	u16 line = 0;

	ilt->start_line = FUNC_ILT_BASE(BP_FUNC(bp));
	DP(BNX2X_MSG_SP, "ilt starts at line %d\n", ilt->start_line);

	/* CDU */
	ilt_client = &ilt->clients[ILT_CLIENT_CDU];
	ilt_client->client_num = ILT_CLIENT_CDU;
	ilt_client->page_size = CDU_ILT_PAGE_SZ;
	ilt_client->flags = ILT_CLIENT_SKIP_MEM;
	ilt_client->start = line;
7263
	line += bnx2x_cid_ilt_lines(bp);
7264 7265 7266 7267 7268
#ifdef BCM_CNIC
	line += CNIC_ILT_LINES;
#endif
	ilt_client->end = line - 1;

M
Merav Sicron 已提交
7269
	DP(NETIF_MSG_IFUP, "ilt client[CDU]: start %d, end %d, psz 0x%x, flags 0x%x, hw psz %d\n",
7270 7271 7272 7273 7274 7275 7276 7277 7278 7279 7280 7281 7282 7283 7284 7285 7286 7287 7288 7289
	   ilt_client->start,
	   ilt_client->end,
	   ilt_client->page_size,
	   ilt_client->flags,
	   ilog2(ilt_client->page_size >> 12));

	/* QM */
	if (QM_INIT(bp->qm_cid_count)) {
		ilt_client = &ilt->clients[ILT_CLIENT_QM];
		ilt_client->client_num = ILT_CLIENT_QM;
		ilt_client->page_size = QM_ILT_PAGE_SZ;
		ilt_client->flags = 0;
		ilt_client->start = line;

		/* 4 bytes for each cid */
		line += DIV_ROUND_UP(bp->qm_cid_count * QM_QUEUES_PER_FUNC * 4,
							 QM_ILT_PAGE_SZ);

		ilt_client->end = line - 1;

M
Merav Sicron 已提交
7290 7291
		DP(NETIF_MSG_IFUP,
		   "ilt client[QM]: start %d, end %d, psz 0x%x, flags 0x%x, hw psz %d\n",
7292 7293 7294 7295 7296 7297 7298 7299 7300 7301 7302 7303 7304 7305 7306 7307 7308
		   ilt_client->start,
		   ilt_client->end,
		   ilt_client->page_size,
		   ilt_client->flags,
		   ilog2(ilt_client->page_size >> 12));

	}
	/* SRC */
	ilt_client = &ilt->clients[ILT_CLIENT_SRC];
#ifdef BCM_CNIC
	ilt_client->client_num = ILT_CLIENT_SRC;
	ilt_client->page_size = SRC_ILT_PAGE_SZ;
	ilt_client->flags = 0;
	ilt_client->start = line;
	line += SRC_ILT_LINES;
	ilt_client->end = line - 1;

M
Merav Sicron 已提交
7309 7310
	DP(NETIF_MSG_IFUP,
	   "ilt client[SRC]: start %d, end %d, psz 0x%x, flags 0x%x, hw psz %d\n",
7311 7312 7313 7314 7315 7316 7317 7318 7319
	   ilt_client->start,
	   ilt_client->end,
	   ilt_client->page_size,
	   ilt_client->flags,
	   ilog2(ilt_client->page_size >> 12));

#else
	ilt_client->flags = (ILT_CLIENT_SKIP_INIT | ILT_CLIENT_SKIP_MEM);
#endif
D
Dmitry Kravkov 已提交
7320

7321 7322 7323 7324 7325 7326 7327 7328 7329 7330
	/* TM */
	ilt_client = &ilt->clients[ILT_CLIENT_TM];
#ifdef BCM_CNIC
	ilt_client->client_num = ILT_CLIENT_TM;
	ilt_client->page_size = TM_ILT_PAGE_SZ;
	ilt_client->flags = 0;
	ilt_client->start = line;
	line += TM_ILT_LINES;
	ilt_client->end = line - 1;

M
Merav Sicron 已提交
7331 7332
	DP(NETIF_MSG_IFUP,
	   "ilt client[TM]: start %d, end %d, psz 0x%x, flags 0x%x, hw psz %d\n",
7333 7334 7335 7336 7337
	   ilt_client->start,
	   ilt_client->end,
	   ilt_client->page_size,
	   ilt_client->flags,
	   ilog2(ilt_client->page_size >> 12));
D
Dmitry Kravkov 已提交
7338

7339 7340 7341
#else
	ilt_client->flags = (ILT_CLIENT_SKIP_INIT | ILT_CLIENT_SKIP_MEM);
#endif
7342
	BUG_ON(line > ILT_MAX_LINES);
7343
}
D
Dmitry Kravkov 已提交
7344

7345 7346 7347 7348 7349 7350 7351 7352 7353 7354 7355 7356 7357
/**
 * bnx2x_pf_q_prep_init - prepare INIT transition parameters
 *
 * @bp:			driver handle
 * @fp:			pointer to fastpath
 * @init_params:	pointer to parameters structure
 *
 * parameters configured:
 *      - HC configuration
 *      - Queue's CDU context
 */
static inline void bnx2x_pf_q_prep_init(struct bnx2x *bp,
	struct bnx2x_fastpath *fp, struct bnx2x_queue_init_params *init_params)
E
Eliezer Tamir 已提交
7358
{
7359 7360

	u8 cos;
7361 7362 7363 7364 7365 7366 7367 7368 7369 7370 7371 7372 7373 7374 7375 7376 7377 7378 7379 7380 7381 7382 7383 7384 7385
	/* FCoE Queue uses Default SB, thus has no HC capabilities */
	if (!IS_FCOE_FP(fp)) {
		__set_bit(BNX2X_Q_FLG_HC, &init_params->rx.flags);
		__set_bit(BNX2X_Q_FLG_HC, &init_params->tx.flags);

		/* If HC is supporterd, enable host coalescing in the transition
		 * to INIT state.
		 */
		__set_bit(BNX2X_Q_FLG_HC_EN, &init_params->rx.flags);
		__set_bit(BNX2X_Q_FLG_HC_EN, &init_params->tx.flags);

		/* HC rate */
		init_params->rx.hc_rate = bp->rx_ticks ?
			(1000000 / bp->rx_ticks) : 0;
		init_params->tx.hc_rate = bp->tx_ticks ?
			(1000000 / bp->tx_ticks) : 0;

		/* FW SB ID */
		init_params->rx.fw_sb_id = init_params->tx.fw_sb_id =
			fp->fw_sb_id;

		/*
		 * CQ index among the SB indices: FCoE clients uses the default
		 * SB, therefore it's different.
		 */
7386 7387
		init_params->rx.sb_cq_index = HC_INDEX_ETH_RX_CQ_CONS;
		init_params->tx.sb_cq_index = HC_INDEX_ETH_FIRST_TX_CQ_CONS;
7388 7389
	}

7390 7391 7392
	/* set maximum number of COSs supported by this queue */
	init_params->max_cos = fp->max_cos;

M
Merav Sicron 已提交
7393
	DP(NETIF_MSG_IFUP, "fp: %d setting queue params max cos to: %d\n",
7394 7395 7396 7397 7398 7399
	    fp->index, init_params->max_cos);

	/* set the context pointers queue object */
	for (cos = FIRST_TX_COS_INDEX; cos < init_params->max_cos; cos++)
		init_params->cxts[cos] =
			&bp->context.vcxt[fp->txdata[cos].cid].eth;
7400 7401
}

7402 7403 7404 7405 7406 7407 7408 7409 7410 7411 7412 7413 7414 7415 7416 7417 7418 7419 7420 7421 7422 7423
int bnx2x_setup_tx_only(struct bnx2x *bp, struct bnx2x_fastpath *fp,
			struct bnx2x_queue_state_params *q_params,
			struct bnx2x_queue_setup_tx_only_params *tx_only_params,
			int tx_index, bool leading)
{
	memset(tx_only_params, 0, sizeof(*tx_only_params));

	/* Set the command */
	q_params->cmd = BNX2X_Q_CMD_SETUP_TX_ONLY;

	/* Set tx-only QUEUE flags: don't zero statistics */
	tx_only_params->flags = bnx2x_get_common_flags(bp, fp, false);

	/* choose the index of the cid to send the slow path on */
	tx_only_params->cid_index = tx_index;

	/* Set general TX_ONLY_SETUP parameters */
	bnx2x_pf_q_prep_general(bp, fp, &tx_only_params->gen_params, tx_index);

	/* Set Tx TX_ONLY_SETUP parameters */
	bnx2x_pf_tx_q_prep(bp, fp, &tx_only_params->txq_params, tx_index);

M
Merav Sicron 已提交
7424 7425
	DP(NETIF_MSG_IFUP,
	   "preparing to send tx-only ramrod for connection: cos %d, primary cid %d, cid %d, client id %d, sp-client id %d, flags %lx\n",
7426 7427 7428 7429 7430 7431 7432 7433 7434
	   tx_index, q_params->q_obj->cids[FIRST_TX_COS_INDEX],
	   q_params->q_obj->cids[tx_index], q_params->q_obj->cl_id,
	   tx_only_params->gen_params.spcl_id, tx_only_params->flags);

	/* send the ramrod */
	return bnx2x_queue_state_change(bp, q_params);
}


7435 7436 7437 7438 7439 7440 7441 7442 7443 7444 7445 7446 7447 7448
/**
 * bnx2x_setup_queue - setup queue
 *
 * @bp:		driver handle
 * @fp:		pointer to fastpath
 * @leading:	is leading
 *
 * This function performs 2 steps in a Queue state machine
 *      actually: 1) RESET->INIT 2) INIT->SETUP
 */

int bnx2x_setup_queue(struct bnx2x *bp, struct bnx2x_fastpath *fp,
		       bool leading)
{
Y
Yuval Mintz 已提交
7449
	struct bnx2x_queue_state_params q_params = {NULL};
7450 7451
	struct bnx2x_queue_setup_params *setup_params =
						&q_params.params.setup;
7452 7453
	struct bnx2x_queue_setup_tx_only_params *tx_only_params =
						&q_params.params.tx_only;
E
Eliezer Tamir 已提交
7454
	int rc;
7455 7456
	u8 tx_index;

M
Merav Sicron 已提交
7457
	DP(NETIF_MSG_IFUP, "setting up queue %d\n", fp->index);
E
Eliezer Tamir 已提交
7458

V
Vladislav Zolotarov 已提交
7459 7460 7461
	/* reset IGU state skip FCoE L2 queue */
	if (!IS_FCOE_FP(fp))
		bnx2x_ack_sb(bp, fp->igu_sb_id, USTORM_ID, 0,
7462
			     IGU_INT_ENABLE, 0);
E
Eliezer Tamir 已提交
7463

7464 7465 7466
	q_params.q_obj = &fp->q_obj;
	/* We want to wait for completion in this context */
	__set_bit(RAMROD_COMP_WAIT, &q_params.ramrod_flags);
E
Eliezer Tamir 已提交
7467

7468 7469
	/* Prepare the INIT parameters */
	bnx2x_pf_q_prep_init(bp, fp, &q_params.params.init);
V
Vladislav Zolotarov 已提交
7470

7471 7472 7473 7474 7475 7476
	/* Set the command */
	q_params.cmd = BNX2X_Q_CMD_INIT;

	/* Change the state to INIT */
	rc = bnx2x_queue_state_change(bp, &q_params);
	if (rc) {
7477
		BNX2X_ERR("Queue(%d) INIT failed\n", fp->index);
7478 7479
		return rc;
	}
V
Vladislav Zolotarov 已提交
7480

M
Merav Sicron 已提交
7481
	DP(NETIF_MSG_IFUP, "init complete\n");
7482 7483


7484 7485
	/* Now move the Queue to the SETUP state... */
	memset(setup_params, 0, sizeof(*setup_params));
E
Eliezer Tamir 已提交
7486

7487 7488
	/* Set QUEUE flags */
	setup_params->flags = bnx2x_get_q_flags(bp, fp, leading);
7489

7490
	/* Set general SETUP parameters */
7491 7492
	bnx2x_pf_q_prep_general(bp, fp, &setup_params->gen_params,
				FIRST_TX_COS_INDEX);
7493

7494
	bnx2x_pf_rx_q_prep(bp, fp, &setup_params->pause_params,
7495 7496
			    &setup_params->rxq_params);

7497 7498
	bnx2x_pf_tx_q_prep(bp, fp, &setup_params->txq_params,
			   FIRST_TX_COS_INDEX);
7499 7500 7501 7502 7503 7504

	/* Set the command */
	q_params.cmd = BNX2X_Q_CMD_SETUP;

	/* Change the state to SETUP */
	rc = bnx2x_queue_state_change(bp, &q_params);
7505 7506 7507 7508 7509 7510 7511 7512 7513 7514 7515 7516 7517 7518 7519 7520 7521 7522 7523
	if (rc) {
		BNX2X_ERR("Queue(%d) SETUP failed\n", fp->index);
		return rc;
	}

	/* loop through the relevant tx-only indices */
	for (tx_index = FIRST_TX_ONLY_COS_INDEX;
	      tx_index < fp->max_cos;
	      tx_index++) {

		/* prepare and send tx-only ramrod*/
		rc = bnx2x_setup_tx_only(bp, fp, &q_params,
					  tx_only_params, tx_index, leading);
		if (rc) {
			BNX2X_ERR("Queue(%d.%d) TX_ONLY_SETUP failed\n",
				  fp->index, tx_index);
			return rc;
		}
	}
7524

7525
	return rc;
E
Eliezer Tamir 已提交
7526 7527
}

7528
static int bnx2x_stop_queue(struct bnx2x *bp, int index)
E
Eliezer Tamir 已提交
7529
{
7530
	struct bnx2x_fastpath *fp = &bp->fp[index];
7531
	struct bnx2x_fp_txdata *txdata;
Y
Yuval Mintz 已提交
7532
	struct bnx2x_queue_state_params q_params = {NULL};
7533 7534
	int rc, tx_index;

M
Merav Sicron 已提交
7535
	DP(NETIF_MSG_IFDOWN, "stopping queue %d cid %d\n", index, fp->cid);
E
Eliezer Tamir 已提交
7536

7537 7538 7539
	q_params.q_obj = &fp->q_obj;
	/* We want to wait for completion in this context */
	__set_bit(RAMROD_COMP_WAIT, &q_params.ramrod_flags);
E
Eliezer Tamir 已提交
7540

7541 7542 7543 7544 7545 7546 7547 7548 7549

	/* close tx-only connections */
	for (tx_index = FIRST_TX_ONLY_COS_INDEX;
	     tx_index < fp->max_cos;
	     tx_index++){

		/* ascertain this is a normal queue*/
		txdata = &fp->txdata[tx_index];

M
Merav Sicron 已提交
7550
		DP(NETIF_MSG_IFDOWN, "stopping tx-only queue %d\n",
7551 7552 7553 7554 7555 7556 7557 7558 7559 7560 7561 7562 7563 7564 7565 7566 7567 7568 7569 7570 7571 7572 7573
							txdata->txq_index);

		/* send halt terminate on tx-only connection */
		q_params.cmd = BNX2X_Q_CMD_TERMINATE;
		memset(&q_params.params.terminate, 0,
		       sizeof(q_params.params.terminate));
		q_params.params.terminate.cid_index = tx_index;

		rc = bnx2x_queue_state_change(bp, &q_params);
		if (rc)
			return rc;

		/* send halt terminate on tx-only connection */
		q_params.cmd = BNX2X_Q_CMD_CFC_DEL;
		memset(&q_params.params.cfc_del, 0,
		       sizeof(q_params.params.cfc_del));
		q_params.params.cfc_del.cid_index = tx_index;
		rc = bnx2x_queue_state_change(bp, &q_params);
		if (rc)
			return rc;
	}
	/* Stop the primary connection: */
	/* ...halt the connection */
7574 7575 7576
	q_params.cmd = BNX2X_Q_CMD_HALT;
	rc = bnx2x_queue_state_change(bp, &q_params);
	if (rc)
7577
		return rc;
E
Eliezer Tamir 已提交
7578

7579
	/* ...terminate the connection */
7580
	q_params.cmd = BNX2X_Q_CMD_TERMINATE;
7581 7582 7583
	memset(&q_params.params.terminate, 0,
	       sizeof(q_params.params.terminate));
	q_params.params.terminate.cid_index = FIRST_TX_COS_INDEX;
7584 7585
	rc = bnx2x_queue_state_change(bp, &q_params);
	if (rc)
7586
		return rc;
7587
	/* ...delete cfc entry */
7588
	q_params.cmd = BNX2X_Q_CMD_CFC_DEL;
7589 7590 7591
	memset(&q_params.params.cfc_del, 0,
	       sizeof(q_params.params.cfc_del));
	q_params.params.cfc_del.cid_index = FIRST_TX_COS_INDEX;
7592
	return bnx2x_queue_state_change(bp, &q_params);
7593 7594 7595
}


7596 7597 7598 7599
static void bnx2x_reset_func(struct bnx2x *bp)
{
	int port = BP_PORT(bp);
	int func = BP_FUNC(bp);
D
Dmitry Kravkov 已提交
7600
	int i;
7601 7602 7603 7604 7605 7606 7607 7608

	/* Disable the function in the FW */
	REG_WR8(bp, BAR_XSTRORM_INTMEM + XSTORM_FUNC_EN_OFFSET(func), 0);
	REG_WR8(bp, BAR_CSTRORM_INTMEM + CSTORM_FUNC_EN_OFFSET(func), 0);
	REG_WR8(bp, BAR_TSTRORM_INTMEM + TSTORM_FUNC_EN_OFFSET(func), 0);
	REG_WR8(bp, BAR_USTRORM_INTMEM + USTORM_FUNC_EN_OFFSET(func), 0);

	/* FP SBs */
V
Vladislav Zolotarov 已提交
7609
	for_each_eth_queue(bp, i) {
7610
		struct bnx2x_fastpath *fp = &bp->fp[i];
7611
		REG_WR8(bp, BAR_CSTRORM_INTMEM +
7612 7613
			   CSTORM_STATUS_BLOCK_DATA_STATE_OFFSET(fp->fw_sb_id),
			   SB_DISABLED);
7614 7615
	}

7616 7617 7618 7619 7620 7621
#ifdef BCM_CNIC
	/* CNIC SB */
	REG_WR8(bp, BAR_CSTRORM_INTMEM +
		CSTORM_STATUS_BLOCK_DATA_STATE_OFFSET(bnx2x_cnic_fw_sb_id(bp)),
		SB_DISABLED);
#endif
7622
	/* SP SB */
7623
	REG_WR8(bp, BAR_CSTRORM_INTMEM +
7624 7625
		   CSTORM_SP_STATUS_BLOCK_DATA_STATE_OFFSET(func),
		   SB_DISABLED);
7626 7627 7628 7629

	for (i = 0; i < XSTORM_SPQ_DATA_SIZE / 4; i++)
		REG_WR(bp, BAR_XSTRORM_INTMEM + XSTORM_SPQ_DATA_OFFSET(func),
		       0);
7630 7631

	/* Configure IGU */
D
Dmitry Kravkov 已提交
7632 7633 7634 7635 7636 7637 7638
	if (bp->common.int_block == INT_BLOCK_HC) {
		REG_WR(bp, HC_REG_LEADING_EDGE_0 + port*8, 0);
		REG_WR(bp, HC_REG_TRAILING_EDGE_0 + port*8, 0);
	} else {
		REG_WR(bp, IGU_REG_LEADING_EDGE_LATCH, 0);
		REG_WR(bp, IGU_REG_TRAILING_EDGE_LATCH, 0);
	}
7639

7640 7641 7642 7643 7644 7645 7646 7647 7648 7649 7650 7651 7652
#ifdef BCM_CNIC
	/* Disable Timer scan */
	REG_WR(bp, TM_REG_EN_LINEAR0_TIMER + port*4, 0);
	/*
	 * Wait for at least 10ms and up to 2 second for the timers scan to
	 * complete
	 */
	for (i = 0; i < 200; i++) {
		msleep(10);
		if (!REG_RD(bp, TM_REG_LIN0_SCAN_ON + port*4))
			break;
	}
#endif
7653
	/* Clear ILT */
D
Dmitry Kravkov 已提交
7654 7655 7656 7657 7658
	bnx2x_clear_func_ilt(bp, func);

	/* Timers workaround bug for E2: if this is vnic-3,
	 * we need to set the entire ilt range for this timers.
	 */
7659
	if (!CHIP_IS_E1x(bp) && BP_VN(bp) == 3) {
D
Dmitry Kravkov 已提交
7660 7661 7662 7663 7664 7665 7666 7667 7668 7669 7670
		struct ilt_client_info ilt_cli;
		/* use dummy TM client */
		memset(&ilt_cli, 0, sizeof(struct ilt_client_info));
		ilt_cli.start = 0;
		ilt_cli.end = ILT_NUM_PAGE_ENTRIES - 1;
		ilt_cli.client_num = ILT_CLIENT_TM;

		bnx2x_ilt_boundry_init_op(bp, &ilt_cli, 0, INITOP_CLEAR);
	}

	/* this assumes that reset_port() called before reset_func()*/
7671
	if (!CHIP_IS_E1x(bp))
D
Dmitry Kravkov 已提交
7672
		bnx2x_pf_disable(bp);
7673 7674

	bp->dmae_ready = 0;
7675 7676 7677 7678 7679 7680 7681
}

static void bnx2x_reset_port(struct bnx2x *bp)
{
	int port = BP_PORT(bp);
	u32 val;

7682 7683 7684
	/* Reset physical Link */
	bnx2x__link_reset(bp);

7685 7686 7687 7688 7689 7690 7691 7692 7693 7694 7695 7696 7697 7698 7699 7700
	REG_WR(bp, NIG_REG_MASK_INTERRUPT_PORT0 + port*4, 0);

	/* Do not rcv packets to BRB */
	REG_WR(bp, NIG_REG_LLH0_BRB1_DRV_MASK + port*4, 0x0);
	/* Do not direct rcv packets that are not for MCP to the BRB */
	REG_WR(bp, (port ? NIG_REG_LLH1_BRB1_NOT_MCP :
			   NIG_REG_LLH0_BRB1_NOT_MCP), 0x0);

	/* Configure AEU */
	REG_WR(bp, MISC_REG_AEU_MASK_ATTN_FUNC_0 + port*4, 0);

	msleep(100);
	/* Check for BRB port occupancy */
	val = REG_RD(bp, BRB1_REG_PORT_NUM_OCC_BLOCKS_0 + port*4);
	if (val)
		DP(NETIF_MSG_IFDOWN,
E
Eilon Greenstein 已提交
7701
		   "BRB1 is not empty  %d blocks are occupied\n", val);
7702 7703 7704 7705

	/* TODO: Close Doorbell port? */
}

7706
static inline int bnx2x_reset_hw(struct bnx2x *bp, u32 load_code)
7707
{
Y
Yuval Mintz 已提交
7708
	struct bnx2x_func_state_params func_params = {NULL};
7709

7710 7711
	/* Prepare parameters for function state transitions */
	__set_bit(RAMROD_COMP_WAIT, &func_params.ramrod_flags);
7712

7713 7714
	func_params.f_obj = &bp->func_obj;
	func_params.cmd = BNX2X_F_CMD_HW_RESET;
7715

7716
	func_params.params.hw_init.load_phase = load_code;
7717

7718
	return bnx2x_func_state_change(bp, &func_params);
7719 7720
}

7721
static inline int bnx2x_func_stop(struct bnx2x *bp)
V
Vladislav Zolotarov 已提交
7722
{
Y
Yuval Mintz 已提交
7723
	struct bnx2x_func_state_params func_params = {NULL};
7724
	int rc;
7725

7726 7727 7728 7729
	/* Prepare parameters for function state transitions */
	__set_bit(RAMROD_COMP_WAIT, &func_params.ramrod_flags);
	func_params.f_obj = &bp->func_obj;
	func_params.cmd = BNX2X_F_CMD_STOP;
7730

7731 7732 7733 7734 7735 7736 7737 7738
	/*
	 * Try to stop the function the 'good way'. If fails (in case
	 * of a parity error during bnx2x_chip_cleanup()) and we are
	 * not in a debug mode, perform a state transaction in order to
	 * enable further HW_RESET transaction.
	 */
	rc = bnx2x_func_state_change(bp, &func_params);
	if (rc) {
7739
#ifdef BNX2X_STOP_ON_ERROR
7740
		return rc;
7741
#else
M
Merav Sicron 已提交
7742
		BNX2X_ERR("FUNC_STOP ramrod failed. Running a dry transaction\n");
7743 7744
		__set_bit(RAMROD_DRV_CLR_ONLY, &func_params.ramrod_flags);
		return bnx2x_func_state_change(bp, &func_params);
7745
#endif
7746
	}
E
Eliezer Tamir 已提交
7747

7748 7749
	return 0;
}
7750

7751 7752 7753 7754 7755 7756 7757 7758 7759 7760 7761 7762
/**
 * bnx2x_send_unload_req - request unload mode from the MCP.
 *
 * @bp:			driver handle
 * @unload_mode:	requested function's unload mode
 *
 * Return unload mode returned by the MCP: COMMON, PORT or FUNC.
 */
u32 bnx2x_send_unload_req(struct bnx2x *bp, int unload_mode)
{
	u32 reset_code = 0;
	int port = BP_PORT(bp);
7763

7764
	/* Select the UNLOAD request mode */
7765 7766 7767
	if (unload_mode == UNLOAD_NORMAL)
		reset_code = DRV_MSG_CODE_UNLOAD_REQ_WOL_DIS;

7768
	else if (bp->flags & NO_WOL_FLAG)
7769 7770
		reset_code = DRV_MSG_CODE_UNLOAD_REQ_WOL_MCP;

7771
	else if (bp->wol) {
7772 7773 7774
		u32 emac_base = port ? GRCBASE_EMAC1 : GRCBASE_EMAC0;
		u8 *mac_addr = bp->dev->dev_addr;
		u32 val;
7775 7776
		u16 pmc;

7777
		/* The mac address is written to entries 1-4 to
7778 7779
		 * preserve entry 0 which is used by the PMF
		 */
7780
		u8 entry = (BP_VN(bp) + 1)*8;
7781 7782 7783 7784 7785 7786 7787 7788

		val = (mac_addr[0] << 8) | mac_addr[1];
		EMAC_WR(bp, EMAC_REG_EMAC_MAC_MATCH + entry, val);

		val = (mac_addr[2] << 24) | (mac_addr[3] << 16) |
		      (mac_addr[4] << 8) | mac_addr[5];
		EMAC_WR(bp, EMAC_REG_EMAC_MAC_MATCH + entry + 4, val);

7789 7790 7791 7792 7793
		/* Enable the PME and clear the status */
		pci_read_config_word(bp->pdev, bp->pm_cap + PCI_PM_CTRL, &pmc);
		pmc |= PCI_PM_CTRL_PME_ENABLE | PCI_PM_CTRL_PME_STATUS;
		pci_write_config_word(bp->pdev, bp->pm_cap + PCI_PM_CTRL, pmc);

7794 7795 7796 7797
		reset_code = DRV_MSG_CODE_UNLOAD_REQ_WOL_EN;

	} else
		reset_code = DRV_MSG_CODE_UNLOAD_REQ_WOL_DIS;
7798

7799 7800 7801 7802 7803 7804
	/* Send the request to the MCP */
	if (!BP_NOMCP(bp))
		reset_code = bnx2x_fw_command(bp, reset_code, 0);
	else {
		int path = BP_PATH(bp);

M
Merav Sicron 已提交
7805
		DP(NETIF_MSG_IFDOWN, "NO MCP - load counts[%d]      %d, %d, %d\n",
7806 7807 7808 7809
		   path, load_count[path][0], load_count[path][1],
		   load_count[path][2]);
		load_count[path][0]--;
		load_count[path][1 + port]--;
M
Merav Sicron 已提交
7810
		DP(NETIF_MSG_IFDOWN, "NO MCP - new load counts[%d]  %d, %d, %d\n",
7811 7812 7813 7814 7815 7816 7817 7818 7819 7820 7821 7822 7823 7824 7825 7826 7827 7828 7829 7830 7831 7832 7833 7834 7835
		   path, load_count[path][0], load_count[path][1],
		   load_count[path][2]);
		if (load_count[path][0] == 0)
			reset_code = FW_MSG_CODE_DRV_UNLOAD_COMMON;
		else if (load_count[path][1 + port] == 0)
			reset_code = FW_MSG_CODE_DRV_UNLOAD_PORT;
		else
			reset_code = FW_MSG_CODE_DRV_UNLOAD_FUNCTION;
	}

	return reset_code;
}

/**
 * bnx2x_send_unload_done - send UNLOAD_DONE command to the MCP.
 *
 * @bp:		driver handle
 */
void bnx2x_send_unload_done(struct bnx2x *bp)
{
	/* Report UNLOAD_DONE to MCP */
	if (!BP_NOMCP(bp))
		bnx2x_fw_command(bp, DRV_MSG_CODE_UNLOAD_DONE, 0);
}

D
Dmitry Kravkov 已提交
7836 7837 7838 7839 7840 7841 7842 7843 7844 7845 7846 7847 7848 7849 7850 7851 7852 7853 7854 7855 7856 7857 7858 7859 7860 7861 7862 7863 7864 7865 7866 7867 7868 7869 7870 7871 7872
static inline int bnx2x_func_wait_started(struct bnx2x *bp)
{
	int tout = 50;
	int msix = (bp->flags & USING_MSIX_FLAG) ? 1 : 0;

	if (!bp->port.pmf)
		return 0;

	/*
	 * (assumption: No Attention from MCP at this stage)
	 * PMF probably in the middle of TXdisable/enable transaction
	 * 1. Sync IRS for default SB
	 * 2. Sync SP queue - this guarantes us that attention handling started
	 * 3. Wait, that TXdisable/enable transaction completes
	 *
	 * 1+2 guranty that if DCBx attention was scheduled it already changed
	 * pending bit of transaction from STARTED-->TX_STOPPED, if we alredy
	 * received complettion for the transaction the state is TX_STOPPED.
	 * State will return to STARTED after completion of TX_STOPPED-->STARTED
	 * transaction.
	 */

	/* make sure default SB ISR is done */
	if (msix)
		synchronize_irq(bp->msix_table[0].vector);
	else
		synchronize_irq(bp->pdev->irq);

	flush_workqueue(bnx2x_wq);

	while (bnx2x_func_get_state(bp, &bp->func_obj) !=
				BNX2X_F_STATE_STARTED && tout--)
		msleep(20);

	if (bnx2x_func_get_state(bp, &bp->func_obj) !=
						BNX2X_F_STATE_STARTED) {
#ifdef BNX2X_STOP_ON_ERROR
M
Merav Sicron 已提交
7873
		BNX2X_ERR("Wrong function state\n");
D
Dmitry Kravkov 已提交
7874 7875 7876 7877 7878 7879
		return -EBUSY;
#else
		/*
		 * Failed to complete the transaction in a "good way"
		 * Force both transactions with CLR bit
		 */
Y
Yuval Mintz 已提交
7880
		struct bnx2x_func_state_params func_params = {NULL};
D
Dmitry Kravkov 已提交
7881

M
Merav Sicron 已提交
7882 7883
		DP(NETIF_MSG_IFDOWN,
		   "Hmmm... unexpected function state! Forcing STARTED-->TX_ST0PPED-->STARTED\n");
D
Dmitry Kravkov 已提交
7884 7885 7886 7887 7888 7889 7890 7891 7892 7893 7894 7895 7896 7897 7898 7899 7900 7901

		func_params.f_obj = &bp->func_obj;
		__set_bit(RAMROD_DRV_CLR_ONLY,
					&func_params.ramrod_flags);

		/* STARTED-->TX_ST0PPED */
		func_params.cmd = BNX2X_F_CMD_TX_STOP;
		bnx2x_func_state_change(bp, &func_params);

		/* TX_ST0PPED-->STARTED */
		func_params.cmd = BNX2X_F_CMD_TX_START;
		return bnx2x_func_state_change(bp, &func_params);
#endif
	}

	return 0;
}

7902 7903 7904
void bnx2x_chip_cleanup(struct bnx2x *bp, int unload_mode)
{
	int port = BP_PORT(bp);
7905 7906
	int i, rc = 0;
	u8 cos;
Y
Yuval Mintz 已提交
7907
	struct bnx2x_mcast_ramrod_params rparam = {NULL};
7908 7909 7910 7911 7912 7913
	u32 reset_code;

	/* Wait until tx fastpath tasks complete */
	for_each_tx_queue(bp, i) {
		struct bnx2x_fastpath *fp = &bp->fp[i];

7914 7915
		for_each_cos_in_tx_queue(fp, cos)
			rc = bnx2x_clean_tx_queue(bp, &fp->txdata[cos]);
7916 7917 7918 7919 7920 7921 7922 7923 7924 7925 7926 7927 7928 7929 7930 7931 7932 7933
#ifdef BNX2X_STOP_ON_ERROR
		if (rc)
			return;
#endif
	}

	/* Give HW time to discard old tx messages */
	usleep_range(1000, 1000);

	/* Clean all ETH MACs */
	rc = bnx2x_del_all_macs(bp, &bp->fp[0].mac_obj, BNX2X_ETH_MAC, false);
	if (rc < 0)
		BNX2X_ERR("Failed to delete all ETH macs: %d\n", rc);

	/* Clean up UC list  */
	rc = bnx2x_del_all_macs(bp, &bp->fp[0].mac_obj, BNX2X_UC_LIST_MAC,
				true);
	if (rc < 0)
M
Merav Sicron 已提交
7934 7935
		BNX2X_ERR("Failed to schedule DEL commands for UC MACs list: %d\n",
			  rc);
7936 7937 7938 7939 7940 7941 7942 7943 7944 7945 7946 7947 7948 7949 7950 7951 7952 7953 7954 7955 7956 7957 7958 7959 7960

	/* Disable LLH */
	if (!CHIP_IS_E1(bp))
		REG_WR(bp, NIG_REG_LLH0_FUNC_EN + port*8, 0);

	/* Set "drop all" (stop Rx).
	 * We need to take a netif_addr_lock() here in order to prevent
	 * a race between the completion code and this code.
	 */
	netif_addr_lock_bh(bp->dev);
	/* Schedule the rx_mode command */
	if (test_bit(BNX2X_FILTER_RX_MODE_PENDING, &bp->sp_state))
		set_bit(BNX2X_FILTER_RX_MODE_SCHED, &bp->sp_state);
	else
		bnx2x_set_storm_rx_mode(bp);

	/* Cleanup multicast configuration */
	rparam.mcast_obj = &bp->mcast_obj;
	rc = bnx2x_config_mcast(bp, &rparam, BNX2X_MCAST_CMD_DEL);
	if (rc < 0)
		BNX2X_ERR("Failed to send DEL multicast command: %d\n", rc);

	netif_addr_unlock_bh(bp->dev);


D
Dmitry Kravkov 已提交
7961 7962 7963 7964 7965 7966 7967 7968 7969 7970 7971 7972 7973 7974 7975 7976 7977 7978 7979 7980

	/*
	 * Send the UNLOAD_REQUEST to the MCP. This will return if
	 * this function should perform FUNC, PORT or COMMON HW
	 * reset.
	 */
	reset_code = bnx2x_send_unload_req(bp, unload_mode);

	/*
	 * (assumption: No Attention from MCP at this stage)
	 * PMF probably in the middle of TXdisable/enable transaction
	 */
	rc = bnx2x_func_wait_started(bp);
	if (rc) {
		BNX2X_ERR("bnx2x_func_wait_started failed\n");
#ifdef BNX2X_STOP_ON_ERROR
		return;
#endif
	}

7981
	/* Close multi and leading connections
7982 7983
	 * Completions for ramrods are collected in a synchronous way
	 */
7984
	for_each_queue(bp, i)
7985
		if (bnx2x_stop_queue(bp, i))
7986 7987 7988
#ifdef BNX2X_STOP_ON_ERROR
			return;
#else
7989
			goto unload_error;
7990
#endif
7991 7992 7993 7994 7995
	/* If SP settings didn't get completed so far - something
	 * very wrong has happen.
	 */
	if (!bnx2x_wait_sp_comp(bp, ~0x0UL))
		BNX2X_ERR("Hmmm... Common slow path ramrods got stuck!\n");
E
Eliezer Tamir 已提交
7996

7997 7998 7999
#ifndef BNX2X_STOP_ON_ERROR
unload_error:
#endif
8000
	rc = bnx2x_func_stop(bp);
8001
	if (rc) {
8002
		BNX2X_ERR("Function stop failed!\n");
8003
#ifdef BNX2X_STOP_ON_ERROR
8004 8005
		return;
#endif
8006
	}
E
Eliezer Tamir 已提交
8007

8008 8009 8010 8011
	/* Disable HW interrupts, NAPI */
	bnx2x_netif_stop(bp, 1);

	/* Release IRQs */
8012
	bnx2x_free_irq(bp);
8013

E
Eliezer Tamir 已提交
8014
	/* Reset the chip */
8015 8016 8017
	rc = bnx2x_reset_hw(bp, reset_code);
	if (rc)
		BNX2X_ERR("HW_RESET failed\n");
E
Eliezer Tamir 已提交
8018

E
Eilon Greenstein 已提交
8019

8020 8021
	/* Report UNLOAD_DONE to MCP */
	bnx2x_send_unload_done(bp);
8022 8023
}

D
Dmitry Kravkov 已提交
8024
void bnx2x_disable_close_the_gate(struct bnx2x *bp)
8025 8026 8027
{
	u32 val;

M
Merav Sicron 已提交
8028
	DP(NETIF_MSG_IFDOWN, "Disabling \"close the gates\"\n");
8029 8030 8031 8032 8033 8034 8035 8036 8037

	if (CHIP_IS_E1(bp)) {
		int port = BP_PORT(bp);
		u32 addr = port ? MISC_REG_AEU_MASK_ATTN_FUNC_1 :
			MISC_REG_AEU_MASK_ATTN_FUNC_0;

		val = REG_RD(bp, addr);
		val &= ~(0x300);
		REG_WR(bp, addr, val);
8038
	} else {
8039 8040 8041 8042 8043 8044 8045 8046 8047 8048
		val = REG_RD(bp, MISC_REG_AEU_GENERAL_MASK);
		val &= ~(MISC_AEU_GENERAL_MASK_REG_AEU_PXP_CLOSE_MASK |
			 MISC_AEU_GENERAL_MASK_REG_AEU_NIG_CLOSE_MASK);
		REG_WR(bp, MISC_REG_AEU_GENERAL_MASK, val);
	}
}

/* Close gates #2, #3 and #4: */
static void bnx2x_set_234_gates(struct bnx2x *bp, bool close)
{
8049
	u32 val;
8050 8051 8052 8053

	/* Gates #2 and #4a are closed/opened for "not E1" only */
	if (!CHIP_IS_E1(bp)) {
		/* #4 */
8054
		REG_WR(bp, PXP_REG_HST_DISCARD_DOORBELLS, !!close);
8055
		/* #2 */
8056
		REG_WR(bp, PXP_REG_HST_DISCARD_INTERNAL_WRITES, !!close);
8057 8058 8059
	}

	/* #3 */
8060 8061 8062 8063 8064 8065 8066 8067 8068 8069 8070 8071 8072 8073 8074 8075 8076 8077 8078 8079
	if (CHIP_IS_E1x(bp)) {
		/* Prevent interrupts from HC on both ports */
		val = REG_RD(bp, HC_REG_CONFIG_1);
		REG_WR(bp, HC_REG_CONFIG_1,
		       (!close) ? (val | HC_CONFIG_1_REG_BLOCK_DISABLE_1) :
		       (val & ~(u32)HC_CONFIG_1_REG_BLOCK_DISABLE_1));

		val = REG_RD(bp, HC_REG_CONFIG_0);
		REG_WR(bp, HC_REG_CONFIG_0,
		       (!close) ? (val | HC_CONFIG_0_REG_BLOCK_DISABLE_0) :
		       (val & ~(u32)HC_CONFIG_0_REG_BLOCK_DISABLE_0));
	} else {
		/* Prevent incomming interrupts in IGU */
		val = REG_RD(bp, IGU_REG_BLOCK_CONFIGURATION);

		REG_WR(bp, IGU_REG_BLOCK_CONFIGURATION,
		       (!close) ?
		       (val | IGU_BLOCK_CONFIGURATION_REG_BLOCK_ENABLE) :
		       (val & ~(u32)IGU_BLOCK_CONFIGURATION_REG_BLOCK_ENABLE));
	}
8080

M
Merav Sicron 已提交
8081
	DP(NETIF_MSG_HW | NETIF_MSG_IFUP, "%s gates #2, #3 and #4\n",
8082 8083 8084 8085 8086 8087 8088 8089 8090 8091 8092 8093 8094 8095
		close ? "closing" : "opening");
	mmiowb();
}

#define SHARED_MF_CLP_MAGIC  0x80000000 /* `magic' bit */

static void bnx2x_clp_reset_prep(struct bnx2x *bp, u32 *magic_val)
{
	/* Do some magic... */
	u32 val = MF_CFG_RD(bp, shared_mf_config.clp_mb);
	*magic_val = val & SHARED_MF_CLP_MAGIC;
	MF_CFG_WR(bp, shared_mf_config.clp_mb, val | SHARED_MF_CLP_MAGIC);
}

8096 8097
/**
 * bnx2x_clp_reset_done - restore the value of the `magic' bit.
8098
 *
8099 8100
 * @bp:		driver handle
 * @magic_val:	old value of the `magic' bit.
8101 8102 8103 8104 8105 8106 8107 8108 8109
 */
static void bnx2x_clp_reset_done(struct bnx2x *bp, u32 magic_val)
{
	/* Restore the `magic' bit value... */
	u32 val = MF_CFG_RD(bp, shared_mf_config.clp_mb);
	MF_CFG_WR(bp, shared_mf_config.clp_mb,
		(val & (~SHARED_MF_CLP_MAGIC)) | magic_val);
}

D
Dmitry Kravkov 已提交
8110
/**
8111
 * bnx2x_reset_mcp_prep - prepare for MCP reset.
8112
 *
8113 8114 8115 8116
 * @bp:		driver handle
 * @magic_val:	old value of 'magic' bit.
 *
 * Takes care of CLP configurations.
8117 8118 8119 8120 8121 8122
 */
static void bnx2x_reset_mcp_prep(struct bnx2x *bp, u32 *magic_val)
{
	u32 shmem;
	u32 validity_offset;

M
Merav Sicron 已提交
8123
	DP(NETIF_MSG_HW | NETIF_MSG_IFUP, "Starting\n");
8124 8125 8126 8127 8128 8129 8130 8131 8132 8133 8134 8135 8136 8137 8138 8139 8140

	/* Set `magic' bit in order to save MF config */
	if (!CHIP_IS_E1(bp))
		bnx2x_clp_reset_prep(bp, magic_val);

	/* Get shmem offset */
	shmem = REG_RD(bp, MISC_REG_SHARED_MEM_ADDR);
	validity_offset = offsetof(struct shmem_region, validity_map[0]);

	/* Clear validity map flags */
	if (shmem > 0)
		REG_WR(bp, shmem + validity_offset, 0);
}

#define MCP_TIMEOUT      5000   /* 5 seconds (in ms) */
#define MCP_ONE_TIMEOUT  100    /* 100 ms */

8141 8142
/**
 * bnx2x_mcp_wait_one - wait for MCP_ONE_TIMEOUT
8143
 *
8144
 * @bp:	driver handle
8145 8146 8147 8148 8149 8150 8151 8152 8153 8154 8155
 */
static inline void bnx2x_mcp_wait_one(struct bnx2x *bp)
{
	/* special handling for emulation and FPGA,
	   wait 10 times longer */
	if (CHIP_REV_IS_SLOW(bp))
		msleep(MCP_ONE_TIMEOUT*10);
	else
		msleep(MCP_ONE_TIMEOUT);
}

8156 8157 8158 8159
/*
 * initializes bp->common.shmem_base and waits for validity signature to appear
 */
static int bnx2x_init_shmem(struct bnx2x *bp)
8160
{
8161 8162
	int cnt = 0;
	u32 val = 0;
8163

8164 8165 8166 8167 8168 8169 8170
	do {
		bp->common.shmem_base = REG_RD(bp, MISC_REG_SHARED_MEM_ADDR);
		if (bp->common.shmem_base) {
			val = SHMEM_RD(bp, validity_map[BP_PORT(bp)]);
			if (val & SHR_MEM_VALIDITY_MB)
				return 0;
		}
8171

8172
		bnx2x_mcp_wait_one(bp);
8173

8174
	} while (cnt++ < (MCP_TIMEOUT / MCP_ONE_TIMEOUT));
8175

8176
	BNX2X_ERR("BAD MCP validity signature\n");
8177

8178 8179
	return -ENODEV;
}
8180

8181 8182 8183
static int bnx2x_reset_mcp_comp(struct bnx2x *bp, u32 magic_val)
{
	int rc = bnx2x_init_shmem(bp);
8184 8185 8186 8187 8188 8189 8190 8191 8192 8193 8194 8195 8196 8197 8198 8199 8200 8201 8202 8203 8204 8205 8206 8207 8208 8209 8210

	/* Restore the `magic' bit value */
	if (!CHIP_IS_E1(bp))
		bnx2x_clp_reset_done(bp, magic_val);

	return rc;
}

static void bnx2x_pxp_prep(struct bnx2x *bp)
{
	if (!CHIP_IS_E1(bp)) {
		REG_WR(bp, PXP2_REG_RD_START_INIT, 0);
		REG_WR(bp, PXP2_REG_RQ_RBC_DONE, 0);
		mmiowb();
	}
}

/*
 * Reset the whole chip except for:
 *      - PCIE core
 *      - PCI Glue, PSWHST, PXP/PXP2 RF (all controlled by
 *              one reset bit)
 *      - IGU
 *      - MISC (including AEU)
 *      - GRC
 *      - RBCN, RBCP
 */
8211
static void bnx2x_process_kill_chip_reset(struct bnx2x *bp, bool global)
8212 8213
{
	u32 not_reset_mask1, reset_mask1, not_reset_mask2, reset_mask2;
8214
	u32 global_bits2, stay_reset2;
8215 8216 8217 8218 8219 8220 8221 8222

	/*
	 * Bits that have to be set in reset_mask2 if we want to reset 'global'
	 * (per chip) blocks.
	 */
	global_bits2 =
		MISC_REGISTERS_RESET_REG_2_RST_MCP_N_RESET_CMN_CPU |
		MISC_REGISTERS_RESET_REG_2_RST_MCP_N_RESET_CMN_CORE;
8223

8224
	/* Don't reset the following blocks */
8225 8226 8227 8228 8229 8230
	not_reset_mask1 =
		MISC_REGISTERS_RESET_REG_1_RST_HC |
		MISC_REGISTERS_RESET_REG_1_RST_PXPV |
		MISC_REGISTERS_RESET_REG_1_RST_PXP;

	not_reset_mask2 =
8231
		MISC_REGISTERS_RESET_REG_2_RST_PCI_MDIO |
8232 8233 8234 8235 8236 8237
		MISC_REGISTERS_RESET_REG_2_RST_EMAC0_HARD_CORE |
		MISC_REGISTERS_RESET_REG_2_RST_EMAC1_HARD_CORE |
		MISC_REGISTERS_RESET_REG_2_RST_MISC_CORE |
		MISC_REGISTERS_RESET_REG_2_RST_RBCN |
		MISC_REGISTERS_RESET_REG_2_RST_GRC  |
		MISC_REGISTERS_RESET_REG_2_RST_MCP_N_RESET_REG_HARD_CORE |
8238 8239 8240
		MISC_REGISTERS_RESET_REG_2_RST_MCP_N_HARD_CORE_RST_B |
		MISC_REGISTERS_RESET_REG_2_RST_ATC |
		MISC_REGISTERS_RESET_REG_2_PGLC;
8241

8242 8243 8244 8245 8246 8247 8248 8249 8250 8251 8252 8253 8254 8255 8256
	/*
	 * Keep the following blocks in reset:
	 *  - all xxMACs are handled by the bnx2x_link code.
	 */
	stay_reset2 =
		MISC_REGISTERS_RESET_REG_2_RST_BMAC0 |
		MISC_REGISTERS_RESET_REG_2_RST_BMAC1 |
		MISC_REGISTERS_RESET_REG_2_RST_EMAC0 |
		MISC_REGISTERS_RESET_REG_2_RST_EMAC1 |
		MISC_REGISTERS_RESET_REG_2_UMAC0 |
		MISC_REGISTERS_RESET_REG_2_UMAC1 |
		MISC_REGISTERS_RESET_REG_2_XMAC |
		MISC_REGISTERS_RESET_REG_2_XMAC_SOFT;

	/* Full reset masks according to the chip */
8257 8258 8259 8260
	reset_mask1 = 0xffffffff;

	if (CHIP_IS_E1(bp))
		reset_mask2 = 0xffff;
8261
	else if (CHIP_IS_E1H(bp))
8262
		reset_mask2 = 0x1ffff;
8263 8264 8265 8266
	else if (CHIP_IS_E2(bp))
		reset_mask2 = 0xfffff;
	else /* CHIP_IS_E3 */
		reset_mask2 = 0x3ffffff;
8267 8268 8269 8270 8271 8272 8273 8274 8275 8276 8277 8278 8279 8280 8281 8282 8283 8284 8285

	/* Don't reset global blocks unless we need to */
	if (!global)
		reset_mask2 &= ~global_bits2;

	/*
	 * In case of attention in the QM, we need to reset PXP
	 * (MISC_REGISTERS_RESET_REG_2_RST_PXP_RQ_RD_WR) before QM
	 * because otherwise QM reset would release 'close the gates' shortly
	 * before resetting the PXP, then the PSWRQ would send a write
	 * request to PGLUE. Then when PXP is reset, PGLUE would try to
	 * read the payload data from PSWWR, but PSWWR would not
	 * respond. The write queue in PGLUE would stuck, dmae commands
	 * would not return. Therefore it's important to reset the second
	 * reset register (containing the
	 * MISC_REGISTERS_RESET_REG_2_RST_PXP_RQ_RD_WR bit) before the
	 * first one (containing the MISC_REGISTERS_RESET_REG_1_RST_QM
	 * bit).
	 */
8286 8287 8288
	REG_WR(bp, GRCBASE_MISC + MISC_REGISTERS_RESET_REG_2_CLEAR,
	       reset_mask2 & (~not_reset_mask2));

8289 8290 8291
	REG_WR(bp, GRCBASE_MISC + MISC_REGISTERS_RESET_REG_1_CLEAR,
	       reset_mask1 & (~not_reset_mask1));

8292 8293 8294
	barrier();
	mmiowb();

8295 8296 8297 8298 8299 8300
	REG_WR(bp, GRCBASE_MISC + MISC_REGISTERS_RESET_REG_2_SET,
	       reset_mask2 & (~stay_reset2));

	barrier();
	mmiowb();

8301
	REG_WR(bp, GRCBASE_MISC + MISC_REGISTERS_RESET_REG_1_SET, reset_mask1);
8302 8303 8304
	mmiowb();
}

8305 8306 8307 8308 8309 8310 8311 8312 8313 8314 8315 8316 8317 8318 8319 8320 8321 8322 8323 8324 8325 8326 8327 8328 8329 8330 8331 8332 8333 8334 8335 8336 8337
/**
 * bnx2x_er_poll_igu_vq - poll for pending writes bit.
 * It should get cleared in no more than 1s.
 *
 * @bp:	driver handle
 *
 * It should get cleared in no more than 1s. Returns 0 if
 * pending writes bit gets cleared.
 */
static int bnx2x_er_poll_igu_vq(struct bnx2x *bp)
{
	u32 cnt = 1000;
	u32 pend_bits = 0;

	do {
		pend_bits  = REG_RD(bp, IGU_REG_PENDING_BITS_STATUS);

		if (pend_bits == 0)
			break;

		usleep_range(1000, 1000);
	} while (cnt-- > 0);

	if (cnt <= 0) {
		BNX2X_ERR("Still pending IGU requests pend_bits=%x!\n",
			  pend_bits);
		return -EBUSY;
	}

	return 0;
}

static int bnx2x_process_kill(struct bnx2x *bp, bool global)
8338 8339 8340 8341 8342 8343 8344 8345 8346 8347 8348 8349 8350 8351 8352 8353 8354 8355
{
	int cnt = 1000;
	u32 val = 0;
	u32 sr_cnt, blk_cnt, port_is_idle_0, port_is_idle_1, pgl_exp_rom2;


	/* Empty the Tetris buffer, wait for 1s */
	do {
		sr_cnt  = REG_RD(bp, PXP2_REG_RD_SR_CNT);
		blk_cnt = REG_RD(bp, PXP2_REG_RD_BLK_CNT);
		port_is_idle_0 = REG_RD(bp, PXP2_REG_RD_PORT_IS_IDLE_0);
		port_is_idle_1 = REG_RD(bp, PXP2_REG_RD_PORT_IS_IDLE_1);
		pgl_exp_rom2 = REG_RD(bp, PXP2_REG_PGL_EXP_ROM2);
		if ((sr_cnt == 0x7e) && (blk_cnt == 0xa0) &&
		    ((port_is_idle_0 & 0x1) == 0x1) &&
		    ((port_is_idle_1 & 0x1) == 0x1) &&
		    (pgl_exp_rom2 == 0xffffffff))
			break;
8356
		usleep_range(1000, 1000);
8357 8358 8359
	} while (cnt-- > 0);

	if (cnt <= 0) {
M
Merav Sicron 已提交
8360 8361
		BNX2X_ERR("Tetris buffer didn't get empty or there are still outstanding read requests after 1s!\n");
		BNX2X_ERR("sr_cnt=0x%08x, blk_cnt=0x%08x, port_is_idle_0=0x%08x, port_is_idle_1=0x%08x, pgl_exp_rom2=0x%08x\n",
8362 8363 8364 8365 8366 8367 8368 8369 8370 8371
			  sr_cnt, blk_cnt, port_is_idle_0, port_is_idle_1,
			  pgl_exp_rom2);
		return -EAGAIN;
	}

	barrier();

	/* Close gates #2, #3 and #4 */
	bnx2x_set_234_gates(bp, true);

8372 8373 8374 8375 8376
	/* Poll for IGU VQs for 57712 and newer chips */
	if (!CHIP_IS_E1x(bp) && bnx2x_er_poll_igu_vq(bp))
		return -EAGAIN;


8377 8378 8379 8380 8381 8382 8383 8384 8385 8386 8387 8388
	/* TBD: Indicate that "process kill" is in progress to MCP */

	/* Clear "unprepared" bit */
	REG_WR(bp, MISC_REG_UNPREPARED, 0);
	barrier();

	/* Make sure all is written to the chip before the reset */
	mmiowb();

	/* Wait for 1ms to empty GLUE and PCI-E core queues,
	 * PSWHST, GRC and PSWRD Tetris buffer.
	 */
8389
	usleep_range(1000, 1000);
8390 8391 8392

	/* Prepare to chip reset: */
	/* MCP */
8393 8394
	if (global)
		bnx2x_reset_mcp_prep(bp, &val);
8395 8396 8397 8398 8399 8400

	/* PXP */
	bnx2x_pxp_prep(bp);
	barrier();

	/* reset the chip */
8401
	bnx2x_process_kill_chip_reset(bp, global);
8402 8403 8404 8405
	barrier();

	/* Recover after reset: */
	/* MCP */
8406
	if (global && bnx2x_reset_mcp_comp(bp, val))
8407 8408
		return -EAGAIN;

8409 8410
	/* TBD: Add resetting the NO_MCP mode DB here */

8411 8412 8413 8414 8415 8416 8417 8418 8419
	/* PXP */
	bnx2x_pxp_prep(bp);

	/* Open the gates #2, #3 and #4 */
	bnx2x_set_234_gates(bp, false);

	/* TBD: IGU/AEU preparation bring back the AEU/IGU to a
	 * reset state, re-enable attentions. */

E
Eliezer Tamir 已提交
8420 8421 8422
	return 0;
}

8423
int bnx2x_leader_reset(struct bnx2x *bp)
8424 8425
{
	int rc = 0;
8426
	bool global = bnx2x_reset_is_global(bp);
A
Ariel Elior 已提交
8427 8428 8429 8430 8431 8432 8433 8434 8435 8436 8437 8438 8439 8440 8441 8442 8443 8444 8445 8446 8447 8448 8449 8450 8451
	u32 load_code;

	/* if not going to reset MCP - load "fake" driver to reset HW while
	 * driver is owner of the HW
	 */
	if (!global && !BP_NOMCP(bp)) {
		load_code = bnx2x_fw_command(bp, DRV_MSG_CODE_LOAD_REQ, 0);
		if (!load_code) {
			BNX2X_ERR("MCP response failure, aborting\n");
			rc = -EAGAIN;
			goto exit_leader_reset;
		}
		if ((load_code != FW_MSG_CODE_DRV_LOAD_COMMON_CHIP) &&
		    (load_code != FW_MSG_CODE_DRV_LOAD_COMMON)) {
			BNX2X_ERR("MCP unexpected resp, aborting\n");
			rc = -EAGAIN;
			goto exit_leader_reset2;
		}
		load_code = bnx2x_fw_command(bp, DRV_MSG_CODE_LOAD_DONE, 0);
		if (!load_code) {
			BNX2X_ERR("MCP response failure, aborting\n");
			rc = -EAGAIN;
			goto exit_leader_reset2;
		}
	}
8452

8453
	/* Try to recover after the failure */
8454
	if (bnx2x_process_kill(bp, global)) {
M
Merav Sicron 已提交
8455 8456
		BNX2X_ERR("Something bad had happen on engine %d! Aii!\n",
			  BP_PATH(bp));
8457
		rc = -EAGAIN;
A
Ariel Elior 已提交
8458
		goto exit_leader_reset2;
8459 8460
	}

8461 8462 8463 8464
	/*
	 * Clear RESET_IN_PROGRES and RESET_GLOBAL bits and update the driver
	 * state.
	 */
8465
	bnx2x_set_reset_done(bp);
8466 8467
	if (global)
		bnx2x_clear_reset_global(bp);
8468

A
Ariel Elior 已提交
8469 8470 8471 8472 8473 8474
exit_leader_reset2:
	/* unload "fake driver" if it was loaded */
	if (!global && !BP_NOMCP(bp)) {
		bnx2x_fw_command(bp, DRV_MSG_CODE_UNLOAD_REQ_WOL_MCP, 0);
		bnx2x_fw_command(bp, DRV_MSG_CODE_UNLOAD_DONE, 0);
	}
8475 8476
exit_leader_reset:
	bp->is_leader = 0;
8477 8478
	bnx2x_release_leader_lock(bp);
	smp_mb();
8479 8480 8481
	return rc;
}

8482 8483 8484 8485 8486 8487 8488 8489 8490 8491 8492 8493 8494 8495 8496 8497 8498 8499 8500 8501 8502 8503 8504
static inline void bnx2x_recovery_failed(struct bnx2x *bp)
{
	netdev_err(bp->dev, "Recovery has failed. Power cycle is needed.\n");

	/* Disconnect this device */
	netif_device_detach(bp->dev);

	/*
	 * Block ifup for all function on this engine until "process kill"
	 * or power cycle.
	 */
	bnx2x_set_reset_in_progress(bp);

	/* Shut down the power */
	bnx2x_set_power_state(bp, PCI_D3hot);

	bp->recovery_state = BNX2X_RECOVERY_FAILED;

	smp_mb();
}

/*
 * Assumption: runs under rtnl lock. This together with the fact
8505
 * that it's called only from bnx2x_sp_rtnl() ensure that it
8506 8507 8508 8509
 * will never be called when netif_running(bp->dev) is false.
 */
static void bnx2x_parity_recover(struct bnx2x *bp)
{
8510
	bool global = false;
8511
	u32 error_recovered, error_unrecovered;
A
Ariel Elior 已提交
8512
	bool is_parity;
8513

8514 8515 8516 8517 8518
	DP(NETIF_MSG_HW, "Handling parity\n");
	while (1) {
		switch (bp->recovery_state) {
		case BNX2X_RECOVERY_INIT:
			DP(NETIF_MSG_HW, "State is BNX2X_RECOVERY_INIT\n");
A
Ariel Elior 已提交
8519 8520
			is_parity = bnx2x_chk_parity_attn(bp, &global, false);
			WARN_ON(!is_parity);
8521

8522
			/* Try to get a LEADER_LOCK HW lock */
8523 8524 8525 8526 8527 8528 8529 8530 8531 8532 8533
			if (bnx2x_trylock_leader_lock(bp)) {
				bnx2x_set_reset_in_progress(bp);
				/*
				 * Check if there is a global attention and if
				 * there was a global attention, set the global
				 * reset bit.
				 */

				if (global)
					bnx2x_set_reset_global(bp);

8534
				bp->is_leader = 1;
8535
			}
8536 8537 8538 8539 8540 8541 8542

			/* Stop the driver */
			/* If interface has been removed - break */
			if (bnx2x_nic_unload(bp, UNLOAD_RECOVERY))
				return;

			bp->recovery_state = BNX2X_RECOVERY_WAIT;
8543 8544 8545 8546

			/* Ensure "is_leader", MCP command sequence and
			 * "recovery_state" update values are seen on other
			 * CPUs.
8547
			 */
8548
			smp_mb();
8549 8550 8551 8552 8553
			break;

		case BNX2X_RECOVERY_WAIT:
			DP(NETIF_MSG_HW, "State is BNX2X_RECOVERY_WAIT\n");
			if (bp->is_leader) {
8554
				int other_engine = BP_PATH(bp) ? 0 : 1;
8555 8556 8557 8558
				bool other_load_status =
					bnx2x_get_load_status(bp, other_engine);
				bool load_status =
					bnx2x_get_load_status(bp, BP_PATH(bp));
8559 8560 8561 8562 8563 8564 8565 8566 8567 8568
				global = bnx2x_reset_is_global(bp);

				/*
				 * In case of a parity in a global block, let
				 * the first leader that performs a
				 * leader_reset() reset the global blocks in
				 * order to clear global attentions. Otherwise
				 * the the gates will remain closed for that
				 * engine.
				 */
8569 8570
				if (load_status ||
				    (global && other_load_status)) {
8571 8572 8573
					/* Wait until all other functions get
					 * down.
					 */
8574
					schedule_delayed_work(&bp->sp_rtnl_task,
8575 8576 8577 8578 8579 8580 8581 8582
								HZ/10);
					return;
				} else {
					/* If all other functions got down -
					 * try to bring the chip back to
					 * normal. In any case it's an exit
					 * point for a leader.
					 */
8583 8584
					if (bnx2x_leader_reset(bp)) {
						bnx2x_recovery_failed(bp);
8585 8586 8587
						return;
					}

8588 8589 8590 8591 8592 8593
					/* If we are here, means that the
					 * leader has succeeded and doesn't
					 * want to be a leader any more. Try
					 * to continue as a none-leader.
					 */
					break;
8594 8595
				}
			} else { /* non-leader */
8596
				if (!bnx2x_reset_is_done(bp, BP_PATH(bp))) {
8597 8598 8599 8600 8601 8602
					/* Try to get a LEADER_LOCK HW lock as
					 * long as a former leader may have
					 * been unloaded by the user or
					 * released a leadership by another
					 * reason.
					 */
8603
					if (bnx2x_trylock_leader_lock(bp)) {
8604 8605 8606 8607 8608 8609 8610
						/* I'm a leader now! Restart a
						 * switch case.
						 */
						bp->is_leader = 1;
						break;
					}

8611
					schedule_delayed_work(&bp->sp_rtnl_task,
8612 8613 8614
								HZ/10);
					return;

8615 8616 8617 8618 8619 8620 8621
				} else {
					/*
					 * If there was a global attention, wait
					 * for it to be cleared.
					 */
					if (bnx2x_reset_is_global(bp)) {
						schedule_delayed_work(
8622 8623
							&bp->sp_rtnl_task,
							HZ/10);
8624 8625 8626
						return;
					}

8627 8628 8629 8630
					error_recovered =
					  bp->eth_stats.recoverable_error;
					error_unrecovered =
					  bp->eth_stats.unrecoverable_error;
A
Ariel Elior 已提交
8631 8632 8633
					bp->recovery_state =
						BNX2X_RECOVERY_NIC_LOADING;
					if (bnx2x_nic_load(bp, LOAD_NORMAL)) {
8634
						error_unrecovered++;
A
Ariel Elior 已提交
8635
						netdev_err(bp->dev,
M
Merav Sicron 已提交
8636
							   "Recovery failed. Power cycle needed\n");
A
Ariel Elior 已提交
8637 8638 8639 8640 8641 8642 8643
						/* Disconnect this device */
						netif_device_detach(bp->dev);
						/* Shut down the power */
						bnx2x_set_power_state(
							bp, PCI_D3hot);
						smp_mb();
					} else {
8644 8645
						bp->recovery_state =
							BNX2X_RECOVERY_DONE;
8646
						error_recovered++;
8647 8648
						smp_mb();
					}
8649 8650 8651 8652
					bp->eth_stats.recoverable_error =
						error_recovered;
					bp->eth_stats.unrecoverable_error =
						error_unrecovered;
8653

8654 8655 8656 8657 8658 8659 8660 8661 8662
					return;
				}
			}
		default:
			return;
		}
	}
}

8663 8664
static int bnx2x_close(struct net_device *dev);

8665 8666 8667
/* bnx2x_nic_unload() flushes the bnx2x_wq, thus reset task is
 * scheduled on a general queue in order to prevent a dead lock.
 */
8668
static void bnx2x_sp_rtnl_task(struct work_struct *work)
8669
{
8670
	struct bnx2x *bp = container_of(work, struct bnx2x, sp_rtnl_task.work);
8671 8672 8673 8674

	rtnl_lock();

	if (!netif_running(bp->dev))
8675 8676 8677 8678
		goto sp_rtnl_exit;

	/* if stop on error is defined no recovery flows should be executed */
#ifdef BNX2X_STOP_ON_ERROR
M
Merav Sicron 已提交
8679
	BNX2X_ERR("recovery flow called but STOP_ON_ERROR defined so reset not done to allow debug dump,\n"
8680
		  "you will need to reboot when done\n");
8681
	goto sp_rtnl_not_reset;
8682
#endif
8683

8684 8685
	if (unlikely(bp->recovery_state != BNX2X_RECOVERY_DONE)) {
		/*
8686 8687
		 * Clear all pending SP commands as we are going to reset the
		 * function anyway.
8688
		 */
8689 8690 8691
		bp->sp_rtnl_state = 0;
		smp_mb();

8692
		bnx2x_parity_recover(bp);
8693 8694 8695 8696 8697 8698 8699 8700 8701 8702 8703 8704

		goto sp_rtnl_exit;
	}

	if (test_and_clear_bit(BNX2X_SP_RTNL_TX_TIMEOUT, &bp->sp_rtnl_state)) {
		/*
		 * Clear all pending SP commands as we are going to reset the
		 * function anyway.
		 */
		bp->sp_rtnl_state = 0;
		smp_mb();

8705 8706
		bnx2x_nic_unload(bp, UNLOAD_NORMAL);
		bnx2x_nic_load(bp, LOAD_NORMAL);
8707 8708

		goto sp_rtnl_exit;
8709
	}
8710 8711 8712 8713 8714
#ifdef BNX2X_STOP_ON_ERROR
sp_rtnl_not_reset:
#endif
	if (test_and_clear_bit(BNX2X_SP_RTNL_SETUP_TC, &bp->sp_rtnl_state))
		bnx2x_setup_tc(bp->dev, bp->dcbx_port_params.ets.num_of_cos);
8715

8716 8717 8718 8719 8720 8721
	/*
	 * in case of fan failure we need to reset id if the "stop on error"
	 * debug flag is set, since we trying to prevent permanent overheating
	 * damage
	 */
	if (test_and_clear_bit(BNX2X_SP_RTNL_FAN_FAILURE, &bp->sp_rtnl_state)) {
M
Merav Sicron 已提交
8722
		DP(NETIF_MSG_HW, "fan failure detected. Unloading driver\n");
8723 8724 8725 8726
		netif_device_detach(bp->dev);
		bnx2x_close(bp->dev);
	}

8727
sp_rtnl_exit:
8728 8729 8730
	rtnl_unlock();
}

E
Eliezer Tamir 已提交
8731 8732
/* end of nic load/unload */

8733 8734 8735 8736 8737 8738 8739 8740 8741 8742 8743 8744 8745 8746 8747 8748 8749 8750 8751 8752 8753 8754 8755 8756 8757 8758 8759 8760 8761 8762 8763
static void bnx2x_period_task(struct work_struct *work)
{
	struct bnx2x *bp = container_of(work, struct bnx2x, period_task.work);

	if (!netif_running(bp->dev))
		goto period_task_exit;

	if (CHIP_REV_IS_SLOW(bp)) {
		BNX2X_ERR("period task called on emulation, ignoring\n");
		goto period_task_exit;
	}

	bnx2x_acquire_phy_lock(bp);
	/*
	 * The barrier is needed to ensure the ordering between the writing to
	 * the bp->port.pmf in the bnx2x_nic_load() or bnx2x_pmf_update() and
	 * the reading here.
	 */
	smp_mb();
	if (bp->port.pmf) {
		bnx2x_period_func(&bp->link_params, &bp->link_vars);

		/* Re-queue task in 1 sec */
		queue_delayed_work(bnx2x_wq, &bp->period_task, 1*HZ);
	}

	bnx2x_release_phy_lock(bp);
period_task_exit:
	return;
}

E
Eliezer Tamir 已提交
8764 8765 8766 8767
/*
 * Init service functions
 */

8768
static u32 bnx2x_get_pretend_reg(struct bnx2x *bp)
D
Dmitry Kravkov 已提交
8769 8770 8771 8772
{
	u32 base = PXP2_REG_PGL_PRETEND_FUNC_F0;
	u32 stride = PXP2_REG_PGL_PRETEND_FUNC_F1 - base;
	return base + (BP_ABS_FUNC(bp)) * stride;
8773 8774
}

D
Dmitry Kravkov 已提交
8775
static void bnx2x_undi_int_disable_e1h(struct bnx2x *bp)
8776
{
D
Dmitry Kravkov 已提交
8777
	u32 reg = bnx2x_get_pretend_reg(bp);
8778 8779 8780 8781 8782 8783

	/* Flush all outstanding writes */
	mmiowb();

	/* Pretend to be function 0 */
	REG_WR(bp, reg, 0);
D
Dmitry Kravkov 已提交
8784
	REG_RD(bp, reg);	/* Flush the GRC transaction (in the chip) */
8785 8786 8787 8788 8789 8790 8791

	/* From now we are in the "like-E1" mode */
	bnx2x_int_disable(bp);

	/* Flush all outstanding writes */
	mmiowb();

D
Dmitry Kravkov 已提交
8792 8793 8794
	/* Restore the original function */
	REG_WR(bp, reg, BP_ABS_FUNC(bp));
	REG_RD(bp, reg);
8795 8796
}

D
Dmitry Kravkov 已提交
8797
static inline void bnx2x_undi_int_disable(struct bnx2x *bp)
8798
{
D
Dmitry Kravkov 已提交
8799
	if (CHIP_IS_E1(bp))
8800
		bnx2x_int_disable(bp);
D
Dmitry Kravkov 已提交
8801 8802
	else
		bnx2x_undi_int_disable_e1h(bp);
8803 8804
}

8805 8806 8807 8808
static void __devinit bnx2x_undi_unload(struct bnx2x *bp)
{
	u32 val;

A
Ariel Elior 已提交
8809 8810 8811 8812 8813 8814
	/* possibly another driver is trying to reset the chip */
	bnx2x_acquire_hw_lock(bp, HW_LOCK_RESOURCE_RESET);

	/* check if doorbell queue is reset */
	if (REG_RD(bp, GRCBASE_MISC + MISC_REGISTERS_RESET_REG_1_SET)
	    & MISC_REGISTERS_RESET_REG_1_RST_DORQ) {
8815 8816 8817

		/*
		 * Check if it is the UNDI driver
8818 8819 8820 8821 8822
		 * UNDI driver initializes CID offset for normal bell to 0x7
		 */
		val = REG_RD(bp, DORQ_REG_NORM_CID_OFST);
		if (val == 0x7) {
			u32 reset_code = DRV_MSG_CODE_UNLOAD_REQ_WOL_DIS;
D
Dmitry Kravkov 已提交
8823 8824
			/* save our pf_num */
			int orig_pf_num = bp->pf_num;
8825 8826
			int port;
			u32 swap_en, swap_val, value;
8827

8828 8829 8830
			/* clear the UNDI indication */
			REG_WR(bp, DORQ_REG_NORM_CID_OFST, 0);

8831 8832 8833
			BNX2X_DEV_INFO("UNDI is active! reset device\n");

			/* try unload UNDI on port 0 */
D
Dmitry Kravkov 已提交
8834
			bp->pf_num = 0;
8835
			bp->fw_seq =
D
Dmitry Kravkov 已提交
8836
			      (SHMEM_RD(bp, func_mb[bp->pf_num].drv_mb_header) &
8837
				DRV_MSG_SEQ_NUMBER_MASK);
Y
Yaniv Rosner 已提交
8838
			reset_code = bnx2x_fw_command(bp, reset_code, 0);
8839 8840 8841 8842

			/* if UNDI is loaded on the other port */
			if (reset_code != FW_MSG_CODE_DRV_UNLOAD_COMMON) {

8843
				/* send "DONE" for previous unload */
Y
Yaniv Rosner 已提交
8844 8845
				bnx2x_fw_command(bp,
						 DRV_MSG_CODE_UNLOAD_DONE, 0);
8846 8847

				/* unload UNDI on port 1 */
D
Dmitry Kravkov 已提交
8848
				bp->pf_num = 1;
8849
				bp->fw_seq =
D
Dmitry Kravkov 已提交
8850
			      (SHMEM_RD(bp, func_mb[bp->pf_num].drv_mb_header) &
8851 8852 8853
					DRV_MSG_SEQ_NUMBER_MASK);
				reset_code = DRV_MSG_CODE_UNLOAD_REQ_WOL_DIS;

Y
Yaniv Rosner 已提交
8854
				bnx2x_fw_command(bp, reset_code, 0);
8855 8856
			}

D
Dmitry Kravkov 已提交
8857
			bnx2x_undi_int_disable(bp);
8858
			port = BP_PORT(bp);
8859 8860 8861

			/* close input traffic and wait for it */
			/* Do not rcv packets to BRB */
8862 8863
			REG_WR(bp, (port ? NIG_REG_LLH1_BRB1_DRV_MASK :
					   NIG_REG_LLH0_BRB1_DRV_MASK), 0x0);
8864 8865
			/* Do not direct rcv packets that are not for MCP to
			 * the BRB */
8866 8867
			REG_WR(bp, (port ? NIG_REG_LLH1_BRB1_NOT_MCP :
					   NIG_REG_LLH0_BRB1_NOT_MCP), 0x0);
8868
			/* clear AEU */
8869 8870
			REG_WR(bp, (port ? MISC_REG_AEU_MASK_ATTN_FUNC_1 :
					   MISC_REG_AEU_MASK_ATTN_FUNC_0), 0);
8871 8872 8873 8874 8875
			msleep(10);

			/* save NIG port swap info */
			swap_val = REG_RD(bp, NIG_REG_PORT_SWAP);
			swap_en = REG_RD(bp, NIG_REG_STRAP_OVERRIDE);
8876 8877 8878
			/* reset device */
			REG_WR(bp,
			       GRCBASE_MISC + MISC_REGISTERS_RESET_REG_1_CLEAR,
8879
			       0xd3ffffff);
8880 8881 8882 8883 8884 8885 8886

			value = 0x1400;
			if (CHIP_IS_E3(bp)) {
				value |= MISC_REGISTERS_RESET_REG_2_MSTAT0;
				value |= MISC_REGISTERS_RESET_REG_2_MSTAT1;
			}

8887 8888
			REG_WR(bp,
			       GRCBASE_MISC + MISC_REGISTERS_RESET_REG_2_CLEAR,
8889 8890
			       value);

8891 8892 8893 8894 8895 8896 8897 8898
			/* take the NIG out of reset and restore swap values */
			REG_WR(bp,
			       GRCBASE_MISC + MISC_REGISTERS_RESET_REG_1_SET,
			       MISC_REGISTERS_RESET_REG_1_RST_NIG);
			REG_WR(bp, NIG_REG_PORT_SWAP, swap_val);
			REG_WR(bp, NIG_REG_STRAP_OVERRIDE, swap_en);

			/* send unload done to the MCP */
Y
Yaniv Rosner 已提交
8899
			bnx2x_fw_command(bp, DRV_MSG_CODE_UNLOAD_DONE, 0);
8900 8901

			/* restore our func and fw_seq */
D
Dmitry Kravkov 已提交
8902
			bp->pf_num = orig_pf_num;
8903
		}
8904
	}
A
Ariel Elior 已提交
8905 8906 8907

	/* now it's safe to release the lock */
	bnx2x_release_hw_lock(bp, HW_LOCK_RESOURCE_RESET);
8908 8909 8910 8911
}

static void __devinit bnx2x_get_common_hwinfo(struct bnx2x *bp)
{
8912
	u32 val, val2, val3, val4, id, boot_mode;
E
Eilon Greenstein 已提交
8913
	u16 pmc;
8914 8915 8916 8917 8918 8919 8920 8921 8922

	/* Get the chip revision id and number. */
	/* chip num:16-31, rev:12-15, metal:4-11, bond_id:0-3 */
	val = REG_RD(bp, MISC_REG_CHIP_NUM);
	id = ((val & 0xffff) << 16);
	val = REG_RD(bp, MISC_REG_CHIP_REV);
	id |= ((val & 0xf) << 12);
	val = REG_RD(bp, MISC_REG_CHIP_METAL);
	id |= ((val & 0xff) << 4);
E
Eilon Greenstein 已提交
8923
	val = REG_RD(bp, MISC_REG_BOND_ID);
8924 8925
	id |= (val & 0xf);
	bp->common.chip_id = id;
8926 8927 8928 8929

	/* Set doorbell size */
	bp->db_size = (1 << BNX2X_DB_SHIFT);

8930
	if (!CHIP_IS_E1x(bp)) {
D
Dmitry Kravkov 已提交
8931 8932 8933 8934 8935 8936 8937 8938 8939 8940 8941 8942 8943 8944 8945 8946 8947 8948 8949
		val = REG_RD(bp, MISC_REG_PORT4MODE_EN_OVWR);
		if ((val & 1) == 0)
			val = REG_RD(bp, MISC_REG_PORT4MODE_EN);
		else
			val = (val >> 1) & 1;
		BNX2X_DEV_INFO("chip is in %s\n", val ? "4_PORT_MODE" :
						       "2_PORT_MODE");
		bp->common.chip_port_mode = val ? CHIP_4_PORT_MODE :
						 CHIP_2_PORT_MODE;

		if (CHIP_MODE_IS_4_PORT(bp))
			bp->pfid = (bp->pf_num >> 1);	/* 0..3 */
		else
			bp->pfid = (bp->pf_num & 0x6);	/* 0, 2, 4, 6 */
	} else {
		bp->common.chip_port_mode = CHIP_PORT_MODE_NONE; /* N/A */
		bp->pfid = bp->pf_num;			/* 0..7 */
	}

M
Merav Sicron 已提交
8950 8951
	BNX2X_DEV_INFO("pf_id: %x", bp->pfid);

D
Dmitry Kravkov 已提交
8952 8953
	bp->link_params.chip_id = bp->common.chip_id;
	BNX2X_DEV_INFO("chip ID is 0x%x\n", id);
8954

8955 8956 8957 8958 8959 8960 8961
	val = (REG_RD(bp, 0x2874) & 0x55);
	if ((bp->common.chip_id & 0x1) ||
	    (CHIP_IS_E1(bp) && val) || (CHIP_IS_E1H(bp) && (val == 0x55))) {
		bp->flags |= ONE_PORT_FLAG;
		BNX2X_DEV_INFO("single port device\n");
	}

8962
	val = REG_RD(bp, MCP_REG_MCPR_NVM_CFG4);
D
Dmitry Kravkov 已提交
8963
	bp->common.flash_size = (BNX2X_NVRAM_1MB_SIZE <<
8964 8965 8966 8967
				 (val & MCPR_NVM_CFG4_FLASH_SIZE));
	BNX2X_DEV_INFO("flash_size 0x%x (%d)\n",
		       bp->common.flash_size, bp->common.flash_size);

8968 8969
	bnx2x_init_shmem(bp);

8970 8971


D
Dmitry Kravkov 已提交
8972 8973 8974
	bp->common.shmem2_base = REG_RD(bp, (BP_PATH(bp) ?
					MISC_REG_GENERIC_CR_1 :
					MISC_REG_GENERIC_CR_0));
8975

8976
	bp->link_params.shmem_base = bp->common.shmem_base;
Y
Yaniv Rosner 已提交
8977
	bp->link_params.shmem2_base = bp->common.shmem2_base;
8978 8979
	BNX2X_DEV_INFO("shmem offset 0x%x  shmem2 offset 0x%x\n",
		       bp->common.shmem_base, bp->common.shmem2_base);
8980

D
Dmitry Kravkov 已提交
8981
	if (!bp->common.shmem_base) {
8982 8983 8984 8985 8986 8987
		BNX2X_DEV_INFO("MCP not active\n");
		bp->flags |= NO_MCP_FLAG;
		return;
	}

	bp->common.hw_config = SHMEM_RD(bp, dev_info.shared_hw_config.config);
8988
	BNX2X_DEV_INFO("hw_config 0x%08x\n", bp->common.hw_config);
8989 8990 8991 8992 8993

	bp->link_params.hw_led_mode = ((bp->common.hw_config &
					SHARED_HW_CFG_LED_MODE_MASK) >>
				       SHARED_HW_CFG_LED_MODE_SHIFT);

8994 8995 8996 8997 8998 8999 9000 9001 9002
	bp->link_params.feature_config_flags = 0;
	val = SHMEM_RD(bp, dev_info.shared_feature_config.config);
	if (val & SHARED_FEAT_CFG_OVERRIDE_PREEMPHASIS_CFG_ENABLED)
		bp->link_params.feature_config_flags |=
				FEATURE_CONFIG_OVERRIDE_PREEMPHASIS_ENABLED;
	else
		bp->link_params.feature_config_flags &=
				~FEATURE_CONFIG_OVERRIDE_PREEMPHASIS_ENABLED;

9003 9004 9005 9006 9007 9008
	val = SHMEM_RD(bp, dev_info.bc_rev) >> 8;
	bp->common.bc_ver = val;
	BNX2X_DEV_INFO("bc_ver %X\n", val);
	if (val < BNX2X_BC_VER) {
		/* for now only warn
		 * later we might need to enforce this */
M
Merav Sicron 已提交
9009 9010
		BNX2X_ERR("This driver needs bc_ver %X but found %X, please upgrade BC\n",
			  BNX2X_BC_VER, val);
9011
	}
E
Eilon Greenstein 已提交
9012
	bp->link_params.feature_config_flags |=
Y
Yaniv Rosner 已提交
9013
				(val >= REQ_BC_VER_4_VRFY_FIRST_PHY_OPT_MDL) ?
D
Dmitry Kravkov 已提交
9014 9015
				FEATURE_CONFIG_BC_SUPPORTS_OPT_MDL_VRFY : 0;

Y
Yaniv Rosner 已提交
9016 9017 9018
	bp->link_params.feature_config_flags |=
		(val >= REQ_BC_VER_4_VRFY_SPECIFIC_PHY_OPT_MDL) ?
		FEATURE_CONFIG_BC_SUPPORTS_DUAL_PHY_OPT_MDL_VRFY : 0;
E
Eilon Greenstein 已提交
9019

9020 9021 9022
	bp->link_params.feature_config_flags |=
		(val >= REQ_BC_VER_4_SFP_TX_DISABLE_SUPPORTED) ?
		FEATURE_CONFIG_BC_SUPPORTS_SFP_TX_DISABLED : 0;
B
Barak Witkowski 已提交
9023 9024
	bp->flags |= (val >= REQ_BC_VER_4_PFC_STATS_SUPPORTED) ?
			BC_SUPPORTS_PFC_STATS : 0;
9025

9026 9027 9028 9029 9030 9031 9032 9033 9034 9035 9036 9037 9038 9039 9040 9041 9042 9043
	boot_mode = SHMEM_RD(bp,
			dev_info.port_feature_config[BP_PORT(bp)].mba_config) &
			PORT_FEATURE_MBA_BOOT_AGENT_TYPE_MASK;
	switch (boot_mode) {
	case PORT_FEATURE_MBA_BOOT_AGENT_TYPE_PXE:
		bp->common.boot_mode = FEATURE_ETH_BOOTMODE_PXE;
		break;
	case PORT_FEATURE_MBA_BOOT_AGENT_TYPE_ISCSIB:
		bp->common.boot_mode = FEATURE_ETH_BOOTMODE_ISCSI;
		break;
	case PORT_FEATURE_MBA_BOOT_AGENT_TYPE_FCOE_BOOT:
		bp->common.boot_mode = FEATURE_ETH_BOOTMODE_FCOE;
		break;
	case PORT_FEATURE_MBA_BOOT_AGENT_TYPE_NONE:
		bp->common.boot_mode = FEATURE_ETH_BOOTMODE_NONE;
		break;
	}

9044 9045 9046
	pci_read_config_word(bp->pdev, bp->pm_cap + PCI_PM_PMC, &pmc);
	bp->flags |= (pmc & PCI_PM_CAP_PME_D3cold) ? 0 : NO_WOL_FLAG;

E
Eilon Greenstein 已提交
9047
	BNX2X_DEV_INFO("%sWoL capable\n",
E
Eilon Greenstein 已提交
9048
		       (bp->flags & NO_WOL_FLAG) ? "not " : "");
9049 9050 9051 9052 9053 9054

	val = SHMEM_RD(bp, dev_info.shared_hw_config.part_num);
	val2 = SHMEM_RD(bp, dev_info.shared_hw_config.part_num[4]);
	val3 = SHMEM_RD(bp, dev_info.shared_hw_config.part_num[8]);
	val4 = SHMEM_RD(bp, dev_info.shared_hw_config.part_num[12]);

V
Vladislav Zolotarov 已提交
9055 9056
	dev_info(&bp->pdev->dev, "part number %X-%X-%X-%X\n",
		 val, val2, val3, val4);
9057 9058
}

D
Dmitry Kravkov 已提交
9059 9060 9061 9062 9063 9064 9065 9066
#define IGU_FID(val)	GET_FIELD((val), IGU_REG_MAPPING_MEMORY_FID)
#define IGU_VEC(val)	GET_FIELD((val), IGU_REG_MAPPING_MEMORY_VECTOR)

static void __devinit bnx2x_get_igu_cam_info(struct bnx2x *bp)
{
	int pfid = BP_FUNC(bp);
	int igu_sb_id;
	u32 val;
9067
	u8 fid, igu_sb_cnt = 0;
D
Dmitry Kravkov 已提交
9068 9069 9070

	bp->igu_base_sb = 0xff;
	if (CHIP_INT_MODE_IS_BC(bp)) {
9071
		int vn = BP_VN(bp);
9072
		igu_sb_cnt = bp->igu_sb_cnt;
D
Dmitry Kravkov 已提交
9073 9074 9075 9076 9077 9078 9079 9080 9081 9082 9083 9084 9085 9086 9087 9088 9089 9090 9091 9092 9093 9094 9095 9096 9097
		bp->igu_base_sb = (CHIP_MODE_IS_4_PORT(bp) ? pfid : vn) *
			FP_SB_MAX_E1x;

		bp->igu_dsb_id =  E1HVN_MAX * FP_SB_MAX_E1x +
			(CHIP_MODE_IS_4_PORT(bp) ? pfid : vn);

		return;
	}

	/* IGU in normal mode - read CAM */
	for (igu_sb_id = 0; igu_sb_id < IGU_REG_MAPPING_MEMORY_SIZE;
	     igu_sb_id++) {
		val = REG_RD(bp, IGU_REG_MAPPING_MEMORY + igu_sb_id * 4);
		if (!(val & IGU_REG_MAPPING_MEMORY_VALID))
			continue;
		fid = IGU_FID(val);
		if ((fid & IGU_FID_ENCODE_IS_PF)) {
			if ((fid & IGU_FID_PF_NUM_MASK) != pfid)
				continue;
			if (IGU_VEC(val) == 0)
				/* default status block */
				bp->igu_dsb_id = igu_sb_id;
			else {
				if (bp->igu_base_sb == 0xff)
					bp->igu_base_sb = igu_sb_id;
9098
				igu_sb_cnt++;
D
Dmitry Kravkov 已提交
9099 9100 9101
			}
		}
	}
9102

9103 9104 9105 9106 9107
#ifdef CONFIG_PCI_MSI
	/*
	 * It's expected that number of CAM entries for this functions is equal
	 * to the number evaluated based on the MSI-X table size. We want a
	 * harsh warning if these values are different!
9108
	 */
9109 9110
	WARN_ON(bp->igu_sb_cnt != igu_sb_cnt);
#endif
9111

9112
	if (igu_sb_cnt == 0)
D
Dmitry Kravkov 已提交
9113 9114 9115
		BNX2X_ERR("CAM configuration error\n");
}

9116 9117
static void __devinit bnx2x_link_settings_supported(struct bnx2x *bp,
						    u32 switch_cfg)
E
Eliezer Tamir 已提交
9118
{
Y
Yaniv Rosner 已提交
9119 9120 9121 9122 9123
	int cfg_size = 0, idx, port = BP_PORT(bp);

	/* Aggregation of supported attributes of all external phys */
	bp->port.supported[0] = 0;
	bp->port.supported[1] = 0;
Y
Yaniv Rosner 已提交
9124 9125
	switch (bp->link_params.num_phys) {
	case 1:
Y
Yaniv Rosner 已提交
9126 9127 9128
		bp->port.supported[0] = bp->link_params.phy[INT_PHY].supported;
		cfg_size = 1;
		break;
Y
Yaniv Rosner 已提交
9129
	case 2:
Y
Yaniv Rosner 已提交
9130 9131 9132 9133 9134 9135 9136 9137 9138 9139 9140 9141 9142 9143 9144 9145 9146 9147
		bp->port.supported[0] = bp->link_params.phy[EXT_PHY1].supported;
		cfg_size = 1;
		break;
	case 3:
		if (bp->link_params.multi_phy_config &
		    PORT_HW_CFG_PHY_SWAPPED_ENABLED) {
			bp->port.supported[1] =
				bp->link_params.phy[EXT_PHY1].supported;
			bp->port.supported[0] =
				bp->link_params.phy[EXT_PHY2].supported;
		} else {
			bp->port.supported[0] =
				bp->link_params.phy[EXT_PHY1].supported;
			bp->port.supported[1] =
				bp->link_params.phy[EXT_PHY2].supported;
		}
		cfg_size = 2;
		break;
Y
Yaniv Rosner 已提交
9148
	}
E
Eliezer Tamir 已提交
9149

Y
Yaniv Rosner 已提交
9150
	if (!(bp->port.supported[0] || bp->port.supported[1])) {
M
Merav Sicron 已提交
9151
		BNX2X_ERR("NVRAM config error. BAD phy config. PHY1 config 0x%x, PHY2 config 0x%x\n",
Y
Yaniv Rosner 已提交
9152
			   SHMEM_RD(bp,
Y
Yaniv Rosner 已提交
9153 9154 9155
			   dev_info.port_hw_config[port].external_phy_config),
			   SHMEM_RD(bp,
			   dev_info.port_hw_config[port].external_phy_config2));
E
Eliezer Tamir 已提交
9156
			return;
D
Dmitry Kravkov 已提交
9157
	}
E
Eliezer Tamir 已提交
9158

9159 9160 9161 9162 9163 9164 9165 9166 9167 9168 9169 9170 9171 9172 9173 9174 9175
	if (CHIP_IS_E3(bp))
		bp->port.phy_addr = REG_RD(bp, MISC_REG_WC0_CTRL_PHY_ADDR);
	else {
		switch (switch_cfg) {
		case SWITCH_CFG_1G:
			bp->port.phy_addr = REG_RD(
				bp, NIG_REG_SERDES0_CTRL_PHY_ADDR + port*0x10);
			break;
		case SWITCH_CFG_10G:
			bp->port.phy_addr = REG_RD(
				bp, NIG_REG_XGXS0_CTRL_PHY_ADDR + port*0x18);
			break;
		default:
			BNX2X_ERR("BAD switch_cfg link_config 0x%x\n",
				  bp->port.link_config[0]);
			return;
		}
E
Eliezer Tamir 已提交
9176
	}
9177
	BNX2X_DEV_INFO("phy_addr 0x%x\n", bp->port.phy_addr);
Y
Yaniv Rosner 已提交
9178 9179 9180
	/* mask what we support according to speed_cap_mask per configuration */
	for (idx = 0; idx < cfg_size; idx++) {
		if (!(bp->link_params.speed_cap_mask[idx] &
Y
Yaniv Rosner 已提交
9181
				PORT_HW_CFG_SPEED_CAPABILITY_D0_10M_HALF))
Y
Yaniv Rosner 已提交
9182
			bp->port.supported[idx] &= ~SUPPORTED_10baseT_Half;
E
Eliezer Tamir 已提交
9183

Y
Yaniv Rosner 已提交
9184
		if (!(bp->link_params.speed_cap_mask[idx] &
Y
Yaniv Rosner 已提交
9185
				PORT_HW_CFG_SPEED_CAPABILITY_D0_10M_FULL))
Y
Yaniv Rosner 已提交
9186
			bp->port.supported[idx] &= ~SUPPORTED_10baseT_Full;
E
Eliezer Tamir 已提交
9187

Y
Yaniv Rosner 已提交
9188
		if (!(bp->link_params.speed_cap_mask[idx] &
Y
Yaniv Rosner 已提交
9189
				PORT_HW_CFG_SPEED_CAPABILITY_D0_100M_HALF))
Y
Yaniv Rosner 已提交
9190
			bp->port.supported[idx] &= ~SUPPORTED_100baseT_Half;
E
Eliezer Tamir 已提交
9191

Y
Yaniv Rosner 已提交
9192
		if (!(bp->link_params.speed_cap_mask[idx] &
Y
Yaniv Rosner 已提交
9193
				PORT_HW_CFG_SPEED_CAPABILITY_D0_100M_FULL))
Y
Yaniv Rosner 已提交
9194
			bp->port.supported[idx] &= ~SUPPORTED_100baseT_Full;
E
Eliezer Tamir 已提交
9195

Y
Yaniv Rosner 已提交
9196
		if (!(bp->link_params.speed_cap_mask[idx] &
Y
Yaniv Rosner 已提交
9197
					PORT_HW_CFG_SPEED_CAPABILITY_D0_1G))
Y
Yaniv Rosner 已提交
9198
			bp->port.supported[idx] &= ~(SUPPORTED_1000baseT_Half |
D
Dmitry Kravkov 已提交
9199
						     SUPPORTED_1000baseT_Full);
E
Eliezer Tamir 已提交
9200

Y
Yaniv Rosner 已提交
9201
		if (!(bp->link_params.speed_cap_mask[idx] &
Y
Yaniv Rosner 已提交
9202
					PORT_HW_CFG_SPEED_CAPABILITY_D0_2_5G))
Y
Yaniv Rosner 已提交
9203
			bp->port.supported[idx] &= ~SUPPORTED_2500baseX_Full;
E
Eliezer Tamir 已提交
9204

Y
Yaniv Rosner 已提交
9205
		if (!(bp->link_params.speed_cap_mask[idx] &
Y
Yaniv Rosner 已提交
9206
					PORT_HW_CFG_SPEED_CAPABILITY_D0_10G))
Y
Yaniv Rosner 已提交
9207 9208 9209
			bp->port.supported[idx] &= ~SUPPORTED_10000baseT_Full;

	}
E
Eliezer Tamir 已提交
9210

Y
Yaniv Rosner 已提交
9211 9212
	BNX2X_DEV_INFO("supported 0x%x 0x%x\n", bp->port.supported[0],
		       bp->port.supported[1]);
E
Eliezer Tamir 已提交
9213 9214
}

9215
static void __devinit bnx2x_link_settings_requested(struct bnx2x *bp)
E
Eliezer Tamir 已提交
9216
{
Y
Yaniv Rosner 已提交
9217 9218 9219 9220 9221 9222 9223 9224 9225 9226 9227 9228 9229 9230 9231 9232
	u32 link_config, idx, cfg_size = 0;
	bp->port.advertising[0] = 0;
	bp->port.advertising[1] = 0;
	switch (bp->link_params.num_phys) {
	case 1:
	case 2:
		cfg_size = 1;
		break;
	case 3:
		cfg_size = 2;
		break;
	}
	for (idx = 0; idx < cfg_size; idx++) {
		bp->link_params.req_duplex[idx] = DUPLEX_FULL;
		link_config = bp->port.link_config[idx];
		switch (link_config & PORT_FEATURE_LINK_SPEED_MASK) {
D
Dmitry Kravkov 已提交
9233
		case PORT_FEATURE_LINK_SPEED_AUTO:
Y
Yaniv Rosner 已提交
9234 9235 9236 9237 9238
			if (bp->port.supported[idx] & SUPPORTED_Autoneg) {
				bp->link_params.req_line_speed[idx] =
					SPEED_AUTO_NEG;
				bp->port.advertising[idx] |=
					bp->port.supported[idx];
9239 9240 9241 9242 9243
				if (bp->link_params.phy[EXT_PHY1].type ==
				    PORT_HW_CFG_XGXS_EXT_PHY_TYPE_BCM84833)
					bp->port.advertising[idx] |=
					(SUPPORTED_100baseT_Half |
					 SUPPORTED_100baseT_Full);
D
Dmitry Kravkov 已提交
9244 9245
			} else {
				/* force 10G, no AN */
Y
Yaniv Rosner 已提交
9246 9247 9248 9249
				bp->link_params.req_line_speed[idx] =
					SPEED_10000;
				bp->port.advertising[idx] |=
					(ADVERTISED_10000baseT_Full |
D
Dmitry Kravkov 已提交
9250
					 ADVERTISED_FIBRE);
Y
Yaniv Rosner 已提交
9251
				continue;
D
Dmitry Kravkov 已提交
9252 9253
			}
			break;
E
Eliezer Tamir 已提交
9254

D
Dmitry Kravkov 已提交
9255
		case PORT_FEATURE_LINK_SPEED_10M_FULL:
Y
Yaniv Rosner 已提交
9256 9257 9258 9259 9260
			if (bp->port.supported[idx] & SUPPORTED_10baseT_Full) {
				bp->link_params.req_line_speed[idx] =
					SPEED_10;
				bp->port.advertising[idx] |=
					(ADVERTISED_10baseT_Full |
D
Dmitry Kravkov 已提交
9261 9262
					 ADVERTISED_TP);
			} else {
M
Merav Sicron 已提交
9263
				BNX2X_ERR("NVRAM config error. Invalid link_config 0x%x  speed_cap_mask 0x%x\n",
D
Dmitry Kravkov 已提交
9264
					    link_config,
Y
Yaniv Rosner 已提交
9265
				    bp->link_params.speed_cap_mask[idx]);
D
Dmitry Kravkov 已提交
9266 9267 9268
				return;
			}
			break;
E
Eliezer Tamir 已提交
9269

D
Dmitry Kravkov 已提交
9270
		case PORT_FEATURE_LINK_SPEED_10M_HALF:
Y
Yaniv Rosner 已提交
9271 9272 9273 9274 9275 9276 9277
			if (bp->port.supported[idx] & SUPPORTED_10baseT_Half) {
				bp->link_params.req_line_speed[idx] =
					SPEED_10;
				bp->link_params.req_duplex[idx] =
					DUPLEX_HALF;
				bp->port.advertising[idx] |=
					(ADVERTISED_10baseT_Half |
D
Dmitry Kravkov 已提交
9278 9279
					 ADVERTISED_TP);
			} else {
M
Merav Sicron 已提交
9280
				BNX2X_ERR("NVRAM config error. Invalid link_config 0x%x  speed_cap_mask 0x%x\n",
D
Dmitry Kravkov 已提交
9281 9282 9283 9284 9285
					    link_config,
					  bp->link_params.speed_cap_mask[idx]);
				return;
			}
			break;
E
Eliezer Tamir 已提交
9286

D
Dmitry Kravkov 已提交
9287 9288 9289
		case PORT_FEATURE_LINK_SPEED_100M_FULL:
			if (bp->port.supported[idx] &
			    SUPPORTED_100baseT_Full) {
Y
Yaniv Rosner 已提交
9290 9291 9292 9293
				bp->link_params.req_line_speed[idx] =
					SPEED_100;
				bp->port.advertising[idx] |=
					(ADVERTISED_100baseT_Full |
D
Dmitry Kravkov 已提交
9294 9295
					 ADVERTISED_TP);
			} else {
M
Merav Sicron 已提交
9296
				BNX2X_ERR("NVRAM config error. Invalid link_config 0x%x  speed_cap_mask 0x%x\n",
D
Dmitry Kravkov 已提交
9297 9298 9299 9300 9301
					    link_config,
					  bp->link_params.speed_cap_mask[idx]);
				return;
			}
			break;
E
Eliezer Tamir 已提交
9302

D
Dmitry Kravkov 已提交
9303 9304 9305 9306 9307 9308 9309
		case PORT_FEATURE_LINK_SPEED_100M_HALF:
			if (bp->port.supported[idx] &
			    SUPPORTED_100baseT_Half) {
				bp->link_params.req_line_speed[idx] =
								SPEED_100;
				bp->link_params.req_duplex[idx] =
								DUPLEX_HALF;
Y
Yaniv Rosner 已提交
9310 9311
				bp->port.advertising[idx] |=
					(ADVERTISED_100baseT_Half |
D
Dmitry Kravkov 已提交
9312 9313
					 ADVERTISED_TP);
			} else {
M
Merav Sicron 已提交
9314
				BNX2X_ERR("NVRAM config error. Invalid link_config 0x%x  speed_cap_mask 0x%x\n",
Y
Yaniv Rosner 已提交
9315 9316
				    link_config,
				    bp->link_params.speed_cap_mask[idx]);
D
Dmitry Kravkov 已提交
9317 9318 9319
				return;
			}
			break;
E
Eliezer Tamir 已提交
9320

D
Dmitry Kravkov 已提交
9321
		case PORT_FEATURE_LINK_SPEED_1G:
Y
Yaniv Rosner 已提交
9322 9323 9324 9325 9326 9327
			if (bp->port.supported[idx] &
			    SUPPORTED_1000baseT_Full) {
				bp->link_params.req_line_speed[idx] =
					SPEED_1000;
				bp->port.advertising[idx] |=
					(ADVERTISED_1000baseT_Full |
D
Dmitry Kravkov 已提交
9328 9329
					 ADVERTISED_TP);
			} else {
M
Merav Sicron 已提交
9330
				BNX2X_ERR("NVRAM config error. Invalid link_config 0x%x  speed_cap_mask 0x%x\n",
Y
Yaniv Rosner 已提交
9331 9332
				    link_config,
				    bp->link_params.speed_cap_mask[idx]);
D
Dmitry Kravkov 已提交
9333 9334 9335
				return;
			}
			break;
E
Eliezer Tamir 已提交
9336

D
Dmitry Kravkov 已提交
9337
		case PORT_FEATURE_LINK_SPEED_2_5G:
Y
Yaniv Rosner 已提交
9338 9339 9340 9341 9342 9343
			if (bp->port.supported[idx] &
			    SUPPORTED_2500baseX_Full) {
				bp->link_params.req_line_speed[idx] =
					SPEED_2500;
				bp->port.advertising[idx] |=
					(ADVERTISED_2500baseX_Full |
9344
						ADVERTISED_TP);
D
Dmitry Kravkov 已提交
9345
			} else {
M
Merav Sicron 已提交
9346
				BNX2X_ERR("NVRAM config error. Invalid link_config 0x%x  speed_cap_mask 0x%x\n",
Y
Yaniv Rosner 已提交
9347
				    link_config,
D
Dmitry Kravkov 已提交
9348 9349 9350 9351
				    bp->link_params.speed_cap_mask[idx]);
				return;
			}
			break;
E
Eliezer Tamir 已提交
9352

D
Dmitry Kravkov 已提交
9353
		case PORT_FEATURE_LINK_SPEED_10G_CX4:
Y
Yaniv Rosner 已提交
9354 9355 9356 9357 9358 9359
			if (bp->port.supported[idx] &
			    SUPPORTED_10000baseT_Full) {
				bp->link_params.req_line_speed[idx] =
					SPEED_10000;
				bp->port.advertising[idx] |=
					(ADVERTISED_10000baseT_Full |
9360
						ADVERTISED_FIBRE);
D
Dmitry Kravkov 已提交
9361
			} else {
M
Merav Sicron 已提交
9362
				BNX2X_ERR("NVRAM config error. Invalid link_config 0x%x  speed_cap_mask 0x%x\n",
Y
Yaniv Rosner 已提交
9363
				    link_config,
D
Dmitry Kravkov 已提交
9364 9365 9366 9367
				    bp->link_params.speed_cap_mask[idx]);
				return;
			}
			break;
9368 9369
		case PORT_FEATURE_LINK_SPEED_20G:
			bp->link_params.req_line_speed[idx] = SPEED_20000;
E
Eliezer Tamir 已提交
9370

9371
			break;
D
Dmitry Kravkov 已提交
9372
		default:
M
Merav Sicron 已提交
9373
			BNX2X_ERR("NVRAM config error. BAD link speed link_config 0x%x\n",
D
Dmitry Kravkov 已提交
9374
				  link_config);
D
Dmitry Kravkov 已提交
9375 9376 9377 9378 9379 9380
				bp->link_params.req_line_speed[idx] =
							SPEED_AUTO_NEG;
				bp->port.advertising[idx] =
						bp->port.supported[idx];
			break;
		}
E
Eliezer Tamir 已提交
9381

Y
Yaniv Rosner 已提交
9382
		bp->link_params.req_flow_ctrl[idx] = (link_config &
9383
					 PORT_FEATURE_FLOW_CONTROL_MASK);
Y
Yaniv Rosner 已提交
9384 9385 9386 9387 9388 9389
		if ((bp->link_params.req_flow_ctrl[idx] ==
		     BNX2X_FLOW_CTRL_AUTO) &&
		    !(bp->port.supported[idx] & SUPPORTED_Autoneg)) {
			bp->link_params.req_flow_ctrl[idx] =
				BNX2X_FLOW_CTRL_NONE;
		}
E
Eliezer Tamir 已提交
9390

M
Merav Sicron 已提交
9391
		BNX2X_DEV_INFO("req_line_speed %d  req_duplex %d req_flow_ctrl 0x%x advertising 0x%x\n",
Y
Yaniv Rosner 已提交
9392 9393 9394 9395 9396
			       bp->link_params.req_line_speed[idx],
			       bp->link_params.req_duplex[idx],
			       bp->link_params.req_flow_ctrl[idx],
			       bp->port.advertising[idx]);
	}
E
Eliezer Tamir 已提交
9397 9398
}

9399 9400 9401 9402 9403 9404 9405 9406
static void __devinit bnx2x_set_mac_buf(u8 *mac_buf, u32 mac_lo, u16 mac_hi)
{
	mac_hi = cpu_to_be16(mac_hi);
	mac_lo = cpu_to_be32(mac_lo);
	memcpy(mac_buf, &mac_hi, sizeof(mac_hi));
	memcpy(mac_buf + sizeof(mac_hi), &mac_lo, sizeof(mac_lo));
}

9407
static void __devinit bnx2x_get_port_hwinfo(struct bnx2x *bp)
E
Eliezer Tamir 已提交
9408
{
9409
	int port = BP_PORT(bp);
E
Eilon Greenstein 已提交
9410
	u32 config;
9411
	u32 ext_phy_type, ext_phy_config;
E
Eliezer Tamir 已提交
9412

Y
Yaniv Rosner 已提交
9413
	bp->link_params.bp = bp;
9414
	bp->link_params.port = port;
Y
Yaniv Rosner 已提交
9415 9416

	bp->link_params.lane_config =
E
Eliezer Tamir 已提交
9417
		SHMEM_RD(bp, dev_info.port_hw_config[port].lane_config);
E
Eilon Greenstein 已提交
9418

Y
Yaniv Rosner 已提交
9419
	bp->link_params.speed_cap_mask[0] =
E
Eliezer Tamir 已提交
9420 9421
		SHMEM_RD(bp,
			 dev_info.port_hw_config[port].speed_capability_mask);
Y
Yaniv Rosner 已提交
9422 9423 9424 9425
	bp->link_params.speed_cap_mask[1] =
		SHMEM_RD(bp,
			 dev_info.port_hw_config[port].speed_capability_mask2);
	bp->port.link_config[0] =
E
Eliezer Tamir 已提交
9426 9427
		SHMEM_RD(bp, dev_info.port_feature_config[port].link_config);

Y
Yaniv Rosner 已提交
9428 9429
	bp->port.link_config[1] =
		SHMEM_RD(bp, dev_info.port_feature_config[port].link_config2);
9430

Y
Yaniv Rosner 已提交
9431 9432
	bp->link_params.multi_phy_config =
		SHMEM_RD(bp, dev_info.port_hw_config[port].multi_phy_config);
9433 9434 9435
	/* If the device is capable of WoL, set the default state according
	 * to the HW
	 */
E
Eilon Greenstein 已提交
9436
	config = SHMEM_RD(bp, dev_info.port_feature_config[port].config);
9437 9438 9439
	bp->wol = (!(bp->flags & NO_WOL_FLAG) &&
		   (config & PORT_FEATURE_WOL_ENABLED));

M
Merav Sicron 已提交
9440
	BNX2X_DEV_INFO("lane_config 0x%08x  speed_cap_mask0 0x%08x  link_config0 0x%08x\n",
Y
Yaniv Rosner 已提交
9441
		       bp->link_params.lane_config,
Y
Yaniv Rosner 已提交
9442 9443
		       bp->link_params.speed_cap_mask[0],
		       bp->port.link_config[0]);
E
Eliezer Tamir 已提交
9444

Y
Yaniv Rosner 已提交
9445
	bp->link_params.switch_cfg = (bp->port.link_config[0] &
D
Dmitry Kravkov 已提交
9446
				      PORT_FEATURE_CONNECTED_SWITCH_MASK);
Y
Yaniv Rosner 已提交
9447
	bnx2x_phy_probe(&bp->link_params);
Y
Yaniv Rosner 已提交
9448
	bnx2x_link_settings_supported(bp, bp->link_params.switch_cfg);
E
Eliezer Tamir 已提交
9449 9450 9451

	bnx2x_link_settings_requested(bp);

E
Eilon Greenstein 已提交
9452 9453 9454 9455
	/*
	 * If connected directly, work with the internal PHY, otherwise, work
	 * with the external PHY
	 */
Y
Yaniv Rosner 已提交
9456 9457 9458 9459
	ext_phy_config =
		SHMEM_RD(bp,
			 dev_info.port_hw_config[port].external_phy_config);
	ext_phy_type = XGXS_EXT_PHY_TYPE(ext_phy_config);
E
Eilon Greenstein 已提交
9460
	if (ext_phy_type == PORT_HW_CFG_XGXS_EXT_PHY_TYPE_DIRECT)
Y
Yaniv Rosner 已提交
9461
		bp->mdio.prtad = bp->port.phy_addr;
E
Eilon Greenstein 已提交
9462 9463 9464 9465

	else if ((ext_phy_type != PORT_HW_CFG_XGXS_EXT_PHY_TYPE_FAILURE) &&
		 (ext_phy_type != PORT_HW_CFG_XGXS_EXT_PHY_TYPE_NOT_CONN))
		bp->mdio.prtad =
Y
Yaniv Rosner 已提交
9466
			XGXS_EXT_PHY_ADDR(ext_phy_config);
9467 9468 9469 9470 9471 9472 9473 9474 9475 9476 9477

	/*
	 * Check if hw lock is required to access MDC/MDIO bus to the PHY(s)
	 * In MF mode, it is set to cover self test cases
	 */
	if (IS_MF(bp))
		bp->port.need_hw_lock = 1;
	else
		bp->port.need_hw_lock = bnx2x_hw_lock_required(bp,
							bp->common.shmem_base,
							bp->common.shmem2_base);
9478
}
E
Eilon Greenstein 已提交
9479

9480
void bnx2x_get_iscsi_info(struct bnx2x *bp)
9481
{
9482
#ifdef BCM_CNIC
9483 9484
	int port = BP_PORT(bp);

9485
	u32 max_iscsi_conn = FW_ENCODE_32BIT_PATTERN ^ SHMEM_RD(bp,
9486
				drv_lic_key[port].max_iscsi_conn);
9487

9488
	/* Get the number of maximum allowed iSCSI connections */
9489 9490 9491 9492
	bp->cnic_eth_dev.max_iscsi_conn =
		(max_iscsi_conn & BNX2X_MAX_ISCSI_INIT_CONN_MASK) >>
		BNX2X_MAX_ISCSI_INIT_CONN_SHIFT;

9493 9494 9495 9496 9497 9498 9499 9500 9501
	BNX2X_DEV_INFO("max_iscsi_conn 0x%x\n",
		       bp->cnic_eth_dev.max_iscsi_conn);

	/*
	 * If maximum allowed number of connections is zero -
	 * disable the feature.
	 */
	if (!bp->cnic_eth_dev.max_iscsi_conn)
		bp->flags |= NO_ISCSI_FLAG;
9502 9503 9504
#else
	bp->flags |= NO_ISCSI_FLAG;
#endif
9505 9506 9507 9508
}

static void __devinit bnx2x_get_fcoe_info(struct bnx2x *bp)
{
9509
#ifdef BCM_CNIC
9510 9511 9512 9513 9514 9515 9516
	int port = BP_PORT(bp);
	int func = BP_ABS_FUNC(bp);

	u32 max_fcoe_conn = FW_ENCODE_32BIT_PATTERN ^ SHMEM_RD(bp,
				drv_lic_key[port].max_fcoe_conn);

	/* Get the number of maximum allowed FCoE connections */
9517 9518 9519 9520
	bp->cnic_eth_dev.max_fcoe_conn =
		(max_fcoe_conn & BNX2X_MAX_FCOE_INIT_CONN_MASK) >>
		BNX2X_MAX_FCOE_INIT_CONN_SHIFT;

9521 9522 9523 9524 9525 9526 9527 9528 9529 9530 9531 9532 9533 9534 9535 9536 9537 9538 9539 9540 9541 9542 9543 9544 9545 9546 9547 9548 9549 9550 9551 9552 9553 9554 9555 9556 9557 9558 9559 9560 9561 9562 9563 9564 9565 9566 9567
	/* Read the WWN: */
	if (!IS_MF(bp)) {
		/* Port info */
		bp->cnic_eth_dev.fcoe_wwn_port_name_hi =
			SHMEM_RD(bp,
				dev_info.port_hw_config[port].
				 fcoe_wwn_port_name_upper);
		bp->cnic_eth_dev.fcoe_wwn_port_name_lo =
			SHMEM_RD(bp,
				dev_info.port_hw_config[port].
				 fcoe_wwn_port_name_lower);

		/* Node info */
		bp->cnic_eth_dev.fcoe_wwn_node_name_hi =
			SHMEM_RD(bp,
				dev_info.port_hw_config[port].
				 fcoe_wwn_node_name_upper);
		bp->cnic_eth_dev.fcoe_wwn_node_name_lo =
			SHMEM_RD(bp,
				dev_info.port_hw_config[port].
				 fcoe_wwn_node_name_lower);
	} else if (!IS_MF_SD(bp)) {
		u32 cfg = MF_CFG_RD(bp, func_ext_config[func].func_cfg);

		/*
		 * Read the WWN info only if the FCoE feature is enabled for
		 * this function.
		 */
		if (cfg & MACP_FUNC_CFG_FLAGS_FCOE_OFFLOAD) {
			/* Port info */
			bp->cnic_eth_dev.fcoe_wwn_port_name_hi =
				MF_CFG_RD(bp, func_ext_config[func].
						fcoe_wwn_port_name_upper);
			bp->cnic_eth_dev.fcoe_wwn_port_name_lo =
				MF_CFG_RD(bp, func_ext_config[func].
						fcoe_wwn_port_name_lower);

			/* Node info */
			bp->cnic_eth_dev.fcoe_wwn_node_name_hi =
				MF_CFG_RD(bp, func_ext_config[func].
						fcoe_wwn_node_name_upper);
			bp->cnic_eth_dev.fcoe_wwn_node_name_lo =
				MF_CFG_RD(bp, func_ext_config[func].
						fcoe_wwn_node_name_lower);
		}
	}

9568
	BNX2X_DEV_INFO("max_fcoe_conn 0x%x\n", bp->cnic_eth_dev.max_fcoe_conn);
9569

9570 9571
	/*
	 * If maximum allowed number of connections is zero -
9572 9573 9574 9575
	 * disable the feature.
	 */
	if (!bp->cnic_eth_dev.max_fcoe_conn)
		bp->flags |= NO_FCOE_FLAG;
9576 9577 9578
#else
	bp->flags |= NO_FCOE_FLAG;
#endif
9579
}
9580 9581 9582 9583 9584 9585 9586 9587 9588 9589 9590

static void __devinit bnx2x_get_cnic_info(struct bnx2x *bp)
{
	/*
	 * iSCSI may be dynamically disabled but reading
	 * info here we will decrease memory usage by driver
	 * if the feature is disabled for good
	 */
	bnx2x_get_iscsi_info(bp);
	bnx2x_get_fcoe_info(bp);
}
9591

9592 9593 9594 9595 9596
static void __devinit bnx2x_get_mac_hwinfo(struct bnx2x *bp)
{
	u32 val, val2;
	int func = BP_ABS_FUNC(bp);
	int port = BP_PORT(bp);
9597 9598 9599 9600
#ifdef BCM_CNIC
	u8 *iscsi_mac = bp->cnic_eth_dev.iscsi_mac;
	u8 *fip_mac = bp->fip_mac;
#endif
9601

9602 9603 9604
	/* Zero primary MAC configuration */
	memset(bp->dev->dev_addr, 0, ETH_ALEN);

9605 9606
	if (BP_NOMCP(bp)) {
		BNX2X_ERROR("warning: random MAC workaround active\n");
9607
		eth_hw_addr_random(bp->dev);
9608 9609 9610 9611 9612 9613
	} else if (IS_MF(bp)) {
		val2 = MF_CFG_RD(bp, func_mf_config[func].mac_upper);
		val = MF_CFG_RD(bp, func_mf_config[func].mac_lower);
		if ((val2 != FUNC_MF_CFG_UPPERMAC_DEFAULT) &&
		    (val != FUNC_MF_CFG_LOWERMAC_DEFAULT))
			bnx2x_set_mac_buf(bp->dev->dev_addr, val, val2);
9614 9615

#ifdef BCM_CNIC
D
Dmitry Kravkov 已提交
9616 9617
		/*
		 * iSCSI and FCoE NPAR MACs: if there is no either iSCSI or
9618 9619
		 * FCoE MAC then the appropriate feature should be disabled.
		 */
9620 9621 9622 9623 9624 9625 9626
		if (IS_MF_SI(bp)) {
			u32 cfg = MF_CFG_RD(bp, func_ext_config[func].func_cfg);
			if (cfg & MACP_FUNC_CFG_FLAGS_ISCSI_OFFLOAD) {
				val2 = MF_CFG_RD(bp, func_ext_config[func].
						     iscsi_mac_addr_upper);
				val = MF_CFG_RD(bp, func_ext_config[func].
						    iscsi_mac_addr_lower);
9627
				bnx2x_set_mac_buf(iscsi_mac, val, val2);
9628 9629
				BNX2X_DEV_INFO("Read iSCSI MAC: %pM\n",
					       iscsi_mac);
9630 9631 9632 9633 9634 9635 9636 9637 9638
			} else
				bp->flags |= NO_ISCSI_OOO_FLAG | NO_ISCSI_FLAG;

			if (cfg & MACP_FUNC_CFG_FLAGS_FCOE_OFFLOAD) {
				val2 = MF_CFG_RD(bp, func_ext_config[func].
						     fcoe_mac_addr_upper);
				val = MF_CFG_RD(bp, func_ext_config[func].
						    fcoe_mac_addr_lower);
				bnx2x_set_mac_buf(fip_mac, val, val2);
D
Dmitry Kravkov 已提交
9639
				BNX2X_DEV_INFO("Read FCoE L2 MAC: %pM\n",
9640
					       fip_mac);
9641 9642 9643

			} else
				bp->flags |= NO_FCOE_FLAG;
D
Dmitry Kravkov 已提交
9644 9645 9646 9647 9648 9649 9650 9651 9652 9653 9654
		} else { /* SD mode */
			if (BNX2X_IS_MF_PROTOCOL_ISCSI(bp)) {
				/* use primary mac as iscsi mac */
				memcpy(iscsi_mac, bp->dev->dev_addr, ETH_ALEN);
				/* Zero primary MAC configuration */
				memset(bp->dev->dev_addr, 0, ETH_ALEN);

				BNX2X_DEV_INFO("SD ISCSI MODE\n");
				BNX2X_DEV_INFO("Read iSCSI MAC: %pM\n",
					       iscsi_mac);
			}
9655
		}
9656
#endif
9657 9658 9659 9660 9661 9662 9663 9664 9665 9666 9667
	} else {
		/* in SF read MACs from port configuration */
		val2 = SHMEM_RD(bp, dev_info.port_hw_config[port].mac_upper);
		val = SHMEM_RD(bp, dev_info.port_hw_config[port].mac_lower);
		bnx2x_set_mac_buf(bp->dev->dev_addr, val, val2);

#ifdef BCM_CNIC
		val2 = SHMEM_RD(bp, dev_info.port_hw_config[port].
				    iscsi_mac_upper);
		val = SHMEM_RD(bp, dev_info.port_hw_config[port].
				   iscsi_mac_lower);
9668
		bnx2x_set_mac_buf(iscsi_mac, val, val2);
9669 9670 9671 9672 9673 9674

		val2 = SHMEM_RD(bp, dev_info.port_hw_config[port].
				    fcoe_fip_mac_upper);
		val = SHMEM_RD(bp, dev_info.port_hw_config[port].
				   fcoe_fip_mac_lower);
		bnx2x_set_mac_buf(fip_mac, val, val2);
9675 9676 9677 9678 9679 9680
#endif
	}

	memcpy(bp->link_params.mac_addr, bp->dev->dev_addr, ETH_ALEN);
	memcpy(bp->dev->perm_addr, bp->dev->dev_addr, ETH_ALEN);

V
Vladislav Zolotarov 已提交
9681
#ifdef BCM_CNIC
9682 9683 9684
	/* Set the FCoE MAC in MF_SD mode */
	if (!CHIP_IS_E1x(bp) && IS_MF_SD(bp))
		memcpy(fip_mac, bp->dev->dev_addr, ETH_ALEN);
9685 9686 9687 9688 9689 9690 9691 9692 9693 9694 9695 9696 9697 9698 9699 9700

	/* Disable iSCSI if MAC configuration is
	 * invalid.
	 */
	if (!is_valid_ether_addr(iscsi_mac)) {
		bp->flags |= NO_ISCSI_FLAG;
		memset(iscsi_mac, 0, ETH_ALEN);
	}

	/* Disable FCoE if MAC configuration is
	 * invalid.
	 */
	if (!is_valid_ether_addr(fip_mac)) {
		bp->flags |= NO_FCOE_FLAG;
		memset(bp->fip_mac, 0, ETH_ALEN);
	}
V
Vladislav Zolotarov 已提交
9701
#endif
9702

D
Dmitry Kravkov 已提交
9703
	if (!bnx2x_is_valid_ether_addr(bp, bp->dev->dev_addr))
9704
		dev_err(&bp->pdev->dev,
M
Merav Sicron 已提交
9705 9706
			"bad Ethernet MAC address configuration: %pM\n"
			"change it manually before bringing up the appropriate network interface\n",
9707
			bp->dev->dev_addr);
M
Merav Sicron 已提交
9708 9709


9710 9711 9712 9713
}

static int __devinit bnx2x_get_hwinfo(struct bnx2x *bp)
{
9714
	int /*abs*/func = BP_ABS_FUNC(bp);
9715
	int vn;
9716
	u32 val = 0;
9717
	int rc = 0;
E
Eliezer Tamir 已提交
9718

9719
	bnx2x_get_common_hwinfo(bp);
E
Eliezer Tamir 已提交
9720

9721 9722 9723
	/*
	 * initialize IGU parameters
	 */
D
Dmitry Kravkov 已提交
9724 9725 9726 9727 9728 9729 9730
	if (CHIP_IS_E1x(bp)) {
		bp->common.int_block = INT_BLOCK_HC;

		bp->igu_dsb_id = DEF_SB_IGU_ID;
		bp->igu_base_sb = 0;
	} else {
		bp->common.int_block = INT_BLOCK_IGU;
9731 9732 9733 9734

		/* do not allow device reset during IGU info preocessing */
		bnx2x_acquire_hw_lock(bp, HW_LOCK_RESOURCE_RESET);

D
Dmitry Kravkov 已提交
9735
		val = REG_RD(bp, IGU_REG_BLOCK_CONFIGURATION);
9736 9737 9738 9739 9740 9741 9742 9743 9744 9745 9746 9747 9748 9749 9750 9751 9752 9753 9754 9755 9756 9757

		if (val & IGU_BLOCK_CONFIGURATION_REG_BACKWARD_COMP_EN) {
			int tout = 5000;

			BNX2X_DEV_INFO("FORCING Normal Mode\n");

			val &= ~(IGU_BLOCK_CONFIGURATION_REG_BACKWARD_COMP_EN);
			REG_WR(bp, IGU_REG_BLOCK_CONFIGURATION, val);
			REG_WR(bp, IGU_REG_RESET_MEMORIES, 0x7f);

			while (tout && REG_RD(bp, IGU_REG_RESET_MEMORIES)) {
				tout--;
				usleep_range(1000, 1000);
			}

			if (REG_RD(bp, IGU_REG_RESET_MEMORIES)) {
				dev_err(&bp->pdev->dev,
					"FORCING Normal Mode failed!!!\n");
				return -EPERM;
			}
		}

D
Dmitry Kravkov 已提交
9758
		if (val & IGU_BLOCK_CONFIGURATION_REG_BACKWARD_COMP_EN) {
9759
			BNX2X_DEV_INFO("IGU Backward Compatible Mode\n");
D
Dmitry Kravkov 已提交
9760 9761
			bp->common.int_block |= INT_BLOCK_MODE_BW_COMP;
		} else
9762
			BNX2X_DEV_INFO("IGU Normal Mode\n");
9763

D
Dmitry Kravkov 已提交
9764 9765
		bnx2x_get_igu_cam_info(bp);

9766
		bnx2x_release_hw_lock(bp, HW_LOCK_RESOURCE_RESET);
D
Dmitry Kravkov 已提交
9767
	}
9768 9769 9770 9771 9772 9773 9774 9775 9776 9777 9778 9779 9780 9781 9782 9783 9784 9785

	/*
	 * set base FW non-default (fast path) status block id, this value is
	 * used to initialize the fw_sb_id saved on the fp/queue structure to
	 * determine the id used by the FW.
	 */
	if (CHIP_IS_E1x(bp))
		bp->base_fw_ndsb = BP_PORT(bp) * FP_SB_MAX_E1x + BP_L_ID(bp);
	else /*
	      * 57712 - we currently use one FW SB per IGU SB (Rx and Tx of
	      * the same queue are indicated on the same IGU SB). So we prefer
	      * FW and IGU SBs to be the same value.
	      */
		bp->base_fw_ndsb = bp->igu_base_sb;

	BNX2X_DEV_INFO("igu_dsb_id %d  igu_base_sb %d  igu_sb_cnt %d\n"
		       "base_fw_ndsb %d\n", bp->igu_dsb_id, bp->igu_base_sb,
		       bp->igu_sb_cnt, bp->base_fw_ndsb);
D
Dmitry Kravkov 已提交
9786 9787 9788 9789

	/*
	 * Initialize MF configuration
	 */
9790

D
Dmitry Kravkov 已提交
9791 9792
	bp->mf_ov = 0;
	bp->mf_mode = 0;
9793
	vn = BP_VN(bp);
9794

D
Dmitry Kravkov 已提交
9795
	if (!CHIP_IS_E1(bp) && !BP_NOMCP(bp)) {
9796 9797 9798 9799
		BNX2X_DEV_INFO("shmem2base 0x%x, size %d, mfcfg offset %d\n",
			       bp->common.shmem2_base, SHMEM2_RD(bp, size),
			      (u32)offsetof(struct shmem2_region, mf_cfg_addr));

D
Dmitry Kravkov 已提交
9800 9801 9802 9803
		if (SHMEM2_HAS(bp, mf_cfg_addr))
			bp->common.mf_cfg_base = SHMEM2_RD(bp, mf_cfg_addr);
		else
			bp->common.mf_cfg_base = bp->common.shmem_base +
9804 9805
				offsetof(struct shmem_region, func_mb) +
				E1H_FUNC_MAX * sizeof(struct drv_func_mb);
9806 9807
		/*
		 * get mf configuration:
L
Lucas De Marchi 已提交
9808
		 * 1. existence of MF configuration
9809 9810 9811 9812 9813 9814 9815 9816 9817 9818 9819 9820 9821 9822 9823 9824 9825 9826 9827 9828 9829
		 * 2. MAC address must be legal (check only upper bytes)
		 *    for  Switch-Independent mode;
		 *    OVLAN must be legal for Switch-Dependent mode
		 * 3. SF_MODE configures specific MF mode
		 */
		if (bp->common.mf_cfg_base != SHMEM_MF_CFG_ADDR_NONE) {
			/* get mf configuration */
			val = SHMEM_RD(bp,
				       dev_info.shared_feature_config.config);
			val &= SHARED_FEAT_CFG_FORCE_SF_MODE_MASK;

			switch (val) {
			case SHARED_FEAT_CFG_FORCE_SF_MODE_SWITCH_INDEPT:
				val = MF_CFG_RD(bp, func_mf_config[func].
						mac_upper);
				/* check for legal mac (upper bytes)*/
				if (val != 0xffff) {
					bp->mf_mode = MULTI_FUNCTION_SI;
					bp->mf_config[vn] = MF_CFG_RD(bp,
						   func_mf_config[func].config);
				} else
M
Merav Sicron 已提交
9830
					BNX2X_DEV_INFO("illegal MAC address for SI\n");
9831 9832 9833 9834 9835 9836 9837 9838 9839 9840 9841 9842
				break;
			case SHARED_FEAT_CFG_FORCE_SF_MODE_MF_ALLOWED:
				/* get OV configuration */
				val = MF_CFG_RD(bp,
					func_mf_config[FUNC_0].e1hov_tag);
				val &= FUNC_MF_CFG_E1HOV_TAG_MASK;

				if (val != FUNC_MF_CFG_E1HOV_TAG_DEFAULT) {
					bp->mf_mode = MULTI_FUNCTION_SD;
					bp->mf_config[vn] = MF_CFG_RD(bp,
						func_mf_config[func].config);
				} else
D
Dmitry Kravkov 已提交
9843
					BNX2X_DEV_INFO("illegal OV for SD\n");
9844 9845 9846 9847
				break;
			default:
				/* Unknown configuration: reset mf_config */
				bp->mf_config[vn] = 0;
M
Merav Sicron 已提交
9848
				BNX2X_DEV_INFO("unknown MF mode 0x%x\n", val);
9849 9850
			}
		}
E
Eliezer Tamir 已提交
9851

9852
		BNX2X_DEV_INFO("%s function mode\n",
D
Dmitry Kravkov 已提交
9853
			       IS_MF(bp) ? "multi" : "single");
9854

9855 9856 9857 9858
		switch (bp->mf_mode) {
		case MULTI_FUNCTION_SD:
			val = MF_CFG_RD(bp, func_mf_config[func].e1hov_tag) &
			      FUNC_MF_CFG_E1HOV_TAG_MASK;
9859
			if (val != FUNC_MF_CFG_E1HOV_TAG_DEFAULT) {
D
Dmitry Kravkov 已提交
9860
				bp->mf_ov = val;
9861 9862
				bp->path_has_ovlan = true;

M
Merav Sicron 已提交
9863 9864
				BNX2X_DEV_INFO("MF OV for func %d is %d (0x%04x)\n",
					       func, bp->mf_ov, bp->mf_ov);
9865
			} else {
9866
				dev_err(&bp->pdev->dev,
M
Merav Sicron 已提交
9867 9868
					"No valid MF OV for func %d, aborting\n",
					func);
9869
				return -EPERM;
9870
			}
9871 9872
			break;
		case MULTI_FUNCTION_SI:
M
Merav Sicron 已提交
9873 9874
			BNX2X_DEV_INFO("func %d is in MF switch-independent mode\n",
				       func);
9875 9876 9877
			break;
		default:
			if (vn) {
9878
				dev_err(&bp->pdev->dev,
M
Merav Sicron 已提交
9879 9880
					"VN %d is in a single function mode, aborting\n",
					vn);
9881
				return -EPERM;
9882
			}
9883
			break;
9884
		}
9885

9886 9887 9888 9889 9890 9891 9892 9893 9894 9895 9896 9897 9898 9899 9900 9901
		/* check if other port on the path needs ovlan:
		 * Since MF configuration is shared between ports
		 * Possible mixed modes are only
		 * {SF, SI} {SF, SD} {SD, SF} {SI, SF}
		 */
		if (CHIP_MODE_IS_4_PORT(bp) &&
		    !bp->path_has_ovlan &&
		    !IS_MF(bp) &&
		    bp->common.mf_cfg_base != SHMEM_MF_CFG_ADDR_NONE) {
			u8 other_port = !BP_PORT(bp);
			u8 other_func = BP_PATH(bp) + 2*other_port;
			val = MF_CFG_RD(bp,
					func_mf_config[other_func].e1hov_tag);
			if (val != FUNC_MF_CFG_E1HOV_TAG_DEFAULT)
				bp->path_has_ovlan = true;
		}
9902
	}
E
Eliezer Tamir 已提交
9903

D
Dmitry Kravkov 已提交
9904 9905
	/* adjust igu_sb_cnt to MF for E1x */
	if (CHIP_IS_E1x(bp) && IS_MF(bp))
9906 9907
		bp->igu_sb_cnt /= E1HVN_MAX;

9908 9909
	/* port info */
	bnx2x_get_port_hwinfo(bp);
D
Dmitry Kravkov 已提交
9910

9911 9912
	/* Get MAC addresses */
	bnx2x_get_mac_hwinfo(bp);
E
Eliezer Tamir 已提交
9913

9914 9915
	bnx2x_get_cnic_info(bp);

9916 9917 9918
	return rc;
}

9919 9920 9921
static void __devinit bnx2x_read_fwinfo(struct bnx2x *bp)
{
	int cnt, i, block_end, rodi;
9922
	char vpd_start[BNX2X_VPD_LEN+1];
9923 9924
	char str_id_reg[VENDOR_ID_LEN+1];
	char str_id_cap[VENDOR_ID_LEN+1];
9925 9926
	char *vpd_data;
	char *vpd_extended_data = NULL;
9927 9928
	u8 len;

9929
	cnt = pci_read_vpd(bp->pdev, 0, BNX2X_VPD_LEN, vpd_start);
9930 9931 9932 9933 9934
	memset(bp->fw_ver, 0, sizeof(bp->fw_ver));

	if (cnt < BNX2X_VPD_LEN)
		goto out_not_found;

9935 9936 9937 9938
	/* VPD RO tag should be first tag after identifier string, hence
	 * we should be able to find it in first BNX2X_VPD_LEN chars
	 */
	i = pci_vpd_find_tag(vpd_start, 0, BNX2X_VPD_LEN,
9939 9940 9941 9942 9943
			     PCI_VPD_LRDT_RO_DATA);
	if (i < 0)
		goto out_not_found;

	block_end = i + PCI_VPD_LRDT_TAG_SIZE +
9944
		    pci_vpd_lrdt_size(&vpd_start[i]);
9945 9946 9947

	i += PCI_VPD_LRDT_TAG_SIZE;

9948 9949 9950 9951 9952 9953 9954 9955 9956 9957 9958 9959 9960 9961 9962 9963 9964
	if (block_end > BNX2X_VPD_LEN) {
		vpd_extended_data = kmalloc(block_end, GFP_KERNEL);
		if (vpd_extended_data  == NULL)
			goto out_not_found;

		/* read rest of vpd image into vpd_extended_data */
		memcpy(vpd_extended_data, vpd_start, BNX2X_VPD_LEN);
		cnt = pci_read_vpd(bp->pdev, BNX2X_VPD_LEN,
				   block_end - BNX2X_VPD_LEN,
				   vpd_extended_data + BNX2X_VPD_LEN);
		if (cnt < (block_end - BNX2X_VPD_LEN))
			goto out_not_found;
		vpd_data = vpd_extended_data;
	} else
		vpd_data = vpd_start;

	/* now vpd_data holds full vpd content in both cases */
9965 9966 9967 9968 9969 9970 9971 9972 9973 9974 9975 9976 9977 9978 9979 9980 9981 9982 9983 9984 9985 9986 9987 9988 9989 9990 9991 9992 9993 9994 9995

	rodi = pci_vpd_find_info_keyword(vpd_data, i, block_end,
				   PCI_VPD_RO_KEYWORD_MFR_ID);
	if (rodi < 0)
		goto out_not_found;

	len = pci_vpd_info_field_size(&vpd_data[rodi]);

	if (len != VENDOR_ID_LEN)
		goto out_not_found;

	rodi += PCI_VPD_INFO_FLD_HDR_SIZE;

	/* vendor specific info */
	snprintf(str_id_reg, VENDOR_ID_LEN + 1, "%04x", PCI_VENDOR_ID_DELL);
	snprintf(str_id_cap, VENDOR_ID_LEN + 1, "%04X", PCI_VENDOR_ID_DELL);
	if (!strncmp(str_id_reg, &vpd_data[rodi], VENDOR_ID_LEN) ||
	    !strncmp(str_id_cap, &vpd_data[rodi], VENDOR_ID_LEN)) {

		rodi = pci_vpd_find_info_keyword(vpd_data, i, block_end,
						PCI_VPD_RO_KEYWORD_VENDOR0);
		if (rodi >= 0) {
			len = pci_vpd_info_field_size(&vpd_data[rodi]);

			rodi += PCI_VPD_INFO_FLD_HDR_SIZE;

			if (len < 32 && (len + rodi) <= BNX2X_VPD_LEN) {
				memcpy(bp->fw_ver, &vpd_data[rodi], len);
				bp->fw_ver[len] = ' ';
			}
		}
9996
		kfree(vpd_extended_data);
9997 9998 9999
		return;
	}
out_not_found:
10000
	kfree(vpd_extended_data);
10001 10002 10003
	return;
}

10004 10005 10006 10007 10008 10009 10010 10011 10012 10013 10014 10015 10016 10017 10018 10019 10020 10021 10022 10023 10024 10025
static void __devinit bnx2x_set_modes_bitmap(struct bnx2x *bp)
{
	u32 flags = 0;

	if (CHIP_REV_IS_FPGA(bp))
		SET_FLAGS(flags, MODE_FPGA);
	else if (CHIP_REV_IS_EMUL(bp))
		SET_FLAGS(flags, MODE_EMUL);
	else
		SET_FLAGS(flags, MODE_ASIC);

	if (CHIP_MODE_IS_4_PORT(bp))
		SET_FLAGS(flags, MODE_PORT4);
	else
		SET_FLAGS(flags, MODE_PORT2);

	if (CHIP_IS_E2(bp))
		SET_FLAGS(flags, MODE_E2);
	else if (CHIP_IS_E3(bp)) {
		SET_FLAGS(flags, MODE_E3);
		if (CHIP_REV(bp) == CHIP_REV_Ax)
			SET_FLAGS(flags, MODE_E3_A0);
10026 10027
		else /*if (CHIP_REV(bp) == CHIP_REV_Bx)*/
			SET_FLAGS(flags, MODE_E3_B0 | MODE_COS3);
10028 10029 10030 10031 10032 10033 10034 10035 10036 10037 10038 10039 10040 10041 10042 10043 10044 10045 10046 10047 10048 10049 10050
	}

	if (IS_MF(bp)) {
		SET_FLAGS(flags, MODE_MF);
		switch (bp->mf_mode) {
		case MULTI_FUNCTION_SD:
			SET_FLAGS(flags, MODE_MF_SD);
			break;
		case MULTI_FUNCTION_SI:
			SET_FLAGS(flags, MODE_MF_SI);
			break;
		}
	} else
		SET_FLAGS(flags, MODE_SF);

#if defined(__LITTLE_ENDIAN)
	SET_FLAGS(flags, MODE_LITTLE_ENDIAN);
#else /*(__BIG_ENDIAN)*/
	SET_FLAGS(flags, MODE_BIG_ENDIAN);
#endif
	INIT_MODE_FLAGS(bp) = flags;
}

10051 10052
static int __devinit bnx2x_init_bp(struct bnx2x *bp)
{
D
Dmitry Kravkov 已提交
10053
	int func;
10054 10055 10056
	int rc;

	mutex_init(&bp->port.phy_mutex);
E
Eilon Greenstein 已提交
10057
	mutex_init(&bp->fw_mb_mutex);
10058
	spin_lock_init(&bp->stats_lock);
10059 10060 10061
#ifdef BCM_CNIC
	mutex_init(&bp->cnic_mutex);
#endif
E
Eliezer Tamir 已提交
10062

10063
	INIT_DELAYED_WORK(&bp->sp_task, bnx2x_sp_task);
10064
	INIT_DELAYED_WORK(&bp->sp_rtnl_task, bnx2x_sp_rtnl_task);
10065
	INIT_DELAYED_WORK(&bp->period_task, bnx2x_period_task);
10066
	rc = bnx2x_get_hwinfo(bp);
10067 10068
	if (rc)
		return rc;
10069

10070 10071 10072 10073 10074
	bnx2x_set_modes_bitmap(bp);

	rc = bnx2x_alloc_mem_bp(bp);
	if (rc)
		return rc;
10075

10076
	bnx2x_read_fwinfo(bp);
D
Dmitry Kravkov 已提交
10077 10078 10079

	func = BP_FUNC(bp);

10080 10081 10082 10083 10084
	/* need to reset chip if undi was active */
	if (!BP_NOMCP(bp))
		bnx2x_undi_unload(bp);

	if (CHIP_REV_IS_FPGA(bp))
V
Vladislav Zolotarov 已提交
10085
		dev_err(&bp->pdev->dev, "FPGA detected\n");
10086 10087

	if (BP_NOMCP(bp) && (func == 0))
M
Merav Sicron 已提交
10088
		dev_err(&bp->pdev->dev, "MCP disabled, must load devices in order!\n");
10089

E
Eilon Greenstein 已提交
10090 10091
	bp->multi_mode = multi_mode;

D
Dmitry Kravkov 已提交
10092 10093 10094 10095 10096 10097
	bp->disable_tpa = disable_tpa;

#ifdef BCM_CNIC
	bp->disable_tpa |= IS_MF_ISCSI_SD(bp);
#endif

10098
	/* Set TPA flags */
D
Dmitry Kravkov 已提交
10099
	if (bp->disable_tpa) {
D
Dmitry Kravkov 已提交
10100
		bp->flags &= ~(TPA_ENABLE_FLAG | GRO_ENABLE_FLAG);
10101 10102
		bp->dev->features &= ~NETIF_F_LRO;
	} else {
D
Dmitry Kravkov 已提交
10103
		bp->flags |= (TPA_ENABLE_FLAG | GRO_ENABLE_FLAG);
10104 10105 10106
		bp->dev->features |= NETIF_F_LRO;
	}

10107 10108 10109 10110 10111
	if (CHIP_IS_E1(bp))
		bp->dropless_fc = 0;
	else
		bp->dropless_fc = dropless_fc;

10112
	bp->mrrs = mrrs;
10113

10114 10115
	bp->tx_ring_size = MAX_TX_AVAIL;

10116
	/* make sure that the numbers are in the right granularity */
10117 10118
	bp->tx_ticks = (50 / BNX2X_BTR) * BNX2X_BTR;
	bp->rx_ticks = (25 / BNX2X_BTR) * BNX2X_BTR;
10119

10120
	bp->current_interval = CHIP_REV_IS_SLOW(bp) ? 5*HZ : HZ;
10121 10122 10123 10124 10125 10126

	init_timer(&bp->timer);
	bp->timer.expires = jiffies + bp->current_interval;
	bp->timer.data = (unsigned long) bp;
	bp->timer.function = bnx2x_timer;

S
Shmulik Ravid 已提交
10127
	bnx2x_dcbx_set_state(bp, true, BNX2X_DCBX_ENABLED_ON_NEG_ON);
V
Vladislav Zolotarov 已提交
10128 10129
	bnx2x_dcbx_init_params(bp);

10130 10131 10132 10133 10134 10135 10136
#ifdef BCM_CNIC
	if (CHIP_IS_E1x(bp))
		bp->cnic_base_cl_id = FP_SB_MAX_E1x;
	else
		bp->cnic_base_cl_id = FP_SB_MAX_E2;
#endif

10137 10138 10139 10140 10141 10142 10143 10144
	/* multiple tx priority */
	if (CHIP_IS_E1x(bp))
		bp->max_cos = BNX2X_MULTI_TX_COS_E1X;
	if (CHIP_IS_E2(bp) || CHIP_IS_E3A0(bp))
		bp->max_cos = BNX2X_MULTI_TX_COS_E2_E3A0;
	if (CHIP_IS_E3B0(bp))
		bp->max_cos = BNX2X_MULTI_TX_COS_E3B0;

D
Dmitry Kravkov 已提交
10145 10146
	bp->gro_check = bnx2x_need_gro_check(bp->dev->mtu);

10147
	return rc;
E
Eliezer Tamir 已提交
10148 10149 10150
}


10151 10152 10153
/****************************************************************************
* General service functions
****************************************************************************/
E
Eliezer Tamir 已提交
10154

10155 10156 10157 10158
/*
 * net_device service functions
 */

Y
Yitchak Gertner 已提交
10159
/* called with rtnl_lock */
E
Eliezer Tamir 已提交
10160 10161 10162
static int bnx2x_open(struct net_device *dev)
{
	struct bnx2x *bp = netdev_priv(dev);
10163 10164
	bool global = false;
	int other_engine = BP_PATH(bp) ? 0 : 1;
10165
	bool other_load_status, load_status;
E
Eliezer Tamir 已提交
10166

10167 10168
	bp->stats_init = true;

E
Eilon Greenstein 已提交
10169 10170
	netif_carrier_off(dev);

E
Eliezer Tamir 已提交
10171 10172
	bnx2x_set_power_state(bp, PCI_D0);

10173 10174
	other_load_status = bnx2x_get_load_status(bp, other_engine);
	load_status = bnx2x_get_load_status(bp, BP_PATH(bp));
10175 10176 10177 10178 10179 10180 10181 10182 10183

	/*
	 * If parity had happen during the unload, then attentions
	 * and/or RECOVERY_IN_PROGRES may still be set. In this case we
	 * want the first function loaded on the current engine to
	 * complete the recovery.
	 */
	if (!bnx2x_reset_is_done(bp, BP_PATH(bp)) ||
	    bnx2x_chk_parity_attn(bp, &global, true))
10184
		do {
10185 10186 10187 10188 10189
			/*
			 * If there are attentions and they are in a global
			 * blocks, set the GLOBAL_RESET bit regardless whether
			 * it will be this function that will complete the
			 * recovery or not.
10190
			 */
10191 10192
			if (global)
				bnx2x_set_reset_global(bp);
10193

10194 10195 10196 10197 10198
			/*
			 * Only the first function on the current engine should
			 * try to recover in open. In case of attentions in
			 * global blocks only the first in the chip should try
			 * to recover.
10199
			 */
10200 10201
			if ((!load_status &&
			     (!global || !other_load_status)) &&
10202 10203 10204
			    bnx2x_trylock_leader_lock(bp) &&
			    !bnx2x_leader_reset(bp)) {
				netdev_info(bp->dev, "Recovered in open\n");
10205 10206 10207
				break;
			}

10208
			/* recovery has failed... */
10209
			bnx2x_set_power_state(bp, PCI_D3hot);
10210
			bp->recovery_state = BNX2X_RECOVERY_FAILED;
10211

M
Merav Sicron 已提交
10212 10213
			BNX2X_ERR("Recovery flow hasn't been properly completed yet. Try again later.\n"
				  "If you still see this message after a few retries then power cycle is required.\n");
10214 10215 10216 10217 10218

			return -EAGAIN;
		} while (0);

	bp->recovery_state = BNX2X_RECOVERY_DONE;
Y
Yitchak Gertner 已提交
10219
	return bnx2x_nic_load(bp, LOAD_OPEN);
E
Eliezer Tamir 已提交
10220 10221
}

Y
Yitchak Gertner 已提交
10222
/* called with rtnl_lock */
10223
static int bnx2x_close(struct net_device *dev)
E
Eliezer Tamir 已提交
10224 10225 10226 10227
{
	struct bnx2x *bp = netdev_priv(dev);

	/* Unload the driver, release IRQs */
Y
Yitchak Gertner 已提交
10228
	bnx2x_nic_unload(bp, UNLOAD_CLOSE);
10229 10230

	/* Power off */
10231
	bnx2x_set_power_state(bp, PCI_D3hot);
E
Eliezer Tamir 已提交
10232 10233 10234 10235

	return 0;
}

10236 10237
static inline int bnx2x_init_mcast_macs_list(struct bnx2x *bp,
					 struct bnx2x_mcast_ramrod_params *p)
10238
{
10239 10240 10241 10242
	int mc_count = netdev_mc_count(bp->dev);
	struct bnx2x_mcast_list_elem *mc_mac =
		kzalloc(sizeof(*mc_mac) * mc_count, GFP_ATOMIC);
	struct netdev_hw_addr *ha;
10243

10244 10245
	if (!mc_mac)
		return -ENOMEM;
10246

10247
	INIT_LIST_HEAD(&p->mcast_list);
10248

10249 10250 10251 10252
	netdev_for_each_mc_addr(ha, bp->dev) {
		mc_mac->mac = bnx2x_mc_addr(ha);
		list_add_tail(&mc_mac->link, &p->mcast_list);
		mc_mac++;
10253
	}
10254 10255 10256 10257

	p->mcast_list_len = mc_count;

	return 0;
10258 10259
}

10260 10261 10262 10263 10264 10265 10266 10267 10268 10269 10270 10271 10272 10273 10274
static inline void bnx2x_free_mcast_macs_list(
	struct bnx2x_mcast_ramrod_params *p)
{
	struct bnx2x_mcast_list_elem *mc_mac =
		list_first_entry(&p->mcast_list, struct bnx2x_mcast_list_elem,
				 link);

	WARN_ON(!mc_mac);
	kfree(mc_mac);
}

/**
 * bnx2x_set_uc_list - configure a new unicast MACs list.
 *
 * @bp: driver handle
10275
 *
10276
 * We will use zero (0) as a MAC type for these MACs.
10277
 */
10278
static inline int bnx2x_set_uc_list(struct bnx2x *bp)
10279
{
10280
	int rc;
10281 10282
	struct net_device *dev = bp->dev;
	struct netdev_hw_addr *ha;
10283 10284
	struct bnx2x_vlan_mac_obj *mac_obj = &bp->fp->mac_obj;
	unsigned long ramrod_flags = 0;
10285

10286 10287 10288 10289 10290 10291
	/* First schedule a cleanup up of old configuration */
	rc = bnx2x_del_all_macs(bp, mac_obj, BNX2X_UC_LIST_MAC, false);
	if (rc < 0) {
		BNX2X_ERR("Failed to schedule DELETE operations: %d\n", rc);
		return rc;
	}
10292 10293

	netdev_for_each_uc_addr(ha, dev) {
10294 10295 10296 10297 10298 10299
		rc = bnx2x_set_mac_one(bp, bnx2x_uc_addr(ha), mac_obj, true,
				       BNX2X_UC_LIST_MAC, &ramrod_flags);
		if (rc < 0) {
			BNX2X_ERR("Failed to schedule ADD operations: %d\n",
				  rc);
			return rc;
10300 10301 10302
		}
	}

10303 10304 10305 10306
	/* Execute the pending commands */
	__set_bit(RAMROD_CONT, &ramrod_flags);
	return bnx2x_set_mac_one(bp, NULL, mac_obj, false /* don't care */,
				 BNX2X_UC_LIST_MAC, &ramrod_flags);
10307 10308
}

10309
static inline int bnx2x_set_mc_list(struct bnx2x *bp)
10310
{
10311
	struct net_device *dev = bp->dev;
Y
Yuval Mintz 已提交
10312
	struct bnx2x_mcast_ramrod_params rparam = {NULL};
10313
	int rc = 0;
10314

10315
	rparam.mcast_obj = &bp->mcast_obj;
10316

10317 10318 10319
	/* first, clear all configured multicast MACs */
	rc = bnx2x_config_mcast(bp, &rparam, BNX2X_MCAST_CMD_DEL);
	if (rc < 0) {
M
Merav Sicron 已提交
10320
		BNX2X_ERR("Failed to clear multicast configuration: %d\n", rc);
10321 10322
		return rc;
	}
10323

10324 10325 10326 10327
	/* then, configure a new MACs list */
	if (netdev_mc_count(dev)) {
		rc = bnx2x_init_mcast_macs_list(bp, &rparam);
		if (rc) {
M
Merav Sicron 已提交
10328 10329
			BNX2X_ERR("Failed to create multicast MACs list: %d\n",
				  rc);
10330 10331
			return rc;
		}
10332

10333 10334 10335 10336
		/* Now add the new MACs */
		rc = bnx2x_config_mcast(bp, &rparam,
					BNX2X_MCAST_CMD_ADD);
		if (rc < 0)
M
Merav Sicron 已提交
10337 10338
			BNX2X_ERR("Failed to set a new multicast configuration: %d\n",
				  rc);
10339

10340 10341
		bnx2x_free_mcast_macs_list(&rparam);
	}
10342

10343
	return rc;
10344 10345 10346
}


10347
/* If bp->state is OPEN, should be called with netif_addr_lock_bh() */
D
Dmitry Kravkov 已提交
10348
void bnx2x_set_rx_mode(struct net_device *dev)
10349 10350 10351 10352 10353 10354 10355 10356 10357
{
	struct bnx2x *bp = netdev_priv(dev);
	u32 rx_mode = BNX2X_RX_MODE_NORMAL;

	if (bp->state != BNX2X_STATE_OPEN) {
		DP(NETIF_MSG_IFUP, "state is %x, returning\n", bp->state);
		return;
	}

10358
	DP(NETIF_MSG_IFUP, "dev->flags = %x\n", bp->dev->flags);
10359 10360 10361

	if (dev->flags & IFF_PROMISC)
		rx_mode = BNX2X_RX_MODE_PROMISC;
10362 10363 10364
	else if ((dev->flags & IFF_ALLMULTI) ||
		 ((netdev_mc_count(dev) > BNX2X_MAX_MULTICAST) &&
		  CHIP_IS_E1(bp)))
10365
		rx_mode = BNX2X_RX_MODE_ALLMULTI;
10366 10367
	else {
		/* some multicasts */
10368
		if (bnx2x_set_mc_list(bp) < 0)
10369
			rx_mode = BNX2X_RX_MODE_ALLMULTI;
10370

10371
		if (bnx2x_set_uc_list(bp) < 0)
10372
			rx_mode = BNX2X_RX_MODE_PROMISC;
10373 10374 10375
	}

	bp->rx_mode = rx_mode;
D
Dmitry Kravkov 已提交
10376 10377 10378 10379 10380
#ifdef BCM_CNIC
	/* handle ISCSI SD mode */
	if (IS_MF_ISCSI_SD(bp))
		bp->rx_mode = BNX2X_RX_MODE_NONE;
#endif
10381 10382 10383 10384 10385 10386 10387

	/* Schedule the rx_mode command */
	if (test_bit(BNX2X_FILTER_RX_MODE_PENDING, &bp->sp_state)) {
		set_bit(BNX2X_FILTER_RX_MODE_SCHED, &bp->sp_state);
		return;
	}

10388 10389 10390
	bnx2x_set_storm_rx_mode(bp);
}

Y
Yaniv Rosner 已提交
10391
/* called with rtnl_lock */
E
Eilon Greenstein 已提交
10392 10393
static int bnx2x_mdio_read(struct net_device *netdev, int prtad,
			   int devad, u16 addr)
E
Eliezer Tamir 已提交
10394
{
E
Eilon Greenstein 已提交
10395 10396 10397
	struct bnx2x *bp = netdev_priv(netdev);
	u16 value;
	int rc;
E
Eliezer Tamir 已提交
10398

E
Eilon Greenstein 已提交
10399 10400
	DP(NETIF_MSG_LINK, "mdio_read: prtad 0x%x, devad 0x%x, addr 0x%x\n",
	   prtad, devad, addr);
E
Eliezer Tamir 已提交
10401

E
Eilon Greenstein 已提交
10402 10403
	/* The HW expects different devad if CL22 is used */
	devad = (devad == MDIO_DEVAD_NONE) ? DEFAULT_PHY_DEV_ADDR : devad;
Y
Yaniv Rosner 已提交
10404

E
Eilon Greenstein 已提交
10405
	bnx2x_acquire_phy_lock(bp);
Y
Yaniv Rosner 已提交
10406
	rc = bnx2x_phy_read(&bp->link_params, prtad, devad, addr, &value);
E
Eilon Greenstein 已提交
10407 10408
	bnx2x_release_phy_lock(bp);
	DP(NETIF_MSG_LINK, "mdio_read_val 0x%x rc = 0x%x\n", value, rc);
E
Eliezer Tamir 已提交
10409

E
Eilon Greenstein 已提交
10410 10411 10412 10413
	if (!rc)
		rc = value;
	return rc;
}
E
Eliezer Tamir 已提交
10414

E
Eilon Greenstein 已提交
10415 10416 10417 10418 10419 10420 10421
/* called with rtnl_lock */
static int bnx2x_mdio_write(struct net_device *netdev, int prtad, int devad,
			    u16 addr, u16 value)
{
	struct bnx2x *bp = netdev_priv(netdev);
	int rc;

M
Merav Sicron 已提交
10422 10423 10424
	DP(NETIF_MSG_LINK,
	   "mdio_write: prtad 0x%x, devad 0x%x, addr 0x%x, value 0x%x\n",
	   prtad, devad, addr, value);
E
Eilon Greenstein 已提交
10425 10426 10427

	/* The HW expects different devad if CL22 is used */
	devad = (devad == MDIO_DEVAD_NONE) ? DEFAULT_PHY_DEV_ADDR : devad;
E
Eliezer Tamir 已提交
10428

E
Eilon Greenstein 已提交
10429
	bnx2x_acquire_phy_lock(bp);
Y
Yaniv Rosner 已提交
10430
	rc = bnx2x_phy_write(&bp->link_params, prtad, devad, addr, value);
E
Eilon Greenstein 已提交
10431 10432 10433
	bnx2x_release_phy_lock(bp);
	return rc;
}
Y
Yaniv Rosner 已提交
10434

E
Eilon Greenstein 已提交
10435 10436 10437 10438 10439
/* called with rtnl_lock */
static int bnx2x_ioctl(struct net_device *dev, struct ifreq *ifr, int cmd)
{
	struct bnx2x *bp = netdev_priv(dev);
	struct mii_ioctl_data *mdio = if_mii(ifr);
E
Eliezer Tamir 已提交
10440

E
Eilon Greenstein 已提交
10441 10442
	DP(NETIF_MSG_LINK, "ioctl: phy id 0x%x, reg 0x%x, val_in 0x%x\n",
	   mdio->phy_id, mdio->reg_num, mdio->val_in);
E
Eliezer Tamir 已提交
10443

E
Eilon Greenstein 已提交
10444 10445 10446 10447
	if (!netif_running(dev))
		return -EAGAIN;

	return mdio_mii_ioctl(&bp->mdio, mdio, cmd);
E
Eliezer Tamir 已提交
10448 10449
}

A
Alexey Dobriyan 已提交
10450
#ifdef CONFIG_NET_POLL_CONTROLLER
E
Eliezer Tamir 已提交
10451 10452 10453 10454 10455 10456 10457 10458 10459 10460
static void poll_bnx2x(struct net_device *dev)
{
	struct bnx2x *bp = netdev_priv(dev);

	disable_irq(bp->pdev->irq);
	bnx2x_interrupt(bp->pdev->irq, dev);
	enable_irq(bp->pdev->irq);
}
#endif

D
Dmitry Kravkov 已提交
10461 10462 10463 10464
static int bnx2x_validate_addr(struct net_device *dev)
{
	struct bnx2x *bp = netdev_priv(dev);

M
Merav Sicron 已提交
10465 10466
	if (!bnx2x_is_valid_ether_addr(bp, dev->dev_addr)) {
		BNX2X_ERR("Non-valid Ethernet address\n");
D
Dmitry Kravkov 已提交
10467
		return -EADDRNOTAVAIL;
M
Merav Sicron 已提交
10468
	}
D
Dmitry Kravkov 已提交
10469 10470 10471
	return 0;
}

10472 10473 10474 10475
static const struct net_device_ops bnx2x_netdev_ops = {
	.ndo_open		= bnx2x_open,
	.ndo_stop		= bnx2x_close,
	.ndo_start_xmit		= bnx2x_start_xmit,
10476
	.ndo_select_queue	= bnx2x_select_queue,
10477
	.ndo_set_rx_mode	= bnx2x_set_rx_mode,
10478
	.ndo_set_mac_address	= bnx2x_change_mac_addr,
D
Dmitry Kravkov 已提交
10479
	.ndo_validate_addr	= bnx2x_validate_addr,
10480 10481
	.ndo_do_ioctl		= bnx2x_ioctl,
	.ndo_change_mtu		= bnx2x_change_mtu,
10482 10483
	.ndo_fix_features	= bnx2x_fix_features,
	.ndo_set_features	= bnx2x_set_features,
10484
	.ndo_tx_timeout		= bnx2x_tx_timeout,
A
Alexey Dobriyan 已提交
10485
#ifdef CONFIG_NET_POLL_CONTROLLER
10486 10487
	.ndo_poll_controller	= poll_bnx2x,
#endif
10488 10489
	.ndo_setup_tc		= bnx2x_setup_tc,

10490 10491 10492
#if defined(NETDEV_FCOE_WWNN) && defined(BCM_CNIC)
	.ndo_fcoe_get_wwn	= bnx2x_fcoe_get_wwn,
#endif
10493 10494
};

10495 10496 10497 10498 10499 10500 10501
static inline int bnx2x_set_coherency_mask(struct bnx2x *bp)
{
	struct device *dev = &bp->pdev->dev;

	if (dma_set_mask(dev, DMA_BIT_MASK(64)) == 0) {
		bp->flags |= USING_DAC_FLAG;
		if (dma_set_coherent_mask(dev, DMA_BIT_MASK(64)) != 0) {
M
Merav Sicron 已提交
10502
			dev_err(dev, "dma_set_coherent_mask failed, aborting\n");
10503 10504 10505 10506 10507 10508 10509 10510 10511 10512
			return -EIO;
		}
	} else if (dma_set_mask(dev, DMA_BIT_MASK(32)) != 0) {
		dev_err(dev, "System does not support DMA, aborting\n");
		return -EIO;
	}

	return 0;
}

10513
static int __devinit bnx2x_init_dev(struct pci_dev *pdev,
10514 10515
				    struct net_device *dev,
				    unsigned long board_type)
E
Eliezer Tamir 已提交
10516 10517 10518
{
	struct bnx2x *bp;
	int rc;
10519
	u32 pci_cfg_dword;
10520 10521 10522
	bool chip_is_e1x = (board_type == BCM57710 ||
			    board_type == BCM57711 ||
			    board_type == BCM57711E);
E
Eliezer Tamir 已提交
10523 10524 10525 10526

	SET_NETDEV_DEV(dev, &pdev->dev);
	bp = netdev_priv(dev);

10527 10528
	bp->dev = dev;
	bp->pdev = pdev;
E
Eliezer Tamir 已提交
10529 10530 10531 10532
	bp->flags = 0;

	rc = pci_enable_device(pdev);
	if (rc) {
V
Vladislav Zolotarov 已提交
10533 10534
		dev_err(&bp->pdev->dev,
			"Cannot enable PCI device, aborting\n");
E
Eliezer Tamir 已提交
10535 10536 10537 10538
		goto err_out;
	}

	if (!(pci_resource_flags(pdev, 0) & IORESOURCE_MEM)) {
V
Vladislav Zolotarov 已提交
10539 10540
		dev_err(&bp->pdev->dev,
			"Cannot find PCI device base address, aborting\n");
E
Eliezer Tamir 已提交
10541 10542 10543 10544 10545
		rc = -ENODEV;
		goto err_out_disable;
	}

	if (!(pci_resource_flags(pdev, 2) & IORESOURCE_MEM)) {
V
Vladislav Zolotarov 已提交
10546 10547
		dev_err(&bp->pdev->dev, "Cannot find second PCI device"
		       " base address, aborting\n");
E
Eliezer Tamir 已提交
10548 10549 10550 10551
		rc = -ENODEV;
		goto err_out_disable;
	}

10552 10553 10554
	if (atomic_read(&pdev->enable_cnt) == 1) {
		rc = pci_request_regions(pdev, DRV_MODULE_NAME);
		if (rc) {
V
Vladislav Zolotarov 已提交
10555 10556
			dev_err(&bp->pdev->dev,
				"Cannot obtain PCI resources, aborting\n");
10557 10558
			goto err_out_disable;
		}
E
Eliezer Tamir 已提交
10559

10560 10561 10562
		pci_set_master(pdev);
		pci_save_state(pdev);
	}
E
Eliezer Tamir 已提交
10563 10564 10565

	bp->pm_cap = pci_find_capability(pdev, PCI_CAP_ID_PM);
	if (bp->pm_cap == 0) {
V
Vladislav Zolotarov 已提交
10566 10567
		dev_err(&bp->pdev->dev,
			"Cannot find power management capability, aborting\n");
E
Eliezer Tamir 已提交
10568 10569 10570 10571
		rc = -EIO;
		goto err_out_release;
	}

10572
	if (!pci_is_pcie(pdev)) {
M
Merav Sicron 已提交
10573
		dev_err(&bp->pdev->dev, "Not PCI Express, aborting\n");
E
Eliezer Tamir 已提交
10574 10575 10576 10577
		rc = -EIO;
		goto err_out_release;
	}

10578 10579
	rc = bnx2x_set_coherency_mask(bp);
	if (rc)
E
Eliezer Tamir 已提交
10580 10581
		goto err_out_release;

10582 10583 10584
	dev->mem_start = pci_resource_start(pdev, 0);
	dev->base_addr = dev->mem_start;
	dev->mem_end = pci_resource_end(pdev, 0);
E
Eliezer Tamir 已提交
10585 10586 10587

	dev->irq = pdev->irq;

10588
	bp->regview = pci_ioremap_bar(pdev, 0);
E
Eliezer Tamir 已提交
10589
	if (!bp->regview) {
V
Vladislav Zolotarov 已提交
10590 10591
		dev_err(&bp->pdev->dev,
			"Cannot map register space, aborting\n");
E
Eliezer Tamir 已提交
10592 10593 10594 10595
		rc = -ENOMEM;
		goto err_out_release;
	}

10596 10597 10598 10599 10600 10601 10602 10603 10604 10605 10606 10607 10608
	/* In E1/E1H use pci device function given by kernel.
	 * In E2/E3 read physical function from ME register since these chips
	 * support Physical Device Assignment where kernel BDF maybe arbitrary
	 * (depending on hypervisor).
	 */
	if (chip_is_e1x)
		bp->pf_num = PCI_FUNC(pdev->devfn);
	else {/* chip is E2/3*/
		pci_read_config_dword(bp->pdev,
				      PCICFG_ME_REGISTER, &pci_cfg_dword);
		bp->pf_num = (u8)((pci_cfg_dword & ME_REG_ABS_PF_NUM) >>
		    ME_REG_ABS_PF_NUM_SHIFT);
	}
M
Merav Sicron 已提交
10609
	BNX2X_DEV_INFO("me reg PF num: %d\n", bp->pf_num);
10610

E
Eliezer Tamir 已提交
10611 10612
	bnx2x_set_power_state(bp, PCI_D0);

10613 10614 10615
	/* clean indirect addresses */
	pci_write_config_dword(bp->pdev, PCICFG_GRC_ADDRESS,
			       PCICFG_VENDOR_ID_OFFSET);
10616 10617
	/*
	 * Clean the following indirect addresses for all functions since it
10618 10619 10620 10621 10622 10623
	 * is not used by the driver.
	 */
	REG_WR(bp, PXP2_REG_PGL_ADDR_88_F0, 0);
	REG_WR(bp, PXP2_REG_PGL_ADDR_8C_F0, 0);
	REG_WR(bp, PXP2_REG_PGL_ADDR_90_F0, 0);
	REG_WR(bp, PXP2_REG_PGL_ADDR_94_F0, 0);
10624

10625
	if (chip_is_e1x) {
10626 10627 10628 10629 10630
		REG_WR(bp, PXP2_REG_PGL_ADDR_88_F1, 0);
		REG_WR(bp, PXP2_REG_PGL_ADDR_8C_F1, 0);
		REG_WR(bp, PXP2_REG_PGL_ADDR_90_F1, 0);
		REG_WR(bp, PXP2_REG_PGL_ADDR_94_F1, 0);
	}
E
Eliezer Tamir 已提交
10631

10632
	/*
10633
	 * Enable internal target-read (in case we are probed after PF FLR).
10634
	 * Must be done prior to any BAR read access. Only for 57712 and up
10635
	 */
10636
	if (!chip_is_e1x)
10637
		REG_WR(bp, PGLUE_B_REG_INTERNAL_PFID_ENABLE_TARGET_READ, 1);
10638

10639
	/* Reset the load counter */
10640
	bnx2x_clear_load_status(bp);
10641

10642
	dev->watchdog_timeo = TX_TIMEOUT;
E
Eliezer Tamir 已提交
10643

10644
	dev->netdev_ops = &bnx2x_netdev_ops;
10645
	bnx2x_set_ethtool_ops(dev);
E
Eilon Greenstein 已提交
10646

10647 10648
	dev->priv_flags |= IFF_UNICAST_FLT;

10649
	dev->hw_features = NETIF_F_SG | NETIF_F_IP_CSUM | NETIF_F_IPV6_CSUM |
D
Dmitry Kravkov 已提交
10650 10651 10652
		NETIF_F_TSO | NETIF_F_TSO_ECN | NETIF_F_TSO6 |
		NETIF_F_RXCSUM | NETIF_F_LRO | NETIF_F_GRO |
		NETIF_F_RXHASH | NETIF_F_HW_VLAN_TX;
10653 10654 10655 10656 10657

	dev->vlan_features = NETIF_F_SG | NETIF_F_IP_CSUM | NETIF_F_IPV6_CSUM |
		NETIF_F_TSO | NETIF_F_TSO_ECN | NETIF_F_TSO6 | NETIF_F_HIGHDMA;

	dev->features |= dev->hw_features | NETIF_F_HW_VLAN_RX;
E
Eilon Greenstein 已提交
10658
	if (bp->flags & USING_DAC_FLAG)
10659
		dev->features |= NETIF_F_HIGHDMA;
E
Eliezer Tamir 已提交
10660

10661 10662 10663
	/* Add Loopback capability to the device */
	dev->hw_features |= NETIF_F_LOOPBACK;

10664
#ifdef BCM_DCBNL
S
Shmulik Ravid 已提交
10665 10666 10667
	dev->dcbnl_ops = &bnx2x_dcbnl_ops;
#endif

E
Eilon Greenstein 已提交
10668 10669 10670 10671 10672 10673 10674 10675
	/* get_port_hwinfo() will set prtad and mmds properly */
	bp->mdio.prtad = MDIO_PRTAD_NONE;
	bp->mdio.mmds = 0;
	bp->mdio.mode_support = MDIO_SUPPORTS_C45 | MDIO_EMULATE_C22;
	bp->mdio.dev = dev;
	bp->mdio.mdio_read = bnx2x_mdio_read;
	bp->mdio.mdio_write = bnx2x_mdio_write;

E
Eliezer Tamir 已提交
10676 10677 10678
	return 0;

err_out_release:
10679 10680
	if (atomic_read(&pdev->enable_cnt) == 1)
		pci_release_regions(pdev);
E
Eliezer Tamir 已提交
10681 10682 10683 10684 10685 10686 10687 10688 10689

err_out_disable:
	pci_disable_device(pdev);
	pci_set_drvdata(pdev, NULL);

err_out:
	return rc;
}

10690 10691
static void __devinit bnx2x_get_pcie_width_speed(struct bnx2x *bp,
						 int *width, int *speed)
E
Eliezer Tamir 已提交
10692 10693 10694
{
	u32 val = REG_RD(bp, PCICFG_OFFSET + PCICFG_LINK_CONTROL);

10695
	*width = (val & PCICFG_LINK_WIDTH) >> PCICFG_LINK_WIDTH_SHIFT;
E
Eliezer Tamir 已提交
10696

10697 10698
	/* return value of 1=2.5GHz 2=5GHz */
	*speed = (val & PCICFG_LINK_SPEED) >> PCICFG_LINK_SPEED_SHIFT;
E
Eliezer Tamir 已提交
10699
}
10700

10701
static int bnx2x_check_firmware(struct bnx2x *bp)
10702
{
10703
	const struct firmware *firmware = bp->firmware;
10704 10705 10706
	struct bnx2x_fw_file_hdr *fw_hdr;
	struct bnx2x_fw_file_section *sections;
	u32 offset, len, num_ops;
10707
	u16 *ops_offsets;
10708
	int i;
10709
	const u8 *fw_ver;
10710

M
Merav Sicron 已提交
10711 10712
	if (firmware->size < sizeof(struct bnx2x_fw_file_hdr)) {
		BNX2X_ERR("Wrong FW size\n");
10713
		return -EINVAL;
M
Merav Sicron 已提交
10714
	}
10715 10716 10717 10718 10719 10720 10721 10722 10723 10724

	fw_hdr = (struct bnx2x_fw_file_hdr *)firmware->data;
	sections = (struct bnx2x_fw_file_section *)fw_hdr;

	/* Make sure none of the offsets and sizes make us read beyond
	 * the end of the firmware data */
	for (i = 0; i < sizeof(*fw_hdr) / sizeof(*sections); i++) {
		offset = be32_to_cpu(sections[i].offset);
		len = be32_to_cpu(sections[i].len);
		if (offset + len > firmware->size) {
M
Merav Sicron 已提交
10725
			BNX2X_ERR("Section %d length is out of bounds\n", i);
10726 10727 10728 10729 10730 10731 10732 10733 10734 10735 10736
			return -EINVAL;
		}
	}

	/* Likewise for the init_ops offsets */
	offset = be32_to_cpu(fw_hdr->init_ops_offsets.offset);
	ops_offsets = (u16 *)(firmware->data + offset);
	num_ops = be32_to_cpu(fw_hdr->init_ops.len) / sizeof(struct raw_op);

	for (i = 0; i < be32_to_cpu(fw_hdr->init_ops_offsets.len) / 2; i++) {
		if (be16_to_cpu(ops_offsets[i]) > num_ops) {
M
Merav Sicron 已提交
10737
			BNX2X_ERR("Section offset %d is out of bounds\n", i);
10738 10739 10740 10741 10742 10743 10744 10745 10746 10747 10748
			return -EINVAL;
		}
	}

	/* Check FW version */
	offset = be32_to_cpu(fw_hdr->fw_version.offset);
	fw_ver = firmware->data + offset;
	if ((fw_ver[0] != BCM_5710_FW_MAJOR_VERSION) ||
	    (fw_ver[1] != BCM_5710_FW_MINOR_VERSION) ||
	    (fw_ver[2] != BCM_5710_FW_REVISION_VERSION) ||
	    (fw_ver[3] != BCM_5710_FW_ENGINEERING_VERSION)) {
M
Merav Sicron 已提交
10749 10750 10751
		BNX2X_ERR("Bad FW version:%d.%d.%d.%d. Should be %d.%d.%d.%d\n",
		       fw_ver[0], fw_ver[1], fw_ver[2], fw_ver[3],
		       BCM_5710_FW_MAJOR_VERSION,
10752 10753 10754
		       BCM_5710_FW_MINOR_VERSION,
		       BCM_5710_FW_REVISION_VERSION,
		       BCM_5710_FW_ENGINEERING_VERSION);
10755
		return -EINVAL;
10756 10757 10758 10759 10760
	}

	return 0;
}

10761
static inline void be32_to_cpu_n(const u8 *_source, u8 *_target, u32 n)
10762
{
10763 10764
	const __be32 *source = (const __be32 *)_source;
	u32 *target = (u32 *)_target;
10765 10766 10767 10768 10769 10770 10771 10772 10773 10774
	u32 i;

	for (i = 0; i < n/4; i++)
		target[i] = be32_to_cpu(source[i]);
}

/*
   Ops array is stored in the following format:
   {op(8bit), offset(24bit, big endian), data(32bit, big endian)}
 */
10775
static inline void bnx2x_prep_ops(const u8 *_source, u8 *_target, u32 n)
10776
{
10777 10778
	const __be32 *source = (const __be32 *)_source;
	struct raw_op *target = (struct raw_op *)_target;
10779 10780
	u32 i, j, tmp;

10781
	for (i = 0, j = 0; i < n/8; i++, j += 2) {
10782 10783
		tmp = be32_to_cpu(source[j]);
		target[i].op = (tmp >> 24) & 0xff;
V
Vladislav Zolotarov 已提交
10784 10785
		target[i].offset = tmp & 0xffffff;
		target[i].raw_data = be32_to_cpu(source[j + 1]);
10786 10787
	}
}
10788

10789 10790 10791 10792 10793 10794 10795 10796 10797 10798 10799 10800 10801 10802 10803 10804 10805 10806 10807 10808 10809 10810 10811 10812
/**
 * IRO array is stored in the following format:
 * {base(24bit), m1(16bit), m2(16bit), m3(16bit), size(16bit) }
 */
static inline void bnx2x_prep_iro(const u8 *_source, u8 *_target, u32 n)
{
	const __be32 *source = (const __be32 *)_source;
	struct iro *target = (struct iro *)_target;
	u32 i, j, tmp;

	for (i = 0, j = 0; i < n/sizeof(struct iro); i++) {
		target[i].base = be32_to_cpu(source[j]);
		j++;
		tmp = be32_to_cpu(source[j]);
		target[i].m1 = (tmp >> 16) & 0xffff;
		target[i].m2 = tmp & 0xffff;
		j++;
		tmp = be32_to_cpu(source[j]);
		target[i].m3 = (tmp >> 16) & 0xffff;
		target[i].size = tmp & 0xffff;
		j++;
	}
}

10813
static inline void be16_to_cpu_n(const u8 *_source, u8 *_target, u32 n)
10814
{
10815 10816
	const __be16 *source = (const __be16 *)_source;
	u16 *target = (u16 *)_target;
10817 10818 10819 10820 10821 10822
	u32 i;

	for (i = 0; i < n/2; i++)
		target[i] = be16_to_cpu(source[i]);
}

10823 10824 10825 10826
#define BNX2X_ALLOC_AND_SET(arr, lbl, func)				\
do {									\
	u32 len = be32_to_cpu(fw_hdr->arr.len);				\
	bp->arr = kmalloc(len, GFP_KERNEL);				\
10827
	if (!bp->arr)							\
10828 10829 10830 10831
		goto lbl;						\
	func(bp->firmware->data + be32_to_cpu(fw_hdr->arr.offset),	\
	     (u8 *)bp->arr, len);					\
} while (0)
10832

Y
Yuval Mintz 已提交
10833
static int bnx2x_init_firmware(struct bnx2x *bp)
10834
{
10835
	const char *fw_file_name;
10836
	struct bnx2x_fw_file_hdr *fw_hdr;
B
Ben Hutchings 已提交
10837
	int rc;
10838

10839 10840
	if (bp->firmware)
		return 0;
10841

10842 10843 10844 10845 10846 10847 10848 10849 10850 10851 10852
	if (CHIP_IS_E1(bp))
		fw_file_name = FW_FILE_NAME_E1;
	else if (CHIP_IS_E1H(bp))
		fw_file_name = FW_FILE_NAME_E1H;
	else if (!CHIP_IS_E1x(bp))
		fw_file_name = FW_FILE_NAME_E2;
	else {
		BNX2X_ERR("Unsupported chip revision\n");
		return -EINVAL;
	}
	BNX2X_DEV_INFO("Loading %s\n", fw_file_name);
10853

10854 10855 10856 10857 10858 10859
	rc = request_firmware(&bp->firmware, fw_file_name, &bp->pdev->dev);
	if (rc) {
		BNX2X_ERR("Can't load firmware file %s\n",
			  fw_file_name);
		goto request_firmware_exit;
	}
10860

10861 10862 10863 10864
	rc = bnx2x_check_firmware(bp);
	if (rc) {
		BNX2X_ERR("Corrupt firmware file %s\n", fw_file_name);
		goto request_firmware_exit;
10865 10866 10867 10868 10869 10870 10871 10872 10873 10874 10875 10876
	}

	fw_hdr = (struct bnx2x_fw_file_hdr *)bp->firmware->data;

	/* Initialize the pointers to the init arrays */
	/* Blob */
	BNX2X_ALLOC_AND_SET(init_data, request_firmware_exit, be32_to_cpu_n);

	/* Opcodes */
	BNX2X_ALLOC_AND_SET(init_ops, init_ops_alloc_err, bnx2x_prep_ops);

	/* Offsets */
10877 10878
	BNX2X_ALLOC_AND_SET(init_ops_offsets, init_offsets_alloc_err,
			    be16_to_cpu_n);
10879 10880

	/* STORMs firmware */
10881 10882 10883 10884 10885 10886 10887 10888 10889 10890 10891 10892 10893 10894 10895 10896
	INIT_TSEM_INT_TABLE_DATA(bp) = bp->firmware->data +
			be32_to_cpu(fw_hdr->tsem_int_table_data.offset);
	INIT_TSEM_PRAM_DATA(bp)      = bp->firmware->data +
			be32_to_cpu(fw_hdr->tsem_pram_data.offset);
	INIT_USEM_INT_TABLE_DATA(bp) = bp->firmware->data +
			be32_to_cpu(fw_hdr->usem_int_table_data.offset);
	INIT_USEM_PRAM_DATA(bp)      = bp->firmware->data +
			be32_to_cpu(fw_hdr->usem_pram_data.offset);
	INIT_XSEM_INT_TABLE_DATA(bp) = bp->firmware->data +
			be32_to_cpu(fw_hdr->xsem_int_table_data.offset);
	INIT_XSEM_PRAM_DATA(bp)      = bp->firmware->data +
			be32_to_cpu(fw_hdr->xsem_pram_data.offset);
	INIT_CSEM_INT_TABLE_DATA(bp) = bp->firmware->data +
			be32_to_cpu(fw_hdr->csem_int_table_data.offset);
	INIT_CSEM_PRAM_DATA(bp)      = bp->firmware->data +
			be32_to_cpu(fw_hdr->csem_pram_data.offset);
10897 10898
	/* IRO */
	BNX2X_ALLOC_AND_SET(iro_arr, iro_alloc_err, bnx2x_prep_iro);
10899 10900

	return 0;
10901

10902 10903
iro_alloc_err:
	kfree(bp->init_ops_offsets);
10904 10905 10906 10907 10908 10909
init_offsets_alloc_err:
	kfree(bp->init_ops);
init_ops_alloc_err:
	kfree(bp->init_data);
request_firmware_exit:
	release_firmware(bp->firmware);
10910
	bp->firmware = NULL;
10911 10912 10913 10914

	return rc;
}

10915 10916 10917 10918 10919 10920
static void bnx2x_release_firmware(struct bnx2x *bp)
{
	kfree(bp->init_ops_offsets);
	kfree(bp->init_ops);
	kfree(bp->init_data);
	release_firmware(bp->firmware);
10921
	bp->firmware = NULL;
10922 10923 10924 10925 10926 10927 10928 10929 10930 10931 10932 10933 10934 10935 10936 10937 10938 10939 10940 10941 10942 10943 10944 10945 10946 10947 10948 10949 10950 10951 10952 10953
}


static struct bnx2x_func_sp_drv_ops bnx2x_func_sp_drv = {
	.init_hw_cmn_chip = bnx2x_init_hw_common_chip,
	.init_hw_cmn      = bnx2x_init_hw_common,
	.init_hw_port     = bnx2x_init_hw_port,
	.init_hw_func     = bnx2x_init_hw_func,

	.reset_hw_cmn     = bnx2x_reset_common,
	.reset_hw_port    = bnx2x_reset_port,
	.reset_hw_func    = bnx2x_reset_func,

	.gunzip_init      = bnx2x_gunzip_init,
	.gunzip_end       = bnx2x_gunzip_end,

	.init_fw          = bnx2x_init_firmware,
	.release_fw       = bnx2x_release_firmware,
};

void bnx2x__init_func_obj(struct bnx2x *bp)
{
	/* Prepare DMAE related driver resources */
	bnx2x_setup_dmae(bp);

	bnx2x_init_func_obj(bp, &bp->func_obj,
			    bnx2x_sp(bp, func_rdata),
			    bnx2x_sp_mapping(bp, func_rdata),
			    &bnx2x_func_sp_drv);
}

/* must be called after sriov-enable */
10954
static inline int bnx2x_set_qm_cid_count(struct bnx2x *bp)
10955
{
10956
	int cid_count = BNX2X_L2_CID_COUNT(bp);
10957

10958 10959 10960 10961 10962
#ifdef BCM_CNIC
	cid_count += CNIC_CID_MAX;
#endif
	return roundup(cid_count, QM_CID_ROUND);
}
D
Dmitry Kravkov 已提交
10963

10964
/**
10965
 * bnx2x_get_num_none_def_sbs - return the number of none default SBs
10966 10967 10968 10969
 *
 * @dev:	pci device
 *
 */
10970
static inline int bnx2x_get_num_non_def_sbs(struct pci_dev *pdev)
10971 10972 10973 10974 10975
{
	int pos;
	u16 control;

	pos = pci_find_capability(pdev, PCI_CAP_ID_MSIX);
10976 10977 10978 10979 10980

	/*
	 * If MSI-X is not supported - return number of SBs needed to support
	 * one fast path queue: one FP queue + SB for CNIC
	 */
10981
	if (!pos)
10982
		return 1 + CNIC_PRESENT;
10983

10984 10985 10986 10987 10988 10989
	/*
	 * The value in the PCI configuration space is the index of the last
	 * entry, namely one less than the actual size of the table, which is
	 * exactly what we want to return from this function: number of all SBs
	 * without the default SB.
	 */
10990
	pci_read_config_word(pdev, pos  + PCI_MSI_FLAGS, &control);
10991
	return control & PCI_MSIX_FLAGS_QSIZE;
10992 10993
}

E
Eliezer Tamir 已提交
10994 10995 10996 10997 10998
static int __devinit bnx2x_init_one(struct pci_dev *pdev,
				    const struct pci_device_id *ent)
{
	struct net_device *dev = NULL;
	struct bnx2x *bp;
10999
	int pcie_width, pcie_speed;
11000 11001 11002 11003 11004 11005 11006 11007 11008 11009 11010 11011
	int rc, max_non_def_sbs;
	int rx_count, tx_count, rss_count;
	/*
	 * An estimated maximum supported CoS number according to the chip
	 * version.
	 * We will try to roughly estimate the maximum number of CoSes this chip
	 * may support in order to minimize the memory allocated for Tx
	 * netdev_queue's. This number will be accurately calculated during the
	 * initialization of bp->max_cos based on the chip versions AND chip
	 * revision in the bnx2x_init_bp().
	 */
	u8 max_cos_est = 0;
11012

D
Dmitry Kravkov 已提交
11013 11014 11015 11016
	switch (ent->driver_data) {
	case BCM57710:
	case BCM57711:
	case BCM57711E:
11017 11018 11019
		max_cos_est = BNX2X_MULTI_TX_COS_E1X;
		break;

D
Dmitry Kravkov 已提交
11020
	case BCM57712:
11021
	case BCM57712_MF:
11022 11023 11024
		max_cos_est = BNX2X_MULTI_TX_COS_E2_E3A0;
		break;

11025 11026 11027 11028 11029 11030
	case BCM57800:
	case BCM57800_MF:
	case BCM57810:
	case BCM57810_MF:
	case BCM57840:
	case BCM57840_MF:
11031
		max_cos_est = BNX2X_MULTI_TX_COS_E3B0;
D
Dmitry Kravkov 已提交
11032
		break;
E
Eliezer Tamir 已提交
11033

D
Dmitry Kravkov 已提交
11034 11035 11036
	default:
		pr_err("Unknown board_type (%ld), aborting\n",
			   ent->driver_data);
V
Vasiliy Kulikov 已提交
11037
		return -ENODEV;
D
Dmitry Kravkov 已提交
11038 11039
	}

11040 11041 11042 11043 11044 11045 11046 11047 11048 11049 11050 11051 11052 11053 11054 11055 11056 11057 11058 11059 11060 11061
	max_non_def_sbs = bnx2x_get_num_non_def_sbs(pdev);

	/* !!! FIXME !!!
	 * Do not allow the maximum SB count to grow above 16
	 * since Special CIDs starts from 16*BNX2X_MULTI_TX_COS=48.
	 * We will use the FP_SB_MAX_E1x macro for this matter.
	 */
	max_non_def_sbs = min_t(int, FP_SB_MAX_E1x, max_non_def_sbs);

	WARN_ON(!max_non_def_sbs);

	/* Maximum number of RSS queues: one IGU SB goes to CNIC */
	rss_count = max_non_def_sbs - CNIC_PRESENT;

	/* Maximum number of netdev Rx queues: RSS + FCoE L2 */
	rx_count = rss_count + FCOE_PRESENT;

	/*
	 * Maximum number of netdev Tx queues:
	 *      Maximum TSS queues * Maximum supported number of CoS  + FCoE L2
	 */
	tx_count = MAX_TXQS_PER_COS * max_cos_est + FCOE_PRESENT;
D
Dmitry Kravkov 已提交
11062

E
Eliezer Tamir 已提交
11063
	/* dev zeroed in init_etherdev */
11064
	dev = alloc_etherdev_mqs(sizeof(*bp), tx_count, rx_count);
11065
	if (!dev)
E
Eliezer Tamir 已提交
11066 11067 11068 11069
		return -ENOMEM;

	bp = netdev_priv(dev);

M
Merav Sicron 已提交
11070
	BNX2X_DEV_INFO("Allocated netdev with %d tx and %d rx queues\n",
11071
			  tx_count, rx_count);
11072

11073 11074 11075
	bp->igu_sb_cnt = max_non_def_sbs;
	bp->msg_enable = debug;
	pci_set_drvdata(pdev, dev);
11076

11077
	rc = bnx2x_init_dev(pdev, dev, ent->driver_data);
E
Eliezer Tamir 已提交
11078 11079 11080 11081 11082
	if (rc < 0) {
		free_netdev(dev);
		return rc;
	}

M
Merav Sicron 已提交
11083
	BNX2X_DEV_INFO("max_non_def_sbs %d\n", max_non_def_sbs);
11084

11085
	rc = bnx2x_init_bp(bp);
11086 11087 11088
	if (rc)
		goto init_one_exit;

11089 11090 11091 11092 11093 11094 11095 11096 11097 11098 11099 11100 11101 11102
	/*
	 * Map doorbels here as we need the real value of bp->max_cos which
	 * is initialized in bnx2x_init_bp().
	 */
	bp->doorbells = ioremap_nocache(pci_resource_start(pdev, 2),
					min_t(u64, BNX2X_DB_SIZE(bp),
					      pci_resource_len(pdev, 2)));
	if (!bp->doorbells) {
		dev_err(&bp->pdev->dev,
			"Cannot map doorbell space, aborting\n");
		rc = -ENOMEM;
		goto init_one_exit;
	}

11103
	/* calc qm_cid_count */
11104
	bp->qm_cid_count = bnx2x_set_qm_cid_count(bp);
11105

V
Vladislav Zolotarov 已提交
11106
#ifdef BCM_CNIC
11107 11108
	/* disable FCOE L2 queue for E1x */
	if (CHIP_IS_E1x(bp))
V
Vladislav Zolotarov 已提交
11109 11110 11111 11112
		bp->flags |= NO_FCOE_FLAG;

#endif

L
Lucas De Marchi 已提交
11113
	/* Configure interrupt mode: try to enable MSI-X/MSI if
11114 11115 11116 11117 11118 11119 11120
	 * needed, set bp->num_queues appropriately.
	 */
	bnx2x_set_int_mode(bp);

	/* Add all NAPI objects */
	bnx2x_add_all_napi(bp);

11121 11122 11123 11124 11125 11126
	rc = register_netdev(dev);
	if (rc) {
		dev_err(&pdev->dev, "Cannot register net device\n");
		goto init_one_exit;
	}

V
Vladislav Zolotarov 已提交
11127 11128 11129 11130 11131 11132 11133 11134 11135
#ifdef BCM_CNIC
	if (!NO_FCOE(bp)) {
		/* Add storage MAC address */
		rtnl_lock();
		dev_addr_add(bp->dev, bp->fip_mac, NETDEV_HW_ADDR_T_SAN);
		rtnl_unlock();
	}
#endif

11136
	bnx2x_get_pcie_width_speed(bp, &pcie_width, &pcie_speed);
11137

M
Merav Sicron 已提交
11138 11139
	BNX2X_DEV_INFO(
		"%s (%c%d) PCI-E x%d %s found at mem %lx, IRQ %d, node addr %pM\n",
11140 11141 11142 11143 11144 11145 11146
		    board_info[ent->driver_data].name,
		    (CHIP_REV(bp) >> 12) + 'A', (CHIP_METAL(bp) >> 4),
		    pcie_width,
		    ((!CHIP_IS_E2(bp) && pcie_speed == 2) ||
		     (CHIP_IS_E2(bp) && pcie_speed == 1)) ?
		    "5GHz (Gen2)" : "2.5GHz",
		    dev->base_addr, bp->pdev->irq, dev->dev_addr);
E
Eilon Greenstein 已提交
11147

E
Eliezer Tamir 已提交
11148
	return 0;
11149 11150 11151 11152 11153 11154 11155 11156 11157 11158 11159 11160 11161 11162 11163 11164 11165

init_one_exit:
	if (bp->regview)
		iounmap(bp->regview);

	if (bp->doorbells)
		iounmap(bp->doorbells);

	free_netdev(dev);

	if (atomic_read(&pdev->enable_cnt) == 1)
		pci_release_regions(pdev);

	pci_disable_device(pdev);
	pci_set_drvdata(pdev, NULL);

	return rc;
E
Eliezer Tamir 已提交
11166 11167 11168 11169 11170
}

static void __devexit bnx2x_remove_one(struct pci_dev *pdev)
{
	struct net_device *dev = pci_get_drvdata(pdev);
11171 11172 11173
	struct bnx2x *bp;

	if (!dev) {
V
Vladislav Zolotarov 已提交
11174
		dev_err(&pdev->dev, "BAD net device from bnx2x_init_one\n");
11175 11176 11177
		return;
	}
	bp = netdev_priv(dev);
E
Eliezer Tamir 已提交
11178

V
Vladislav Zolotarov 已提交
11179 11180 11181 11182 11183 11184 11185 11186 11187
#ifdef BCM_CNIC
	/* Delete storage MAC address */
	if (!NO_FCOE(bp)) {
		rtnl_lock();
		dev_addr_del(bp->dev, bp->fip_mac, NETDEV_HW_ADDR_T_SAN);
		rtnl_unlock();
	}
#endif

11188 11189 11190 11191 11192
#ifdef BCM_DCBNL
	/* Delete app tlvs from dcbnl */
	bnx2x_dcbnl_update_applist(bp, true);
#endif

E
Eliezer Tamir 已提交
11193 11194
	unregister_netdev(dev);

11195 11196 11197
	/* Delete all NAPI objects */
	bnx2x_del_all_napi(bp);

11198 11199 11200
	/* Power on: we can't let PCI layer write to us while we are in D3 */
	bnx2x_set_power_state(bp, PCI_D0);

11201 11202
	/* Disable MSI/MSI-X */
	bnx2x_disable_msi(bp);
D
Dmitry Kravkov 已提交
11203

11204 11205 11206
	/* Power off */
	bnx2x_set_power_state(bp, PCI_D3hot);

11207
	/* Make sure RESET task is not scheduled before continuing */
11208
	cancel_delayed_work_sync(&bp->sp_rtnl_task);
11209

E
Eliezer Tamir 已提交
11210 11211 11212 11213 11214 11215
	if (bp->regview)
		iounmap(bp->regview);

	if (bp->doorbells)
		iounmap(bp->doorbells);

11216 11217
	bnx2x_release_firmware(bp);

11218 11219
	bnx2x_free_mem_bp(bp);

E
Eliezer Tamir 已提交
11220
	free_netdev(dev);
11221 11222 11223 11224

	if (atomic_read(&pdev->enable_cnt) == 1)
		pci_release_regions(pdev);

E
Eliezer Tamir 已提交
11225 11226 11227 11228
	pci_disable_device(pdev);
	pci_set_drvdata(pdev, NULL);
}

Y
Yitchak Gertner 已提交
11229 11230 11231 11232 11233 11234 11235 11236
static int bnx2x_eeh_nic_unload(struct bnx2x *bp)
{
	int i;

	bp->state = BNX2X_STATE_ERROR;

	bp->rx_mode = BNX2X_RX_MODE_NONE;

11237 11238 11239 11240 11241 11242
#ifdef BCM_CNIC
	bnx2x_cnic_notify(bp, CNIC_CTL_STOP_CMD);
#endif
	/* Stop Tx */
	bnx2x_tx_disable(bp);

Y
Yitchak Gertner 已提交
11243 11244 11245
	bnx2x_netif_stop(bp, 0);

	del_timer_sync(&bp->timer);
11246 11247

	bnx2x_stats_handle(bp, STATS_EVENT_STOP);
Y
Yitchak Gertner 已提交
11248 11249

	/* Release IRQs */
11250
	bnx2x_free_irq(bp);
Y
Yitchak Gertner 已提交
11251 11252 11253

	/* Free SKBs, SGEs, TPA pool and driver internals */
	bnx2x_free_skbs(bp);
11254

V
Vladislav Zolotarov 已提交
11255
	for_each_rx_queue(bp, i)
Y
Yitchak Gertner 已提交
11256
		bnx2x_free_rx_sge_range(bp, bp->fp + i, NUM_RX_SGE);
11257

Y
Yitchak Gertner 已提交
11258 11259 11260 11261
	bnx2x_free_mem(bp);

	bp->state = BNX2X_STATE_CLOSED;

11262 11263
	netif_carrier_off(bp->dev);

Y
Yitchak Gertner 已提交
11264 11265 11266 11267 11268 11269 11270 11271 11272 11273 11274 11275 11276 11277 11278 11279
	return 0;
}

static void bnx2x_eeh_recover(struct bnx2x *bp)
{
	u32 val;

	mutex_init(&bp->port.phy_mutex);


	val = SHMEM_RD(bp, validity_map[BP_PORT(bp)]);
	if ((val & (SHR_MEM_VALIDITY_DEV_INFO | SHR_MEM_VALIDITY_MB))
		!= (SHR_MEM_VALIDITY_DEV_INFO | SHR_MEM_VALIDITY_MB))
		BNX2X_ERR("BAD MCP validity signature\n");
}

W
Wendy Xiong 已提交
11280 11281 11282 11283 11284 11285 11286 11287 11288 11289 11290 11291 11292 11293 11294 11295 11296 11297
/**
 * bnx2x_io_error_detected - called when PCI error is detected
 * @pdev: Pointer to PCI device
 * @state: The current pci connection state
 *
 * This function is called after a PCI bus error affecting
 * this device has been detected.
 */
static pci_ers_result_t bnx2x_io_error_detected(struct pci_dev *pdev,
						pci_channel_state_t state)
{
	struct net_device *dev = pci_get_drvdata(pdev);
	struct bnx2x *bp = netdev_priv(dev);

	rtnl_lock();

	netif_device_detach(dev);

11298 11299 11300 11301 11302
	if (state == pci_channel_io_perm_failure) {
		rtnl_unlock();
		return PCI_ERS_RESULT_DISCONNECT;
	}

W
Wendy Xiong 已提交
11303
	if (netif_running(dev))
Y
Yitchak Gertner 已提交
11304
		bnx2x_eeh_nic_unload(bp);
W
Wendy Xiong 已提交
11305 11306 11307 11308 11309 11310 11311 11312 11313 11314 11315 11316 11317 11318 11319 11320 11321 11322 11323 11324 11325 11326 11327 11328 11329 11330 11331 11332 11333 11334 11335 11336 11337 11338 11339 11340 11341 11342 11343 11344 11345 11346 11347 11348 11349 11350 11351 11352 11353 11354 11355 11356

	pci_disable_device(pdev);

	rtnl_unlock();

	/* Request a slot reset */
	return PCI_ERS_RESULT_NEED_RESET;
}

/**
 * bnx2x_io_slot_reset - called after the PCI bus has been reset
 * @pdev: Pointer to PCI device
 *
 * Restart the card from scratch, as if from a cold-boot.
 */
static pci_ers_result_t bnx2x_io_slot_reset(struct pci_dev *pdev)
{
	struct net_device *dev = pci_get_drvdata(pdev);
	struct bnx2x *bp = netdev_priv(dev);

	rtnl_lock();

	if (pci_enable_device(pdev)) {
		dev_err(&pdev->dev,
			"Cannot re-enable PCI device after reset\n");
		rtnl_unlock();
		return PCI_ERS_RESULT_DISCONNECT;
	}

	pci_set_master(pdev);
	pci_restore_state(pdev);

	if (netif_running(dev))
		bnx2x_set_power_state(bp, PCI_D0);

	rtnl_unlock();

	return PCI_ERS_RESULT_RECOVERED;
}

/**
 * bnx2x_io_resume - called when traffic can start flowing again
 * @pdev: Pointer to PCI device
 *
 * This callback is called when the error recovery driver tells us that
 * its OK to resume normal operation.
 */
static void bnx2x_io_resume(struct pci_dev *pdev)
{
	struct net_device *dev = pci_get_drvdata(pdev);
	struct bnx2x *bp = netdev_priv(dev);

11357
	if (bp->recovery_state != BNX2X_RECOVERY_DONE) {
M
Merav Sicron 已提交
11358
		netdev_err(bp->dev, "Handling parity error recovery. Try again later\n");
11359 11360 11361
		return;
	}

W
Wendy Xiong 已提交
11362 11363
	rtnl_lock();

Y
Yitchak Gertner 已提交
11364 11365
	bnx2x_eeh_recover(bp);

W
Wendy Xiong 已提交
11366
	if (netif_running(dev))
Y
Yitchak Gertner 已提交
11367
		bnx2x_nic_load(bp, LOAD_NORMAL);
W
Wendy Xiong 已提交
11368 11369 11370 11371 11372 11373 11374 11375

	netif_device_attach(dev);

	rtnl_unlock();
}

static struct pci_error_handlers bnx2x_err_handler = {
	.error_detected = bnx2x_io_error_detected,
E
Eilon Greenstein 已提交
11376 11377
	.slot_reset     = bnx2x_io_slot_reset,
	.resume         = bnx2x_io_resume,
W
Wendy Xiong 已提交
11378 11379
};

E
Eliezer Tamir 已提交
11380
static struct pci_driver bnx2x_pci_driver = {
W
Wendy Xiong 已提交
11381 11382 11383 11384 11385 11386 11387
	.name        = DRV_MODULE_NAME,
	.id_table    = bnx2x_pci_tbl,
	.probe       = bnx2x_init_one,
	.remove      = __devexit_p(bnx2x_remove_one),
	.suspend     = bnx2x_suspend,
	.resume      = bnx2x_resume,
	.err_handler = &bnx2x_err_handler,
E
Eliezer Tamir 已提交
11388 11389 11390 11391
};

static int __init bnx2x_init(void)
{
11392 11393
	int ret;

11394
	pr_info("%s", version);
11395

11396 11397
	bnx2x_wq = create_singlethread_workqueue("bnx2x");
	if (bnx2x_wq == NULL) {
11398
		pr_err("Cannot create workqueue\n");
11399 11400 11401
		return -ENOMEM;
	}

11402 11403
	ret = pci_register_driver(&bnx2x_pci_driver);
	if (ret) {
11404
		pr_err("Cannot register driver\n");
11405 11406 11407
		destroy_workqueue(bnx2x_wq);
	}
	return ret;
E
Eliezer Tamir 已提交
11408 11409 11410 11411 11412
}

static void __exit bnx2x_cleanup(void)
{
	pci_unregister_driver(&bnx2x_pci_driver);
11413 11414

	destroy_workqueue(bnx2x_wq);
E
Eliezer Tamir 已提交
11415 11416
}

11417 11418 11419 11420 11421
void bnx2x_notify_link_changed(struct bnx2x *bp)
{
	REG_WR(bp, MISC_REG_AEU_GENERAL_ATTN_12 + BP_FUNC(bp)*sizeof(u32), 1);
}

E
Eliezer Tamir 已提交
11422 11423 11424
module_init(bnx2x_init);
module_exit(bnx2x_cleanup);

11425
#ifdef BCM_CNIC
11426 11427 11428 11429 11430 11431 11432 11433 11434 11435 11436 11437 11438 11439 11440 11441 11442 11443
/**
 * bnx2x_set_iscsi_eth_mac_addr - set iSCSI MAC(s).
 *
 * @bp:		driver handle
 * @set:	set or clear the CAM entry
 *
 * This function will wait until the ramdord completion returns.
 * Return 0 if success, -ENODEV if ramrod doesn't return.
 */
static inline int bnx2x_set_iscsi_eth_mac_addr(struct bnx2x *bp)
{
	unsigned long ramrod_flags = 0;

	__set_bit(RAMROD_COMP_WAIT, &ramrod_flags);
	return bnx2x_set_mac_one(bp, bp->cnic_eth_dev.iscsi_mac,
				 &bp->iscsi_l2_mac_obj, true,
				 BNX2X_ISCSI_ETH_MAC, &ramrod_flags);
}
11444 11445 11446 11447 11448 11449 11450 11451 11452 11453 11454 11455

/* count denotes the number of new completions we have seen */
static void bnx2x_cnic_sp_post(struct bnx2x *bp, int count)
{
	struct eth_spe *spe;

#ifdef BNX2X_STOP_ON_ERROR
	if (unlikely(bp->panic))
		return;
#endif

	spin_lock_bh(&bp->spq_lock);
11456
	BUG_ON(bp->cnic_spq_pending < count);
11457 11458 11459
	bp->cnic_spq_pending -= count;


11460 11461 11462 11463
	for (; bp->cnic_kwq_pending; bp->cnic_kwq_pending--) {
		u16 type =  (le16_to_cpu(bp->cnic_kwq_cons->hdr.type)
				& SPE_HDR_CONN_TYPE) >>
				SPE_HDR_CONN_TYPE_SHIFT;
11464 11465
		u8 cmd = (le32_to_cpu(bp->cnic_kwq_cons->hdr.conn_and_cmd_data)
				>> SPE_HDR_CMD_ID_SHIFT) & 0xff;
11466 11467 11468 11469 11470 11471

		/* Set validation for iSCSI L2 client before sending SETUP
		 *  ramrod
		 */
		if (type == ETH_CONNECTION_TYPE) {
			if (cmd == RAMROD_CMD_ID_ETH_CLIENT_SETUP)
11472 11473 11474
				bnx2x_set_ctx_validation(bp, &bp->context.
					vcxt[BNX2X_ISCSI_ETH_CID].eth,
					BNX2X_ISCSI_ETH_CID);
11475 11476
		}

11477 11478 11479
		/*
		 * There may be not more than 8 L2, not more than 8 L5 SPEs
		 * and in the air. We also check that number of outstanding
11480 11481
		 * COMMON ramrods is not more than the EQ and SPQ can
		 * accommodate.
11482
		 */
11483 11484 11485 11486 11487 11488 11489
		if (type == ETH_CONNECTION_TYPE) {
			if (!atomic_read(&bp->cq_spq_left))
				break;
			else
				atomic_dec(&bp->cq_spq_left);
		} else if (type == NONE_CONNECTION_TYPE) {
			if (!atomic_read(&bp->eq_spq_left))
11490 11491
				break;
			else
11492
				atomic_dec(&bp->eq_spq_left);
V
Vladislav Zolotarov 已提交
11493 11494
		} else if ((type == ISCSI_CONNECTION_TYPE) ||
			   (type == FCOE_CONNECTION_TYPE)) {
11495 11496 11497 11498 11499 11500 11501 11502
			if (bp->cnic_spq_pending >=
			    bp->cnic_eth_dev.max_kwqe_pending)
				break;
			else
				bp->cnic_spq_pending++;
		} else {
			BNX2X_ERR("Unknown SPE type: %d\n", type);
			bnx2x_panic();
11503
			break;
11504
		}
11505 11506 11507 11508

		spe = bnx2x_sp_get_next(bp);
		*spe = *bp->cnic_kwq_cons;

M
Merav Sicron 已提交
11509
		DP(BNX2X_MSG_SP, "pending on SPQ %d, on KWQ %d count %d\n",
11510 11511 11512 11513 11514 11515 11516 11517 11518 11519 11520 11521 11522 11523 11524 11525 11526 11527
		   bp->cnic_spq_pending, bp->cnic_kwq_pending, count);

		if (bp->cnic_kwq_cons == bp->cnic_kwq_last)
			bp->cnic_kwq_cons = bp->cnic_kwq;
		else
			bp->cnic_kwq_cons++;
	}
	bnx2x_sp_prod_update(bp);
	spin_unlock_bh(&bp->spq_lock);
}

static int bnx2x_cnic_sp_queue(struct net_device *dev,
			       struct kwqe_16 *kwqes[], u32 count)
{
	struct bnx2x *bp = netdev_priv(dev);
	int i;

#ifdef BNX2X_STOP_ON_ERROR
M
Merav Sicron 已提交
11528 11529
	if (unlikely(bp->panic)) {
		BNX2X_ERR("Can't post to SP queue while panic\n");
11530
		return -EIO;
M
Merav Sicron 已提交
11531
	}
11532 11533
#endif

A
Ariel Elior 已提交
11534 11535
	if ((bp->recovery_state != BNX2X_RECOVERY_DONE) &&
	    (bp->recovery_state != BNX2X_RECOVERY_NIC_LOADING)) {
M
Merav Sicron 已提交
11536
		BNX2X_ERR("Handling parity error recovery. Try again later\n");
A
Ariel Elior 已提交
11537 11538 11539
		return -EAGAIN;
	}

11540 11541 11542 11543 11544 11545 11546 11547 11548 11549 11550 11551
	spin_lock_bh(&bp->spq_lock);

	for (i = 0; i < count; i++) {
		struct eth_spe *spe = (struct eth_spe *)kwqes[i];

		if (bp->cnic_kwq_pending == MAX_SP_DESC_CNT)
			break;

		*bp->cnic_kwq_prod = *spe;

		bp->cnic_kwq_pending++;

M
Merav Sicron 已提交
11552
		DP(BNX2X_MSG_SP, "L5 SPQE %x %x %x:%x pos %d\n",
11553
		   spe->hdr.conn_and_cmd_data, spe->hdr.type,
11554 11555
		   spe->data.update_data_addr.hi,
		   spe->data.update_data_addr.lo,
11556 11557 11558 11559 11560 11561 11562 11563 11564 11565 11566 11567 11568 11569 11570 11571 11572 11573 11574 11575 11576 11577
		   bp->cnic_kwq_pending);

		if (bp->cnic_kwq_prod == bp->cnic_kwq_last)
			bp->cnic_kwq_prod = bp->cnic_kwq;
		else
			bp->cnic_kwq_prod++;
	}

	spin_unlock_bh(&bp->spq_lock);

	if (bp->cnic_spq_pending < bp->cnic_eth_dev.max_kwqe_pending)
		bnx2x_cnic_sp_post(bp, 0);

	return i;
}

static int bnx2x_cnic_ctl_send(struct bnx2x *bp, struct cnic_ctl_info *ctl)
{
	struct cnic_ops *c_ops;
	int rc = 0;

	mutex_lock(&bp->cnic_mutex);
11578 11579
	c_ops = rcu_dereference_protected(bp->cnic_ops,
					  lockdep_is_held(&bp->cnic_mutex));
11580 11581 11582 11583 11584 11585 11586 11587 11588 11589 11590 11591 11592 11593 11594 11595 11596 11597 11598 11599 11600 11601 11602 11603
	if (c_ops)
		rc = c_ops->cnic_ctl(bp->cnic_data, ctl);
	mutex_unlock(&bp->cnic_mutex);

	return rc;
}

static int bnx2x_cnic_ctl_send_bh(struct bnx2x *bp, struct cnic_ctl_info *ctl)
{
	struct cnic_ops *c_ops;
	int rc = 0;

	rcu_read_lock();
	c_ops = rcu_dereference(bp->cnic_ops);
	if (c_ops)
		rc = c_ops->cnic_ctl(bp->cnic_data, ctl);
	rcu_read_unlock();

	return rc;
}

/*
 * for commands that have no data
 */
D
Dmitry Kravkov 已提交
11604
int bnx2x_cnic_notify(struct bnx2x *bp, int cmd)
11605 11606 11607 11608 11609 11610 11611 11612
{
	struct cnic_ctl_info ctl = {0};

	ctl.cmd = cmd;

	return bnx2x_cnic_ctl_send(bp, &ctl);
}

11613
static void bnx2x_cnic_cfc_comp(struct bnx2x *bp, int cid, u8 err)
11614
{
11615
	struct cnic_ctl_info ctl = {0};
11616 11617 11618 11619

	/* first we tell CNIC and only then we count this as a completion */
	ctl.cmd = CNIC_CTL_COMPLETION_CMD;
	ctl.data.comp.cid = cid;
11620
	ctl.data.comp.error = err;
11621 11622

	bnx2x_cnic_ctl_send_bh(bp, &ctl);
11623
	bnx2x_cnic_sp_post(bp, 0);
11624 11625
}

11626 11627 11628 11629 11630 11631 11632 11633 11634 11635 11636 11637 11638 11639 11640 11641 11642 11643 11644 11645 11646 11647 11648 11649 11650 11651 11652 11653 11654 11655 11656 11657 11658 11659 11660 11661 11662 11663 11664 11665 11666 11667

/* Called with netif_addr_lock_bh() taken.
 * Sets an rx_mode config for an iSCSI ETH client.
 * Doesn't block.
 * Completion should be checked outside.
 */
static void bnx2x_set_iscsi_eth_rx_mode(struct bnx2x *bp, bool start)
{
	unsigned long accept_flags = 0, ramrod_flags = 0;
	u8 cl_id = bnx2x_cnic_eth_cl_id(bp, BNX2X_ISCSI_ETH_CL_ID_IDX);
	int sched_state = BNX2X_FILTER_ISCSI_ETH_STOP_SCHED;

	if (start) {
		/* Start accepting on iSCSI L2 ring. Accept all multicasts
		 * because it's the only way for UIO Queue to accept
		 * multicasts (in non-promiscuous mode only one Queue per
		 * function will receive multicast packets (leading in our
		 * case).
		 */
		__set_bit(BNX2X_ACCEPT_UNICAST, &accept_flags);
		__set_bit(BNX2X_ACCEPT_ALL_MULTICAST, &accept_flags);
		__set_bit(BNX2X_ACCEPT_BROADCAST, &accept_flags);
		__set_bit(BNX2X_ACCEPT_ANY_VLAN, &accept_flags);

		/* Clear STOP_PENDING bit if START is requested */
		clear_bit(BNX2X_FILTER_ISCSI_ETH_STOP_SCHED, &bp->sp_state);

		sched_state = BNX2X_FILTER_ISCSI_ETH_START_SCHED;
	} else
		/* Clear START_PENDING bit if STOP is requested */
		clear_bit(BNX2X_FILTER_ISCSI_ETH_START_SCHED, &bp->sp_state);

	if (test_bit(BNX2X_FILTER_RX_MODE_PENDING, &bp->sp_state))
		set_bit(sched_state, &bp->sp_state);
	else {
		__set_bit(RAMROD_RX, &ramrod_flags);
		bnx2x_set_q_rx_mode(bp, cl_id, 0, accept_flags, 0,
				    ramrod_flags);
	}
}


11668 11669 11670 11671 11672 11673 11674 11675 11676 11677 11678 11679 11680 11681
static int bnx2x_drv_ctl(struct net_device *dev, struct drv_ctl_info *ctl)
{
	struct bnx2x *bp = netdev_priv(dev);
	int rc = 0;

	switch (ctl->cmd) {
	case DRV_CTL_CTXTBL_WR_CMD: {
		u32 index = ctl->data.io.offset;
		dma_addr_t addr = ctl->data.io.dma_addr;

		bnx2x_ilt_wr(bp, index, addr);
		break;
	}

11682 11683
	case DRV_CTL_RET_L5_SPQ_CREDIT_CMD: {
		int count = ctl->data.credit.credit_count;
11684 11685 11686 11687 11688 11689 11690

		bnx2x_cnic_sp_post(bp, count);
		break;
	}

	/* rtnl_lock is held.  */
	case DRV_CTL_START_L2_CMD: {
11691 11692 11693 11694 11695 11696 11697 11698 11699 11700 11701 11702
		struct cnic_eth_dev *cp = &bp->cnic_eth_dev;
		unsigned long sp_bits = 0;

		/* Configure the iSCSI classification object */
		bnx2x_init_mac_obj(bp, &bp->iscsi_l2_mac_obj,
				   cp->iscsi_l2_client_id,
				   cp->iscsi_l2_cid, BP_FUNC(bp),
				   bnx2x_sp(bp, mac_rdata),
				   bnx2x_sp_mapping(bp, mac_rdata),
				   BNX2X_FILTER_MAC_PENDING,
				   &bp->sp_state, BNX2X_OBJ_TYPE_RX,
				   &bp->macs_pool);
V
Vladislav Zolotarov 已提交
11703

11704
		/* Set iSCSI MAC address */
11705 11706 11707
		rc = bnx2x_set_iscsi_eth_mac_addr(bp);
		if (rc)
			break;
11708 11709 11710 11711

		mmiowb();
		barrier();

11712 11713 11714 11715 11716 11717 11718 11719 11720 11721 11722 11723
		/* Start accepting on iSCSI L2 ring */

		netif_addr_lock_bh(dev);
		bnx2x_set_iscsi_eth_rx_mode(bp, true);
		netif_addr_unlock_bh(dev);

		/* bits to wait on */
		__set_bit(BNX2X_FILTER_RX_MODE_PENDING, &sp_bits);
		__set_bit(BNX2X_FILTER_ISCSI_ETH_START_SCHED, &sp_bits);

		if (!bnx2x_wait_sp_comp(bp, sp_bits))
			BNX2X_ERR("rx_mode completion timed out!\n");
11724

11725 11726 11727 11728 11729
		break;
	}

	/* rtnl_lock is held.  */
	case DRV_CTL_STOP_L2_CMD: {
11730
		unsigned long sp_bits = 0;
11731

11732
		/* Stop accepting on iSCSI L2 ring */
11733 11734 11735 11736 11737 11738 11739 11740 11741 11742
		netif_addr_lock_bh(dev);
		bnx2x_set_iscsi_eth_rx_mode(bp, false);
		netif_addr_unlock_bh(dev);

		/* bits to wait on */
		__set_bit(BNX2X_FILTER_RX_MODE_PENDING, &sp_bits);
		__set_bit(BNX2X_FILTER_ISCSI_ETH_STOP_SCHED, &sp_bits);

		if (!bnx2x_wait_sp_comp(bp, sp_bits))
			BNX2X_ERR("rx_mode completion timed out!\n");
11743 11744 11745 11746 11747

		mmiowb();
		barrier();

		/* Unset iSCSI L2 MAC */
11748 11749
		rc = bnx2x_del_all_macs(bp, &bp->iscsi_l2_mac_obj,
					BNX2X_ISCSI_ETH_MAC, true);
11750 11751
		break;
	}
11752 11753 11754 11755
	case DRV_CTL_RET_L2_SPQ_CREDIT_CMD: {
		int count = ctl->data.credit.credit_count;

		smp_mb__before_atomic_inc();
11756
		atomic_add(count, &bp->cq_spq_left);
11757 11758 11759
		smp_mb__after_atomic_inc();
		break;
	}
11760 11761 11762 11763 11764 11765 11766 11767 11768 11769 11770 11771 11772 11773 11774 11775 11776 11777 11778 11779 11780 11781 11782 11783 11784 11785 11786 11787 11788 11789 11790 11791
	case DRV_CTL_ULP_REGISTER_CMD: {
		int ulp_type = ctl->data.ulp_type;

		if (CHIP_IS_E3(bp)) {
			int idx = BP_FW_MB_IDX(bp);
			u32 cap;

			cap = SHMEM2_RD(bp, drv_capabilities_flag[idx]);
			if (ulp_type == CNIC_ULP_ISCSI)
				cap |= DRV_FLAGS_CAPABILITIES_LOADED_ISCSI;
			else if (ulp_type == CNIC_ULP_FCOE)
				cap |= DRV_FLAGS_CAPABILITIES_LOADED_FCOE;
			SHMEM2_WR(bp, drv_capabilities_flag[idx], cap);
		}
		break;
	}
	case DRV_CTL_ULP_UNREGISTER_CMD: {
		int ulp_type = ctl->data.ulp_type;

		if (CHIP_IS_E3(bp)) {
			int idx = BP_FW_MB_IDX(bp);
			u32 cap;

			cap = SHMEM2_RD(bp, drv_capabilities_flag[idx]);
			if (ulp_type == CNIC_ULP_ISCSI)
				cap &= ~DRV_FLAGS_CAPABILITIES_LOADED_ISCSI;
			else if (ulp_type == CNIC_ULP_FCOE)
				cap &= ~DRV_FLAGS_CAPABILITIES_LOADED_FCOE;
			SHMEM2_WR(bp, drv_capabilities_flag[idx], cap);
		}
		break;
	}
11792 11793 11794 11795 11796 11797 11798 11799 11800

	default:
		BNX2X_ERR("unknown command %x\n", ctl->cmd);
		rc = -EINVAL;
	}

	return rc;
}

D
Dmitry Kravkov 已提交
11801
void bnx2x_setup_cnic_irq_info(struct bnx2x *bp)
11802 11803 11804 11805 11806 11807 11808 11809 11810 11811 11812
{
	struct cnic_eth_dev *cp = &bp->cnic_eth_dev;

	if (bp->flags & USING_MSIX_FLAG) {
		cp->drv_state |= CNIC_DRV_STATE_USING_MSIX;
		cp->irq_arr[0].irq_flags |= CNIC_IRQ_FL_MSIX;
		cp->irq_arr[0].vector = bp->msix_table[1].vector;
	} else {
		cp->drv_state &= ~CNIC_DRV_STATE_USING_MSIX;
		cp->irq_arr[0].irq_flags &= ~CNIC_IRQ_FL_MSIX;
	}
11813
	if (!CHIP_IS_E1x(bp))
D
Dmitry Kravkov 已提交
11814 11815 11816 11817
		cp->irq_arr[0].status_blk = (void *)bp->cnic_sb.e2_sb;
	else
		cp->irq_arr[0].status_blk = (void *)bp->cnic_sb.e1x_sb;

11818 11819
	cp->irq_arr[0].status_blk_num =  bnx2x_cnic_fw_sb_id(bp);
	cp->irq_arr[0].status_blk_num2 = bnx2x_cnic_igu_sb_id(bp);
11820 11821
	cp->irq_arr[1].status_blk = bp->def_status_blk;
	cp->irq_arr[1].status_blk_num = DEF_SB_ID;
11822
	cp->irq_arr[1].status_blk_num2 = DEF_SB_IGU_ID;
11823 11824 11825 11826 11827 11828 11829 11830 11831 11832

	cp->num_irq = 2;
}

static int bnx2x_register_cnic(struct net_device *dev, struct cnic_ops *ops,
			       void *data)
{
	struct bnx2x *bp = netdev_priv(dev);
	struct cnic_eth_dev *cp = &bp->cnic_eth_dev;

M
Merav Sicron 已提交
11833 11834
	if (ops == NULL) {
		BNX2X_ERR("NULL ops received\n");
11835
		return -EINVAL;
M
Merav Sicron 已提交
11836
	}
11837 11838 11839 11840 11841 11842 11843 11844 11845 11846 11847 11848 11849 11850 11851

	bp->cnic_kwq = kzalloc(PAGE_SIZE, GFP_KERNEL);
	if (!bp->cnic_kwq)
		return -ENOMEM;

	bp->cnic_kwq_cons = bp->cnic_kwq;
	bp->cnic_kwq_prod = bp->cnic_kwq;
	bp->cnic_kwq_last = bp->cnic_kwq + MAX_SP_DESC_CNT;

	bp->cnic_spq_pending = 0;
	bp->cnic_kwq_pending = 0;

	bp->cnic_data = data;

	cp->num_irq = 0;
11852
	cp->drv_state |= CNIC_DRV_STATE_REGD;
11853
	cp->iro_arr = bp->iro_arr;
11854 11855

	bnx2x_setup_cnic_irq_info(bp);
11856

11857 11858 11859 11860 11861 11862 11863 11864 11865 11866 11867 11868
	rcu_assign_pointer(bp->cnic_ops, ops);

	return 0;
}

static int bnx2x_unregister_cnic(struct net_device *dev)
{
	struct bnx2x *bp = netdev_priv(dev);
	struct cnic_eth_dev *cp = &bp->cnic_eth_dev;

	mutex_lock(&bp->cnic_mutex);
	cp->drv_state = 0;
11869
	RCU_INIT_POINTER(bp->cnic_ops, NULL);
11870 11871 11872 11873 11874 11875 11876 11877 11878 11879 11880 11881 11882
	mutex_unlock(&bp->cnic_mutex);
	synchronize_rcu();
	kfree(bp->cnic_kwq);
	bp->cnic_kwq = NULL;

	return 0;
}

struct cnic_eth_dev *bnx2x_cnic_probe(struct net_device *dev)
{
	struct bnx2x *bp = netdev_priv(dev);
	struct cnic_eth_dev *cp = &bp->cnic_eth_dev;

11883 11884 11885 11886 11887 11888 11889
	/* If both iSCSI and FCoE are disabled - return NULL in
	 * order to indicate CNIC that it should not try to work
	 * with this device.
	 */
	if (NO_ISCSI(bp) && NO_FCOE(bp))
		return NULL;

11890 11891 11892 11893 11894 11895
	cp->drv_owner = THIS_MODULE;
	cp->chip_id = CHIP_ID(bp);
	cp->pdev = bp->pdev;
	cp->io_base = bp->regview;
	cp->io_base2 = bp->doorbells;
	cp->max_kwqe_pending = 8;
11896
	cp->ctx_blk_size = CDU_ILT_PAGE_SZ;
11897 11898
	cp->ctx_tbl_offset = FUNC_ILT_BASE(BP_FUNC(bp)) +
			     bnx2x_cid_ilt_lines(bp);
11899
	cp->ctx_tbl_len = CNIC_ILT_LINES;
11900
	cp->starting_cid = bnx2x_cid_ilt_lines(bp) * ILT_PAGE_CIDS;
11901 11902 11903 11904
	cp->drv_submit_kwqes_16 = bnx2x_cnic_sp_queue;
	cp->drv_ctl = bnx2x_drv_ctl;
	cp->drv_register_cnic = bnx2x_register_cnic;
	cp->drv_unregister_cnic = bnx2x_unregister_cnic;
V
Vladislav Zolotarov 已提交
11905
	cp->fcoe_init_cid = BNX2X_FCOE_ETH_CID;
11906 11907
	cp->iscsi_l2_client_id =
		bnx2x_cnic_eth_cl_id(bp, BNX2X_ISCSI_ETH_CL_ID_IDX);
11908 11909
	cp->iscsi_l2_cid = BNX2X_ISCSI_ETH_CID;

11910 11911 11912 11913 11914 11915 11916 11917 11918
	if (NO_ISCSI_OOO(bp))
		cp->drv_state |= CNIC_DRV_STATE_NO_ISCSI_OOO;

	if (NO_ISCSI(bp))
		cp->drv_state |= CNIC_DRV_STATE_NO_ISCSI;

	if (NO_FCOE(bp))
		cp->drv_state |= CNIC_DRV_STATE_NO_FCOE;

M
Merav Sicron 已提交
11919 11920
	BNX2X_DEV_INFO(
		"page_size %d, tbl_offset %d, tbl_lines %d, starting cid %d\n",
11921 11922 11923 11924
	   cp->ctx_blk_size,
	   cp->ctx_tbl_offset,
	   cp->ctx_tbl_len,
	   cp->starting_cid);
11925 11926 11927 11928 11929
	return cp;
}
EXPORT_SYMBOL(bnx2x_cnic_probe);

#endif /* BCM_CNIC */
11930