2008-02-15 16:19:42 +08:00
|
|
|
/*
|
2009-06-16 16:30:31 +08:00
|
|
|
* Copyright IBM Corp. 2007, 2009
|
2008-02-15 16:19:42 +08:00
|
|
|
* Author(s): Utz Bacher <utz.bacher@de.ibm.com>,
|
|
|
|
* Frank Pavlic <fpavlic@de.ibm.com>,
|
|
|
|
* Thomas Spatzier <tspat@de.ibm.com>,
|
|
|
|
* Frank Blaschka <frank.blaschka@de.ibm.com>
|
|
|
|
*/
|
|
|
|
|
2008-12-25 20:39:49 +08:00
|
|
|
#define KMSG_COMPONENT "qeth"
|
|
|
|
#define pr_fmt(fmt) KMSG_COMPONENT ": " fmt
|
|
|
|
|
2008-02-15 16:19:42 +08:00
|
|
|
#include <linux/module.h>
|
|
|
|
#include <linux/moduleparam.h>
|
|
|
|
#include <linux/string.h>
|
|
|
|
#include <linux/errno.h>
|
|
|
|
#include <linux/kernel.h>
|
include cleanup: Update gfp.h and slab.h includes to prepare for breaking implicit slab.h inclusion from percpu.h
percpu.h is included by sched.h and module.h and thus ends up being
included when building most .c files. percpu.h includes slab.h which
in turn includes gfp.h making everything defined by the two files
universally available and complicating inclusion dependencies.
percpu.h -> slab.h dependency is about to be removed. Prepare for
this change by updating users of gfp and slab facilities include those
headers directly instead of assuming availability. As this conversion
needs to touch large number of source files, the following script is
used as the basis of conversion.
http://userweb.kernel.org/~tj/misc/slabh-sweep.py
The script does the followings.
* Scan files for gfp and slab usages and update includes such that
only the necessary includes are there. ie. if only gfp is used,
gfp.h, if slab is used, slab.h.
* When the script inserts a new include, it looks at the include
blocks and try to put the new include such that its order conforms
to its surrounding. It's put in the include block which contains
core kernel includes, in the same order that the rest are ordered -
alphabetical, Christmas tree, rev-Xmas-tree or at the end if there
doesn't seem to be any matching order.
* If the script can't find a place to put a new include (mostly
because the file doesn't have fitting include block), it prints out
an error message indicating which .h file needs to be added to the
file.
The conversion was done in the following steps.
1. The initial automatic conversion of all .c files updated slightly
over 4000 files, deleting around 700 includes and adding ~480 gfp.h
and ~3000 slab.h inclusions. The script emitted errors for ~400
files.
2. Each error was manually checked. Some didn't need the inclusion,
some needed manual addition while adding it to implementation .h or
embedding .c file was more appropriate for others. This step added
inclusions to around 150 files.
3. The script was run again and the output was compared to the edits
from #2 to make sure no file was left behind.
4. Several build tests were done and a couple of problems were fixed.
e.g. lib/decompress_*.c used malloc/free() wrappers around slab
APIs requiring slab.h to be added manually.
5. The script was run on all .h files but without automatically
editing them as sprinkling gfp.h and slab.h inclusions around .h
files could easily lead to inclusion dependency hell. Most gfp.h
inclusion directives were ignored as stuff from gfp.h was usually
wildly available and often used in preprocessor macros. Each
slab.h inclusion directive was examined and added manually as
necessary.
6. percpu.h was updated not to include slab.h.
7. Build test were done on the following configurations and failures
were fixed. CONFIG_GCOV_KERNEL was turned off for all tests (as my
distributed build env didn't work with gcov compiles) and a few
more options had to be turned off depending on archs to make things
build (like ipr on powerpc/64 which failed due to missing writeq).
* x86 and x86_64 UP and SMP allmodconfig and a custom test config.
* powerpc and powerpc64 SMP allmodconfig
* sparc and sparc64 SMP allmodconfig
* ia64 SMP allmodconfig
* s390 SMP allmodconfig
* alpha SMP allmodconfig
* um on x86_64 SMP allmodconfig
8. percpu.h modifications were reverted so that it could be applied as
a separate patch and serve as bisection point.
Given the fact that I had only a couple of failures from tests on step
6, I'm fairly confident about the coverage of this conversion patch.
If there is a breakage, it's likely to be something in one of the arch
headers which should be easily discoverable easily on most builds of
the specific arch.
Signed-off-by: Tejun Heo <tj@kernel.org>
Guess-its-ok-by: Christoph Lameter <cl@linux-foundation.org>
Cc: Ingo Molnar <mingo@redhat.com>
Cc: Lee Schermerhorn <Lee.Schermerhorn@hp.com>
2010-03-24 16:04:11 +08:00
|
|
|
#include <linux/slab.h>
|
2008-02-15 16:19:42 +08:00
|
|
|
#include <linux/etherdevice.h>
|
|
|
|
#include <linux/ip.h>
|
2009-05-23 07:22:17 +08:00
|
|
|
#include <linux/list.h>
|
2015-10-06 21:12:28 +08:00
|
|
|
#include <linux/hash.h>
|
|
|
|
#include <linux/hashtable.h>
|
|
|
|
#include <linux/string.h>
|
2008-02-15 16:19:42 +08:00
|
|
|
#include "qeth_core.h"
|
2014-01-14 22:54:11 +08:00
|
|
|
#include "qeth_l2.h"
|
2008-02-15 16:19:42 +08:00
|
|
|
|
|
|
|
static int qeth_l2_set_offline(struct ccwgroup_device *);
|
|
|
|
static int qeth_l2_stop(struct net_device *);
|
2015-10-06 21:12:28 +08:00
|
|
|
static void qeth_l2_set_rx_mode(struct net_device *);
|
2014-01-29 16:23:48 +08:00
|
|
|
static void qeth_bridgeport_query_support(struct qeth_card *card);
|
|
|
|
static void qeth_bridge_state_change(struct qeth_card *card,
|
|
|
|
struct qeth_ipa_cmd *cmd);
|
|
|
|
static void qeth_bridge_host_event(struct qeth_card *card,
|
|
|
|
struct qeth_ipa_cmd *cmd);
|
2008-02-15 16:19:42 +08:00
|
|
|
|
|
|
|
static int qeth_l2_verify_dev(struct net_device *dev)
|
|
|
|
{
|
|
|
|
struct qeth_card *card;
|
|
|
|
unsigned long flags;
|
|
|
|
int rc = 0;
|
|
|
|
|
|
|
|
read_lock_irqsave(&qeth_core_card_list.rwlock, flags);
|
|
|
|
list_for_each_entry(card, &qeth_core_card_list.list, list) {
|
|
|
|
if (card->dev == dev) {
|
|
|
|
rc = QETH_REAL_CARD;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
read_unlock_irqrestore(&qeth_core_card_list.rwlock, flags);
|
|
|
|
|
|
|
|
return rc;
|
|
|
|
}
|
|
|
|
|
|
|
|
static struct net_device *qeth_l2_netdev_by_devno(unsigned char *read_dev_no)
|
|
|
|
{
|
|
|
|
struct qeth_card *card;
|
|
|
|
struct net_device *ndev;
|
2008-06-06 18:37:45 +08:00
|
|
|
__u16 temp_dev_no;
|
2008-02-15 16:19:42 +08:00
|
|
|
unsigned long flags;
|
2008-06-06 18:37:45 +08:00
|
|
|
struct ccw_dev_id read_devid;
|
2008-02-15 16:19:42 +08:00
|
|
|
|
|
|
|
ndev = NULL;
|
|
|
|
memcpy(&temp_dev_no, read_dev_no, 2);
|
|
|
|
read_lock_irqsave(&qeth_core_card_list.rwlock, flags);
|
|
|
|
list_for_each_entry(card, &qeth_core_card_list.list, list) {
|
2008-06-06 18:37:45 +08:00
|
|
|
ccw_device_get_id(CARD_RDEV(card), &read_devid);
|
|
|
|
if (read_devid.devno == temp_dev_no) {
|
2008-02-15 16:19:42 +08:00
|
|
|
ndev = card->dev;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
read_unlock_irqrestore(&qeth_core_card_list.rwlock, flags);
|
|
|
|
return ndev;
|
|
|
|
}
|
|
|
|
|
2015-01-21 20:39:09 +08:00
|
|
|
static int qeth_setdel_makerc(struct qeth_card *card, int retcode)
|
2008-02-15 16:19:42 +08:00
|
|
|
{
|
2015-01-21 20:39:09 +08:00
|
|
|
int rc;
|
2008-02-15 16:19:42 +08:00
|
|
|
|
2015-01-21 20:39:09 +08:00
|
|
|
if (retcode)
|
2015-01-21 20:39:10 +08:00
|
|
|
QETH_CARD_TEXT_(card, 2, "err%04x", retcode);
|
2015-01-21 20:39:09 +08:00
|
|
|
switch (retcode) {
|
|
|
|
case IPA_RC_SUCCESS:
|
|
|
|
rc = 0;
|
|
|
|
break;
|
|
|
|
case IPA_RC_L2_UNSUPPORTED_CMD:
|
2015-05-18 20:27:58 +08:00
|
|
|
rc = -EOPNOTSUPP;
|
2015-01-21 20:39:09 +08:00
|
|
|
break;
|
|
|
|
case IPA_RC_L2_ADDR_TABLE_FULL:
|
|
|
|
rc = -ENOSPC;
|
|
|
|
break;
|
|
|
|
case IPA_RC_L2_DUP_MAC:
|
|
|
|
case IPA_RC_L2_DUP_LAYER3_MAC:
|
|
|
|
rc = -EEXIST;
|
|
|
|
break;
|
|
|
|
case IPA_RC_L2_MAC_NOT_AUTH_BY_HYP:
|
|
|
|
case IPA_RC_L2_MAC_NOT_AUTH_BY_ADP:
|
|
|
|
rc = -EPERM;
|
|
|
|
break;
|
|
|
|
case IPA_RC_L2_MAC_NOT_FOUND:
|
|
|
|
rc = -ENOENT;
|
|
|
|
break;
|
2015-01-21 20:39:10 +08:00
|
|
|
case -ENOMEM:
|
|
|
|
rc = -ENOMEM;
|
|
|
|
break;
|
2015-01-21 20:39:09 +08:00
|
|
|
default:
|
|
|
|
rc = -EIO;
|
|
|
|
break;
|
2008-02-15 16:19:42 +08:00
|
|
|
}
|
2015-01-21 20:39:09 +08:00
|
|
|
return rc;
|
2008-02-15 16:19:42 +08:00
|
|
|
}
|
|
|
|
|
2017-01-12 22:48:41 +08:00
|
|
|
static int qeth_l2_send_setdelmac(struct qeth_card *card, __u8 *mac,
|
|
|
|
enum qeth_ipa_cmds ipacmd)
|
|
|
|
{
|
|
|
|
struct qeth_ipa_cmd *cmd;
|
|
|
|
struct qeth_cmd_buffer *iob;
|
|
|
|
|
|
|
|
QETH_CARD_TEXT(card, 2, "L2sdmac");
|
|
|
|
iob = qeth_get_ipacmd_buffer(card, ipacmd, QETH_PROT_IPV4);
|
|
|
|
if (!iob)
|
|
|
|
return -ENOMEM;
|
|
|
|
cmd = (struct qeth_ipa_cmd *)(iob->data+IPA_PDU_HEADER_SIZE);
|
|
|
|
cmd->data.setdelmac.mac_length = OSA_ADDR_LEN;
|
|
|
|
memcpy(&cmd->data.setdelmac.mac, mac, OSA_ADDR_LEN);
|
|
|
|
return qeth_setdel_makerc(card, qeth_send_ipa_cmd(card, iob,
|
|
|
|
NULL, NULL));
|
|
|
|
}
|
|
|
|
|
|
|
|
static int qeth_l2_send_setmac(struct qeth_card *card, __u8 *mac)
|
|
|
|
{
|
|
|
|
int rc;
|
|
|
|
|
|
|
|
QETH_CARD_TEXT(card, 2, "L2Setmac");
|
|
|
|
rc = qeth_l2_send_setdelmac(card, mac, IPA_CMD_SETVMAC);
|
|
|
|
if (rc == 0) {
|
|
|
|
card->info.mac_bits |= QETH_LAYER2_MAC_REGISTERED;
|
|
|
|
memcpy(card->dev->dev_addr, mac, OSA_ADDR_LEN);
|
|
|
|
dev_info(&card->gdev->dev,
|
|
|
|
"MAC address %pM successfully registered on device %s\n",
|
|
|
|
card->dev->dev_addr, card->dev->name);
|
|
|
|
} else {
|
|
|
|
card->info.mac_bits &= ~QETH_LAYER2_MAC_REGISTERED;
|
|
|
|
switch (rc) {
|
|
|
|
case -EEXIST:
|
|
|
|
dev_warn(&card->gdev->dev,
|
|
|
|
"MAC address %pM already exists\n", mac);
|
|
|
|
break;
|
|
|
|
case -EPERM:
|
|
|
|
dev_warn(&card->gdev->dev,
|
|
|
|
"MAC address %pM is not authorized\n", mac);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return rc;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int qeth_l2_send_delmac(struct qeth_card *card, __u8 *mac)
|
|
|
|
{
|
|
|
|
int rc;
|
|
|
|
|
|
|
|
QETH_CARD_TEXT(card, 2, "L2Delmac");
|
|
|
|
if (!(card->info.mac_bits & QETH_LAYER2_MAC_REGISTERED))
|
|
|
|
return 0;
|
|
|
|
rc = qeth_l2_send_setdelmac(card, mac, IPA_CMD_DELVMAC);
|
|
|
|
if (rc == 0)
|
|
|
|
card->info.mac_bits &= ~QETH_LAYER2_MAC_REGISTERED;
|
|
|
|
return rc;
|
|
|
|
}
|
|
|
|
|
2008-02-15 16:19:42 +08:00
|
|
|
static int qeth_l2_send_setgroupmac(struct qeth_card *card, __u8 *mac)
|
|
|
|
{
|
2015-01-21 20:39:09 +08:00
|
|
|
int rc;
|
2008-02-15 16:19:42 +08:00
|
|
|
|
2015-01-21 20:39:09 +08:00
|
|
|
QETH_CARD_TEXT(card, 2, "L2Sgmac");
|
2017-01-12 22:48:39 +08:00
|
|
|
rc = qeth_l2_send_setdelmac(card, mac, IPA_CMD_SETGMAC);
|
2015-01-21 20:39:09 +08:00
|
|
|
if (rc == -EEXIST)
|
|
|
|
QETH_DBF_MESSAGE(2, "Group MAC %pM already existing on %s\n",
|
|
|
|
mac, QETH_CARD_IFNAME(card));
|
|
|
|
else if (rc)
|
|
|
|
QETH_DBF_MESSAGE(2, "Could not set group MAC %pM on %s: %d\n",
|
|
|
|
mac, QETH_CARD_IFNAME(card), rc);
|
|
|
|
return rc;
|
2008-02-15 16:19:42 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
static int qeth_l2_send_delgroupmac(struct qeth_card *card, __u8 *mac)
|
|
|
|
{
|
2015-01-21 20:39:09 +08:00
|
|
|
int rc;
|
|
|
|
|
2010-06-22 06:57:05 +08:00
|
|
|
QETH_CARD_TEXT(card, 2, "L2Dgmac");
|
2017-01-12 22:48:39 +08:00
|
|
|
rc = qeth_l2_send_setdelmac(card, mac, IPA_CMD_DELGMAC);
|
2015-01-21 20:39:09 +08:00
|
|
|
if (rc)
|
|
|
|
QETH_DBF_MESSAGE(2,
|
|
|
|
"Could not delete group MAC %pM on %s: %d\n",
|
|
|
|
mac, QETH_CARD_IFNAME(card), rc);
|
|
|
|
return rc;
|
2008-02-15 16:19:42 +08:00
|
|
|
}
|
|
|
|
|
2015-10-06 21:12:28 +08:00
|
|
|
static int qeth_l2_write_mac(struct qeth_card *card, struct qeth_mac *mac)
|
|
|
|
{
|
|
|
|
if (mac->is_uc) {
|
2017-01-12 22:48:40 +08:00
|
|
|
return qeth_l2_send_setdelmac(card, mac->mac_addr,
|
2017-01-12 22:48:39 +08:00
|
|
|
IPA_CMD_SETVMAC);
|
2008-08-15 14:02:59 +08:00
|
|
|
} else {
|
2017-01-12 22:48:40 +08:00
|
|
|
return qeth_l2_send_setgroupmac(card, mac->mac_addr);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static int qeth_l2_remove_mac(struct qeth_card *card, struct qeth_mac *mac)
|
|
|
|
{
|
|
|
|
if (mac->is_uc) {
|
|
|
|
return qeth_l2_send_setdelmac(card, mac->mac_addr,
|
|
|
|
IPA_CMD_DELVMAC);
|
|
|
|
} else {
|
|
|
|
return qeth_l2_send_delgroupmac(card, mac->mac_addr);
|
2008-08-15 14:02:59 +08:00
|
|
|
}
|
2008-02-15 16:19:42 +08:00
|
|
|
}
|
|
|
|
|
2017-01-12 22:48:37 +08:00
|
|
|
static void qeth_l2_del_all_macs(struct qeth_card *card)
|
2008-02-15 16:19:42 +08:00
|
|
|
{
|
2015-10-06 21:12:28 +08:00
|
|
|
struct qeth_mac *mac;
|
|
|
|
struct hlist_node *tmp;
|
|
|
|
int i;
|
2008-02-15 16:19:42 +08:00
|
|
|
|
|
|
|
spin_lock_bh(&card->mclock);
|
2015-10-06 21:12:28 +08:00
|
|
|
hash_for_each_safe(card->mac_htable, i, tmp, mac, hnode) {
|
|
|
|
hash_del(&mac->hnode);
|
|
|
|
kfree(mac);
|
2008-02-15 16:19:42 +08:00
|
|
|
}
|
|
|
|
spin_unlock_bh(&card->mclock);
|
|
|
|
}
|
|
|
|
|
2017-01-12 22:48:41 +08:00
|
|
|
static inline u32 qeth_l2_mac_hash(const u8 *addr)
|
|
|
|
{
|
|
|
|
return get_unaligned((u32 *)(&addr[2]));
|
|
|
|
}
|
|
|
|
|
2009-08-26 10:01:08 +08:00
|
|
|
static inline int qeth_l2_get_cast_type(struct qeth_card *card,
|
|
|
|
struct sk_buff *skb)
|
2008-02-15 16:19:42 +08:00
|
|
|
{
|
2009-08-26 10:01:08 +08:00
|
|
|
if (card->info.type == QETH_CARD_TYPE_OSN)
|
|
|
|
return RTN_UNSPEC;
|
|
|
|
if (is_broadcast_ether_addr(skb->data))
|
|
|
|
return RTN_BROADCAST;
|
|
|
|
if (is_multicast_ether_addr(skb->data))
|
|
|
|
return RTN_MULTICAST;
|
|
|
|
return RTN_UNSPEC;
|
2008-02-15 16:19:42 +08:00
|
|
|
}
|
|
|
|
|
2015-09-18 22:06:51 +08:00
|
|
|
static inline void qeth_l2_hdr_csum(struct qeth_card *card,
|
|
|
|
struct qeth_hdr *hdr, struct sk_buff *skb)
|
|
|
|
{
|
|
|
|
struct iphdr *iph = ip_hdr(skb);
|
|
|
|
|
|
|
|
/* tcph->check contains already the pseudo hdr checksum
|
|
|
|
* so just set the header flags
|
|
|
|
*/
|
|
|
|
if (iph->protocol == IPPROTO_UDP)
|
|
|
|
hdr->hdr.l2.flags[1] |= QETH_HDR_EXT_UDP;
|
|
|
|
hdr->hdr.l2.flags[1] |= QETH_HDR_EXT_CSUM_TRANSP_REQ |
|
|
|
|
QETH_HDR_EXT_CSUM_HDR_REQ;
|
|
|
|
iph->check = 0;
|
|
|
|
if (card->options.performance_stats)
|
|
|
|
card->perf_stats.tx_csum++;
|
|
|
|
}
|
|
|
|
|
2008-02-15 16:19:42 +08:00
|
|
|
static void qeth_l2_fill_header(struct qeth_card *card, struct qeth_hdr *hdr,
|
2014-03-19 14:58:01 +08:00
|
|
|
struct sk_buff *skb, int cast_type)
|
2008-02-15 16:19:42 +08:00
|
|
|
{
|
2008-08-01 22:39:13 +08:00
|
|
|
struct vlan_ethhdr *veth = (struct vlan_ethhdr *)skb_mac_header(skb);
|
2008-02-15 16:19:42 +08:00
|
|
|
|
|
|
|
memset(hdr, 0, sizeof(struct qeth_hdr));
|
|
|
|
hdr->hdr.l2.id = QETH_HEADER_TYPE_LAYER2;
|
|
|
|
|
|
|
|
/* set byte byte 3 to casting flags */
|
|
|
|
if (cast_type == RTN_MULTICAST)
|
|
|
|
hdr->hdr.l2.flags[2] |= QETH_LAYER2_FLAG_MULTICAST;
|
|
|
|
else if (cast_type == RTN_BROADCAST)
|
|
|
|
hdr->hdr.l2.flags[2] |= QETH_LAYER2_FLAG_BROADCAST;
|
|
|
|
else
|
2009-08-26 10:01:08 +08:00
|
|
|
hdr->hdr.l2.flags[2] |= QETH_LAYER2_FLAG_UNICAST;
|
2008-02-15 16:19:42 +08:00
|
|
|
|
2017-04-11 22:11:15 +08:00
|
|
|
hdr->hdr.l2.pkt_length = skb->len - sizeof(struct qeth_hdr);
|
2008-02-15 16:19:42 +08:00
|
|
|
/* VSWITCH relies on the VLAN
|
|
|
|
* information to be present in
|
|
|
|
* the QDIO header */
|
|
|
|
if (veth->h_vlan_proto == __constant_htons(ETH_P_8021Q)) {
|
|
|
|
hdr->hdr.l2.flags[2] |= QETH_LAYER2_FLAG_VLAN;
|
|
|
|
hdr->hdr.l2.vlan_id = ntohs(veth->h_vlan_TCI);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static int qeth_l2_send_setdelvlan_cb(struct qeth_card *card,
|
|
|
|
struct qeth_reply *reply, unsigned long data)
|
|
|
|
{
|
|
|
|
struct qeth_ipa_cmd *cmd;
|
|
|
|
|
2010-06-22 06:57:05 +08:00
|
|
|
QETH_CARD_TEXT(card, 2, "L2sdvcb");
|
2008-02-15 16:19:42 +08:00
|
|
|
cmd = (struct qeth_ipa_cmd *) data;
|
|
|
|
if (cmd->hdr.return_code) {
|
2008-06-06 18:37:46 +08:00
|
|
|
QETH_DBF_MESSAGE(2, "Error in processing VLAN %i on %s: 0x%x. "
|
2008-02-15 16:19:42 +08:00
|
|
|
"Continuing\n", cmd->data.setdelvlan.vlan_id,
|
|
|
|
QETH_CARD_IFNAME(card), cmd->hdr.return_code);
|
2010-06-22 06:57:05 +08:00
|
|
|
QETH_CARD_TEXT_(card, 2, "L2VL%4x", cmd->hdr.command);
|
|
|
|
QETH_CARD_TEXT_(card, 2, "err%d", cmd->hdr.return_code);
|
2008-02-15 16:19:42 +08:00
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int qeth_l2_send_setdelvlan(struct qeth_card *card, __u16 i,
|
|
|
|
enum qeth_ipa_cmds ipacmd)
|
|
|
|
{
|
|
|
|
struct qeth_ipa_cmd *cmd;
|
|
|
|
struct qeth_cmd_buffer *iob;
|
|
|
|
|
2010-06-22 06:57:05 +08:00
|
|
|
QETH_CARD_TEXT_(card, 4, "L2sdv%x", ipacmd);
|
2008-02-15 16:19:42 +08:00
|
|
|
iob = qeth_get_ipacmd_buffer(card, ipacmd, QETH_PROT_IPV4);
|
2015-01-21 20:39:10 +08:00
|
|
|
if (!iob)
|
|
|
|
return -ENOMEM;
|
2008-02-15 16:19:42 +08:00
|
|
|
cmd = (struct qeth_ipa_cmd *)(iob->data+IPA_PDU_HEADER_SIZE);
|
|
|
|
cmd->data.setdelvlan.vlan_id = i;
|
|
|
|
return qeth_send_ipa_cmd(card, iob,
|
|
|
|
qeth_l2_send_setdelvlan_cb, NULL);
|
|
|
|
}
|
|
|
|
|
2011-02-27 14:41:36 +08:00
|
|
|
static void qeth_l2_process_vlans(struct qeth_card *card)
|
2008-02-15 16:19:42 +08:00
|
|
|
{
|
|
|
|
struct qeth_vlan_vid *id;
|
2010-06-22 06:57:05 +08:00
|
|
|
QETH_CARD_TEXT(card, 3, "L2prcvln");
|
2008-02-15 16:19:42 +08:00
|
|
|
spin_lock_bh(&card->vlanlock);
|
|
|
|
list_for_each_entry(id, &card->vid_list, list) {
|
2011-02-27 14:41:36 +08:00
|
|
|
qeth_l2_send_setdelvlan(card, id->vid, IPA_CMD_SETVLAN);
|
2008-02-15 16:19:42 +08:00
|
|
|
}
|
|
|
|
spin_unlock_bh(&card->vlanlock);
|
|
|
|
}
|
|
|
|
|
2013-04-19 10:04:28 +08:00
|
|
|
static int qeth_l2_vlan_rx_add_vid(struct net_device *dev,
|
|
|
|
__be16 proto, u16 vid)
|
2008-02-15 16:19:42 +08:00
|
|
|
{
|
2008-07-26 17:24:10 +08:00
|
|
|
struct qeth_card *card = dev->ml_priv;
|
2008-02-15 16:19:42 +08:00
|
|
|
struct qeth_vlan_vid *id;
|
2015-01-21 20:39:10 +08:00
|
|
|
int rc;
|
2008-02-15 16:19:42 +08:00
|
|
|
|
2010-06-22 06:57:05 +08:00
|
|
|
QETH_CARD_TEXT_(card, 4, "aid:%d", vid);
|
2010-10-01 10:51:13 +08:00
|
|
|
if (!vid)
|
2011-12-09 08:52:37 +08:00
|
|
|
return 0;
|
2010-05-17 05:15:14 +08:00
|
|
|
if (card->info.type == QETH_CARD_TYPE_OSM) {
|
2010-06-22 06:57:05 +08:00
|
|
|
QETH_CARD_TEXT(card, 3, "aidOSM");
|
2011-12-09 08:52:37 +08:00
|
|
|
return 0;
|
2010-05-17 05:15:14 +08:00
|
|
|
}
|
2009-03-25 04:57:18 +08:00
|
|
|
if (qeth_wait_for_threads(card, QETH_RECOVER_THREAD)) {
|
2010-06-22 06:57:05 +08:00
|
|
|
QETH_CARD_TEXT(card, 3, "aidREC");
|
2011-12-09 08:52:37 +08:00
|
|
|
return 0;
|
2009-03-25 04:57:18 +08:00
|
|
|
}
|
2008-02-15 16:19:42 +08:00
|
|
|
id = kmalloc(sizeof(struct qeth_vlan_vid), GFP_ATOMIC);
|
|
|
|
if (id) {
|
|
|
|
id->vid = vid;
|
2015-01-21 20:39:10 +08:00
|
|
|
rc = qeth_l2_send_setdelvlan(card, vid, IPA_CMD_SETVLAN);
|
|
|
|
if (rc) {
|
|
|
|
kfree(id);
|
|
|
|
return rc;
|
|
|
|
}
|
2008-02-15 16:19:42 +08:00
|
|
|
spin_lock_bh(&card->vlanlock);
|
|
|
|
list_add_tail(&id->list, &card->vid_list);
|
|
|
|
spin_unlock_bh(&card->vlanlock);
|
2011-12-09 08:52:37 +08:00
|
|
|
} else {
|
|
|
|
return -ENOMEM;
|
2008-02-15 16:19:42 +08:00
|
|
|
}
|
2011-12-09 08:52:37 +08:00
|
|
|
return 0;
|
2008-02-15 16:19:42 +08:00
|
|
|
}
|
|
|
|
|
2013-04-19 10:04:28 +08:00
|
|
|
static int qeth_l2_vlan_rx_kill_vid(struct net_device *dev,
|
|
|
|
__be16 proto, u16 vid)
|
2008-02-15 16:19:42 +08:00
|
|
|
{
|
|
|
|
struct qeth_vlan_vid *id, *tmpid = NULL;
|
2008-07-26 17:24:10 +08:00
|
|
|
struct qeth_card *card = dev->ml_priv;
|
2015-01-21 20:39:10 +08:00
|
|
|
int rc = 0;
|
2008-02-15 16:19:42 +08:00
|
|
|
|
2010-06-22 06:57:05 +08:00
|
|
|
QETH_CARD_TEXT_(card, 4, "kid:%d", vid);
|
2010-05-17 05:15:14 +08:00
|
|
|
if (card->info.type == QETH_CARD_TYPE_OSM) {
|
2010-06-22 06:57:05 +08:00
|
|
|
QETH_CARD_TEXT(card, 3, "kidOSM");
|
2011-12-09 08:52:37 +08:00
|
|
|
return 0;
|
2010-05-17 05:15:14 +08:00
|
|
|
}
|
2009-03-25 04:57:18 +08:00
|
|
|
if (qeth_wait_for_threads(card, QETH_RECOVER_THREAD)) {
|
2010-06-22 06:57:05 +08:00
|
|
|
QETH_CARD_TEXT(card, 3, "kidREC");
|
2011-12-09 08:52:37 +08:00
|
|
|
return 0;
|
2009-03-25 04:57:18 +08:00
|
|
|
}
|
2008-02-15 16:19:42 +08:00
|
|
|
spin_lock_bh(&card->vlanlock);
|
|
|
|
list_for_each_entry(id, &card->vid_list, list) {
|
|
|
|
if (id->vid == vid) {
|
|
|
|
list_del(&id->list);
|
|
|
|
tmpid = id;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
spin_unlock_bh(&card->vlanlock);
|
|
|
|
if (tmpid) {
|
2015-01-21 20:39:10 +08:00
|
|
|
rc = qeth_l2_send_setdelvlan(card, vid, IPA_CMD_DELVLAN);
|
2008-02-15 16:19:42 +08:00
|
|
|
kfree(tmpid);
|
|
|
|
}
|
2015-10-06 21:12:28 +08:00
|
|
|
qeth_l2_set_rx_mode(card->dev);
|
2015-01-21 20:39:10 +08:00
|
|
|
return rc;
|
2008-02-15 16:19:42 +08:00
|
|
|
}
|
|
|
|
|
2015-08-04 23:11:47 +08:00
|
|
|
static void qeth_l2_stop_card(struct qeth_card *card, int recovery_mode)
|
2008-02-15 16:19:42 +08:00
|
|
|
{
|
2008-04-01 16:26:58 +08:00
|
|
|
QETH_DBF_TEXT(SETUP , 2, "stopcard");
|
|
|
|
QETH_DBF_HEX(SETUP, 2, &card, sizeof(void *));
|
2008-02-15 16:19:42 +08:00
|
|
|
|
|
|
|
qeth_set_allowed_threads(card, 0, 1);
|
|
|
|
if (card->read.state == CH_STATE_UP &&
|
|
|
|
card->write.state == CH_STATE_UP &&
|
|
|
|
(card->state == CARD_STATE_UP)) {
|
|
|
|
if (recovery_mode &&
|
|
|
|
card->info.type != QETH_CARD_TYPE_OSN) {
|
|
|
|
qeth_l2_stop(card->dev);
|
|
|
|
} else {
|
|
|
|
rtnl_lock();
|
|
|
|
dev_close(card->dev);
|
|
|
|
rtnl_unlock();
|
|
|
|
}
|
2011-02-27 14:41:36 +08:00
|
|
|
card->info.mac_bits &= ~QETH_LAYER2_MAC_REGISTERED;
|
2008-02-15 16:19:42 +08:00
|
|
|
card->state = CARD_STATE_SOFTSETUP;
|
|
|
|
}
|
|
|
|
if (card->state == CARD_STATE_SOFTSETUP) {
|
2017-01-12 22:48:37 +08:00
|
|
|
qeth_l2_del_all_macs(card);
|
2008-02-15 16:19:42 +08:00
|
|
|
qeth_clear_ipacmd_list(card);
|
|
|
|
card->state = CARD_STATE_HARDSETUP;
|
|
|
|
}
|
|
|
|
if (card->state == CARD_STATE_HARDSETUP) {
|
|
|
|
qeth_qdio_clear_card(card, 0);
|
|
|
|
qeth_clear_qdio_buffers(card);
|
|
|
|
qeth_clear_working_pool_list(card);
|
|
|
|
card->state = CARD_STATE_DOWN;
|
|
|
|
}
|
|
|
|
if (card->state == CARD_STATE_DOWN) {
|
|
|
|
qeth_clear_cmd_buffers(&card->read);
|
|
|
|
qeth_clear_cmd_buffers(&card->write);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2010-09-08 05:14:42 +08:00
|
|
|
static int qeth_l2_process_inbound_buffer(struct qeth_card *card,
|
|
|
|
int budget, int *done)
|
2008-02-15 16:19:42 +08:00
|
|
|
{
|
2010-09-08 05:14:42 +08:00
|
|
|
int work_done = 0;
|
2008-02-15 16:19:42 +08:00
|
|
|
struct sk_buff *skb;
|
|
|
|
struct qeth_hdr *hdr;
|
|
|
|
unsigned int len;
|
|
|
|
|
2010-09-08 05:14:42 +08:00
|
|
|
*done = 0;
|
2012-11-19 10:46:50 +08:00
|
|
|
WARN_ON_ONCE(!budget);
|
2010-09-08 05:14:42 +08:00
|
|
|
while (budget) {
|
|
|
|
skb = qeth_core_get_next_skb(card,
|
2011-08-08 09:33:59 +08:00
|
|
|
&card->qdio.in_q->bufs[card->rx.b_index],
|
2010-09-08 05:14:42 +08:00
|
|
|
&card->rx.b_element, &card->rx.e_offset, &hdr);
|
|
|
|
if (!skb) {
|
|
|
|
*done = 1;
|
|
|
|
break;
|
2008-02-15 16:19:42 +08:00
|
|
|
}
|
2010-09-08 05:14:42 +08:00
|
|
|
skb->dev = card->dev;
|
2008-02-15 16:19:42 +08:00
|
|
|
switch (hdr->hdr.l2.id) {
|
|
|
|
case QETH_HEADER_TYPE_LAYER2:
|
|
|
|
skb->pkt_type = PACKET_HOST;
|
|
|
|
skb->protocol = eth_type_trans(skb, skb->dev);
|
2015-09-18 22:06:51 +08:00
|
|
|
if ((card->dev->features & NETIF_F_RXCSUM)
|
|
|
|
&& ((hdr->hdr.l2.flags[1] &
|
|
|
|
(QETH_HDR_EXT_CSUM_HDR_REQ |
|
|
|
|
QETH_HDR_EXT_CSUM_TRANSP_REQ)) ==
|
|
|
|
(QETH_HDR_EXT_CSUM_HDR_REQ |
|
|
|
|
QETH_HDR_EXT_CSUM_TRANSP_REQ)))
|
|
|
|
skb->ip_summed = CHECKSUM_UNNECESSARY;
|
|
|
|
else
|
|
|
|
skb->ip_summed = CHECKSUM_NONE;
|
2008-04-01 16:26:56 +08:00
|
|
|
if (skb->protocol == htons(ETH_P_802_2))
|
|
|
|
*((__u32 *)skb->cb) = ++card->seqno.pkt_seqno;
|
2008-02-15 16:19:42 +08:00
|
|
|
len = skb->len;
|
2015-10-06 21:12:27 +08:00
|
|
|
napi_gro_receive(&card->napi, skb);
|
2008-02-15 16:19:42 +08:00
|
|
|
break;
|
|
|
|
case QETH_HEADER_TYPE_OSN:
|
2008-10-24 17:16:55 +08:00
|
|
|
if (card->info.type == QETH_CARD_TYPE_OSN) {
|
|
|
|
skb_push(skb, sizeof(struct qeth_hdr));
|
|
|
|
skb_copy_to_linear_data(skb, hdr,
|
2008-02-15 16:19:42 +08:00
|
|
|
sizeof(struct qeth_hdr));
|
2008-10-24 17:16:55 +08:00
|
|
|
len = skb->len;
|
|
|
|
card->osn_info.data_cb(skb);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
/* else unknown */
|
2008-02-15 16:19:42 +08:00
|
|
|
default:
|
|
|
|
dev_kfree_skb_any(skb);
|
2010-06-22 06:57:05 +08:00
|
|
|
QETH_CARD_TEXT(card, 3, "inbunkno");
|
2008-04-01 16:26:58 +08:00
|
|
|
QETH_DBF_HEX(CTRL, 3, hdr, QETH_DBF_CTRL_LEN);
|
2008-02-15 16:19:42 +08:00
|
|
|
continue;
|
|
|
|
}
|
2010-09-08 05:14:42 +08:00
|
|
|
work_done++;
|
|
|
|
budget--;
|
2008-02-15 16:19:42 +08:00
|
|
|
card->stats.rx_packets++;
|
|
|
|
card->stats.rx_bytes += len;
|
|
|
|
}
|
2010-09-08 05:14:42 +08:00
|
|
|
return work_done;
|
|
|
|
}
|
|
|
|
|
2008-02-15 16:19:42 +08:00
|
|
|
static int qeth_l2_request_initial_mac(struct qeth_card *card)
|
|
|
|
{
|
|
|
|
int rc = 0;
|
|
|
|
char vendor_pre[] = {0x02, 0x00, 0x00};
|
|
|
|
|
2008-04-01 16:26:58 +08:00
|
|
|
QETH_DBF_TEXT(SETUP, 2, "doL2init");
|
|
|
|
QETH_DBF_TEXT_(SETUP, 2, "doL2%s", CARD_BUS_ID(card));
|
2008-02-15 16:19:42 +08:00
|
|
|
|
2012-11-13 07:05:16 +08:00
|
|
|
if (qeth_is_supported(card, IPA_SETADAPTERPARMS)) {
|
|
|
|
rc = qeth_query_setadapterparms(card);
|
|
|
|
if (rc) {
|
|
|
|
QETH_DBF_MESSAGE(2, "could not query adapter "
|
|
|
|
"parameters on device %s: x%x\n",
|
|
|
|
CARD_BUS_ID(card), rc);
|
|
|
|
}
|
2008-02-15 16:19:42 +08:00
|
|
|
}
|
|
|
|
|
2010-05-17 05:15:14 +08:00
|
|
|
if (card->info.type == QETH_CARD_TYPE_IQD ||
|
|
|
|
card->info.type == QETH_CARD_TYPE_OSM ||
|
|
|
|
card->info.type == QETH_CARD_TYPE_OSX ||
|
|
|
|
card->info.guestlan) {
|
2008-02-15 16:19:42 +08:00
|
|
|
rc = qeth_setadpparms_change_macaddr(card);
|
|
|
|
if (rc) {
|
2008-06-06 18:37:46 +08:00
|
|
|
QETH_DBF_MESSAGE(2, "couldn't get MAC address on "
|
|
|
|
"device %s: x%x\n", CARD_BUS_ID(card), rc);
|
2015-01-21 20:39:10 +08:00
|
|
|
QETH_DBF_TEXT_(SETUP, 2, "1err%04x", rc);
|
2008-02-15 16:19:42 +08:00
|
|
|
return rc;
|
|
|
|
}
|
2008-04-01 16:26:58 +08:00
|
|
|
QETH_DBF_HEX(SETUP, 2, card->dev->dev_addr, OSA_ADDR_LEN);
|
2008-02-15 16:19:42 +08:00
|
|
|
} else {
|
2012-07-13 13:33:10 +08:00
|
|
|
eth_random_addr(card->dev->dev_addr);
|
2008-02-15 16:19:42 +08:00
|
|
|
memcpy(card->dev->dev_addr, vendor_pre, 3);
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int qeth_l2_set_mac_address(struct net_device *dev, void *p)
|
|
|
|
{
|
|
|
|
struct sockaddr *addr = p;
|
2008-07-26 17:24:10 +08:00
|
|
|
struct qeth_card *card = dev->ml_priv;
|
2008-02-15 16:19:42 +08:00
|
|
|
int rc = 0;
|
|
|
|
|
2010-06-22 06:57:05 +08:00
|
|
|
QETH_CARD_TEXT(card, 3, "setmac");
|
2008-02-15 16:19:42 +08:00
|
|
|
|
|
|
|
if (qeth_l2_verify_dev(dev) != QETH_REAL_CARD) {
|
2010-06-22 06:57:05 +08:00
|
|
|
QETH_CARD_TEXT(card, 3, "setmcINV");
|
2008-02-15 16:19:42 +08:00
|
|
|
return -EOPNOTSUPP;
|
|
|
|
}
|
|
|
|
|
2010-05-17 05:15:14 +08:00
|
|
|
if (card->info.type == QETH_CARD_TYPE_OSN ||
|
|
|
|
card->info.type == QETH_CARD_TYPE_OSM ||
|
|
|
|
card->info.type == QETH_CARD_TYPE_OSX) {
|
2010-06-22 06:57:05 +08:00
|
|
|
QETH_CARD_TEXT(card, 3, "setmcTYP");
|
2008-02-15 16:19:42 +08:00
|
|
|
return -EOPNOTSUPP;
|
|
|
|
}
|
2010-06-22 06:57:05 +08:00
|
|
|
QETH_CARD_HEX(card, 3, addr->sa_data, OSA_ADDR_LEN);
|
2009-03-25 04:57:18 +08:00
|
|
|
if (qeth_wait_for_threads(card, QETH_RECOVER_THREAD)) {
|
2010-06-22 06:57:05 +08:00
|
|
|
QETH_CARD_TEXT(card, 3, "setmcREC");
|
2009-03-25 04:57:18 +08:00
|
|
|
return -ERESTARTSYS;
|
|
|
|
}
|
2008-02-15 16:19:42 +08:00
|
|
|
rc = qeth_l2_send_delmac(card, &card->dev->dev_addr[0]);
|
2015-01-21 20:39:09 +08:00
|
|
|
if (!rc || (rc == -ENOENT))
|
2008-02-15 16:19:42 +08:00
|
|
|
rc = qeth_l2_send_setmac(card, addr->sa_data);
|
2012-03-07 10:06:28 +08:00
|
|
|
return rc ? -EINVAL : 0;
|
2008-02-15 16:19:42 +08:00
|
|
|
}
|
|
|
|
|
2015-05-18 20:27:55 +08:00
|
|
|
static void qeth_promisc_to_bridge(struct qeth_card *card)
|
|
|
|
{
|
|
|
|
struct net_device *dev = card->dev;
|
|
|
|
enum qeth_ipa_promisc_modes promisc_mode;
|
|
|
|
int role;
|
|
|
|
int rc;
|
|
|
|
|
|
|
|
QETH_CARD_TEXT(card, 3, "pmisc2br");
|
|
|
|
|
|
|
|
if (!card->options.sbp.reflect_promisc)
|
|
|
|
return;
|
|
|
|
promisc_mode = (dev->flags & IFF_PROMISC) ? SET_PROMISC_MODE_ON
|
|
|
|
: SET_PROMISC_MODE_OFF;
|
|
|
|
if (promisc_mode == card->info.promisc_mode)
|
|
|
|
return;
|
|
|
|
|
|
|
|
if (promisc_mode == SET_PROMISC_MODE_ON) {
|
|
|
|
if (card->options.sbp.reflect_promisc_primary)
|
|
|
|
role = QETH_SBP_ROLE_PRIMARY;
|
|
|
|
else
|
|
|
|
role = QETH_SBP_ROLE_SECONDARY;
|
|
|
|
} else
|
|
|
|
role = QETH_SBP_ROLE_NONE;
|
|
|
|
|
|
|
|
rc = qeth_bridgeport_setrole(card, role);
|
|
|
|
QETH_DBF_TEXT_(SETUP, 2, "bpm%c%04x",
|
|
|
|
(promisc_mode == SET_PROMISC_MODE_ON) ? '+' : '-', rc);
|
|
|
|
if (!rc) {
|
|
|
|
card->options.sbp.role = role;
|
|
|
|
card->info.promisc_mode = promisc_mode;
|
|
|
|
}
|
2015-10-06 21:12:28 +08:00
|
|
|
|
|
|
|
}
|
|
|
|
/* New MAC address is added to the hash table and marked to be written on card
|
|
|
|
* only if there is not in the hash table storage already
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
static void
|
|
|
|
qeth_l2_add_mac(struct qeth_card *card, struct netdev_hw_addr *ha, u8 is_uc)
|
|
|
|
{
|
|
|
|
struct qeth_mac *mac;
|
|
|
|
|
|
|
|
hash_for_each_possible(card->mac_htable, mac, hnode,
|
|
|
|
qeth_l2_mac_hash(ha->addr)) {
|
|
|
|
if (is_uc == mac->is_uc &&
|
|
|
|
!memcmp(ha->addr, mac->mac_addr, OSA_ADDR_LEN)) {
|
2016-06-16 22:18:58 +08:00
|
|
|
mac->disp_flag = QETH_DISP_ADDR_DO_NOTHING;
|
2015-10-06 21:12:28 +08:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
mac = kzalloc(sizeof(struct qeth_mac), GFP_ATOMIC);
|
|
|
|
|
|
|
|
if (!mac)
|
|
|
|
return;
|
|
|
|
|
|
|
|
memcpy(mac->mac_addr, ha->addr, OSA_ADDR_LEN);
|
|
|
|
mac->is_uc = is_uc;
|
2016-06-16 22:18:58 +08:00
|
|
|
mac->disp_flag = QETH_DISP_ADDR_ADD;
|
2015-10-06 21:12:28 +08:00
|
|
|
|
|
|
|
hash_add(card->mac_htable, &mac->hnode,
|
|
|
|
qeth_l2_mac_hash(mac->mac_addr));
|
|
|
|
|
2015-05-18 20:27:55 +08:00
|
|
|
}
|
|
|
|
|
2015-10-06 21:12:28 +08:00
|
|
|
static void qeth_l2_set_rx_mode(struct net_device *dev)
|
2008-02-15 16:19:42 +08:00
|
|
|
{
|
2008-07-26 17:24:10 +08:00
|
|
|
struct qeth_card *card = dev->ml_priv;
|
2009-05-23 07:22:17 +08:00
|
|
|
struct netdev_hw_addr *ha;
|
2015-10-06 21:12:28 +08:00
|
|
|
struct qeth_mac *mac;
|
|
|
|
struct hlist_node *tmp;
|
|
|
|
int i;
|
|
|
|
int rc;
|
2008-02-15 16:19:42 +08:00
|
|
|
|
|
|
|
if (card->info.type == QETH_CARD_TYPE_OSN)
|
2015-10-06 21:12:28 +08:00
|
|
|
return;
|
2008-02-15 16:19:42 +08:00
|
|
|
|
2010-06-22 06:57:05 +08:00
|
|
|
QETH_CARD_TEXT(card, 3, "setmulti");
|
2009-03-25 04:57:18 +08:00
|
|
|
if (qeth_threads_running(card, QETH_RECOVER_THREAD) &&
|
|
|
|
(card->state != CARD_STATE_UP))
|
|
|
|
return;
|
2015-10-06 21:12:28 +08:00
|
|
|
|
2008-02-15 16:19:42 +08:00
|
|
|
spin_lock_bh(&card->mclock);
|
2015-10-06 21:12:28 +08:00
|
|
|
|
2010-04-02 05:22:57 +08:00
|
|
|
netdev_for_each_mc_addr(ha, dev)
|
2015-10-06 21:12:28 +08:00
|
|
|
qeth_l2_add_mac(card, ha, 0);
|
2008-08-15 14:02:59 +08:00
|
|
|
|
2010-01-26 05:36:10 +08:00
|
|
|
netdev_for_each_uc_addr(ha, dev)
|
2015-10-06 21:12:28 +08:00
|
|
|
qeth_l2_add_mac(card, ha, 1);
|
|
|
|
|
|
|
|
hash_for_each_safe(card->mac_htable, i, tmp, mac, hnode) {
|
2016-06-16 22:18:58 +08:00
|
|
|
if (mac->disp_flag == QETH_DISP_ADDR_DELETE) {
|
2017-01-12 22:48:40 +08:00
|
|
|
qeth_l2_remove_mac(card, mac);
|
2015-10-06 21:12:28 +08:00
|
|
|
hash_del(&mac->hnode);
|
|
|
|
kfree(mac);
|
|
|
|
|
2016-06-16 22:18:58 +08:00
|
|
|
} else if (mac->disp_flag == QETH_DISP_ADDR_ADD) {
|
2015-10-06 21:12:28 +08:00
|
|
|
rc = qeth_l2_write_mac(card, mac);
|
|
|
|
if (rc) {
|
|
|
|
hash_del(&mac->hnode);
|
|
|
|
kfree(mac);
|
|
|
|
} else
|
2016-06-16 22:18:58 +08:00
|
|
|
mac->disp_flag = QETH_DISP_ADDR_DELETE;
|
2015-10-06 21:12:28 +08:00
|
|
|
} else
|
2016-06-16 22:18:58 +08:00
|
|
|
mac->disp_flag = QETH_DISP_ADDR_DELETE;
|
2015-10-06 21:12:28 +08:00
|
|
|
}
|
2008-08-15 14:02:59 +08:00
|
|
|
|
2008-02-15 16:19:42 +08:00
|
|
|
spin_unlock_bh(&card->mclock);
|
2015-10-06 21:12:28 +08:00
|
|
|
|
2015-05-18 20:27:55 +08:00
|
|
|
if (qeth_adp_supported(card, IPA_SETADP_SET_PROMISC_MODE))
|
|
|
|
qeth_setadp_promisc_mode(card);
|
|
|
|
else
|
|
|
|
qeth_promisc_to_bridge(card);
|
2008-02-15 16:19:42 +08:00
|
|
|
}
|
|
|
|
|
2017-04-11 22:11:14 +08:00
|
|
|
static netdev_tx_t qeth_l2_hard_start_xmit(struct sk_buff *skb,
|
|
|
|
struct net_device *dev)
|
2008-02-15 16:19:42 +08:00
|
|
|
{
|
|
|
|
int rc;
|
|
|
|
struct qeth_hdr *hdr = NULL;
|
|
|
|
int elements = 0;
|
2008-07-26 17:24:10 +08:00
|
|
|
struct qeth_card *card = dev->ml_priv;
|
2008-02-15 16:19:42 +08:00
|
|
|
struct sk_buff *new_skb = skb;
|
2009-08-26 10:01:08 +08:00
|
|
|
int cast_type = qeth_l2_get_cast_type(card, skb);
|
2014-04-28 16:05:08 +08:00
|
|
|
struct qeth_qdio_out_q *queue;
|
2008-02-15 16:19:42 +08:00
|
|
|
int tx_bytes = skb->len;
|
2008-08-01 22:39:13 +08:00
|
|
|
int data_offset = -1;
|
|
|
|
int elements_needed = 0;
|
|
|
|
int hd_len = 0;
|
2016-06-16 22:18:55 +08:00
|
|
|
int nr_frags;
|
2008-02-15 16:19:42 +08:00
|
|
|
|
2014-04-28 16:05:08 +08:00
|
|
|
if (card->qdio.do_prio_queueing || (cast_type &&
|
|
|
|
card->info.is_multicast_different))
|
|
|
|
queue = card->qdio.out_qs[qeth_get_priority_queue(card, skb,
|
|
|
|
qeth_get_ip_version(skb), cast_type)];
|
|
|
|
else
|
|
|
|
queue = card->qdio.out_qs[card->qdio.default_out_queue];
|
|
|
|
|
2008-02-15 16:19:42 +08:00
|
|
|
if ((card->state != CARD_STATE_UP) || !card->lan_online) {
|
|
|
|
card->stats.tx_carrier_errors++;
|
|
|
|
goto tx_drop;
|
|
|
|
}
|
|
|
|
|
|
|
|
if ((card->info.type == QETH_CARD_TYPE_OSN) &&
|
|
|
|
(skb->protocol == htons(ETH_P_IPV6)))
|
|
|
|
goto tx_drop;
|
|
|
|
|
|
|
|
if (card->options.performance_stats) {
|
|
|
|
card->perf_stats.outbound_cnt++;
|
|
|
|
card->perf_stats.outbound_start_time = qeth_get_micros();
|
|
|
|
}
|
|
|
|
netif_stop_queue(dev);
|
|
|
|
|
2016-06-16 22:18:55 +08:00
|
|
|
/* fix hardware limitation: as long as we do not have sbal
|
|
|
|
* chaining we can not send long frag lists
|
|
|
|
*/
|
|
|
|
if ((card->info.type != QETH_CARD_TYPE_IQD) &&
|
2017-03-23 21:55:08 +08:00
|
|
|
!qeth_get_elements_no(card, new_skb, 0, 0)) {
|
2016-06-16 22:18:57 +08:00
|
|
|
int lin_rc = skb_linearize(new_skb);
|
|
|
|
|
|
|
|
if (card->options.performance_stats) {
|
|
|
|
if (lin_rc)
|
|
|
|
card->perf_stats.tx_linfail++;
|
|
|
|
else
|
|
|
|
card->perf_stats.tx_lin++;
|
|
|
|
}
|
|
|
|
if (lin_rc)
|
2016-06-16 22:18:55 +08:00
|
|
|
goto tx_drop;
|
|
|
|
}
|
|
|
|
|
2008-02-15 16:19:42 +08:00
|
|
|
if (card->info.type == QETH_CARD_TYPE_OSN)
|
|
|
|
hdr = (struct qeth_hdr *)skb->data;
|
|
|
|
else {
|
2009-03-25 04:57:16 +08:00
|
|
|
if (card->info.type == QETH_CARD_TYPE_IQD) {
|
2008-08-01 22:39:13 +08:00
|
|
|
new_skb = skb;
|
|
|
|
data_offset = ETH_HLEN;
|
|
|
|
hd_len = ETH_HLEN;
|
|
|
|
hdr = kmem_cache_alloc(qeth_core_header_cache,
|
|
|
|
GFP_ATOMIC);
|
|
|
|
if (!hdr)
|
|
|
|
goto tx_drop;
|
|
|
|
elements_needed++;
|
|
|
|
skb_reset_mac_header(new_skb);
|
2014-03-19 14:58:01 +08:00
|
|
|
qeth_l2_fill_header(card, hdr, new_skb, cast_type);
|
2008-08-01 22:39:13 +08:00
|
|
|
hdr->hdr.l2.pkt_length = new_skb->len;
|
|
|
|
memcpy(((char *)hdr) + sizeof(struct qeth_hdr),
|
|
|
|
skb_mac_header(new_skb), ETH_HLEN);
|
|
|
|
} else {
|
|
|
|
/* create a clone with writeable headroom */
|
|
|
|
new_skb = skb_realloc_headroom(skb,
|
|
|
|
sizeof(struct qeth_hdr));
|
|
|
|
if (!new_skb)
|
|
|
|
goto tx_drop;
|
|
|
|
hdr = (struct qeth_hdr *)skb_push(new_skb,
|
2008-04-24 16:15:24 +08:00
|
|
|
sizeof(struct qeth_hdr));
|
2008-08-01 22:39:13 +08:00
|
|
|
skb_set_mac_header(new_skb, sizeof(struct qeth_hdr));
|
2014-03-19 14:58:01 +08:00
|
|
|
qeth_l2_fill_header(card, hdr, new_skb, cast_type);
|
2015-09-18 22:06:51 +08:00
|
|
|
if (new_skb->ip_summed == CHECKSUM_PARTIAL)
|
|
|
|
qeth_l2_hdr_csum(card, hdr, new_skb);
|
2008-08-01 22:39:13 +08:00
|
|
|
}
|
2008-02-15 16:19:42 +08:00
|
|
|
}
|
|
|
|
|
2017-03-23 21:55:08 +08:00
|
|
|
elements = qeth_get_elements_no(card, new_skb, elements_needed,
|
|
|
|
(data_offset > 0) ? data_offset : 0);
|
2009-03-25 04:57:16 +08:00
|
|
|
if (!elements) {
|
|
|
|
if (data_offset >= 0)
|
|
|
|
kmem_cache_free(qeth_core_header_cache, hdr);
|
|
|
|
goto tx_drop;
|
2009-03-25 04:57:15 +08:00
|
|
|
}
|
2008-02-15 16:19:42 +08:00
|
|
|
|
2010-06-22 06:57:10 +08:00
|
|
|
if (card->info.type != QETH_CARD_TYPE_IQD) {
|
2013-04-22 09:12:29 +08:00
|
|
|
if (qeth_hdr_chk_and_bounce(new_skb, &hdr,
|
2010-06-22 06:57:10 +08:00
|
|
|
sizeof(struct qeth_hdr_layer2)))
|
|
|
|
goto tx_drop;
|
2008-02-15 16:19:42 +08:00
|
|
|
rc = qeth_do_send_packet(card, queue, new_skb, hdr,
|
2009-03-25 04:57:16 +08:00
|
|
|
elements);
|
2010-06-22 06:57:10 +08:00
|
|
|
} else
|
2008-02-15 16:19:42 +08:00
|
|
|
rc = qeth_do_send_packet_fast(card, queue, new_skb, hdr,
|
2017-04-11 22:11:13 +08:00
|
|
|
data_offset, hd_len);
|
2008-02-15 16:19:42 +08:00
|
|
|
if (!rc) {
|
|
|
|
card->stats.tx_packets++;
|
|
|
|
card->stats.tx_bytes += tx_bytes;
|
2016-06-16 22:18:55 +08:00
|
|
|
if (card->options.performance_stats) {
|
|
|
|
nr_frags = skb_shinfo(new_skb)->nr_frags;
|
|
|
|
if (nr_frags) {
|
|
|
|
card->perf_stats.sg_skbs_sent++;
|
|
|
|
/* nr_frags + skb->data */
|
|
|
|
card->perf_stats.sg_frags_sent += nr_frags + 1;
|
|
|
|
}
|
|
|
|
}
|
2008-02-15 16:19:42 +08:00
|
|
|
if (new_skb != skb)
|
|
|
|
dev_kfree_skb_any(skb);
|
2009-07-06 10:23:38 +08:00
|
|
|
rc = NETDEV_TX_OK;
|
2008-02-15 16:19:42 +08:00
|
|
|
} else {
|
2008-08-01 22:39:13 +08:00
|
|
|
if (data_offset >= 0)
|
|
|
|
kmem_cache_free(qeth_core_header_cache, hdr);
|
|
|
|
|
2008-02-15 16:19:42 +08:00
|
|
|
if (rc == -EBUSY) {
|
|
|
|
if (new_skb != skb)
|
|
|
|
dev_kfree_skb_any(new_skb);
|
|
|
|
return NETDEV_TX_BUSY;
|
|
|
|
} else
|
|
|
|
goto tx_drop;
|
|
|
|
}
|
|
|
|
|
|
|
|
netif_wake_queue(dev);
|
|
|
|
if (card->options.performance_stats)
|
|
|
|
card->perf_stats.outbound_time += qeth_get_micros() -
|
|
|
|
card->perf_stats.outbound_start_time;
|
|
|
|
return rc;
|
|
|
|
|
|
|
|
tx_drop:
|
|
|
|
card->stats.tx_dropped++;
|
|
|
|
card->stats.tx_errors++;
|
|
|
|
if ((new_skb != skb) && new_skb)
|
|
|
|
dev_kfree_skb_any(new_skb);
|
|
|
|
dev_kfree_skb_any(skb);
|
2008-06-06 18:37:48 +08:00
|
|
|
netif_wake_queue(dev);
|
2008-02-15 16:19:42 +08:00
|
|
|
return NETDEV_TX_OK;
|
|
|
|
}
|
|
|
|
|
2011-01-13 04:42:24 +08:00
|
|
|
static int __qeth_l2_open(struct net_device *dev)
|
2008-02-15 16:19:42 +08:00
|
|
|
{
|
2008-07-26 17:24:10 +08:00
|
|
|
struct qeth_card *card = dev->ml_priv;
|
2010-09-08 05:14:42 +08:00
|
|
|
int rc = 0;
|
2008-02-15 16:19:42 +08:00
|
|
|
|
2010-06-22 06:57:05 +08:00
|
|
|
QETH_CARD_TEXT(card, 4, "qethopen");
|
2011-01-13 04:42:24 +08:00
|
|
|
if (card->state == CARD_STATE_UP)
|
|
|
|
return rc;
|
2008-02-15 16:19:42 +08:00
|
|
|
if (card->state != CARD_STATE_SOFTSETUP)
|
|
|
|
return -ENODEV;
|
|
|
|
|
|
|
|
if ((card->info.type != QETH_CARD_TYPE_OSN) &&
|
|
|
|
(!(card->info.mac_bits & QETH_LAYER2_MAC_REGISTERED))) {
|
2010-06-22 06:57:05 +08:00
|
|
|
QETH_CARD_TEXT(card, 4, "nomacadr");
|
2008-02-15 16:19:42 +08:00
|
|
|
return -EPERM;
|
|
|
|
}
|
|
|
|
card->data.state = CH_STATE_UP;
|
|
|
|
card->state = CARD_STATE_UP;
|
|
|
|
netif_start_queue(dev);
|
|
|
|
|
2010-09-08 05:14:42 +08:00
|
|
|
if (qdio_stop_irq(card->data.ccwdev, 0) >= 0) {
|
|
|
|
napi_enable(&card->napi);
|
|
|
|
napi_schedule(&card->napi);
|
|
|
|
} else
|
|
|
|
rc = -EIO;
|
|
|
|
return rc;
|
2008-02-15 16:19:42 +08:00
|
|
|
}
|
|
|
|
|
2011-01-13 04:42:24 +08:00
|
|
|
static int qeth_l2_open(struct net_device *dev)
|
|
|
|
{
|
|
|
|
struct qeth_card *card = dev->ml_priv;
|
|
|
|
|
|
|
|
QETH_CARD_TEXT(card, 5, "qethope_");
|
|
|
|
if (qeth_wait_for_threads(card, QETH_RECOVER_THREAD)) {
|
|
|
|
QETH_CARD_TEXT(card, 3, "openREC");
|
|
|
|
return -ERESTARTSYS;
|
|
|
|
}
|
|
|
|
return __qeth_l2_open(dev);
|
|
|
|
}
|
|
|
|
|
2008-02-15 16:19:42 +08:00
|
|
|
static int qeth_l2_stop(struct net_device *dev)
|
|
|
|
{
|
2008-07-26 17:24:10 +08:00
|
|
|
struct qeth_card *card = dev->ml_priv;
|
2008-02-15 16:19:42 +08:00
|
|
|
|
2010-06-22 06:57:05 +08:00
|
|
|
QETH_CARD_TEXT(card, 4, "qethstop");
|
2008-02-15 16:19:42 +08:00
|
|
|
netif_tx_disable(dev);
|
2010-09-08 05:14:42 +08:00
|
|
|
if (card->state == CARD_STATE_UP) {
|
2008-02-15 16:19:42 +08:00
|
|
|
card->state = CARD_STATE_SOFTSETUP;
|
2010-09-08 05:14:42 +08:00
|
|
|
napi_disable(&card->napi);
|
|
|
|
}
|
2008-02-15 16:19:42 +08:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2017-05-11 01:07:52 +08:00
|
|
|
static const struct device_type qeth_l2_devtype = {
|
|
|
|
.name = "qeth_layer2",
|
|
|
|
.groups = qeth_l2_attr_groups,
|
|
|
|
};
|
|
|
|
|
2008-02-15 16:19:42 +08:00
|
|
|
static int qeth_l2_probe_device(struct ccwgroup_device *gdev)
|
|
|
|
{
|
|
|
|
struct qeth_card *card = dev_get_drvdata(&gdev->dev);
|
2017-05-11 01:07:51 +08:00
|
|
|
int rc;
|
2008-02-15 16:19:42 +08:00
|
|
|
|
2017-05-11 01:07:52 +08:00
|
|
|
if (gdev->dev.type == &qeth_generic_devtype) {
|
|
|
|
rc = qeth_l2_create_device_attributes(&gdev->dev);
|
|
|
|
if (rc)
|
|
|
|
return rc;
|
|
|
|
}
|
2008-02-15 16:19:42 +08:00
|
|
|
INIT_LIST_HEAD(&card->vid_list);
|
2015-10-06 21:12:28 +08:00
|
|
|
hash_init(card->mac_htable);
|
2008-02-15 16:19:42 +08:00
|
|
|
card->options.layer2 = 1;
|
2011-05-13 02:45:02 +08:00
|
|
|
card->info.hwtrap = 0;
|
2008-02-15 16:19:42 +08:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void qeth_l2_remove_device(struct ccwgroup_device *cgdev)
|
|
|
|
{
|
|
|
|
struct qeth_card *card = dev_get_drvdata(&cgdev->dev);
|
|
|
|
|
2017-05-11 01:07:52 +08:00
|
|
|
if (cgdev->dev.type == &qeth_generic_devtype)
|
|
|
|
qeth_l2_remove_device_attributes(&cgdev->dev);
|
2009-05-20 05:38:37 +08:00
|
|
|
qeth_set_allowed_threads(card, 0, 1);
|
2008-02-15 16:19:42 +08:00
|
|
|
wait_event(card->wait_q, qeth_threads_running(card, 0xffffffff) == 0);
|
|
|
|
|
2011-02-27 14:41:36 +08:00
|
|
|
if (cgdev->state == CCWGROUP_ONLINE)
|
2008-02-15 16:19:42 +08:00
|
|
|
qeth_l2_set_offline(cgdev);
|
|
|
|
|
|
|
|
if (card->dev) {
|
2016-07-04 20:07:16 +08:00
|
|
|
netif_napi_del(&card->napi);
|
2008-02-15 16:19:42 +08:00
|
|
|
unregister_netdev(card->dev);
|
|
|
|
card->dev = NULL;
|
|
|
|
}
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2009-09-02 16:03:33 +08:00
|
|
|
static const struct ethtool_ops qeth_l2_ethtool_ops = {
|
2008-02-15 16:19:42 +08:00
|
|
|
.get_link = ethtool_op_get_link,
|
|
|
|
.get_strings = qeth_core_get_strings,
|
|
|
|
.get_ethtool_stats = qeth_core_get_ethtool_stats,
|
2009-10-01 19:24:32 +08:00
|
|
|
.get_sset_count = qeth_core_get_sset_count,
|
2008-02-15 16:19:42 +08:00
|
|
|
.get_drvinfo = qeth_core_get_drvinfo,
|
2017-04-11 22:11:17 +08:00
|
|
|
.get_link_ksettings = qeth_core_ethtool_get_link_ksettings,
|
2008-02-15 16:19:42 +08:00
|
|
|
};
|
|
|
|
|
2009-09-02 16:03:33 +08:00
|
|
|
static const struct ethtool_ops qeth_l2_osn_ops = {
|
2008-02-15 16:19:42 +08:00
|
|
|
.get_strings = qeth_core_get_strings,
|
|
|
|
.get_ethtool_stats = qeth_core_get_ethtool_stats,
|
2009-10-01 19:24:32 +08:00
|
|
|
.get_sset_count = qeth_core_get_sset_count,
|
2008-02-15 16:19:42 +08:00
|
|
|
.get_drvinfo = qeth_core_get_drvinfo,
|
|
|
|
};
|
|
|
|
|
2009-01-09 11:44:00 +08:00
|
|
|
static const struct net_device_ops qeth_l2_netdev_ops = {
|
2009-01-09 02:50:55 +08:00
|
|
|
.ndo_open = qeth_l2_open,
|
|
|
|
.ndo_stop = qeth_l2_stop,
|
|
|
|
.ndo_get_stats = qeth_get_stats,
|
|
|
|
.ndo_start_xmit = qeth_l2_hard_start_xmit,
|
|
|
|
.ndo_validate_addr = eth_validate_addr,
|
2015-10-06 21:12:28 +08:00
|
|
|
.ndo_set_rx_mode = qeth_l2_set_rx_mode,
|
2017-04-11 22:11:10 +08:00
|
|
|
.ndo_do_ioctl = qeth_do_ioctl,
|
2009-01-09 02:50:55 +08:00
|
|
|
.ndo_set_mac_address = qeth_l2_set_mac_address,
|
|
|
|
.ndo_change_mtu = qeth_change_mtu,
|
|
|
|
.ndo_vlan_rx_add_vid = qeth_l2_vlan_rx_add_vid,
|
|
|
|
.ndo_vlan_rx_kill_vid = qeth_l2_vlan_rx_kill_vid,
|
|
|
|
.ndo_tx_timeout = qeth_tx_timeout,
|
2016-06-16 22:18:59 +08:00
|
|
|
.ndo_fix_features = qeth_fix_features,
|
|
|
|
.ndo_set_features = qeth_set_features
|
2009-01-09 02:50:55 +08:00
|
|
|
};
|
|
|
|
|
2008-02-15 16:19:42 +08:00
|
|
|
static int qeth_l2_setup_netdev(struct qeth_card *card)
|
|
|
|
{
|
|
|
|
switch (card->info.type) {
|
|
|
|
case QETH_CARD_TYPE_IQD:
|
net: set name_assign_type in alloc_netdev()
Extend alloc_netdev{,_mq{,s}}() to take name_assign_type as argument, and convert
all users to pass NET_NAME_UNKNOWN.
Coccinelle patch:
@@
expression sizeof_priv, name, setup, txqs, rxqs, count;
@@
(
-alloc_netdev_mqs(sizeof_priv, name, setup, txqs, rxqs)
+alloc_netdev_mqs(sizeof_priv, name, NET_NAME_UNKNOWN, setup, txqs, rxqs)
|
-alloc_netdev_mq(sizeof_priv, name, setup, count)
+alloc_netdev_mq(sizeof_priv, name, NET_NAME_UNKNOWN, setup, count)
|
-alloc_netdev(sizeof_priv, name, setup)
+alloc_netdev(sizeof_priv, name, NET_NAME_UNKNOWN, setup)
)
v9: move comments here from the wrong commit
Signed-off-by: Tom Gundersen <teg@jklm.no>
Reviewed-by: David Herrmann <dh.herrmann@gmail.com>
Signed-off-by: David S. Miller <davem@davemloft.net>
2014-07-14 22:37:24 +08:00
|
|
|
card->dev = alloc_netdev(0, "hsi%d", NET_NAME_UNKNOWN,
|
|
|
|
ether_setup);
|
2008-02-15 16:19:42 +08:00
|
|
|
break;
|
|
|
|
case QETH_CARD_TYPE_OSN:
|
net: set name_assign_type in alloc_netdev()
Extend alloc_netdev{,_mq{,s}}() to take name_assign_type as argument, and convert
all users to pass NET_NAME_UNKNOWN.
Coccinelle patch:
@@
expression sizeof_priv, name, setup, txqs, rxqs, count;
@@
(
-alloc_netdev_mqs(sizeof_priv, name, setup, txqs, rxqs)
+alloc_netdev_mqs(sizeof_priv, name, NET_NAME_UNKNOWN, setup, txqs, rxqs)
|
-alloc_netdev_mq(sizeof_priv, name, setup, count)
+alloc_netdev_mq(sizeof_priv, name, NET_NAME_UNKNOWN, setup, count)
|
-alloc_netdev(sizeof_priv, name, setup)
+alloc_netdev(sizeof_priv, name, NET_NAME_UNKNOWN, setup)
)
v9: move comments here from the wrong commit
Signed-off-by: Tom Gundersen <teg@jklm.no>
Reviewed-by: David Herrmann <dh.herrmann@gmail.com>
Signed-off-by: David S. Miller <davem@davemloft.net>
2014-07-14 22:37:24 +08:00
|
|
|
card->dev = alloc_netdev(0, "osn%d", NET_NAME_UNKNOWN,
|
|
|
|
ether_setup);
|
2008-02-15 16:19:42 +08:00
|
|
|
card->dev->flags |= IFF_NOARP;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
card->dev = alloc_etherdev(0);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!card->dev)
|
|
|
|
return -ENODEV;
|
|
|
|
|
2008-07-26 17:24:10 +08:00
|
|
|
card->dev->ml_priv = card;
|
2008-02-15 16:19:42 +08:00
|
|
|
card->dev->watchdog_timeo = QETH_TX_TIMEOUT;
|
|
|
|
card->dev->mtu = card->info.initial_mtu;
|
2016-10-21 01:55:23 +08:00
|
|
|
card->dev->min_mtu = 64;
|
|
|
|
card->dev->max_mtu = ETH_MAX_MTU;
|
2009-01-09 02:50:55 +08:00
|
|
|
card->dev->netdev_ops = &qeth_l2_netdev_ops;
|
2014-05-11 08:12:32 +08:00
|
|
|
card->dev->ethtool_ops =
|
|
|
|
(card->info.type != QETH_CARD_TYPE_OSN) ?
|
|
|
|
&qeth_l2_ethtool_ops : &qeth_l2_osn_ops;
|
2013-04-19 10:04:27 +08:00
|
|
|
card->dev->features |= NETIF_F_HW_VLAN_CTAG_FILTER;
|
2015-09-18 22:06:51 +08:00
|
|
|
if (card->info.type == QETH_CARD_TYPE_OSD && !card->info.guestlan) {
|
2016-06-16 22:18:59 +08:00
|
|
|
card->dev->hw_features = NETIF_F_SG;
|
2016-06-16 22:19:00 +08:00
|
|
|
card->dev->vlan_features = NETIF_F_SG;
|
2016-06-16 22:18:59 +08:00
|
|
|
/* OSA 3S and earlier has no RX/TX support */
|
2016-06-16 22:19:00 +08:00
|
|
|
if (qeth_is_supported(card, IPA_OUTBOUND_CHECKSUM)) {
|
2016-06-16 22:18:59 +08:00
|
|
|
card->dev->hw_features |= NETIF_F_IP_CSUM;
|
2016-06-16 22:19:00 +08:00
|
|
|
card->dev->vlan_features |= NETIF_F_IP_CSUM;
|
|
|
|
}
|
|
|
|
if (qeth_is_supported(card, IPA_INBOUND_CHECKSUM)) {
|
2016-06-16 22:18:59 +08:00
|
|
|
card->dev->hw_features |= NETIF_F_RXCSUM;
|
2016-06-16 22:19:00 +08:00
|
|
|
card->dev->vlan_features |= NETIF_F_RXCSUM;
|
|
|
|
}
|
2015-09-18 22:06:51 +08:00
|
|
|
}
|
2008-02-15 16:19:42 +08:00
|
|
|
card->info.broadcast_capable = 1;
|
|
|
|
qeth_l2_request_initial_mac(card);
|
2016-06-16 22:18:55 +08:00
|
|
|
card->dev->gso_max_size = (QETH_MAX_BUFFER_ELEMENTS(card) - 1) *
|
|
|
|
PAGE_SIZE;
|
2008-02-15 16:19:42 +08:00
|
|
|
SET_NETDEV_DEV(card->dev, &card->gdev->dev);
|
2017-04-11 22:11:11 +08:00
|
|
|
netif_napi_add(card->dev, &card->napi, qeth_poll, QETH_NAPI_WEIGHT);
|
2015-12-11 19:27:55 +08:00
|
|
|
netif_carrier_off(card->dev);
|
2008-02-15 16:19:42 +08:00
|
|
|
return register_netdev(card->dev);
|
|
|
|
}
|
|
|
|
|
2015-09-18 22:06:51 +08:00
|
|
|
static int qeth_l2_start_ipassists(struct qeth_card *card)
|
|
|
|
{
|
|
|
|
/* configure isolation level */
|
|
|
|
if (qeth_set_access_ctrl_online(card, 0))
|
|
|
|
return -ENODEV;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2008-02-15 16:19:42 +08:00
|
|
|
static int __qeth_l2_set_online(struct ccwgroup_device *gdev, int recovery_mode)
|
|
|
|
{
|
|
|
|
struct qeth_card *card = dev_get_drvdata(&gdev->dev);
|
|
|
|
int rc = 0;
|
|
|
|
enum qeth_card_states recover_flag;
|
|
|
|
|
2010-07-23 07:15:05 +08:00
|
|
|
mutex_lock(&card->discipline_mutex);
|
2010-05-12 03:34:47 +08:00
|
|
|
mutex_lock(&card->conf_mutex);
|
2008-04-01 16:26:58 +08:00
|
|
|
QETH_DBF_TEXT(SETUP, 2, "setonlin");
|
|
|
|
QETH_DBF_HEX(SETUP, 2, &card, sizeof(void *));
|
2008-02-15 16:19:42 +08:00
|
|
|
|
|
|
|
recover_flag = card->state;
|
|
|
|
rc = qeth_core_hardsetup_card(card);
|
|
|
|
if (rc) {
|
2015-01-21 20:39:10 +08:00
|
|
|
QETH_DBF_TEXT_(SETUP, 2, "2err%04x", rc);
|
2009-11-12 08:11:43 +08:00
|
|
|
rc = -ENODEV;
|
2008-02-15 16:19:42 +08:00
|
|
|
goto out_remove;
|
|
|
|
}
|
2014-01-29 16:23:48 +08:00
|
|
|
qeth_bridgeport_query_support(card);
|
|
|
|
if (card->options.sbp.supported_funcs)
|
|
|
|
dev_info(&card->gdev->dev,
|
2015-05-18 20:27:56 +08:00
|
|
|
"The device represents a Bridge Capable Port\n");
|
2012-11-19 10:46:49 +08:00
|
|
|
qeth_trace_features(card);
|
2008-02-15 16:19:42 +08:00
|
|
|
|
2009-11-12 08:11:43 +08:00
|
|
|
if (!card->dev && qeth_l2_setup_netdev(card)) {
|
|
|
|
rc = -ENODEV;
|
2008-02-15 16:19:42 +08:00
|
|
|
goto out_remove;
|
2009-11-12 08:11:43 +08:00
|
|
|
}
|
2008-02-15 16:19:42 +08:00
|
|
|
|
|
|
|
if (card->info.type != QETH_CARD_TYPE_OSN)
|
|
|
|
qeth_l2_send_setmac(card, &card->dev->dev_addr[0]);
|
|
|
|
|
2011-05-13 02:45:02 +08:00
|
|
|
if (qeth_is_diagass_supported(card, QETH_DIAGS_CMD_TRAP)) {
|
|
|
|
if (card->info.hwtrap &&
|
|
|
|
qeth_hw_trap(card, QETH_DIAGS_TRAP_ARM))
|
|
|
|
card->info.hwtrap = 0;
|
|
|
|
} else
|
|
|
|
card->info.hwtrap = 0;
|
|
|
|
|
2014-01-14 22:54:11 +08:00
|
|
|
qeth_l2_setup_bridgeport_attrs(card);
|
|
|
|
|
2008-02-15 16:19:42 +08:00
|
|
|
card->state = CARD_STATE_HARDSETUP;
|
2010-09-08 05:14:42 +08:00
|
|
|
memset(&card->rx, 0, sizeof(struct qeth_rx));
|
2008-02-15 16:19:42 +08:00
|
|
|
qeth_print_status_message(card);
|
|
|
|
|
|
|
|
/* softsetup */
|
2008-04-01 16:26:58 +08:00
|
|
|
QETH_DBF_TEXT(SETUP, 2, "softsetp");
|
2008-02-15 16:19:42 +08:00
|
|
|
|
2010-05-17 05:15:14 +08:00
|
|
|
if ((card->info.type == QETH_CARD_TYPE_OSD) ||
|
2013-01-21 10:30:20 +08:00
|
|
|
(card->info.type == QETH_CARD_TYPE_OSX)) {
|
2016-06-16 22:18:59 +08:00
|
|
|
rc = qeth_l2_start_ipassists(card);
|
|
|
|
if (rc)
|
2013-01-21 10:30:20 +08:00
|
|
|
goto out_remove;
|
|
|
|
}
|
2010-05-17 05:15:14 +08:00
|
|
|
|
|
|
|
if (card->info.type != QETH_CARD_TYPE_OSN &&
|
|
|
|
card->info.type != QETH_CARD_TYPE_OSM)
|
2011-02-27 14:41:36 +08:00
|
|
|
qeth_l2_process_vlans(card);
|
2008-02-15 16:19:42 +08:00
|
|
|
|
|
|
|
netif_tx_disable(card->dev);
|
|
|
|
|
|
|
|
rc = qeth_init_qdio_queues(card);
|
|
|
|
if (rc) {
|
2008-04-01 16:26:58 +08:00
|
|
|
QETH_DBF_TEXT_(SETUP, 2, "6err%d", rc);
|
2009-11-12 08:11:43 +08:00
|
|
|
rc = -ENODEV;
|
2008-02-15 16:19:42 +08:00
|
|
|
goto out_remove;
|
|
|
|
}
|
|
|
|
card->state = CARD_STATE_SOFTSETUP;
|
2010-11-26 10:41:19 +08:00
|
|
|
if (card->lan_online)
|
|
|
|
netif_carrier_on(card->dev);
|
|
|
|
else
|
|
|
|
netif_carrier_off(card->dev);
|
2008-02-15 16:19:42 +08:00
|
|
|
|
|
|
|
qeth_set_allowed_threads(card, 0xffffffff, 0);
|
|
|
|
if (recover_flag == CARD_STATE_RECOVER) {
|
|
|
|
if (recovery_mode &&
|
|
|
|
card->info.type != QETH_CARD_TYPE_OSN) {
|
2011-01-13 04:42:24 +08:00
|
|
|
__qeth_l2_open(card->dev);
|
2008-02-15 16:19:42 +08:00
|
|
|
} else {
|
|
|
|
rtnl_lock();
|
|
|
|
dev_open(card->dev);
|
|
|
|
rtnl_unlock();
|
|
|
|
}
|
|
|
|
/* this also sets saved unicast addresses */
|
2015-10-06 21:12:28 +08:00
|
|
|
qeth_l2_set_rx_mode(card->dev);
|
2016-09-15 20:39:21 +08:00
|
|
|
rtnl_lock();
|
|
|
|
qeth_recover_features(card->dev);
|
|
|
|
rtnl_unlock();
|
2008-02-15 16:19:42 +08:00
|
|
|
}
|
|
|
|
/* let user_space know that device is online */
|
|
|
|
kobject_uevent(&gdev->dev.kobj, KOBJ_CHANGE);
|
2010-05-12 03:34:47 +08:00
|
|
|
mutex_unlock(&card->conf_mutex);
|
2010-07-23 07:15:05 +08:00
|
|
|
mutex_unlock(&card->discipline_mutex);
|
2008-02-15 16:19:42 +08:00
|
|
|
return 0;
|
2009-11-12 08:11:43 +08:00
|
|
|
|
2008-02-15 16:19:42 +08:00
|
|
|
out_remove:
|
|
|
|
qeth_l2_stop_card(card, 0);
|
|
|
|
ccw_device_set_offline(CARD_DDEV(card));
|
|
|
|
ccw_device_set_offline(CARD_WDEV(card));
|
|
|
|
ccw_device_set_offline(CARD_RDEV(card));
|
2014-02-24 20:12:06 +08:00
|
|
|
qdio_free(CARD_DDEV(card));
|
2008-02-15 16:19:42 +08:00
|
|
|
if (recover_flag == CARD_STATE_RECOVER)
|
|
|
|
card->state = CARD_STATE_RECOVER;
|
|
|
|
else
|
|
|
|
card->state = CARD_STATE_DOWN;
|
2010-05-12 03:34:47 +08:00
|
|
|
mutex_unlock(&card->conf_mutex);
|
2010-07-23 07:15:05 +08:00
|
|
|
mutex_unlock(&card->discipline_mutex);
|
2009-11-12 08:11:43 +08:00
|
|
|
return rc;
|
2008-02-15 16:19:42 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
static int qeth_l2_set_online(struct ccwgroup_device *gdev)
|
|
|
|
{
|
|
|
|
return __qeth_l2_set_online(gdev, 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
static int __qeth_l2_set_offline(struct ccwgroup_device *cgdev,
|
|
|
|
int recovery_mode)
|
|
|
|
{
|
|
|
|
struct qeth_card *card = dev_get_drvdata(&cgdev->dev);
|
|
|
|
int rc = 0, rc2 = 0, rc3 = 0;
|
|
|
|
enum qeth_card_states recover_flag;
|
|
|
|
|
2010-07-23 07:15:05 +08:00
|
|
|
mutex_lock(&card->discipline_mutex);
|
2010-05-12 03:34:47 +08:00
|
|
|
mutex_lock(&card->conf_mutex);
|
2008-04-01 16:26:58 +08:00
|
|
|
QETH_DBF_TEXT(SETUP, 3, "setoffl");
|
|
|
|
QETH_DBF_HEX(SETUP, 3, &card, sizeof(void *));
|
2008-02-15 16:19:42 +08:00
|
|
|
|
|
|
|
if (card->dev && netif_carrier_ok(card->dev))
|
|
|
|
netif_carrier_off(card->dev);
|
|
|
|
recover_flag = card->state;
|
2011-05-13 02:45:02 +08:00
|
|
|
if ((!recovery_mode && card->info.hwtrap) || card->info.hwtrap == 2) {
|
|
|
|
qeth_hw_trap(card, QETH_DIAGS_TRAP_DISARM);
|
|
|
|
card->info.hwtrap = 1;
|
|
|
|
}
|
2008-10-24 17:16:52 +08:00
|
|
|
qeth_l2_stop_card(card, recovery_mode);
|
2008-02-15 16:19:42 +08:00
|
|
|
rc = ccw_device_set_offline(CARD_DDEV(card));
|
|
|
|
rc2 = ccw_device_set_offline(CARD_WDEV(card));
|
|
|
|
rc3 = ccw_device_set_offline(CARD_RDEV(card));
|
|
|
|
if (!rc)
|
|
|
|
rc = (rc2) ? rc2 : rc3;
|
|
|
|
if (rc)
|
2008-04-01 16:26:58 +08:00
|
|
|
QETH_DBF_TEXT_(SETUP, 2, "1err%d", rc);
|
2014-02-24 20:12:06 +08:00
|
|
|
qdio_free(CARD_DDEV(card));
|
2008-02-15 16:19:42 +08:00
|
|
|
if (recover_flag == CARD_STATE_UP)
|
|
|
|
card->state = CARD_STATE_RECOVER;
|
|
|
|
/* let user_space know that device is offline */
|
|
|
|
kobject_uevent(&cgdev->dev.kobj, KOBJ_CHANGE);
|
2010-05-12 03:34:47 +08:00
|
|
|
mutex_unlock(&card->conf_mutex);
|
2010-07-23 07:15:05 +08:00
|
|
|
mutex_unlock(&card->discipline_mutex);
|
2008-02-15 16:19:42 +08:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int qeth_l2_set_offline(struct ccwgroup_device *cgdev)
|
|
|
|
{
|
|
|
|
return __qeth_l2_set_offline(cgdev, 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
static int qeth_l2_recover(void *ptr)
|
|
|
|
{
|
|
|
|
struct qeth_card *card;
|
|
|
|
int rc = 0;
|
|
|
|
|
|
|
|
card = (struct qeth_card *) ptr;
|
2010-06-22 06:57:05 +08:00
|
|
|
QETH_CARD_TEXT(card, 2, "recover1");
|
2008-02-15 16:19:42 +08:00
|
|
|
if (!qeth_do_run_thread(card, QETH_RECOVER_THREAD))
|
|
|
|
return 0;
|
2010-06-22 06:57:05 +08:00
|
|
|
QETH_CARD_TEXT(card, 2, "recover2");
|
2008-12-25 20:39:49 +08:00
|
|
|
dev_warn(&card->gdev->dev,
|
|
|
|
"A recovery process has been started for the device\n");
|
2013-04-08 06:19:27 +08:00
|
|
|
qeth_set_recovery_task(card);
|
2008-02-15 16:19:42 +08:00
|
|
|
__qeth_l2_set_offline(card->gdev, 1);
|
|
|
|
rc = __qeth_l2_set_online(card->gdev, 1);
|
|
|
|
if (!rc)
|
2008-12-25 20:39:49 +08:00
|
|
|
dev_info(&card->gdev->dev,
|
|
|
|
"Device successfully recovered!\n");
|
2008-09-19 18:56:03 +08:00
|
|
|
else {
|
2013-01-21 10:30:20 +08:00
|
|
|
qeth_close_dev(card);
|
|
|
|
dev_warn(&card->gdev->dev, "The qeth device driver "
|
qeth: fix deadlock between recovery and bonding driver
The recovery thread, when failing, tears down the respective interface. To do
so, it needs to obtain the rtnl lock first, as the interface configuration is
changed.
If another process tries to modify an interface setting at the same time, that
process can obtain the rtnl lock first, but the respective callback in the qeth
driver will block until recovery has completed - which cannot happen since the
calling process already obtained it.
In one particular case, the bonding driver acquired the rtnl lock to modify the
card's MAC address, while the recovery failed at the same time due to the card
being removed. Hence qeth_l2_set_mac_address (implicitly holding the rtnl lock)
was waiting on qeth_l2_recover, which deadlocked when waiting on the rtnl lock.
This patch uses rtnl_trylock instead of rtnl_lock in the recovery thread. If the
lock cannot be obtained, the interface will be left up, but the card state
remains in CARD_STATE_RECOVER, which will prevent any further activities on the
card.
Signed-off-by: Stefan Raspl <raspl@linux.vnet.ibm.com>
Signed-off-by: Frank Blaschka <frank.blaschka@de.ibm.com>
Reviewed-by: Ursula Braun <ursula.braun@de.ibm.com>
Signed-off-by: David S. Miller <davem@davemloft.net>
2012-10-16 03:21:18 +08:00
|
|
|
"failed to recover an error on the device\n");
|
2008-09-19 18:56:03 +08:00
|
|
|
}
|
2013-04-08 06:19:27 +08:00
|
|
|
qeth_clear_recovery_task(card);
|
2010-09-08 05:14:42 +08:00
|
|
|
qeth_clear_thread_start_bit(card, QETH_RECOVER_THREAD);
|
|
|
|
qeth_clear_thread_running_bit(card, QETH_RECOVER_THREAD);
|
2008-02-15 16:19:42 +08:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int __init qeth_l2_init(void)
|
|
|
|
{
|
2008-12-25 20:39:49 +08:00
|
|
|
pr_info("register layer 2 discipline\n");
|
2008-02-15 16:19:42 +08:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void __exit qeth_l2_exit(void)
|
|
|
|
{
|
2008-12-25 20:39:49 +08:00
|
|
|
pr_info("unregister layer 2 discipline\n");
|
2008-02-15 16:19:42 +08:00
|
|
|
}
|
|
|
|
|
2009-06-16 16:30:31 +08:00
|
|
|
static int qeth_l2_pm_suspend(struct ccwgroup_device *gdev)
|
|
|
|
{
|
|
|
|
struct qeth_card *card = dev_get_drvdata(&gdev->dev);
|
|
|
|
|
|
|
|
if (card->dev)
|
|
|
|
netif_device_detach(card->dev);
|
|
|
|
qeth_set_allowed_threads(card, 0, 1);
|
|
|
|
wait_event(card->wait_q, qeth_threads_running(card, 0xffffffff) == 0);
|
|
|
|
if (gdev->state == CCWGROUP_OFFLINE)
|
|
|
|
return 0;
|
|
|
|
if (card->state == CARD_STATE_UP) {
|
2011-05-13 02:45:02 +08:00
|
|
|
if (card->info.hwtrap)
|
|
|
|
qeth_hw_trap(card, QETH_DIAGS_TRAP_DISARM);
|
2009-06-16 16:30:31 +08:00
|
|
|
__qeth_l2_set_offline(card->gdev, 1);
|
|
|
|
} else
|
|
|
|
__qeth_l2_set_offline(card->gdev, 0);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int qeth_l2_pm_resume(struct ccwgroup_device *gdev)
|
|
|
|
{
|
|
|
|
struct qeth_card *card = dev_get_drvdata(&gdev->dev);
|
|
|
|
int rc = 0;
|
|
|
|
|
|
|
|
if (gdev->state == CCWGROUP_OFFLINE)
|
|
|
|
goto out;
|
|
|
|
|
|
|
|
if (card->state == CARD_STATE_RECOVER) {
|
|
|
|
rc = __qeth_l2_set_online(card->gdev, 1);
|
|
|
|
if (rc) {
|
2010-03-09 04:36:56 +08:00
|
|
|
rtnl_lock();
|
|
|
|
dev_close(card->dev);
|
|
|
|
rtnl_unlock();
|
2009-06-16 16:30:31 +08:00
|
|
|
}
|
|
|
|
} else
|
|
|
|
rc = __qeth_l2_set_online(card->gdev, 0);
|
|
|
|
out:
|
|
|
|
qeth_set_allowed_threads(card, 0xffffffff, 0);
|
|
|
|
if (card->dev)
|
|
|
|
netif_device_attach(card->dev);
|
|
|
|
if (rc)
|
|
|
|
dev_warn(&card->gdev->dev, "The qeth device driver "
|
|
|
|
"failed to recover an error on the device\n");
|
|
|
|
return rc;
|
|
|
|
}
|
|
|
|
|
2014-01-29 16:23:48 +08:00
|
|
|
/* Returns zero if the command is successfully "consumed" */
|
|
|
|
static int qeth_l2_control_event(struct qeth_card *card,
|
|
|
|
struct qeth_ipa_cmd *cmd)
|
|
|
|
{
|
|
|
|
switch (cmd->hdr.command) {
|
2015-05-18 20:27:56 +08:00
|
|
|
case IPA_CMD_SETBRIDGEPORT_OSA:
|
|
|
|
case IPA_CMD_SETBRIDGEPORT_IQD:
|
2014-01-29 16:23:48 +08:00
|
|
|
if (cmd->data.sbp.hdr.command_code ==
|
|
|
|
IPA_SBP_BRIDGE_PORT_STATE_CHANGE) {
|
|
|
|
qeth_bridge_state_change(card, cmd);
|
|
|
|
return 0;
|
|
|
|
} else
|
|
|
|
return 1;
|
|
|
|
case IPA_CMD_ADDRESS_CHANGE_NOTIF:
|
|
|
|
qeth_bridge_host_event(card, cmd);
|
|
|
|
return 0;
|
|
|
|
default:
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-05-16 00:02:21 +08:00
|
|
|
struct qeth_discipline qeth_l2_discipline = {
|
2017-05-11 01:07:52 +08:00
|
|
|
.devtype = &qeth_l2_devtype,
|
2012-05-16 00:02:21 +08:00
|
|
|
.start_poll = qeth_qdio_start_poll,
|
|
|
|
.input_handler = (qdio_handler_t *) qeth_qdio_input_handler,
|
|
|
|
.output_handler = (qdio_handler_t *) qeth_qdio_output_handler,
|
2017-04-11 22:11:11 +08:00
|
|
|
.process_rx_buffer = qeth_l2_process_inbound_buffer,
|
2012-05-16 00:02:21 +08:00
|
|
|
.recover = qeth_l2_recover,
|
|
|
|
.setup = qeth_l2_probe_device,
|
2008-02-15 16:19:42 +08:00
|
|
|
.remove = qeth_l2_remove_device,
|
|
|
|
.set_online = qeth_l2_set_online,
|
|
|
|
.set_offline = qeth_l2_set_offline,
|
2009-06-16 16:30:31 +08:00
|
|
|
.freeze = qeth_l2_pm_suspend,
|
|
|
|
.thaw = qeth_l2_pm_resume,
|
|
|
|
.restore = qeth_l2_pm_resume,
|
2017-04-11 22:11:10 +08:00
|
|
|
.do_ioctl = NULL,
|
2014-01-29 16:23:48 +08:00
|
|
|
.control_event_handler = qeth_l2_control_event,
|
2008-02-15 16:19:42 +08:00
|
|
|
};
|
2012-05-16 00:02:21 +08:00
|
|
|
EXPORT_SYMBOL_GPL(qeth_l2_discipline);
|
2008-02-15 16:19:42 +08:00
|
|
|
|
|
|
|
static int qeth_osn_send_control_data(struct qeth_card *card, int len,
|
|
|
|
struct qeth_cmd_buffer *iob)
|
|
|
|
{
|
|
|
|
unsigned long flags;
|
|
|
|
int rc = 0;
|
|
|
|
|
2010-06-22 06:57:05 +08:00
|
|
|
QETH_CARD_TEXT(card, 5, "osndctrd");
|
2008-02-15 16:19:42 +08:00
|
|
|
|
|
|
|
wait_event(card->wait_q,
|
|
|
|
atomic_cmpxchg(&card->write.irq_pending, 0, 1) == 0);
|
|
|
|
qeth_prepare_control_data(card, len, iob);
|
2010-06-22 06:57:05 +08:00
|
|
|
QETH_CARD_TEXT(card, 6, "osnoirqp");
|
2008-02-15 16:19:42 +08:00
|
|
|
spin_lock_irqsave(get_ccwdev_lock(card->write.ccwdev), flags);
|
|
|
|
rc = ccw_device_start(card->write.ccwdev, &card->write.ccw,
|
|
|
|
(addr_t) iob, 0, 0);
|
|
|
|
spin_unlock_irqrestore(get_ccwdev_lock(card->write.ccwdev), flags);
|
|
|
|
if (rc) {
|
2008-06-06 18:37:46 +08:00
|
|
|
QETH_DBF_MESSAGE(2, "qeth_osn_send_control_data: "
|
2008-02-15 16:19:42 +08:00
|
|
|
"ccw_device_start rc = %i\n", rc);
|
2010-06-22 06:57:05 +08:00
|
|
|
QETH_CARD_TEXT_(card, 2, " err%d", rc);
|
2008-02-15 16:19:42 +08:00
|
|
|
qeth_release_buffer(iob->channel, iob);
|
|
|
|
atomic_set(&card->write.irq_pending, 0);
|
|
|
|
wake_up(&card->wait_q);
|
|
|
|
}
|
|
|
|
return rc;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int qeth_osn_send_ipa_cmd(struct qeth_card *card,
|
|
|
|
struct qeth_cmd_buffer *iob, int data_len)
|
|
|
|
{
|
|
|
|
u16 s1, s2;
|
|
|
|
|
2010-06-22 06:57:05 +08:00
|
|
|
QETH_CARD_TEXT(card, 4, "osndipa");
|
2008-02-15 16:19:42 +08:00
|
|
|
|
|
|
|
qeth_prepare_ipa_cmd(card, iob, QETH_PROT_OSN2);
|
|
|
|
s1 = (u16)(IPA_PDU_HEADER_SIZE + data_len);
|
|
|
|
s2 = (u16)data_len;
|
|
|
|
memcpy(QETH_IPA_PDU_LEN_TOTAL(iob->data), &s1, 2);
|
|
|
|
memcpy(QETH_IPA_PDU_LEN_PDU1(iob->data), &s2, 2);
|
|
|
|
memcpy(QETH_IPA_PDU_LEN_PDU2(iob->data), &s2, 2);
|
|
|
|
memcpy(QETH_IPA_PDU_LEN_PDU3(iob->data), &s2, 2);
|
|
|
|
return qeth_osn_send_control_data(card, s1, iob);
|
|
|
|
}
|
|
|
|
|
|
|
|
int qeth_osn_assist(struct net_device *dev, void *data, int data_len)
|
|
|
|
{
|
|
|
|
struct qeth_cmd_buffer *iob;
|
|
|
|
struct qeth_card *card;
|
|
|
|
int rc;
|
|
|
|
|
|
|
|
if (!dev)
|
|
|
|
return -ENODEV;
|
2008-07-26 17:24:10 +08:00
|
|
|
card = dev->ml_priv;
|
2008-02-15 16:19:42 +08:00
|
|
|
if (!card)
|
|
|
|
return -ENODEV;
|
2010-06-22 06:57:05 +08:00
|
|
|
QETH_CARD_TEXT(card, 2, "osnsdmc");
|
2015-01-16 21:05:47 +08:00
|
|
|
if (!qeth_card_hw_is_reachable(card))
|
2008-02-15 16:19:42 +08:00
|
|
|
return -ENODEV;
|
|
|
|
iob = qeth_wait_for_buffer(&card->write);
|
|
|
|
memcpy(iob->data+IPA_PDU_HEADER_SIZE, data, data_len);
|
|
|
|
rc = qeth_osn_send_ipa_cmd(card, iob, data_len);
|
|
|
|
return rc;
|
|
|
|
}
|
|
|
|
EXPORT_SYMBOL(qeth_osn_assist);
|
|
|
|
|
|
|
|
int qeth_osn_register(unsigned char *read_dev_no, struct net_device **dev,
|
|
|
|
int (*assist_cb)(struct net_device *, void *),
|
|
|
|
int (*data_cb)(struct sk_buff *))
|
|
|
|
{
|
|
|
|
struct qeth_card *card;
|
|
|
|
|
|
|
|
*dev = qeth_l2_netdev_by_devno(read_dev_no);
|
|
|
|
if (*dev == NULL)
|
|
|
|
return -ENODEV;
|
2008-07-26 17:24:10 +08:00
|
|
|
card = (*dev)->ml_priv;
|
2008-02-15 16:19:42 +08:00
|
|
|
if (!card)
|
|
|
|
return -ENODEV;
|
2010-06-22 06:57:05 +08:00
|
|
|
QETH_CARD_TEXT(card, 2, "osnreg");
|
2008-02-15 16:19:42 +08:00
|
|
|
if ((assist_cb == NULL) || (data_cb == NULL))
|
|
|
|
return -EINVAL;
|
|
|
|
card->osn_info.assist_cb = assist_cb;
|
|
|
|
card->osn_info.data_cb = data_cb;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
EXPORT_SYMBOL(qeth_osn_register);
|
|
|
|
|
|
|
|
void qeth_osn_deregister(struct net_device *dev)
|
|
|
|
{
|
|
|
|
struct qeth_card *card;
|
|
|
|
|
|
|
|
if (!dev)
|
|
|
|
return;
|
2008-07-26 17:24:10 +08:00
|
|
|
card = dev->ml_priv;
|
2008-02-15 16:19:42 +08:00
|
|
|
if (!card)
|
|
|
|
return;
|
2010-06-22 06:57:05 +08:00
|
|
|
QETH_CARD_TEXT(card, 2, "osndereg");
|
2008-02-15 16:19:42 +08:00
|
|
|
card->osn_info.assist_cb = NULL;
|
|
|
|
card->osn_info.data_cb = NULL;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
EXPORT_SYMBOL(qeth_osn_deregister);
|
|
|
|
|
2014-01-14 22:54:11 +08:00
|
|
|
/* SETBRIDGEPORT support, async notifications */
|
|
|
|
|
2014-01-14 22:54:13 +08:00
|
|
|
enum qeth_an_event_type {anev_reg_unreg, anev_abort, anev_reset};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* qeth_bridge_emit_host_event() - bridgeport address change notification
|
|
|
|
* @card: qeth_card structure pointer, for udev events.
|
|
|
|
* @evtype: "normal" register/unregister, or abort, or reset. For abort
|
|
|
|
* and reset token and addr_lnid are unused and may be NULL.
|
|
|
|
* @code: event bitmask: high order bit 0x80 value 1 means removal of an
|
|
|
|
* object, 0 - addition of an object.
|
|
|
|
* 0x01 - VLAN, 0x02 - MAC, 0x03 - VLAN and MAC.
|
|
|
|
* @token: "network token" structure identifying physical address of the port.
|
|
|
|
* @addr_lnid: pointer to structure with MAC address and VLAN ID.
|
|
|
|
*
|
|
|
|
* This function is called when registrations and deregistrations are
|
|
|
|
* reported by the hardware, and also when notifications are enabled -
|
|
|
|
* for all currently registered addresses.
|
|
|
|
*/
|
|
|
|
static void qeth_bridge_emit_host_event(struct qeth_card *card,
|
|
|
|
enum qeth_an_event_type evtype,
|
|
|
|
u8 code, struct net_if_token *token, struct mac_addr_lnid *addr_lnid)
|
|
|
|
{
|
|
|
|
char str[7][32];
|
|
|
|
char *env[8];
|
|
|
|
int i = 0;
|
|
|
|
|
|
|
|
switch (evtype) {
|
|
|
|
case anev_reg_unreg:
|
|
|
|
snprintf(str[i], sizeof(str[i]), "BRIDGEDHOST=%s",
|
|
|
|
(code & IPA_ADDR_CHANGE_CODE_REMOVAL)
|
|
|
|
? "deregister" : "register");
|
|
|
|
env[i] = str[i]; i++;
|
|
|
|
if (code & IPA_ADDR_CHANGE_CODE_VLANID) {
|
|
|
|
snprintf(str[i], sizeof(str[i]), "VLAN=%d",
|
|
|
|
addr_lnid->lnid);
|
|
|
|
env[i] = str[i]; i++;
|
|
|
|
}
|
|
|
|
if (code & IPA_ADDR_CHANGE_CODE_MACADDR) {
|
2015-09-18 22:06:48 +08:00
|
|
|
snprintf(str[i], sizeof(str[i]), "MAC=%pM",
|
2015-09-18 22:06:51 +08:00
|
|
|
addr_lnid->mac);
|
2014-01-14 22:54:13 +08:00
|
|
|
env[i] = str[i]; i++;
|
|
|
|
}
|
|
|
|
snprintf(str[i], sizeof(str[i]), "NTOK_BUSID=%x.%x.%04x",
|
|
|
|
token->cssid, token->ssid, token->devnum);
|
|
|
|
env[i] = str[i]; i++;
|
|
|
|
snprintf(str[i], sizeof(str[i]), "NTOK_IID=%02x", token->iid);
|
|
|
|
env[i] = str[i]; i++;
|
|
|
|
snprintf(str[i], sizeof(str[i]), "NTOK_CHPID=%02x",
|
|
|
|
token->chpid);
|
|
|
|
env[i] = str[i]; i++;
|
|
|
|
snprintf(str[i], sizeof(str[i]), "NTOK_CHID=%04x", token->chid);
|
|
|
|
env[i] = str[i]; i++;
|
|
|
|
break;
|
|
|
|
case anev_abort:
|
|
|
|
snprintf(str[i], sizeof(str[i]), "BRIDGEDHOST=abort");
|
|
|
|
env[i] = str[i]; i++;
|
|
|
|
break;
|
|
|
|
case anev_reset:
|
|
|
|
snprintf(str[i], sizeof(str[i]), "BRIDGEDHOST=reset");
|
|
|
|
env[i] = str[i]; i++;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
env[i] = NULL;
|
|
|
|
kobject_uevent_env(&card->gdev->dev.kobj, KOBJ_CHANGE, env);
|
|
|
|
}
|
|
|
|
|
2014-01-14 22:54:11 +08:00
|
|
|
struct qeth_bridge_state_data {
|
|
|
|
struct work_struct worker;
|
|
|
|
struct qeth_card *card;
|
|
|
|
struct qeth_sbp_state_change qports;
|
|
|
|
};
|
|
|
|
|
|
|
|
static void qeth_bridge_state_change_worker(struct work_struct *work)
|
|
|
|
{
|
|
|
|
struct qeth_bridge_state_data *data =
|
|
|
|
container_of(work, struct qeth_bridge_state_data, worker);
|
|
|
|
/* We are only interested in the first entry - local port */
|
|
|
|
struct qeth_sbp_port_entry *entry = &data->qports.entry[0];
|
|
|
|
char env_locrem[32];
|
|
|
|
char env_role[32];
|
|
|
|
char env_state[32];
|
|
|
|
char *env[] = {
|
|
|
|
env_locrem,
|
|
|
|
env_role,
|
|
|
|
env_state,
|
|
|
|
NULL
|
|
|
|
};
|
|
|
|
|
|
|
|
/* Role should not change by itself, but if it did, */
|
|
|
|
/* information from the hardware is authoritative. */
|
|
|
|
mutex_lock(&data->card->conf_mutex);
|
|
|
|
data->card->options.sbp.role = entry->role;
|
|
|
|
mutex_unlock(&data->card->conf_mutex);
|
|
|
|
|
|
|
|
snprintf(env_locrem, sizeof(env_locrem), "BRIDGEPORT=statechange");
|
|
|
|
snprintf(env_role, sizeof(env_role), "ROLE=%s",
|
|
|
|
(entry->role == QETH_SBP_ROLE_NONE) ? "none" :
|
|
|
|
(entry->role == QETH_SBP_ROLE_PRIMARY) ? "primary" :
|
|
|
|
(entry->role == QETH_SBP_ROLE_SECONDARY) ? "secondary" :
|
|
|
|
"<INVALID>");
|
|
|
|
snprintf(env_state, sizeof(env_state), "STATE=%s",
|
|
|
|
(entry->state == QETH_SBP_STATE_INACTIVE) ? "inactive" :
|
|
|
|
(entry->state == QETH_SBP_STATE_STANDBY) ? "standby" :
|
|
|
|
(entry->state == QETH_SBP_STATE_ACTIVE) ? "active" :
|
|
|
|
"<INVALID>");
|
|
|
|
kobject_uevent_env(&data->card->gdev->dev.kobj,
|
|
|
|
KOBJ_CHANGE, env);
|
|
|
|
kfree(data);
|
|
|
|
}
|
|
|
|
|
2014-01-29 16:23:48 +08:00
|
|
|
static void qeth_bridge_state_change(struct qeth_card *card,
|
|
|
|
struct qeth_ipa_cmd *cmd)
|
2014-01-14 22:54:11 +08:00
|
|
|
{
|
|
|
|
struct qeth_sbp_state_change *qports =
|
|
|
|
&cmd->data.sbp.data.state_change;
|
|
|
|
struct qeth_bridge_state_data *data;
|
|
|
|
int extrasize;
|
|
|
|
|
|
|
|
QETH_CARD_TEXT(card, 2, "brstchng");
|
|
|
|
if (qports->entry_length != sizeof(struct qeth_sbp_port_entry)) {
|
2014-10-22 18:18:03 +08:00
|
|
|
QETH_CARD_TEXT_(card, 2, "BPsz%04x", qports->entry_length);
|
2014-01-14 22:54:11 +08:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
extrasize = sizeof(struct qeth_sbp_port_entry) * qports->num_entries;
|
|
|
|
data = kzalloc(sizeof(struct qeth_bridge_state_data) + extrasize,
|
|
|
|
GFP_ATOMIC);
|
|
|
|
if (!data) {
|
|
|
|
QETH_CARD_TEXT(card, 2, "BPSalloc");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
INIT_WORK(&data->worker, qeth_bridge_state_change_worker);
|
|
|
|
data->card = card;
|
|
|
|
memcpy(&data->qports, qports,
|
|
|
|
sizeof(struct qeth_sbp_state_change) + extrasize);
|
|
|
|
queue_work(qeth_wq, &data->worker);
|
|
|
|
}
|
|
|
|
|
2014-01-14 22:54:13 +08:00
|
|
|
struct qeth_bridge_host_data {
|
|
|
|
struct work_struct worker;
|
|
|
|
struct qeth_card *card;
|
|
|
|
struct qeth_ipacmd_addr_change hostevs;
|
|
|
|
};
|
|
|
|
|
|
|
|
static void qeth_bridge_host_event_worker(struct work_struct *work)
|
|
|
|
{
|
|
|
|
struct qeth_bridge_host_data *data =
|
|
|
|
container_of(work, struct qeth_bridge_host_data, worker);
|
|
|
|
int i;
|
|
|
|
|
|
|
|
if (data->hostevs.lost_event_mask) {
|
|
|
|
dev_info(&data->card->gdev->dev,
|
2015-05-18 20:27:56 +08:00
|
|
|
"Address notification from the Bridge Port stopped %s (%s)\n",
|
2014-01-14 22:54:13 +08:00
|
|
|
data->card->dev->name,
|
|
|
|
(data->hostevs.lost_event_mask == 0x01)
|
|
|
|
? "Overflow"
|
|
|
|
: (data->hostevs.lost_event_mask == 0x02)
|
|
|
|
? "Bridge port state change"
|
|
|
|
: "Unknown reason");
|
|
|
|
mutex_lock(&data->card->conf_mutex);
|
|
|
|
data->card->options.sbp.hostnotification = 0;
|
|
|
|
mutex_unlock(&data->card->conf_mutex);
|
|
|
|
qeth_bridge_emit_host_event(data->card, anev_abort,
|
|
|
|
0, NULL, NULL);
|
|
|
|
} else
|
|
|
|
for (i = 0; i < data->hostevs.num_entries; i++) {
|
|
|
|
struct qeth_ipacmd_addr_change_entry *entry =
|
|
|
|
&data->hostevs.entry[i];
|
|
|
|
qeth_bridge_emit_host_event(data->card,
|
|
|
|
anev_reg_unreg,
|
|
|
|
entry->change_code,
|
|
|
|
&entry->token, &entry->addr_lnid);
|
|
|
|
}
|
|
|
|
kfree(data);
|
|
|
|
}
|
|
|
|
|
2014-01-29 16:23:48 +08:00
|
|
|
static void qeth_bridge_host_event(struct qeth_card *card,
|
|
|
|
struct qeth_ipa_cmd *cmd)
|
2014-01-14 22:54:13 +08:00
|
|
|
{
|
|
|
|
struct qeth_ipacmd_addr_change *hostevs =
|
|
|
|
&cmd->data.addrchange;
|
|
|
|
struct qeth_bridge_host_data *data;
|
|
|
|
int extrasize;
|
|
|
|
|
|
|
|
QETH_CARD_TEXT(card, 2, "brhostev");
|
|
|
|
if (cmd->hdr.return_code != 0x0000) {
|
|
|
|
if (cmd->hdr.return_code == 0x0010) {
|
|
|
|
if (hostevs->lost_event_mask == 0x00)
|
|
|
|
hostevs->lost_event_mask = 0xff;
|
|
|
|
} else {
|
|
|
|
QETH_CARD_TEXT_(card, 2, "BPHe%04x",
|
|
|
|
cmd->hdr.return_code);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
extrasize = sizeof(struct qeth_ipacmd_addr_change_entry) *
|
|
|
|
hostevs->num_entries;
|
|
|
|
data = kzalloc(sizeof(struct qeth_bridge_host_data) + extrasize,
|
|
|
|
GFP_ATOMIC);
|
|
|
|
if (!data) {
|
|
|
|
QETH_CARD_TEXT(card, 2, "BPHalloc");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
INIT_WORK(&data->worker, qeth_bridge_host_event_worker);
|
|
|
|
data->card = card;
|
|
|
|
memcpy(&data->hostevs, hostevs,
|
|
|
|
sizeof(struct qeth_ipacmd_addr_change) + extrasize);
|
|
|
|
queue_work(qeth_wq, &data->worker);
|
|
|
|
}
|
|
|
|
|
2014-01-14 22:54:11 +08:00
|
|
|
/* SETBRIDGEPORT support; sending commands */
|
|
|
|
|
|
|
|
struct _qeth_sbp_cbctl {
|
|
|
|
u16 ipa_rc;
|
|
|
|
u16 cmd_rc;
|
|
|
|
union {
|
|
|
|
u32 supported;
|
|
|
|
struct {
|
|
|
|
enum qeth_sbp_roles *role;
|
|
|
|
enum qeth_sbp_states *state;
|
|
|
|
} qports;
|
|
|
|
} data;
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* qeth_bridgeport_makerc() - derive "traditional" error from hardware codes.
|
|
|
|
* @card: qeth_card structure pointer, for debug messages.
|
|
|
|
* @cbctl: state structure with hardware return codes.
|
|
|
|
* @setcmd: IPA command code
|
|
|
|
*
|
|
|
|
* Returns negative errno-compatible error indication or 0 on success.
|
|
|
|
*/
|
|
|
|
static int qeth_bridgeport_makerc(struct qeth_card *card,
|
|
|
|
struct _qeth_sbp_cbctl *cbctl, enum qeth_ipa_sbp_cmd setcmd)
|
|
|
|
{
|
|
|
|
int rc;
|
2015-05-18 20:27:56 +08:00
|
|
|
int is_iqd = (card->info.type == QETH_CARD_TYPE_IQD);
|
2014-01-14 22:54:11 +08:00
|
|
|
|
2015-05-18 20:27:56 +08:00
|
|
|
if ((is_iqd && (cbctl->ipa_rc == IPA_RC_SUCCESS)) ||
|
|
|
|
(!is_iqd && (cbctl->ipa_rc == cbctl->cmd_rc)))
|
2014-01-14 22:54:11 +08:00
|
|
|
switch (cbctl->cmd_rc) {
|
|
|
|
case 0x0000:
|
|
|
|
rc = 0;
|
|
|
|
break;
|
2015-05-18 20:27:56 +08:00
|
|
|
case 0x2B04:
|
2014-01-14 22:54:11 +08:00
|
|
|
case 0x0004:
|
2015-05-18 20:27:58 +08:00
|
|
|
rc = -EOPNOTSUPP;
|
2014-01-14 22:54:11 +08:00
|
|
|
break;
|
2015-05-18 20:27:56 +08:00
|
|
|
case 0x2B0C:
|
2014-01-14 22:54:11 +08:00
|
|
|
case 0x000C: /* Not configured as bridge Port */
|
|
|
|
rc = -ENODEV; /* maybe not the best code here? */
|
|
|
|
dev_err(&card->gdev->dev,
|
2015-05-18 20:27:56 +08:00
|
|
|
"The device is not configured as a Bridge Port\n");
|
2014-01-14 22:54:11 +08:00
|
|
|
break;
|
2016-06-16 22:18:51 +08:00
|
|
|
case 0x2B10:
|
|
|
|
case 0x0010: /* OS mismatch */
|
|
|
|
rc = -EPERM;
|
|
|
|
dev_err(&card->gdev->dev,
|
|
|
|
"A Bridge Port is already configured by a different operating system\n");
|
|
|
|
break;
|
2015-05-18 20:27:56 +08:00
|
|
|
case 0x2B14:
|
2014-01-14 22:54:11 +08:00
|
|
|
case 0x0014: /* Another device is Primary */
|
|
|
|
switch (setcmd) {
|
|
|
|
case IPA_SBP_SET_PRIMARY_BRIDGE_PORT:
|
|
|
|
rc = -EEXIST;
|
|
|
|
dev_err(&card->gdev->dev,
|
2015-05-18 20:27:56 +08:00
|
|
|
"The LAN already has a primary Bridge Port\n");
|
2014-01-14 22:54:11 +08:00
|
|
|
break;
|
|
|
|
case IPA_SBP_SET_SECONDARY_BRIDGE_PORT:
|
|
|
|
rc = -EBUSY;
|
|
|
|
dev_err(&card->gdev->dev,
|
2015-05-18 20:27:56 +08:00
|
|
|
"The device is already a primary Bridge Port\n");
|
2014-01-14 22:54:11 +08:00
|
|
|
break;
|
|
|
|
default:
|
|
|
|
rc = -EIO;
|
|
|
|
}
|
|
|
|
break;
|
2015-05-18 20:27:56 +08:00
|
|
|
case 0x2B18:
|
2014-01-14 22:54:11 +08:00
|
|
|
case 0x0018: /* This device is currently Secondary */
|
|
|
|
rc = -EBUSY;
|
|
|
|
dev_err(&card->gdev->dev,
|
2015-05-18 20:27:56 +08:00
|
|
|
"The device is already a secondary Bridge Port\n");
|
2014-01-14 22:54:11 +08:00
|
|
|
break;
|
2015-05-18 20:27:56 +08:00
|
|
|
case 0x2B1C:
|
2014-01-14 22:54:11 +08:00
|
|
|
case 0x001C: /* Limit for Secondary devices reached */
|
|
|
|
rc = -EEXIST;
|
|
|
|
dev_err(&card->gdev->dev,
|
2015-05-18 20:27:56 +08:00
|
|
|
"The LAN cannot have more secondary Bridge Ports\n");
|
2014-01-14 22:54:11 +08:00
|
|
|
break;
|
2015-05-18 20:27:56 +08:00
|
|
|
case 0x2B24:
|
2014-01-14 22:54:11 +08:00
|
|
|
case 0x0024: /* This device is currently Primary */
|
|
|
|
rc = -EBUSY;
|
|
|
|
dev_err(&card->gdev->dev,
|
2015-05-18 20:27:56 +08:00
|
|
|
"The device is already a primary Bridge Port\n");
|
2014-01-14 22:54:11 +08:00
|
|
|
break;
|
2015-05-18 20:27:56 +08:00
|
|
|
case 0x2B20:
|
2014-01-14 22:54:11 +08:00
|
|
|
case 0x0020: /* Not authorized by zManager */
|
|
|
|
rc = -EACCES;
|
|
|
|
dev_err(&card->gdev->dev,
|
2015-05-18 20:27:56 +08:00
|
|
|
"The device is not authorized to be a Bridge Port\n");
|
2014-01-14 22:54:11 +08:00
|
|
|
break;
|
|
|
|
default:
|
|
|
|
rc = -EIO;
|
|
|
|
}
|
2015-05-18 20:27:56 +08:00
|
|
|
else
|
|
|
|
switch (cbctl->ipa_rc) {
|
|
|
|
case IPA_RC_NOTSUPP:
|
2015-05-18 20:27:58 +08:00
|
|
|
rc = -EOPNOTSUPP;
|
2015-05-18 20:27:56 +08:00
|
|
|
break;
|
|
|
|
case IPA_RC_UNSUPPORTED_COMMAND:
|
2015-05-18 20:27:58 +08:00
|
|
|
rc = -EOPNOTSUPP;
|
2015-05-18 20:27:56 +08:00
|
|
|
break;
|
|
|
|
default:
|
|
|
|
rc = -EIO;
|
|
|
|
}
|
|
|
|
|
2014-01-14 22:54:11 +08:00
|
|
|
if (rc) {
|
|
|
|
QETH_CARD_TEXT_(card, 2, "SBPi%04x", cbctl->ipa_rc);
|
|
|
|
QETH_CARD_TEXT_(card, 2, "SBPc%04x", cbctl->cmd_rc);
|
|
|
|
}
|
|
|
|
return rc;
|
|
|
|
}
|
|
|
|
|
2015-05-18 20:27:56 +08:00
|
|
|
static inline int ipa_cmd_sbp(struct qeth_card *card)
|
|
|
|
{
|
|
|
|
return (card->info.type == QETH_CARD_TYPE_IQD) ?
|
|
|
|
IPA_CMD_SETBRIDGEPORT_IQD :
|
|
|
|
IPA_CMD_SETBRIDGEPORT_OSA;
|
|
|
|
}
|
|
|
|
|
2014-01-14 22:54:11 +08:00
|
|
|
static int qeth_bridgeport_query_support_cb(struct qeth_card *card,
|
|
|
|
struct qeth_reply *reply, unsigned long data)
|
|
|
|
{
|
|
|
|
struct qeth_ipa_cmd *cmd = (struct qeth_ipa_cmd *) data;
|
|
|
|
struct _qeth_sbp_cbctl *cbctl = (struct _qeth_sbp_cbctl *)reply->param;
|
|
|
|
QETH_CARD_TEXT(card, 2, "brqsupcb");
|
|
|
|
cbctl->ipa_rc = cmd->hdr.return_code;
|
|
|
|
cbctl->cmd_rc = cmd->data.sbp.hdr.return_code;
|
|
|
|
if ((cbctl->ipa_rc == 0) && (cbctl->cmd_rc == 0)) {
|
|
|
|
cbctl->data.supported =
|
|
|
|
cmd->data.sbp.data.query_cmds_supp.supported_cmds;
|
|
|
|
} else {
|
|
|
|
cbctl->data.supported = 0;
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* qeth_bridgeport_query_support() - store bitmask of supported subfunctions.
|
|
|
|
* @card: qeth_card structure pointer.
|
|
|
|
*
|
|
|
|
* Sets bitmask of supported setbridgeport subfunctions in the qeth_card
|
|
|
|
* strucutre: card->options.sbp.supported_funcs.
|
|
|
|
*/
|
2014-01-29 16:23:48 +08:00
|
|
|
static void qeth_bridgeport_query_support(struct qeth_card *card)
|
2014-01-14 22:54:11 +08:00
|
|
|
{
|
|
|
|
struct qeth_cmd_buffer *iob;
|
|
|
|
struct qeth_ipa_cmd *cmd;
|
|
|
|
struct _qeth_sbp_cbctl cbctl;
|
|
|
|
|
|
|
|
QETH_CARD_TEXT(card, 2, "brqsuppo");
|
2015-05-18 20:27:56 +08:00
|
|
|
iob = qeth_get_ipacmd_buffer(card, ipa_cmd_sbp(card), 0);
|
2015-01-21 20:39:10 +08:00
|
|
|
if (!iob)
|
|
|
|
return;
|
2014-01-14 22:54:11 +08:00
|
|
|
cmd = (struct qeth_ipa_cmd *)(iob->data+IPA_PDU_HEADER_SIZE);
|
|
|
|
cmd->data.sbp.hdr.cmdlength =
|
|
|
|
sizeof(struct qeth_ipacmd_sbp_hdr) +
|
|
|
|
sizeof(struct qeth_sbp_query_cmds_supp);
|
|
|
|
cmd->data.sbp.hdr.command_code =
|
|
|
|
IPA_SBP_QUERY_COMMANDS_SUPPORTED;
|
|
|
|
cmd->data.sbp.hdr.used_total = 1;
|
|
|
|
cmd->data.sbp.hdr.seq_no = 1;
|
|
|
|
if (qeth_send_ipa_cmd(card, iob, qeth_bridgeport_query_support_cb,
|
|
|
|
(void *)&cbctl) ||
|
|
|
|
qeth_bridgeport_makerc(card, &cbctl,
|
|
|
|
IPA_SBP_QUERY_COMMANDS_SUPPORTED)) {
|
|
|
|
/* non-zero makerc signifies failure, and produce messages */
|
|
|
|
card->options.sbp.role = QETH_SBP_ROLE_NONE;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
card->options.sbp.supported_funcs = cbctl.data.supported;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int qeth_bridgeport_query_ports_cb(struct qeth_card *card,
|
|
|
|
struct qeth_reply *reply, unsigned long data)
|
|
|
|
{
|
|
|
|
struct qeth_ipa_cmd *cmd = (struct qeth_ipa_cmd *) data;
|
|
|
|
struct qeth_sbp_query_ports *qports = &cmd->data.sbp.data.query_ports;
|
|
|
|
struct _qeth_sbp_cbctl *cbctl = (struct _qeth_sbp_cbctl *)reply->param;
|
|
|
|
|
|
|
|
QETH_CARD_TEXT(card, 2, "brqprtcb");
|
|
|
|
cbctl->ipa_rc = cmd->hdr.return_code;
|
|
|
|
cbctl->cmd_rc = cmd->data.sbp.hdr.return_code;
|
|
|
|
if ((cbctl->ipa_rc != 0) || (cbctl->cmd_rc != 0))
|
|
|
|
return 0;
|
|
|
|
if (qports->entry_length != sizeof(struct qeth_sbp_port_entry)) {
|
|
|
|
cbctl->cmd_rc = 0xffff;
|
|
|
|
QETH_CARD_TEXT_(card, 2, "SBPs%04x", qports->entry_length);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
/* first entry contains the state of the local port */
|
|
|
|
if (qports->num_entries > 0) {
|
|
|
|
if (cbctl->data.qports.role)
|
|
|
|
*cbctl->data.qports.role = qports->entry[0].role;
|
|
|
|
if (cbctl->data.qports.state)
|
|
|
|
*cbctl->data.qports.state = qports->entry[0].state;
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* qeth_bridgeport_query_ports() - query local bridgeport status.
|
|
|
|
* @card: qeth_card structure pointer.
|
|
|
|
* @role: Role of the port: 0-none, 1-primary, 2-secondary.
|
|
|
|
* @state: State of the port: 0-inactive, 1-standby, 2-active.
|
|
|
|
*
|
|
|
|
* Returns negative errno-compatible error indication or 0 on success.
|
|
|
|
*
|
|
|
|
* 'role' and 'state' are not updated in case of hardware operation failure.
|
|
|
|
*/
|
|
|
|
int qeth_bridgeport_query_ports(struct qeth_card *card,
|
|
|
|
enum qeth_sbp_roles *role, enum qeth_sbp_states *state)
|
|
|
|
{
|
|
|
|
int rc = 0;
|
|
|
|
struct qeth_cmd_buffer *iob;
|
|
|
|
struct qeth_ipa_cmd *cmd;
|
|
|
|
struct _qeth_sbp_cbctl cbctl = {
|
|
|
|
.data = {
|
|
|
|
.qports = {
|
|
|
|
.role = role,
|
|
|
|
.state = state,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
};
|
|
|
|
|
|
|
|
QETH_CARD_TEXT(card, 2, "brqports");
|
|
|
|
if (!(card->options.sbp.supported_funcs & IPA_SBP_QUERY_BRIDGE_PORTS))
|
|
|
|
return -EOPNOTSUPP;
|
2015-05-18 20:27:56 +08:00
|
|
|
iob = qeth_get_ipacmd_buffer(card, ipa_cmd_sbp(card), 0);
|
2015-01-21 20:39:10 +08:00
|
|
|
if (!iob)
|
|
|
|
return -ENOMEM;
|
2014-01-14 22:54:11 +08:00
|
|
|
cmd = (struct qeth_ipa_cmd *)(iob->data+IPA_PDU_HEADER_SIZE);
|
|
|
|
cmd->data.sbp.hdr.cmdlength =
|
|
|
|
sizeof(struct qeth_ipacmd_sbp_hdr);
|
|
|
|
cmd->data.sbp.hdr.command_code =
|
|
|
|
IPA_SBP_QUERY_BRIDGE_PORTS;
|
|
|
|
cmd->data.sbp.hdr.used_total = 1;
|
|
|
|
cmd->data.sbp.hdr.seq_no = 1;
|
|
|
|
rc = qeth_send_ipa_cmd(card, iob, qeth_bridgeport_query_ports_cb,
|
|
|
|
(void *)&cbctl);
|
2015-05-18 20:27:53 +08:00
|
|
|
if (rc < 0)
|
2014-01-14 22:54:11 +08:00
|
|
|
return rc;
|
2015-05-18 20:27:53 +08:00
|
|
|
return qeth_bridgeport_makerc(card, &cbctl, IPA_SBP_QUERY_BRIDGE_PORTS);
|
2014-01-14 22:54:11 +08:00
|
|
|
}
|
|
|
|
EXPORT_SYMBOL_GPL(qeth_bridgeport_query_ports);
|
|
|
|
|
|
|
|
static int qeth_bridgeport_set_cb(struct qeth_card *card,
|
|
|
|
struct qeth_reply *reply, unsigned long data)
|
|
|
|
{
|
|
|
|
struct qeth_ipa_cmd *cmd = (struct qeth_ipa_cmd *)data;
|
|
|
|
struct _qeth_sbp_cbctl *cbctl = (struct _qeth_sbp_cbctl *)reply->param;
|
|
|
|
QETH_CARD_TEXT(card, 2, "brsetrcb");
|
|
|
|
cbctl->ipa_rc = cmd->hdr.return_code;
|
|
|
|
cbctl->cmd_rc = cmd->data.sbp.hdr.return_code;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* qeth_bridgeport_setrole() - Assign primary role to the port.
|
|
|
|
* @card: qeth_card structure pointer.
|
|
|
|
* @role: Role to assign.
|
|
|
|
*
|
|
|
|
* Returns negative errno-compatible error indication or 0 on success.
|
|
|
|
*/
|
|
|
|
int qeth_bridgeport_setrole(struct qeth_card *card, enum qeth_sbp_roles role)
|
|
|
|
{
|
|
|
|
int rc = 0;
|
|
|
|
int cmdlength;
|
|
|
|
struct qeth_cmd_buffer *iob;
|
|
|
|
struct qeth_ipa_cmd *cmd;
|
|
|
|
struct _qeth_sbp_cbctl cbctl;
|
|
|
|
enum qeth_ipa_sbp_cmd setcmd;
|
|
|
|
|
|
|
|
QETH_CARD_TEXT(card, 2, "brsetrol");
|
|
|
|
switch (role) {
|
|
|
|
case QETH_SBP_ROLE_NONE:
|
|
|
|
setcmd = IPA_SBP_RESET_BRIDGE_PORT_ROLE;
|
|
|
|
cmdlength = sizeof(struct qeth_ipacmd_sbp_hdr) +
|
|
|
|
sizeof(struct qeth_sbp_reset_role);
|
|
|
|
break;
|
|
|
|
case QETH_SBP_ROLE_PRIMARY:
|
|
|
|
setcmd = IPA_SBP_SET_PRIMARY_BRIDGE_PORT;
|
|
|
|
cmdlength = sizeof(struct qeth_ipacmd_sbp_hdr) +
|
|
|
|
sizeof(struct qeth_sbp_set_primary);
|
|
|
|
break;
|
|
|
|
case QETH_SBP_ROLE_SECONDARY:
|
|
|
|
setcmd = IPA_SBP_SET_SECONDARY_BRIDGE_PORT;
|
|
|
|
cmdlength = sizeof(struct qeth_ipacmd_sbp_hdr) +
|
|
|
|
sizeof(struct qeth_sbp_set_secondary);
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
if (!(card->options.sbp.supported_funcs & setcmd))
|
|
|
|
return -EOPNOTSUPP;
|
2015-05-18 20:27:56 +08:00
|
|
|
iob = qeth_get_ipacmd_buffer(card, ipa_cmd_sbp(card), 0);
|
2015-01-21 20:39:10 +08:00
|
|
|
if (!iob)
|
|
|
|
return -ENOMEM;
|
2014-01-14 22:54:11 +08:00
|
|
|
cmd = (struct qeth_ipa_cmd *)(iob->data+IPA_PDU_HEADER_SIZE);
|
|
|
|
cmd->data.sbp.hdr.cmdlength = cmdlength;
|
|
|
|
cmd->data.sbp.hdr.command_code = setcmd;
|
|
|
|
cmd->data.sbp.hdr.used_total = 1;
|
|
|
|
cmd->data.sbp.hdr.seq_no = 1;
|
|
|
|
rc = qeth_send_ipa_cmd(card, iob, qeth_bridgeport_set_cb,
|
|
|
|
(void *)&cbctl);
|
2015-05-18 20:27:53 +08:00
|
|
|
if (rc < 0)
|
2014-01-14 22:54:11 +08:00
|
|
|
return rc;
|
2015-05-18 20:27:53 +08:00
|
|
|
return qeth_bridgeport_makerc(card, &cbctl, setcmd);
|
2014-01-14 22:54:11 +08:00
|
|
|
}
|
|
|
|
|
2014-01-14 22:54:13 +08:00
|
|
|
/**
|
|
|
|
* qeth_anset_makerc() - derive "traditional" error from hardware codes.
|
|
|
|
* @card: qeth_card structure pointer, for debug messages.
|
|
|
|
*
|
|
|
|
* Returns negative errno-compatible error indication or 0 on success.
|
|
|
|
*/
|
|
|
|
static int qeth_anset_makerc(struct qeth_card *card, int pnso_rc, u16 response)
|
|
|
|
{
|
|
|
|
int rc;
|
|
|
|
|
|
|
|
if (pnso_rc == 0)
|
|
|
|
switch (response) {
|
|
|
|
case 0x0001:
|
|
|
|
rc = 0;
|
|
|
|
break;
|
|
|
|
case 0x0004:
|
|
|
|
case 0x0100:
|
|
|
|
case 0x0106:
|
2015-05-18 20:27:58 +08:00
|
|
|
rc = -EOPNOTSUPP;
|
2014-01-14 22:54:13 +08:00
|
|
|
dev_err(&card->gdev->dev,
|
|
|
|
"Setting address notification failed\n");
|
|
|
|
break;
|
|
|
|
case 0x0107:
|
|
|
|
rc = -EAGAIN;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
rc = -EIO;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
rc = -EIO;
|
|
|
|
|
|
|
|
if (rc) {
|
|
|
|
QETH_CARD_TEXT_(card, 2, "SBPp%04x", pnso_rc);
|
|
|
|
QETH_CARD_TEXT_(card, 2, "SBPr%04x", response);
|
|
|
|
}
|
|
|
|
return rc;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void qeth_bridgeport_an_set_cb(void *priv,
|
|
|
|
enum qdio_brinfo_entry_type type, void *entry)
|
|
|
|
{
|
|
|
|
struct qeth_card *card = (struct qeth_card *)priv;
|
|
|
|
struct qdio_brinfo_entry_l2 *l2entry;
|
|
|
|
u8 code;
|
|
|
|
|
|
|
|
if (type != l2_addr_lnid) {
|
|
|
|
WARN_ON_ONCE(1);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
l2entry = (struct qdio_brinfo_entry_l2 *)entry;
|
|
|
|
code = IPA_ADDR_CHANGE_CODE_MACADDR;
|
|
|
|
if (l2entry->addr_lnid.lnid)
|
|
|
|
code |= IPA_ADDR_CHANGE_CODE_VLANID;
|
|
|
|
qeth_bridge_emit_host_event(card, anev_reg_unreg, code,
|
|
|
|
(struct net_if_token *)&l2entry->nit,
|
|
|
|
(struct mac_addr_lnid *)&l2entry->addr_lnid);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* qeth_bridgeport_an_set() - Enable or disable bridgeport address notification
|
|
|
|
* @card: qeth_card structure pointer.
|
|
|
|
* @enable: 0 - disable, non-zero - enable notifications
|
|
|
|
*
|
|
|
|
* Returns negative errno-compatible error indication or 0 on success.
|
|
|
|
*
|
|
|
|
* On enable, emits a series of address notifications udev events for all
|
|
|
|
* currently registered hosts.
|
|
|
|
*/
|
|
|
|
int qeth_bridgeport_an_set(struct qeth_card *card, int enable)
|
|
|
|
{
|
|
|
|
int rc;
|
|
|
|
u16 response;
|
|
|
|
struct ccw_device *ddev;
|
|
|
|
struct subchannel_id schid;
|
|
|
|
|
|
|
|
if (!card)
|
|
|
|
return -EINVAL;
|
|
|
|
if (!card->options.sbp.supported_funcs)
|
|
|
|
return -EOPNOTSUPP;
|
|
|
|
ddev = CARD_DDEV(card);
|
|
|
|
ccw_device_get_schid(ddev, &schid);
|
|
|
|
|
|
|
|
if (enable) {
|
|
|
|
qeth_bridge_emit_host_event(card, anev_reset, 0, NULL, NULL);
|
|
|
|
rc = qdio_pnso_brinfo(schid, 1, &response,
|
|
|
|
qeth_bridgeport_an_set_cb, card);
|
|
|
|
} else
|
|
|
|
rc = qdio_pnso_brinfo(schid, 0, &response, NULL, NULL);
|
|
|
|
return qeth_anset_makerc(card, rc, response);
|
|
|
|
}
|
|
|
|
EXPORT_SYMBOL_GPL(qeth_bridgeport_an_set);
|
|
|
|
|
2008-02-15 16:19:42 +08:00
|
|
|
module_init(qeth_l2_init);
|
|
|
|
module_exit(qeth_l2_exit);
|
|
|
|
MODULE_AUTHOR("Frank Blaschka <frank.blaschka@de.ibm.com>");
|
|
|
|
MODULE_DESCRIPTION("qeth layer 2 discipline");
|
|
|
|
MODULE_LICENSE("GPL");
|