xref: /openbmc/linux/drivers/net/ethernet/ibm/ehea/ehea_main.c (revision c900529f3d9161bfde5cca0754f83b4d3c3e0220)
1de6cc651SThomas Gleixner // SPDX-License-Identifier: GPL-2.0-or-later
29aa32835SJeff Kirsher /*
33396c782SPaul Gortmaker  *  linux/drivers/net/ethernet/ibm/ehea/ehea_main.c
49aa32835SJeff Kirsher  *
59aa32835SJeff Kirsher  *  eHEA ethernet device driver for IBM eServer System p
69aa32835SJeff Kirsher  *
79aa32835SJeff Kirsher  *  (C) Copyright IBM Corp. 2006
89aa32835SJeff Kirsher  *
99aa32835SJeff Kirsher  *  Authors:
109aa32835SJeff Kirsher  *	 Christoph Raisch <raisch@de.ibm.com>
119aa32835SJeff Kirsher  *	 Jan-Bernd Themann <themann@de.ibm.com>
129aa32835SJeff Kirsher  *	 Thomas Klein <tklein@de.ibm.com>
139aa32835SJeff Kirsher  */
149aa32835SJeff Kirsher 
159aa32835SJeff Kirsher #define pr_fmt(fmt) KBUILD_MODNAME ": " fmt
169aa32835SJeff Kirsher 
1709b38aa1SHimangi Saraogi #include <linux/device.h>
189aa32835SJeff Kirsher #include <linux/in.h>
199aa32835SJeff Kirsher #include <linux/ip.h>
209aa32835SJeff Kirsher #include <linux/tcp.h>
219aa32835SJeff Kirsher #include <linux/udp.h>
229aa32835SJeff Kirsher #include <linux/if.h>
239aa32835SJeff Kirsher #include <linux/list.h>
249aa32835SJeff Kirsher #include <linux/slab.h>
259aa32835SJeff Kirsher #include <linux/if_ether.h>
269aa32835SJeff Kirsher #include <linux/notifier.h>
279aa32835SJeff Kirsher #include <linux/reboot.h>
289aa32835SJeff Kirsher #include <linux/memory.h>
299aa32835SJeff Kirsher #include <asm/kexec.h>
309aa32835SJeff Kirsher #include <linux/mutex.h>
319aa32835SJeff Kirsher #include <linux/prefetch.h>
326bff3ffcSChristophe Leroy #include <linux/of.h>
336bff3ffcSChristophe Leroy #include <linux/of_device.h>
34*3d40aed8SRob Herring #include <linux/platform_device.h>
359aa32835SJeff Kirsher 
369aa32835SJeff Kirsher #include <net/ip.h>
379aa32835SJeff Kirsher 
389aa32835SJeff Kirsher #include "ehea.h"
399aa32835SJeff Kirsher #include "ehea_qmr.h"
409aa32835SJeff Kirsher #include "ehea_phyp.h"
419aa32835SJeff Kirsher 
429aa32835SJeff Kirsher 
439aa32835SJeff Kirsher MODULE_LICENSE("GPL");
449aa32835SJeff Kirsher MODULE_AUTHOR("Christoph Raisch <raisch@de.ibm.com>");
459aa32835SJeff Kirsher MODULE_DESCRIPTION("IBM eServer HEA Driver");
469aa32835SJeff Kirsher MODULE_VERSION(DRV_VERSION);
479aa32835SJeff Kirsher 
489aa32835SJeff Kirsher 
499aa32835SJeff Kirsher static int msg_level = -1;
509aa32835SJeff Kirsher static int rq1_entries = EHEA_DEF_ENTRIES_RQ1;
519aa32835SJeff Kirsher static int rq2_entries = EHEA_DEF_ENTRIES_RQ2;
529aa32835SJeff Kirsher static int rq3_entries = EHEA_DEF_ENTRIES_RQ3;
539aa32835SJeff Kirsher static int sq_entries = EHEA_DEF_ENTRIES_SQ;
54b9564468SAnton Blanchard static int use_mcs = 1;
559aa32835SJeff Kirsher static int prop_carrier_state;
569aa32835SJeff Kirsher 
579aa32835SJeff Kirsher module_param(msg_level, int, 0);
589aa32835SJeff Kirsher module_param(rq1_entries, int, 0);
599aa32835SJeff Kirsher module_param(rq2_entries, int, 0);
609aa32835SJeff Kirsher module_param(rq3_entries, int, 0);
619aa32835SJeff Kirsher module_param(sq_entries, int, 0);
629aa32835SJeff Kirsher module_param(prop_carrier_state, int, 0);
639aa32835SJeff Kirsher module_param(use_mcs, int, 0);
649aa32835SJeff Kirsher 
659aa32835SJeff Kirsher MODULE_PARM_DESC(msg_level, "msg_level");
669aa32835SJeff Kirsher MODULE_PARM_DESC(prop_carrier_state, "Propagate carrier state of physical "
679aa32835SJeff Kirsher 		 "port to stack. 1:yes, 0:no.  Default = 0 ");
689aa32835SJeff Kirsher MODULE_PARM_DESC(rq3_entries, "Number of entries for Receive Queue 3 "
6931d17536SDave Young 		 "[2^x - 1], x = [7..14]. Default = "
709aa32835SJeff Kirsher 		 __MODULE_STRING(EHEA_DEF_ENTRIES_RQ3) ")");
719aa32835SJeff Kirsher MODULE_PARM_DESC(rq2_entries, "Number of entries for Receive Queue 2 "
7231d17536SDave Young 		 "[2^x - 1], x = [7..14]. Default = "
739aa32835SJeff Kirsher 		 __MODULE_STRING(EHEA_DEF_ENTRIES_RQ2) ")");
749aa32835SJeff Kirsher MODULE_PARM_DESC(rq1_entries, "Number of entries for Receive Queue 1 "
7531d17536SDave Young 		 "[2^x - 1], x = [7..14]. Default = "
769aa32835SJeff Kirsher 		 __MODULE_STRING(EHEA_DEF_ENTRIES_RQ1) ")");
779aa32835SJeff Kirsher MODULE_PARM_DESC(sq_entries, " Number of entries for the Send Queue  "
7831d17536SDave Young 		 "[2^x - 1], x = [7..14]. Default = "
799aa32835SJeff Kirsher 		 __MODULE_STRING(EHEA_DEF_ENTRIES_SQ) ")");
80b9564468SAnton Blanchard MODULE_PARM_DESC(use_mcs, " Multiple receive queues, 1: enable, 0: disable, "
81b9564468SAnton Blanchard 		 "Default = 1");
829aa32835SJeff Kirsher 
839aa32835SJeff Kirsher static int port_name_cnt;
849aa32835SJeff Kirsher static LIST_HEAD(adapter_list);
859aa32835SJeff Kirsher static unsigned long ehea_driver_flags;
869aa32835SJeff Kirsher static DEFINE_MUTEX(dlpar_mem_lock);
871886e5d2SThadeu Lima de Souza Cascardo static struct ehea_fw_handle_array ehea_fw_handles;
881886e5d2SThadeu Lima de Souza Cascardo static struct ehea_bcmc_reg_array ehea_bcmc_regs;
899aa32835SJeff Kirsher 
909aa32835SJeff Kirsher 
91c45640e4SRob Herring static int ehea_probe_adapter(struct platform_device *dev);
929aa32835SJeff Kirsher 
930297be07SBill Pemberton static int ehea_remove(struct platform_device *dev);
949aa32835SJeff Kirsher 
95abae1e07SFabian Frederick static const struct of_device_id ehea_module_device_table[] = {
967a3a6212SOlaf Hering 	{
977a3a6212SOlaf Hering 		.name = "lhea",
987a3a6212SOlaf Hering 		.compatible = "IBM,lhea",
997a3a6212SOlaf Hering 	},
1007a3a6212SOlaf Hering 	{
1017a3a6212SOlaf Hering 		.type = "network",
1027a3a6212SOlaf Hering 		.compatible = "IBM,lhea-ethernet",
1037a3a6212SOlaf Hering 	},
1047a3a6212SOlaf Hering 	{},
1057a3a6212SOlaf Hering };
1067a3a6212SOlaf Hering MODULE_DEVICE_TABLE(of, ehea_module_device_table);
1077a3a6212SOlaf Hering 
108abae1e07SFabian Frederick static const struct of_device_id ehea_device_table[] = {
1099aa32835SJeff Kirsher 	{
1109aa32835SJeff Kirsher 		.name = "lhea",
1119aa32835SJeff Kirsher 		.compatible = "IBM,lhea",
1129aa32835SJeff Kirsher 	},
1139aa32835SJeff Kirsher 	{},
1149aa32835SJeff Kirsher };
11595291cedSQiheng Lin MODULE_DEVICE_TABLE(of, ehea_device_table);
1169aa32835SJeff Kirsher 
117c45640e4SRob Herring static struct platform_driver ehea_driver = {
1189aa32835SJeff Kirsher 	.driver = {
1199aa32835SJeff Kirsher 		.name = "ehea",
1209aa32835SJeff Kirsher 		.owner = THIS_MODULE,
1219aa32835SJeff Kirsher 		.of_match_table = ehea_device_table,
1229aa32835SJeff Kirsher 	},
1239aa32835SJeff Kirsher 	.probe = ehea_probe_adapter,
1249aa32835SJeff Kirsher 	.remove = ehea_remove,
1259aa32835SJeff Kirsher };
1269aa32835SJeff Kirsher 
ehea_dump(void * adr,int len,char * msg)1279aa32835SJeff Kirsher void ehea_dump(void *adr, int len, char *msg)
1289aa32835SJeff Kirsher {
1299aa32835SJeff Kirsher 	int x;
1309aa32835SJeff Kirsher 	unsigned char *deb = adr;
1319aa32835SJeff Kirsher 	for (x = 0; x < len; x += 16) {
1329aa32835SJeff Kirsher 		pr_info("%s adr=%p ofs=%04x %016llx %016llx\n",
1339aa32835SJeff Kirsher 			msg, deb, x, *((u64 *)&deb[0]), *((u64 *)&deb[8]));
1349aa32835SJeff Kirsher 		deb += 16;
1359aa32835SJeff Kirsher 	}
1369aa32835SJeff Kirsher }
1379aa32835SJeff Kirsher 
ehea_schedule_port_reset(struct ehea_port * port)1381886e5d2SThadeu Lima de Souza Cascardo static void ehea_schedule_port_reset(struct ehea_port *port)
1399aa32835SJeff Kirsher {
1409aa32835SJeff Kirsher 	if (!test_bit(__EHEA_DISABLE_PORT_RESET, &port->flags))
1419aa32835SJeff Kirsher 		schedule_work(&port->reset_task);
1429aa32835SJeff Kirsher }
1439aa32835SJeff Kirsher 
ehea_update_firmware_handles(void)1449aa32835SJeff Kirsher static void ehea_update_firmware_handles(void)
1459aa32835SJeff Kirsher {
1469aa32835SJeff Kirsher 	struct ehea_fw_handle_entry *arr = NULL;
1479aa32835SJeff Kirsher 	struct ehea_adapter *adapter;
1489aa32835SJeff Kirsher 	int num_adapters = 0;
1499aa32835SJeff Kirsher 	int num_ports = 0;
1509aa32835SJeff Kirsher 	int num_portres = 0;
1519aa32835SJeff Kirsher 	int i = 0;
1529aa32835SJeff Kirsher 	int num_fw_handles, k, l;
1539aa32835SJeff Kirsher 
1549aa32835SJeff Kirsher 	/* Determine number of handles */
1559aa32835SJeff Kirsher 	mutex_lock(&ehea_fw_handles.lock);
1569aa32835SJeff Kirsher 
1579aa32835SJeff Kirsher 	list_for_each_entry(adapter, &adapter_list, list) {
1589aa32835SJeff Kirsher 		num_adapters++;
1599aa32835SJeff Kirsher 
1609aa32835SJeff Kirsher 		for (k = 0; k < EHEA_MAX_PORTS; k++) {
1619aa32835SJeff Kirsher 			struct ehea_port *port = adapter->port[k];
1629aa32835SJeff Kirsher 
1639aa32835SJeff Kirsher 			if (!port || (port->state != EHEA_PORT_UP))
1649aa32835SJeff Kirsher 				continue;
1659aa32835SJeff Kirsher 
1669aa32835SJeff Kirsher 			num_ports++;
167723f28e4SAnton Blanchard 			num_portres += port->num_def_qps;
1689aa32835SJeff Kirsher 		}
1699aa32835SJeff Kirsher 	}
1709aa32835SJeff Kirsher 
1719aa32835SJeff Kirsher 	num_fw_handles = num_adapters * EHEA_NUM_ADAPTER_FW_HANDLES +
1729aa32835SJeff Kirsher 			 num_ports * EHEA_NUM_PORT_FW_HANDLES +
1739aa32835SJeff Kirsher 			 num_portres * EHEA_NUM_PORTRES_FW_HANDLES;
1749aa32835SJeff Kirsher 
1759aa32835SJeff Kirsher 	if (num_fw_handles) {
1769aa32835SJeff Kirsher 		arr = kcalloc(num_fw_handles, sizeof(*arr), GFP_KERNEL);
1779aa32835SJeff Kirsher 		if (!arr)
1789aa32835SJeff Kirsher 			goto out;  /* Keep the existing array */
1799aa32835SJeff Kirsher 	} else
1809aa32835SJeff Kirsher 		goto out_update;
1819aa32835SJeff Kirsher 
1829aa32835SJeff Kirsher 	list_for_each_entry(adapter, &adapter_list, list) {
1839aa32835SJeff Kirsher 		if (num_adapters == 0)
1849aa32835SJeff Kirsher 			break;
1859aa32835SJeff Kirsher 
1869aa32835SJeff Kirsher 		for (k = 0; k < EHEA_MAX_PORTS; k++) {
1879aa32835SJeff Kirsher 			struct ehea_port *port = adapter->port[k];
1889aa32835SJeff Kirsher 
1899aa32835SJeff Kirsher 			if (!port || (port->state != EHEA_PORT_UP) ||
1909aa32835SJeff Kirsher 			    (num_ports == 0))
1919aa32835SJeff Kirsher 				continue;
1929aa32835SJeff Kirsher 
193723f28e4SAnton Blanchard 			for (l = 0; l < port->num_def_qps; l++) {
1949aa32835SJeff Kirsher 				struct ehea_port_res *pr = &port->port_res[l];
1959aa32835SJeff Kirsher 
1969aa32835SJeff Kirsher 				arr[i].adh = adapter->handle;
1979aa32835SJeff Kirsher 				arr[i++].fwh = pr->qp->fw_handle;
1989aa32835SJeff Kirsher 				arr[i].adh = adapter->handle;
1999aa32835SJeff Kirsher 				arr[i++].fwh = pr->send_cq->fw_handle;
2009aa32835SJeff Kirsher 				arr[i].adh = adapter->handle;
2019aa32835SJeff Kirsher 				arr[i++].fwh = pr->recv_cq->fw_handle;
2029aa32835SJeff Kirsher 				arr[i].adh = adapter->handle;
2039aa32835SJeff Kirsher 				arr[i++].fwh = pr->eq->fw_handle;
2049aa32835SJeff Kirsher 				arr[i].adh = adapter->handle;
2059aa32835SJeff Kirsher 				arr[i++].fwh = pr->send_mr.handle;
2069aa32835SJeff Kirsher 				arr[i].adh = adapter->handle;
2079aa32835SJeff Kirsher 				arr[i++].fwh = pr->recv_mr.handle;
2089aa32835SJeff Kirsher 			}
2099aa32835SJeff Kirsher 			arr[i].adh = adapter->handle;
2109aa32835SJeff Kirsher 			arr[i++].fwh = port->qp_eq->fw_handle;
2119aa32835SJeff Kirsher 			num_ports--;
2129aa32835SJeff Kirsher 		}
2139aa32835SJeff Kirsher 
2149aa32835SJeff Kirsher 		arr[i].adh = adapter->handle;
2159aa32835SJeff Kirsher 		arr[i++].fwh = adapter->neq->fw_handle;
2169aa32835SJeff Kirsher 
2179aa32835SJeff Kirsher 		if (adapter->mr.handle) {
2189aa32835SJeff Kirsher 			arr[i].adh = adapter->handle;
2199aa32835SJeff Kirsher 			arr[i++].fwh = adapter->mr.handle;
2209aa32835SJeff Kirsher 		}
2219aa32835SJeff Kirsher 		num_adapters--;
2229aa32835SJeff Kirsher 	}
2239aa32835SJeff Kirsher 
2249aa32835SJeff Kirsher out_update:
2259aa32835SJeff Kirsher 	kfree(ehea_fw_handles.arr);
2269aa32835SJeff Kirsher 	ehea_fw_handles.arr = arr;
2279aa32835SJeff Kirsher 	ehea_fw_handles.num_entries = i;
2289aa32835SJeff Kirsher out:
2299aa32835SJeff Kirsher 	mutex_unlock(&ehea_fw_handles.lock);
2309aa32835SJeff Kirsher }
2319aa32835SJeff Kirsher 
ehea_update_bcmc_registrations(void)2329aa32835SJeff Kirsher static void ehea_update_bcmc_registrations(void)
2339aa32835SJeff Kirsher {
2349aa32835SJeff Kirsher 	unsigned long flags;
2359aa32835SJeff Kirsher 	struct ehea_bcmc_reg_entry *arr = NULL;
2369aa32835SJeff Kirsher 	struct ehea_adapter *adapter;
2379aa32835SJeff Kirsher 	struct ehea_mc_list *mc_entry;
2389aa32835SJeff Kirsher 	int num_registrations = 0;
2399aa32835SJeff Kirsher 	int i = 0;
2409aa32835SJeff Kirsher 	int k;
2419aa32835SJeff Kirsher 
2429aa32835SJeff Kirsher 	spin_lock_irqsave(&ehea_bcmc_regs.lock, flags);
2439aa32835SJeff Kirsher 
2449aa32835SJeff Kirsher 	/* Determine number of registrations */
2459aa32835SJeff Kirsher 	list_for_each_entry(adapter, &adapter_list, list)
2469aa32835SJeff Kirsher 		for (k = 0; k < EHEA_MAX_PORTS; k++) {
2479aa32835SJeff Kirsher 			struct ehea_port *port = adapter->port[k];
2489aa32835SJeff Kirsher 
2499aa32835SJeff Kirsher 			if (!port || (port->state != EHEA_PORT_UP))
2509aa32835SJeff Kirsher 				continue;
2519aa32835SJeff Kirsher 
2529aa32835SJeff Kirsher 			num_registrations += 2;	/* Broadcast registrations */
2539aa32835SJeff Kirsher 
2549aa32835SJeff Kirsher 			list_for_each_entry(mc_entry, &port->mc_list->list,list)
2559aa32835SJeff Kirsher 				num_registrations += 2;
2569aa32835SJeff Kirsher 		}
2579aa32835SJeff Kirsher 
2589aa32835SJeff Kirsher 	if (num_registrations) {
2599aa32835SJeff Kirsher 		arr = kcalloc(num_registrations, sizeof(*arr), GFP_ATOMIC);
2609aa32835SJeff Kirsher 		if (!arr)
2619aa32835SJeff Kirsher 			goto out;  /* Keep the existing array */
2629aa32835SJeff Kirsher 	} else
2639aa32835SJeff Kirsher 		goto out_update;
2649aa32835SJeff Kirsher 
2659aa32835SJeff Kirsher 	list_for_each_entry(adapter, &adapter_list, list) {
2669aa32835SJeff Kirsher 		for (k = 0; k < EHEA_MAX_PORTS; k++) {
2679aa32835SJeff Kirsher 			struct ehea_port *port = adapter->port[k];
2689aa32835SJeff Kirsher 
2699aa32835SJeff Kirsher 			if (!port || (port->state != EHEA_PORT_UP))
2709aa32835SJeff Kirsher 				continue;
2719aa32835SJeff Kirsher 
2729aa32835SJeff Kirsher 			if (num_registrations == 0)
2739aa32835SJeff Kirsher 				goto out_update;
2749aa32835SJeff Kirsher 
2759aa32835SJeff Kirsher 			arr[i].adh = adapter->handle;
2769aa32835SJeff Kirsher 			arr[i].port_id = port->logical_port_id;
2779aa32835SJeff Kirsher 			arr[i].reg_type = EHEA_BCMC_BROADCAST |
2789aa32835SJeff Kirsher 					  EHEA_BCMC_UNTAGGED;
2799aa32835SJeff Kirsher 			arr[i++].macaddr = port->mac_addr;
2809aa32835SJeff Kirsher 
2819aa32835SJeff Kirsher 			arr[i].adh = adapter->handle;
2829aa32835SJeff Kirsher 			arr[i].port_id = port->logical_port_id;
2839aa32835SJeff Kirsher 			arr[i].reg_type = EHEA_BCMC_BROADCAST |
2849aa32835SJeff Kirsher 					  EHEA_BCMC_VLANID_ALL;
2859aa32835SJeff Kirsher 			arr[i++].macaddr = port->mac_addr;
2869aa32835SJeff Kirsher 			num_registrations -= 2;
2879aa32835SJeff Kirsher 
2889aa32835SJeff Kirsher 			list_for_each_entry(mc_entry,
2899aa32835SJeff Kirsher 					    &port->mc_list->list, list) {
2909aa32835SJeff Kirsher 				if (num_registrations == 0)
2919aa32835SJeff Kirsher 					goto out_update;
2929aa32835SJeff Kirsher 
2939aa32835SJeff Kirsher 				arr[i].adh = adapter->handle;
2949aa32835SJeff Kirsher 				arr[i].port_id = port->logical_port_id;
2955d384574SThadeu Lima de Souza Cascardo 				arr[i].reg_type = EHEA_BCMC_MULTICAST |
2969aa32835SJeff Kirsher 						  EHEA_BCMC_UNTAGGED;
2975d384574SThadeu Lima de Souza Cascardo 				if (mc_entry->macaddr == 0)
2985d384574SThadeu Lima de Souza Cascardo 					arr[i].reg_type |= EHEA_BCMC_SCOPE_ALL;
2999aa32835SJeff Kirsher 				arr[i++].macaddr = mc_entry->macaddr;
3009aa32835SJeff Kirsher 
3019aa32835SJeff Kirsher 				arr[i].adh = adapter->handle;
3029aa32835SJeff Kirsher 				arr[i].port_id = port->logical_port_id;
3035d384574SThadeu Lima de Souza Cascardo 				arr[i].reg_type = EHEA_BCMC_MULTICAST |
3049aa32835SJeff Kirsher 						  EHEA_BCMC_VLANID_ALL;
3055d384574SThadeu Lima de Souza Cascardo 				if (mc_entry->macaddr == 0)
3065d384574SThadeu Lima de Souza Cascardo 					arr[i].reg_type |= EHEA_BCMC_SCOPE_ALL;
3079aa32835SJeff Kirsher 				arr[i++].macaddr = mc_entry->macaddr;
3089aa32835SJeff Kirsher 				num_registrations -= 2;
3099aa32835SJeff Kirsher 			}
3109aa32835SJeff Kirsher 		}
3119aa32835SJeff Kirsher 	}
3129aa32835SJeff Kirsher 
3139aa32835SJeff Kirsher out_update:
3149aa32835SJeff Kirsher 	kfree(ehea_bcmc_regs.arr);
3159aa32835SJeff Kirsher 	ehea_bcmc_regs.arr = arr;
3169aa32835SJeff Kirsher 	ehea_bcmc_regs.num_entries = i;
3179aa32835SJeff Kirsher out:
3189aa32835SJeff Kirsher 	spin_unlock_irqrestore(&ehea_bcmc_regs.lock, flags);
3199aa32835SJeff Kirsher }
3209aa32835SJeff Kirsher 
ehea_get_stats64(struct net_device * dev,struct rtnl_link_stats64 * stats)321bc1f4470Sstephen hemminger static void ehea_get_stats64(struct net_device *dev,
322239c562cSAnton Blanchard 			     struct rtnl_link_stats64 *stats)
3239aa32835SJeff Kirsher {
3249aa32835SJeff Kirsher 	struct ehea_port *port = netdev_priv(dev);
3252aefcad8Sbrenohl@br.ibm.com 	u64 rx_packets = 0, tx_packets = 0, rx_bytes = 0, tx_bytes = 0;
3269aa32835SJeff Kirsher 	int i;
3279aa32835SJeff Kirsher 
3282aefcad8Sbrenohl@br.ibm.com 	for (i = 0; i < port->num_def_qps; i++) {
3292aefcad8Sbrenohl@br.ibm.com 		rx_packets += port->port_res[i].rx_packets;
3302aefcad8Sbrenohl@br.ibm.com 		rx_bytes   += port->port_res[i].rx_bytes;
3312aefcad8Sbrenohl@br.ibm.com 	}
3322aefcad8Sbrenohl@br.ibm.com 
333723f28e4SAnton Blanchard 	for (i = 0; i < port->num_def_qps; i++) {
3342aefcad8Sbrenohl@br.ibm.com 		tx_packets += port->port_res[i].tx_packets;
3352aefcad8Sbrenohl@br.ibm.com 		tx_bytes   += port->port_res[i].tx_bytes;
3362aefcad8Sbrenohl@br.ibm.com 	}
3372aefcad8Sbrenohl@br.ibm.com 
3382aefcad8Sbrenohl@br.ibm.com 	stats->tx_packets = tx_packets;
3392aefcad8Sbrenohl@br.ibm.com 	stats->rx_bytes = rx_bytes;
3402aefcad8Sbrenohl@br.ibm.com 	stats->tx_bytes = tx_bytes;
3412aefcad8Sbrenohl@br.ibm.com 	stats->rx_packets = rx_packets;
3422aefcad8Sbrenohl@br.ibm.com 
34309c1d446SEric Dumazet 	stats->multicast = port->stats.multicast;
34409c1d446SEric Dumazet 	stats->rx_errors = port->stats.rx_errors;
3452aefcad8Sbrenohl@br.ibm.com }
3462aefcad8Sbrenohl@br.ibm.com 
ehea_update_stats(struct work_struct * work)3472aefcad8Sbrenohl@br.ibm.com static void ehea_update_stats(struct work_struct *work)
3482aefcad8Sbrenohl@br.ibm.com {
3492aefcad8Sbrenohl@br.ibm.com 	struct ehea_port *port =
3502aefcad8Sbrenohl@br.ibm.com 		container_of(work, struct ehea_port, stats_work.work);
3512aefcad8Sbrenohl@br.ibm.com 	struct net_device *dev = port->netdev;
352239c562cSAnton Blanchard 	struct rtnl_link_stats64 *stats = &port->stats;
3532aefcad8Sbrenohl@br.ibm.com 	struct hcp_ehea_port_cb2 *cb2;
3542aefcad8Sbrenohl@br.ibm.com 	u64 hret;
3559aa32835SJeff Kirsher 
3569aa32835SJeff Kirsher 	cb2 = (void *)get_zeroed_page(GFP_KERNEL);
3579aa32835SJeff Kirsher 	if (!cb2) {
3582aefcad8Sbrenohl@br.ibm.com 		netdev_err(dev, "No mem for cb2. Some interface statistics were not updated\n");
3592aefcad8Sbrenohl@br.ibm.com 		goto resched;
3609aa32835SJeff Kirsher 	}
3619aa32835SJeff Kirsher 
3629aa32835SJeff Kirsher 	hret = ehea_h_query_ehea_port(port->adapter->handle,
3639aa32835SJeff Kirsher 				      port->logical_port_id,
3649aa32835SJeff Kirsher 				      H_PORT_CB2, H_PORT_CB2_ALL, cb2);
3659aa32835SJeff Kirsher 	if (hret != H_SUCCESS) {
3669aa32835SJeff Kirsher 		netdev_err(dev, "query_ehea_port failed\n");
3679aa32835SJeff Kirsher 		goto out_herr;
3689aa32835SJeff Kirsher 	}
3699aa32835SJeff Kirsher 
3709aa32835SJeff Kirsher 	if (netif_msg_hw(port))
3719aa32835SJeff Kirsher 		ehea_dump(cb2, sizeof(*cb2), "net_device_stats");
3729aa32835SJeff Kirsher 
3739aa32835SJeff Kirsher 	stats->multicast = cb2->rxmcp;
3749aa32835SJeff Kirsher 	stats->rx_errors = cb2->rxuerr;
3759aa32835SJeff Kirsher 
3769aa32835SJeff Kirsher out_herr:
3779aa32835SJeff Kirsher 	free_page((unsigned long)cb2);
3782aefcad8Sbrenohl@br.ibm.com resched:
37967c170a2SAnton Blanchard 	schedule_delayed_work(&port->stats_work,
38067c170a2SAnton Blanchard 			      round_jiffies_relative(msecs_to_jiffies(1000)));
3819aa32835SJeff Kirsher }
3829aa32835SJeff Kirsher 
ehea_refill_rq1(struct ehea_port_res * pr,int index,int nr_of_wqes)3839aa32835SJeff Kirsher static void ehea_refill_rq1(struct ehea_port_res *pr, int index, int nr_of_wqes)
3849aa32835SJeff Kirsher {
3859aa32835SJeff Kirsher 	struct sk_buff **skb_arr_rq1 = pr->rq1_skba.arr;
3869aa32835SJeff Kirsher 	struct net_device *dev = pr->port->netdev;
3879aa32835SJeff Kirsher 	int max_index_mask = pr->rq1_skba.len - 1;
3889aa32835SJeff Kirsher 	int fill_wqes = pr->rq1_skba.os_skbs + nr_of_wqes;
3899aa32835SJeff Kirsher 	int adder = 0;
3909aa32835SJeff Kirsher 	int i;
3919aa32835SJeff Kirsher 
3929aa32835SJeff Kirsher 	pr->rq1_skba.os_skbs = 0;
3939aa32835SJeff Kirsher 
3949aa32835SJeff Kirsher 	if (unlikely(test_bit(__EHEA_STOP_XFER, &ehea_driver_flags))) {
3959aa32835SJeff Kirsher 		if (nr_of_wqes > 0)
3969aa32835SJeff Kirsher 			pr->rq1_skba.index = index;
3979aa32835SJeff Kirsher 		pr->rq1_skba.os_skbs = fill_wqes;
3989aa32835SJeff Kirsher 		return;
3999aa32835SJeff Kirsher 	}
4009aa32835SJeff Kirsher 
4019aa32835SJeff Kirsher 	for (i = 0; i < fill_wqes; i++) {
4029aa32835SJeff Kirsher 		if (!skb_arr_rq1[index]) {
4039aa32835SJeff Kirsher 			skb_arr_rq1[index] = netdev_alloc_skb(dev,
4049aa32835SJeff Kirsher 							      EHEA_L_PKT_SIZE);
4059aa32835SJeff Kirsher 			if (!skb_arr_rq1[index]) {
4069aa32835SJeff Kirsher 				pr->rq1_skba.os_skbs = fill_wqes - i;
4079aa32835SJeff Kirsher 				break;
4089aa32835SJeff Kirsher 			}
4099aa32835SJeff Kirsher 		}
4109aa32835SJeff Kirsher 		index--;
4119aa32835SJeff Kirsher 		index &= max_index_mask;
4129aa32835SJeff Kirsher 		adder++;
4139aa32835SJeff Kirsher 	}
4149aa32835SJeff Kirsher 
4159aa32835SJeff Kirsher 	if (adder == 0)
4169aa32835SJeff Kirsher 		return;
4179aa32835SJeff Kirsher 
4189aa32835SJeff Kirsher 	/* Ring doorbell */
4199aa32835SJeff Kirsher 	ehea_update_rq1a(pr->qp, adder);
4209aa32835SJeff Kirsher }
4219aa32835SJeff Kirsher 
ehea_init_fill_rq1(struct ehea_port_res * pr,int nr_rq1a)4229aa32835SJeff Kirsher static void ehea_init_fill_rq1(struct ehea_port_res *pr, int nr_rq1a)
4239aa32835SJeff Kirsher {
4249aa32835SJeff Kirsher 	struct sk_buff **skb_arr_rq1 = pr->rq1_skba.arr;
4259aa32835SJeff Kirsher 	struct net_device *dev = pr->port->netdev;
4269aa32835SJeff Kirsher 	int i;
4279aa32835SJeff Kirsher 
4289aa32835SJeff Kirsher 	if (nr_rq1a > pr->rq1_skba.len) {
4299aa32835SJeff Kirsher 		netdev_err(dev, "NR_RQ1A bigger than skb array len\n");
4309aa32835SJeff Kirsher 		return;
4319aa32835SJeff Kirsher 	}
4329aa32835SJeff Kirsher 
4339aa32835SJeff Kirsher 	for (i = 0; i < nr_rq1a; i++) {
4349aa32835SJeff Kirsher 		skb_arr_rq1[i] = netdev_alloc_skb(dev, EHEA_L_PKT_SIZE);
435720a43efSJoe Perches 		if (!skb_arr_rq1[i])
4369aa32835SJeff Kirsher 			break;
4379aa32835SJeff Kirsher 	}
4389aa32835SJeff Kirsher 	/* Ring doorbell */
4399aa32835SJeff Kirsher 	ehea_update_rq1a(pr->qp, i - 1);
4409aa32835SJeff Kirsher }
4419aa32835SJeff Kirsher 
ehea_refill_rq_def(struct ehea_port_res * pr,struct ehea_q_skb_arr * q_skba,int rq_nr,int num_wqes,int wqe_type,int packet_size)4429aa32835SJeff Kirsher static int ehea_refill_rq_def(struct ehea_port_res *pr,
4439aa32835SJeff Kirsher 			      struct ehea_q_skb_arr *q_skba, int rq_nr,
4449aa32835SJeff Kirsher 			      int num_wqes, int wqe_type, int packet_size)
4459aa32835SJeff Kirsher {
4469aa32835SJeff Kirsher 	struct net_device *dev = pr->port->netdev;
4479aa32835SJeff Kirsher 	struct ehea_qp *qp = pr->qp;
4489aa32835SJeff Kirsher 	struct sk_buff **skb_arr = q_skba->arr;
4499aa32835SJeff Kirsher 	struct ehea_rwqe *rwqe;
4509aa32835SJeff Kirsher 	int i, index, max_index_mask, fill_wqes;
4519aa32835SJeff Kirsher 	int adder = 0;
4529aa32835SJeff Kirsher 	int ret = 0;
4539aa32835SJeff Kirsher 
4549aa32835SJeff Kirsher 	fill_wqes = q_skba->os_skbs + num_wqes;
4559aa32835SJeff Kirsher 	q_skba->os_skbs = 0;
4569aa32835SJeff Kirsher 
4579aa32835SJeff Kirsher 	if (unlikely(test_bit(__EHEA_STOP_XFER, &ehea_driver_flags))) {
4589aa32835SJeff Kirsher 		q_skba->os_skbs = fill_wqes;
4599aa32835SJeff Kirsher 		return ret;
4609aa32835SJeff Kirsher 	}
4619aa32835SJeff Kirsher 
4629aa32835SJeff Kirsher 	index = q_skba->index;
4639aa32835SJeff Kirsher 	max_index_mask = q_skba->len - 1;
4649aa32835SJeff Kirsher 	for (i = 0; i < fill_wqes; i++) {
4659aa32835SJeff Kirsher 		u64 tmp_addr;
4669aa32835SJeff Kirsher 		struct sk_buff *skb;
4679aa32835SJeff Kirsher 
4689aa32835SJeff Kirsher 		skb = netdev_alloc_skb_ip_align(dev, packet_size);
4699aa32835SJeff Kirsher 		if (!skb) {
4709aa32835SJeff Kirsher 			q_skba->os_skbs = fill_wqes - i;
4719aa32835SJeff Kirsher 			if (q_skba->os_skbs == q_skba->len - 2) {
4729aa32835SJeff Kirsher 				netdev_info(pr->port->netdev,
4739aa32835SJeff Kirsher 					    "rq%i ran dry - no mem for skb\n",
4749aa32835SJeff Kirsher 					    rq_nr);
4759aa32835SJeff Kirsher 				ret = -ENOMEM;
4769aa32835SJeff Kirsher 			}
4779aa32835SJeff Kirsher 			break;
4789aa32835SJeff Kirsher 		}
4799aa32835SJeff Kirsher 
4809aa32835SJeff Kirsher 		skb_arr[index] = skb;
4819aa32835SJeff Kirsher 		tmp_addr = ehea_map_vaddr(skb->data);
4829aa32835SJeff Kirsher 		if (tmp_addr == -1) {
483cfbe4063SEric W. Biederman 			dev_consume_skb_any(skb);
4849aa32835SJeff Kirsher 			q_skba->os_skbs = fill_wqes - i;
4859aa32835SJeff Kirsher 			ret = 0;
4869aa32835SJeff Kirsher 			break;
4879aa32835SJeff Kirsher 		}
4889aa32835SJeff Kirsher 
4899aa32835SJeff Kirsher 		rwqe = ehea_get_next_rwqe(qp, rq_nr);
4909aa32835SJeff Kirsher 		rwqe->wr_id = EHEA_BMASK_SET(EHEA_WR_ID_TYPE, wqe_type)
4919aa32835SJeff Kirsher 			    | EHEA_BMASK_SET(EHEA_WR_ID_INDEX, index);
4929aa32835SJeff Kirsher 		rwqe->sg_list[0].l_key = pr->recv_mr.lkey;
4939aa32835SJeff Kirsher 		rwqe->sg_list[0].vaddr = tmp_addr;
4949aa32835SJeff Kirsher 		rwqe->sg_list[0].len = packet_size;
4959aa32835SJeff Kirsher 		rwqe->data_segments = 1;
4969aa32835SJeff Kirsher 
4979aa32835SJeff Kirsher 		index++;
4989aa32835SJeff Kirsher 		index &= max_index_mask;
4999aa32835SJeff Kirsher 		adder++;
5009aa32835SJeff Kirsher 	}
5019aa32835SJeff Kirsher 
5029aa32835SJeff Kirsher 	q_skba->index = index;
5039aa32835SJeff Kirsher 	if (adder == 0)
5049aa32835SJeff Kirsher 		goto out;
5059aa32835SJeff Kirsher 
5069aa32835SJeff Kirsher 	/* Ring doorbell */
5079aa32835SJeff Kirsher 	iosync();
5089aa32835SJeff Kirsher 	if (rq_nr == 2)
5099aa32835SJeff Kirsher 		ehea_update_rq2a(pr->qp, adder);
5109aa32835SJeff Kirsher 	else
5119aa32835SJeff Kirsher 		ehea_update_rq3a(pr->qp, adder);
5129aa32835SJeff Kirsher out:
5139aa32835SJeff Kirsher 	return ret;
5149aa32835SJeff Kirsher }
5159aa32835SJeff Kirsher 
5169aa32835SJeff Kirsher 
ehea_refill_rq2(struct ehea_port_res * pr,int nr_of_wqes)5179aa32835SJeff Kirsher static int ehea_refill_rq2(struct ehea_port_res *pr, int nr_of_wqes)
5189aa32835SJeff Kirsher {
5199aa32835SJeff Kirsher 	return ehea_refill_rq_def(pr, &pr->rq2_skba, 2,
5209aa32835SJeff Kirsher 				  nr_of_wqes, EHEA_RWQE2_TYPE,
5219aa32835SJeff Kirsher 				  EHEA_RQ2_PKT_SIZE);
5229aa32835SJeff Kirsher }
5239aa32835SJeff Kirsher 
5249aa32835SJeff Kirsher 
ehea_refill_rq3(struct ehea_port_res * pr,int nr_of_wqes)5259aa32835SJeff Kirsher static int ehea_refill_rq3(struct ehea_port_res *pr, int nr_of_wqes)
5269aa32835SJeff Kirsher {
5279aa32835SJeff Kirsher 	return ehea_refill_rq_def(pr, &pr->rq3_skba, 3,
5289aa32835SJeff Kirsher 				  nr_of_wqes, EHEA_RWQE3_TYPE,
5299aa32835SJeff Kirsher 				  EHEA_MAX_PACKET_SIZE);
5309aa32835SJeff Kirsher }
5319aa32835SJeff Kirsher 
ehea_check_cqe(struct ehea_cqe * cqe,int * rq_num)5329aa32835SJeff Kirsher static inline int ehea_check_cqe(struct ehea_cqe *cqe, int *rq_num)
5339aa32835SJeff Kirsher {
5349aa32835SJeff Kirsher 	*rq_num = (cqe->type & EHEA_CQE_TYPE_RQ) >> 5;
5359aa32835SJeff Kirsher 	if ((cqe->status & EHEA_CQE_STAT_ERR_MASK) == 0)
5369aa32835SJeff Kirsher 		return 0;
5379aa32835SJeff Kirsher 	if (((cqe->status & EHEA_CQE_STAT_ERR_TCP) != 0) &&
5389aa32835SJeff Kirsher 	    (cqe->header_length == 0))
5399aa32835SJeff Kirsher 		return 0;
5409aa32835SJeff Kirsher 	return -EINVAL;
5419aa32835SJeff Kirsher }
5429aa32835SJeff Kirsher 
ehea_fill_skb(struct net_device * dev,struct sk_buff * skb,struct ehea_cqe * cqe,struct ehea_port_res * pr)5439aa32835SJeff Kirsher static inline void ehea_fill_skb(struct net_device *dev,
544b9564468SAnton Blanchard 				 struct sk_buff *skb, struct ehea_cqe *cqe,
545b9564468SAnton Blanchard 				 struct ehea_port_res *pr)
5469aa32835SJeff Kirsher {
5479aa32835SJeff Kirsher 	int length = cqe->num_bytes_transfered - 4;	/*remove CRC */
5489aa32835SJeff Kirsher 
5499aa32835SJeff Kirsher 	skb_put(skb, length);
5509aa32835SJeff Kirsher 	skb->protocol = eth_type_trans(skb, dev);
5519aa32835SJeff Kirsher 
5529aa32835SJeff Kirsher 	/* The packet was not an IPV4 packet so a complemented checksum was
5539aa32835SJeff Kirsher 	   calculated. The value is found in the Internet Checksum field. */
5549aa32835SJeff Kirsher 	if (cqe->status & EHEA_CQE_BLIND_CKSUM) {
5559aa32835SJeff Kirsher 		skb->ip_summed = CHECKSUM_COMPLETE;
5569aa32835SJeff Kirsher 		skb->csum = csum_unfold(~cqe->inet_checksum_value);
5579aa32835SJeff Kirsher 	} else
5589aa32835SJeff Kirsher 		skb->ip_summed = CHECKSUM_UNNECESSARY;
559b9564468SAnton Blanchard 
560b9564468SAnton Blanchard 	skb_record_rx_queue(skb, pr - &pr->port->port_res[0]);
5619aa32835SJeff Kirsher }
5629aa32835SJeff Kirsher 
get_skb_by_index(struct sk_buff ** skb_array,int arr_len,struct ehea_cqe * cqe)5639aa32835SJeff Kirsher static inline struct sk_buff *get_skb_by_index(struct sk_buff **skb_array,
5649aa32835SJeff Kirsher 					       int arr_len,
5659aa32835SJeff Kirsher 					       struct ehea_cqe *cqe)
5669aa32835SJeff Kirsher {
5679aa32835SJeff Kirsher 	int skb_index = EHEA_BMASK_GET(EHEA_WR_ID_INDEX, cqe->wr_id);
5689aa32835SJeff Kirsher 	struct sk_buff *skb;
5699aa32835SJeff Kirsher 	void *pref;
5709aa32835SJeff Kirsher 	int x;
5719aa32835SJeff Kirsher 
5729aa32835SJeff Kirsher 	x = skb_index + 1;
5739aa32835SJeff Kirsher 	x &= (arr_len - 1);
5749aa32835SJeff Kirsher 
5759aa32835SJeff Kirsher 	pref = skb_array[x];
5769aa32835SJeff Kirsher 	if (pref) {
5779aa32835SJeff Kirsher 		prefetchw(pref);
5789aa32835SJeff Kirsher 		prefetchw(pref + EHEA_CACHE_LINE);
5799aa32835SJeff Kirsher 
5809aa32835SJeff Kirsher 		pref = (skb_array[x]->data);
5819aa32835SJeff Kirsher 		prefetch(pref);
5829aa32835SJeff Kirsher 		prefetch(pref + EHEA_CACHE_LINE);
5839aa32835SJeff Kirsher 		prefetch(pref + EHEA_CACHE_LINE * 2);
5849aa32835SJeff Kirsher 		prefetch(pref + EHEA_CACHE_LINE * 3);
5859aa32835SJeff Kirsher 	}
5869aa32835SJeff Kirsher 
5879aa32835SJeff Kirsher 	skb = skb_array[skb_index];
5889aa32835SJeff Kirsher 	skb_array[skb_index] = NULL;
5899aa32835SJeff Kirsher 	return skb;
5909aa32835SJeff Kirsher }
5919aa32835SJeff Kirsher 
get_skb_by_index_ll(struct sk_buff ** skb_array,int arr_len,int wqe_index)5929aa32835SJeff Kirsher static inline struct sk_buff *get_skb_by_index_ll(struct sk_buff **skb_array,
5939aa32835SJeff Kirsher 						  int arr_len, int wqe_index)
5949aa32835SJeff Kirsher {
5959aa32835SJeff Kirsher 	struct sk_buff *skb;
5969aa32835SJeff Kirsher 	void *pref;
5979aa32835SJeff Kirsher 	int x;
5989aa32835SJeff Kirsher 
5999aa32835SJeff Kirsher 	x = wqe_index + 1;
6009aa32835SJeff Kirsher 	x &= (arr_len - 1);
6019aa32835SJeff Kirsher 
6029aa32835SJeff Kirsher 	pref = skb_array[x];
6039aa32835SJeff Kirsher 	if (pref) {
6049aa32835SJeff Kirsher 		prefetchw(pref);
6059aa32835SJeff Kirsher 		prefetchw(pref + EHEA_CACHE_LINE);
6069aa32835SJeff Kirsher 
6079aa32835SJeff Kirsher 		pref = (skb_array[x]->data);
6089aa32835SJeff Kirsher 		prefetchw(pref);
6099aa32835SJeff Kirsher 		prefetchw(pref + EHEA_CACHE_LINE);
6109aa32835SJeff Kirsher 	}
6119aa32835SJeff Kirsher 
6129aa32835SJeff Kirsher 	skb = skb_array[wqe_index];
6139aa32835SJeff Kirsher 	skb_array[wqe_index] = NULL;
6149aa32835SJeff Kirsher 	return skb;
6159aa32835SJeff Kirsher }
6169aa32835SJeff Kirsher 
ehea_treat_poll_error(struct ehea_port_res * pr,int rq,struct ehea_cqe * cqe,int * processed_rq2,int * processed_rq3)6179aa32835SJeff Kirsher static int ehea_treat_poll_error(struct ehea_port_res *pr, int rq,
6189aa32835SJeff Kirsher 				 struct ehea_cqe *cqe, int *processed_rq2,
6199aa32835SJeff Kirsher 				 int *processed_rq3)
6209aa32835SJeff Kirsher {
6219aa32835SJeff Kirsher 	struct sk_buff *skb;
6229aa32835SJeff Kirsher 
6239aa32835SJeff Kirsher 	if (cqe->status & EHEA_CQE_STAT_ERR_TCP)
6249aa32835SJeff Kirsher 		pr->p_stats.err_tcp_cksum++;
6259aa32835SJeff Kirsher 	if (cqe->status & EHEA_CQE_STAT_ERR_IP)
6269aa32835SJeff Kirsher 		pr->p_stats.err_ip_cksum++;
6279aa32835SJeff Kirsher 	if (cqe->status & EHEA_CQE_STAT_ERR_CRC)
6289aa32835SJeff Kirsher 		pr->p_stats.err_frame_crc++;
6299aa32835SJeff Kirsher 
6309aa32835SJeff Kirsher 	if (rq == 2) {
6319aa32835SJeff Kirsher 		*processed_rq2 += 1;
6329aa32835SJeff Kirsher 		skb = get_skb_by_index(pr->rq2_skba.arr, pr->rq2_skba.len, cqe);
6339aa32835SJeff Kirsher 		dev_kfree_skb(skb);
6349aa32835SJeff Kirsher 	} else if (rq == 3) {
6359aa32835SJeff Kirsher 		*processed_rq3 += 1;
6369aa32835SJeff Kirsher 		skb = get_skb_by_index(pr->rq3_skba.arr, pr->rq3_skba.len, cqe);
6379aa32835SJeff Kirsher 		dev_kfree_skb(skb);
6389aa32835SJeff Kirsher 	}
6399aa32835SJeff Kirsher 
6409aa32835SJeff Kirsher 	if (cqe->status & EHEA_CQE_STAT_FAT_ERR_MASK) {
6419aa32835SJeff Kirsher 		if (netif_msg_rx_err(pr->port)) {
6429aa32835SJeff Kirsher 			pr_err("Critical receive error for QP %d. Resetting port.\n",
6439aa32835SJeff Kirsher 			       pr->qp->init_attr.qp_nr);
6449aa32835SJeff Kirsher 			ehea_dump(cqe, sizeof(*cqe), "CQE");
6459aa32835SJeff Kirsher 		}
6469aa32835SJeff Kirsher 		ehea_schedule_port_reset(pr->port);
6479aa32835SJeff Kirsher 		return 1;
6489aa32835SJeff Kirsher 	}
6499aa32835SJeff Kirsher 
6509aa32835SJeff Kirsher 	return 0;
6519aa32835SJeff Kirsher }
6529aa32835SJeff Kirsher 
ehea_proc_rwqes(struct net_device * dev,struct ehea_port_res * pr,int budget)6539aa32835SJeff Kirsher static int ehea_proc_rwqes(struct net_device *dev,
6549aa32835SJeff Kirsher 			   struct ehea_port_res *pr,
6559aa32835SJeff Kirsher 			   int budget)
6569aa32835SJeff Kirsher {
6579aa32835SJeff Kirsher 	struct ehea_port *port = pr->port;
6589aa32835SJeff Kirsher 	struct ehea_qp *qp = pr->qp;
6599aa32835SJeff Kirsher 	struct ehea_cqe *cqe;
6609aa32835SJeff Kirsher 	struct sk_buff *skb;
6619aa32835SJeff Kirsher 	struct sk_buff **skb_arr_rq1 = pr->rq1_skba.arr;
6629aa32835SJeff Kirsher 	struct sk_buff **skb_arr_rq2 = pr->rq2_skba.arr;
6639aa32835SJeff Kirsher 	struct sk_buff **skb_arr_rq3 = pr->rq3_skba.arr;
6649aa32835SJeff Kirsher 	int skb_arr_rq1_len = pr->rq1_skba.len;
6659aa32835SJeff Kirsher 	int skb_arr_rq2_len = pr->rq2_skba.len;
6669aa32835SJeff Kirsher 	int skb_arr_rq3_len = pr->rq3_skba.len;
6679aa32835SJeff Kirsher 	int processed, processed_rq1, processed_rq2, processed_rq3;
6689aa32835SJeff Kirsher 	u64 processed_bytes = 0;
6699aa32835SJeff Kirsher 	int wqe_index, last_wqe_index, rq, port_reset;
6709aa32835SJeff Kirsher 
6719aa32835SJeff Kirsher 	processed = processed_rq1 = processed_rq2 = processed_rq3 = 0;
6729aa32835SJeff Kirsher 	last_wqe_index = 0;
6739aa32835SJeff Kirsher 
6749aa32835SJeff Kirsher 	cqe = ehea_poll_rq1(qp, &wqe_index);
6759aa32835SJeff Kirsher 	while ((processed < budget) && cqe) {
6769aa32835SJeff Kirsher 		ehea_inc_rq1(qp);
6779aa32835SJeff Kirsher 		processed_rq1++;
6789aa32835SJeff Kirsher 		processed++;
6799aa32835SJeff Kirsher 		if (netif_msg_rx_status(port))
6809aa32835SJeff Kirsher 			ehea_dump(cqe, sizeof(*cqe), "CQE");
6819aa32835SJeff Kirsher 
6829aa32835SJeff Kirsher 		last_wqe_index = wqe_index;
6839aa32835SJeff Kirsher 		rmb();
6849aa32835SJeff Kirsher 		if (!ehea_check_cqe(cqe, &rq)) {
6859aa32835SJeff Kirsher 			if (rq == 1) {
6869aa32835SJeff Kirsher 				/* LL RQ1 */
6879aa32835SJeff Kirsher 				skb = get_skb_by_index_ll(skb_arr_rq1,
6889aa32835SJeff Kirsher 							  skb_arr_rq1_len,
6899aa32835SJeff Kirsher 							  wqe_index);
6909aa32835SJeff Kirsher 				if (unlikely(!skb)) {
6919aa32835SJeff Kirsher 					netif_info(port, rx_err, dev,
6929aa32835SJeff Kirsher 						  "LL rq1: skb=NULL\n");
6939aa32835SJeff Kirsher 
6949aa32835SJeff Kirsher 					skb = netdev_alloc_skb(dev,
6959aa32835SJeff Kirsher 							       EHEA_L_PKT_SIZE);
696720a43efSJoe Perches 					if (!skb)
6979aa32835SJeff Kirsher 						break;
6989aa32835SJeff Kirsher 				}
6999aa32835SJeff Kirsher 				skb_copy_to_linear_data(skb, ((char *)cqe) + 64,
7009aa32835SJeff Kirsher 						 cqe->num_bytes_transfered - 4);
701b9564468SAnton Blanchard 				ehea_fill_skb(dev, skb, cqe, pr);
7029aa32835SJeff Kirsher 			} else if (rq == 2) {
7039aa32835SJeff Kirsher 				/* RQ2 */
7049aa32835SJeff Kirsher 				skb = get_skb_by_index(skb_arr_rq2,
7059aa32835SJeff Kirsher 						       skb_arr_rq2_len, cqe);
7069aa32835SJeff Kirsher 				if (unlikely(!skb)) {
7079aa32835SJeff Kirsher 					netif_err(port, rx_err, dev,
7089aa32835SJeff Kirsher 						  "rq2: skb=NULL\n");
7099aa32835SJeff Kirsher 					break;
7109aa32835SJeff Kirsher 				}
711b9564468SAnton Blanchard 				ehea_fill_skb(dev, skb, cqe, pr);
7129aa32835SJeff Kirsher 				processed_rq2++;
7139aa32835SJeff Kirsher 			} else {
7149aa32835SJeff Kirsher 				/* RQ3 */
7159aa32835SJeff Kirsher 				skb = get_skb_by_index(skb_arr_rq3,
7169aa32835SJeff Kirsher 						       skb_arr_rq3_len, cqe);
7179aa32835SJeff Kirsher 				if (unlikely(!skb)) {
7189aa32835SJeff Kirsher 					netif_err(port, rx_err, dev,
7199aa32835SJeff Kirsher 						  "rq3: skb=NULL\n");
7209aa32835SJeff Kirsher 					break;
7219aa32835SJeff Kirsher 				}
722b9564468SAnton Blanchard 				ehea_fill_skb(dev, skb, cqe, pr);
7239aa32835SJeff Kirsher 				processed_rq3++;
7249aa32835SJeff Kirsher 			}
7259aa32835SJeff Kirsher 
7269aa32835SJeff Kirsher 			processed_bytes += skb->len;
7273428414fSAnton Blanchard 
7283428414fSAnton Blanchard 			if (cqe->status & EHEA_CQE_VLAN_TAG_XTRACT)
729e5905c83SDavid S. Miller 				__vlan_hwaccel_put_tag(skb, htons(ETH_P_8021Q),
730e5905c83SDavid S. Miller 						       cqe->vlan_tag);
7313428414fSAnton Blanchard 
7323428414fSAnton Blanchard 			napi_gro_receive(&pr->napi, skb);
7339aa32835SJeff Kirsher 		} else {
7349aa32835SJeff Kirsher 			pr->p_stats.poll_receive_errors++;
7359aa32835SJeff Kirsher 			port_reset = ehea_treat_poll_error(pr, rq, cqe,
7369aa32835SJeff Kirsher 							   &processed_rq2,
7379aa32835SJeff Kirsher 							   &processed_rq3);
7389aa32835SJeff Kirsher 			if (port_reset)
7399aa32835SJeff Kirsher 				break;
7409aa32835SJeff Kirsher 		}
7419aa32835SJeff Kirsher 		cqe = ehea_poll_rq1(qp, &wqe_index);
7429aa32835SJeff Kirsher 	}
7439aa32835SJeff Kirsher 
7449aa32835SJeff Kirsher 	pr->rx_packets += processed;
7459aa32835SJeff Kirsher 	pr->rx_bytes += processed_bytes;
7469aa32835SJeff Kirsher 
7479aa32835SJeff Kirsher 	ehea_refill_rq1(pr, last_wqe_index, processed_rq1);
7489aa32835SJeff Kirsher 	ehea_refill_rq2(pr, processed_rq2);
7499aa32835SJeff Kirsher 	ehea_refill_rq3(pr, processed_rq3);
7509aa32835SJeff Kirsher 
7519aa32835SJeff Kirsher 	return processed;
7529aa32835SJeff Kirsher }
7539aa32835SJeff Kirsher 
7549aa32835SJeff Kirsher #define SWQE_RESTART_CHECK 0xdeadbeaff00d0000ull
7559aa32835SJeff Kirsher 
reset_sq_restart_flag(struct ehea_port * port)7569aa32835SJeff Kirsher static void reset_sq_restart_flag(struct ehea_port *port)
7579aa32835SJeff Kirsher {
7589aa32835SJeff Kirsher 	int i;
7599aa32835SJeff Kirsher 
760723f28e4SAnton Blanchard 	for (i = 0; i < port->num_def_qps; i++) {
7619aa32835SJeff Kirsher 		struct ehea_port_res *pr = &port->port_res[i];
7629aa32835SJeff Kirsher 		pr->sq_restart_flag = 0;
7639aa32835SJeff Kirsher 	}
7649aa32835SJeff Kirsher 	wake_up(&port->restart_wq);
7659aa32835SJeff Kirsher }
7669aa32835SJeff Kirsher 
check_sqs(struct ehea_port * port)7679aa32835SJeff Kirsher static void check_sqs(struct ehea_port *port)
7689aa32835SJeff Kirsher {
7699aa32835SJeff Kirsher 	struct ehea_swqe *swqe;
7709aa32835SJeff Kirsher 	int swqe_index;
771c1f9a93dSzhong jiang 	int i;
7729aa32835SJeff Kirsher 
773723f28e4SAnton Blanchard 	for (i = 0; i < port->num_def_qps; i++) {
7749aa32835SJeff Kirsher 		struct ehea_port_res *pr = &port->port_res[i];
7759aa32835SJeff Kirsher 		int ret;
7769aa32835SJeff Kirsher 		swqe = ehea_get_swqe(pr->qp, &swqe_index);
7779aa32835SJeff Kirsher 		memset(swqe, 0, SWQE_HEADER_SIZE);
7789aa32835SJeff Kirsher 		atomic_dec(&pr->swqe_avail);
7799aa32835SJeff Kirsher 
7809aa32835SJeff Kirsher 		swqe->tx_control |= EHEA_SWQE_PURGE;
7819aa32835SJeff Kirsher 		swqe->wr_id = SWQE_RESTART_CHECK;
7829aa32835SJeff Kirsher 		swqe->tx_control |= EHEA_SWQE_SIGNALLED_COMPLETION;
7839aa32835SJeff Kirsher 		swqe->tx_control |= EHEA_SWQE_IMM_DATA_PRESENT;
7849aa32835SJeff Kirsher 		swqe->immediate_data_length = 80;
7859aa32835SJeff Kirsher 
7869aa32835SJeff Kirsher 		ehea_post_swqe(pr->qp, swqe);
7879aa32835SJeff Kirsher 
7889aa32835SJeff Kirsher 		ret = wait_event_timeout(port->restart_wq,
7899aa32835SJeff Kirsher 					 pr->sq_restart_flag == 0,
7909aa32835SJeff Kirsher 					 msecs_to_jiffies(100));
7919aa32835SJeff Kirsher 
7929aa32835SJeff Kirsher 		if (!ret) {
7939aa32835SJeff Kirsher 			pr_err("HW/SW queues out of sync\n");
7949aa32835SJeff Kirsher 			ehea_schedule_port_reset(pr->port);
7959aa32835SJeff Kirsher 			return;
7969aa32835SJeff Kirsher 		}
7979aa32835SJeff Kirsher 	}
7989aa32835SJeff Kirsher }
7999aa32835SJeff Kirsher 
8009aa32835SJeff Kirsher 
ehea_proc_cqes(struct ehea_port_res * pr,int my_quota)8019aa32835SJeff Kirsher static struct ehea_cqe *ehea_proc_cqes(struct ehea_port_res *pr, int my_quota)
8029aa32835SJeff Kirsher {
8039aa32835SJeff Kirsher 	struct sk_buff *skb;
8049aa32835SJeff Kirsher 	struct ehea_cq *send_cq = pr->send_cq;
8059aa32835SJeff Kirsher 	struct ehea_cqe *cqe;
8069aa32835SJeff Kirsher 	int quota = my_quota;
8079aa32835SJeff Kirsher 	int cqe_counter = 0;
8089aa32835SJeff Kirsher 	int swqe_av = 0;
8099aa32835SJeff Kirsher 	int index;
810b9564468SAnton Blanchard 	struct netdev_queue *txq = netdev_get_tx_queue(pr->port->netdev,
811b9564468SAnton Blanchard 						pr - &pr->port->port_res[0]);
8129aa32835SJeff Kirsher 
8139aa32835SJeff Kirsher 	cqe = ehea_poll_cq(send_cq);
8149aa32835SJeff Kirsher 	while (cqe && (quota > 0)) {
8159aa32835SJeff Kirsher 		ehea_inc_cq(send_cq);
8169aa32835SJeff Kirsher 
8179aa32835SJeff Kirsher 		cqe_counter++;
8189aa32835SJeff Kirsher 		rmb();
8199aa32835SJeff Kirsher 
8209aa32835SJeff Kirsher 		if (cqe->wr_id == SWQE_RESTART_CHECK) {
8219aa32835SJeff Kirsher 			pr->sq_restart_flag = 1;
8229aa32835SJeff Kirsher 			swqe_av++;
8239aa32835SJeff Kirsher 			break;
8249aa32835SJeff Kirsher 		}
8259aa32835SJeff Kirsher 
8269aa32835SJeff Kirsher 		if (cqe->status & EHEA_CQE_STAT_ERR_MASK) {
8279aa32835SJeff Kirsher 			pr_err("Bad send completion status=0x%04X\n",
8289aa32835SJeff Kirsher 			       cqe->status);
8299aa32835SJeff Kirsher 
8309aa32835SJeff Kirsher 			if (netif_msg_tx_err(pr->port))
8319aa32835SJeff Kirsher 				ehea_dump(cqe, sizeof(*cqe), "Send CQE");
8329aa32835SJeff Kirsher 
8339aa32835SJeff Kirsher 			if (cqe->status & EHEA_CQE_STAT_RESET_MASK) {
8349aa32835SJeff Kirsher 				pr_err("Resetting port\n");
8359aa32835SJeff Kirsher 				ehea_schedule_port_reset(pr->port);
8369aa32835SJeff Kirsher 				break;
8379aa32835SJeff Kirsher 			}
8389aa32835SJeff Kirsher 		}
8399aa32835SJeff Kirsher 
8409aa32835SJeff Kirsher 		if (netif_msg_tx_done(pr->port))
8419aa32835SJeff Kirsher 			ehea_dump(cqe, sizeof(*cqe), "CQE");
8429aa32835SJeff Kirsher 
8439aa32835SJeff Kirsher 		if (likely(EHEA_BMASK_GET(EHEA_WR_ID_TYPE, cqe->wr_id)
8449aa32835SJeff Kirsher 			   == EHEA_SWQE2_TYPE)) {
8459aa32835SJeff Kirsher 
8469aa32835SJeff Kirsher 			index = EHEA_BMASK_GET(EHEA_WR_ID_INDEX, cqe->wr_id);
8479aa32835SJeff Kirsher 			skb = pr->sq_skba.arr[index];
848cfbe4063SEric W. Biederman 			dev_consume_skb_any(skb);
8499aa32835SJeff Kirsher 			pr->sq_skba.arr[index] = NULL;
8509aa32835SJeff Kirsher 		}
8519aa32835SJeff Kirsher 
8529aa32835SJeff Kirsher 		swqe_av += EHEA_BMASK_GET(EHEA_WR_ID_REFILL, cqe->wr_id);
8539aa32835SJeff Kirsher 		quota--;
8549aa32835SJeff Kirsher 
8559aa32835SJeff Kirsher 		cqe = ehea_poll_cq(send_cq);
8569aa32835SJeff Kirsher 	}
8579aa32835SJeff Kirsher 
8589aa32835SJeff Kirsher 	ehea_update_feca(send_cq, cqe_counter);
8599aa32835SJeff Kirsher 	atomic_add(swqe_av, &pr->swqe_avail);
8609aa32835SJeff Kirsher 
861b9564468SAnton Blanchard 	if (unlikely(netif_tx_queue_stopped(txq) &&
862b9564468SAnton Blanchard 		     (atomic_read(&pr->swqe_avail) >= pr->swqe_refill_th))) {
863b9564468SAnton Blanchard 		__netif_tx_lock(txq, smp_processor_id());
864b9564468SAnton Blanchard 		if (netif_tx_queue_stopped(txq) &&
865b9564468SAnton Blanchard 		    (atomic_read(&pr->swqe_avail) >= pr->swqe_refill_th))
866b9564468SAnton Blanchard 			netif_tx_wake_queue(txq);
867b9564468SAnton Blanchard 		__netif_tx_unlock(txq);
8689aa32835SJeff Kirsher 	}
869b9564468SAnton Blanchard 
8709aa32835SJeff Kirsher 	wake_up(&pr->port->swqe_avail_wq);
8719aa32835SJeff Kirsher 
8729aa32835SJeff Kirsher 	return cqe;
8739aa32835SJeff Kirsher }
8749aa32835SJeff Kirsher 
8759aa32835SJeff Kirsher #define EHEA_POLL_MAX_CQES 65535
8769aa32835SJeff Kirsher 
ehea_poll(struct napi_struct * napi,int budget)8779aa32835SJeff Kirsher static int ehea_poll(struct napi_struct *napi, int budget)
8789aa32835SJeff Kirsher {
8799aa32835SJeff Kirsher 	struct ehea_port_res *pr = container_of(napi, struct ehea_port_res,
8809aa32835SJeff Kirsher 						napi);
8819aa32835SJeff Kirsher 	struct net_device *dev = pr->port->netdev;
8829aa32835SJeff Kirsher 	struct ehea_cqe *cqe;
8839aa32835SJeff Kirsher 	struct ehea_cqe *cqe_skb = NULL;
884222ca96bSAnton Blanchard 	int wqe_index;
8859aa32835SJeff Kirsher 	int rx = 0;
8869aa32835SJeff Kirsher 
8879aa32835SJeff Kirsher 	cqe_skb = ehea_proc_cqes(pr, EHEA_POLL_MAX_CQES);
8889aa32835SJeff Kirsher 	rx += ehea_proc_rwqes(dev, pr, budget - rx);
8899aa32835SJeff Kirsher 
890222ca96bSAnton Blanchard 	while (rx != budget) {
8919aa32835SJeff Kirsher 		napi_complete(napi);
8929aa32835SJeff Kirsher 		ehea_reset_cq_ep(pr->recv_cq);
8939aa32835SJeff Kirsher 		ehea_reset_cq_ep(pr->send_cq);
8949aa32835SJeff Kirsher 		ehea_reset_cq_n1(pr->recv_cq);
8959aa32835SJeff Kirsher 		ehea_reset_cq_n1(pr->send_cq);
8969aa32835SJeff Kirsher 		rmb();
8979aa32835SJeff Kirsher 		cqe = ehea_poll_rq1(pr->qp, &wqe_index);
8989aa32835SJeff Kirsher 		cqe_skb = ehea_poll_cq(pr->send_cq);
8999aa32835SJeff Kirsher 
9009aa32835SJeff Kirsher 		if (!cqe && !cqe_skb)
9019aa32835SJeff Kirsher 			return rx;
9029aa32835SJeff Kirsher 
9039aa32835SJeff Kirsher 		if (!napi_reschedule(napi))
9049aa32835SJeff Kirsher 			return rx;
9059aa32835SJeff Kirsher 
9069aa32835SJeff Kirsher 		cqe_skb = ehea_proc_cqes(pr, EHEA_POLL_MAX_CQES);
9079aa32835SJeff Kirsher 		rx += ehea_proc_rwqes(dev, pr, budget - rx);
9089aa32835SJeff Kirsher 	}
9099aa32835SJeff Kirsher 
9109aa32835SJeff Kirsher 	return rx;
9119aa32835SJeff Kirsher }
9129aa32835SJeff Kirsher 
ehea_recv_irq_handler(int irq,void * param)9139aa32835SJeff Kirsher static irqreturn_t ehea_recv_irq_handler(int irq, void *param)
9149aa32835SJeff Kirsher {
9159aa32835SJeff Kirsher 	struct ehea_port_res *pr = param;
9169aa32835SJeff Kirsher 
9179aa32835SJeff Kirsher 	napi_schedule(&pr->napi);
9189aa32835SJeff Kirsher 
9199aa32835SJeff Kirsher 	return IRQ_HANDLED;
9209aa32835SJeff Kirsher }
9219aa32835SJeff Kirsher 
ehea_qp_aff_irq_handler(int irq,void * param)9229aa32835SJeff Kirsher static irqreturn_t ehea_qp_aff_irq_handler(int irq, void *param)
9239aa32835SJeff Kirsher {
9249aa32835SJeff Kirsher 	struct ehea_port *port = param;
9259aa32835SJeff Kirsher 	struct ehea_eqe *eqe;
9269aa32835SJeff Kirsher 	struct ehea_qp *qp;
9279aa32835SJeff Kirsher 	u32 qp_token;
9289aa32835SJeff Kirsher 	u64 resource_type, aer, aerr;
9299aa32835SJeff Kirsher 	int reset_port = 0;
9309aa32835SJeff Kirsher 
9319aa32835SJeff Kirsher 	eqe = ehea_poll_eq(port->qp_eq);
9329aa32835SJeff Kirsher 
9339aa32835SJeff Kirsher 	while (eqe) {
9349aa32835SJeff Kirsher 		qp_token = EHEA_BMASK_GET(EHEA_EQE_QP_TOKEN, eqe->entry);
9359aa32835SJeff Kirsher 		pr_err("QP aff_err: entry=0x%llx, token=0x%x\n",
9369aa32835SJeff Kirsher 		       eqe->entry, qp_token);
9379aa32835SJeff Kirsher 
9389aa32835SJeff Kirsher 		qp = port->port_res[qp_token].qp;
9399aa32835SJeff Kirsher 
9409aa32835SJeff Kirsher 		resource_type = ehea_error_data(port->adapter, qp->fw_handle,
9419aa32835SJeff Kirsher 						&aer, &aerr);
9429aa32835SJeff Kirsher 
9439aa32835SJeff Kirsher 		if (resource_type == EHEA_AER_RESTYPE_QP) {
9449aa32835SJeff Kirsher 			if ((aer & EHEA_AER_RESET_MASK) ||
9459aa32835SJeff Kirsher 			    (aerr & EHEA_AERR_RESET_MASK))
9469aa32835SJeff Kirsher 				 reset_port = 1;
9479aa32835SJeff Kirsher 		} else
9489aa32835SJeff Kirsher 			reset_port = 1;   /* Reset in case of CQ or EQ error */
9499aa32835SJeff Kirsher 
9509aa32835SJeff Kirsher 		eqe = ehea_poll_eq(port->qp_eq);
9519aa32835SJeff Kirsher 	}
9529aa32835SJeff Kirsher 
9539aa32835SJeff Kirsher 	if (reset_port) {
9549aa32835SJeff Kirsher 		pr_err("Resetting port\n");
9559aa32835SJeff Kirsher 		ehea_schedule_port_reset(port);
9569aa32835SJeff Kirsher 	}
9579aa32835SJeff Kirsher 
9589aa32835SJeff Kirsher 	return IRQ_HANDLED;
9599aa32835SJeff Kirsher }
9609aa32835SJeff Kirsher 
ehea_get_port(struct ehea_adapter * adapter,int logical_port)9619aa32835SJeff Kirsher static struct ehea_port *ehea_get_port(struct ehea_adapter *adapter,
9629aa32835SJeff Kirsher 				       int logical_port)
9639aa32835SJeff Kirsher {
9649aa32835SJeff Kirsher 	int i;
9659aa32835SJeff Kirsher 
9669aa32835SJeff Kirsher 	for (i = 0; i < EHEA_MAX_PORTS; i++)
9679aa32835SJeff Kirsher 		if (adapter->port[i])
9689aa32835SJeff Kirsher 			if (adapter->port[i]->logical_port_id == logical_port)
9699aa32835SJeff Kirsher 				return adapter->port[i];
9709aa32835SJeff Kirsher 	return NULL;
9719aa32835SJeff Kirsher }
9729aa32835SJeff Kirsher 
ehea_sense_port_attr(struct ehea_port * port)9739aa32835SJeff Kirsher int ehea_sense_port_attr(struct ehea_port *port)
9749aa32835SJeff Kirsher {
9759aa32835SJeff Kirsher 	int ret;
9769aa32835SJeff Kirsher 	u64 hret;
9779aa32835SJeff Kirsher 	struct hcp_ehea_port_cb0 *cb0;
9789aa32835SJeff Kirsher 
9799aa32835SJeff Kirsher 	/* may be called via ehea_neq_tasklet() */
9809aa32835SJeff Kirsher 	cb0 = (void *)get_zeroed_page(GFP_ATOMIC);
9819aa32835SJeff Kirsher 	if (!cb0) {
9829aa32835SJeff Kirsher 		pr_err("no mem for cb0\n");
9839aa32835SJeff Kirsher 		ret = -ENOMEM;
9849aa32835SJeff Kirsher 		goto out;
9859aa32835SJeff Kirsher 	}
9869aa32835SJeff Kirsher 
9879aa32835SJeff Kirsher 	hret = ehea_h_query_ehea_port(port->adapter->handle,
9889aa32835SJeff Kirsher 				      port->logical_port_id, H_PORT_CB0,
9899aa32835SJeff Kirsher 				      EHEA_BMASK_SET(H_PORT_CB0_ALL, 0xFFFF),
9909aa32835SJeff Kirsher 				      cb0);
9919aa32835SJeff Kirsher 	if (hret != H_SUCCESS) {
9929aa32835SJeff Kirsher 		ret = -EIO;
9939aa32835SJeff Kirsher 		goto out_free;
9949aa32835SJeff Kirsher 	}
9959aa32835SJeff Kirsher 
9969aa32835SJeff Kirsher 	/* MAC address */
9979aa32835SJeff Kirsher 	port->mac_addr = cb0->port_mac_addr << 16;
9989aa32835SJeff Kirsher 
9999aa32835SJeff Kirsher 	if (!is_valid_ether_addr((u8 *)&port->mac_addr)) {
10009aa32835SJeff Kirsher 		ret = -EADDRNOTAVAIL;
10019aa32835SJeff Kirsher 		goto out_free;
10029aa32835SJeff Kirsher 	}
10039aa32835SJeff Kirsher 
10049aa32835SJeff Kirsher 	/* Port speed */
10059aa32835SJeff Kirsher 	switch (cb0->port_speed) {
10069aa32835SJeff Kirsher 	case H_SPEED_10M_H:
10079aa32835SJeff Kirsher 		port->port_speed = EHEA_SPEED_10M;
10089aa32835SJeff Kirsher 		port->full_duplex = 0;
10099aa32835SJeff Kirsher 		break;
10109aa32835SJeff Kirsher 	case H_SPEED_10M_F:
10119aa32835SJeff Kirsher 		port->port_speed = EHEA_SPEED_10M;
10129aa32835SJeff Kirsher 		port->full_duplex = 1;
10139aa32835SJeff Kirsher 		break;
10149aa32835SJeff Kirsher 	case H_SPEED_100M_H:
10159aa32835SJeff Kirsher 		port->port_speed = EHEA_SPEED_100M;
10169aa32835SJeff Kirsher 		port->full_duplex = 0;
10179aa32835SJeff Kirsher 		break;
10189aa32835SJeff Kirsher 	case H_SPEED_100M_F:
10199aa32835SJeff Kirsher 		port->port_speed = EHEA_SPEED_100M;
10209aa32835SJeff Kirsher 		port->full_duplex = 1;
10219aa32835SJeff Kirsher 		break;
10229aa32835SJeff Kirsher 	case H_SPEED_1G_F:
10239aa32835SJeff Kirsher 		port->port_speed = EHEA_SPEED_1G;
10249aa32835SJeff Kirsher 		port->full_duplex = 1;
10259aa32835SJeff Kirsher 		break;
10269aa32835SJeff Kirsher 	case H_SPEED_10G_F:
10279aa32835SJeff Kirsher 		port->port_speed = EHEA_SPEED_10G;
10289aa32835SJeff Kirsher 		port->full_duplex = 1;
10299aa32835SJeff Kirsher 		break;
10309aa32835SJeff Kirsher 	default:
10319aa32835SJeff Kirsher 		port->port_speed = 0;
10329aa32835SJeff Kirsher 		port->full_duplex = 0;
10339aa32835SJeff Kirsher 		break;
10349aa32835SJeff Kirsher 	}
10359aa32835SJeff Kirsher 
10369aa32835SJeff Kirsher 	port->autoneg = 1;
10379aa32835SJeff Kirsher 	port->num_mcs = cb0->num_default_qps;
10389aa32835SJeff Kirsher 
10399aa32835SJeff Kirsher 	/* Number of default QPs */
10409aa32835SJeff Kirsher 	if (use_mcs)
10419aa32835SJeff Kirsher 		port->num_def_qps = cb0->num_default_qps;
10429aa32835SJeff Kirsher 	else
10439aa32835SJeff Kirsher 		port->num_def_qps = 1;
10449aa32835SJeff Kirsher 
10459aa32835SJeff Kirsher 	if (!port->num_def_qps) {
10469aa32835SJeff Kirsher 		ret = -EINVAL;
10479aa32835SJeff Kirsher 		goto out_free;
10489aa32835SJeff Kirsher 	}
10499aa32835SJeff Kirsher 
10509aa32835SJeff Kirsher 	ret = 0;
10519aa32835SJeff Kirsher out_free:
10529aa32835SJeff Kirsher 	if (ret || netif_msg_probe(port))
10539aa32835SJeff Kirsher 		ehea_dump(cb0, sizeof(*cb0), "ehea_sense_port_attr");
10549aa32835SJeff Kirsher 	free_page((unsigned long)cb0);
10559aa32835SJeff Kirsher out:
10569aa32835SJeff Kirsher 	return ret;
10579aa32835SJeff Kirsher }
10589aa32835SJeff Kirsher 
ehea_set_portspeed(struct ehea_port * port,u32 port_speed)10599aa32835SJeff Kirsher int ehea_set_portspeed(struct ehea_port *port, u32 port_speed)
10609aa32835SJeff Kirsher {
10619aa32835SJeff Kirsher 	struct hcp_ehea_port_cb4 *cb4;
10629aa32835SJeff Kirsher 	u64 hret;
10639aa32835SJeff Kirsher 	int ret = 0;
10649aa32835SJeff Kirsher 
10659aa32835SJeff Kirsher 	cb4 = (void *)get_zeroed_page(GFP_KERNEL);
10669aa32835SJeff Kirsher 	if (!cb4) {
10679aa32835SJeff Kirsher 		pr_err("no mem for cb4\n");
10689aa32835SJeff Kirsher 		ret = -ENOMEM;
10699aa32835SJeff Kirsher 		goto out;
10709aa32835SJeff Kirsher 	}
10719aa32835SJeff Kirsher 
10729aa32835SJeff Kirsher 	cb4->port_speed = port_speed;
10739aa32835SJeff Kirsher 
10749aa32835SJeff Kirsher 	netif_carrier_off(port->netdev);
10759aa32835SJeff Kirsher 
10769aa32835SJeff Kirsher 	hret = ehea_h_modify_ehea_port(port->adapter->handle,
10779aa32835SJeff Kirsher 				       port->logical_port_id,
10789aa32835SJeff Kirsher 				       H_PORT_CB4, H_PORT_CB4_SPEED, cb4);
10799aa32835SJeff Kirsher 	if (hret == H_SUCCESS) {
10809aa32835SJeff Kirsher 		port->autoneg = port_speed == EHEA_SPEED_AUTONEG ? 1 : 0;
10819aa32835SJeff Kirsher 
10829aa32835SJeff Kirsher 		hret = ehea_h_query_ehea_port(port->adapter->handle,
10839aa32835SJeff Kirsher 					      port->logical_port_id,
10849aa32835SJeff Kirsher 					      H_PORT_CB4, H_PORT_CB4_SPEED,
10859aa32835SJeff Kirsher 					      cb4);
10869aa32835SJeff Kirsher 		if (hret == H_SUCCESS) {
10879aa32835SJeff Kirsher 			switch (cb4->port_speed) {
10889aa32835SJeff Kirsher 			case H_SPEED_10M_H:
10899aa32835SJeff Kirsher 				port->port_speed = EHEA_SPEED_10M;
10909aa32835SJeff Kirsher 				port->full_duplex = 0;
10919aa32835SJeff Kirsher 				break;
10929aa32835SJeff Kirsher 			case H_SPEED_10M_F:
10939aa32835SJeff Kirsher 				port->port_speed = EHEA_SPEED_10M;
10949aa32835SJeff Kirsher 				port->full_duplex = 1;
10959aa32835SJeff Kirsher 				break;
10969aa32835SJeff Kirsher 			case H_SPEED_100M_H:
10979aa32835SJeff Kirsher 				port->port_speed = EHEA_SPEED_100M;
10989aa32835SJeff Kirsher 				port->full_duplex = 0;
10999aa32835SJeff Kirsher 				break;
11009aa32835SJeff Kirsher 			case H_SPEED_100M_F:
11019aa32835SJeff Kirsher 				port->port_speed = EHEA_SPEED_100M;
11029aa32835SJeff Kirsher 				port->full_duplex = 1;
11039aa32835SJeff Kirsher 				break;
11049aa32835SJeff Kirsher 			case H_SPEED_1G_F:
11059aa32835SJeff Kirsher 				port->port_speed = EHEA_SPEED_1G;
11069aa32835SJeff Kirsher 				port->full_duplex = 1;
11079aa32835SJeff Kirsher 				break;
11089aa32835SJeff Kirsher 			case H_SPEED_10G_F:
11099aa32835SJeff Kirsher 				port->port_speed = EHEA_SPEED_10G;
11109aa32835SJeff Kirsher 				port->full_duplex = 1;
11119aa32835SJeff Kirsher 				break;
11129aa32835SJeff Kirsher 			default:
11139aa32835SJeff Kirsher 				port->port_speed = 0;
11149aa32835SJeff Kirsher 				port->full_duplex = 0;
11159aa32835SJeff Kirsher 				break;
11169aa32835SJeff Kirsher 			}
11179aa32835SJeff Kirsher 		} else {
11189aa32835SJeff Kirsher 			pr_err("Failed sensing port speed\n");
11199aa32835SJeff Kirsher 			ret = -EIO;
11209aa32835SJeff Kirsher 		}
11219aa32835SJeff Kirsher 	} else {
11229aa32835SJeff Kirsher 		if (hret == H_AUTHORITY) {
11239aa32835SJeff Kirsher 			pr_info("Hypervisor denied setting port speed\n");
11249aa32835SJeff Kirsher 			ret = -EPERM;
11259aa32835SJeff Kirsher 		} else {
11269aa32835SJeff Kirsher 			ret = -EIO;
11279aa32835SJeff Kirsher 			pr_err("Failed setting port speed\n");
11289aa32835SJeff Kirsher 		}
11299aa32835SJeff Kirsher 	}
11309aa32835SJeff Kirsher 	if (!prop_carrier_state || (port->phy_link == EHEA_PHY_LINK_UP))
11319aa32835SJeff Kirsher 		netif_carrier_on(port->netdev);
11329aa32835SJeff Kirsher 
11339aa32835SJeff Kirsher 	free_page((unsigned long)cb4);
11349aa32835SJeff Kirsher out:
11359aa32835SJeff Kirsher 	return ret;
11369aa32835SJeff Kirsher }
11379aa32835SJeff Kirsher 
ehea_parse_eqe(struct ehea_adapter * adapter,u64 eqe)11389aa32835SJeff Kirsher static void ehea_parse_eqe(struct ehea_adapter *adapter, u64 eqe)
11399aa32835SJeff Kirsher {
11409aa32835SJeff Kirsher 	int ret;
11419aa32835SJeff Kirsher 	u8 ec;
11429aa32835SJeff Kirsher 	u8 portnum;
11439aa32835SJeff Kirsher 	struct ehea_port *port;
11449aa32835SJeff Kirsher 	struct net_device *dev;
11459aa32835SJeff Kirsher 
11469aa32835SJeff Kirsher 	ec = EHEA_BMASK_GET(NEQE_EVENT_CODE, eqe);
11479aa32835SJeff Kirsher 	portnum = EHEA_BMASK_GET(NEQE_PORTNUM, eqe);
11489aa32835SJeff Kirsher 	port = ehea_get_port(adapter, portnum);
11491740c29aSxypron.glpk@gmx.de 	if (!port) {
11501740c29aSxypron.glpk@gmx.de 		netdev_err(NULL, "unknown portnum %x\n", portnum);
11511740c29aSxypron.glpk@gmx.de 		return;
11521740c29aSxypron.glpk@gmx.de 	}
11539aa32835SJeff Kirsher 	dev = port->netdev;
11549aa32835SJeff Kirsher 
11559aa32835SJeff Kirsher 	switch (ec) {
11569aa32835SJeff Kirsher 	case EHEA_EC_PORTSTATE_CHG:	/* port state change */
11579aa32835SJeff Kirsher 
11589aa32835SJeff Kirsher 		if (EHEA_BMASK_GET(NEQE_PORT_UP, eqe)) {
11599aa32835SJeff Kirsher 			if (!netif_carrier_ok(dev)) {
11609aa32835SJeff Kirsher 				ret = ehea_sense_port_attr(port);
11619aa32835SJeff Kirsher 				if (ret) {
11629aa32835SJeff Kirsher 					netdev_err(dev, "failed resensing port attributes\n");
11639aa32835SJeff Kirsher 					break;
11649aa32835SJeff Kirsher 				}
11659aa32835SJeff Kirsher 
11669aa32835SJeff Kirsher 				netif_info(port, link, dev,
11679aa32835SJeff Kirsher 					   "Logical port up: %dMbps %s Duplex\n",
11689aa32835SJeff Kirsher 					   port->port_speed,
11699aa32835SJeff Kirsher 					   port->full_duplex == 1 ?
11709aa32835SJeff Kirsher 					   "Full" : "Half");
11719aa32835SJeff Kirsher 
11729aa32835SJeff Kirsher 				netif_carrier_on(dev);
11739aa32835SJeff Kirsher 				netif_wake_queue(dev);
11749aa32835SJeff Kirsher 			}
11759aa32835SJeff Kirsher 		} else
11769aa32835SJeff Kirsher 			if (netif_carrier_ok(dev)) {
11779aa32835SJeff Kirsher 				netif_info(port, link, dev,
11789aa32835SJeff Kirsher 					   "Logical port down\n");
11799aa32835SJeff Kirsher 				netif_carrier_off(dev);
1180b9564468SAnton Blanchard 				netif_tx_disable(dev);
11819aa32835SJeff Kirsher 			}
11829aa32835SJeff Kirsher 
11839aa32835SJeff Kirsher 		if (EHEA_BMASK_GET(NEQE_EXTSWITCH_PORT_UP, eqe)) {
11849aa32835SJeff Kirsher 			port->phy_link = EHEA_PHY_LINK_UP;
11859aa32835SJeff Kirsher 			netif_info(port, link, dev,
11869aa32835SJeff Kirsher 				   "Physical port up\n");
11879aa32835SJeff Kirsher 			if (prop_carrier_state)
11889aa32835SJeff Kirsher 				netif_carrier_on(dev);
11899aa32835SJeff Kirsher 		} else {
11909aa32835SJeff Kirsher 			port->phy_link = EHEA_PHY_LINK_DOWN;
11919aa32835SJeff Kirsher 			netif_info(port, link, dev,
11929aa32835SJeff Kirsher 				   "Physical port down\n");
11939aa32835SJeff Kirsher 			if (prop_carrier_state)
11949aa32835SJeff Kirsher 				netif_carrier_off(dev);
11959aa32835SJeff Kirsher 		}
11969aa32835SJeff Kirsher 
11979aa32835SJeff Kirsher 		if (EHEA_BMASK_GET(NEQE_EXTSWITCH_PRIMARY, eqe))
11989aa32835SJeff Kirsher 			netdev_info(dev,
11999aa32835SJeff Kirsher 				    "External switch port is primary port\n");
12009aa32835SJeff Kirsher 		else
12019aa32835SJeff Kirsher 			netdev_info(dev,
12029aa32835SJeff Kirsher 				    "External switch port is backup port\n");
12039aa32835SJeff Kirsher 
12049aa32835SJeff Kirsher 		break;
12059aa32835SJeff Kirsher 	case EHEA_EC_ADAPTER_MALFUNC:
12069aa32835SJeff Kirsher 		netdev_err(dev, "Adapter malfunction\n");
12079aa32835SJeff Kirsher 		break;
12089aa32835SJeff Kirsher 	case EHEA_EC_PORT_MALFUNC:
12099aa32835SJeff Kirsher 		netdev_info(dev, "Port malfunction\n");
12109aa32835SJeff Kirsher 		netif_carrier_off(dev);
1211b9564468SAnton Blanchard 		netif_tx_disable(dev);
12129aa32835SJeff Kirsher 		break;
12139aa32835SJeff Kirsher 	default:
12149aa32835SJeff Kirsher 		netdev_err(dev, "unknown event code %x, eqe=0x%llX\n", ec, eqe);
12159aa32835SJeff Kirsher 		break;
12169aa32835SJeff Kirsher 	}
12179aa32835SJeff Kirsher }
12189aa32835SJeff Kirsher 
ehea_neq_tasklet(struct tasklet_struct * t)12199da7cfc3SAllen Pais static void ehea_neq_tasklet(struct tasklet_struct *t)
12209aa32835SJeff Kirsher {
12219da7cfc3SAllen Pais 	struct ehea_adapter *adapter = from_tasklet(adapter, t, neq_tasklet);
12229aa32835SJeff Kirsher 	struct ehea_eqe *eqe;
12239aa32835SJeff Kirsher 	u64 event_mask;
12249aa32835SJeff Kirsher 
12259aa32835SJeff Kirsher 	eqe = ehea_poll_eq(adapter->neq);
12269aa32835SJeff Kirsher 	pr_debug("eqe=%p\n", eqe);
12279aa32835SJeff Kirsher 
12289aa32835SJeff Kirsher 	while (eqe) {
12299aa32835SJeff Kirsher 		pr_debug("*eqe=%lx\n", (unsigned long) eqe->entry);
12309aa32835SJeff Kirsher 		ehea_parse_eqe(adapter, eqe->entry);
12319aa32835SJeff Kirsher 		eqe = ehea_poll_eq(adapter->neq);
12329aa32835SJeff Kirsher 		pr_debug("next eqe=%p\n", eqe);
12339aa32835SJeff Kirsher 	}
12349aa32835SJeff Kirsher 
12359aa32835SJeff Kirsher 	event_mask = EHEA_BMASK_SET(NELR_PORTSTATE_CHG, 1)
12369aa32835SJeff Kirsher 		   | EHEA_BMASK_SET(NELR_ADAPTER_MALFUNC, 1)
12379aa32835SJeff Kirsher 		   | EHEA_BMASK_SET(NELR_PORT_MALFUNC, 1);
12389aa32835SJeff Kirsher 
12399aa32835SJeff Kirsher 	ehea_h_reset_events(adapter->handle,
12409aa32835SJeff Kirsher 			    adapter->neq->fw_handle, event_mask);
12419aa32835SJeff Kirsher }
12429aa32835SJeff Kirsher 
ehea_interrupt_neq(int irq,void * param)12439aa32835SJeff Kirsher static irqreturn_t ehea_interrupt_neq(int irq, void *param)
12449aa32835SJeff Kirsher {
12459aa32835SJeff Kirsher 	struct ehea_adapter *adapter = param;
12469aa32835SJeff Kirsher 	tasklet_hi_schedule(&adapter->neq_tasklet);
12479aa32835SJeff Kirsher 	return IRQ_HANDLED;
12489aa32835SJeff Kirsher }
12499aa32835SJeff Kirsher 
12509aa32835SJeff Kirsher 
ehea_fill_port_res(struct ehea_port_res * pr)12519aa32835SJeff Kirsher static int ehea_fill_port_res(struct ehea_port_res *pr)
12529aa32835SJeff Kirsher {
12539aa32835SJeff Kirsher 	int ret;
12549aa32835SJeff Kirsher 	struct ehea_qp_init_attr *init_attr = &pr->qp->init_attr;
12559aa32835SJeff Kirsher 
12569aa32835SJeff Kirsher 	ehea_init_fill_rq1(pr, pr->rq1_skba.len);
12579aa32835SJeff Kirsher 
12589aa32835SJeff Kirsher 	ret = ehea_refill_rq2(pr, init_attr->act_nr_rwqes_rq2 - 1);
12599aa32835SJeff Kirsher 
12609aa32835SJeff Kirsher 	ret |= ehea_refill_rq3(pr, init_attr->act_nr_rwqes_rq3 - 1);
12619aa32835SJeff Kirsher 
12629aa32835SJeff Kirsher 	return ret;
12639aa32835SJeff Kirsher }
12649aa32835SJeff Kirsher 
ehea_reg_interrupts(struct net_device * dev)12659aa32835SJeff Kirsher static int ehea_reg_interrupts(struct net_device *dev)
12669aa32835SJeff Kirsher {
12679aa32835SJeff Kirsher 	struct ehea_port *port = netdev_priv(dev);
12689aa32835SJeff Kirsher 	struct ehea_port_res *pr;
12699aa32835SJeff Kirsher 	int i, ret;
12709aa32835SJeff Kirsher 
12719aa32835SJeff Kirsher 
12729aa32835SJeff Kirsher 	snprintf(port->int_aff_name, EHEA_IRQ_NAME_SIZE - 1, "%s-aff",
12739aa32835SJeff Kirsher 		 dev->name);
12749aa32835SJeff Kirsher 
12759aa32835SJeff Kirsher 	ret = ibmebus_request_irq(port->qp_eq->attr.ist1,
12769aa32835SJeff Kirsher 				  ehea_qp_aff_irq_handler,
127746c915f8SMichael Opdenacker 				  0, port->int_aff_name, port);
12789aa32835SJeff Kirsher 	if (ret) {
12799aa32835SJeff Kirsher 		netdev_err(dev, "failed registering irq for qp_aff_irq_handler:ist=%X\n",
12809aa32835SJeff Kirsher 			   port->qp_eq->attr.ist1);
12819aa32835SJeff Kirsher 		goto out_free_qpeq;
12829aa32835SJeff Kirsher 	}
12839aa32835SJeff Kirsher 
12849aa32835SJeff Kirsher 	netif_info(port, ifup, dev,
12859aa32835SJeff Kirsher 		   "irq_handle 0x%X for function qp_aff_irq_handler registered\n",
12869aa32835SJeff Kirsher 		   port->qp_eq->attr.ist1);
12879aa32835SJeff Kirsher 
12889aa32835SJeff Kirsher 
1289723f28e4SAnton Blanchard 	for (i = 0; i < port->num_def_qps; i++) {
12909aa32835SJeff Kirsher 		pr = &port->port_res[i];
12919aa32835SJeff Kirsher 		snprintf(pr->int_send_name, EHEA_IRQ_NAME_SIZE - 1,
12929aa32835SJeff Kirsher 			 "%s-queue%d", dev->name, i);
12939aa32835SJeff Kirsher 		ret = ibmebus_request_irq(pr->eq->attr.ist1,
12949aa32835SJeff Kirsher 					  ehea_recv_irq_handler,
129546c915f8SMichael Opdenacker 					  0, pr->int_send_name, pr);
12969aa32835SJeff Kirsher 		if (ret) {
12979aa32835SJeff Kirsher 			netdev_err(dev, "failed registering irq for ehea_queue port_res_nr:%d, ist=%X\n",
12989aa32835SJeff Kirsher 				   i, pr->eq->attr.ist1);
12999aa32835SJeff Kirsher 			goto out_free_req;
13009aa32835SJeff Kirsher 		}
13019aa32835SJeff Kirsher 		netif_info(port, ifup, dev,
13029aa32835SJeff Kirsher 			   "irq_handle 0x%X for function ehea_queue_int %d registered\n",
13039aa32835SJeff Kirsher 			   pr->eq->attr.ist1, i);
13049aa32835SJeff Kirsher 	}
13059aa32835SJeff Kirsher out:
13069aa32835SJeff Kirsher 	return ret;
13079aa32835SJeff Kirsher 
13089aa32835SJeff Kirsher 
13099aa32835SJeff Kirsher out_free_req:
13109aa32835SJeff Kirsher 	while (--i >= 0) {
13119aa32835SJeff Kirsher 		u32 ist = port->port_res[i].eq->attr.ist1;
13129aa32835SJeff Kirsher 		ibmebus_free_irq(ist, &port->port_res[i]);
13139aa32835SJeff Kirsher 	}
13149aa32835SJeff Kirsher 
13159aa32835SJeff Kirsher out_free_qpeq:
13169aa32835SJeff Kirsher 	ibmebus_free_irq(port->qp_eq->attr.ist1, port);
13179aa32835SJeff Kirsher 	i = port->num_def_qps;
13189aa32835SJeff Kirsher 
13199aa32835SJeff Kirsher 	goto out;
13209aa32835SJeff Kirsher 
13219aa32835SJeff Kirsher }
13229aa32835SJeff Kirsher 
ehea_free_interrupts(struct net_device * dev)13239aa32835SJeff Kirsher static void ehea_free_interrupts(struct net_device *dev)
13249aa32835SJeff Kirsher {
13259aa32835SJeff Kirsher 	struct ehea_port *port = netdev_priv(dev);
13269aa32835SJeff Kirsher 	struct ehea_port_res *pr;
13279aa32835SJeff Kirsher 	int i;
13289aa32835SJeff Kirsher 
13299aa32835SJeff Kirsher 	/* send */
13309aa32835SJeff Kirsher 
1331723f28e4SAnton Blanchard 	for (i = 0; i < port->num_def_qps; i++) {
13329aa32835SJeff Kirsher 		pr = &port->port_res[i];
13339aa32835SJeff Kirsher 		ibmebus_free_irq(pr->eq->attr.ist1, pr);
13349aa32835SJeff Kirsher 		netif_info(port, intr, dev,
13359aa32835SJeff Kirsher 			   "free send irq for res %d with handle 0x%X\n",
13369aa32835SJeff Kirsher 			   i, pr->eq->attr.ist1);
13379aa32835SJeff Kirsher 	}
13389aa32835SJeff Kirsher 
13399aa32835SJeff Kirsher 	/* associated events */
13409aa32835SJeff Kirsher 	ibmebus_free_irq(port->qp_eq->attr.ist1, port);
13419aa32835SJeff Kirsher 	netif_info(port, intr, dev,
13429aa32835SJeff Kirsher 		   "associated event interrupt for handle 0x%X freed\n",
13439aa32835SJeff Kirsher 		   port->qp_eq->attr.ist1);
13449aa32835SJeff Kirsher }
13459aa32835SJeff Kirsher 
ehea_configure_port(struct ehea_port * port)13469aa32835SJeff Kirsher static int ehea_configure_port(struct ehea_port *port)
13479aa32835SJeff Kirsher {
13489aa32835SJeff Kirsher 	int ret, i;
13499aa32835SJeff Kirsher 	u64 hret, mask;
13509aa32835SJeff Kirsher 	struct hcp_ehea_port_cb0 *cb0;
13519aa32835SJeff Kirsher 
13529aa32835SJeff Kirsher 	ret = -ENOMEM;
13539aa32835SJeff Kirsher 	cb0 = (void *)get_zeroed_page(GFP_KERNEL);
13549aa32835SJeff Kirsher 	if (!cb0)
13559aa32835SJeff Kirsher 		goto out;
13569aa32835SJeff Kirsher 
13579aa32835SJeff Kirsher 	cb0->port_rc = EHEA_BMASK_SET(PXLY_RC_VALID, 1)
13589aa32835SJeff Kirsher 		     | EHEA_BMASK_SET(PXLY_RC_IP_CHKSUM, 1)
13599aa32835SJeff Kirsher 		     | EHEA_BMASK_SET(PXLY_RC_TCP_UDP_CHKSUM, 1)
13609aa32835SJeff Kirsher 		     | EHEA_BMASK_SET(PXLY_RC_VLAN_XTRACT, 1)
13619aa32835SJeff Kirsher 		     | EHEA_BMASK_SET(PXLY_RC_VLAN_TAG_FILTER,
13629aa32835SJeff Kirsher 				      PXLY_RC_VLAN_FILTER)
13639aa32835SJeff Kirsher 		     | EHEA_BMASK_SET(PXLY_RC_JUMBO_FRAME, 1);
13649aa32835SJeff Kirsher 
13659aa32835SJeff Kirsher 	for (i = 0; i < port->num_mcs; i++)
13669aa32835SJeff Kirsher 		if (use_mcs)
13679aa32835SJeff Kirsher 			cb0->default_qpn_arr[i] =
13689aa32835SJeff Kirsher 				port->port_res[i].qp->init_attr.qp_nr;
13699aa32835SJeff Kirsher 		else
13709aa32835SJeff Kirsher 			cb0->default_qpn_arr[i] =
13719aa32835SJeff Kirsher 				port->port_res[0].qp->init_attr.qp_nr;
13729aa32835SJeff Kirsher 
13739aa32835SJeff Kirsher 	if (netif_msg_ifup(port))
13749aa32835SJeff Kirsher 		ehea_dump(cb0, sizeof(*cb0), "ehea_configure_port");
13759aa32835SJeff Kirsher 
13769aa32835SJeff Kirsher 	mask = EHEA_BMASK_SET(H_PORT_CB0_PRC, 1)
13779aa32835SJeff Kirsher 	     | EHEA_BMASK_SET(H_PORT_CB0_DEFQPNARRAY, 1);
13789aa32835SJeff Kirsher 
13799aa32835SJeff Kirsher 	hret = ehea_h_modify_ehea_port(port->adapter->handle,
13809aa32835SJeff Kirsher 				       port->logical_port_id,
13819aa32835SJeff Kirsher 				       H_PORT_CB0, mask, cb0);
13829aa32835SJeff Kirsher 	ret = -EIO;
13839aa32835SJeff Kirsher 	if (hret != H_SUCCESS)
13849aa32835SJeff Kirsher 		goto out_free;
13859aa32835SJeff Kirsher 
13869aa32835SJeff Kirsher 	ret = 0;
13879aa32835SJeff Kirsher 
13889aa32835SJeff Kirsher out_free:
13899aa32835SJeff Kirsher 	free_page((unsigned long)cb0);
13909aa32835SJeff Kirsher out:
13919aa32835SJeff Kirsher 	return ret;
13929aa32835SJeff Kirsher }
13939aa32835SJeff Kirsher 
ehea_gen_smrs(struct ehea_port_res * pr)13941886e5d2SThadeu Lima de Souza Cascardo static int ehea_gen_smrs(struct ehea_port_res *pr)
13959aa32835SJeff Kirsher {
13969aa32835SJeff Kirsher 	int ret;
13979aa32835SJeff Kirsher 	struct ehea_adapter *adapter = pr->port->adapter;
13989aa32835SJeff Kirsher 
13999aa32835SJeff Kirsher 	ret = ehea_gen_smr(adapter, &adapter->mr, &pr->send_mr);
14009aa32835SJeff Kirsher 	if (ret)
14019aa32835SJeff Kirsher 		goto out;
14029aa32835SJeff Kirsher 
14039aa32835SJeff Kirsher 	ret = ehea_gen_smr(adapter, &adapter->mr, &pr->recv_mr);
14049aa32835SJeff Kirsher 	if (ret)
14059aa32835SJeff Kirsher 		goto out_free;
14069aa32835SJeff Kirsher 
14079aa32835SJeff Kirsher 	return 0;
14089aa32835SJeff Kirsher 
14099aa32835SJeff Kirsher out_free:
14109aa32835SJeff Kirsher 	ehea_rem_mr(&pr->send_mr);
14119aa32835SJeff Kirsher out:
14129aa32835SJeff Kirsher 	pr_err("Generating SMRS failed\n");
14139aa32835SJeff Kirsher 	return -EIO;
14149aa32835SJeff Kirsher }
14159aa32835SJeff Kirsher 
ehea_rem_smrs(struct ehea_port_res * pr)14161886e5d2SThadeu Lima de Souza Cascardo static int ehea_rem_smrs(struct ehea_port_res *pr)
14179aa32835SJeff Kirsher {
14189aa32835SJeff Kirsher 	if ((ehea_rem_mr(&pr->send_mr)) ||
14199aa32835SJeff Kirsher 	    (ehea_rem_mr(&pr->recv_mr)))
14209aa32835SJeff Kirsher 		return -EIO;
14219aa32835SJeff Kirsher 	else
14229aa32835SJeff Kirsher 		return 0;
14239aa32835SJeff Kirsher }
14249aa32835SJeff Kirsher 
ehea_init_q_skba(struct ehea_q_skb_arr * q_skba,int max_q_entries)14259aa32835SJeff Kirsher static int ehea_init_q_skba(struct ehea_q_skb_arr *q_skba, int max_q_entries)
14269aa32835SJeff Kirsher {
14279aa32835SJeff Kirsher 	int arr_size = sizeof(void *) * max_q_entries;
14289aa32835SJeff Kirsher 
14299aa32835SJeff Kirsher 	q_skba->arr = vzalloc(arr_size);
14309aa32835SJeff Kirsher 	if (!q_skba->arr)
14319aa32835SJeff Kirsher 		return -ENOMEM;
14329aa32835SJeff Kirsher 
14339aa32835SJeff Kirsher 	q_skba->len = max_q_entries;
14349aa32835SJeff Kirsher 	q_skba->index = 0;
14359aa32835SJeff Kirsher 	q_skba->os_skbs = 0;
14369aa32835SJeff Kirsher 
14379aa32835SJeff Kirsher 	return 0;
14389aa32835SJeff Kirsher }
14399aa32835SJeff Kirsher 
ehea_init_port_res(struct ehea_port * port,struct ehea_port_res * pr,struct port_res_cfg * pr_cfg,int queue_token)14409aa32835SJeff Kirsher static int ehea_init_port_res(struct ehea_port *port, struct ehea_port_res *pr,
14419aa32835SJeff Kirsher 			      struct port_res_cfg *pr_cfg, int queue_token)
14429aa32835SJeff Kirsher {
14439aa32835SJeff Kirsher 	struct ehea_adapter *adapter = port->adapter;
14449aa32835SJeff Kirsher 	enum ehea_eq_type eq_type = EHEA_EQ;
14459aa32835SJeff Kirsher 	struct ehea_qp_init_attr *init_attr = NULL;
14469aa32835SJeff Kirsher 	int ret = -EIO;
14479aa32835SJeff Kirsher 	u64 tx_bytes, rx_bytes, tx_packets, rx_packets;
14489aa32835SJeff Kirsher 
14499aa32835SJeff Kirsher 	tx_bytes = pr->tx_bytes;
14509aa32835SJeff Kirsher 	tx_packets = pr->tx_packets;
14519aa32835SJeff Kirsher 	rx_bytes = pr->rx_bytes;
14529aa32835SJeff Kirsher 	rx_packets = pr->rx_packets;
14539aa32835SJeff Kirsher 
14549aa32835SJeff Kirsher 	memset(pr, 0, sizeof(struct ehea_port_res));
14559aa32835SJeff Kirsher 
1456c8f19128SYueHaibing 	pr->tx_bytes = tx_bytes;
14579aa32835SJeff Kirsher 	pr->tx_packets = tx_packets;
14589aa32835SJeff Kirsher 	pr->rx_bytes = rx_bytes;
14599aa32835SJeff Kirsher 	pr->rx_packets = rx_packets;
14609aa32835SJeff Kirsher 
14619aa32835SJeff Kirsher 	pr->port = port;
14629aa32835SJeff Kirsher 
14639aa32835SJeff Kirsher 	pr->eq = ehea_create_eq(adapter, eq_type, EHEA_MAX_ENTRIES_EQ, 0);
14649aa32835SJeff Kirsher 	if (!pr->eq) {
14659aa32835SJeff Kirsher 		pr_err("create_eq failed (eq)\n");
14669aa32835SJeff Kirsher 		goto out_free;
14679aa32835SJeff Kirsher 	}
14689aa32835SJeff Kirsher 
14699aa32835SJeff Kirsher 	pr->recv_cq = ehea_create_cq(adapter, pr_cfg->max_entries_rcq,
14709aa32835SJeff Kirsher 				     pr->eq->fw_handle,
14719aa32835SJeff Kirsher 				     port->logical_port_id);
14729aa32835SJeff Kirsher 	if (!pr->recv_cq) {
14739aa32835SJeff Kirsher 		pr_err("create_cq failed (cq_recv)\n");
14749aa32835SJeff Kirsher 		goto out_free;
14759aa32835SJeff Kirsher 	}
14769aa32835SJeff Kirsher 
14779aa32835SJeff Kirsher 	pr->send_cq = ehea_create_cq(adapter, pr_cfg->max_entries_scq,
14789aa32835SJeff Kirsher 				     pr->eq->fw_handle,
14799aa32835SJeff Kirsher 				     port->logical_port_id);
14809aa32835SJeff Kirsher 	if (!pr->send_cq) {
14819aa32835SJeff Kirsher 		pr_err("create_cq failed (cq_send)\n");
14829aa32835SJeff Kirsher 		goto out_free;
14839aa32835SJeff Kirsher 	}
14849aa32835SJeff Kirsher 
14859aa32835SJeff Kirsher 	if (netif_msg_ifup(port))
14869aa32835SJeff Kirsher 		pr_info("Send CQ: act_nr_cqes=%d, Recv CQ: act_nr_cqes=%d\n",
14879aa32835SJeff Kirsher 			pr->send_cq->attr.act_nr_of_cqes,
14889aa32835SJeff Kirsher 			pr->recv_cq->attr.act_nr_of_cqes);
14899aa32835SJeff Kirsher 
14909aa32835SJeff Kirsher 	init_attr = kzalloc(sizeof(*init_attr), GFP_KERNEL);
14919aa32835SJeff Kirsher 	if (!init_attr) {
14929aa32835SJeff Kirsher 		ret = -ENOMEM;
14939aa32835SJeff Kirsher 		pr_err("no mem for ehea_qp_init_attr\n");
14949aa32835SJeff Kirsher 		goto out_free;
14959aa32835SJeff Kirsher 	}
14969aa32835SJeff Kirsher 
14979aa32835SJeff Kirsher 	init_attr->low_lat_rq1 = 1;
14989aa32835SJeff Kirsher 	init_attr->signalingtype = 1;	/* generate CQE if specified in WQE */
14999aa32835SJeff Kirsher 	init_attr->rq_count = 3;
15009aa32835SJeff Kirsher 	init_attr->qp_token = queue_token;
15019aa32835SJeff Kirsher 	init_attr->max_nr_send_wqes = pr_cfg->max_entries_sq;
15029aa32835SJeff Kirsher 	init_attr->max_nr_rwqes_rq1 = pr_cfg->max_entries_rq1;
15039aa32835SJeff Kirsher 	init_attr->max_nr_rwqes_rq2 = pr_cfg->max_entries_rq2;
15049aa32835SJeff Kirsher 	init_attr->max_nr_rwqes_rq3 = pr_cfg->max_entries_rq3;
15059aa32835SJeff Kirsher 	init_attr->wqe_size_enc_sq = EHEA_SG_SQ;
15069aa32835SJeff Kirsher 	init_attr->wqe_size_enc_rq1 = EHEA_SG_RQ1;
15079aa32835SJeff Kirsher 	init_attr->wqe_size_enc_rq2 = EHEA_SG_RQ2;
15089aa32835SJeff Kirsher 	init_attr->wqe_size_enc_rq3 = EHEA_SG_RQ3;
15099aa32835SJeff Kirsher 	init_attr->rq2_threshold = EHEA_RQ2_THRESHOLD;
15109aa32835SJeff Kirsher 	init_attr->rq3_threshold = EHEA_RQ3_THRESHOLD;
15119aa32835SJeff Kirsher 	init_attr->port_nr = port->logical_port_id;
15129aa32835SJeff Kirsher 	init_attr->send_cq_handle = pr->send_cq->fw_handle;
15139aa32835SJeff Kirsher 	init_attr->recv_cq_handle = pr->recv_cq->fw_handle;
15149aa32835SJeff Kirsher 	init_attr->aff_eq_handle = port->qp_eq->fw_handle;
15159aa32835SJeff Kirsher 
15169aa32835SJeff Kirsher 	pr->qp = ehea_create_qp(adapter, adapter->pd, init_attr);
15179aa32835SJeff Kirsher 	if (!pr->qp) {
15189aa32835SJeff Kirsher 		pr_err("create_qp failed\n");
15199aa32835SJeff Kirsher 		ret = -EIO;
15209aa32835SJeff Kirsher 		goto out_free;
15219aa32835SJeff Kirsher 	}
15229aa32835SJeff Kirsher 
15239aa32835SJeff Kirsher 	if (netif_msg_ifup(port))
15249aa32835SJeff Kirsher 		pr_info("QP: qp_nr=%d\n act_nr_snd_wqe=%d\n nr_rwqe_rq1=%d\n nr_rwqe_rq2=%d\n nr_rwqe_rq3=%d\n",
15259aa32835SJeff Kirsher 			init_attr->qp_nr,
15269aa32835SJeff Kirsher 			init_attr->act_nr_send_wqes,
15279aa32835SJeff Kirsher 			init_attr->act_nr_rwqes_rq1,
15289aa32835SJeff Kirsher 			init_attr->act_nr_rwqes_rq2,
15299aa32835SJeff Kirsher 			init_attr->act_nr_rwqes_rq3);
15309aa32835SJeff Kirsher 
15319aa32835SJeff Kirsher 	pr->sq_skba_size = init_attr->act_nr_send_wqes + 1;
15329aa32835SJeff Kirsher 
15339aa32835SJeff Kirsher 	ret = ehea_init_q_skba(&pr->sq_skba, pr->sq_skba_size);
15349aa32835SJeff Kirsher 	ret |= ehea_init_q_skba(&pr->rq1_skba, init_attr->act_nr_rwqes_rq1 + 1);
15359aa32835SJeff Kirsher 	ret |= ehea_init_q_skba(&pr->rq2_skba, init_attr->act_nr_rwqes_rq2 + 1);
15369aa32835SJeff Kirsher 	ret |= ehea_init_q_skba(&pr->rq3_skba, init_attr->act_nr_rwqes_rq3 + 1);
15379aa32835SJeff Kirsher 	if (ret)
15389aa32835SJeff Kirsher 		goto out_free;
15399aa32835SJeff Kirsher 
15409aa32835SJeff Kirsher 	pr->swqe_refill_th = init_attr->act_nr_send_wqes / 10;
15419aa32835SJeff Kirsher 	if (ehea_gen_smrs(pr) != 0) {
15429aa32835SJeff Kirsher 		ret = -EIO;
15439aa32835SJeff Kirsher 		goto out_free;
15449aa32835SJeff Kirsher 	}
15459aa32835SJeff Kirsher 
15469aa32835SJeff Kirsher 	atomic_set(&pr->swqe_avail, init_attr->act_nr_send_wqes - 1);
15479aa32835SJeff Kirsher 
15489aa32835SJeff Kirsher 	kfree(init_attr);
15499aa32835SJeff Kirsher 
1550b48b89f9SJakub Kicinski 	netif_napi_add(pr->port->netdev, &pr->napi, ehea_poll);
15519aa32835SJeff Kirsher 
15529aa32835SJeff Kirsher 	ret = 0;
15539aa32835SJeff Kirsher 	goto out;
15549aa32835SJeff Kirsher 
15559aa32835SJeff Kirsher out_free:
15569aa32835SJeff Kirsher 	kfree(init_attr);
15579aa32835SJeff Kirsher 	vfree(pr->sq_skba.arr);
15589aa32835SJeff Kirsher 	vfree(pr->rq1_skba.arr);
15599aa32835SJeff Kirsher 	vfree(pr->rq2_skba.arr);
15609aa32835SJeff Kirsher 	vfree(pr->rq3_skba.arr);
15619aa32835SJeff Kirsher 	ehea_destroy_qp(pr->qp);
15629aa32835SJeff Kirsher 	ehea_destroy_cq(pr->send_cq);
15639aa32835SJeff Kirsher 	ehea_destroy_cq(pr->recv_cq);
15649aa32835SJeff Kirsher 	ehea_destroy_eq(pr->eq);
15659aa32835SJeff Kirsher out:
15669aa32835SJeff Kirsher 	return ret;
15679aa32835SJeff Kirsher }
15689aa32835SJeff Kirsher 
ehea_clean_portres(struct ehea_port * port,struct ehea_port_res * pr)15699aa32835SJeff Kirsher static int ehea_clean_portres(struct ehea_port *port, struct ehea_port_res *pr)
15709aa32835SJeff Kirsher {
15719aa32835SJeff Kirsher 	int ret, i;
15729aa32835SJeff Kirsher 
15739aa32835SJeff Kirsher 	if (pr->qp)
15749aa32835SJeff Kirsher 		netif_napi_del(&pr->napi);
15759aa32835SJeff Kirsher 
15769aa32835SJeff Kirsher 	ret = ehea_destroy_qp(pr->qp);
15779aa32835SJeff Kirsher 
15789aa32835SJeff Kirsher 	if (!ret) {
15799aa32835SJeff Kirsher 		ehea_destroy_cq(pr->send_cq);
15809aa32835SJeff Kirsher 		ehea_destroy_cq(pr->recv_cq);
15819aa32835SJeff Kirsher 		ehea_destroy_eq(pr->eq);
15829aa32835SJeff Kirsher 
15839aa32835SJeff Kirsher 		for (i = 0; i < pr->rq1_skba.len; i++)
15849aa32835SJeff Kirsher 			dev_kfree_skb(pr->rq1_skba.arr[i]);
15859aa32835SJeff Kirsher 
15869aa32835SJeff Kirsher 		for (i = 0; i < pr->rq2_skba.len; i++)
15879aa32835SJeff Kirsher 			dev_kfree_skb(pr->rq2_skba.arr[i]);
15889aa32835SJeff Kirsher 
15899aa32835SJeff Kirsher 		for (i = 0; i < pr->rq3_skba.len; i++)
15909aa32835SJeff Kirsher 			dev_kfree_skb(pr->rq3_skba.arr[i]);
15919aa32835SJeff Kirsher 
15929aa32835SJeff Kirsher 		for (i = 0; i < pr->sq_skba.len; i++)
15939aa32835SJeff Kirsher 			dev_kfree_skb(pr->sq_skba.arr[i]);
15949aa32835SJeff Kirsher 
15959aa32835SJeff Kirsher 		vfree(pr->rq1_skba.arr);
15969aa32835SJeff Kirsher 		vfree(pr->rq2_skba.arr);
15979aa32835SJeff Kirsher 		vfree(pr->rq3_skba.arr);
15989aa32835SJeff Kirsher 		vfree(pr->sq_skba.arr);
15999aa32835SJeff Kirsher 		ret = ehea_rem_smrs(pr);
16009aa32835SJeff Kirsher 	}
16019aa32835SJeff Kirsher 	return ret;
16029aa32835SJeff Kirsher }
16039aa32835SJeff Kirsher 
write_swqe2_immediate(struct sk_buff * skb,struct ehea_swqe * swqe,u32 lkey)160413946f5eSAnton Blanchard static void write_swqe2_immediate(struct sk_buff *skb, struct ehea_swqe *swqe,
160513946f5eSAnton Blanchard 				  u32 lkey)
16069aa32835SJeff Kirsher {
16079aa32835SJeff Kirsher 	int skb_data_size = skb_headlen(skb);
160813946f5eSAnton Blanchard 	u8 *imm_data = &swqe->u.immdata_desc.immediate_data[0];
160913946f5eSAnton Blanchard 	struct ehea_vsgentry *sg1entry = &swqe->u.immdata_desc.sg_entry;
161013946f5eSAnton Blanchard 	unsigned int immediate_len = SWQE2_MAX_IMM;
16119aa32835SJeff Kirsher 
161213946f5eSAnton Blanchard 	swqe->descriptors = 0;
161313946f5eSAnton Blanchard 
161413946f5eSAnton Blanchard 	if (skb_is_gso(skb)) {
16159aa32835SJeff Kirsher 		swqe->tx_control |= EHEA_SWQE_TSO;
16169aa32835SJeff Kirsher 		swqe->mss = skb_shinfo(skb)->gso_size;
161713946f5eSAnton Blanchard 		/*
161813946f5eSAnton Blanchard 		 * For TSO packets we only copy the headers into the
161913946f5eSAnton Blanchard 		 * immediate area.
16209aa32835SJeff Kirsher 		 */
1621504148feSEric Dumazet 		immediate_len = skb_tcp_all_headers(skb);
16229aa32835SJeff Kirsher 	}
16239aa32835SJeff Kirsher 
162413946f5eSAnton Blanchard 	if (skb_is_gso(skb) || skb_data_size >= SWQE2_MAX_IMM) {
162513946f5eSAnton Blanchard 		skb_copy_from_linear_data(skb, imm_data, immediate_len);
162613946f5eSAnton Blanchard 		swqe->immediate_data_length = immediate_len;
16279aa32835SJeff Kirsher 
162813946f5eSAnton Blanchard 		if (skb_data_size > immediate_len) {
16299aa32835SJeff Kirsher 			sg1entry->l_key = lkey;
163013946f5eSAnton Blanchard 			sg1entry->len = skb_data_size - immediate_len;
16319aa32835SJeff Kirsher 			sg1entry->vaddr =
163213946f5eSAnton Blanchard 				ehea_map_vaddr(skb->data + immediate_len);
16339aa32835SJeff Kirsher 			swqe->descriptors++;
16349aa32835SJeff Kirsher 		}
16359aa32835SJeff Kirsher 	} else {
16369aa32835SJeff Kirsher 		skb_copy_from_linear_data(skb, imm_data, skb_data_size);
16379aa32835SJeff Kirsher 		swqe->immediate_data_length = skb_data_size;
16389aa32835SJeff Kirsher 	}
16399aa32835SJeff Kirsher }
16409aa32835SJeff Kirsher 
write_swqe2_data(struct sk_buff * skb,struct net_device * dev,struct ehea_swqe * swqe,u32 lkey)16419aa32835SJeff Kirsher static inline void write_swqe2_data(struct sk_buff *skb, struct net_device *dev,
16429aa32835SJeff Kirsher 				    struct ehea_swqe *swqe, u32 lkey)
16439aa32835SJeff Kirsher {
16449aa32835SJeff Kirsher 	struct ehea_vsgentry *sg_list, *sg1entry, *sgentry;
16459aa32835SJeff Kirsher 	skb_frag_t *frag;
16469aa32835SJeff Kirsher 	int nfrags, sg1entry_contains_frag_data, i;
16479aa32835SJeff Kirsher 
16489aa32835SJeff Kirsher 	nfrags = skb_shinfo(skb)->nr_frags;
16499aa32835SJeff Kirsher 	sg1entry = &swqe->u.immdata_desc.sg_entry;
16509aa32835SJeff Kirsher 	sg_list = (struct ehea_vsgentry *)&swqe->u.immdata_desc.sg_list;
16519aa32835SJeff Kirsher 	sg1entry_contains_frag_data = 0;
16529aa32835SJeff Kirsher 
165313946f5eSAnton Blanchard 	write_swqe2_immediate(skb, swqe, lkey);
16549aa32835SJeff Kirsher 
16559aa32835SJeff Kirsher 	/* write descriptors */
16569aa32835SJeff Kirsher 	if (nfrags > 0) {
16579aa32835SJeff Kirsher 		if (swqe->descriptors == 0) {
16589aa32835SJeff Kirsher 			/* sg1entry not yet used */
16599aa32835SJeff Kirsher 			frag = &skb_shinfo(skb)->frags[0];
16609aa32835SJeff Kirsher 
16619aa32835SJeff Kirsher 			/* copy sg1entry data */
16629aa32835SJeff Kirsher 			sg1entry->l_key = lkey;
16639e903e08SEric Dumazet 			sg1entry->len = skb_frag_size(frag);
16649aa32835SJeff Kirsher 			sg1entry->vaddr =
1665618c4a0aSIan Campbell 				ehea_map_vaddr(skb_frag_address(frag));
16669aa32835SJeff Kirsher 			swqe->descriptors++;
16679aa32835SJeff Kirsher 			sg1entry_contains_frag_data = 1;
16689aa32835SJeff Kirsher 		}
16699aa32835SJeff Kirsher 
16709aa32835SJeff Kirsher 		for (i = sg1entry_contains_frag_data; i < nfrags; i++) {
16719aa32835SJeff Kirsher 
16729aa32835SJeff Kirsher 			frag = &skb_shinfo(skb)->frags[i];
16739aa32835SJeff Kirsher 			sgentry = &sg_list[i - sg1entry_contains_frag_data];
16749aa32835SJeff Kirsher 
16759aa32835SJeff Kirsher 			sgentry->l_key = lkey;
16760110bba5SEric Dumazet 			sgentry->len = skb_frag_size(frag);
1677618c4a0aSIan Campbell 			sgentry->vaddr = ehea_map_vaddr(skb_frag_address(frag));
16789aa32835SJeff Kirsher 			swqe->descriptors++;
16799aa32835SJeff Kirsher 		}
16809aa32835SJeff Kirsher 	}
16819aa32835SJeff Kirsher }
16829aa32835SJeff Kirsher 
ehea_broadcast_reg_helper(struct ehea_port * port,u32 hcallid)16839aa32835SJeff Kirsher static int ehea_broadcast_reg_helper(struct ehea_port *port, u32 hcallid)
16849aa32835SJeff Kirsher {
16859aa32835SJeff Kirsher 	int ret = 0;
16869aa32835SJeff Kirsher 	u64 hret;
16879aa32835SJeff Kirsher 	u8 reg_type;
16889aa32835SJeff Kirsher 
16899aa32835SJeff Kirsher 	/* De/Register untagged packets */
16909aa32835SJeff Kirsher 	reg_type = EHEA_BCMC_BROADCAST | EHEA_BCMC_UNTAGGED;
16919aa32835SJeff Kirsher 	hret = ehea_h_reg_dereg_bcmc(port->adapter->handle,
16929aa32835SJeff Kirsher 				     port->logical_port_id,
16939aa32835SJeff Kirsher 				     reg_type, port->mac_addr, 0, hcallid);
16949aa32835SJeff Kirsher 	if (hret != H_SUCCESS) {
16959aa32835SJeff Kirsher 		pr_err("%sregistering bc address failed (tagged)\n",
16969aa32835SJeff Kirsher 		       hcallid == H_REG_BCMC ? "" : "de");
16979aa32835SJeff Kirsher 		ret = -EIO;
16989aa32835SJeff Kirsher 		goto out_herr;
16999aa32835SJeff Kirsher 	}
17009aa32835SJeff Kirsher 
17019aa32835SJeff Kirsher 	/* De/Register VLAN packets */
17029aa32835SJeff Kirsher 	reg_type = EHEA_BCMC_BROADCAST | EHEA_BCMC_VLANID_ALL;
17039aa32835SJeff Kirsher 	hret = ehea_h_reg_dereg_bcmc(port->adapter->handle,
17049aa32835SJeff Kirsher 				     port->logical_port_id,
17059aa32835SJeff Kirsher 				     reg_type, port->mac_addr, 0, hcallid);
17069aa32835SJeff Kirsher 	if (hret != H_SUCCESS) {
17079aa32835SJeff Kirsher 		pr_err("%sregistering bc address failed (vlan)\n",
17089aa32835SJeff Kirsher 		       hcallid == H_REG_BCMC ? "" : "de");
17099aa32835SJeff Kirsher 		ret = -EIO;
17109aa32835SJeff Kirsher 	}
17119aa32835SJeff Kirsher out_herr:
17129aa32835SJeff Kirsher 	return ret;
17139aa32835SJeff Kirsher }
17149aa32835SJeff Kirsher 
ehea_set_mac_addr(struct net_device * dev,void * sa)17159aa32835SJeff Kirsher static int ehea_set_mac_addr(struct net_device *dev, void *sa)
17169aa32835SJeff Kirsher {
17179aa32835SJeff Kirsher 	struct ehea_port *port = netdev_priv(dev);
17189aa32835SJeff Kirsher 	struct sockaddr *mac_addr = sa;
17199aa32835SJeff Kirsher 	struct hcp_ehea_port_cb0 *cb0;
17209aa32835SJeff Kirsher 	int ret;
17219aa32835SJeff Kirsher 	u64 hret;
17229aa32835SJeff Kirsher 
17239aa32835SJeff Kirsher 	if (!is_valid_ether_addr(mac_addr->sa_data)) {
17249aa32835SJeff Kirsher 		ret = -EADDRNOTAVAIL;
17259aa32835SJeff Kirsher 		goto out;
17269aa32835SJeff Kirsher 	}
17279aa32835SJeff Kirsher 
17289aa32835SJeff Kirsher 	cb0 = (void *)get_zeroed_page(GFP_KERNEL);
17299aa32835SJeff Kirsher 	if (!cb0) {
17309aa32835SJeff Kirsher 		pr_err("no mem for cb0\n");
17319aa32835SJeff Kirsher 		ret = -ENOMEM;
17329aa32835SJeff Kirsher 		goto out;
17339aa32835SJeff Kirsher 	}
17349aa32835SJeff Kirsher 
17359aa32835SJeff Kirsher 	memcpy(&(cb0->port_mac_addr), &(mac_addr->sa_data[0]), ETH_ALEN);
17369aa32835SJeff Kirsher 
17379aa32835SJeff Kirsher 	cb0->port_mac_addr = cb0->port_mac_addr >> 16;
17389aa32835SJeff Kirsher 
17399aa32835SJeff Kirsher 	hret = ehea_h_modify_ehea_port(port->adapter->handle,
17409aa32835SJeff Kirsher 				       port->logical_port_id, H_PORT_CB0,
17419aa32835SJeff Kirsher 				       EHEA_BMASK_SET(H_PORT_CB0_MAC, 1), cb0);
17429aa32835SJeff Kirsher 	if (hret != H_SUCCESS) {
17439aa32835SJeff Kirsher 		ret = -EIO;
17449aa32835SJeff Kirsher 		goto out_free;
17459aa32835SJeff Kirsher 	}
17469aa32835SJeff Kirsher 
1747a05e4c0aSJakub Kicinski 	eth_hw_addr_set(dev, mac_addr->sa_data);
17489aa32835SJeff Kirsher 
17499aa32835SJeff Kirsher 	/* Deregister old MAC in pHYP */
17509aa32835SJeff Kirsher 	if (port->state == EHEA_PORT_UP) {
17519aa32835SJeff Kirsher 		ret = ehea_broadcast_reg_helper(port, H_DEREG_BCMC);
17529aa32835SJeff Kirsher 		if (ret)
17539aa32835SJeff Kirsher 			goto out_upregs;
17549aa32835SJeff Kirsher 	}
17559aa32835SJeff Kirsher 
17569aa32835SJeff Kirsher 	port->mac_addr = cb0->port_mac_addr << 16;
17579aa32835SJeff Kirsher 
17589aa32835SJeff Kirsher 	/* Register new MAC in pHYP */
17599aa32835SJeff Kirsher 	if (port->state == EHEA_PORT_UP) {
17609aa32835SJeff Kirsher 		ret = ehea_broadcast_reg_helper(port, H_REG_BCMC);
17619aa32835SJeff Kirsher 		if (ret)
17629aa32835SJeff Kirsher 			goto out_upregs;
17639aa32835SJeff Kirsher 	}
17649aa32835SJeff Kirsher 
17659aa32835SJeff Kirsher 	ret = 0;
17669aa32835SJeff Kirsher 
17679aa32835SJeff Kirsher out_upregs:
17689aa32835SJeff Kirsher 	ehea_update_bcmc_registrations();
17699aa32835SJeff Kirsher out_free:
17709aa32835SJeff Kirsher 	free_page((unsigned long)cb0);
17719aa32835SJeff Kirsher out:
17729aa32835SJeff Kirsher 	return ret;
17739aa32835SJeff Kirsher }
17749aa32835SJeff Kirsher 
ehea_promiscuous_error(u64 hret,int enable)17759aa32835SJeff Kirsher static void ehea_promiscuous_error(u64 hret, int enable)
17769aa32835SJeff Kirsher {
17779aa32835SJeff Kirsher 	if (hret == H_AUTHORITY)
17789aa32835SJeff Kirsher 		pr_info("Hypervisor denied %sabling promiscuous mode\n",
17799aa32835SJeff Kirsher 			enable == 1 ? "en" : "dis");
17809aa32835SJeff Kirsher 	else
17819aa32835SJeff Kirsher 		pr_err("failed %sabling promiscuous mode\n",
17829aa32835SJeff Kirsher 		       enable == 1 ? "en" : "dis");
17839aa32835SJeff Kirsher }
17849aa32835SJeff Kirsher 
ehea_promiscuous(struct net_device * dev,int enable)17859aa32835SJeff Kirsher static void ehea_promiscuous(struct net_device *dev, int enable)
17869aa32835SJeff Kirsher {
17879aa32835SJeff Kirsher 	struct ehea_port *port = netdev_priv(dev);
17889aa32835SJeff Kirsher 	struct hcp_ehea_port_cb7 *cb7;
17899aa32835SJeff Kirsher 	u64 hret;
17909aa32835SJeff Kirsher 
17919aa32835SJeff Kirsher 	if (enable == port->promisc)
17929aa32835SJeff Kirsher 		return;
17939aa32835SJeff Kirsher 
17949aa32835SJeff Kirsher 	cb7 = (void *)get_zeroed_page(GFP_ATOMIC);
17959aa32835SJeff Kirsher 	if (!cb7) {
17969aa32835SJeff Kirsher 		pr_err("no mem for cb7\n");
17979aa32835SJeff Kirsher 		goto out;
17989aa32835SJeff Kirsher 	}
17999aa32835SJeff Kirsher 
18009aa32835SJeff Kirsher 	/* Modify Pxs_DUCQPN in CB7 */
18019aa32835SJeff Kirsher 	cb7->def_uc_qpn = enable == 1 ? port->port_res[0].qp->fw_handle : 0;
18029aa32835SJeff Kirsher 
18039aa32835SJeff Kirsher 	hret = ehea_h_modify_ehea_port(port->adapter->handle,
18049aa32835SJeff Kirsher 				       port->logical_port_id,
18059aa32835SJeff Kirsher 				       H_PORT_CB7, H_PORT_CB7_DUCQPN, cb7);
18069aa32835SJeff Kirsher 	if (hret) {
18079aa32835SJeff Kirsher 		ehea_promiscuous_error(hret, enable);
18089aa32835SJeff Kirsher 		goto out;
18099aa32835SJeff Kirsher 	}
18109aa32835SJeff Kirsher 
18119aa32835SJeff Kirsher 	port->promisc = enable;
18129aa32835SJeff Kirsher out:
18139aa32835SJeff Kirsher 	free_page((unsigned long)cb7);
18149aa32835SJeff Kirsher }
18159aa32835SJeff Kirsher 
ehea_multicast_reg_helper(struct ehea_port * port,u64 mc_mac_addr,u32 hcallid)18169aa32835SJeff Kirsher static u64 ehea_multicast_reg_helper(struct ehea_port *port, u64 mc_mac_addr,
18179aa32835SJeff Kirsher 				     u32 hcallid)
18189aa32835SJeff Kirsher {
18199aa32835SJeff Kirsher 	u64 hret;
18209aa32835SJeff Kirsher 	u8 reg_type;
18219aa32835SJeff Kirsher 
18225d384574SThadeu Lima de Souza Cascardo 	reg_type = EHEA_BCMC_MULTICAST | EHEA_BCMC_UNTAGGED;
18235d384574SThadeu Lima de Souza Cascardo 	if (mc_mac_addr == 0)
18245d384574SThadeu Lima de Souza Cascardo 		reg_type |= EHEA_BCMC_SCOPE_ALL;
18259aa32835SJeff Kirsher 
18269aa32835SJeff Kirsher 	hret = ehea_h_reg_dereg_bcmc(port->adapter->handle,
18279aa32835SJeff Kirsher 				     port->logical_port_id,
18289aa32835SJeff Kirsher 				     reg_type, mc_mac_addr, 0, hcallid);
18299aa32835SJeff Kirsher 	if (hret)
18309aa32835SJeff Kirsher 		goto out;
18319aa32835SJeff Kirsher 
18325d384574SThadeu Lima de Souza Cascardo 	reg_type = EHEA_BCMC_MULTICAST | EHEA_BCMC_VLANID_ALL;
18335d384574SThadeu Lima de Souza Cascardo 	if (mc_mac_addr == 0)
18345d384574SThadeu Lima de Souza Cascardo 		reg_type |= EHEA_BCMC_SCOPE_ALL;
18359aa32835SJeff Kirsher 
18369aa32835SJeff Kirsher 	hret = ehea_h_reg_dereg_bcmc(port->adapter->handle,
18379aa32835SJeff Kirsher 				     port->logical_port_id,
18389aa32835SJeff Kirsher 				     reg_type, mc_mac_addr, 0, hcallid);
18399aa32835SJeff Kirsher out:
18409aa32835SJeff Kirsher 	return hret;
18419aa32835SJeff Kirsher }
18429aa32835SJeff Kirsher 
ehea_drop_multicast_list(struct net_device * dev)18439aa32835SJeff Kirsher static int ehea_drop_multicast_list(struct net_device *dev)
18449aa32835SJeff Kirsher {
18459aa32835SJeff Kirsher 	struct ehea_port *port = netdev_priv(dev);
18469aa32835SJeff Kirsher 	struct ehea_mc_list *mc_entry = port->mc_list;
18479aa32835SJeff Kirsher 	struct list_head *pos;
18489aa32835SJeff Kirsher 	struct list_head *temp;
18499aa32835SJeff Kirsher 	int ret = 0;
18509aa32835SJeff Kirsher 	u64 hret;
18519aa32835SJeff Kirsher 
18529aa32835SJeff Kirsher 	list_for_each_safe(pos, temp, &(port->mc_list->list)) {
18539aa32835SJeff Kirsher 		mc_entry = list_entry(pos, struct ehea_mc_list, list);
18549aa32835SJeff Kirsher 
18559aa32835SJeff Kirsher 		hret = ehea_multicast_reg_helper(port, mc_entry->macaddr,
18569aa32835SJeff Kirsher 						 H_DEREG_BCMC);
18579aa32835SJeff Kirsher 		if (hret) {
18589aa32835SJeff Kirsher 			pr_err("failed deregistering mcast MAC\n");
18599aa32835SJeff Kirsher 			ret = -EIO;
18609aa32835SJeff Kirsher 		}
18619aa32835SJeff Kirsher 
18629aa32835SJeff Kirsher 		list_del(pos);
18639aa32835SJeff Kirsher 		kfree(mc_entry);
18649aa32835SJeff Kirsher 	}
18659aa32835SJeff Kirsher 	return ret;
18669aa32835SJeff Kirsher }
18679aa32835SJeff Kirsher 
ehea_allmulti(struct net_device * dev,int enable)18689aa32835SJeff Kirsher static void ehea_allmulti(struct net_device *dev, int enable)
18699aa32835SJeff Kirsher {
18709aa32835SJeff Kirsher 	struct ehea_port *port = netdev_priv(dev);
18719aa32835SJeff Kirsher 	u64 hret;
18729aa32835SJeff Kirsher 
18739aa32835SJeff Kirsher 	if (!port->allmulti) {
18749aa32835SJeff Kirsher 		if (enable) {
18759aa32835SJeff Kirsher 			/* Enable ALLMULTI */
18769aa32835SJeff Kirsher 			ehea_drop_multicast_list(dev);
18779aa32835SJeff Kirsher 			hret = ehea_multicast_reg_helper(port, 0, H_REG_BCMC);
18789aa32835SJeff Kirsher 			if (!hret)
18799aa32835SJeff Kirsher 				port->allmulti = 1;
18809aa32835SJeff Kirsher 			else
18819aa32835SJeff Kirsher 				netdev_err(dev,
18829aa32835SJeff Kirsher 					   "failed enabling IFF_ALLMULTI\n");
18839aa32835SJeff Kirsher 		}
18845d384574SThadeu Lima de Souza Cascardo 	} else {
18859aa32835SJeff Kirsher 		if (!enable) {
18869aa32835SJeff Kirsher 			/* Disable ALLMULTI */
18879aa32835SJeff Kirsher 			hret = ehea_multicast_reg_helper(port, 0, H_DEREG_BCMC);
18889aa32835SJeff Kirsher 			if (!hret)
18899aa32835SJeff Kirsher 				port->allmulti = 0;
18909aa32835SJeff Kirsher 			else
18919aa32835SJeff Kirsher 				netdev_err(dev,
18929aa32835SJeff Kirsher 					   "failed disabling IFF_ALLMULTI\n");
18939aa32835SJeff Kirsher 		}
18949aa32835SJeff Kirsher 	}
18955d384574SThadeu Lima de Souza Cascardo }
18969aa32835SJeff Kirsher 
ehea_add_multicast_entry(struct ehea_port * port,u8 * mc_mac_addr)18979aa32835SJeff Kirsher static void ehea_add_multicast_entry(struct ehea_port *port, u8 *mc_mac_addr)
18989aa32835SJeff Kirsher {
18999aa32835SJeff Kirsher 	struct ehea_mc_list *ehea_mcl_entry;
19009aa32835SJeff Kirsher 	u64 hret;
19019aa32835SJeff Kirsher 
19029aa32835SJeff Kirsher 	ehea_mcl_entry = kzalloc(sizeof(*ehea_mcl_entry), GFP_ATOMIC);
1903b2adaca9SJoe Perches 	if (!ehea_mcl_entry)
19049aa32835SJeff Kirsher 		return;
19059aa32835SJeff Kirsher 
19069aa32835SJeff Kirsher 	INIT_LIST_HEAD(&ehea_mcl_entry->list);
19079aa32835SJeff Kirsher 
19089aa32835SJeff Kirsher 	memcpy(&ehea_mcl_entry->macaddr, mc_mac_addr, ETH_ALEN);
19099aa32835SJeff Kirsher 
19109aa32835SJeff Kirsher 	hret = ehea_multicast_reg_helper(port, ehea_mcl_entry->macaddr,
19119aa32835SJeff Kirsher 					 H_REG_BCMC);
19129aa32835SJeff Kirsher 	if (!hret)
19139aa32835SJeff Kirsher 		list_add(&ehea_mcl_entry->list, &port->mc_list->list);
19149aa32835SJeff Kirsher 	else {
19159aa32835SJeff Kirsher 		pr_err("failed registering mcast MAC\n");
19169aa32835SJeff Kirsher 		kfree(ehea_mcl_entry);
19179aa32835SJeff Kirsher 	}
19189aa32835SJeff Kirsher }
19199aa32835SJeff Kirsher 
ehea_set_multicast_list(struct net_device * dev)19209aa32835SJeff Kirsher static void ehea_set_multicast_list(struct net_device *dev)
19219aa32835SJeff Kirsher {
19229aa32835SJeff Kirsher 	struct ehea_port *port = netdev_priv(dev);
19239aa32835SJeff Kirsher 	struct netdev_hw_addr *ha;
19249aa32835SJeff Kirsher 	int ret;
19259aa32835SJeff Kirsher 
192647d59d01SThadeu Lima de Souza Cascardo 	ehea_promiscuous(dev, !!(dev->flags & IFF_PROMISC));
19279aa32835SJeff Kirsher 
19289aa32835SJeff Kirsher 	if (dev->flags & IFF_ALLMULTI) {
19299aa32835SJeff Kirsher 		ehea_allmulti(dev, 1);
19309aa32835SJeff Kirsher 		goto out;
19319aa32835SJeff Kirsher 	}
19329aa32835SJeff Kirsher 	ehea_allmulti(dev, 0);
19339aa32835SJeff Kirsher 
19349aa32835SJeff Kirsher 	if (!netdev_mc_empty(dev)) {
19359aa32835SJeff Kirsher 		ret = ehea_drop_multicast_list(dev);
19369aa32835SJeff Kirsher 		if (ret) {
19379aa32835SJeff Kirsher 			/* Dropping the current multicast list failed.
19389aa32835SJeff Kirsher 			 * Enabling ALL_MULTI is the best we can do.
19399aa32835SJeff Kirsher 			 */
19409aa32835SJeff Kirsher 			ehea_allmulti(dev, 1);
19419aa32835SJeff Kirsher 		}
19429aa32835SJeff Kirsher 
19439aa32835SJeff Kirsher 		if (netdev_mc_count(dev) > port->adapter->max_mc_mac) {
19449aa32835SJeff Kirsher 			pr_info("Mcast registration limit reached (0x%llx). Use ALLMULTI!\n",
19459aa32835SJeff Kirsher 				port->adapter->max_mc_mac);
19469aa32835SJeff Kirsher 			goto out;
19479aa32835SJeff Kirsher 		}
19489aa32835SJeff Kirsher 
19499aa32835SJeff Kirsher 		netdev_for_each_mc_addr(ha, dev)
19509aa32835SJeff Kirsher 			ehea_add_multicast_entry(port, ha->addr);
19519aa32835SJeff Kirsher 
19529aa32835SJeff Kirsher 	}
19539aa32835SJeff Kirsher out:
19549aa32835SJeff Kirsher 	ehea_update_bcmc_registrations();
19559aa32835SJeff Kirsher }
19569aa32835SJeff Kirsher 
xmit_common(struct sk_buff * skb,struct ehea_swqe * swqe)1957d695c335SAnton Blanchard static void xmit_common(struct sk_buff *skb, struct ehea_swqe *swqe)
1958d695c335SAnton Blanchard {
1959d695c335SAnton Blanchard 	swqe->tx_control |= EHEA_SWQE_IMM_DATA_PRESENT | EHEA_SWQE_CRC;
1960d695c335SAnton Blanchard 
1961be1d1486SVlad Yasevich 	if (vlan_get_protocol(skb) != htons(ETH_P_IP))
1962d695c335SAnton Blanchard 		return;
1963d695c335SAnton Blanchard 
1964d695c335SAnton Blanchard 	if (skb->ip_summed == CHECKSUM_PARTIAL)
1965d695c335SAnton Blanchard 		swqe->tx_control |= EHEA_SWQE_IP_CHECKSUM;
1966d695c335SAnton Blanchard 
1967d695c335SAnton Blanchard 	swqe->ip_start = skb_network_offset(skb);
1968d695c335SAnton Blanchard 	swqe->ip_end = swqe->ip_start + ip_hdrlen(skb) - 1;
1969d695c335SAnton Blanchard 
1970d695c335SAnton Blanchard 	switch (ip_hdr(skb)->protocol) {
1971d695c335SAnton Blanchard 	case IPPROTO_UDP:
1972d695c335SAnton Blanchard 		if (skb->ip_summed == CHECKSUM_PARTIAL)
1973d695c335SAnton Blanchard 			swqe->tx_control |= EHEA_SWQE_TCP_CHECKSUM;
1974d695c335SAnton Blanchard 
1975d695c335SAnton Blanchard 		swqe->tcp_offset = swqe->ip_end + 1 +
1976d695c335SAnton Blanchard 				   offsetof(struct udphdr, check);
1977d695c335SAnton Blanchard 		break;
1978d695c335SAnton Blanchard 
1979d695c335SAnton Blanchard 	case IPPROTO_TCP:
1980d695c335SAnton Blanchard 		if (skb->ip_summed == CHECKSUM_PARTIAL)
1981d695c335SAnton Blanchard 			swqe->tx_control |= EHEA_SWQE_TCP_CHECKSUM;
1982d695c335SAnton Blanchard 
1983d695c335SAnton Blanchard 		swqe->tcp_offset = swqe->ip_end + 1 +
1984d695c335SAnton Blanchard 				   offsetof(struct tcphdr, check);
1985d695c335SAnton Blanchard 		break;
1986d695c335SAnton Blanchard 	}
1987d695c335SAnton Blanchard }
1988d695c335SAnton Blanchard 
ehea_xmit2(struct sk_buff * skb,struct net_device * dev,struct ehea_swqe * swqe,u32 lkey)19899aa32835SJeff Kirsher static void ehea_xmit2(struct sk_buff *skb, struct net_device *dev,
19909aa32835SJeff Kirsher 		       struct ehea_swqe *swqe, u32 lkey)
19919aa32835SJeff Kirsher {
1992d695c335SAnton Blanchard 	swqe->tx_control |= EHEA_SWQE_DESCRIPTORS_PRESENT;
19939aa32835SJeff Kirsher 
1994d695c335SAnton Blanchard 	xmit_common(skb, swqe);
19959aa32835SJeff Kirsher 
19969aa32835SJeff Kirsher 	write_swqe2_data(skb, dev, swqe, lkey);
19979aa32835SJeff Kirsher }
19989aa32835SJeff Kirsher 
ehea_xmit3(struct sk_buff * skb,struct net_device * dev,struct ehea_swqe * swqe)19999aa32835SJeff Kirsher static void ehea_xmit3(struct sk_buff *skb, struct net_device *dev,
20009aa32835SJeff Kirsher 		       struct ehea_swqe *swqe)
20019aa32835SJeff Kirsher {
20029aa32835SJeff Kirsher 	u8 *imm_data = &swqe->u.immdata_nodesc.immediate_data[0];
20039aa32835SJeff Kirsher 
2004d695c335SAnton Blanchard 	xmit_common(skb, swqe);
20059aa32835SJeff Kirsher 
200630e2e90bSAnton Blanchard 	if (!skb->data_len)
20079aa32835SJeff Kirsher 		skb_copy_from_linear_data(skb, imm_data, skb->len);
200830e2e90bSAnton Blanchard 	else
200930e2e90bSAnton Blanchard 		skb_copy_bits(skb, 0, imm_data, skb->len);
2010d695c335SAnton Blanchard 
20119aa32835SJeff Kirsher 	swqe->immediate_data_length = skb->len;
2012cfbe4063SEric W. Biederman 	dev_consume_skb_any(skb);
20139aa32835SJeff Kirsher }
20149aa32835SJeff Kirsher 
ehea_start_xmit(struct sk_buff * skb,struct net_device * dev)201594b2bb28SYueHaibing static netdev_tx_t ehea_start_xmit(struct sk_buff *skb, struct net_device *dev)
20169aa32835SJeff Kirsher {
20179aa32835SJeff Kirsher 	struct ehea_port *port = netdev_priv(dev);
20189aa32835SJeff Kirsher 	struct ehea_swqe *swqe;
20199aa32835SJeff Kirsher 	u32 lkey;
20209aa32835SJeff Kirsher 	int swqe_index;
20219aa32835SJeff Kirsher 	struct ehea_port_res *pr;
2022b9564468SAnton Blanchard 	struct netdev_queue *txq;
20239aa32835SJeff Kirsher 
2024b9564468SAnton Blanchard 	pr = &port->port_res[skb_get_queue_mapping(skb)];
2025b9564468SAnton Blanchard 	txq = netdev_get_tx_queue(dev, skb_get_queue_mapping(skb));
20269aa32835SJeff Kirsher 
20279aa32835SJeff Kirsher 	swqe = ehea_get_swqe(pr->qp, &swqe_index);
20289aa32835SJeff Kirsher 	memset(swqe, 0, SWQE_HEADER_SIZE);
20299aa32835SJeff Kirsher 	atomic_dec(&pr->swqe_avail);
20309aa32835SJeff Kirsher 
2031df8a39deSJiri Pirko 	if (skb_vlan_tag_present(skb)) {
20329aa32835SJeff Kirsher 		swqe->tx_control |= EHEA_SWQE_VLAN_INSERT;
2033df8a39deSJiri Pirko 		swqe->vlan_tag = skb_vlan_tag_get(skb);
20349aa32835SJeff Kirsher 	}
20359aa32835SJeff Kirsher 
20369aa32835SJeff Kirsher 	pr->tx_packets++;
20379aa32835SJeff Kirsher 	pr->tx_bytes += skb->len;
20389aa32835SJeff Kirsher 
20399aa32835SJeff Kirsher 	if (skb->len <= SWQE3_MAX_IMM) {
20409aa32835SJeff Kirsher 		u32 sig_iv = port->sig_comp_iv;
20419aa32835SJeff Kirsher 		u32 swqe_num = pr->swqe_id_counter;
20429aa32835SJeff Kirsher 		ehea_xmit3(skb, dev, swqe);
20439aa32835SJeff Kirsher 		swqe->wr_id = EHEA_BMASK_SET(EHEA_WR_ID_TYPE, EHEA_SWQE3_TYPE)
20449aa32835SJeff Kirsher 			| EHEA_BMASK_SET(EHEA_WR_ID_COUNT, swqe_num);
20459aa32835SJeff Kirsher 		if (pr->swqe_ll_count >= (sig_iv - 1)) {
20469aa32835SJeff Kirsher 			swqe->wr_id |= EHEA_BMASK_SET(EHEA_WR_ID_REFILL,
20479aa32835SJeff Kirsher 						      sig_iv);
20489aa32835SJeff Kirsher 			swqe->tx_control |= EHEA_SWQE_SIGNALLED_COMPLETION;
20499aa32835SJeff Kirsher 			pr->swqe_ll_count = 0;
20509aa32835SJeff Kirsher 		} else
20519aa32835SJeff Kirsher 			pr->swqe_ll_count += 1;
20529aa32835SJeff Kirsher 	} else {
20539aa32835SJeff Kirsher 		swqe->wr_id =
20549aa32835SJeff Kirsher 			EHEA_BMASK_SET(EHEA_WR_ID_TYPE, EHEA_SWQE2_TYPE)
20559aa32835SJeff Kirsher 		      | EHEA_BMASK_SET(EHEA_WR_ID_COUNT, pr->swqe_id_counter)
20569aa32835SJeff Kirsher 		      | EHEA_BMASK_SET(EHEA_WR_ID_REFILL, 1)
20579aa32835SJeff Kirsher 		      | EHEA_BMASK_SET(EHEA_WR_ID_INDEX, pr->sq_skba.index);
20589aa32835SJeff Kirsher 		pr->sq_skba.arr[pr->sq_skba.index] = skb;
20599aa32835SJeff Kirsher 
20609aa32835SJeff Kirsher 		pr->sq_skba.index++;
20619aa32835SJeff Kirsher 		pr->sq_skba.index &= (pr->sq_skba.len - 1);
20629aa32835SJeff Kirsher 
20639aa32835SJeff Kirsher 		lkey = pr->send_mr.lkey;
20649aa32835SJeff Kirsher 		ehea_xmit2(skb, dev, swqe, lkey);
20659aa32835SJeff Kirsher 		swqe->tx_control |= EHEA_SWQE_SIGNALLED_COMPLETION;
20669aa32835SJeff Kirsher 	}
20679aa32835SJeff Kirsher 	pr->swqe_id_counter += 1;
20689aa32835SJeff Kirsher 
20699aa32835SJeff Kirsher 	netif_info(port, tx_queued, dev,
20709aa32835SJeff Kirsher 		   "post swqe on QP %d\n", pr->qp->init_attr.qp_nr);
20719aa32835SJeff Kirsher 	if (netif_msg_tx_queued(port))
20729aa32835SJeff Kirsher 		ehea_dump(swqe, 512, "swqe");
20739aa32835SJeff Kirsher 
20749aa32835SJeff Kirsher 	if (unlikely(test_bit(__EHEA_STOP_XFER, &ehea_driver_flags))) {
2075b9564468SAnton Blanchard 		netif_tx_stop_queue(txq);
20769aa32835SJeff Kirsher 		swqe->tx_control |= EHEA_SWQE_PURGE;
20779aa32835SJeff Kirsher 	}
20789aa32835SJeff Kirsher 
20799aa32835SJeff Kirsher 	ehea_post_swqe(pr->qp, swqe);
20809aa32835SJeff Kirsher 
20819aa32835SJeff Kirsher 	if (unlikely(atomic_read(&pr->swqe_avail) <= 1)) {
20829aa32835SJeff Kirsher 		pr->p_stats.queue_stopped++;
2083b9564468SAnton Blanchard 		netif_tx_stop_queue(txq);
20849aa32835SJeff Kirsher 	}
20859aa32835SJeff Kirsher 
20869aa32835SJeff Kirsher 	return NETDEV_TX_OK;
20879aa32835SJeff Kirsher }
20889aa32835SJeff Kirsher 
ehea_vlan_rx_add_vid(struct net_device * dev,__be16 proto,u16 vid)208980d5c368SPatrick McHardy static int ehea_vlan_rx_add_vid(struct net_device *dev, __be16 proto, u16 vid)
20909aa32835SJeff Kirsher {
20919aa32835SJeff Kirsher 	struct ehea_port *port = netdev_priv(dev);
20929aa32835SJeff Kirsher 	struct ehea_adapter *adapter = port->adapter;
20939aa32835SJeff Kirsher 	struct hcp_ehea_port_cb1 *cb1;
20949aa32835SJeff Kirsher 	int index;
20959aa32835SJeff Kirsher 	u64 hret;
20968e586137SJiri Pirko 	int err = 0;
20979aa32835SJeff Kirsher 
20989aa32835SJeff Kirsher 	cb1 = (void *)get_zeroed_page(GFP_KERNEL);
20999aa32835SJeff Kirsher 	if (!cb1) {
21009aa32835SJeff Kirsher 		pr_err("no mem for cb1\n");
21018e586137SJiri Pirko 		err = -ENOMEM;
21029aa32835SJeff Kirsher 		goto out;
21039aa32835SJeff Kirsher 	}
21049aa32835SJeff Kirsher 
21059aa32835SJeff Kirsher 	hret = ehea_h_query_ehea_port(adapter->handle, port->logical_port_id,
21069aa32835SJeff Kirsher 				      H_PORT_CB1, H_PORT_CB1_ALL, cb1);
21079aa32835SJeff Kirsher 	if (hret != H_SUCCESS) {
21089aa32835SJeff Kirsher 		pr_err("query_ehea_port failed\n");
21098e586137SJiri Pirko 		err = -EINVAL;
21109aa32835SJeff Kirsher 		goto out;
21119aa32835SJeff Kirsher 	}
21129aa32835SJeff Kirsher 
21139aa32835SJeff Kirsher 	index = (vid / 64);
21149aa32835SJeff Kirsher 	cb1->vlan_filter[index] |= ((u64)(0x8000000000000000 >> (vid & 0x3F)));
21159aa32835SJeff Kirsher 
21169aa32835SJeff Kirsher 	hret = ehea_h_modify_ehea_port(adapter->handle, port->logical_port_id,
21179aa32835SJeff Kirsher 				       H_PORT_CB1, H_PORT_CB1_ALL, cb1);
21188e586137SJiri Pirko 	if (hret != H_SUCCESS) {
21199aa32835SJeff Kirsher 		pr_err("modify_ehea_port failed\n");
21208e586137SJiri Pirko 		err = -EINVAL;
21218e586137SJiri Pirko 	}
21229aa32835SJeff Kirsher out:
21239aa32835SJeff Kirsher 	free_page((unsigned long)cb1);
21248e586137SJiri Pirko 	return err;
21259aa32835SJeff Kirsher }
21269aa32835SJeff Kirsher 
ehea_vlan_rx_kill_vid(struct net_device * dev,__be16 proto,u16 vid)212780d5c368SPatrick McHardy static int ehea_vlan_rx_kill_vid(struct net_device *dev, __be16 proto, u16 vid)
21289aa32835SJeff Kirsher {
21299aa32835SJeff Kirsher 	struct ehea_port *port = netdev_priv(dev);
21309aa32835SJeff Kirsher 	struct ehea_adapter *adapter = port->adapter;
21319aa32835SJeff Kirsher 	struct hcp_ehea_port_cb1 *cb1;
21329aa32835SJeff Kirsher 	int index;
21339aa32835SJeff Kirsher 	u64 hret;
21348e586137SJiri Pirko 	int err = 0;
21359aa32835SJeff Kirsher 
21369aa32835SJeff Kirsher 	cb1 = (void *)get_zeroed_page(GFP_KERNEL);
21379aa32835SJeff Kirsher 	if (!cb1) {
21389aa32835SJeff Kirsher 		pr_err("no mem for cb1\n");
21398e586137SJiri Pirko 		err = -ENOMEM;
21409aa32835SJeff Kirsher 		goto out;
21419aa32835SJeff Kirsher 	}
21429aa32835SJeff Kirsher 
21439aa32835SJeff Kirsher 	hret = ehea_h_query_ehea_port(adapter->handle, port->logical_port_id,
21449aa32835SJeff Kirsher 				      H_PORT_CB1, H_PORT_CB1_ALL, cb1);
21459aa32835SJeff Kirsher 	if (hret != H_SUCCESS) {
21469aa32835SJeff Kirsher 		pr_err("query_ehea_port failed\n");
21478e586137SJiri Pirko 		err = -EINVAL;
21489aa32835SJeff Kirsher 		goto out;
21499aa32835SJeff Kirsher 	}
21509aa32835SJeff Kirsher 
21519aa32835SJeff Kirsher 	index = (vid / 64);
21529aa32835SJeff Kirsher 	cb1->vlan_filter[index] &= ~((u64)(0x8000000000000000 >> (vid & 0x3F)));
21539aa32835SJeff Kirsher 
21549aa32835SJeff Kirsher 	hret = ehea_h_modify_ehea_port(adapter->handle, port->logical_port_id,
21559aa32835SJeff Kirsher 				       H_PORT_CB1, H_PORT_CB1_ALL, cb1);
21568e586137SJiri Pirko 	if (hret != H_SUCCESS) {
21579aa32835SJeff Kirsher 		pr_err("modify_ehea_port failed\n");
21588e586137SJiri Pirko 		err = -EINVAL;
21598e586137SJiri Pirko 	}
21609aa32835SJeff Kirsher out:
21619aa32835SJeff Kirsher 	free_page((unsigned long)cb1);
21628e586137SJiri Pirko 	return err;
21639aa32835SJeff Kirsher }
21649aa32835SJeff Kirsher 
ehea_activate_qp(struct ehea_adapter * adapter,struct ehea_qp * qp)21651886e5d2SThadeu Lima de Souza Cascardo static int ehea_activate_qp(struct ehea_adapter *adapter, struct ehea_qp *qp)
21669aa32835SJeff Kirsher {
21679aa32835SJeff Kirsher 	int ret = -EIO;
21689aa32835SJeff Kirsher 	u64 hret;
21699aa32835SJeff Kirsher 	u16 dummy16 = 0;
21709aa32835SJeff Kirsher 	u64 dummy64 = 0;
21719aa32835SJeff Kirsher 	struct hcp_modify_qp_cb0 *cb0;
21729aa32835SJeff Kirsher 
21739aa32835SJeff Kirsher 	cb0 = (void *)get_zeroed_page(GFP_KERNEL);
21749aa32835SJeff Kirsher 	if (!cb0) {
21759aa32835SJeff Kirsher 		ret = -ENOMEM;
21769aa32835SJeff Kirsher 		goto out;
21779aa32835SJeff Kirsher 	}
21789aa32835SJeff Kirsher 
21799aa32835SJeff Kirsher 	hret = ehea_h_query_ehea_qp(adapter->handle, 0, qp->fw_handle,
21809aa32835SJeff Kirsher 				    EHEA_BMASK_SET(H_QPCB0_ALL, 0xFFFF), cb0);
21819aa32835SJeff Kirsher 	if (hret != H_SUCCESS) {
21829aa32835SJeff Kirsher 		pr_err("query_ehea_qp failed (1)\n");
21839aa32835SJeff Kirsher 		goto out;
21849aa32835SJeff Kirsher 	}
21859aa32835SJeff Kirsher 
21869aa32835SJeff Kirsher 	cb0->qp_ctl_reg = H_QP_CR_STATE_INITIALIZED;
21879aa32835SJeff Kirsher 	hret = ehea_h_modify_ehea_qp(adapter->handle, 0, qp->fw_handle,
21889aa32835SJeff Kirsher 				     EHEA_BMASK_SET(H_QPCB0_QP_CTL_REG, 1), cb0,
21899aa32835SJeff Kirsher 				     &dummy64, &dummy64, &dummy16, &dummy16);
21909aa32835SJeff Kirsher 	if (hret != H_SUCCESS) {
21919aa32835SJeff Kirsher 		pr_err("modify_ehea_qp failed (1)\n");
21929aa32835SJeff Kirsher 		goto out;
21939aa32835SJeff Kirsher 	}
21949aa32835SJeff Kirsher 
21959aa32835SJeff Kirsher 	hret = ehea_h_query_ehea_qp(adapter->handle, 0, qp->fw_handle,
21969aa32835SJeff Kirsher 				    EHEA_BMASK_SET(H_QPCB0_ALL, 0xFFFF), cb0);
21979aa32835SJeff Kirsher 	if (hret != H_SUCCESS) {
21989aa32835SJeff Kirsher 		pr_err("query_ehea_qp failed (2)\n");
21999aa32835SJeff Kirsher 		goto out;
22009aa32835SJeff Kirsher 	}
22019aa32835SJeff Kirsher 
22029aa32835SJeff Kirsher 	cb0->qp_ctl_reg = H_QP_CR_ENABLED | H_QP_CR_STATE_INITIALIZED;
22039aa32835SJeff Kirsher 	hret = ehea_h_modify_ehea_qp(adapter->handle, 0, qp->fw_handle,
22049aa32835SJeff Kirsher 				     EHEA_BMASK_SET(H_QPCB0_QP_CTL_REG, 1), cb0,
22059aa32835SJeff Kirsher 				     &dummy64, &dummy64, &dummy16, &dummy16);
22069aa32835SJeff Kirsher 	if (hret != H_SUCCESS) {
22079aa32835SJeff Kirsher 		pr_err("modify_ehea_qp failed (2)\n");
22089aa32835SJeff Kirsher 		goto out;
22099aa32835SJeff Kirsher 	}
22109aa32835SJeff Kirsher 
22119aa32835SJeff Kirsher 	hret = ehea_h_query_ehea_qp(adapter->handle, 0, qp->fw_handle,
22129aa32835SJeff Kirsher 				    EHEA_BMASK_SET(H_QPCB0_ALL, 0xFFFF), cb0);
22139aa32835SJeff Kirsher 	if (hret != H_SUCCESS) {
22149aa32835SJeff Kirsher 		pr_err("query_ehea_qp failed (3)\n");
22159aa32835SJeff Kirsher 		goto out;
22169aa32835SJeff Kirsher 	}
22179aa32835SJeff Kirsher 
22189aa32835SJeff Kirsher 	cb0->qp_ctl_reg = H_QP_CR_ENABLED | H_QP_CR_STATE_RDY2SND;
22199aa32835SJeff Kirsher 	hret = ehea_h_modify_ehea_qp(adapter->handle, 0, qp->fw_handle,
22209aa32835SJeff Kirsher 				     EHEA_BMASK_SET(H_QPCB0_QP_CTL_REG, 1), cb0,
22219aa32835SJeff Kirsher 				     &dummy64, &dummy64, &dummy16, &dummy16);
22229aa32835SJeff Kirsher 	if (hret != H_SUCCESS) {
22239aa32835SJeff Kirsher 		pr_err("modify_ehea_qp failed (3)\n");
22249aa32835SJeff Kirsher 		goto out;
22259aa32835SJeff Kirsher 	}
22269aa32835SJeff Kirsher 
22279aa32835SJeff Kirsher 	hret = ehea_h_query_ehea_qp(adapter->handle, 0, qp->fw_handle,
22289aa32835SJeff Kirsher 				    EHEA_BMASK_SET(H_QPCB0_ALL, 0xFFFF), cb0);
22299aa32835SJeff Kirsher 	if (hret != H_SUCCESS) {
22309aa32835SJeff Kirsher 		pr_err("query_ehea_qp failed (4)\n");
22319aa32835SJeff Kirsher 		goto out;
22329aa32835SJeff Kirsher 	}
22339aa32835SJeff Kirsher 
22349aa32835SJeff Kirsher 	ret = 0;
22359aa32835SJeff Kirsher out:
22369aa32835SJeff Kirsher 	free_page((unsigned long)cb0);
22379aa32835SJeff Kirsher 	return ret;
22389aa32835SJeff Kirsher }
22399aa32835SJeff Kirsher 
ehea_port_res_setup(struct ehea_port * port,int def_qps)2240723f28e4SAnton Blanchard static int ehea_port_res_setup(struct ehea_port *port, int def_qps)
22419aa32835SJeff Kirsher {
22429aa32835SJeff Kirsher 	int ret, i;
22439aa32835SJeff Kirsher 	struct port_res_cfg pr_cfg, pr_cfg_small_rx;
22449aa32835SJeff Kirsher 	enum ehea_eq_type eq_type = EHEA_EQ;
22459aa32835SJeff Kirsher 
22469aa32835SJeff Kirsher 	port->qp_eq = ehea_create_eq(port->adapter, eq_type,
22479aa32835SJeff Kirsher 				   EHEA_MAX_ENTRIES_EQ, 1);
22489aa32835SJeff Kirsher 	if (!port->qp_eq) {
22499aa32835SJeff Kirsher 		ret = -EINVAL;
22509aa32835SJeff Kirsher 		pr_err("ehea_create_eq failed (qp_eq)\n");
22519aa32835SJeff Kirsher 		goto out_kill_eq;
22529aa32835SJeff Kirsher 	}
22539aa32835SJeff Kirsher 
22549aa32835SJeff Kirsher 	pr_cfg.max_entries_rcq = rq1_entries + rq2_entries + rq3_entries;
22559aa32835SJeff Kirsher 	pr_cfg.max_entries_scq = sq_entries * 2;
22569aa32835SJeff Kirsher 	pr_cfg.max_entries_sq = sq_entries;
22579aa32835SJeff Kirsher 	pr_cfg.max_entries_rq1 = rq1_entries;
22589aa32835SJeff Kirsher 	pr_cfg.max_entries_rq2 = rq2_entries;
22599aa32835SJeff Kirsher 	pr_cfg.max_entries_rq3 = rq3_entries;
22609aa32835SJeff Kirsher 
22619aa32835SJeff Kirsher 	pr_cfg_small_rx.max_entries_rcq = 1;
22629aa32835SJeff Kirsher 	pr_cfg_small_rx.max_entries_scq = sq_entries;
22639aa32835SJeff Kirsher 	pr_cfg_small_rx.max_entries_sq = sq_entries;
22649aa32835SJeff Kirsher 	pr_cfg_small_rx.max_entries_rq1 = 1;
22659aa32835SJeff Kirsher 	pr_cfg_small_rx.max_entries_rq2 = 1;
22669aa32835SJeff Kirsher 	pr_cfg_small_rx.max_entries_rq3 = 1;
22679aa32835SJeff Kirsher 
22689aa32835SJeff Kirsher 	for (i = 0; i < def_qps; i++) {
22699aa32835SJeff Kirsher 		ret = ehea_init_port_res(port, &port->port_res[i], &pr_cfg, i);
22709aa32835SJeff Kirsher 		if (ret)
22719aa32835SJeff Kirsher 			goto out_clean_pr;
22729aa32835SJeff Kirsher 	}
2273723f28e4SAnton Blanchard 	for (i = def_qps; i < def_qps; i++) {
22749aa32835SJeff Kirsher 		ret = ehea_init_port_res(port, &port->port_res[i],
22759aa32835SJeff Kirsher 					 &pr_cfg_small_rx, i);
22769aa32835SJeff Kirsher 		if (ret)
22779aa32835SJeff Kirsher 			goto out_clean_pr;
22789aa32835SJeff Kirsher 	}
22799aa32835SJeff Kirsher 
22809aa32835SJeff Kirsher 	return 0;
22819aa32835SJeff Kirsher 
22829aa32835SJeff Kirsher out_clean_pr:
22839aa32835SJeff Kirsher 	while (--i >= 0)
22849aa32835SJeff Kirsher 		ehea_clean_portres(port, &port->port_res[i]);
22859aa32835SJeff Kirsher 
22869aa32835SJeff Kirsher out_kill_eq:
22879aa32835SJeff Kirsher 	ehea_destroy_eq(port->qp_eq);
22889aa32835SJeff Kirsher 	return ret;
22899aa32835SJeff Kirsher }
22909aa32835SJeff Kirsher 
ehea_clean_all_portres(struct ehea_port * port)22919aa32835SJeff Kirsher static int ehea_clean_all_portres(struct ehea_port *port)
22929aa32835SJeff Kirsher {
22939aa32835SJeff Kirsher 	int ret = 0;
22949aa32835SJeff Kirsher 	int i;
22959aa32835SJeff Kirsher 
2296723f28e4SAnton Blanchard 	for (i = 0; i < port->num_def_qps; i++)
22979aa32835SJeff Kirsher 		ret |= ehea_clean_portres(port, &port->port_res[i]);
22989aa32835SJeff Kirsher 
22999aa32835SJeff Kirsher 	ret |= ehea_destroy_eq(port->qp_eq);
23009aa32835SJeff Kirsher 
23019aa32835SJeff Kirsher 	return ret;
23029aa32835SJeff Kirsher }
23039aa32835SJeff Kirsher 
ehea_remove_adapter_mr(struct ehea_adapter * adapter)23049aa32835SJeff Kirsher static void ehea_remove_adapter_mr(struct ehea_adapter *adapter)
23059aa32835SJeff Kirsher {
23069aa32835SJeff Kirsher 	if (adapter->active_ports)
23079aa32835SJeff Kirsher 		return;
23089aa32835SJeff Kirsher 
23099aa32835SJeff Kirsher 	ehea_rem_mr(&adapter->mr);
23109aa32835SJeff Kirsher }
23119aa32835SJeff Kirsher 
ehea_add_adapter_mr(struct ehea_adapter * adapter)23129aa32835SJeff Kirsher static int ehea_add_adapter_mr(struct ehea_adapter *adapter)
23139aa32835SJeff Kirsher {
23149aa32835SJeff Kirsher 	if (adapter->active_ports)
23159aa32835SJeff Kirsher 		return 0;
23169aa32835SJeff Kirsher 
23179aa32835SJeff Kirsher 	return ehea_reg_kernel_mr(adapter, &adapter->mr);
23189aa32835SJeff Kirsher }
23199aa32835SJeff Kirsher 
ehea_up(struct net_device * dev)23209aa32835SJeff Kirsher static int ehea_up(struct net_device *dev)
23219aa32835SJeff Kirsher {
23229aa32835SJeff Kirsher 	int ret, i;
23239aa32835SJeff Kirsher 	struct ehea_port *port = netdev_priv(dev);
23249aa32835SJeff Kirsher 
23259aa32835SJeff Kirsher 	if (port->state == EHEA_PORT_UP)
23269aa32835SJeff Kirsher 		return 0;
23279aa32835SJeff Kirsher 
2328723f28e4SAnton Blanchard 	ret = ehea_port_res_setup(port, port->num_def_qps);
23299aa32835SJeff Kirsher 	if (ret) {
23309aa32835SJeff Kirsher 		netdev_err(dev, "port_res_failed\n");
23319aa32835SJeff Kirsher 		goto out;
23329aa32835SJeff Kirsher 	}
23339aa32835SJeff Kirsher 
23349aa32835SJeff Kirsher 	/* Set default QP for this port */
23359aa32835SJeff Kirsher 	ret = ehea_configure_port(port);
23369aa32835SJeff Kirsher 	if (ret) {
23379aa32835SJeff Kirsher 		netdev_err(dev, "ehea_configure_port failed. ret:%d\n", ret);
23389aa32835SJeff Kirsher 		goto out_clean_pr;
23399aa32835SJeff Kirsher 	}
23409aa32835SJeff Kirsher 
23419aa32835SJeff Kirsher 	ret = ehea_reg_interrupts(dev);
23429aa32835SJeff Kirsher 	if (ret) {
23439aa32835SJeff Kirsher 		netdev_err(dev, "reg_interrupts failed. ret:%d\n", ret);
23449aa32835SJeff Kirsher 		goto out_clean_pr;
23459aa32835SJeff Kirsher 	}
23469aa32835SJeff Kirsher 
2347723f28e4SAnton Blanchard 	for (i = 0; i < port->num_def_qps; i++) {
23489aa32835SJeff Kirsher 		ret = ehea_activate_qp(port->adapter, port->port_res[i].qp);
23499aa32835SJeff Kirsher 		if (ret) {
23509aa32835SJeff Kirsher 			netdev_err(dev, "activate_qp failed\n");
23519aa32835SJeff Kirsher 			goto out_free_irqs;
23529aa32835SJeff Kirsher 		}
23539aa32835SJeff Kirsher 	}
23549aa32835SJeff Kirsher 
23559aa32835SJeff Kirsher 	for (i = 0; i < port->num_def_qps; i++) {
23569aa32835SJeff Kirsher 		ret = ehea_fill_port_res(&port->port_res[i]);
23579aa32835SJeff Kirsher 		if (ret) {
23589aa32835SJeff Kirsher 			netdev_err(dev, "out_free_irqs\n");
23599aa32835SJeff Kirsher 			goto out_free_irqs;
23609aa32835SJeff Kirsher 		}
23619aa32835SJeff Kirsher 	}
23629aa32835SJeff Kirsher 
23639aa32835SJeff Kirsher 	ret = ehea_broadcast_reg_helper(port, H_REG_BCMC);
23649aa32835SJeff Kirsher 	if (ret) {
23659aa32835SJeff Kirsher 		ret = -EIO;
23669aa32835SJeff Kirsher 		goto out_free_irqs;
23679aa32835SJeff Kirsher 	}
23689aa32835SJeff Kirsher 
23699aa32835SJeff Kirsher 	port->state = EHEA_PORT_UP;
23709aa32835SJeff Kirsher 
23719aa32835SJeff Kirsher 	ret = 0;
23729aa32835SJeff Kirsher 	goto out;
23739aa32835SJeff Kirsher 
23749aa32835SJeff Kirsher out_free_irqs:
23759aa32835SJeff Kirsher 	ehea_free_interrupts(dev);
23769aa32835SJeff Kirsher 
23779aa32835SJeff Kirsher out_clean_pr:
23789aa32835SJeff Kirsher 	ehea_clean_all_portres(port);
23799aa32835SJeff Kirsher out:
23809aa32835SJeff Kirsher 	if (ret)
23819aa32835SJeff Kirsher 		netdev_info(dev, "Failed starting. ret=%i\n", ret);
23829aa32835SJeff Kirsher 
23839aa32835SJeff Kirsher 	ehea_update_bcmc_registrations();
23849aa32835SJeff Kirsher 	ehea_update_firmware_handles();
23859aa32835SJeff Kirsher 
23869aa32835SJeff Kirsher 	return ret;
23879aa32835SJeff Kirsher }
23889aa32835SJeff Kirsher 
port_napi_disable(struct ehea_port * port)23899aa32835SJeff Kirsher static void port_napi_disable(struct ehea_port *port)
23909aa32835SJeff Kirsher {
23919aa32835SJeff Kirsher 	int i;
23929aa32835SJeff Kirsher 
2393723f28e4SAnton Blanchard 	for (i = 0; i < port->num_def_qps; i++)
23949aa32835SJeff Kirsher 		napi_disable(&port->port_res[i].napi);
23959aa32835SJeff Kirsher }
23969aa32835SJeff Kirsher 
port_napi_enable(struct ehea_port * port)23979aa32835SJeff Kirsher static void port_napi_enable(struct ehea_port *port)
23989aa32835SJeff Kirsher {
23999aa32835SJeff Kirsher 	int i;
24009aa32835SJeff Kirsher 
2401723f28e4SAnton Blanchard 	for (i = 0; i < port->num_def_qps; i++)
24029aa32835SJeff Kirsher 		napi_enable(&port->port_res[i].napi);
24039aa32835SJeff Kirsher }
24049aa32835SJeff Kirsher 
ehea_open(struct net_device * dev)24059aa32835SJeff Kirsher static int ehea_open(struct net_device *dev)
24069aa32835SJeff Kirsher {
24079aa32835SJeff Kirsher 	int ret;
24089aa32835SJeff Kirsher 	struct ehea_port *port = netdev_priv(dev);
24099aa32835SJeff Kirsher 
24109aa32835SJeff Kirsher 	mutex_lock(&port->port_lock);
24119aa32835SJeff Kirsher 
24129aa32835SJeff Kirsher 	netif_info(port, ifup, dev, "enabling port\n");
24139aa32835SJeff Kirsher 
241429ab5a3bSGuilherme G. Piccoli 	netif_carrier_off(dev);
241529ab5a3bSGuilherme G. Piccoli 
24169aa32835SJeff Kirsher 	ret = ehea_up(dev);
24179aa32835SJeff Kirsher 	if (!ret) {
24189aa32835SJeff Kirsher 		port_napi_enable(port);
2419b9564468SAnton Blanchard 		netif_tx_start_all_queues(dev);
24209aa32835SJeff Kirsher 	}
24219aa32835SJeff Kirsher 
24229aa32835SJeff Kirsher 	mutex_unlock(&port->port_lock);
242367c170a2SAnton Blanchard 	schedule_delayed_work(&port->stats_work,
242467c170a2SAnton Blanchard 			      round_jiffies_relative(msecs_to_jiffies(1000)));
24259aa32835SJeff Kirsher 
24269aa32835SJeff Kirsher 	return ret;
24279aa32835SJeff Kirsher }
24289aa32835SJeff Kirsher 
ehea_down(struct net_device * dev)24299aa32835SJeff Kirsher static int ehea_down(struct net_device *dev)
24309aa32835SJeff Kirsher {
24319aa32835SJeff Kirsher 	int ret;
24329aa32835SJeff Kirsher 	struct ehea_port *port = netdev_priv(dev);
24339aa32835SJeff Kirsher 
24349aa32835SJeff Kirsher 	if (port->state == EHEA_PORT_DOWN)
24359aa32835SJeff Kirsher 		return 0;
24369aa32835SJeff Kirsher 
24379aa32835SJeff Kirsher 	ehea_drop_multicast_list(dev);
24385d384574SThadeu Lima de Souza Cascardo 	ehea_allmulti(dev, 0);
24399aa32835SJeff Kirsher 	ehea_broadcast_reg_helper(port, H_DEREG_BCMC);
24409aa32835SJeff Kirsher 
24419aa32835SJeff Kirsher 	ehea_free_interrupts(dev);
24429aa32835SJeff Kirsher 
24439aa32835SJeff Kirsher 	port->state = EHEA_PORT_DOWN;
24449aa32835SJeff Kirsher 
24459aa32835SJeff Kirsher 	ehea_update_bcmc_registrations();
24469aa32835SJeff Kirsher 
24479aa32835SJeff Kirsher 	ret = ehea_clean_all_portres(port);
24489aa32835SJeff Kirsher 	if (ret)
24499aa32835SJeff Kirsher 		netdev_info(dev, "Failed freeing resources. ret=%i\n", ret);
24509aa32835SJeff Kirsher 
24519aa32835SJeff Kirsher 	ehea_update_firmware_handles();
24529aa32835SJeff Kirsher 
24539aa32835SJeff Kirsher 	return ret;
24549aa32835SJeff Kirsher }
24559aa32835SJeff Kirsher 
ehea_stop(struct net_device * dev)24569aa32835SJeff Kirsher static int ehea_stop(struct net_device *dev)
24579aa32835SJeff Kirsher {
24589aa32835SJeff Kirsher 	int ret;
24599aa32835SJeff Kirsher 	struct ehea_port *port = netdev_priv(dev);
24609aa32835SJeff Kirsher 
24619aa32835SJeff Kirsher 	netif_info(port, ifdown, dev, "disabling port\n");
24629aa32835SJeff Kirsher 
24639aa32835SJeff Kirsher 	set_bit(__EHEA_DISABLE_PORT_RESET, &port->flags);
24649aa32835SJeff Kirsher 	cancel_work_sync(&port->reset_task);
24652aefcad8Sbrenohl@br.ibm.com 	cancel_delayed_work_sync(&port->stats_work);
24669aa32835SJeff Kirsher 	mutex_lock(&port->port_lock);
2467b9564468SAnton Blanchard 	netif_tx_stop_all_queues(dev);
24689aa32835SJeff Kirsher 	port_napi_disable(port);
24699aa32835SJeff Kirsher 	ret = ehea_down(dev);
24709aa32835SJeff Kirsher 	mutex_unlock(&port->port_lock);
24719aa32835SJeff Kirsher 	clear_bit(__EHEA_DISABLE_PORT_RESET, &port->flags);
24729aa32835SJeff Kirsher 	return ret;
24739aa32835SJeff Kirsher }
24749aa32835SJeff Kirsher 
ehea_purge_sq(struct ehea_qp * orig_qp)24759aa32835SJeff Kirsher static void ehea_purge_sq(struct ehea_qp *orig_qp)
24769aa32835SJeff Kirsher {
24779aa32835SJeff Kirsher 	struct ehea_qp qp = *orig_qp;
24789aa32835SJeff Kirsher 	struct ehea_qp_init_attr *init_attr = &qp.init_attr;
24799aa32835SJeff Kirsher 	struct ehea_swqe *swqe;
24809aa32835SJeff Kirsher 	int wqe_index;
24819aa32835SJeff Kirsher 	int i;
24829aa32835SJeff Kirsher 
24839aa32835SJeff Kirsher 	for (i = 0; i < init_attr->act_nr_send_wqes; i++) {
24849aa32835SJeff Kirsher 		swqe = ehea_get_swqe(&qp, &wqe_index);
24859aa32835SJeff Kirsher 		swqe->tx_control |= EHEA_SWQE_PURGE;
24869aa32835SJeff Kirsher 	}
24879aa32835SJeff Kirsher }
24889aa32835SJeff Kirsher 
ehea_flush_sq(struct ehea_port * port)24899aa32835SJeff Kirsher static void ehea_flush_sq(struct ehea_port *port)
24909aa32835SJeff Kirsher {
24919aa32835SJeff Kirsher 	int i;
24929aa32835SJeff Kirsher 
2493723f28e4SAnton Blanchard 	for (i = 0; i < port->num_def_qps; i++) {
24949aa32835SJeff Kirsher 		struct ehea_port_res *pr = &port->port_res[i];
24959aa32835SJeff Kirsher 		int swqe_max = pr->sq_skba_size - 2 - pr->swqe_ll_count;
24969aa32835SJeff Kirsher 		int ret;
24979aa32835SJeff Kirsher 
24989aa32835SJeff Kirsher 		ret = wait_event_timeout(port->swqe_avail_wq,
24999aa32835SJeff Kirsher 			 atomic_read(&pr->swqe_avail) >= swqe_max,
25009aa32835SJeff Kirsher 			 msecs_to_jiffies(100));
25019aa32835SJeff Kirsher 
25029aa32835SJeff Kirsher 		if (!ret) {
25039aa32835SJeff Kirsher 			pr_err("WARNING: sq not flushed completely\n");
25049aa32835SJeff Kirsher 			break;
25059aa32835SJeff Kirsher 		}
25069aa32835SJeff Kirsher 	}
25079aa32835SJeff Kirsher }
25089aa32835SJeff Kirsher 
ehea_stop_qps(struct net_device * dev)25091886e5d2SThadeu Lima de Souza Cascardo static int ehea_stop_qps(struct net_device *dev)
25109aa32835SJeff Kirsher {
25119aa32835SJeff Kirsher 	struct ehea_port *port = netdev_priv(dev);
25129aa32835SJeff Kirsher 	struct ehea_adapter *adapter = port->adapter;
25139aa32835SJeff Kirsher 	struct hcp_modify_qp_cb0 *cb0;
25149aa32835SJeff Kirsher 	int ret = -EIO;
25159aa32835SJeff Kirsher 	int dret;
25169aa32835SJeff Kirsher 	int i;
25179aa32835SJeff Kirsher 	u64 hret;
25189aa32835SJeff Kirsher 	u64 dummy64 = 0;
25199aa32835SJeff Kirsher 	u16 dummy16 = 0;
25209aa32835SJeff Kirsher 
25219aa32835SJeff Kirsher 	cb0 = (void *)get_zeroed_page(GFP_KERNEL);
25229aa32835SJeff Kirsher 	if (!cb0) {
25239aa32835SJeff Kirsher 		ret = -ENOMEM;
25249aa32835SJeff Kirsher 		goto out;
25259aa32835SJeff Kirsher 	}
25269aa32835SJeff Kirsher 
2527723f28e4SAnton Blanchard 	for (i = 0; i < (port->num_def_qps); i++) {
25289aa32835SJeff Kirsher 		struct ehea_port_res *pr =  &port->port_res[i];
25299aa32835SJeff Kirsher 		struct ehea_qp *qp = pr->qp;
25309aa32835SJeff Kirsher 
25319aa32835SJeff Kirsher 		/* Purge send queue */
25329aa32835SJeff Kirsher 		ehea_purge_sq(qp);
25339aa32835SJeff Kirsher 
25349aa32835SJeff Kirsher 		/* Disable queue pair */
25359aa32835SJeff Kirsher 		hret = ehea_h_query_ehea_qp(adapter->handle, 0, qp->fw_handle,
25369aa32835SJeff Kirsher 					    EHEA_BMASK_SET(H_QPCB0_ALL, 0xFFFF),
25379aa32835SJeff Kirsher 					    cb0);
25389aa32835SJeff Kirsher 		if (hret != H_SUCCESS) {
25399aa32835SJeff Kirsher 			pr_err("query_ehea_qp failed (1)\n");
25409aa32835SJeff Kirsher 			goto out;
25419aa32835SJeff Kirsher 		}
25429aa32835SJeff Kirsher 
25439aa32835SJeff Kirsher 		cb0->qp_ctl_reg = (cb0->qp_ctl_reg & H_QP_CR_RES_STATE) << 8;
25449aa32835SJeff Kirsher 		cb0->qp_ctl_reg &= ~H_QP_CR_ENABLED;
25459aa32835SJeff Kirsher 
25469aa32835SJeff Kirsher 		hret = ehea_h_modify_ehea_qp(adapter->handle, 0, qp->fw_handle,
25479aa32835SJeff Kirsher 					     EHEA_BMASK_SET(H_QPCB0_QP_CTL_REG,
25489aa32835SJeff Kirsher 							    1), cb0, &dummy64,
25499aa32835SJeff Kirsher 					     &dummy64, &dummy16, &dummy16);
25509aa32835SJeff Kirsher 		if (hret != H_SUCCESS) {
25519aa32835SJeff Kirsher 			pr_err("modify_ehea_qp failed (1)\n");
25529aa32835SJeff Kirsher 			goto out;
25539aa32835SJeff Kirsher 		}
25549aa32835SJeff Kirsher 
25559aa32835SJeff Kirsher 		hret = ehea_h_query_ehea_qp(adapter->handle, 0, qp->fw_handle,
25569aa32835SJeff Kirsher 					    EHEA_BMASK_SET(H_QPCB0_ALL, 0xFFFF),
25579aa32835SJeff Kirsher 					    cb0);
25589aa32835SJeff Kirsher 		if (hret != H_SUCCESS) {
25599aa32835SJeff Kirsher 			pr_err("query_ehea_qp failed (2)\n");
25609aa32835SJeff Kirsher 			goto out;
25619aa32835SJeff Kirsher 		}
25629aa32835SJeff Kirsher 
25639aa32835SJeff Kirsher 		/* deregister shared memory regions */
25649aa32835SJeff Kirsher 		dret = ehea_rem_smrs(pr);
25659aa32835SJeff Kirsher 		if (dret) {
25669aa32835SJeff Kirsher 			pr_err("unreg shared memory region failed\n");
25679aa32835SJeff Kirsher 			goto out;
25689aa32835SJeff Kirsher 		}
25699aa32835SJeff Kirsher 	}
25709aa32835SJeff Kirsher 
25719aa32835SJeff Kirsher 	ret = 0;
25729aa32835SJeff Kirsher out:
25739aa32835SJeff Kirsher 	free_page((unsigned long)cb0);
25749aa32835SJeff Kirsher 
25759aa32835SJeff Kirsher 	return ret;
25769aa32835SJeff Kirsher }
25779aa32835SJeff Kirsher 
ehea_update_rqs(struct ehea_qp * orig_qp,struct ehea_port_res * pr)25781886e5d2SThadeu Lima de Souza Cascardo static void ehea_update_rqs(struct ehea_qp *orig_qp, struct ehea_port_res *pr)
25799aa32835SJeff Kirsher {
25809aa32835SJeff Kirsher 	struct ehea_qp qp = *orig_qp;
25819aa32835SJeff Kirsher 	struct ehea_qp_init_attr *init_attr = &qp.init_attr;
25829aa32835SJeff Kirsher 	struct ehea_rwqe *rwqe;
25839aa32835SJeff Kirsher 	struct sk_buff **skba_rq2 = pr->rq2_skba.arr;
25849aa32835SJeff Kirsher 	struct sk_buff **skba_rq3 = pr->rq3_skba.arr;
25859aa32835SJeff Kirsher 	struct sk_buff *skb;
25869aa32835SJeff Kirsher 	u32 lkey = pr->recv_mr.lkey;
25879aa32835SJeff Kirsher 
25889aa32835SJeff Kirsher 
25899aa32835SJeff Kirsher 	int i;
25909aa32835SJeff Kirsher 	int index;
25919aa32835SJeff Kirsher 
25929aa32835SJeff Kirsher 	for (i = 0; i < init_attr->act_nr_rwqes_rq2 + 1; i++) {
25939aa32835SJeff Kirsher 		rwqe = ehea_get_next_rwqe(&qp, 2);
25949aa32835SJeff Kirsher 		rwqe->sg_list[0].l_key = lkey;
25959aa32835SJeff Kirsher 		index = EHEA_BMASK_GET(EHEA_WR_ID_INDEX, rwqe->wr_id);
25969aa32835SJeff Kirsher 		skb = skba_rq2[index];
25979aa32835SJeff Kirsher 		if (skb)
25989aa32835SJeff Kirsher 			rwqe->sg_list[0].vaddr = ehea_map_vaddr(skb->data);
25999aa32835SJeff Kirsher 	}
26009aa32835SJeff Kirsher 
26019aa32835SJeff Kirsher 	for (i = 0; i < init_attr->act_nr_rwqes_rq3 + 1; i++) {
26029aa32835SJeff Kirsher 		rwqe = ehea_get_next_rwqe(&qp, 3);
26039aa32835SJeff Kirsher 		rwqe->sg_list[0].l_key = lkey;
26049aa32835SJeff Kirsher 		index = EHEA_BMASK_GET(EHEA_WR_ID_INDEX, rwqe->wr_id);
26059aa32835SJeff Kirsher 		skb = skba_rq3[index];
26069aa32835SJeff Kirsher 		if (skb)
26079aa32835SJeff Kirsher 			rwqe->sg_list[0].vaddr = ehea_map_vaddr(skb->data);
26089aa32835SJeff Kirsher 	}
26099aa32835SJeff Kirsher }
26109aa32835SJeff Kirsher 
ehea_restart_qps(struct net_device * dev)26111886e5d2SThadeu Lima de Souza Cascardo static int ehea_restart_qps(struct net_device *dev)
26129aa32835SJeff Kirsher {
26139aa32835SJeff Kirsher 	struct ehea_port *port = netdev_priv(dev);
26149aa32835SJeff Kirsher 	struct ehea_adapter *adapter = port->adapter;
26159aa32835SJeff Kirsher 	int ret = 0;
26169aa32835SJeff Kirsher 	int i;
26179aa32835SJeff Kirsher 
26189aa32835SJeff Kirsher 	struct hcp_modify_qp_cb0 *cb0;
26199aa32835SJeff Kirsher 	u64 hret;
26209aa32835SJeff Kirsher 	u64 dummy64 = 0;
26219aa32835SJeff Kirsher 	u16 dummy16 = 0;
26229aa32835SJeff Kirsher 
26239aa32835SJeff Kirsher 	cb0 = (void *)get_zeroed_page(GFP_KERNEL);
2624015dbf56SZhen Lei 	if (!cb0)
2625015dbf56SZhen Lei 		return -ENOMEM;
26269aa32835SJeff Kirsher 
2627723f28e4SAnton Blanchard 	for (i = 0; i < (port->num_def_qps); i++) {
26289aa32835SJeff Kirsher 		struct ehea_port_res *pr =  &port->port_res[i];
26299aa32835SJeff Kirsher 		struct ehea_qp *qp = pr->qp;
26309aa32835SJeff Kirsher 
26319aa32835SJeff Kirsher 		ret = ehea_gen_smrs(pr);
26329aa32835SJeff Kirsher 		if (ret) {
26339aa32835SJeff Kirsher 			netdev_err(dev, "creation of shared memory regions failed\n");
26349aa32835SJeff Kirsher 			goto out;
26359aa32835SJeff Kirsher 		}
26369aa32835SJeff Kirsher 
26379aa32835SJeff Kirsher 		ehea_update_rqs(qp, pr);
26389aa32835SJeff Kirsher 
26399aa32835SJeff Kirsher 		/* Enable queue pair */
26409aa32835SJeff Kirsher 		hret = ehea_h_query_ehea_qp(adapter->handle, 0, qp->fw_handle,
26419aa32835SJeff Kirsher 					    EHEA_BMASK_SET(H_QPCB0_ALL, 0xFFFF),
26429aa32835SJeff Kirsher 					    cb0);
26439aa32835SJeff Kirsher 		if (hret != H_SUCCESS) {
26449aa32835SJeff Kirsher 			netdev_err(dev, "query_ehea_qp failed (1)\n");
2645015dbf56SZhen Lei 			ret = -EFAULT;
26469aa32835SJeff Kirsher 			goto out;
26479aa32835SJeff Kirsher 		}
26489aa32835SJeff Kirsher 
26499aa32835SJeff Kirsher 		cb0->qp_ctl_reg = (cb0->qp_ctl_reg & H_QP_CR_RES_STATE) << 8;
26509aa32835SJeff Kirsher 		cb0->qp_ctl_reg |= H_QP_CR_ENABLED;
26519aa32835SJeff Kirsher 
26529aa32835SJeff Kirsher 		hret = ehea_h_modify_ehea_qp(adapter->handle, 0, qp->fw_handle,
26539aa32835SJeff Kirsher 					     EHEA_BMASK_SET(H_QPCB0_QP_CTL_REG,
26549aa32835SJeff Kirsher 							    1), cb0, &dummy64,
26559aa32835SJeff Kirsher 					     &dummy64, &dummy16, &dummy16);
26569aa32835SJeff Kirsher 		if (hret != H_SUCCESS) {
26579aa32835SJeff Kirsher 			netdev_err(dev, "modify_ehea_qp failed (1)\n");
2658015dbf56SZhen Lei 			ret = -EFAULT;
26599aa32835SJeff Kirsher 			goto out;
26609aa32835SJeff Kirsher 		}
26619aa32835SJeff Kirsher 
26629aa32835SJeff Kirsher 		hret = ehea_h_query_ehea_qp(adapter->handle, 0, qp->fw_handle,
26639aa32835SJeff Kirsher 					    EHEA_BMASK_SET(H_QPCB0_ALL, 0xFFFF),
26649aa32835SJeff Kirsher 					    cb0);
26659aa32835SJeff Kirsher 		if (hret != H_SUCCESS) {
26669aa32835SJeff Kirsher 			netdev_err(dev, "query_ehea_qp failed (2)\n");
2667015dbf56SZhen Lei 			ret = -EFAULT;
26689aa32835SJeff Kirsher 			goto out;
26699aa32835SJeff Kirsher 		}
26709aa32835SJeff Kirsher 
26719aa32835SJeff Kirsher 		/* refill entire queue */
26729aa32835SJeff Kirsher 		ehea_refill_rq1(pr, pr->rq1_skba.index, 0);
26739aa32835SJeff Kirsher 		ehea_refill_rq2(pr, 0);
26749aa32835SJeff Kirsher 		ehea_refill_rq3(pr, 0);
26759aa32835SJeff Kirsher 	}
26769aa32835SJeff Kirsher out:
26779aa32835SJeff Kirsher 	free_page((unsigned long)cb0);
26789aa32835SJeff Kirsher 
26799aa32835SJeff Kirsher 	return ret;
26809aa32835SJeff Kirsher }
26819aa32835SJeff Kirsher 
ehea_reset_port(struct work_struct * work)26829aa32835SJeff Kirsher static void ehea_reset_port(struct work_struct *work)
26839aa32835SJeff Kirsher {
26849aa32835SJeff Kirsher 	int ret;
26859aa32835SJeff Kirsher 	struct ehea_port *port =
26869aa32835SJeff Kirsher 		container_of(work, struct ehea_port, reset_task);
26879aa32835SJeff Kirsher 	struct net_device *dev = port->netdev;
26889aa32835SJeff Kirsher 
26899aa32835SJeff Kirsher 	mutex_lock(&dlpar_mem_lock);
26909aa32835SJeff Kirsher 	port->resets++;
26919aa32835SJeff Kirsher 	mutex_lock(&port->port_lock);
2692b9564468SAnton Blanchard 	netif_tx_disable(dev);
26939aa32835SJeff Kirsher 
26949aa32835SJeff Kirsher 	port_napi_disable(port);
26959aa32835SJeff Kirsher 
26969aa32835SJeff Kirsher 	ehea_down(dev);
26979aa32835SJeff Kirsher 
26989aa32835SJeff Kirsher 	ret = ehea_up(dev);
26999aa32835SJeff Kirsher 	if (ret)
27009aa32835SJeff Kirsher 		goto out;
27019aa32835SJeff Kirsher 
27029aa32835SJeff Kirsher 	ehea_set_multicast_list(dev);
27039aa32835SJeff Kirsher 
27049aa32835SJeff Kirsher 	netif_info(port, timer, dev, "reset successful\n");
27059aa32835SJeff Kirsher 
27069aa32835SJeff Kirsher 	port_napi_enable(port);
27079aa32835SJeff Kirsher 
2708b9564468SAnton Blanchard 	netif_tx_wake_all_queues(dev);
27099aa32835SJeff Kirsher out:
27109aa32835SJeff Kirsher 	mutex_unlock(&port->port_lock);
27119aa32835SJeff Kirsher 	mutex_unlock(&dlpar_mem_lock);
27129aa32835SJeff Kirsher }
27139aa32835SJeff Kirsher 
ehea_rereg_mrs(void)27149aa32835SJeff Kirsher static void ehea_rereg_mrs(void)
27159aa32835SJeff Kirsher {
27169aa32835SJeff Kirsher 	int ret, i;
27179aa32835SJeff Kirsher 	struct ehea_adapter *adapter;
27189aa32835SJeff Kirsher 
27199aa32835SJeff Kirsher 	pr_info("LPAR memory changed - re-initializing driver\n");
27209aa32835SJeff Kirsher 
27219aa32835SJeff Kirsher 	list_for_each_entry(adapter, &adapter_list, list)
27229aa32835SJeff Kirsher 		if (adapter->active_ports) {
27239aa32835SJeff Kirsher 			/* Shutdown all ports */
27249aa32835SJeff Kirsher 			for (i = 0; i < EHEA_MAX_PORTS; i++) {
27259aa32835SJeff Kirsher 				struct ehea_port *port = adapter->port[i];
27269aa32835SJeff Kirsher 				struct net_device *dev;
27279aa32835SJeff Kirsher 
27289aa32835SJeff Kirsher 				if (!port)
27299aa32835SJeff Kirsher 					continue;
27309aa32835SJeff Kirsher 
27319aa32835SJeff Kirsher 				dev = port->netdev;
27329aa32835SJeff Kirsher 
27339aa32835SJeff Kirsher 				if (dev->flags & IFF_UP) {
27349aa32835SJeff Kirsher 					mutex_lock(&port->port_lock);
2735b9564468SAnton Blanchard 					netif_tx_disable(dev);
27369aa32835SJeff Kirsher 					ehea_flush_sq(port);
27379aa32835SJeff Kirsher 					ret = ehea_stop_qps(dev);
27389aa32835SJeff Kirsher 					if (ret) {
27399aa32835SJeff Kirsher 						mutex_unlock(&port->port_lock);
27409aa32835SJeff Kirsher 						goto out;
27419aa32835SJeff Kirsher 					}
27429aa32835SJeff Kirsher 					port_napi_disable(port);
27439aa32835SJeff Kirsher 					mutex_unlock(&port->port_lock);
27449aa32835SJeff Kirsher 				}
27459aa32835SJeff Kirsher 				reset_sq_restart_flag(port);
27469aa32835SJeff Kirsher 			}
27479aa32835SJeff Kirsher 
27489aa32835SJeff Kirsher 			/* Unregister old memory region */
27499aa32835SJeff Kirsher 			ret = ehea_rem_mr(&adapter->mr);
27509aa32835SJeff Kirsher 			if (ret) {
27519aa32835SJeff Kirsher 				pr_err("unregister MR failed - driver inoperable!\n");
27529aa32835SJeff Kirsher 				goto out;
27539aa32835SJeff Kirsher 			}
27549aa32835SJeff Kirsher 		}
27559aa32835SJeff Kirsher 
27569aa32835SJeff Kirsher 	clear_bit(__EHEA_STOP_XFER, &ehea_driver_flags);
27579aa32835SJeff Kirsher 
27589aa32835SJeff Kirsher 	list_for_each_entry(adapter, &adapter_list, list)
27599aa32835SJeff Kirsher 		if (adapter->active_ports) {
27609aa32835SJeff Kirsher 			/* Register new memory region */
27619aa32835SJeff Kirsher 			ret = ehea_reg_kernel_mr(adapter, &adapter->mr);
27629aa32835SJeff Kirsher 			if (ret) {
27639aa32835SJeff Kirsher 				pr_err("register MR failed - driver inoperable!\n");
27649aa32835SJeff Kirsher 				goto out;
27659aa32835SJeff Kirsher 			}
27669aa32835SJeff Kirsher 
27679aa32835SJeff Kirsher 			/* Restart all ports */
27689aa32835SJeff Kirsher 			for (i = 0; i < EHEA_MAX_PORTS; i++) {
27699aa32835SJeff Kirsher 				struct ehea_port *port = adapter->port[i];
27709aa32835SJeff Kirsher 
27719aa32835SJeff Kirsher 				if (port) {
27729aa32835SJeff Kirsher 					struct net_device *dev = port->netdev;
27739aa32835SJeff Kirsher 
27749aa32835SJeff Kirsher 					if (dev->flags & IFF_UP) {
27759aa32835SJeff Kirsher 						mutex_lock(&port->port_lock);
27769aa32835SJeff Kirsher 						ret = ehea_restart_qps(dev);
27779aa32835SJeff Kirsher 						if (!ret) {
27789aa32835SJeff Kirsher 							check_sqs(port);
27799aa32835SJeff Kirsher 							port_napi_enable(port);
2780b9564468SAnton Blanchard 							netif_tx_wake_all_queues(dev);
27819aa32835SJeff Kirsher 						} else {
27829aa32835SJeff Kirsher 							netdev_err(dev, "Unable to restart QPS\n");
27839aa32835SJeff Kirsher 						}
27849aa32835SJeff Kirsher 						mutex_unlock(&port->port_lock);
27859aa32835SJeff Kirsher 					}
27869aa32835SJeff Kirsher 				}
27879aa32835SJeff Kirsher 			}
27889aa32835SJeff Kirsher 		}
27899aa32835SJeff Kirsher 	pr_info("re-initializing driver complete\n");
27909aa32835SJeff Kirsher out:
27919aa32835SJeff Kirsher 	return;
27929aa32835SJeff Kirsher }
27939aa32835SJeff Kirsher 
ehea_tx_watchdog(struct net_device * dev,unsigned int txqueue)27940290bd29SMichael S. Tsirkin static void ehea_tx_watchdog(struct net_device *dev, unsigned int txqueue)
27959aa32835SJeff Kirsher {
27969aa32835SJeff Kirsher 	struct ehea_port *port = netdev_priv(dev);
27979aa32835SJeff Kirsher 
27989aa32835SJeff Kirsher 	if (netif_carrier_ok(dev) &&
27999aa32835SJeff Kirsher 	    !test_bit(__EHEA_STOP_XFER, &ehea_driver_flags))
28009aa32835SJeff Kirsher 		ehea_schedule_port_reset(port);
28019aa32835SJeff Kirsher }
28029aa32835SJeff Kirsher 
ehea_sense_adapter_attr(struct ehea_adapter * adapter)28031886e5d2SThadeu Lima de Souza Cascardo static int ehea_sense_adapter_attr(struct ehea_adapter *adapter)
28049aa32835SJeff Kirsher {
28059aa32835SJeff Kirsher 	struct hcp_query_ehea *cb;
28069aa32835SJeff Kirsher 	u64 hret;
28079aa32835SJeff Kirsher 	int ret;
28089aa32835SJeff Kirsher 
28099aa32835SJeff Kirsher 	cb = (void *)get_zeroed_page(GFP_KERNEL);
28109aa32835SJeff Kirsher 	if (!cb) {
28119aa32835SJeff Kirsher 		ret = -ENOMEM;
28129aa32835SJeff Kirsher 		goto out;
28139aa32835SJeff Kirsher 	}
28149aa32835SJeff Kirsher 
28159aa32835SJeff Kirsher 	hret = ehea_h_query_ehea(adapter->handle, cb);
28169aa32835SJeff Kirsher 
28179aa32835SJeff Kirsher 	if (hret != H_SUCCESS) {
28189aa32835SJeff Kirsher 		ret = -EIO;
28199aa32835SJeff Kirsher 		goto out_herr;
28209aa32835SJeff Kirsher 	}
28219aa32835SJeff Kirsher 
28229aa32835SJeff Kirsher 	adapter->max_mc_mac = cb->max_mc_mac - 1;
28239aa32835SJeff Kirsher 	ret = 0;
28249aa32835SJeff Kirsher 
28259aa32835SJeff Kirsher out_herr:
28269aa32835SJeff Kirsher 	free_page((unsigned long)cb);
28279aa32835SJeff Kirsher out:
28289aa32835SJeff Kirsher 	return ret;
28299aa32835SJeff Kirsher }
28309aa32835SJeff Kirsher 
ehea_get_jumboframe_status(struct ehea_port * port,int * jumbo)28311886e5d2SThadeu Lima de Souza Cascardo static int ehea_get_jumboframe_status(struct ehea_port *port, int *jumbo)
28329aa32835SJeff Kirsher {
28339aa32835SJeff Kirsher 	struct hcp_ehea_port_cb4 *cb4;
28349aa32835SJeff Kirsher 	u64 hret;
28359aa32835SJeff Kirsher 	int ret = 0;
28369aa32835SJeff Kirsher 
28379aa32835SJeff Kirsher 	*jumbo = 0;
28389aa32835SJeff Kirsher 
28399aa32835SJeff Kirsher 	/* (Try to) enable *jumbo frames */
28409aa32835SJeff Kirsher 	cb4 = (void *)get_zeroed_page(GFP_KERNEL);
28419aa32835SJeff Kirsher 	if (!cb4) {
28429aa32835SJeff Kirsher 		pr_err("no mem for cb4\n");
28439aa32835SJeff Kirsher 		ret = -ENOMEM;
28449aa32835SJeff Kirsher 		goto out;
28459aa32835SJeff Kirsher 	} else {
28469aa32835SJeff Kirsher 		hret = ehea_h_query_ehea_port(port->adapter->handle,
28479aa32835SJeff Kirsher 					      port->logical_port_id,
28489aa32835SJeff Kirsher 					      H_PORT_CB4,
28499aa32835SJeff Kirsher 					      H_PORT_CB4_JUMBO, cb4);
28509aa32835SJeff Kirsher 		if (hret == H_SUCCESS) {
28519aa32835SJeff Kirsher 			if (cb4->jumbo_frame)
28529aa32835SJeff Kirsher 				*jumbo = 1;
28539aa32835SJeff Kirsher 			else {
28549aa32835SJeff Kirsher 				cb4->jumbo_frame = 1;
28559aa32835SJeff Kirsher 				hret = ehea_h_modify_ehea_port(port->adapter->
28569aa32835SJeff Kirsher 							       handle,
28579aa32835SJeff Kirsher 							       port->
28589aa32835SJeff Kirsher 							       logical_port_id,
28599aa32835SJeff Kirsher 							       H_PORT_CB4,
28609aa32835SJeff Kirsher 							       H_PORT_CB4_JUMBO,
28619aa32835SJeff Kirsher 							       cb4);
28629aa32835SJeff Kirsher 				if (hret == H_SUCCESS)
28639aa32835SJeff Kirsher 					*jumbo = 1;
28649aa32835SJeff Kirsher 			}
28659aa32835SJeff Kirsher 		} else
28669aa32835SJeff Kirsher 			ret = -EINVAL;
28679aa32835SJeff Kirsher 
28689aa32835SJeff Kirsher 		free_page((unsigned long)cb4);
28699aa32835SJeff Kirsher 	}
28709aa32835SJeff Kirsher out:
28719aa32835SJeff Kirsher 	return ret;
28729aa32835SJeff Kirsher }
28739aa32835SJeff Kirsher 
log_port_id_show(struct device * dev,struct device_attribute * attr,char * buf)28740056982fSYueHaibing static ssize_t log_port_id_show(struct device *dev,
28759aa32835SJeff Kirsher 				struct device_attribute *attr, char *buf)
28769aa32835SJeff Kirsher {
28779aa32835SJeff Kirsher 	struct ehea_port *port = container_of(dev, struct ehea_port, ofdev.dev);
28789aa32835SJeff Kirsher 	return sprintf(buf, "%d", port->logical_port_id);
28799aa32835SJeff Kirsher }
28809aa32835SJeff Kirsher 
28810056982fSYueHaibing static DEVICE_ATTR_RO(log_port_id);
28829aa32835SJeff Kirsher 
logical_port_release(struct device * dev)28830297be07SBill Pemberton static void logical_port_release(struct device *dev)
28849aa32835SJeff Kirsher {
28859aa32835SJeff Kirsher 	struct ehea_port *port = container_of(dev, struct ehea_port, ofdev.dev);
28869aa32835SJeff Kirsher 	of_node_put(port->ofdev.dev.of_node);
28879aa32835SJeff Kirsher }
28889aa32835SJeff Kirsher 
ehea_register_port(struct ehea_port * port,struct device_node * dn)28899aa32835SJeff Kirsher static struct device *ehea_register_port(struct ehea_port *port,
28909aa32835SJeff Kirsher 					 struct device_node *dn)
28919aa32835SJeff Kirsher {
28929aa32835SJeff Kirsher 	int ret;
28939aa32835SJeff Kirsher 
28949aa32835SJeff Kirsher 	port->ofdev.dev.of_node = of_node_get(dn);
28959aa32835SJeff Kirsher 	port->ofdev.dev.parent = &port->adapter->ofdev->dev;
28969aa32835SJeff Kirsher 	port->ofdev.dev.bus = &ibmebus_bus_type;
28979aa32835SJeff Kirsher 
28989aa32835SJeff Kirsher 	dev_set_name(&port->ofdev.dev, "port%d", port_name_cnt++);
28999aa32835SJeff Kirsher 	port->ofdev.dev.release = logical_port_release;
29009aa32835SJeff Kirsher 
29019aa32835SJeff Kirsher 	ret = of_device_register(&port->ofdev);
29029aa32835SJeff Kirsher 	if (ret) {
29039aa32835SJeff Kirsher 		pr_err("failed to register device. ret=%d\n", ret);
29040e7ce23aSYang Yingliang 		put_device(&port->ofdev.dev);
29059aa32835SJeff Kirsher 		goto out;
29069aa32835SJeff Kirsher 	}
29079aa32835SJeff Kirsher 
29089aa32835SJeff Kirsher 	ret = device_create_file(&port->ofdev.dev, &dev_attr_log_port_id);
29099aa32835SJeff Kirsher 	if (ret) {
29109aa32835SJeff Kirsher 		pr_err("failed to register attributes, ret=%d\n", ret);
29119aa32835SJeff Kirsher 		goto out_unreg_of_dev;
29129aa32835SJeff Kirsher 	}
29139aa32835SJeff Kirsher 
29149aa32835SJeff Kirsher 	return &port->ofdev.dev;
29159aa32835SJeff Kirsher 
29169aa32835SJeff Kirsher out_unreg_of_dev:
29179aa32835SJeff Kirsher 	of_device_unregister(&port->ofdev);
29189aa32835SJeff Kirsher out:
29199aa32835SJeff Kirsher 	return NULL;
29209aa32835SJeff Kirsher }
29219aa32835SJeff Kirsher 
ehea_unregister_port(struct ehea_port * port)29229aa32835SJeff Kirsher static void ehea_unregister_port(struct ehea_port *port)
29239aa32835SJeff Kirsher {
29249aa32835SJeff Kirsher 	device_remove_file(&port->ofdev.dev, &dev_attr_log_port_id);
29259aa32835SJeff Kirsher 	of_device_unregister(&port->ofdev);
29269aa32835SJeff Kirsher }
29279aa32835SJeff Kirsher 
29289aa32835SJeff Kirsher static const struct net_device_ops ehea_netdev_ops = {
29299aa32835SJeff Kirsher 	.ndo_open		= ehea_open,
29309aa32835SJeff Kirsher 	.ndo_stop		= ehea_stop,
29319aa32835SJeff Kirsher 	.ndo_start_xmit		= ehea_start_xmit,
2932239c562cSAnton Blanchard 	.ndo_get_stats64	= ehea_get_stats64,
29339aa32835SJeff Kirsher 	.ndo_set_mac_address	= ehea_set_mac_addr,
29349aa32835SJeff Kirsher 	.ndo_validate_addr	= eth_validate_addr,
2935afc4b13dSJiri Pirko 	.ndo_set_rx_mode	= ehea_set_multicast_list,
29369aa32835SJeff Kirsher 	.ndo_vlan_rx_add_vid	= ehea_vlan_rx_add_vid,
29379aa32835SJeff Kirsher 	.ndo_vlan_rx_kill_vid	= ehea_vlan_rx_kill_vid,
29389aa32835SJeff Kirsher 	.ndo_tx_timeout		= ehea_tx_watchdog,
29399aa32835SJeff Kirsher };
29409aa32835SJeff Kirsher 
ehea_setup_single_port(struct ehea_adapter * adapter,u32 logical_port_id,struct device_node * dn)29411886e5d2SThadeu Lima de Souza Cascardo static struct ehea_port *ehea_setup_single_port(struct ehea_adapter *adapter,
29429aa32835SJeff Kirsher 					 u32 logical_port_id,
29439aa32835SJeff Kirsher 					 struct device_node *dn)
29449aa32835SJeff Kirsher {
29459aa32835SJeff Kirsher 	int ret;
29469aa32835SJeff Kirsher 	struct net_device *dev;
29479aa32835SJeff Kirsher 	struct ehea_port *port;
29489aa32835SJeff Kirsher 	struct device *port_dev;
29499aa32835SJeff Kirsher 	int jumbo;
29509aa32835SJeff Kirsher 
29519aa32835SJeff Kirsher 	/* allocate memory for the port structures */
2952b9564468SAnton Blanchard 	dev = alloc_etherdev_mq(sizeof(struct ehea_port), EHEA_MAX_PORT_RES);
29539aa32835SJeff Kirsher 
29549aa32835SJeff Kirsher 	if (!dev) {
29559aa32835SJeff Kirsher 		ret = -ENOMEM;
29569aa32835SJeff Kirsher 		goto out_err;
29579aa32835SJeff Kirsher 	}
29589aa32835SJeff Kirsher 
29599aa32835SJeff Kirsher 	port = netdev_priv(dev);
29609aa32835SJeff Kirsher 
29619aa32835SJeff Kirsher 	mutex_init(&port->port_lock);
29629aa32835SJeff Kirsher 	port->state = EHEA_PORT_DOWN;
29639aa32835SJeff Kirsher 	port->sig_comp_iv = sq_entries / 10;
29649aa32835SJeff Kirsher 
29659aa32835SJeff Kirsher 	port->adapter = adapter;
29669aa32835SJeff Kirsher 	port->netdev = dev;
29679aa32835SJeff Kirsher 	port->logical_port_id = logical_port_id;
29689aa32835SJeff Kirsher 
29699aa32835SJeff Kirsher 	port->msg_enable = netif_msg_init(msg_level, EHEA_MSG_DEFAULT);
29709aa32835SJeff Kirsher 
29719aa32835SJeff Kirsher 	port->mc_list = kzalloc(sizeof(struct ehea_mc_list), GFP_KERNEL);
29729aa32835SJeff Kirsher 	if (!port->mc_list) {
29739aa32835SJeff Kirsher 		ret = -ENOMEM;
29749aa32835SJeff Kirsher 		goto out_free_ethdev;
29759aa32835SJeff Kirsher 	}
29769aa32835SJeff Kirsher 
29779aa32835SJeff Kirsher 	INIT_LIST_HEAD(&port->mc_list->list);
29789aa32835SJeff Kirsher 
29799aa32835SJeff Kirsher 	ret = ehea_sense_port_attr(port);
29809aa32835SJeff Kirsher 	if (ret)
29819aa32835SJeff Kirsher 		goto out_free_mc_list;
29829aa32835SJeff Kirsher 
2983b9564468SAnton Blanchard 	netif_set_real_num_rx_queues(dev, port->num_def_qps);
2984723f28e4SAnton Blanchard 	netif_set_real_num_tx_queues(dev, port->num_def_qps);
2985b9564468SAnton Blanchard 
29869aa32835SJeff Kirsher 	port_dev = ehea_register_port(port, dn);
29879aa32835SJeff Kirsher 	if (!port_dev)
29889aa32835SJeff Kirsher 		goto out_free_mc_list;
29899aa32835SJeff Kirsher 
29909aa32835SJeff Kirsher 	SET_NETDEV_DEV(dev, port_dev);
29919aa32835SJeff Kirsher 
29929aa32835SJeff Kirsher 	/* initialize net_device structure */
2993ceca777dSJakub Kicinski 	eth_hw_addr_set(dev, (u8 *)&port->mac_addr);
29949aa32835SJeff Kirsher 
29959aa32835SJeff Kirsher 	dev->netdev_ops = &ehea_netdev_ops;
29969aa32835SJeff Kirsher 	ehea_set_ethtool_ops(dev);
29979aa32835SJeff Kirsher 
299855fbbe46SDavid S. Miller 	dev->hw_features = NETIF_F_SG | NETIF_F_TSO |
299955fbbe46SDavid S. Miller 		      NETIF_F_IP_CSUM | NETIF_F_HW_VLAN_CTAG_TX;
300028e24c62SEric Dumazet 	dev->features = NETIF_F_SG | NETIF_F_TSO |
300155fbbe46SDavid S. Miller 		      NETIF_F_HIGHDMA | NETIF_F_IP_CSUM |
300255fbbe46SDavid S. Miller 		      NETIF_F_HW_VLAN_CTAG_TX | NETIF_F_HW_VLAN_CTAG_RX |
300355fbbe46SDavid S. Miller 		      NETIF_F_HW_VLAN_CTAG_FILTER | NETIF_F_RXCSUM;
3004076f2032SAnton Blanchard 	dev->vlan_features = NETIF_F_SG | NETIF_F_TSO | NETIF_F_HIGHDMA |
3005076f2032SAnton Blanchard 			NETIF_F_IP_CSUM;
30069aa32835SJeff Kirsher 	dev->watchdog_timeo = EHEA_WATCH_DOG_TIMEOUT;
30079aa32835SJeff Kirsher 
30083d5d96acSJarod Wilson 	/* MTU range: 68 - 9022 */
30093d5d96acSJarod Wilson 	dev->min_mtu = ETH_MIN_MTU;
30103d5d96acSJarod Wilson 	dev->max_mtu = EHEA_MAX_PACKET_SIZE;
30113d5d96acSJarod Wilson 
30129aa32835SJeff Kirsher 	INIT_WORK(&port->reset_task, ehea_reset_port);
30132aefcad8Sbrenohl@br.ibm.com 	INIT_DELAYED_WORK(&port->stats_work, ehea_update_stats);
30149aa32835SJeff Kirsher 
30159aa32835SJeff Kirsher 	init_waitqueue_head(&port->swqe_avail_wq);
30169aa32835SJeff Kirsher 	init_waitqueue_head(&port->restart_wq);
30179aa32835SJeff Kirsher 
30189aa32835SJeff Kirsher 	ret = register_netdev(dev);
30199aa32835SJeff Kirsher 	if (ret) {
30209aa32835SJeff Kirsher 		pr_err("register_netdev failed. ret=%d\n", ret);
30219aa32835SJeff Kirsher 		goto out_unreg_port;
30229aa32835SJeff Kirsher 	}
30239aa32835SJeff Kirsher 
30249aa32835SJeff Kirsher 	ret = ehea_get_jumboframe_status(port, &jumbo);
30259aa32835SJeff Kirsher 	if (ret)
30269aa32835SJeff Kirsher 		netdev_err(dev, "failed determining jumbo frame status\n");
30279aa32835SJeff Kirsher 
30289aa32835SJeff Kirsher 	netdev_info(dev, "Jumbo frames are %sabled\n",
30299aa32835SJeff Kirsher 		    jumbo == 1 ? "en" : "dis");
30309aa32835SJeff Kirsher 
30319aa32835SJeff Kirsher 	adapter->active_ports++;
30329aa32835SJeff Kirsher 
30339aa32835SJeff Kirsher 	return port;
30349aa32835SJeff Kirsher 
30359aa32835SJeff Kirsher out_unreg_port:
30369aa32835SJeff Kirsher 	ehea_unregister_port(port);
30379aa32835SJeff Kirsher 
30389aa32835SJeff Kirsher out_free_mc_list:
30399aa32835SJeff Kirsher 	kfree(port->mc_list);
30409aa32835SJeff Kirsher 
30419aa32835SJeff Kirsher out_free_ethdev:
30429aa32835SJeff Kirsher 	free_netdev(dev);
30439aa32835SJeff Kirsher 
30449aa32835SJeff Kirsher out_err:
30459aa32835SJeff Kirsher 	pr_err("setting up logical port with id=%d failed, ret=%d\n",
30469aa32835SJeff Kirsher 	       logical_port_id, ret);
30479aa32835SJeff Kirsher 	return NULL;
30489aa32835SJeff Kirsher }
30499aa32835SJeff Kirsher 
ehea_shutdown_single_port(struct ehea_port * port)30509aa32835SJeff Kirsher static void ehea_shutdown_single_port(struct ehea_port *port)
30519aa32835SJeff Kirsher {
30529aa32835SJeff Kirsher 	struct ehea_adapter *adapter = port->adapter;
30539aa32835SJeff Kirsher 
30549aa32835SJeff Kirsher 	cancel_work_sync(&port->reset_task);
30552aefcad8Sbrenohl@br.ibm.com 	cancel_delayed_work_sync(&port->stats_work);
30569aa32835SJeff Kirsher 	unregister_netdev(port->netdev);
30579aa32835SJeff Kirsher 	ehea_unregister_port(port);
30589aa32835SJeff Kirsher 	kfree(port->mc_list);
30599aa32835SJeff Kirsher 	free_netdev(port->netdev);
30609aa32835SJeff Kirsher 	adapter->active_ports--;
30619aa32835SJeff Kirsher }
30629aa32835SJeff Kirsher 
ehea_setup_ports(struct ehea_adapter * adapter)30639aa32835SJeff Kirsher static int ehea_setup_ports(struct ehea_adapter *adapter)
30649aa32835SJeff Kirsher {
30659aa32835SJeff Kirsher 	struct device_node *lhea_dn;
30669aa32835SJeff Kirsher 	struct device_node *eth_dn = NULL;
30679aa32835SJeff Kirsher 
30689aa32835SJeff Kirsher 	const u32 *dn_log_port_id;
30699aa32835SJeff Kirsher 	int i = 0;
30709aa32835SJeff Kirsher 
30719aa32835SJeff Kirsher 	lhea_dn = adapter->ofdev->dev.of_node;
30729aa32835SJeff Kirsher 	while ((eth_dn = of_get_next_child(lhea_dn, eth_dn))) {
30739aa32835SJeff Kirsher 
30749aa32835SJeff Kirsher 		dn_log_port_id = of_get_property(eth_dn, "ibm,hea-port-no",
30759aa32835SJeff Kirsher 						 NULL);
30769aa32835SJeff Kirsher 		if (!dn_log_port_id) {
3077f7ce9103SRob Herring 			pr_err("bad device node: eth_dn name=%pOF\n", eth_dn);
30789aa32835SJeff Kirsher 			continue;
30799aa32835SJeff Kirsher 		}
30809aa32835SJeff Kirsher 
30819aa32835SJeff Kirsher 		if (ehea_add_adapter_mr(adapter)) {
30829aa32835SJeff Kirsher 			pr_err("creating MR failed\n");
30839aa32835SJeff Kirsher 			of_node_put(eth_dn);
30849aa32835SJeff Kirsher 			return -EIO;
30859aa32835SJeff Kirsher 		}
30869aa32835SJeff Kirsher 
30879aa32835SJeff Kirsher 		adapter->port[i] = ehea_setup_single_port(adapter,
30889aa32835SJeff Kirsher 							  *dn_log_port_id,
30899aa32835SJeff Kirsher 							  eth_dn);
30909aa32835SJeff Kirsher 		if (adapter->port[i])
30919aa32835SJeff Kirsher 			netdev_info(adapter->port[i]->netdev,
30929aa32835SJeff Kirsher 				    "logical port id #%d\n", *dn_log_port_id);
30939aa32835SJeff Kirsher 		else
30949aa32835SJeff Kirsher 			ehea_remove_adapter_mr(adapter);
30959aa32835SJeff Kirsher 
30969aa32835SJeff Kirsher 		i++;
30979aa32835SJeff Kirsher 	}
30989aa32835SJeff Kirsher 	return 0;
30999aa32835SJeff Kirsher }
31009aa32835SJeff Kirsher 
ehea_get_eth_dn(struct ehea_adapter * adapter,u32 logical_port_id)31019aa32835SJeff Kirsher static struct device_node *ehea_get_eth_dn(struct ehea_adapter *adapter,
31029aa32835SJeff Kirsher 					   u32 logical_port_id)
31039aa32835SJeff Kirsher {
31049aa32835SJeff Kirsher 	struct device_node *lhea_dn;
31059aa32835SJeff Kirsher 	struct device_node *eth_dn = NULL;
31069aa32835SJeff Kirsher 	const u32 *dn_log_port_id;
31079aa32835SJeff Kirsher 
31089aa32835SJeff Kirsher 	lhea_dn = adapter->ofdev->dev.of_node;
31099aa32835SJeff Kirsher 	while ((eth_dn = of_get_next_child(lhea_dn, eth_dn))) {
31109aa32835SJeff Kirsher 
31119aa32835SJeff Kirsher 		dn_log_port_id = of_get_property(eth_dn, "ibm,hea-port-no",
31129aa32835SJeff Kirsher 						 NULL);
31139aa32835SJeff Kirsher 		if (dn_log_port_id)
31149aa32835SJeff Kirsher 			if (*dn_log_port_id == logical_port_id)
31159aa32835SJeff Kirsher 				return eth_dn;
31169aa32835SJeff Kirsher 	}
31179aa32835SJeff Kirsher 
31189aa32835SJeff Kirsher 	return NULL;
31199aa32835SJeff Kirsher }
31209aa32835SJeff Kirsher 
probe_port_store(struct device * dev,struct device_attribute * attr,const char * buf,size_t count)31210056982fSYueHaibing static ssize_t probe_port_store(struct device *dev,
31229aa32835SJeff Kirsher 			       struct device_attribute *attr,
31239aa32835SJeff Kirsher 			       const char *buf, size_t count)
31249aa32835SJeff Kirsher {
31259aa32835SJeff Kirsher 	struct ehea_adapter *adapter = dev_get_drvdata(dev);
31269aa32835SJeff Kirsher 	struct ehea_port *port;
31279aa32835SJeff Kirsher 	struct device_node *eth_dn = NULL;
31289aa32835SJeff Kirsher 	int i;
31299aa32835SJeff Kirsher 
31309aa32835SJeff Kirsher 	u32 logical_port_id;
31319aa32835SJeff Kirsher 
31329aa32835SJeff Kirsher 	sscanf(buf, "%d", &logical_port_id);
31339aa32835SJeff Kirsher 
31349aa32835SJeff Kirsher 	port = ehea_get_port(adapter, logical_port_id);
31359aa32835SJeff Kirsher 
31369aa32835SJeff Kirsher 	if (port) {
31379aa32835SJeff Kirsher 		netdev_info(port->netdev, "adding port with logical port id=%d failed: port already configured\n",
31389aa32835SJeff Kirsher 			    logical_port_id);
31399aa32835SJeff Kirsher 		return -EINVAL;
31409aa32835SJeff Kirsher 	}
31419aa32835SJeff Kirsher 
31429aa32835SJeff Kirsher 	eth_dn = ehea_get_eth_dn(adapter, logical_port_id);
31439aa32835SJeff Kirsher 
31449aa32835SJeff Kirsher 	if (!eth_dn) {
31459aa32835SJeff Kirsher 		pr_info("no logical port with id %d found\n", logical_port_id);
31469aa32835SJeff Kirsher 		return -EINVAL;
31479aa32835SJeff Kirsher 	}
31489aa32835SJeff Kirsher 
31499aa32835SJeff Kirsher 	if (ehea_add_adapter_mr(adapter)) {
31509aa32835SJeff Kirsher 		pr_err("creating MR failed\n");
3151be693df3SWen Yang 		of_node_put(eth_dn);
31529aa32835SJeff Kirsher 		return -EIO;
31539aa32835SJeff Kirsher 	}
31549aa32835SJeff Kirsher 
31559aa32835SJeff Kirsher 	port = ehea_setup_single_port(adapter, logical_port_id, eth_dn);
31569aa32835SJeff Kirsher 
31579aa32835SJeff Kirsher 	of_node_put(eth_dn);
31589aa32835SJeff Kirsher 
31599aa32835SJeff Kirsher 	if (port) {
31609aa32835SJeff Kirsher 		for (i = 0; i < EHEA_MAX_PORTS; i++)
31619aa32835SJeff Kirsher 			if (!adapter->port[i]) {
31629aa32835SJeff Kirsher 				adapter->port[i] = port;
31639aa32835SJeff Kirsher 				break;
31649aa32835SJeff Kirsher 			}
31659aa32835SJeff Kirsher 
31669aa32835SJeff Kirsher 		netdev_info(port->netdev, "added: (logical port id=%d)\n",
31679aa32835SJeff Kirsher 			    logical_port_id);
31689aa32835SJeff Kirsher 	} else {
31699aa32835SJeff Kirsher 		ehea_remove_adapter_mr(adapter);
31709aa32835SJeff Kirsher 		return -EIO;
31719aa32835SJeff Kirsher 	}
31729aa32835SJeff Kirsher 
31739aa32835SJeff Kirsher 	return (ssize_t) count;
31749aa32835SJeff Kirsher }
31759aa32835SJeff Kirsher 
remove_port_store(struct device * dev,struct device_attribute * attr,const char * buf,size_t count)31760056982fSYueHaibing static ssize_t remove_port_store(struct device *dev,
31779aa32835SJeff Kirsher 				 struct device_attribute *attr,
31789aa32835SJeff Kirsher 				 const char *buf, size_t count)
31799aa32835SJeff Kirsher {
31809aa32835SJeff Kirsher 	struct ehea_adapter *adapter = dev_get_drvdata(dev);
31819aa32835SJeff Kirsher 	struct ehea_port *port;
31829aa32835SJeff Kirsher 	int i;
31839aa32835SJeff Kirsher 	u32 logical_port_id;
31849aa32835SJeff Kirsher 
31859aa32835SJeff Kirsher 	sscanf(buf, "%d", &logical_port_id);
31869aa32835SJeff Kirsher 
31879aa32835SJeff Kirsher 	port = ehea_get_port(adapter, logical_port_id);
31889aa32835SJeff Kirsher 
31899aa32835SJeff Kirsher 	if (port) {
31909aa32835SJeff Kirsher 		netdev_info(port->netdev, "removed: (logical port id=%d)\n",
31919aa32835SJeff Kirsher 			    logical_port_id);
31929aa32835SJeff Kirsher 
31939aa32835SJeff Kirsher 		ehea_shutdown_single_port(port);
31949aa32835SJeff Kirsher 
31959aa32835SJeff Kirsher 		for (i = 0; i < EHEA_MAX_PORTS; i++)
31969aa32835SJeff Kirsher 			if (adapter->port[i] == port) {
31979aa32835SJeff Kirsher 				adapter->port[i] = NULL;
31989aa32835SJeff Kirsher 				break;
31999aa32835SJeff Kirsher 			}
32009aa32835SJeff Kirsher 	} else {
32019aa32835SJeff Kirsher 		pr_err("removing port with logical port id=%d failed. port not configured.\n",
32029aa32835SJeff Kirsher 		       logical_port_id);
32039aa32835SJeff Kirsher 		return -EINVAL;
32049aa32835SJeff Kirsher 	}
32059aa32835SJeff Kirsher 
32069aa32835SJeff Kirsher 	ehea_remove_adapter_mr(adapter);
32079aa32835SJeff Kirsher 
32089aa32835SJeff Kirsher 	return (ssize_t) count;
32099aa32835SJeff Kirsher }
32109aa32835SJeff Kirsher 
32110056982fSYueHaibing static DEVICE_ATTR_WO(probe_port);
32120056982fSYueHaibing static DEVICE_ATTR_WO(remove_port);
32139aa32835SJeff Kirsher 
ehea_create_device_sysfs(struct platform_device * dev)32141886e5d2SThadeu Lima de Souza Cascardo static int ehea_create_device_sysfs(struct platform_device *dev)
32159aa32835SJeff Kirsher {
32169aa32835SJeff Kirsher 	int ret = device_create_file(&dev->dev, &dev_attr_probe_port);
32179aa32835SJeff Kirsher 	if (ret)
32189aa32835SJeff Kirsher 		goto out;
32199aa32835SJeff Kirsher 
32209aa32835SJeff Kirsher 	ret = device_create_file(&dev->dev, &dev_attr_remove_port);
32219aa32835SJeff Kirsher out:
32229aa32835SJeff Kirsher 	return ret;
32239aa32835SJeff Kirsher }
32249aa32835SJeff Kirsher 
ehea_remove_device_sysfs(struct platform_device * dev)32251886e5d2SThadeu Lima de Souza Cascardo static void ehea_remove_device_sysfs(struct platform_device *dev)
32269aa32835SJeff Kirsher {
32279aa32835SJeff Kirsher 	device_remove_file(&dev->dev, &dev_attr_probe_port);
32289aa32835SJeff Kirsher 	device_remove_file(&dev->dev, &dev_attr_remove_port);
32299aa32835SJeff Kirsher }
32309aa32835SJeff Kirsher 
ehea_reboot_notifier(struct notifier_block * nb,unsigned long action,void * unused)3231aa183323SAnton Blanchard static int ehea_reboot_notifier(struct notifier_block *nb,
3232aa183323SAnton Blanchard 				unsigned long action, void *unused)
3233aa183323SAnton Blanchard {
3234aa183323SAnton Blanchard 	if (action == SYS_RESTART) {
3235aa183323SAnton Blanchard 		pr_info("Reboot: freeing all eHEA resources\n");
3236aa183323SAnton Blanchard 		ibmebus_unregister_driver(&ehea_driver);
3237aa183323SAnton Blanchard 	}
3238aa183323SAnton Blanchard 	return NOTIFY_DONE;
3239aa183323SAnton Blanchard }
3240aa183323SAnton Blanchard 
3241aa183323SAnton Blanchard static struct notifier_block ehea_reboot_nb = {
3242aa183323SAnton Blanchard 	.notifier_call = ehea_reboot_notifier,
3243aa183323SAnton Blanchard };
3244aa183323SAnton Blanchard 
ehea_mem_notifier(struct notifier_block * nb,unsigned long action,void * data)3245aa183323SAnton Blanchard static int ehea_mem_notifier(struct notifier_block *nb,
3246aa183323SAnton Blanchard 			     unsigned long action, void *data)
3247aa183323SAnton Blanchard {
3248aa183323SAnton Blanchard 	int ret = NOTIFY_BAD;
3249aa183323SAnton Blanchard 	struct memory_notify *arg = data;
3250aa183323SAnton Blanchard 
3251aa183323SAnton Blanchard 	mutex_lock(&dlpar_mem_lock);
3252aa183323SAnton Blanchard 
3253aa183323SAnton Blanchard 	switch (action) {
3254aa183323SAnton Blanchard 	case MEM_CANCEL_OFFLINE:
3255aa183323SAnton Blanchard 		pr_info("memory offlining canceled");
3256df561f66SGustavo A. R. Silva 		fallthrough;	/* re-add canceled memory block */
3257aa183323SAnton Blanchard 
3258aa183323SAnton Blanchard 	case MEM_ONLINE:
3259aa183323SAnton Blanchard 		pr_info("memory is going online");
3260aa183323SAnton Blanchard 		set_bit(__EHEA_STOP_XFER, &ehea_driver_flags);
3261aa183323SAnton Blanchard 		if (ehea_add_sect_bmap(arg->start_pfn, arg->nr_pages))
3262aa183323SAnton Blanchard 			goto out_unlock;
3263aa183323SAnton Blanchard 		ehea_rereg_mrs();
3264aa183323SAnton Blanchard 		break;
3265aa183323SAnton Blanchard 
3266aa183323SAnton Blanchard 	case MEM_GOING_OFFLINE:
3267aa183323SAnton Blanchard 		pr_info("memory is going offline");
3268aa183323SAnton Blanchard 		set_bit(__EHEA_STOP_XFER, &ehea_driver_flags);
3269aa183323SAnton Blanchard 		if (ehea_rem_sect_bmap(arg->start_pfn, arg->nr_pages))
3270aa183323SAnton Blanchard 			goto out_unlock;
3271aa183323SAnton Blanchard 		ehea_rereg_mrs();
3272aa183323SAnton Blanchard 		break;
3273aa183323SAnton Blanchard 
3274aa183323SAnton Blanchard 	default:
3275aa183323SAnton Blanchard 		break;
3276aa183323SAnton Blanchard 	}
3277aa183323SAnton Blanchard 
3278aa183323SAnton Blanchard 	ehea_update_firmware_handles();
3279aa183323SAnton Blanchard 	ret = NOTIFY_OK;
3280aa183323SAnton Blanchard 
3281aa183323SAnton Blanchard out_unlock:
3282aa183323SAnton Blanchard 	mutex_unlock(&dlpar_mem_lock);
3283aa183323SAnton Blanchard 	return ret;
3284aa183323SAnton Blanchard }
3285aa183323SAnton Blanchard 
3286aa183323SAnton Blanchard static struct notifier_block ehea_mem_nb = {
3287aa183323SAnton Blanchard 	.notifier_call = ehea_mem_notifier,
3288aa183323SAnton Blanchard };
3289aa183323SAnton Blanchard 
ehea_crash_handler(void)3290aa183323SAnton Blanchard static void ehea_crash_handler(void)
3291aa183323SAnton Blanchard {
3292aa183323SAnton Blanchard 	int i;
3293aa183323SAnton Blanchard 
3294aa183323SAnton Blanchard 	if (ehea_fw_handles.arr)
3295aa183323SAnton Blanchard 		for (i = 0; i < ehea_fw_handles.num_entries; i++)
3296aa183323SAnton Blanchard 			ehea_h_free_resource(ehea_fw_handles.arr[i].adh,
3297aa183323SAnton Blanchard 					     ehea_fw_handles.arr[i].fwh,
3298aa183323SAnton Blanchard 					     FORCE_FREE);
3299aa183323SAnton Blanchard 
3300aa183323SAnton Blanchard 	if (ehea_bcmc_regs.arr)
3301aa183323SAnton Blanchard 		for (i = 0; i < ehea_bcmc_regs.num_entries; i++)
3302aa183323SAnton Blanchard 			ehea_h_reg_dereg_bcmc(ehea_bcmc_regs.arr[i].adh,
3303aa183323SAnton Blanchard 					      ehea_bcmc_regs.arr[i].port_id,
3304aa183323SAnton Blanchard 					      ehea_bcmc_regs.arr[i].reg_type,
3305aa183323SAnton Blanchard 					      ehea_bcmc_regs.arr[i].macaddr,
3306aa183323SAnton Blanchard 					      0, H_DEREG_BCMC);
3307aa183323SAnton Blanchard }
3308aa183323SAnton Blanchard 
3309aa183323SAnton Blanchard static atomic_t ehea_memory_hooks_registered;
3310aa183323SAnton Blanchard 
3311aa183323SAnton Blanchard /* Register memory hooks on probe of first adapter */
ehea_register_memory_hooks(void)3312aa183323SAnton Blanchard static int ehea_register_memory_hooks(void)
3313aa183323SAnton Blanchard {
3314aa183323SAnton Blanchard 	int ret = 0;
3315aa183323SAnton Blanchard 
33163051f392SMichael Ellerman 	if (atomic_inc_return(&ehea_memory_hooks_registered) > 1)
3317aa183323SAnton Blanchard 		return 0;
3318aa183323SAnton Blanchard 
3319aa183323SAnton Blanchard 	ret = ehea_create_busmap();
3320aa183323SAnton Blanchard 	if (ret) {
3321aa183323SAnton Blanchard 		pr_info("ehea_create_busmap failed\n");
3322aa183323SAnton Blanchard 		goto out;
3323aa183323SAnton Blanchard 	}
3324aa183323SAnton Blanchard 
3325aa183323SAnton Blanchard 	ret = register_reboot_notifier(&ehea_reboot_nb);
3326aa183323SAnton Blanchard 	if (ret) {
3327aa183323SAnton Blanchard 		pr_info("register_reboot_notifier failed\n");
3328aa183323SAnton Blanchard 		goto out;
3329aa183323SAnton Blanchard 	}
3330aa183323SAnton Blanchard 
3331aa183323SAnton Blanchard 	ret = register_memory_notifier(&ehea_mem_nb);
3332aa183323SAnton Blanchard 	if (ret) {
3333aa183323SAnton Blanchard 		pr_info("register_memory_notifier failed\n");
3334aa183323SAnton Blanchard 		goto out2;
3335aa183323SAnton Blanchard 	}
3336aa183323SAnton Blanchard 
3337aa183323SAnton Blanchard 	ret = crash_shutdown_register(ehea_crash_handler);
3338aa183323SAnton Blanchard 	if (ret) {
3339aa183323SAnton Blanchard 		pr_info("crash_shutdown_register failed\n");
3340aa183323SAnton Blanchard 		goto out3;
3341aa183323SAnton Blanchard 	}
3342aa183323SAnton Blanchard 
3343aa183323SAnton Blanchard 	return 0;
3344aa183323SAnton Blanchard 
3345aa183323SAnton Blanchard out3:
3346aa183323SAnton Blanchard 	unregister_memory_notifier(&ehea_mem_nb);
3347aa183323SAnton Blanchard out2:
3348aa183323SAnton Blanchard 	unregister_reboot_notifier(&ehea_reboot_nb);
3349aa183323SAnton Blanchard out:
33503051f392SMichael Ellerman 	atomic_dec(&ehea_memory_hooks_registered);
3351aa183323SAnton Blanchard 	return ret;
3352aa183323SAnton Blanchard }
3353aa183323SAnton Blanchard 
ehea_unregister_memory_hooks(void)3354aa183323SAnton Blanchard static void ehea_unregister_memory_hooks(void)
3355aa183323SAnton Blanchard {
33563051f392SMichael Ellerman 	/* Only remove the hooks if we've registered them */
33573051f392SMichael Ellerman 	if (atomic_read(&ehea_memory_hooks_registered) == 0)
3358aa183323SAnton Blanchard 		return;
3359aa183323SAnton Blanchard 
3360aa183323SAnton Blanchard 	unregister_reboot_notifier(&ehea_reboot_nb);
3361aa183323SAnton Blanchard 	if (crash_shutdown_unregister(ehea_crash_handler))
3362aa183323SAnton Blanchard 		pr_info("failed unregistering crash handler\n");
3363aa183323SAnton Blanchard 	unregister_memory_notifier(&ehea_mem_nb);
3364aa183323SAnton Blanchard }
3365aa183323SAnton Blanchard 
ehea_probe_adapter(struct platform_device * dev)3366c45640e4SRob Herring static int ehea_probe_adapter(struct platform_device *dev)
33679aa32835SJeff Kirsher {
33689aa32835SJeff Kirsher 	struct ehea_adapter *adapter;
33699aa32835SJeff Kirsher 	const u64 *adapter_handle;
33709aa32835SJeff Kirsher 	int ret;
3371c2f1244bSThadeu Lima de Souza Cascardo 	int i;
33729aa32835SJeff Kirsher 
3373aa183323SAnton Blanchard 	ret = ehea_register_memory_hooks();
3374aa183323SAnton Blanchard 	if (ret)
3375aa183323SAnton Blanchard 		return ret;
3376aa183323SAnton Blanchard 
33779aa32835SJeff Kirsher 	if (!dev || !dev->dev.of_node) {
33789aa32835SJeff Kirsher 		pr_err("Invalid ibmebus device probed\n");
33799aa32835SJeff Kirsher 		return -EINVAL;
33809aa32835SJeff Kirsher 	}
33819aa32835SJeff Kirsher 
338209b38aa1SHimangi Saraogi 	adapter = devm_kzalloc(&dev->dev, sizeof(*adapter), GFP_KERNEL);
33839aa32835SJeff Kirsher 	if (!adapter) {
33849aa32835SJeff Kirsher 		ret = -ENOMEM;
33859aa32835SJeff Kirsher 		dev_err(&dev->dev, "no mem for ehea_adapter\n");
33869aa32835SJeff Kirsher 		goto out;
33879aa32835SJeff Kirsher 	}
33889aa32835SJeff Kirsher 
33899aa32835SJeff Kirsher 	list_add(&adapter->list, &adapter_list);
33909aa32835SJeff Kirsher 
33919aa32835SJeff Kirsher 	adapter->ofdev = dev;
33929aa32835SJeff Kirsher 
33939aa32835SJeff Kirsher 	adapter_handle = of_get_property(dev->dev.of_node, "ibm,hea-handle",
33949aa32835SJeff Kirsher 					 NULL);
33959aa32835SJeff Kirsher 	if (adapter_handle)
33969aa32835SJeff Kirsher 		adapter->handle = *adapter_handle;
33979aa32835SJeff Kirsher 
33989aa32835SJeff Kirsher 	if (!adapter->handle) {
33999aa32835SJeff Kirsher 		dev_err(&dev->dev, "failed getting handle for adapter"
3400f7ce9103SRob Herring 			" '%pOF'\n", dev->dev.of_node);
34019aa32835SJeff Kirsher 		ret = -ENODEV;
34029aa32835SJeff Kirsher 		goto out_free_ad;
34039aa32835SJeff Kirsher 	}
34049aa32835SJeff Kirsher 
34059aa32835SJeff Kirsher 	adapter->pd = EHEA_PD_ID;
34069aa32835SJeff Kirsher 
34078513fbd8SJingoo Han 	platform_set_drvdata(dev, adapter);
34089aa32835SJeff Kirsher 
34099aa32835SJeff Kirsher 
34109aa32835SJeff Kirsher 	/* initialize adapter and ports */
34119aa32835SJeff Kirsher 	/* get adapter properties */
34129aa32835SJeff Kirsher 	ret = ehea_sense_adapter_attr(adapter);
34139aa32835SJeff Kirsher 	if (ret) {
34149aa32835SJeff Kirsher 		dev_err(&dev->dev, "sense_adapter_attr failed: %d\n", ret);
34159aa32835SJeff Kirsher 		goto out_free_ad;
34169aa32835SJeff Kirsher 	}
34179aa32835SJeff Kirsher 
34189aa32835SJeff Kirsher 	adapter->neq = ehea_create_eq(adapter,
34199aa32835SJeff Kirsher 				      EHEA_NEQ, EHEA_MAX_ENTRIES_EQ, 1);
34209aa32835SJeff Kirsher 	if (!adapter->neq) {
34219aa32835SJeff Kirsher 		ret = -EIO;
34229aa32835SJeff Kirsher 		dev_err(&dev->dev, "NEQ creation failed\n");
34239aa32835SJeff Kirsher 		goto out_free_ad;
34249aa32835SJeff Kirsher 	}
34259aa32835SJeff Kirsher 
34269da7cfc3SAllen Pais 	tasklet_setup(&adapter->neq_tasklet, ehea_neq_tasklet);
34279aa32835SJeff Kirsher 
34289aa32835SJeff Kirsher 	ret = ehea_create_device_sysfs(dev);
34299aa32835SJeff Kirsher 	if (ret)
3430c2f1244bSThadeu Lima de Souza Cascardo 		goto out_kill_eq;
34319aa32835SJeff Kirsher 
34329aa32835SJeff Kirsher 	ret = ehea_setup_ports(adapter);
34339aa32835SJeff Kirsher 	if (ret) {
34349aa32835SJeff Kirsher 		dev_err(&dev->dev, "setup_ports failed\n");
34359aa32835SJeff Kirsher 		goto out_rem_dev_sysfs;
34369aa32835SJeff Kirsher 	}
34379aa32835SJeff Kirsher 
3438c2f1244bSThadeu Lima de Souza Cascardo 	ret = ibmebus_request_irq(adapter->neq->attr.ist1,
343946c915f8SMichael Opdenacker 				  ehea_interrupt_neq, 0,
3440c2f1244bSThadeu Lima de Souza Cascardo 				  "ehea_neq", adapter);
3441c2f1244bSThadeu Lima de Souza Cascardo 	if (ret) {
3442c2f1244bSThadeu Lima de Souza Cascardo 		dev_err(&dev->dev, "requesting NEQ IRQ failed\n");
3443c2f1244bSThadeu Lima de Souza Cascardo 		goto out_shutdown_ports;
3444c2f1244bSThadeu Lima de Souza Cascardo 	}
3445c2f1244bSThadeu Lima de Souza Cascardo 
3446380ec964SThadeu Lima de Souza Cascardo 	/* Handle any events that might be pending. */
3447380ec964SThadeu Lima de Souza Cascardo 	tasklet_hi_schedule(&adapter->neq_tasklet);
3448c2f1244bSThadeu Lima de Souza Cascardo 
34499aa32835SJeff Kirsher 	ret = 0;
34509aa32835SJeff Kirsher 	goto out;
34519aa32835SJeff Kirsher 
3452c2f1244bSThadeu Lima de Souza Cascardo out_shutdown_ports:
3453c2f1244bSThadeu Lima de Souza Cascardo 	for (i = 0; i < EHEA_MAX_PORTS; i++)
3454c2f1244bSThadeu Lima de Souza Cascardo 		if (adapter->port[i]) {
3455c2f1244bSThadeu Lima de Souza Cascardo 			ehea_shutdown_single_port(adapter->port[i]);
3456c2f1244bSThadeu Lima de Souza Cascardo 			adapter->port[i] = NULL;
3457c2f1244bSThadeu Lima de Souza Cascardo 		}
3458c2f1244bSThadeu Lima de Souza Cascardo 
34599aa32835SJeff Kirsher out_rem_dev_sysfs:
34609aa32835SJeff Kirsher 	ehea_remove_device_sysfs(dev);
34619aa32835SJeff Kirsher 
34629aa32835SJeff Kirsher out_kill_eq:
34639aa32835SJeff Kirsher 	ehea_destroy_eq(adapter->neq);
34649aa32835SJeff Kirsher 
34659aa32835SJeff Kirsher out_free_ad:
34669aa32835SJeff Kirsher 	list_del(&adapter->list);
34679aa32835SJeff Kirsher 
34689aa32835SJeff Kirsher out:
34699aa32835SJeff Kirsher 	ehea_update_firmware_handles();
34709aa32835SJeff Kirsher 
34719aa32835SJeff Kirsher 	return ret;
34729aa32835SJeff Kirsher }
34739aa32835SJeff Kirsher 
ehea_remove(struct platform_device * dev)34740297be07SBill Pemberton static int ehea_remove(struct platform_device *dev)
34759aa32835SJeff Kirsher {
34768513fbd8SJingoo Han 	struct ehea_adapter *adapter = platform_get_drvdata(dev);
34779aa32835SJeff Kirsher 	int i;
34789aa32835SJeff Kirsher 
34799aa32835SJeff Kirsher 	for (i = 0; i < EHEA_MAX_PORTS; i++)
34809aa32835SJeff Kirsher 		if (adapter->port[i]) {
34819aa32835SJeff Kirsher 			ehea_shutdown_single_port(adapter->port[i]);
34829aa32835SJeff Kirsher 			adapter->port[i] = NULL;
34839aa32835SJeff Kirsher 		}
34849aa32835SJeff Kirsher 
34859aa32835SJeff Kirsher 	ehea_remove_device_sysfs(dev);
34869aa32835SJeff Kirsher 
34879aa32835SJeff Kirsher 	ibmebus_free_irq(adapter->neq->attr.ist1, adapter);
34889aa32835SJeff Kirsher 	tasklet_kill(&adapter->neq_tasklet);
34899aa32835SJeff Kirsher 
34909aa32835SJeff Kirsher 	ehea_destroy_eq(adapter->neq);
34919aa32835SJeff Kirsher 	ehea_remove_adapter_mr(adapter);
34929aa32835SJeff Kirsher 	list_del(&adapter->list);
34939aa32835SJeff Kirsher 
34949aa32835SJeff Kirsher 	ehea_update_firmware_handles();
34959aa32835SJeff Kirsher 
34969aa32835SJeff Kirsher 	return 0;
34979aa32835SJeff Kirsher }
34989aa32835SJeff Kirsher 
check_module_parm(void)34999aa32835SJeff Kirsher static int check_module_parm(void)
35009aa32835SJeff Kirsher {
35019aa32835SJeff Kirsher 	int ret = 0;
35029aa32835SJeff Kirsher 
35039aa32835SJeff Kirsher 	if ((rq1_entries < EHEA_MIN_ENTRIES_QP) ||
35049aa32835SJeff Kirsher 	    (rq1_entries > EHEA_MAX_ENTRIES_RQ1)) {
35059aa32835SJeff Kirsher 		pr_info("Bad parameter: rq1_entries\n");
35069aa32835SJeff Kirsher 		ret = -EINVAL;
35079aa32835SJeff Kirsher 	}
35089aa32835SJeff Kirsher 	if ((rq2_entries < EHEA_MIN_ENTRIES_QP) ||
35099aa32835SJeff Kirsher 	    (rq2_entries > EHEA_MAX_ENTRIES_RQ2)) {
35109aa32835SJeff Kirsher 		pr_info("Bad parameter: rq2_entries\n");
35119aa32835SJeff Kirsher 		ret = -EINVAL;
35129aa32835SJeff Kirsher 	}
35139aa32835SJeff Kirsher 	if ((rq3_entries < EHEA_MIN_ENTRIES_QP) ||
35149aa32835SJeff Kirsher 	    (rq3_entries > EHEA_MAX_ENTRIES_RQ3)) {
35159aa32835SJeff Kirsher 		pr_info("Bad parameter: rq3_entries\n");
35169aa32835SJeff Kirsher 		ret = -EINVAL;
35179aa32835SJeff Kirsher 	}
35189aa32835SJeff Kirsher 	if ((sq_entries < EHEA_MIN_ENTRIES_QP) ||
35199aa32835SJeff Kirsher 	    (sq_entries > EHEA_MAX_ENTRIES_SQ)) {
35209aa32835SJeff Kirsher 		pr_info("Bad parameter: sq_entries\n");
35219aa32835SJeff Kirsher 		ret = -EINVAL;
35229aa32835SJeff Kirsher 	}
35239aa32835SJeff Kirsher 
35249aa32835SJeff Kirsher 	return ret;
35259aa32835SJeff Kirsher }
35269aa32835SJeff Kirsher 
capabilities_show(struct device_driver * drv,char * buf)3527ce8e0cd8SGreg Kroah-Hartman static ssize_t capabilities_show(struct device_driver *drv, char *buf)
35289aa32835SJeff Kirsher {
35299aa32835SJeff Kirsher 	return sprintf(buf, "%d", EHEA_CAPABILITIES);
35309aa32835SJeff Kirsher }
35319aa32835SJeff Kirsher 
3532ce8e0cd8SGreg Kroah-Hartman static DRIVER_ATTR_RO(capabilities);
35339aa32835SJeff Kirsher 
ehea_module_init(void)35341886e5d2SThadeu Lima de Souza Cascardo static int __init ehea_module_init(void)
35359aa32835SJeff Kirsher {
35369aa32835SJeff Kirsher 	int ret;
35379aa32835SJeff Kirsher 
35389aa32835SJeff Kirsher 	pr_info("IBM eHEA ethernet device driver (Release %s)\n", DRV_VERSION);
35399aa32835SJeff Kirsher 
35409aa32835SJeff Kirsher 	memset(&ehea_fw_handles, 0, sizeof(ehea_fw_handles));
35419aa32835SJeff Kirsher 	memset(&ehea_bcmc_regs, 0, sizeof(ehea_bcmc_regs));
35429aa32835SJeff Kirsher 
35439aa32835SJeff Kirsher 	mutex_init(&ehea_fw_handles.lock);
35449aa32835SJeff Kirsher 	spin_lock_init(&ehea_bcmc_regs.lock);
35459aa32835SJeff Kirsher 
35469aa32835SJeff Kirsher 	ret = check_module_parm();
35479aa32835SJeff Kirsher 	if (ret)
35489aa32835SJeff Kirsher 		goto out;
35499aa32835SJeff Kirsher 
35509aa32835SJeff Kirsher 	ret = ibmebus_register_driver(&ehea_driver);
35519aa32835SJeff Kirsher 	if (ret) {
35529aa32835SJeff Kirsher 		pr_err("failed registering eHEA device driver on ebus\n");
3553aa183323SAnton Blanchard 		goto out;
35549aa32835SJeff Kirsher 	}
35559aa32835SJeff Kirsher 
35569aa32835SJeff Kirsher 	ret = driver_create_file(&ehea_driver.driver,
35579aa32835SJeff Kirsher 				 &driver_attr_capabilities);
35589aa32835SJeff Kirsher 	if (ret) {
35599aa32835SJeff Kirsher 		pr_err("failed to register capabilities attribute, ret=%d\n",
35609aa32835SJeff Kirsher 		       ret);
3561aa183323SAnton Blanchard 		goto out2;
35629aa32835SJeff Kirsher 	}
35639aa32835SJeff Kirsher 
35649aa32835SJeff Kirsher 	return ret;
35659aa32835SJeff Kirsher 
35669aa32835SJeff Kirsher out2:
3567aa183323SAnton Blanchard 	ibmebus_unregister_driver(&ehea_driver);
35689aa32835SJeff Kirsher out:
35699aa32835SJeff Kirsher 	return ret;
35709aa32835SJeff Kirsher }
35719aa32835SJeff Kirsher 
ehea_module_exit(void)35729aa32835SJeff Kirsher static void __exit ehea_module_exit(void)
35739aa32835SJeff Kirsher {
35749aa32835SJeff Kirsher 	driver_remove_file(&ehea_driver.driver, &driver_attr_capabilities);
35759aa32835SJeff Kirsher 	ibmebus_unregister_driver(&ehea_driver);
3576aa183323SAnton Blanchard 	ehea_unregister_memory_hooks();
35779aa32835SJeff Kirsher 	kfree(ehea_fw_handles.arr);
35789aa32835SJeff Kirsher 	kfree(ehea_bcmc_regs.arr);
35799aa32835SJeff Kirsher 	ehea_destroy_busmap();
35809aa32835SJeff Kirsher }
35819aa32835SJeff Kirsher 
35829aa32835SJeff Kirsher module_init(ehea_module_init);
35839aa32835SJeff Kirsher module_exit(ehea_module_exit);
3584