| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | /*
 | 
					
						
							|  |  |  |  *  drivers/s390/net/qeth_l3_main.c | 
					
						
							|  |  |  |  * | 
					
						
							| 
									
										
										
										
											2009-06-16 10:30:31 +02:00
										 |  |  |  *    Copyright IBM Corp. 2007, 2009 | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01: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 13:39:49 +01:00
										 |  |  | #define KMSG_COMPONENT "qeth"
 | 
					
						
							|  |  |  | #define pr_fmt(fmt) KMSG_COMPONENT ": " fmt
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | #include <linux/module.h>
 | 
					
						
							|  |  |  | #include <linux/moduleparam.h>
 | 
					
						
							|  |  |  | #include <linux/string.h>
 | 
					
						
							|  |  |  | #include <linux/errno.h>
 | 
					
						
							|  |  |  | #include <linux/kernel.h>
 | 
					
						
							|  |  |  | #include <linux/etherdevice.h>
 | 
					
						
							|  |  |  | #include <linux/mii.h>
 | 
					
						
							|  |  |  | #include <linux/ip.h>
 | 
					
						
							| 
									
										
										
										
											2009-03-24 20:57:16 +00:00
										 |  |  | #include <linux/ipv6.h>
 | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | #include <linux/inetdevice.h>
 | 
					
						
							|  |  |  | #include <linux/igmp.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 17:04:11 +09:00
										 |  |  | #include <linux/slab.h>
 | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | #include <net/ip.h>
 | 
					
						
							|  |  |  | #include <net/arp.h>
 | 
					
						
							| 
									
										
										
										
											2009-03-24 20:57:16 +00:00
										 |  |  | #include <net/ip6_checksum.h>
 | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | #include "qeth_l3.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int qeth_l3_set_offline(struct ccwgroup_device *); | 
					
						
							|  |  |  | static int qeth_l3_recover(void *); | 
					
						
							|  |  |  | static int qeth_l3_stop(struct net_device *); | 
					
						
							|  |  |  | static void qeth_l3_set_multicast_list(struct net_device *); | 
					
						
							|  |  |  | static int qeth_l3_neigh_setup(struct net_device *, struct neigh_parms *); | 
					
						
							|  |  |  | static int qeth_l3_register_addr_entry(struct qeth_card *, | 
					
						
							|  |  |  | 		struct qeth_ipaddr *); | 
					
						
							|  |  |  | static int qeth_l3_deregister_addr_entry(struct qeth_card *, | 
					
						
							|  |  |  | 		struct qeth_ipaddr *); | 
					
						
							|  |  |  | static int __qeth_l3_set_online(struct ccwgroup_device *, int); | 
					
						
							|  |  |  | static int __qeth_l3_set_offline(struct ccwgroup_device *, int); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-11-12 00:11:44 +00:00
										 |  |  | int qeth_l3_set_large_send(struct qeth_card *card, | 
					
						
							|  |  |  | 		enum qeth_large_send_types type) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	int rc = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	card->options.large_send = type; | 
					
						
							|  |  |  | 	if (card->dev == NULL) | 
					
						
							|  |  |  | 		return 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (card->options.large_send == QETH_LARGE_SEND_TSO) { | 
					
						
							|  |  |  | 		if (qeth_is_supported(card, IPA_OUTBOUND_TSO)) { | 
					
						
							|  |  |  | 			card->dev->features |= NETIF_F_TSO | NETIF_F_SG | | 
					
						
							| 
									
										
										
										
											2010-05-11 19:34:45 +00:00
										 |  |  | 					NETIF_F_IP_CSUM; | 
					
						
							| 
									
										
										
										
											2009-11-12 00:11:44 +00:00
										 |  |  | 		} else { | 
					
						
							|  |  |  | 			card->dev->features &= ~(NETIF_F_TSO | NETIF_F_SG | | 
					
						
							| 
									
										
										
										
											2010-05-11 19:34:45 +00:00
										 |  |  | 					NETIF_F_IP_CSUM); | 
					
						
							| 
									
										
										
										
											2009-11-12 00:11:44 +00:00
										 |  |  | 			card->options.large_send = QETH_LARGE_SEND_NO; | 
					
						
							|  |  |  | 			rc = -EOPNOTSUPP; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} else { | 
					
						
							|  |  |  | 		card->dev->features &= ~(NETIF_F_TSO | NETIF_F_SG | | 
					
						
							| 
									
										
										
										
											2010-05-11 19:34:45 +00:00
										 |  |  | 					NETIF_F_IP_CSUM); | 
					
						
							| 
									
										
										
										
											2009-11-12 00:11:44 +00:00
										 |  |  | 		card->options.large_send = QETH_LARGE_SEND_NO; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return rc; | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | static int qeth_l3_isxdigit(char *buf) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	while (*buf) { | 
					
						
							|  |  |  | 		if (!isxdigit(*buf++)) | 
					
						
							|  |  |  | 			return 0; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return 1; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void qeth_l3_ipaddr4_to_string(const __u8 *addr, char *buf) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	sprintf(buf, "%i.%i.%i.%i", addr[0], addr[1], addr[2], addr[3]); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | int qeth_l3_string_to_ipaddr4(const char *buf, __u8 *addr) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	int count = 0, rc = 0; | 
					
						
							|  |  |  | 	int in[4]; | 
					
						
							|  |  |  | 	char c; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	rc = sscanf(buf, "%u.%u.%u.%u%c", | 
					
						
							|  |  |  | 		    &in[0], &in[1], &in[2], &in[3], &c); | 
					
						
							|  |  |  | 	if (rc != 4 && (rc != 5 || c != '\n')) | 
					
						
							|  |  |  | 		return -EINVAL; | 
					
						
							|  |  |  | 	for (count = 0; count < 4; count++) { | 
					
						
							|  |  |  | 		if (in[count] > 255) | 
					
						
							|  |  |  | 			return -EINVAL; | 
					
						
							|  |  |  | 		addr[count] = in[count]; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void qeth_l3_ipaddr6_to_string(const __u8 *addr, char *buf) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	sprintf(buf, "%02x%02x:%02x%02x:%02x%02x:%02x%02x" | 
					
						
							|  |  |  | 		     ":%02x%02x:%02x%02x:%02x%02x:%02x%02x", | 
					
						
							|  |  |  | 		     addr[0], addr[1], addr[2], addr[3], | 
					
						
							|  |  |  | 		     addr[4], addr[5], addr[6], addr[7], | 
					
						
							|  |  |  | 		     addr[8], addr[9], addr[10], addr[11], | 
					
						
							|  |  |  | 		     addr[12], addr[13], addr[14], addr[15]); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | int qeth_l3_string_to_ipaddr6(const char *buf, __u8 *addr) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	const char *end, *end_tmp, *start; | 
					
						
							|  |  |  | 	__u16 *in; | 
					
						
							|  |  |  | 	char num[5]; | 
					
						
							|  |  |  | 	int num2, cnt, out, found, save_cnt; | 
					
						
							|  |  |  | 	unsigned short in_tmp[8] = {0, }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	cnt = out = found = save_cnt = num2 = 0; | 
					
						
							|  |  |  | 	end = start = buf; | 
					
						
							|  |  |  | 	in = (__u16 *) addr; | 
					
						
							|  |  |  | 	memset(in, 0, 16); | 
					
						
							|  |  |  | 	while (*end) { | 
					
						
							|  |  |  | 		end = strchr(start, ':'); | 
					
						
							|  |  |  | 		if (end == NULL) { | 
					
						
							|  |  |  | 			end = buf + strlen(buf); | 
					
						
							|  |  |  | 			end_tmp = strchr(start, '\n'); | 
					
						
							|  |  |  | 			if (end_tmp != NULL) | 
					
						
							|  |  |  | 				end = end_tmp; | 
					
						
							|  |  |  | 			out = 1; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		if ((end - start)) { | 
					
						
							|  |  |  | 			memset(num, 0, 5); | 
					
						
							|  |  |  | 			if ((end - start) > 4) | 
					
						
							|  |  |  | 				return -EINVAL; | 
					
						
							|  |  |  | 			memcpy(num, start, end - start); | 
					
						
							|  |  |  | 			if (!qeth_l3_isxdigit(num)) | 
					
						
							|  |  |  | 				return -EINVAL; | 
					
						
							|  |  |  | 			sscanf(start, "%x", &num2); | 
					
						
							|  |  |  | 			if (found) | 
					
						
							|  |  |  | 				in_tmp[save_cnt++] = num2; | 
					
						
							|  |  |  | 			else | 
					
						
							|  |  |  | 				in[cnt++] = num2; | 
					
						
							|  |  |  | 			if (out) | 
					
						
							|  |  |  | 				break; | 
					
						
							|  |  |  | 		} else { | 
					
						
							|  |  |  | 			if (found) | 
					
						
							|  |  |  | 				return -EINVAL; | 
					
						
							|  |  |  | 			found = 1; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		start = ++end; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	if (cnt + save_cnt > 8) | 
					
						
							|  |  |  | 		return -EINVAL; | 
					
						
							|  |  |  | 	cnt = 7; | 
					
						
							|  |  |  | 	while (save_cnt) | 
					
						
							|  |  |  | 		in[cnt--] = in_tmp[--save_cnt]; | 
					
						
							|  |  |  | 	return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void qeth_l3_ipaddr_to_string(enum qeth_prot_versions proto, const __u8 *addr, | 
					
						
							|  |  |  | 				char *buf) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	if (proto == QETH_PROT_IPV4) | 
					
						
							|  |  |  | 		qeth_l3_ipaddr4_to_string(addr, buf); | 
					
						
							|  |  |  | 	else if (proto == QETH_PROT_IPV6) | 
					
						
							|  |  |  | 		qeth_l3_ipaddr6_to_string(addr, buf); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | int qeth_l3_string_to_ipaddr(const char *buf, enum qeth_prot_versions proto, | 
					
						
							|  |  |  | 				__u8 *addr) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	if (proto == QETH_PROT_IPV4) | 
					
						
							|  |  |  | 		return qeth_l3_string_to_ipaddr4(buf, addr); | 
					
						
							|  |  |  | 	else if (proto == QETH_PROT_IPV6) | 
					
						
							|  |  |  | 		return qeth_l3_string_to_ipaddr6(buf, addr); | 
					
						
							|  |  |  | 	else | 
					
						
							|  |  |  | 		return -EINVAL; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void qeth_l3_convert_addr_to_bits(u8 *addr, u8 *bits, int len) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	int i, j; | 
					
						
							|  |  |  | 	u8 octet; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	for (i = 0; i < len; ++i) { | 
					
						
							|  |  |  | 		octet = addr[i]; | 
					
						
							|  |  |  | 		for (j = 7; j >= 0; --j) { | 
					
						
							|  |  |  | 			bits[i*8 + j] = octet & 1; | 
					
						
							|  |  |  | 			octet >>= 1; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-07-22 23:15:03 +00:00
										 |  |  | int qeth_l3_is_addr_covered_by_ipato(struct qeth_card *card, | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 						struct qeth_ipaddr *addr) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct qeth_ipato_entry *ipatoe; | 
					
						
							|  |  |  | 	u8 addr_bits[128] = {0, }; | 
					
						
							|  |  |  | 	u8 ipatoe_bits[128] = {0, }; | 
					
						
							|  |  |  | 	int rc = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (!card->ipato.enabled) | 
					
						
							|  |  |  | 		return 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	qeth_l3_convert_addr_to_bits((u8 *) &addr->u, addr_bits, | 
					
						
							|  |  |  | 				  (addr->proto == QETH_PROT_IPV4)? 4:16); | 
					
						
							|  |  |  | 	list_for_each_entry(ipatoe, &card->ipato.entries, entry) { | 
					
						
							|  |  |  | 		if (addr->proto != ipatoe->proto) | 
					
						
							|  |  |  | 			continue; | 
					
						
							|  |  |  | 		qeth_l3_convert_addr_to_bits(ipatoe->addr, ipatoe_bits, | 
					
						
							|  |  |  | 					  (ipatoe->proto == QETH_PROT_IPV4) ? | 
					
						
							|  |  |  | 					  4 : 16); | 
					
						
							|  |  |  | 		if (addr->proto == QETH_PROT_IPV4) | 
					
						
							|  |  |  | 			rc = !memcmp(addr_bits, ipatoe_bits, | 
					
						
							|  |  |  | 				     min(32, ipatoe->mask_bits)); | 
					
						
							|  |  |  | 		else | 
					
						
							|  |  |  | 			rc = !memcmp(addr_bits, ipatoe_bits, | 
					
						
							|  |  |  | 				     min(128, ipatoe->mask_bits)); | 
					
						
							|  |  |  | 		if (rc) | 
					
						
							|  |  |  | 			break; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	/* invert? */ | 
					
						
							|  |  |  | 	if ((addr->proto == QETH_PROT_IPV4) && card->ipato.invert4) | 
					
						
							|  |  |  | 		rc = !rc; | 
					
						
							|  |  |  | 	else if ((addr->proto == QETH_PROT_IPV6) && card->ipato.invert6) | 
					
						
							|  |  |  | 		rc = !rc; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return rc; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /*
 | 
					
						
							|  |  |  |  * Add IP to be added to todo list. If there is already an "add todo" | 
					
						
							|  |  |  |  * in this list we just incremenent the reference count. | 
					
						
							|  |  |  |  * Returns 0 if we  just incremented reference count. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | static int __qeth_l3_insert_ip_todo(struct qeth_card *card, | 
					
						
							|  |  |  | 					struct qeth_ipaddr *addr, int add) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct qeth_ipaddr *tmp, *t; | 
					
						
							|  |  |  | 	int found = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-01-11 02:50:50 +00:00
										 |  |  | 	if (card->options.sniffer) | 
					
						
							|  |  |  | 		return 0; | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 	list_for_each_entry_safe(tmp, t, card->ip_tbd_list, entry) { | 
					
						
							|  |  |  | 		if ((addr->type == QETH_IP_TYPE_DEL_ALL_MC) && | 
					
						
							|  |  |  | 		    (tmp->type == QETH_IP_TYPE_DEL_ALL_MC)) | 
					
						
							|  |  |  | 			return 0; | 
					
						
							|  |  |  | 		if ((tmp->proto        == QETH_PROT_IPV4)     && | 
					
						
							|  |  |  | 		    (addr->proto       == QETH_PROT_IPV4)     && | 
					
						
							|  |  |  | 		    (tmp->type         == addr->type)         && | 
					
						
							|  |  |  | 		    (tmp->is_multicast == addr->is_multicast) && | 
					
						
							|  |  |  | 		    (tmp->u.a4.addr    == addr->u.a4.addr)    && | 
					
						
							|  |  |  | 		    (tmp->u.a4.mask    == addr->u.a4.mask)) { | 
					
						
							|  |  |  | 			found = 1; | 
					
						
							|  |  |  | 			break; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		if ((tmp->proto        == QETH_PROT_IPV6)      && | 
					
						
							|  |  |  | 		    (addr->proto       == QETH_PROT_IPV6)      && | 
					
						
							|  |  |  | 		    (tmp->type         == addr->type)          && | 
					
						
							|  |  |  | 		    (tmp->is_multicast == addr->is_multicast)  && | 
					
						
							|  |  |  | 		    (tmp->u.a6.pfxlen  == addr->u.a6.pfxlen)   && | 
					
						
							|  |  |  | 		    (memcmp(&tmp->u.a6.addr, &addr->u.a6.addr, | 
					
						
							|  |  |  | 			    sizeof(struct in6_addr)) == 0)) { | 
					
						
							|  |  |  | 			found = 1; | 
					
						
							|  |  |  | 			break; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	if (found) { | 
					
						
							|  |  |  | 		if (addr->users != 0) | 
					
						
							|  |  |  | 			tmp->users += addr->users; | 
					
						
							|  |  |  | 		else | 
					
						
							|  |  |  | 			tmp->users += add ? 1 : -1; | 
					
						
							|  |  |  | 		if (tmp->users == 0) { | 
					
						
							|  |  |  | 			list_del(&tmp->entry); | 
					
						
							|  |  |  | 			kfree(tmp); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		return 0; | 
					
						
							|  |  |  | 	} else { | 
					
						
							|  |  |  | 		if (addr->type == QETH_IP_TYPE_DEL_ALL_MC) | 
					
						
							|  |  |  | 			list_add(&addr->entry, card->ip_tbd_list); | 
					
						
							|  |  |  | 		else { | 
					
						
							|  |  |  | 			if (addr->users == 0) | 
					
						
							|  |  |  | 				addr->users += add ? 1 : -1; | 
					
						
							|  |  |  | 			if (add && (addr->type == QETH_IP_TYPE_NORMAL) && | 
					
						
							|  |  |  | 			    qeth_l3_is_addr_covered_by_ipato(card, addr)) { | 
					
						
							| 
									
										
										
										
											2010-06-21 22:57:05 +00:00
										 |  |  | 				QETH_CARD_TEXT(card, 2, "tkovaddr"); | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 				addr->set_flags |= QETH_IPA_SETIP_TAKEOVER_FLAG; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			list_add_tail(&addr->entry, card->ip_tbd_list); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		return 1; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int qeth_l3_delete_ip(struct qeth_card *card, struct qeth_ipaddr *addr) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	unsigned long flags; | 
					
						
							|  |  |  | 	int rc = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-06-21 22:57:05 +00:00
										 |  |  | 	QETH_CARD_TEXT(card, 4, "delip"); | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	if (addr->proto == QETH_PROT_IPV4) | 
					
						
							| 
									
										
										
										
											2010-06-21 22:57:05 +00:00
										 |  |  | 		QETH_CARD_HEX(card, 4, &addr->u.a4.addr, 4); | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 	else { | 
					
						
							| 
									
										
										
										
											2010-06-21 22:57:05 +00:00
										 |  |  | 		QETH_CARD_HEX(card, 4, &addr->u.a6.addr, 8); | 
					
						
							|  |  |  | 		QETH_CARD_HEX(card, 4, ((char *)&addr->u.a6.addr) + 8, 8); | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 	} | 
					
						
							|  |  |  | 	spin_lock_irqsave(&card->ip_lock, flags); | 
					
						
							|  |  |  | 	rc = __qeth_l3_insert_ip_todo(card, addr, 0); | 
					
						
							|  |  |  | 	spin_unlock_irqrestore(&card->ip_lock, flags); | 
					
						
							|  |  |  | 	return rc; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int qeth_l3_add_ip(struct qeth_card *card, struct qeth_ipaddr *addr) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	unsigned long flags; | 
					
						
							|  |  |  | 	int rc = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-06-21 22:57:05 +00:00
										 |  |  | 	QETH_CARD_TEXT(card, 4, "addip"); | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 	if (addr->proto == QETH_PROT_IPV4) | 
					
						
							| 
									
										
										
										
											2010-06-21 22:57:05 +00:00
										 |  |  | 		QETH_CARD_HEX(card, 4, &addr->u.a4.addr, 4); | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 	else { | 
					
						
							| 
									
										
										
										
											2010-06-21 22:57:05 +00:00
										 |  |  | 		QETH_CARD_HEX(card, 4, &addr->u.a6.addr, 8); | 
					
						
							|  |  |  | 		QETH_CARD_HEX(card, 4, ((char *)&addr->u.a6.addr) + 8, 8); | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 	} | 
					
						
							|  |  |  | 	spin_lock_irqsave(&card->ip_lock, flags); | 
					
						
							|  |  |  | 	rc = __qeth_l3_insert_ip_todo(card, addr, 1); | 
					
						
							|  |  |  | 	spin_unlock_irqrestore(&card->ip_lock, flags); | 
					
						
							|  |  |  | 	return rc; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static struct qeth_ipaddr *qeth_l3_get_addr_buffer( | 
					
						
							|  |  |  | 				enum qeth_prot_versions prot) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct qeth_ipaddr *addr; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	addr = kzalloc(sizeof(struct qeth_ipaddr), GFP_ATOMIC); | 
					
						
							|  |  |  | 	if (addr == NULL) { | 
					
						
							|  |  |  | 		return NULL; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	addr->type = QETH_IP_TYPE_NORMAL; | 
					
						
							|  |  |  | 	addr->proto = prot; | 
					
						
							|  |  |  | 	return addr; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void qeth_l3_delete_mc_addresses(struct qeth_card *card) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct qeth_ipaddr *iptodo; | 
					
						
							|  |  |  | 	unsigned long flags; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-06-21 22:57:05 +00:00
										 |  |  | 	QETH_CARD_TEXT(card, 4, "delmc"); | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 	iptodo = qeth_l3_get_addr_buffer(QETH_PROT_IPV4); | 
					
						
							|  |  |  | 	if (!iptodo) { | 
					
						
							| 
									
										
										
										
											2010-06-21 22:57:05 +00:00
										 |  |  | 		QETH_CARD_TEXT(card, 2, "dmcnomem"); | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 		return; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	iptodo->type = QETH_IP_TYPE_DEL_ALL_MC; | 
					
						
							|  |  |  | 	spin_lock_irqsave(&card->ip_lock, flags); | 
					
						
							|  |  |  | 	if (!__qeth_l3_insert_ip_todo(card, iptodo, 0)) | 
					
						
							|  |  |  | 		kfree(iptodo); | 
					
						
							|  |  |  | 	spin_unlock_irqrestore(&card->ip_lock, flags); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /*
 | 
					
						
							|  |  |  |  * Add/remove address to/from card's ip list, i.e. try to add or remove | 
					
						
							|  |  |  |  * reference to/from an IP address that is already registered on the card. | 
					
						
							|  |  |  |  * Returns: | 
					
						
							|  |  |  |  *	0  address was on card and its reference count has been adjusted, | 
					
						
							|  |  |  |  *	   but is still > 0, so nothing has to be done | 
					
						
							|  |  |  |  *	   also returns 0 if card was not on card and the todo was to delete | 
					
						
							|  |  |  |  *	   the address -> there is also nothing to be done | 
					
						
							|  |  |  |  *	1  address was not on card and the todo is to add it to the card's ip | 
					
						
							|  |  |  |  *	   list | 
					
						
							|  |  |  |  *	-1 address was on card and its reference count has been decremented | 
					
						
							|  |  |  |  *	   to <= 0 by the todo -> address must be removed from card | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | static int __qeth_l3_ref_ip_on_card(struct qeth_card *card, | 
					
						
							|  |  |  | 		struct qeth_ipaddr *todo, struct qeth_ipaddr **__addr) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct qeth_ipaddr *addr; | 
					
						
							|  |  |  | 	int found = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	list_for_each_entry(addr, &card->ip_list, entry) { | 
					
						
							|  |  |  | 		if ((addr->proto == QETH_PROT_IPV4) && | 
					
						
							|  |  |  | 		    (todo->proto == QETH_PROT_IPV4) && | 
					
						
							|  |  |  | 		    (addr->type == todo->type) && | 
					
						
							|  |  |  | 		    (addr->u.a4.addr == todo->u.a4.addr) && | 
					
						
							|  |  |  | 		    (addr->u.a4.mask == todo->u.a4.mask)) { | 
					
						
							|  |  |  | 			found = 1; | 
					
						
							|  |  |  | 			break; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		if ((addr->proto == QETH_PROT_IPV6) && | 
					
						
							|  |  |  | 		    (todo->proto == QETH_PROT_IPV6) && | 
					
						
							|  |  |  | 		    (addr->type == todo->type) && | 
					
						
							|  |  |  | 		    (addr->u.a6.pfxlen == todo->u.a6.pfxlen) && | 
					
						
							|  |  |  | 		    (memcmp(&addr->u.a6.addr, &todo->u.a6.addr, | 
					
						
							|  |  |  | 			    sizeof(struct in6_addr)) == 0)) { | 
					
						
							|  |  |  | 			found = 1; | 
					
						
							|  |  |  | 			break; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	if (found) { | 
					
						
							|  |  |  | 		addr->users += todo->users; | 
					
						
							|  |  |  | 		if (addr->users <= 0) { | 
					
						
							|  |  |  | 			*__addr = addr; | 
					
						
							|  |  |  | 			return -1; | 
					
						
							|  |  |  | 		} else { | 
					
						
							|  |  |  | 			/* for VIPA and RXIP limit refcount to 1 */ | 
					
						
							|  |  |  | 			if (addr->type != QETH_IP_TYPE_NORMAL) | 
					
						
							|  |  |  | 				addr->users = 1; | 
					
						
							|  |  |  | 			return 0; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	if (todo->users > 0) { | 
					
						
							|  |  |  | 		/* for VIPA and RXIP limit refcount to 1 */ | 
					
						
							|  |  |  | 		if (todo->type != QETH_IP_TYPE_NORMAL) | 
					
						
							|  |  |  | 			todo->users = 1; | 
					
						
							|  |  |  | 		return 1; | 
					
						
							|  |  |  | 	} else | 
					
						
							|  |  |  | 		return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void __qeth_l3_delete_all_mc(struct qeth_card *card, | 
					
						
							|  |  |  | 					unsigned long *flags) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2008-04-01 10:26:53 +02:00
										 |  |  | 	struct list_head fail_list; | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 	struct qeth_ipaddr *addr, *tmp; | 
					
						
							|  |  |  | 	int rc; | 
					
						
							| 
									
										
										
										
											2008-04-01 10:26:53 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	INIT_LIST_HEAD(&fail_list); | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | again: | 
					
						
							|  |  |  | 	list_for_each_entry_safe(addr, tmp, &card->ip_list, entry) { | 
					
						
							|  |  |  | 		if (addr->is_multicast) { | 
					
						
							|  |  |  | 			list_del(&addr->entry); | 
					
						
							|  |  |  | 			spin_unlock_irqrestore(&card->ip_lock, *flags); | 
					
						
							|  |  |  | 			rc = qeth_l3_deregister_addr_entry(card, addr); | 
					
						
							|  |  |  | 			spin_lock_irqsave(&card->ip_lock, *flags); | 
					
						
							| 
									
										
										
										
											2008-04-01 10:26:53 +02:00
										 |  |  | 			if (!rc || (rc == IPA_RC_MC_ADDR_NOT_FOUND)) | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 				kfree(addr); | 
					
						
							| 
									
										
										
										
											2008-04-01 10:26:53 +02:00
										 |  |  | 			else | 
					
						
							|  |  |  | 				list_add_tail(&addr->entry, &fail_list); | 
					
						
							|  |  |  | 			goto again; | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2008-04-01 10:26:53 +02:00
										 |  |  | 	list_splice(&fail_list, &card->ip_list); | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void qeth_l3_set_ip_addr_list(struct qeth_card *card) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct list_head *tbd_list; | 
					
						
							|  |  |  | 	struct qeth_ipaddr *todo, *addr; | 
					
						
							|  |  |  | 	unsigned long flags; | 
					
						
							|  |  |  | 	int rc; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-06-21 22:57:05 +00:00
										 |  |  | 	QETH_CARD_TEXT(card, 2, "sdiplist"); | 
					
						
							|  |  |  | 	QETH_CARD_HEX(card, 2, &card, sizeof(void *)); | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-01-11 02:50:50 +00:00
										 |  |  | 	if (card->options.sniffer) | 
					
						
							|  |  |  | 		return; | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 	spin_lock_irqsave(&card->ip_lock, flags); | 
					
						
							|  |  |  | 	tbd_list = card->ip_tbd_list; | 
					
						
							|  |  |  | 	card->ip_tbd_list = kmalloc(sizeof(struct list_head), GFP_ATOMIC); | 
					
						
							|  |  |  | 	if (!card->ip_tbd_list) { | 
					
						
							| 
									
										
										
										
											2010-06-21 22:57:05 +00:00
										 |  |  | 		QETH_CARD_TEXT(card, 0, "silnomem"); | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 		card->ip_tbd_list = tbd_list; | 
					
						
							|  |  |  | 		spin_unlock_irqrestore(&card->ip_lock, flags); | 
					
						
							|  |  |  | 		return; | 
					
						
							|  |  |  | 	} else | 
					
						
							|  |  |  | 		INIT_LIST_HEAD(card->ip_tbd_list); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	while (!list_empty(tbd_list)) { | 
					
						
							|  |  |  | 		todo = list_entry(tbd_list->next, struct qeth_ipaddr, entry); | 
					
						
							|  |  |  | 		list_del(&todo->entry); | 
					
						
							|  |  |  | 		if (todo->type == QETH_IP_TYPE_DEL_ALL_MC) { | 
					
						
							|  |  |  | 			__qeth_l3_delete_all_mc(card, &flags); | 
					
						
							|  |  |  | 			kfree(todo); | 
					
						
							|  |  |  | 			continue; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		rc = __qeth_l3_ref_ip_on_card(card, todo, &addr); | 
					
						
							|  |  |  | 		if (rc == 0) { | 
					
						
							|  |  |  | 			/* nothing to be done; only adjusted refcount */ | 
					
						
							|  |  |  | 			kfree(todo); | 
					
						
							|  |  |  | 		} else if (rc == 1) { | 
					
						
							|  |  |  | 			/* new entry to be added to on-card list */ | 
					
						
							|  |  |  | 			spin_unlock_irqrestore(&card->ip_lock, flags); | 
					
						
							|  |  |  | 			rc = qeth_l3_register_addr_entry(card, todo); | 
					
						
							|  |  |  | 			spin_lock_irqsave(&card->ip_lock, flags); | 
					
						
							| 
									
										
										
										
											2008-04-01 10:27:00 +02:00
										 |  |  | 			if (!rc || (rc == IPA_RC_LAN_OFFLINE)) | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 				list_add_tail(&todo->entry, &card->ip_list); | 
					
						
							|  |  |  | 			else | 
					
						
							|  |  |  | 				kfree(todo); | 
					
						
							|  |  |  | 		} else if (rc == -1) { | 
					
						
							|  |  |  | 			/* on-card entry to be removed */ | 
					
						
							|  |  |  | 			list_del_init(&addr->entry); | 
					
						
							|  |  |  | 			spin_unlock_irqrestore(&card->ip_lock, flags); | 
					
						
							|  |  |  | 			rc = qeth_l3_deregister_addr_entry(card, addr); | 
					
						
							|  |  |  | 			spin_lock_irqsave(&card->ip_lock, flags); | 
					
						
							| 
									
										
										
										
											2010-01-11 02:50:50 +00:00
										 |  |  | 			if (!rc || (rc == IPA_RC_IP_ADDRESS_NOT_DEFINED)) | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 				kfree(addr); | 
					
						
							|  |  |  | 			else | 
					
						
							|  |  |  | 				list_add_tail(&addr->entry, &card->ip_list); | 
					
						
							|  |  |  | 			kfree(todo); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	spin_unlock_irqrestore(&card->ip_lock, flags); | 
					
						
							|  |  |  | 	kfree(tbd_list); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void qeth_l3_clear_ip_list(struct qeth_card *card, int clean, | 
					
						
							|  |  |  | 					int recover) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct qeth_ipaddr *addr, *tmp; | 
					
						
							|  |  |  | 	unsigned long flags; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-06-21 22:57:05 +00:00
										 |  |  | 	QETH_CARD_TEXT(card, 4, "clearip"); | 
					
						
							| 
									
										
										
										
											2010-01-11 02:50:50 +00:00
										 |  |  | 	if (recover && card->options.sniffer) | 
					
						
							|  |  |  | 		return; | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 	spin_lock_irqsave(&card->ip_lock, flags); | 
					
						
							|  |  |  | 	/* clear todo list */ | 
					
						
							|  |  |  | 	list_for_each_entry_safe(addr, tmp, card->ip_tbd_list, entry) { | 
					
						
							|  |  |  | 		list_del(&addr->entry); | 
					
						
							|  |  |  | 		kfree(addr); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	while (!list_empty(&card->ip_list)) { | 
					
						
							|  |  |  | 		addr = list_entry(card->ip_list.next, | 
					
						
							|  |  |  | 				  struct qeth_ipaddr, entry); | 
					
						
							|  |  |  | 		list_del_init(&addr->entry); | 
					
						
							|  |  |  | 		if (clean) { | 
					
						
							|  |  |  | 			spin_unlock_irqrestore(&card->ip_lock, flags); | 
					
						
							|  |  |  | 			qeth_l3_deregister_addr_entry(card, addr); | 
					
						
							|  |  |  | 			spin_lock_irqsave(&card->ip_lock, flags); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		if (!recover || addr->is_multicast) { | 
					
						
							|  |  |  | 			kfree(addr); | 
					
						
							|  |  |  | 			continue; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		list_add_tail(&addr->entry, card->ip_tbd_list); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	spin_unlock_irqrestore(&card->ip_lock, flags); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int qeth_l3_address_exists_in_list(struct list_head *list, | 
					
						
							|  |  |  | 		struct qeth_ipaddr *addr, int same_type) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct qeth_ipaddr *tmp; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	list_for_each_entry(tmp, list, entry) { | 
					
						
							|  |  |  | 		if ((tmp->proto == QETH_PROT_IPV4) && | 
					
						
							|  |  |  | 		    (addr->proto == QETH_PROT_IPV4) && | 
					
						
							|  |  |  | 		    ((same_type && (tmp->type == addr->type)) || | 
					
						
							|  |  |  | 		    (!same_type && (tmp->type != addr->type))) && | 
					
						
							|  |  |  | 		    (tmp->u.a4.addr == addr->u.a4.addr)) | 
					
						
							|  |  |  | 			return 1; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		if ((tmp->proto == QETH_PROT_IPV6) && | 
					
						
							|  |  |  | 		    (addr->proto == QETH_PROT_IPV6) && | 
					
						
							|  |  |  | 		    ((same_type && (tmp->type == addr->type)) || | 
					
						
							|  |  |  | 		    (!same_type && (tmp->type != addr->type))) && | 
					
						
							|  |  |  | 		    (memcmp(&tmp->u.a6.addr, &addr->u.a6.addr, | 
					
						
							|  |  |  | 			    sizeof(struct in6_addr)) == 0)) | 
					
						
							|  |  |  | 			return 1; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int qeth_l3_send_setdelmc(struct qeth_card *card, | 
					
						
							|  |  |  | 			struct qeth_ipaddr *addr, int ipacmd) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	int rc; | 
					
						
							|  |  |  | 	struct qeth_cmd_buffer *iob; | 
					
						
							|  |  |  | 	struct qeth_ipa_cmd *cmd; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-06-21 22:57:05 +00:00
										 |  |  | 	QETH_CARD_TEXT(card, 4, "setdelmc"); | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	iob = qeth_get_ipacmd_buffer(card, ipacmd, addr->proto); | 
					
						
							|  |  |  | 	cmd = (struct qeth_ipa_cmd *)(iob->data+IPA_PDU_HEADER_SIZE); | 
					
						
							|  |  |  | 	memcpy(&cmd->data.setdelipm.mac, addr->mac, OSA_ADDR_LEN); | 
					
						
							|  |  |  | 	if (addr->proto == QETH_PROT_IPV6) | 
					
						
							|  |  |  | 		memcpy(cmd->data.setdelipm.ip6, &addr->u.a6.addr, | 
					
						
							|  |  |  | 		       sizeof(struct in6_addr)); | 
					
						
							|  |  |  | 	else | 
					
						
							|  |  |  | 		memcpy(&cmd->data.setdelipm.ip4, &addr->u.a4.addr, 4); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	rc = qeth_send_ipa_cmd(card, iob, NULL, NULL); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return rc; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void qeth_l3_fill_netmask(u8 *netmask, unsigned int len) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	int i, j; | 
					
						
							|  |  |  | 	for (i = 0; i < 16; i++) { | 
					
						
							|  |  |  | 		j = (len) - (i * 8); | 
					
						
							|  |  |  | 		if (j >= 8) | 
					
						
							|  |  |  | 			netmask[i] = 0xff; | 
					
						
							|  |  |  | 		else if (j > 0) | 
					
						
							|  |  |  | 			netmask[i] = (u8)(0xFF00 >> j); | 
					
						
							|  |  |  | 		else | 
					
						
							|  |  |  | 			netmask[i] = 0; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int qeth_l3_send_setdelip(struct qeth_card *card, | 
					
						
							|  |  |  | 		struct qeth_ipaddr *addr, int ipacmd, unsigned int flags) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	int rc; | 
					
						
							|  |  |  | 	struct qeth_cmd_buffer *iob; | 
					
						
							|  |  |  | 	struct qeth_ipa_cmd *cmd; | 
					
						
							|  |  |  | 	__u8 netmask[16]; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-06-21 22:57:05 +00:00
										 |  |  | 	QETH_CARD_TEXT(card, 4, "setdelip"); | 
					
						
							|  |  |  | 	QETH_CARD_TEXT_(card, 4, "flags%02X", flags); | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	iob = qeth_get_ipacmd_buffer(card, ipacmd, addr->proto); | 
					
						
							|  |  |  | 	cmd = (struct qeth_ipa_cmd *)(iob->data+IPA_PDU_HEADER_SIZE); | 
					
						
							|  |  |  | 	if (addr->proto == QETH_PROT_IPV6) { | 
					
						
							|  |  |  | 		memcpy(cmd->data.setdelip6.ip_addr, &addr->u.a6.addr, | 
					
						
							|  |  |  | 		       sizeof(struct in6_addr)); | 
					
						
							|  |  |  | 		qeth_l3_fill_netmask(netmask, addr->u.a6.pfxlen); | 
					
						
							|  |  |  | 		memcpy(cmd->data.setdelip6.mask, netmask, | 
					
						
							|  |  |  | 		       sizeof(struct in6_addr)); | 
					
						
							|  |  |  | 		cmd->data.setdelip6.flags = flags; | 
					
						
							|  |  |  | 	} else { | 
					
						
							|  |  |  | 		memcpy(cmd->data.setdelip4.ip_addr, &addr->u.a4.addr, 4); | 
					
						
							|  |  |  | 		memcpy(cmd->data.setdelip4.mask, &addr->u.a4.mask, 4); | 
					
						
							|  |  |  | 		cmd->data.setdelip4.flags = flags; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	rc = qeth_send_ipa_cmd(card, iob, NULL, NULL); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return rc; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int qeth_l3_send_setrouting(struct qeth_card *card, | 
					
						
							|  |  |  | 	enum qeth_routing_types type, enum qeth_prot_versions prot) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	int rc; | 
					
						
							|  |  |  | 	struct qeth_ipa_cmd *cmd; | 
					
						
							|  |  |  | 	struct qeth_cmd_buffer *iob; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-06-21 22:57:05 +00:00
										 |  |  | 	QETH_CARD_TEXT(card, 4, "setroutg"); | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 	iob = qeth_get_ipacmd_buffer(card, IPA_CMD_SETRTG, prot); | 
					
						
							|  |  |  | 	cmd = (struct qeth_ipa_cmd *)(iob->data+IPA_PDU_HEADER_SIZE); | 
					
						
							|  |  |  | 	cmd->data.setrtg.type = (type); | 
					
						
							|  |  |  | 	rc = qeth_send_ipa_cmd(card, iob, NULL, NULL); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return rc; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void qeth_l3_correct_routing_type(struct qeth_card *card, | 
					
						
							|  |  |  | 		enum qeth_routing_types *type, enum qeth_prot_versions prot) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	if (card->info.type == QETH_CARD_TYPE_IQD) { | 
					
						
							|  |  |  | 		switch (*type) { | 
					
						
							|  |  |  | 		case NO_ROUTER: | 
					
						
							|  |  |  | 		case PRIMARY_CONNECTOR: | 
					
						
							|  |  |  | 		case SECONDARY_CONNECTOR: | 
					
						
							|  |  |  | 		case MULTICAST_ROUTER: | 
					
						
							|  |  |  | 			return; | 
					
						
							|  |  |  | 		default: | 
					
						
							|  |  |  | 			goto out_inval; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} else { | 
					
						
							|  |  |  | 		switch (*type) { | 
					
						
							|  |  |  | 		case NO_ROUTER: | 
					
						
							|  |  |  | 		case PRIMARY_ROUTER: | 
					
						
							|  |  |  | 		case SECONDARY_ROUTER: | 
					
						
							|  |  |  | 			return; | 
					
						
							|  |  |  | 		case MULTICAST_ROUTER: | 
					
						
							|  |  |  | 			if (qeth_is_ipafunc_supported(card, prot, | 
					
						
							|  |  |  | 						      IPA_OSA_MC_ROUTER)) | 
					
						
							|  |  |  | 				return; | 
					
						
							|  |  |  | 		default: | 
					
						
							|  |  |  | 			goto out_inval; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | out_inval: | 
					
						
							|  |  |  | 	*type = NO_ROUTER; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | int qeth_l3_setrouting_v4(struct qeth_card *card) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	int rc; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-06-21 22:57:05 +00:00
										 |  |  | 	QETH_CARD_TEXT(card, 3, "setrtg4"); | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	qeth_l3_correct_routing_type(card, &card->options.route4.type, | 
					
						
							|  |  |  | 				  QETH_PROT_IPV4); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	rc = qeth_l3_send_setrouting(card, card->options.route4.type, | 
					
						
							|  |  |  | 				  QETH_PROT_IPV4); | 
					
						
							|  |  |  | 	if (rc) { | 
					
						
							|  |  |  | 		card->options.route4.type = NO_ROUTER; | 
					
						
							| 
									
										
										
										
											2008-06-06 12:37:46 +02:00
										 |  |  | 		QETH_DBF_MESSAGE(2, "Error (0x%04x) while setting routing type" | 
					
						
							|  |  |  | 			" on %s. Type set to 'no router'.\n", rc, | 
					
						
							|  |  |  | 			QETH_CARD_IFNAME(card)); | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 	} | 
					
						
							|  |  |  | 	return rc; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | int qeth_l3_setrouting_v6(struct qeth_card *card) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	int rc = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-06-21 22:57:05 +00:00
										 |  |  | 	QETH_CARD_TEXT(card, 3, "setrtg6"); | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | #ifdef CONFIG_QETH_IPV6
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (!qeth_is_supported(card, IPA_IPV6)) | 
					
						
							|  |  |  | 		return 0; | 
					
						
							|  |  |  | 	qeth_l3_correct_routing_type(card, &card->options.route6.type, | 
					
						
							|  |  |  | 				  QETH_PROT_IPV6); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	rc = qeth_l3_send_setrouting(card, card->options.route6.type, | 
					
						
							|  |  |  | 				  QETH_PROT_IPV6); | 
					
						
							|  |  |  | 	if (rc) { | 
					
						
							|  |  |  | 		card->options.route6.type = NO_ROUTER; | 
					
						
							| 
									
										
										
										
											2008-06-06 12:37:46 +02:00
										 |  |  | 		QETH_DBF_MESSAGE(2, "Error (0x%04x) while setting routing type" | 
					
						
							|  |  |  | 			" on %s. Type set to 'no router'.\n", rc, | 
					
						
							|  |  |  | 			QETH_CARD_IFNAME(card)); | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 	} | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 	return rc; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /*
 | 
					
						
							|  |  |  |  * IP address takeover related functions | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | static void qeth_l3_clear_ipato_list(struct qeth_card *card) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	struct qeth_ipato_entry *ipatoe, *tmp; | 
					
						
							|  |  |  | 	unsigned long flags; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	spin_lock_irqsave(&card->ip_lock, flags); | 
					
						
							|  |  |  | 	list_for_each_entry_safe(ipatoe, tmp, &card->ipato.entries, entry) { | 
					
						
							|  |  |  | 		list_del(&ipatoe->entry); | 
					
						
							|  |  |  | 		kfree(ipatoe); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	spin_unlock_irqrestore(&card->ip_lock, flags); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | int qeth_l3_add_ipato_entry(struct qeth_card *card, | 
					
						
							|  |  |  | 				struct qeth_ipato_entry *new) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct qeth_ipato_entry *ipatoe; | 
					
						
							|  |  |  | 	unsigned long flags; | 
					
						
							|  |  |  | 	int rc = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-06-21 22:57:05 +00:00
										 |  |  | 	QETH_CARD_TEXT(card, 2, "addipato"); | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 	spin_lock_irqsave(&card->ip_lock, flags); | 
					
						
							|  |  |  | 	list_for_each_entry(ipatoe, &card->ipato.entries, entry) { | 
					
						
							|  |  |  | 		if (ipatoe->proto != new->proto) | 
					
						
							|  |  |  | 			continue; | 
					
						
							|  |  |  | 		if (!memcmp(ipatoe->addr, new->addr, | 
					
						
							|  |  |  | 			    (ipatoe->proto == QETH_PROT_IPV4)? 4:16) && | 
					
						
							|  |  |  | 		    (ipatoe->mask_bits == new->mask_bits)) { | 
					
						
							|  |  |  | 			rc = -EEXIST; | 
					
						
							|  |  |  | 			break; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	if (!rc) | 
					
						
							|  |  |  | 		list_add_tail(&new->entry, &card->ipato.entries); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	spin_unlock_irqrestore(&card->ip_lock, flags); | 
					
						
							|  |  |  | 	return rc; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void qeth_l3_del_ipato_entry(struct qeth_card *card, | 
					
						
							|  |  |  | 		enum qeth_prot_versions proto, u8 *addr, int mask_bits) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct qeth_ipato_entry *ipatoe, *tmp; | 
					
						
							|  |  |  | 	unsigned long flags; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-06-21 22:57:05 +00:00
										 |  |  | 	QETH_CARD_TEXT(card, 2, "delipato"); | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 	spin_lock_irqsave(&card->ip_lock, flags); | 
					
						
							|  |  |  | 	list_for_each_entry_safe(ipatoe, tmp, &card->ipato.entries, entry) { | 
					
						
							|  |  |  | 		if (ipatoe->proto != proto) | 
					
						
							|  |  |  | 			continue; | 
					
						
							|  |  |  | 		if (!memcmp(ipatoe->addr, addr, | 
					
						
							|  |  |  | 			    (proto == QETH_PROT_IPV4)? 4:16) && | 
					
						
							|  |  |  | 		    (ipatoe->mask_bits == mask_bits)) { | 
					
						
							|  |  |  | 			list_del(&ipatoe->entry); | 
					
						
							|  |  |  | 			kfree(ipatoe); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	spin_unlock_irqrestore(&card->ip_lock, flags); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /*
 | 
					
						
							|  |  |  |  * VIPA related functions | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | int qeth_l3_add_vipa(struct qeth_card *card, enum qeth_prot_versions proto, | 
					
						
							|  |  |  | 	      const u8 *addr) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct qeth_ipaddr *ipaddr; | 
					
						
							|  |  |  | 	unsigned long flags; | 
					
						
							|  |  |  | 	int rc = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	ipaddr = qeth_l3_get_addr_buffer(proto); | 
					
						
							|  |  |  | 	if (ipaddr) { | 
					
						
							|  |  |  | 		if (proto == QETH_PROT_IPV4) { | 
					
						
							| 
									
										
										
										
											2010-06-21 22:57:05 +00:00
										 |  |  | 			QETH_CARD_TEXT(card, 2, "addvipa4"); | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 			memcpy(&ipaddr->u.a4.addr, addr, 4); | 
					
						
							|  |  |  | 			ipaddr->u.a4.mask = 0; | 
					
						
							|  |  |  | 		} else if (proto == QETH_PROT_IPV6) { | 
					
						
							| 
									
										
										
										
											2010-06-21 22:57:05 +00:00
										 |  |  | 			QETH_CARD_TEXT(card, 2, "addvipa6"); | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 			memcpy(&ipaddr->u.a6.addr, addr, 16); | 
					
						
							|  |  |  | 			ipaddr->u.a6.pfxlen = 0; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		ipaddr->type = QETH_IP_TYPE_VIPA; | 
					
						
							|  |  |  | 		ipaddr->set_flags = QETH_IPA_SETIP_VIPA_FLAG; | 
					
						
							|  |  |  | 		ipaddr->del_flags = QETH_IPA_DELIP_VIPA_FLAG; | 
					
						
							|  |  |  | 	} else | 
					
						
							|  |  |  | 		return -ENOMEM; | 
					
						
							|  |  |  | 	spin_lock_irqsave(&card->ip_lock, flags); | 
					
						
							|  |  |  | 	if (qeth_l3_address_exists_in_list(&card->ip_list, ipaddr, 0) || | 
					
						
							|  |  |  | 	    qeth_l3_address_exists_in_list(card->ip_tbd_list, ipaddr, 0)) | 
					
						
							|  |  |  | 		rc = -EEXIST; | 
					
						
							|  |  |  | 	spin_unlock_irqrestore(&card->ip_lock, flags); | 
					
						
							|  |  |  | 	if (rc) { | 
					
						
							|  |  |  | 		return rc; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	if (!qeth_l3_add_ip(card, ipaddr)) | 
					
						
							|  |  |  | 		kfree(ipaddr); | 
					
						
							|  |  |  | 	qeth_l3_set_ip_addr_list(card); | 
					
						
							|  |  |  | 	return rc; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void qeth_l3_del_vipa(struct qeth_card *card, enum qeth_prot_versions proto, | 
					
						
							|  |  |  | 	      const u8 *addr) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct qeth_ipaddr *ipaddr; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	ipaddr = qeth_l3_get_addr_buffer(proto); | 
					
						
							|  |  |  | 	if (ipaddr) { | 
					
						
							|  |  |  | 		if (proto == QETH_PROT_IPV4) { | 
					
						
							| 
									
										
										
										
											2010-06-21 22:57:05 +00:00
										 |  |  | 			QETH_CARD_TEXT(card, 2, "delvipa4"); | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 			memcpy(&ipaddr->u.a4.addr, addr, 4); | 
					
						
							|  |  |  | 			ipaddr->u.a4.mask = 0; | 
					
						
							|  |  |  | 		} else if (proto == QETH_PROT_IPV6) { | 
					
						
							| 
									
										
										
										
											2010-06-21 22:57:05 +00:00
										 |  |  | 			QETH_CARD_TEXT(card, 2, "delvipa6"); | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 			memcpy(&ipaddr->u.a6.addr, addr, 16); | 
					
						
							|  |  |  | 			ipaddr->u.a6.pfxlen = 0; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		ipaddr->type = QETH_IP_TYPE_VIPA; | 
					
						
							|  |  |  | 	} else | 
					
						
							|  |  |  | 		return; | 
					
						
							|  |  |  | 	if (!qeth_l3_delete_ip(card, ipaddr)) | 
					
						
							|  |  |  | 		kfree(ipaddr); | 
					
						
							|  |  |  | 	qeth_l3_set_ip_addr_list(card); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /*
 | 
					
						
							|  |  |  |  * proxy ARP related functions | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | int qeth_l3_add_rxip(struct qeth_card *card, enum qeth_prot_versions proto, | 
					
						
							|  |  |  | 	      const u8 *addr) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct qeth_ipaddr *ipaddr; | 
					
						
							|  |  |  | 	unsigned long flags; | 
					
						
							|  |  |  | 	int rc = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	ipaddr = qeth_l3_get_addr_buffer(proto); | 
					
						
							|  |  |  | 	if (ipaddr) { | 
					
						
							|  |  |  | 		if (proto == QETH_PROT_IPV4) { | 
					
						
							| 
									
										
										
										
											2010-06-21 22:57:05 +00:00
										 |  |  | 			QETH_CARD_TEXT(card, 2, "addrxip4"); | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 			memcpy(&ipaddr->u.a4.addr, addr, 4); | 
					
						
							|  |  |  | 			ipaddr->u.a4.mask = 0; | 
					
						
							|  |  |  | 		} else if (proto == QETH_PROT_IPV6) { | 
					
						
							| 
									
										
										
										
											2010-06-21 22:57:05 +00:00
										 |  |  | 			QETH_CARD_TEXT(card, 2, "addrxip6"); | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 			memcpy(&ipaddr->u.a6.addr, addr, 16); | 
					
						
							|  |  |  | 			ipaddr->u.a6.pfxlen = 0; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		ipaddr->type = QETH_IP_TYPE_RXIP; | 
					
						
							|  |  |  | 		ipaddr->set_flags = QETH_IPA_SETIP_TAKEOVER_FLAG; | 
					
						
							|  |  |  | 		ipaddr->del_flags = 0; | 
					
						
							|  |  |  | 	} else | 
					
						
							|  |  |  | 		return -ENOMEM; | 
					
						
							|  |  |  | 	spin_lock_irqsave(&card->ip_lock, flags); | 
					
						
							|  |  |  | 	if (qeth_l3_address_exists_in_list(&card->ip_list, ipaddr, 0) || | 
					
						
							|  |  |  | 	    qeth_l3_address_exists_in_list(card->ip_tbd_list, ipaddr, 0)) | 
					
						
							|  |  |  | 		rc = -EEXIST; | 
					
						
							|  |  |  | 	spin_unlock_irqrestore(&card->ip_lock, flags); | 
					
						
							|  |  |  | 	if (rc) { | 
					
						
							|  |  |  | 		return rc; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	if (!qeth_l3_add_ip(card, ipaddr)) | 
					
						
							|  |  |  | 		kfree(ipaddr); | 
					
						
							|  |  |  | 	qeth_l3_set_ip_addr_list(card); | 
					
						
							|  |  |  | 	return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void qeth_l3_del_rxip(struct qeth_card *card, enum qeth_prot_versions proto, | 
					
						
							|  |  |  | 			const u8 *addr) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct qeth_ipaddr *ipaddr; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	ipaddr = qeth_l3_get_addr_buffer(proto); | 
					
						
							|  |  |  | 	if (ipaddr) { | 
					
						
							|  |  |  | 		if (proto == QETH_PROT_IPV4) { | 
					
						
							| 
									
										
										
										
											2010-06-21 22:57:05 +00:00
										 |  |  | 			QETH_CARD_TEXT(card, 2, "addrxip4"); | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 			memcpy(&ipaddr->u.a4.addr, addr, 4); | 
					
						
							|  |  |  | 			ipaddr->u.a4.mask = 0; | 
					
						
							|  |  |  | 		} else if (proto == QETH_PROT_IPV6) { | 
					
						
							| 
									
										
										
										
											2010-06-21 22:57:05 +00:00
										 |  |  | 			QETH_CARD_TEXT(card, 2, "addrxip6"); | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 			memcpy(&ipaddr->u.a6.addr, addr, 16); | 
					
						
							|  |  |  | 			ipaddr->u.a6.pfxlen = 0; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		ipaddr->type = QETH_IP_TYPE_RXIP; | 
					
						
							|  |  |  | 	} else | 
					
						
							|  |  |  | 		return; | 
					
						
							|  |  |  | 	if (!qeth_l3_delete_ip(card, ipaddr)) | 
					
						
							|  |  |  | 		kfree(ipaddr); | 
					
						
							|  |  |  | 	qeth_l3_set_ip_addr_list(card); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int qeth_l3_register_addr_entry(struct qeth_card *card, | 
					
						
							|  |  |  | 				struct qeth_ipaddr *addr) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	char buf[50]; | 
					
						
							|  |  |  | 	int rc = 0; | 
					
						
							|  |  |  | 	int cnt = 3; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (addr->proto == QETH_PROT_IPV4) { | 
					
						
							| 
									
										
										
										
											2010-06-21 22:57:05 +00:00
										 |  |  | 		QETH_CARD_TEXT(card, 2, "setaddr4"); | 
					
						
							|  |  |  | 		QETH_CARD_HEX(card, 3, &addr->u.a4.addr, sizeof(int)); | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 	} else if (addr->proto == QETH_PROT_IPV6) { | 
					
						
							| 
									
										
										
										
											2010-06-21 22:57:05 +00:00
										 |  |  | 		QETH_CARD_TEXT(card, 2, "setaddr6"); | 
					
						
							|  |  |  | 		QETH_CARD_HEX(card, 3, &addr->u.a6.addr, 8); | 
					
						
							|  |  |  | 		QETH_CARD_HEX(card, 3, ((char *)&addr->u.a6.addr) + 8, 8); | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 	} else { | 
					
						
							| 
									
										
										
										
											2010-06-21 22:57:05 +00:00
										 |  |  | 		QETH_CARD_TEXT(card, 2, "setaddr?"); | 
					
						
							|  |  |  | 		QETH_CARD_HEX(card, 3, addr, sizeof(struct qeth_ipaddr)); | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 	} | 
					
						
							|  |  |  | 	do { | 
					
						
							|  |  |  | 		if (addr->is_multicast) | 
					
						
							|  |  |  | 			rc =  qeth_l3_send_setdelmc(card, addr, IPA_CMD_SETIPM); | 
					
						
							|  |  |  | 		else | 
					
						
							|  |  |  | 			rc = qeth_l3_send_setdelip(card, addr, IPA_CMD_SETIP, | 
					
						
							|  |  |  | 					addr->set_flags); | 
					
						
							|  |  |  | 		if (rc) | 
					
						
							| 
									
										
										
										
											2010-06-21 22:57:05 +00:00
										 |  |  | 			QETH_CARD_TEXT(card, 2, "failed"); | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 	} while ((--cnt > 0) && rc); | 
					
						
							|  |  |  | 	if (rc) { | 
					
						
							| 
									
										
										
										
											2010-06-21 22:57:05 +00:00
										 |  |  | 		QETH_CARD_TEXT(card, 2, "FAILED"); | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 		qeth_l3_ipaddr_to_string(addr->proto, (u8 *)&addr->u, buf); | 
					
						
							| 
									
										
										
										
											2008-12-25 13:39:49 +01:00
										 |  |  | 		dev_warn(&card->gdev->dev, | 
					
						
							|  |  |  | 			"Registering IP address %s failed\n", buf); | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 	} | 
					
						
							|  |  |  | 	return rc; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int qeth_l3_deregister_addr_entry(struct qeth_card *card, | 
					
						
							|  |  |  | 						struct qeth_ipaddr *addr) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	int rc = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (addr->proto == QETH_PROT_IPV4) { | 
					
						
							| 
									
										
										
										
											2010-06-21 22:57:05 +00:00
										 |  |  | 		QETH_CARD_TEXT(card, 2, "deladdr4"); | 
					
						
							|  |  |  | 		QETH_CARD_HEX(card, 3, &addr->u.a4.addr, sizeof(int)); | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 	} else if (addr->proto == QETH_PROT_IPV6) { | 
					
						
							| 
									
										
										
										
											2010-06-21 22:57:05 +00:00
										 |  |  | 		QETH_CARD_TEXT(card, 2, "deladdr6"); | 
					
						
							|  |  |  | 		QETH_CARD_HEX(card, 3, &addr->u.a6.addr, 8); | 
					
						
							|  |  |  | 		QETH_CARD_HEX(card, 3, ((char *)&addr->u.a6.addr) + 8, 8); | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 	} else { | 
					
						
							| 
									
										
										
										
											2010-06-21 22:57:05 +00:00
										 |  |  | 		QETH_CARD_TEXT(card, 2, "deladdr?"); | 
					
						
							|  |  |  | 		QETH_CARD_HEX(card, 3, addr, sizeof(struct qeth_ipaddr)); | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 	} | 
					
						
							|  |  |  | 	if (addr->is_multicast) | 
					
						
							|  |  |  | 		rc = qeth_l3_send_setdelmc(card, addr, IPA_CMD_DELIPM); | 
					
						
							|  |  |  | 	else | 
					
						
							|  |  |  | 		rc = qeth_l3_send_setdelip(card, addr, IPA_CMD_DELIP, | 
					
						
							|  |  |  | 					addr->del_flags); | 
					
						
							| 
									
										
										
										
											2008-07-14 09:59:31 +02:00
										 |  |  | 	if (rc) | 
					
						
							| 
									
										
										
										
											2010-06-21 22:57:05 +00:00
										 |  |  | 		QETH_CARD_TEXT(card, 2, "failed"); | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	return rc; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static inline u8 qeth_l3_get_qeth_hdr_flags4(int cast_type) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	if (cast_type == RTN_MULTICAST) | 
					
						
							|  |  |  | 		return QETH_CAST_MULTICAST; | 
					
						
							|  |  |  | 	if (cast_type == RTN_BROADCAST) | 
					
						
							|  |  |  | 		return QETH_CAST_BROADCAST; | 
					
						
							|  |  |  | 	return QETH_CAST_UNICAST; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static inline u8 qeth_l3_get_qeth_hdr_flags6(int cast_type) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	u8 ct = QETH_HDR_PASSTHRU | QETH_HDR_IPV6; | 
					
						
							|  |  |  | 	if (cast_type == RTN_MULTICAST) | 
					
						
							|  |  |  | 		return ct | QETH_CAST_MULTICAST; | 
					
						
							|  |  |  | 	if (cast_type == RTN_ANYCAST) | 
					
						
							|  |  |  | 		return ct | QETH_CAST_ANYCAST; | 
					
						
							|  |  |  | 	if (cast_type == RTN_BROADCAST) | 
					
						
							|  |  |  | 		return ct | QETH_CAST_BROADCAST; | 
					
						
							|  |  |  | 	return ct | QETH_CAST_UNICAST; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int qeth_l3_send_setadp_mode(struct qeth_card *card, __u32 command, | 
					
						
							|  |  |  | 					__u32 mode) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	int rc; | 
					
						
							|  |  |  | 	struct qeth_cmd_buffer *iob; | 
					
						
							|  |  |  | 	struct qeth_ipa_cmd *cmd; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-06-21 22:57:05 +00:00
										 |  |  | 	QETH_CARD_TEXT(card, 4, "adpmode"); | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	iob = qeth_get_adapter_cmd(card, command, | 
					
						
							|  |  |  | 				   sizeof(struct qeth_ipacmd_setadpparms)); | 
					
						
							|  |  |  | 	cmd = (struct qeth_ipa_cmd *)(iob->data+IPA_PDU_HEADER_SIZE); | 
					
						
							|  |  |  | 	cmd->data.setadapterparms.data.mode = mode; | 
					
						
							|  |  |  | 	rc = qeth_send_ipa_cmd(card, iob, qeth_default_setadapterparms_cb, | 
					
						
							|  |  |  | 			       NULL); | 
					
						
							|  |  |  | 	return rc; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int qeth_l3_setadapter_hstr(struct qeth_card *card) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	int rc; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-06-21 22:57:05 +00:00
										 |  |  | 	QETH_CARD_TEXT(card, 4, "adphstr"); | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	if (qeth_adp_supported(card, IPA_SETADP_SET_BROADCAST_MODE)) { | 
					
						
							|  |  |  | 		rc = qeth_l3_send_setadp_mode(card, | 
					
						
							|  |  |  | 					IPA_SETADP_SET_BROADCAST_MODE, | 
					
						
							|  |  |  | 					card->options.broadcast_mode); | 
					
						
							|  |  |  | 		if (rc) | 
					
						
							| 
									
										
										
										
											2008-06-06 12:37:46 +02:00
										 |  |  | 			QETH_DBF_MESSAGE(2, "couldn't set broadcast mode on " | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 				   "device %s: x%x\n", | 
					
						
							|  |  |  | 				   CARD_BUS_ID(card), rc); | 
					
						
							|  |  |  | 		rc = qeth_l3_send_setadp_mode(card, | 
					
						
							|  |  |  | 					IPA_SETADP_ALTER_MAC_ADDRESS, | 
					
						
							|  |  |  | 					card->options.macaddr_mode); | 
					
						
							|  |  |  | 		if (rc) | 
					
						
							| 
									
										
										
										
											2008-06-06 12:37:46 +02:00
										 |  |  | 			QETH_DBF_MESSAGE(2, "couldn't set macaddr mode on " | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 				   "device %s: x%x\n", CARD_BUS_ID(card), rc); | 
					
						
							|  |  |  | 		return rc; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	if (card->options.broadcast_mode == QETH_TR_BROADCAST_LOCAL) | 
					
						
							| 
									
										
										
										
											2008-06-06 12:37:46 +02:00
										 |  |  | 		QETH_DBF_MESSAGE(2, "set adapter parameters not available " | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 			   "to set broadcast mode, using ALLRINGS " | 
					
						
							|  |  |  | 			   "on device %s:\n", CARD_BUS_ID(card)); | 
					
						
							|  |  |  | 	if (card->options.macaddr_mode == QETH_TR_MACADDR_CANONICAL) | 
					
						
							| 
									
										
										
										
											2008-06-06 12:37:46 +02:00
										 |  |  | 		QETH_DBF_MESSAGE(2, "set adapter parameters not available " | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 			   "to set macaddr mode, using NONCANONICAL " | 
					
						
							|  |  |  | 			   "on device %s:\n", CARD_BUS_ID(card)); | 
					
						
							|  |  |  | 	return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int qeth_l3_setadapter_parms(struct qeth_card *card) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	int rc; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-04-01 10:26:58 +02:00
										 |  |  | 	QETH_DBF_TEXT(SETUP, 2, "setadprm"); | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	if (!qeth_is_supported(card, IPA_SETADAPTERPARMS)) { | 
					
						
							| 
									
										
										
										
											2008-12-25 13:39:49 +01:00
										 |  |  | 		dev_info(&card->gdev->dev, | 
					
						
							|  |  |  | 			"set adapter parameters not supported.\n"); | 
					
						
							| 
									
										
										
										
											2008-04-01 10:26:58 +02:00
										 |  |  | 		QETH_DBF_TEXT(SETUP, 2, " notsupp"); | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 		return 0; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	rc = qeth_query_setadapterparms(card); | 
					
						
							|  |  |  | 	if (rc) { | 
					
						
							| 
									
										
										
										
											2008-12-25 13:39:49 +01:00
										 |  |  | 		QETH_DBF_MESSAGE(2, "%s couldn't set adapter parameters: " | 
					
						
							| 
									
										
										
										
											2009-03-24 20:57:14 +00:00
										 |  |  | 			"0x%x\n", dev_name(&card->gdev->dev), rc); | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 		return rc; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	if (qeth_adp_supported(card, IPA_SETADP_ALTER_MAC_ADDRESS)) { | 
					
						
							|  |  |  | 		rc = qeth_setadpparms_change_macaddr(card); | 
					
						
							|  |  |  | 		if (rc) | 
					
						
							| 
									
										
										
										
											2008-12-25 13:39:49 +01:00
										 |  |  | 			dev_warn(&card->gdev->dev, "Reading the adapter MAC" | 
					
						
							| 
									
										
										
										
											2009-01-04 17:36:32 -08:00
										 |  |  | 				" address failed\n"); | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if ((card->info.link_type == QETH_LINK_TYPE_HSTR) || | 
					
						
							|  |  |  | 	    (card->info.link_type == QETH_LINK_TYPE_LANE_TR)) | 
					
						
							|  |  |  | 		rc = qeth_l3_setadapter_hstr(card); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return rc; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int qeth_l3_default_setassparms_cb(struct qeth_card *card, | 
					
						
							|  |  |  | 			struct qeth_reply *reply, unsigned long data) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct qeth_ipa_cmd *cmd; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-06-21 22:57:05 +00:00
										 |  |  | 	QETH_CARD_TEXT(card, 4, "defadpcb"); | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	cmd = (struct qeth_ipa_cmd *) data; | 
					
						
							|  |  |  | 	if (cmd->hdr.return_code == 0) { | 
					
						
							|  |  |  | 		cmd->hdr.return_code = cmd->data.setassparms.hdr.return_code; | 
					
						
							|  |  |  | 		if (cmd->hdr.prot_version == QETH_PROT_IPV4) | 
					
						
							|  |  |  | 			card->options.ipa4.enabled_funcs = cmd->hdr.ipa_enabled; | 
					
						
							|  |  |  | 		if (cmd->hdr.prot_version == QETH_PROT_IPV6) | 
					
						
							|  |  |  | 			card->options.ipa6.enabled_funcs = cmd->hdr.ipa_enabled; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	if (cmd->data.setassparms.hdr.assist_no == IPA_INBOUND_CHECKSUM && | 
					
						
							|  |  |  | 	    cmd->data.setassparms.hdr.command_code == IPA_CMD_ASS_START) { | 
					
						
							|  |  |  | 		card->info.csum_mask = cmd->data.setassparms.data.flags_32bit; | 
					
						
							| 
									
										
										
										
											2010-06-21 22:57:05 +00:00
										 |  |  | 		QETH_CARD_TEXT_(card, 3, "csum:%d", card->info.csum_mask); | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2010-05-11 19:34:45 +00:00
										 |  |  | 	if (cmd->data.setassparms.hdr.assist_no == IPA_OUTBOUND_CHECKSUM && | 
					
						
							|  |  |  | 	    cmd->data.setassparms.hdr.command_code == IPA_CMD_ASS_START) { | 
					
						
							|  |  |  | 		card->info.tx_csum_mask = | 
					
						
							|  |  |  | 			cmd->data.setassparms.data.flags_32bit; | 
					
						
							| 
									
										
										
										
											2010-06-21 22:57:05 +00:00
										 |  |  | 		QETH_CARD_TEXT_(card, 3, "tcsu:%d", card->info.tx_csum_mask); | 
					
						
							| 
									
										
										
										
											2010-05-11 19:34:45 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 	return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static struct qeth_cmd_buffer *qeth_l3_get_setassparms_cmd( | 
					
						
							|  |  |  | 	struct qeth_card *card, enum qeth_ipa_funcs ipa_func, __u16 cmd_code, | 
					
						
							|  |  |  | 	__u16 len, enum qeth_prot_versions prot) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct qeth_cmd_buffer *iob; | 
					
						
							|  |  |  | 	struct qeth_ipa_cmd *cmd; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-06-21 22:57:05 +00:00
										 |  |  | 	QETH_CARD_TEXT(card, 4, "getasscm"); | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 	iob = qeth_get_ipacmd_buffer(card, IPA_CMD_SETASSPARMS, prot); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	cmd = (struct qeth_ipa_cmd *)(iob->data+IPA_PDU_HEADER_SIZE); | 
					
						
							|  |  |  | 	cmd->data.setassparms.hdr.assist_no = ipa_func; | 
					
						
							|  |  |  | 	cmd->data.setassparms.hdr.length = 8 + len; | 
					
						
							|  |  |  | 	cmd->data.setassparms.hdr.command_code = cmd_code; | 
					
						
							|  |  |  | 	cmd->data.setassparms.hdr.return_code = 0; | 
					
						
							|  |  |  | 	cmd->data.setassparms.hdr.seq_no = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return iob; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int qeth_l3_send_setassparms(struct qeth_card *card, | 
					
						
							|  |  |  | 	struct qeth_cmd_buffer *iob, __u16 len, long data, | 
					
						
							|  |  |  | 	int (*reply_cb)(struct qeth_card *, struct qeth_reply *, | 
					
						
							|  |  |  | 		unsigned long), | 
					
						
							|  |  |  | 	void *reply_param) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	int rc; | 
					
						
							|  |  |  | 	struct qeth_ipa_cmd *cmd; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-06-21 22:57:05 +00:00
										 |  |  | 	QETH_CARD_TEXT(card, 4, "sendassp"); | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	cmd = (struct qeth_ipa_cmd *)(iob->data+IPA_PDU_HEADER_SIZE); | 
					
						
							|  |  |  | 	if (len <= sizeof(__u32)) | 
					
						
							|  |  |  | 		cmd->data.setassparms.data.flags_32bit = (__u32) data; | 
					
						
							|  |  |  | 	else   /* (len > sizeof(__u32)) */ | 
					
						
							|  |  |  | 		memcpy(&cmd->data.setassparms.data, (void *) data, len); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	rc = qeth_send_ipa_cmd(card, iob, reply_cb, reply_param); | 
					
						
							|  |  |  | 	return rc; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #ifdef CONFIG_QETH_IPV6
 | 
					
						
							|  |  |  | static int qeth_l3_send_simple_setassparms_ipv6(struct qeth_card *card, | 
					
						
							|  |  |  | 		enum qeth_ipa_funcs ipa_func, __u16 cmd_code) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	int rc; | 
					
						
							|  |  |  | 	struct qeth_cmd_buffer *iob; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-06-21 22:57:05 +00:00
										 |  |  | 	QETH_CARD_TEXT(card, 4, "simassp6"); | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 	iob = qeth_l3_get_setassparms_cmd(card, ipa_func, cmd_code, | 
					
						
							|  |  |  | 				       0, QETH_PROT_IPV6); | 
					
						
							|  |  |  | 	rc = qeth_l3_send_setassparms(card, iob, 0, 0, | 
					
						
							|  |  |  | 				   qeth_l3_default_setassparms_cb, NULL); | 
					
						
							|  |  |  | 	return rc; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int qeth_l3_send_simple_setassparms(struct qeth_card *card, | 
					
						
							|  |  |  | 		enum qeth_ipa_funcs ipa_func, __u16 cmd_code, long data) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	int rc; | 
					
						
							|  |  |  | 	int length = 0; | 
					
						
							|  |  |  | 	struct qeth_cmd_buffer *iob; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-06-21 22:57:05 +00:00
										 |  |  | 	QETH_CARD_TEXT(card, 4, "simassp4"); | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 	if (data) | 
					
						
							|  |  |  | 		length = sizeof(__u32); | 
					
						
							|  |  |  | 	iob = qeth_l3_get_setassparms_cmd(card, ipa_func, cmd_code, | 
					
						
							|  |  |  | 				       length, QETH_PROT_IPV4); | 
					
						
							|  |  |  | 	rc = qeth_l3_send_setassparms(card, iob, length, data, | 
					
						
							|  |  |  | 				   qeth_l3_default_setassparms_cb, NULL); | 
					
						
							|  |  |  | 	return rc; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int qeth_l3_start_ipa_arp_processing(struct qeth_card *card) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	int rc; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-06-21 22:57:05 +00:00
										 |  |  | 	QETH_CARD_TEXT(card, 3, "ipaarp"); | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	if (!qeth_is_supported(card, IPA_ARP_PROCESSING)) { | 
					
						
							| 
									
										
										
										
											2008-12-25 13:39:49 +01:00
										 |  |  | 		dev_info(&card->gdev->dev, | 
					
						
							|  |  |  | 			"ARP processing not supported on %s!\n", | 
					
						
							|  |  |  | 			QETH_CARD_IFNAME(card)); | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 		return 0; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	rc = qeth_l3_send_simple_setassparms(card, IPA_ARP_PROCESSING, | 
					
						
							|  |  |  | 					IPA_CMD_ASS_START, 0); | 
					
						
							|  |  |  | 	if (rc) { | 
					
						
							| 
									
										
										
										
											2008-12-25 13:39:49 +01:00
										 |  |  | 		dev_warn(&card->gdev->dev, | 
					
						
							|  |  |  | 			"Starting ARP processing support for %s failed\n", | 
					
						
							|  |  |  | 			QETH_CARD_IFNAME(card)); | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 	} | 
					
						
							|  |  |  | 	return rc; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int qeth_l3_start_ipa_ip_fragmentation(struct qeth_card *card) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	int rc; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-06-21 22:57:05 +00:00
										 |  |  | 	QETH_CARD_TEXT(card, 3, "ipaipfrg"); | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	if (!qeth_is_supported(card, IPA_IP_FRAGMENTATION)) { | 
					
						
							| 
									
										
										
										
											2008-12-25 13:39:49 +01:00
										 |  |  | 		dev_info(&card->gdev->dev, | 
					
						
							|  |  |  | 			"Hardware IP fragmentation not supported on %s\n", | 
					
						
							|  |  |  | 			QETH_CARD_IFNAME(card)); | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 		return  -EOPNOTSUPP; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	rc = qeth_l3_send_simple_setassparms(card, IPA_IP_FRAGMENTATION, | 
					
						
							|  |  |  | 					  IPA_CMD_ASS_START, 0); | 
					
						
							|  |  |  | 	if (rc) { | 
					
						
							| 
									
										
										
										
											2008-12-25 13:39:49 +01:00
										 |  |  | 		dev_warn(&card->gdev->dev, | 
					
						
							|  |  |  | 			"Starting IP fragmentation support for %s failed\n", | 
					
						
							|  |  |  | 			QETH_CARD_IFNAME(card)); | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 	} else | 
					
						
							| 
									
										
										
										
											2008-12-25 13:39:49 +01:00
										 |  |  | 		dev_info(&card->gdev->dev, | 
					
						
							|  |  |  | 			"Hardware IP fragmentation enabled \n"); | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 	return rc; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int qeth_l3_start_ipa_source_mac(struct qeth_card *card) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	int rc; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-06-21 22:57:05 +00:00
										 |  |  | 	QETH_CARD_TEXT(card, 3, "stsrcmac"); | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	if (!qeth_is_supported(card, IPA_SOURCE_MAC)) { | 
					
						
							| 
									
										
										
										
											2008-12-25 13:39:49 +01:00
										 |  |  | 		dev_info(&card->gdev->dev, | 
					
						
							| 
									
										
										
										
											2009-01-04 17:34:52 -08:00
										 |  |  | 			"Inbound source MAC-address not supported on %s\n", | 
					
						
							| 
									
										
										
										
											2008-12-25 13:39:49 +01:00
										 |  |  | 			QETH_CARD_IFNAME(card)); | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 		return -EOPNOTSUPP; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	rc = qeth_l3_send_simple_setassparms(card, IPA_SOURCE_MAC, | 
					
						
							|  |  |  | 					  IPA_CMD_ASS_START, 0); | 
					
						
							|  |  |  | 	if (rc) | 
					
						
							| 
									
										
										
										
											2008-12-25 13:39:49 +01:00
										 |  |  | 		dev_warn(&card->gdev->dev, | 
					
						
							| 
									
										
										
										
											2009-01-04 17:34:52 -08:00
										 |  |  | 			"Starting source MAC-address support for %s failed\n", | 
					
						
							| 
									
										
										
										
											2008-12-25 13:39:49 +01:00
										 |  |  | 			QETH_CARD_IFNAME(card)); | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 	return rc; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int qeth_l3_start_ipa_vlan(struct qeth_card *card) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	int rc = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-06-21 22:57:05 +00:00
										 |  |  | 	QETH_CARD_TEXT(card, 3, "strtvlan"); | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	if (!qeth_is_supported(card, IPA_FULL_VLAN)) { | 
					
						
							| 
									
										
										
										
											2008-12-25 13:39:49 +01:00
										 |  |  | 		dev_info(&card->gdev->dev, | 
					
						
							|  |  |  | 			"VLAN not supported on %s\n", QETH_CARD_IFNAME(card)); | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 		return -EOPNOTSUPP; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	rc = qeth_l3_send_simple_setassparms(card, IPA_VLAN_PRIO, | 
					
						
							|  |  |  | 					  IPA_CMD_ASS_START, 0); | 
					
						
							|  |  |  | 	if (rc) { | 
					
						
							| 
									
										
										
										
											2008-12-25 13:39:49 +01:00
										 |  |  | 		dev_warn(&card->gdev->dev, | 
					
						
							|  |  |  | 			"Starting VLAN support for %s failed\n", | 
					
						
							|  |  |  | 			QETH_CARD_IFNAME(card)); | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 	} else { | 
					
						
							| 
									
										
										
										
											2008-12-25 13:39:49 +01:00
										 |  |  | 		dev_info(&card->gdev->dev, "VLAN enabled\n"); | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 	} | 
					
						
							|  |  |  | 	return rc; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int qeth_l3_start_ipa_multicast(struct qeth_card *card) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	int rc; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-06-21 22:57:05 +00:00
										 |  |  | 	QETH_CARD_TEXT(card, 3, "stmcast"); | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	if (!qeth_is_supported(card, IPA_MULTICASTING)) { | 
					
						
							| 
									
										
										
										
											2008-12-25 13:39:49 +01:00
										 |  |  | 		dev_info(&card->gdev->dev, | 
					
						
							|  |  |  | 			"Multicast not supported on %s\n", | 
					
						
							|  |  |  | 			QETH_CARD_IFNAME(card)); | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 		return -EOPNOTSUPP; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	rc = qeth_l3_send_simple_setassparms(card, IPA_MULTICASTING, | 
					
						
							|  |  |  | 					  IPA_CMD_ASS_START, 0); | 
					
						
							|  |  |  | 	if (rc) { | 
					
						
							| 
									
										
										
										
											2008-12-25 13:39:49 +01:00
										 |  |  | 		dev_warn(&card->gdev->dev, | 
					
						
							|  |  |  | 			"Starting multicast support for %s failed\n", | 
					
						
							|  |  |  | 			QETH_CARD_IFNAME(card)); | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 	} else { | 
					
						
							| 
									
										
										
										
											2008-12-25 13:39:49 +01:00
										 |  |  | 		dev_info(&card->gdev->dev, "Multicast enabled\n"); | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 		card->dev->flags |= IFF_MULTICAST; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return rc; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int qeth_l3_query_ipassists_cb(struct qeth_card *card, | 
					
						
							|  |  |  | 		struct qeth_reply *reply, unsigned long data) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct qeth_ipa_cmd *cmd; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-04-01 10:26:58 +02:00
										 |  |  | 	QETH_DBF_TEXT(SETUP, 2, "qipasscb"); | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	cmd = (struct qeth_ipa_cmd *) data; | 
					
						
							|  |  |  | 	if (cmd->hdr.prot_version == QETH_PROT_IPV4) { | 
					
						
							|  |  |  | 		card->options.ipa4.supported_funcs = cmd->hdr.ipa_supported; | 
					
						
							|  |  |  | 		card->options.ipa4.enabled_funcs = cmd->hdr.ipa_enabled; | 
					
						
							|  |  |  | 	} else { | 
					
						
							|  |  |  | 		card->options.ipa6.supported_funcs = cmd->hdr.ipa_supported; | 
					
						
							|  |  |  | 		card->options.ipa6.enabled_funcs = cmd->hdr.ipa_enabled; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2008-04-01 10:26:58 +02:00
										 |  |  | 	QETH_DBF_TEXT(SETUP, 2, "suppenbl"); | 
					
						
							|  |  |  | 	QETH_DBF_TEXT_(SETUP, 2, "%x", cmd->hdr.ipa_supported); | 
					
						
							|  |  |  | 	QETH_DBF_TEXT_(SETUP, 2, "%x", cmd->hdr.ipa_enabled); | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 	return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int qeth_l3_query_ipassists(struct qeth_card *card, | 
					
						
							|  |  |  | 				enum qeth_prot_versions prot) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	int rc; | 
					
						
							|  |  |  | 	struct qeth_cmd_buffer *iob; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-04-01 10:26:58 +02:00
										 |  |  | 	QETH_DBF_TEXT_(SETUP, 2, "qipassi%i", prot); | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 	iob = qeth_get_ipacmd_buffer(card, IPA_CMD_QIPASSIST, prot); | 
					
						
							|  |  |  | 	rc = qeth_send_ipa_cmd(card, iob, qeth_l3_query_ipassists_cb, NULL); | 
					
						
							|  |  |  | 	return rc; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #ifdef CONFIG_QETH_IPV6
 | 
					
						
							|  |  |  | static int qeth_l3_softsetup_ipv6(struct qeth_card *card) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	int rc; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-06-21 22:57:05 +00:00
										 |  |  | 	QETH_CARD_TEXT(card, 3, "softipv6"); | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	if (card->info.type == QETH_CARD_TYPE_IQD) | 
					
						
							|  |  |  | 		goto out; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	rc = qeth_l3_query_ipassists(card, QETH_PROT_IPV6); | 
					
						
							|  |  |  | 	if (rc) { | 
					
						
							| 
									
										
										
										
											2008-12-25 13:39:49 +01:00
										 |  |  | 		dev_err(&card->gdev->dev, | 
					
						
							|  |  |  | 			"Activating IPv6 support for %s failed\n", | 
					
						
							|  |  |  | 			QETH_CARD_IFNAME(card)); | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 		return rc; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	rc = qeth_l3_send_simple_setassparms(card, IPA_IPV6, | 
					
						
							|  |  |  | 					  IPA_CMD_ASS_START, 3); | 
					
						
							|  |  |  | 	if (rc) { | 
					
						
							| 
									
										
										
										
											2008-12-25 13:39:49 +01:00
										 |  |  | 		dev_err(&card->gdev->dev, | 
					
						
							|  |  |  | 			"Activating IPv6 support for %s failed\n", | 
					
						
							|  |  |  | 			QETH_CARD_IFNAME(card)); | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 		return rc; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	rc = qeth_l3_send_simple_setassparms_ipv6(card, IPA_IPV6, | 
					
						
							|  |  |  | 					       IPA_CMD_ASS_START); | 
					
						
							|  |  |  | 	if (rc) { | 
					
						
							| 
									
										
										
										
											2008-12-25 13:39:49 +01:00
										 |  |  | 		dev_err(&card->gdev->dev, | 
					
						
							|  |  |  | 			"Activating IPv6 support for %s failed\n", | 
					
						
							|  |  |  | 			 QETH_CARD_IFNAME(card)); | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 		return rc; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	rc = qeth_l3_send_simple_setassparms_ipv6(card, IPA_PASSTHRU, | 
					
						
							|  |  |  | 					       IPA_CMD_ASS_START); | 
					
						
							|  |  |  | 	if (rc) { | 
					
						
							| 
									
										
										
										
											2008-12-25 13:39:49 +01:00
										 |  |  | 		dev_warn(&card->gdev->dev, | 
					
						
							|  |  |  | 			"Enabling the passthrough mode for %s failed\n", | 
					
						
							|  |  |  | 			QETH_CARD_IFNAME(card)); | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 		return rc; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | out: | 
					
						
							| 
									
										
										
										
											2008-12-25 13:39:49 +01:00
										 |  |  | 	dev_info(&card->gdev->dev, "IPV6 enabled\n"); | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 	return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int qeth_l3_start_ipa_ipv6(struct qeth_card *card) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	int rc = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-06-21 22:57:05 +00:00
										 |  |  | 	QETH_CARD_TEXT(card, 3, "strtipv6"); | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	if (!qeth_is_supported(card, IPA_IPV6)) { | 
					
						
							| 
									
										
										
										
											2008-12-25 13:39:49 +01:00
										 |  |  | 		dev_info(&card->gdev->dev, | 
					
						
							|  |  |  | 			"IPv6 not supported on %s\n", QETH_CARD_IFNAME(card)); | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 		return 0; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | #ifdef CONFIG_QETH_IPV6
 | 
					
						
							|  |  |  | 	rc = qeth_l3_softsetup_ipv6(card); | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 	return rc ; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int qeth_l3_start_ipa_broadcast(struct qeth_card *card) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	int rc; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-06-21 22:57:05 +00:00
										 |  |  | 	QETH_CARD_TEXT(card, 3, "stbrdcst"); | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 	card->info.broadcast_capable = 0; | 
					
						
							|  |  |  | 	if (!qeth_is_supported(card, IPA_FILTERING)) { | 
					
						
							| 
									
										
										
										
											2008-12-25 13:39:49 +01:00
										 |  |  | 		dev_info(&card->gdev->dev, | 
					
						
							|  |  |  | 			"Broadcast not supported on %s\n", | 
					
						
							|  |  |  | 			QETH_CARD_IFNAME(card)); | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 		rc = -EOPNOTSUPP; | 
					
						
							|  |  |  | 		goto out; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	rc = qeth_l3_send_simple_setassparms(card, IPA_FILTERING, | 
					
						
							|  |  |  | 					  IPA_CMD_ASS_START, 0); | 
					
						
							|  |  |  | 	if (rc) { | 
					
						
							| 
									
										
										
										
											2008-12-25 13:39:49 +01:00
										 |  |  | 		dev_warn(&card->gdev->dev, "Enabling broadcast filtering for " | 
					
						
							|  |  |  | 			"%s failed\n", QETH_CARD_IFNAME(card)); | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 		goto out; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	rc = qeth_l3_send_simple_setassparms(card, IPA_FILTERING, | 
					
						
							|  |  |  | 					  IPA_CMD_ASS_CONFIGURE, 1); | 
					
						
							|  |  |  | 	if (rc) { | 
					
						
							| 
									
										
										
										
											2008-12-25 13:39:49 +01:00
										 |  |  | 		dev_warn(&card->gdev->dev, | 
					
						
							|  |  |  | 			"Setting up broadcast filtering for %s failed\n", | 
					
						
							|  |  |  | 			QETH_CARD_IFNAME(card)); | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 		goto out; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	card->info.broadcast_capable = QETH_BROADCAST_WITH_ECHO; | 
					
						
							| 
									
										
										
										
											2008-12-25 13:39:49 +01:00
										 |  |  | 	dev_info(&card->gdev->dev, "Broadcast enabled\n"); | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 	rc = qeth_l3_send_simple_setassparms(card, IPA_FILTERING, | 
					
						
							|  |  |  | 					  IPA_CMD_ASS_ENABLE, 1); | 
					
						
							|  |  |  | 	if (rc) { | 
					
						
							| 
									
										
										
										
											2008-12-25 13:39:49 +01:00
										 |  |  | 		dev_warn(&card->gdev->dev, "Setting up broadcast echo " | 
					
						
							|  |  |  | 			"filtering for %s failed\n", QETH_CARD_IFNAME(card)); | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 		goto out; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	card->info.broadcast_capable = QETH_BROADCAST_WITHOUT_ECHO; | 
					
						
							|  |  |  | out: | 
					
						
							|  |  |  | 	if (card->info.broadcast_capable) | 
					
						
							|  |  |  | 		card->dev->flags |= IFF_BROADCAST; | 
					
						
							|  |  |  | 	else | 
					
						
							|  |  |  | 		card->dev->flags &= ~IFF_BROADCAST; | 
					
						
							|  |  |  | 	return rc; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int qeth_l3_send_checksum_command(struct qeth_card *card) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	int rc; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	rc = qeth_l3_send_simple_setassparms(card, IPA_INBOUND_CHECKSUM, | 
					
						
							|  |  |  | 					  IPA_CMD_ASS_START, 0); | 
					
						
							|  |  |  | 	if (rc) { | 
					
						
							| 
									
										
										
										
											2008-12-25 13:39:49 +01:00
										 |  |  | 		dev_warn(&card->gdev->dev, "Starting HW checksumming for %s " | 
					
						
							|  |  |  | 			"failed, using SW checksumming\n", | 
					
						
							|  |  |  | 			QETH_CARD_IFNAME(card)); | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 		return rc; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	rc = qeth_l3_send_simple_setassparms(card, IPA_INBOUND_CHECKSUM, | 
					
						
							|  |  |  | 					  IPA_CMD_ASS_ENABLE, | 
					
						
							|  |  |  | 					  card->info.csum_mask); | 
					
						
							|  |  |  | 	if (rc) { | 
					
						
							| 
									
										
										
										
											2008-12-25 13:39:49 +01:00
										 |  |  | 		dev_warn(&card->gdev->dev, "Enabling HW checksumming for %s " | 
					
						
							|  |  |  | 			"failed, using SW checksumming\n", | 
					
						
							|  |  |  | 			QETH_CARD_IFNAME(card)); | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 		return rc; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-11-12 00:11:45 +00:00
										 |  |  | int qeth_l3_set_rx_csum(struct qeth_card *card, | 
					
						
							|  |  |  | 	enum qeth_checksum_types csum_type) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	int rc = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (card->options.checksum_type == HW_CHECKSUMMING) { | 
					
						
							|  |  |  | 		if ((csum_type != HW_CHECKSUMMING) && | 
					
						
							|  |  |  | 			(card->state != CARD_STATE_DOWN)) { | 
					
						
							|  |  |  | 			rc = qeth_l3_send_simple_setassparms(card, | 
					
						
							|  |  |  | 				IPA_INBOUND_CHECKSUM, IPA_CMD_ASS_STOP, 0); | 
					
						
							|  |  |  | 			if (rc) | 
					
						
							|  |  |  | 				return -EIO; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} else { | 
					
						
							|  |  |  | 		if (csum_type == HW_CHECKSUMMING) { | 
					
						
							|  |  |  | 			if (card->state != CARD_STATE_DOWN) { | 
					
						
							|  |  |  | 				if (!qeth_is_supported(card, | 
					
						
							|  |  |  | 				    IPA_INBOUND_CHECKSUM)) | 
					
						
							|  |  |  | 					return -EPERM; | 
					
						
							|  |  |  | 				rc = qeth_l3_send_checksum_command(card); | 
					
						
							|  |  |  | 				if (rc) | 
					
						
							|  |  |  | 					return -EIO; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	card->options.checksum_type = csum_type; | 
					
						
							|  |  |  | 	return rc; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | static int qeth_l3_start_ipa_checksum(struct qeth_card *card) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	int rc = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-06-21 22:57:05 +00:00
										 |  |  | 	QETH_CARD_TEXT(card, 3, "strtcsum"); | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	if (card->options.checksum_type == NO_CHECKSUMMING) { | 
					
						
							| 
									
										
										
										
											2008-12-25 13:39:49 +01:00
										 |  |  | 		dev_info(&card->gdev->dev, | 
					
						
							|  |  |  | 			"Using no checksumming on %s.\n", | 
					
						
							|  |  |  | 			QETH_CARD_IFNAME(card)); | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 		return 0; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	if (card->options.checksum_type == SW_CHECKSUMMING) { | 
					
						
							| 
									
										
										
										
											2008-12-25 13:39:49 +01:00
										 |  |  | 		dev_info(&card->gdev->dev, | 
					
						
							|  |  |  | 			"Using SW checksumming on %s.\n", | 
					
						
							|  |  |  | 			QETH_CARD_IFNAME(card)); | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 		return 0; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	if (!qeth_is_supported(card, IPA_INBOUND_CHECKSUM)) { | 
					
						
							| 
									
										
										
										
											2008-12-25 13:39:49 +01:00
										 |  |  | 		dev_info(&card->gdev->dev, | 
					
						
							|  |  |  | 			"Inbound HW Checksumming not " | 
					
						
							|  |  |  | 			"supported on %s,\ncontinuing " | 
					
						
							|  |  |  | 			"using Inbound SW Checksumming\n", | 
					
						
							|  |  |  | 			QETH_CARD_IFNAME(card)); | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 		card->options.checksum_type = SW_CHECKSUMMING; | 
					
						
							|  |  |  | 		return 0; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	rc = qeth_l3_send_checksum_command(card); | 
					
						
							|  |  |  | 	if (!rc) | 
					
						
							| 
									
										
										
										
											2008-12-25 13:39:49 +01:00
										 |  |  | 		dev_info(&card->gdev->dev, | 
					
						
							|  |  |  | 			"HW Checksumming (inbound) enabled\n"); | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	return rc; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-05-11 19:34:45 +00:00
										 |  |  | static int qeth_l3_start_ipa_tx_checksum(struct qeth_card *card) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	int rc = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (!qeth_is_supported(card, IPA_OUTBOUND_CHECKSUM)) | 
					
						
							|  |  |  | 		return rc; | 
					
						
							|  |  |  | 	rc = qeth_l3_send_simple_setassparms(card, IPA_OUTBOUND_CHECKSUM, | 
					
						
							|  |  |  | 			  IPA_CMD_ASS_START, 0); | 
					
						
							|  |  |  | 	if (rc) | 
					
						
							|  |  |  | 		goto err_out; | 
					
						
							|  |  |  | 	rc = qeth_l3_send_simple_setassparms(card, IPA_OUTBOUND_CHECKSUM, | 
					
						
							|  |  |  | 			  IPA_CMD_ASS_ENABLE, card->info.tx_csum_mask); | 
					
						
							|  |  |  | 	if (rc) | 
					
						
							|  |  |  | 		goto err_out; | 
					
						
							|  |  |  | 	dev_info(&card->gdev->dev, "HW TX Checksumming enabled\n"); | 
					
						
							|  |  |  | 	return rc; | 
					
						
							|  |  |  | err_out: | 
					
						
							|  |  |  | 	dev_warn(&card->gdev->dev, "Enabling HW TX checksumming for %s " | 
					
						
							|  |  |  | 		"failed, using SW TX checksumming\n", QETH_CARD_IFNAME(card)); | 
					
						
							|  |  |  | 	return rc; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | static int qeth_l3_start_ipa_tso(struct qeth_card *card) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	int rc; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-06-21 22:57:05 +00:00
										 |  |  | 	QETH_CARD_TEXT(card, 3, "sttso"); | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	if (!qeth_is_supported(card, IPA_OUTBOUND_TSO)) { | 
					
						
							| 
									
										
										
										
											2008-12-25 13:39:49 +01:00
										 |  |  | 		dev_info(&card->gdev->dev, | 
					
						
							|  |  |  | 			"Outbound TSO not supported on %s\n", | 
					
						
							|  |  |  | 			QETH_CARD_IFNAME(card)); | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 		rc = -EOPNOTSUPP; | 
					
						
							|  |  |  | 	} else { | 
					
						
							|  |  |  | 		rc = qeth_l3_send_simple_setassparms(card, IPA_OUTBOUND_TSO, | 
					
						
							|  |  |  | 						IPA_CMD_ASS_START, 0); | 
					
						
							|  |  |  | 		if (rc) | 
					
						
							| 
									
										
										
										
											2008-12-25 13:39:49 +01:00
										 |  |  | 			dev_warn(&card->gdev->dev, "Starting outbound TCP " | 
					
						
							|  |  |  | 				"segmentation offload for %s failed\n", | 
					
						
							|  |  |  | 				QETH_CARD_IFNAME(card)); | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 		else | 
					
						
							| 
									
										
										
										
											2008-12-25 13:39:49 +01:00
										 |  |  | 			dev_info(&card->gdev->dev, | 
					
						
							|  |  |  | 				"Outbound TSO enabled\n"); | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 	} | 
					
						
							|  |  |  | 	if (rc && (card->options.large_send == QETH_LARGE_SEND_TSO)) { | 
					
						
							|  |  |  | 		card->options.large_send = QETH_LARGE_SEND_NO; | 
					
						
							|  |  |  | 		card->dev->features &= ~(NETIF_F_TSO | NETIF_F_SG); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return rc; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int qeth_l3_start_ipassists(struct qeth_card *card) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2010-06-21 22:57:05 +00:00
										 |  |  | 	QETH_CARD_TEXT(card, 3, "strtipas"); | 
					
						
							| 
									
										
										
										
											2009-11-12 00:11:41 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	qeth_set_access_ctrl_online(card);	/* go on*/ | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 	qeth_l3_start_ipa_arp_processing(card);	/* go on*/ | 
					
						
							|  |  |  | 	qeth_l3_start_ipa_ip_fragmentation(card);	/* go on*/ | 
					
						
							|  |  |  | 	qeth_l3_start_ipa_source_mac(card);	/* go on*/ | 
					
						
							|  |  |  | 	qeth_l3_start_ipa_vlan(card);		/* go on*/ | 
					
						
							|  |  |  | 	qeth_l3_start_ipa_multicast(card);		/* go on*/ | 
					
						
							|  |  |  | 	qeth_l3_start_ipa_ipv6(card);		/* go on*/ | 
					
						
							|  |  |  | 	qeth_l3_start_ipa_broadcast(card);		/* go on*/ | 
					
						
							|  |  |  | 	qeth_l3_start_ipa_checksum(card);		/* go on*/ | 
					
						
							| 
									
										
										
										
											2010-05-11 19:34:45 +00:00
										 |  |  | 	qeth_l3_start_ipa_tx_checksum(card); | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 	qeth_l3_start_ipa_tso(card);		/* go on*/ | 
					
						
							|  |  |  | 	return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int qeth_l3_put_unique_id(struct qeth_card *card) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	int rc = 0; | 
					
						
							|  |  |  | 	struct qeth_cmd_buffer *iob; | 
					
						
							|  |  |  | 	struct qeth_ipa_cmd *cmd; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-06-21 22:57:05 +00:00
										 |  |  | 	QETH_CARD_TEXT(card, 2, "puniqeid"); | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	if ((card->info.unique_id & UNIQUE_ID_NOT_BY_CARD) == | 
					
						
							|  |  |  | 		UNIQUE_ID_NOT_BY_CARD) | 
					
						
							|  |  |  | 		return -1; | 
					
						
							|  |  |  | 	iob = qeth_get_ipacmd_buffer(card, IPA_CMD_DESTROY_ADDR, | 
					
						
							|  |  |  | 				     QETH_PROT_IPV6); | 
					
						
							|  |  |  | 	cmd = (struct qeth_ipa_cmd *)(iob->data+IPA_PDU_HEADER_SIZE); | 
					
						
							|  |  |  | 	*((__u16 *) &cmd->data.create_destroy_addr.unique_id[6]) = | 
					
						
							|  |  |  | 				card->info.unique_id; | 
					
						
							|  |  |  | 	memcpy(&cmd->data.create_destroy_addr.unique_id[0], | 
					
						
							|  |  |  | 	       card->dev->dev_addr, OSA_ADDR_LEN); | 
					
						
							|  |  |  | 	rc = qeth_send_ipa_cmd(card, iob, NULL, NULL); | 
					
						
							|  |  |  | 	return rc; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int qeth_l3_iqd_read_initial_mac_cb(struct qeth_card *card, | 
					
						
							|  |  |  | 		struct qeth_reply *reply, unsigned long data) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct qeth_ipa_cmd *cmd; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	cmd = (struct qeth_ipa_cmd *) data; | 
					
						
							|  |  |  | 	if (cmd->hdr.return_code == 0) | 
					
						
							|  |  |  | 		memcpy(card->dev->dev_addr, | 
					
						
							|  |  |  | 			cmd->data.create_destroy_addr.unique_id, ETH_ALEN); | 
					
						
							|  |  |  | 	else | 
					
						
							|  |  |  | 		random_ether_addr(card->dev->dev_addr); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int qeth_l3_iqd_read_initial_mac(struct qeth_card *card) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	int rc = 0; | 
					
						
							|  |  |  | 	struct qeth_cmd_buffer *iob; | 
					
						
							|  |  |  | 	struct qeth_ipa_cmd *cmd; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-04-01 10:26:58 +02:00
										 |  |  | 	QETH_DBF_TEXT(SETUP, 2, "hsrmac"); | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	iob = qeth_get_ipacmd_buffer(card, IPA_CMD_CREATE_ADDR, | 
					
						
							|  |  |  | 				     QETH_PROT_IPV6); | 
					
						
							|  |  |  | 	cmd = (struct qeth_ipa_cmd *)(iob->data+IPA_PDU_HEADER_SIZE); | 
					
						
							|  |  |  | 	*((__u16 *) &cmd->data.create_destroy_addr.unique_id[6]) = | 
					
						
							|  |  |  | 			card->info.unique_id; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	rc = qeth_send_ipa_cmd(card, iob, qeth_l3_iqd_read_initial_mac_cb, | 
					
						
							|  |  |  | 				NULL); | 
					
						
							|  |  |  | 	return rc; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int qeth_l3_get_unique_id_cb(struct qeth_card *card, | 
					
						
							|  |  |  | 		struct qeth_reply *reply, unsigned long data) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct qeth_ipa_cmd *cmd; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	cmd = (struct qeth_ipa_cmd *) data; | 
					
						
							|  |  |  | 	if (cmd->hdr.return_code == 0) | 
					
						
							|  |  |  | 		card->info.unique_id = *((__u16 *) | 
					
						
							|  |  |  | 				&cmd->data.create_destroy_addr.unique_id[6]); | 
					
						
							|  |  |  | 	else { | 
					
						
							|  |  |  | 		card->info.unique_id =  UNIQUE_ID_IF_CREATE_ADDR_FAILED | | 
					
						
							|  |  |  | 					UNIQUE_ID_NOT_BY_CARD; | 
					
						
							| 
									
										
										
										
											2008-12-25 13:39:49 +01:00
										 |  |  | 		dev_warn(&card->gdev->dev, "The network adapter failed to " | 
					
						
							|  |  |  | 			"generate a unique ID\n"); | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 	} | 
					
						
							|  |  |  | 	return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int qeth_l3_get_unique_id(struct qeth_card *card) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	int rc = 0; | 
					
						
							|  |  |  | 	struct qeth_cmd_buffer *iob; | 
					
						
							|  |  |  | 	struct qeth_ipa_cmd *cmd; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-04-01 10:26:58 +02:00
										 |  |  | 	QETH_DBF_TEXT(SETUP, 2, "guniqeid"); | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	if (!qeth_is_supported(card, IPA_IPV6)) { | 
					
						
							|  |  |  | 		card->info.unique_id =  UNIQUE_ID_IF_CREATE_ADDR_FAILED | | 
					
						
							|  |  |  | 					UNIQUE_ID_NOT_BY_CARD; | 
					
						
							|  |  |  | 		return 0; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	iob = qeth_get_ipacmd_buffer(card, IPA_CMD_CREATE_ADDR, | 
					
						
							|  |  |  | 				     QETH_PROT_IPV6); | 
					
						
							|  |  |  | 	cmd = (struct qeth_ipa_cmd *)(iob->data+IPA_PDU_HEADER_SIZE); | 
					
						
							|  |  |  | 	*((__u16 *) &cmd->data.create_destroy_addr.unique_id[6]) = | 
					
						
							|  |  |  | 			card->info.unique_id; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	rc = qeth_send_ipa_cmd(card, iob, qeth_l3_get_unique_id_cb, NULL); | 
					
						
							|  |  |  | 	return rc; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-01-11 02:50:50 +00:00
										 |  |  | static int | 
					
						
							|  |  |  | qeth_diags_trace_cb(struct qeth_card *card, struct qeth_reply *reply, | 
					
						
							|  |  |  | 			    unsigned long data) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct qeth_ipa_cmd	   *cmd; | 
					
						
							|  |  |  | 	__u16 rc; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	QETH_DBF_TEXT(SETUP, 2, "diastrcb"); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	cmd = (struct qeth_ipa_cmd *)data; | 
					
						
							|  |  |  | 	rc = cmd->hdr.return_code; | 
					
						
							| 
									
										
										
										
											2010-03-08 20:36:55 +00:00
										 |  |  | 	if (rc) | 
					
						
							| 
									
										
										
										
											2010-06-21 22:57:05 +00:00
										 |  |  | 		QETH_CARD_TEXT_(card, 2, "dxter%x", rc); | 
					
						
							| 
									
										
										
										
											2010-01-11 02:50:50 +00:00
										 |  |  | 	switch (cmd->data.diagass.action) { | 
					
						
							|  |  |  | 	case QETH_DIAGS_CMD_TRACE_QUERY: | 
					
						
							|  |  |  | 		break; | 
					
						
							|  |  |  | 	case QETH_DIAGS_CMD_TRACE_DISABLE: | 
					
						
							| 
									
										
										
										
											2010-03-08 20:36:55 +00:00
										 |  |  | 		switch (rc) { | 
					
						
							|  |  |  | 		case 0: | 
					
						
							|  |  |  | 		case IPA_RC_INVALID_SUBCMD: | 
					
						
							|  |  |  | 			card->info.promisc_mode = SET_PROMISC_MODE_OFF; | 
					
						
							|  |  |  | 			dev_info(&card->gdev->dev, "The HiperSockets network " | 
					
						
							|  |  |  | 				"traffic analyzer is deactivated\n"); | 
					
						
							|  |  |  | 			break; | 
					
						
							|  |  |  | 		default: | 
					
						
							|  |  |  | 			break; | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2010-01-11 02:50:50 +00:00
										 |  |  | 		break; | 
					
						
							|  |  |  | 	case QETH_DIAGS_CMD_TRACE_ENABLE: | 
					
						
							| 
									
										
										
										
											2010-03-08 20:36:55 +00:00
										 |  |  | 		switch (rc) { | 
					
						
							|  |  |  | 		case 0: | 
					
						
							|  |  |  | 			card->info.promisc_mode = SET_PROMISC_MODE_ON; | 
					
						
							|  |  |  | 			dev_info(&card->gdev->dev, "The HiperSockets network " | 
					
						
							|  |  |  | 				"traffic analyzer is activated\n"); | 
					
						
							|  |  |  | 			break; | 
					
						
							|  |  |  | 		case IPA_RC_HARDWARE_AUTH_ERROR: | 
					
						
							|  |  |  | 			dev_warn(&card->gdev->dev, "The device is not " | 
					
						
							|  |  |  | 				"authorized to run as a HiperSockets network " | 
					
						
							|  |  |  | 				"traffic analyzer\n"); | 
					
						
							|  |  |  | 			break; | 
					
						
							|  |  |  | 		case IPA_RC_TRACE_ALREADY_ACTIVE: | 
					
						
							|  |  |  | 			dev_warn(&card->gdev->dev, "A HiperSockets " | 
					
						
							|  |  |  | 				"network traffic analyzer is already " | 
					
						
							|  |  |  | 				"active in the HiperSockets LAN\n"); | 
					
						
							|  |  |  | 			break; | 
					
						
							|  |  |  | 		default: | 
					
						
							|  |  |  | 			break; | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2010-01-11 02:50:50 +00:00
										 |  |  | 		break; | 
					
						
							|  |  |  | 	default: | 
					
						
							|  |  |  | 		QETH_DBF_MESSAGE(2, "Unknown sniffer action (0x%04x) on %s\n", | 
					
						
							|  |  |  | 			cmd->data.diagass.action, QETH_CARD_IFNAME(card)); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int | 
					
						
							|  |  |  | qeth_diags_trace(struct qeth_card *card, enum qeth_diags_trace_cmds diags_cmd) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct qeth_cmd_buffer *iob; | 
					
						
							|  |  |  | 	struct qeth_ipa_cmd    *cmd; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	QETH_DBF_TEXT(SETUP, 2, "diagtrac"); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	iob = qeth_get_ipacmd_buffer(card, IPA_CMD_SET_DIAG_ASS, 0); | 
					
						
							|  |  |  | 	cmd = (struct qeth_ipa_cmd *)(iob->data+IPA_PDU_HEADER_SIZE); | 
					
						
							|  |  |  | 	cmd->data.diagass.subcmd_len = 16; | 
					
						
							|  |  |  | 	cmd->data.diagass.subcmd = QETH_DIAGS_CMD_TRACE; | 
					
						
							|  |  |  | 	cmd->data.diagass.type = QETH_DIAGS_TYPE_HIPERSOCKET; | 
					
						
							|  |  |  | 	cmd->data.diagass.action = diags_cmd; | 
					
						
							|  |  |  | 	return qeth_send_ipa_cmd(card, iob, qeth_diags_trace_cb, NULL); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | static void qeth_l3_get_mac_for_ipm(__u32 ipm, char *mac, | 
					
						
							|  |  |  | 				struct net_device *dev) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	if (dev->type == ARPHRD_IEEE802_TR) | 
					
						
							|  |  |  | 		ip_tr_mc_map(ipm, mac); | 
					
						
							|  |  |  | 	else | 
					
						
							|  |  |  | 		ip_eth_mc_map(ipm, mac); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void qeth_l3_add_mc(struct qeth_card *card, struct in_device *in4_dev) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct qeth_ipaddr *ipm; | 
					
						
							|  |  |  | 	struct ip_mc_list *im4; | 
					
						
							|  |  |  | 	char buf[MAX_ADDR_LEN]; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-06-21 22:57:05 +00:00
										 |  |  | 	QETH_CARD_TEXT(card, 4, "addmc"); | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 	for (im4 = in4_dev->mc_list; im4; im4 = im4->next) { | 
					
						
							|  |  |  | 		qeth_l3_get_mac_for_ipm(im4->multiaddr, buf, in4_dev->dev); | 
					
						
							|  |  |  | 		ipm = qeth_l3_get_addr_buffer(QETH_PROT_IPV4); | 
					
						
							|  |  |  | 		if (!ipm) | 
					
						
							|  |  |  | 			continue; | 
					
						
							|  |  |  | 		ipm->u.a4.addr = im4->multiaddr; | 
					
						
							|  |  |  | 		memcpy(ipm->mac, buf, OSA_ADDR_LEN); | 
					
						
							|  |  |  | 		ipm->is_multicast = 1; | 
					
						
							|  |  |  | 		if (!qeth_l3_add_ip(card, ipm)) | 
					
						
							|  |  |  | 			kfree(ipm); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void qeth_l3_add_vlan_mc(struct qeth_card *card) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct in_device *in_dev; | 
					
						
							|  |  |  | 	struct vlan_group *vg; | 
					
						
							|  |  |  | 	int i; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-06-21 22:57:05 +00:00
										 |  |  | 	QETH_CARD_TEXT(card, 4, "addmcvl"); | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 	if (!qeth_is_supported(card, IPA_FULL_VLAN) || (card->vlangrp == NULL)) | 
					
						
							|  |  |  | 		return; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	vg = card->vlangrp; | 
					
						
							|  |  |  | 	for (i = 0; i < VLAN_GROUP_ARRAY_LEN; i++) { | 
					
						
							|  |  |  | 		struct net_device *netdev = vlan_group_get_device(vg, i); | 
					
						
							|  |  |  | 		if (netdev == NULL || | 
					
						
							|  |  |  | 		    !(netdev->flags & IFF_UP)) | 
					
						
							|  |  |  | 			continue; | 
					
						
							|  |  |  | 		in_dev = in_dev_get(netdev); | 
					
						
							|  |  |  | 		if (!in_dev) | 
					
						
							|  |  |  | 			continue; | 
					
						
							|  |  |  | 		read_lock(&in_dev->mc_list_lock); | 
					
						
							|  |  |  | 		qeth_l3_add_mc(card, in_dev); | 
					
						
							|  |  |  | 		read_unlock(&in_dev->mc_list_lock); | 
					
						
							|  |  |  | 		in_dev_put(in_dev); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void qeth_l3_add_multicast_ipv4(struct qeth_card *card) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct in_device *in4_dev; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-06-21 22:57:05 +00:00
										 |  |  | 	QETH_CARD_TEXT(card, 4, "chkmcv4"); | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 	in4_dev = in_dev_get(card->dev); | 
					
						
							|  |  |  | 	if (in4_dev == NULL) | 
					
						
							|  |  |  | 		return; | 
					
						
							|  |  |  | 	read_lock(&in4_dev->mc_list_lock); | 
					
						
							|  |  |  | 	qeth_l3_add_mc(card, in4_dev); | 
					
						
							|  |  |  | 	qeth_l3_add_vlan_mc(card); | 
					
						
							|  |  |  | 	read_unlock(&in4_dev->mc_list_lock); | 
					
						
							|  |  |  | 	in_dev_put(in4_dev); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #ifdef CONFIG_QETH_IPV6
 | 
					
						
							|  |  |  | static void qeth_l3_add_mc6(struct qeth_card *card, struct inet6_dev *in6_dev) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct qeth_ipaddr *ipm; | 
					
						
							|  |  |  | 	struct ifmcaddr6 *im6; | 
					
						
							|  |  |  | 	char buf[MAX_ADDR_LEN]; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-06-21 22:57:05 +00:00
										 |  |  | 	QETH_CARD_TEXT(card, 4, "addmc6"); | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 	for (im6 = in6_dev->mc_list; im6 != NULL; im6 = im6->next) { | 
					
						
							|  |  |  | 		ndisc_mc_map(&im6->mca_addr, buf, in6_dev->dev, 0); | 
					
						
							|  |  |  | 		ipm = qeth_l3_get_addr_buffer(QETH_PROT_IPV6); | 
					
						
							|  |  |  | 		if (!ipm) | 
					
						
							|  |  |  | 			continue; | 
					
						
							|  |  |  | 		ipm->is_multicast = 1; | 
					
						
							|  |  |  | 		memcpy(ipm->mac, buf, OSA_ADDR_LEN); | 
					
						
							|  |  |  | 		memcpy(&ipm->u.a6.addr, &im6->mca_addr.s6_addr, | 
					
						
							|  |  |  | 		       sizeof(struct in6_addr)); | 
					
						
							|  |  |  | 		if (!qeth_l3_add_ip(card, ipm)) | 
					
						
							|  |  |  | 			kfree(ipm); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void qeth_l3_add_vlan_mc6(struct qeth_card *card) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct inet6_dev *in_dev; | 
					
						
							|  |  |  | 	struct vlan_group *vg; | 
					
						
							|  |  |  | 	int i; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-06-21 22:57:05 +00:00
										 |  |  | 	QETH_CARD_TEXT(card, 4, "admc6vl"); | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 	if (!qeth_is_supported(card, IPA_FULL_VLAN) || (card->vlangrp == NULL)) | 
					
						
							|  |  |  | 		return; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	vg = card->vlangrp; | 
					
						
							|  |  |  | 	for (i = 0; i < VLAN_GROUP_ARRAY_LEN; i++) { | 
					
						
							|  |  |  | 		struct net_device *netdev = vlan_group_get_device(vg, i); | 
					
						
							|  |  |  | 		if (netdev == NULL || | 
					
						
							|  |  |  | 		    !(netdev->flags & IFF_UP)) | 
					
						
							|  |  |  | 			continue; | 
					
						
							|  |  |  | 		in_dev = in6_dev_get(netdev); | 
					
						
							|  |  |  | 		if (!in_dev) | 
					
						
							|  |  |  | 			continue; | 
					
						
							|  |  |  | 		read_lock_bh(&in_dev->lock); | 
					
						
							|  |  |  | 		qeth_l3_add_mc6(card, in_dev); | 
					
						
							|  |  |  | 		read_unlock_bh(&in_dev->lock); | 
					
						
							|  |  |  | 		in6_dev_put(in_dev); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void qeth_l3_add_multicast_ipv6(struct qeth_card *card) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct inet6_dev *in6_dev; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-06-21 22:57:05 +00:00
										 |  |  | 	QETH_CARD_TEXT(card, 4, "chkmcv6"); | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 	if (!qeth_is_supported(card, IPA_IPV6)) | 
					
						
							|  |  |  | 		return ; | 
					
						
							|  |  |  | 	in6_dev = in6_dev_get(card->dev); | 
					
						
							|  |  |  | 	if (in6_dev == NULL) | 
					
						
							|  |  |  | 		return; | 
					
						
							|  |  |  | 	read_lock_bh(&in6_dev->lock); | 
					
						
							|  |  |  | 	qeth_l3_add_mc6(card, in6_dev); | 
					
						
							|  |  |  | 	qeth_l3_add_vlan_mc6(card); | 
					
						
							|  |  |  | 	read_unlock_bh(&in6_dev->lock); | 
					
						
							|  |  |  | 	in6_dev_put(in6_dev); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | #endif /* CONFIG_QETH_IPV6 */
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void qeth_l3_free_vlan_addresses4(struct qeth_card *card, | 
					
						
							|  |  |  | 			unsigned short vid) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct in_device *in_dev; | 
					
						
							|  |  |  | 	struct in_ifaddr *ifa; | 
					
						
							|  |  |  | 	struct qeth_ipaddr *addr; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-06-21 22:57:05 +00:00
										 |  |  | 	QETH_CARD_TEXT(card, 4, "frvaddr4"); | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	in_dev = in_dev_get(vlan_group_get_device(card->vlangrp, vid)); | 
					
						
							|  |  |  | 	if (!in_dev) | 
					
						
							|  |  |  | 		return; | 
					
						
							|  |  |  | 	for (ifa = in_dev->ifa_list; ifa; ifa = ifa->ifa_next) { | 
					
						
							|  |  |  | 		addr = qeth_l3_get_addr_buffer(QETH_PROT_IPV4); | 
					
						
							|  |  |  | 		if (addr) { | 
					
						
							|  |  |  | 			addr->u.a4.addr = ifa->ifa_address; | 
					
						
							|  |  |  | 			addr->u.a4.mask = ifa->ifa_mask; | 
					
						
							|  |  |  | 			addr->type = QETH_IP_TYPE_NORMAL; | 
					
						
							|  |  |  | 			if (!qeth_l3_delete_ip(card, addr)) | 
					
						
							|  |  |  | 				kfree(addr); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	in_dev_put(in_dev); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void qeth_l3_free_vlan_addresses6(struct qeth_card *card, | 
					
						
							|  |  |  | 			unsigned short vid) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | #ifdef CONFIG_QETH_IPV6
 | 
					
						
							|  |  |  | 	struct inet6_dev *in6_dev; | 
					
						
							|  |  |  | 	struct inet6_ifaddr *ifa; | 
					
						
							|  |  |  | 	struct qeth_ipaddr *addr; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-06-21 22:57:05 +00:00
										 |  |  | 	QETH_CARD_TEXT(card, 4, "frvaddr6"); | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	in6_dev = in6_dev_get(vlan_group_get_device(card->vlangrp, vid)); | 
					
						
							|  |  |  | 	if (!in6_dev) | 
					
						
							|  |  |  | 		return; | 
					
						
							| 
									
										
										
										
											2010-03-22 09:59:48 -07:00
										 |  |  | 	list_for_each_entry(ifa, &in6_dev->addr_list, if_list) { | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 		addr = qeth_l3_get_addr_buffer(QETH_PROT_IPV6); | 
					
						
							|  |  |  | 		if (addr) { | 
					
						
							|  |  |  | 			memcpy(&addr->u.a6.addr, &ifa->addr, | 
					
						
							|  |  |  | 			       sizeof(struct in6_addr)); | 
					
						
							|  |  |  | 			addr->u.a6.pfxlen = ifa->prefix_len; | 
					
						
							|  |  |  | 			addr->type = QETH_IP_TYPE_NORMAL; | 
					
						
							|  |  |  | 			if (!qeth_l3_delete_ip(card, addr)) | 
					
						
							|  |  |  | 				kfree(addr); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	in6_dev_put(in6_dev); | 
					
						
							|  |  |  | #endif /* CONFIG_QETH_IPV6 */
 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void qeth_l3_free_vlan_addresses(struct qeth_card *card, | 
					
						
							|  |  |  | 			unsigned short vid) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	if (!card->vlangrp) | 
					
						
							|  |  |  | 		return; | 
					
						
							|  |  |  | 	qeth_l3_free_vlan_addresses4(card, vid); | 
					
						
							|  |  |  | 	qeth_l3_free_vlan_addresses6(card, vid); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void qeth_l3_vlan_rx_register(struct net_device *dev, | 
					
						
							|  |  |  | 			struct vlan_group *grp) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2008-07-26 02:24:10 -07:00
										 |  |  | 	struct qeth_card *card = dev->ml_priv; | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 	unsigned long flags; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-06-21 22:57:05 +00:00
										 |  |  | 	QETH_CARD_TEXT(card, 4, "vlanreg"); | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 	spin_lock_irqsave(&card->vlanlock, flags); | 
					
						
							|  |  |  | 	card->vlangrp = grp; | 
					
						
							|  |  |  | 	spin_unlock_irqrestore(&card->vlanlock, flags); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void qeth_l3_vlan_rx_add_vid(struct net_device *dev, unsigned short vid) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	return; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void qeth_l3_vlan_rx_kill_vid(struct net_device *dev, unsigned short vid) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2008-07-26 02:24:10 -07:00
										 |  |  | 	struct qeth_card *card = dev->ml_priv; | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 	unsigned long flags; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-06-21 22:57:05 +00:00
										 |  |  | 	QETH_CARD_TEXT_(card, 4, "kid:%d", vid); | 
					
						
							| 
									
										
										
										
											2009-03-24 20:57:18 +00:00
										 |  |  | 	if (qeth_wait_for_threads(card, QETH_RECOVER_THREAD)) { | 
					
						
							| 
									
										
										
										
											2010-06-21 22:57:05 +00:00
										 |  |  | 		QETH_CARD_TEXT(card, 3, "kidREC"); | 
					
						
							| 
									
										
										
										
											2009-03-24 20:57:18 +00:00
										 |  |  | 		return; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 	spin_lock_irqsave(&card->vlanlock, flags); | 
					
						
							|  |  |  | 	/* unregister IP addresses of vlan device */ | 
					
						
							|  |  |  | 	qeth_l3_free_vlan_addresses(card, vid); | 
					
						
							|  |  |  | 	vlan_group_set_device(card->vlangrp, vid, NULL); | 
					
						
							|  |  |  | 	spin_unlock_irqrestore(&card->vlanlock, flags); | 
					
						
							|  |  |  | 	qeth_l3_set_multicast_list(card->dev); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static inline __u16 qeth_l3_rebuild_skb(struct qeth_card *card, | 
					
						
							|  |  |  | 			struct sk_buff *skb, struct qeth_hdr *hdr) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	unsigned short vlan_id = 0; | 
					
						
							|  |  |  | 	__be16 prot; | 
					
						
							|  |  |  | 	struct iphdr *ip_hdr; | 
					
						
							|  |  |  | 	unsigned char tg_addr[MAX_ADDR_LEN]; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (!(hdr->hdr.l3.flags & QETH_HDR_PASSTHRU)) { | 
					
						
							|  |  |  | 		prot = htons((hdr->hdr.l3.flags & QETH_HDR_IPV6)? ETH_P_IPV6 : | 
					
						
							|  |  |  | 			      ETH_P_IP); | 
					
						
							|  |  |  | 		switch (hdr->hdr.l3.flags & QETH_HDR_CAST_MASK) { | 
					
						
							|  |  |  | 		case QETH_CAST_MULTICAST: | 
					
						
							|  |  |  | 			switch (prot) { | 
					
						
							|  |  |  | #ifdef CONFIG_QETH_IPV6
 | 
					
						
							|  |  |  | 			case __constant_htons(ETH_P_IPV6): | 
					
						
							|  |  |  | 				ndisc_mc_map((struct in6_addr *) | 
					
						
							|  |  |  | 				     skb->data + 24, | 
					
						
							|  |  |  | 				     tg_addr, card->dev, 0); | 
					
						
							|  |  |  | 				break; | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 			case __constant_htons(ETH_P_IP): | 
					
						
							|  |  |  | 				ip_hdr = (struct iphdr *)skb->data; | 
					
						
							|  |  |  | 				(card->dev->type == ARPHRD_IEEE802_TR) ? | 
					
						
							|  |  |  | 				ip_tr_mc_map(ip_hdr->daddr, tg_addr): | 
					
						
							|  |  |  | 				ip_eth_mc_map(ip_hdr->daddr, tg_addr); | 
					
						
							|  |  |  | 				break; | 
					
						
							|  |  |  | 			default: | 
					
						
							|  |  |  | 				memcpy(tg_addr, card->dev->broadcast, | 
					
						
							|  |  |  | 					card->dev->addr_len); | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			card->stats.multicast++; | 
					
						
							|  |  |  | 			skb->pkt_type = PACKET_MULTICAST; | 
					
						
							|  |  |  | 			break; | 
					
						
							|  |  |  | 		case QETH_CAST_BROADCAST: | 
					
						
							|  |  |  | 			memcpy(tg_addr, card->dev->broadcast, | 
					
						
							|  |  |  | 				card->dev->addr_len); | 
					
						
							|  |  |  | 			card->stats.multicast++; | 
					
						
							|  |  |  | 			skb->pkt_type = PACKET_BROADCAST; | 
					
						
							|  |  |  | 			break; | 
					
						
							|  |  |  | 		case QETH_CAST_UNICAST: | 
					
						
							|  |  |  | 		case QETH_CAST_ANYCAST: | 
					
						
							|  |  |  | 		case QETH_CAST_NOCAST: | 
					
						
							|  |  |  | 		default: | 
					
						
							| 
									
										
										
										
											2010-01-11 02:50:50 +00:00
										 |  |  | 			if (card->options.sniffer) | 
					
						
							|  |  |  | 				skb->pkt_type = PACKET_OTHERHOST; | 
					
						
							|  |  |  | 			else | 
					
						
							|  |  |  | 				skb->pkt_type = PACKET_HOST; | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 			memcpy(tg_addr, card->dev->dev_addr, | 
					
						
							|  |  |  | 				card->dev->addr_len); | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2009-01-04 17:34:52 -08:00
										 |  |  | 		if (hdr->hdr.l3.ext_flags & QETH_HDR_EXT_SRC_MAC_ADDR) | 
					
						
							|  |  |  | 			card->dev->header_ops->create(skb, card->dev, prot, | 
					
						
							|  |  |  | 				tg_addr, &hdr->hdr.l3.dest_addr[2], | 
					
						
							|  |  |  | 				card->dev->addr_len); | 
					
						
							|  |  |  | 		else | 
					
						
							|  |  |  | 			card->dev->header_ops->create(skb, card->dev, prot, | 
					
						
							|  |  |  | 				tg_addr, "FAKELL", card->dev->addr_len); | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #ifdef CONFIG_TR
 | 
					
						
							|  |  |  | 	if (card->dev->type == ARPHRD_IEEE802_TR) | 
					
						
							|  |  |  | 		skb->protocol = tr_type_trans(skb, card->dev); | 
					
						
							|  |  |  | 	else | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 		skb->protocol = eth_type_trans(skb, card->dev); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (hdr->hdr.l3.ext_flags & | 
					
						
							|  |  |  | 	    (QETH_HDR_EXT_VLAN_FRAME | QETH_HDR_EXT_INCLUDE_VLAN_TAG)) { | 
					
						
							|  |  |  | 		vlan_id = (hdr->hdr.l3.ext_flags & QETH_HDR_EXT_VLAN_FRAME)? | 
					
						
							|  |  |  | 		 hdr->hdr.l3.vlan_id : *((u16 *)&hdr->hdr.l3.dest_addr[12]); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-05-19 21:38:40 +00:00
										 |  |  | 	switch (card->options.checksum_type) { | 
					
						
							|  |  |  | 	case SW_CHECKSUMMING: | 
					
						
							|  |  |  | 		skb->ip_summed = CHECKSUM_NONE; | 
					
						
							|  |  |  | 		break; | 
					
						
							|  |  |  | 	case NO_CHECKSUMMING: | 
					
						
							|  |  |  | 		skb->ip_summed = CHECKSUM_UNNECESSARY; | 
					
						
							|  |  |  | 		break; | 
					
						
							|  |  |  | 	case HW_CHECKSUMMING: | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 		if ((hdr->hdr.l3.ext_flags & | 
					
						
							| 
									
										
										
										
											2009-05-19 21:38:40 +00:00
										 |  |  | 		    (QETH_HDR_EXT_CSUM_HDR_REQ | | 
					
						
							|  |  |  | 		     QETH_HDR_EXT_CSUM_TRANSP_REQ)) == | 
					
						
							|  |  |  | 		    (QETH_HDR_EXT_CSUM_HDR_REQ | | 
					
						
							|  |  |  | 		     QETH_HDR_EXT_CSUM_TRANSP_REQ)) | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 			skb->ip_summed = CHECKSUM_UNNECESSARY; | 
					
						
							|  |  |  | 		else | 
					
						
							| 
									
										
										
										
											2009-05-19 21:38:40 +00:00
										 |  |  | 			skb->ip_summed = CHECKSUM_NONE; | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return vlan_id; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void qeth_l3_process_inbound_buffer(struct qeth_card *card, | 
					
						
							|  |  |  | 			    struct qeth_qdio_buffer *buf, int index) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct qdio_buffer_element *element; | 
					
						
							|  |  |  | 	struct sk_buff *skb; | 
					
						
							|  |  |  | 	struct qeth_hdr *hdr; | 
					
						
							|  |  |  | 	int offset; | 
					
						
							|  |  |  | 	__u16 vlan_tag = 0; | 
					
						
							|  |  |  | 	unsigned int len; | 
					
						
							|  |  |  | 	/* get first element of current buffer */ | 
					
						
							|  |  |  | 	element = (struct qdio_buffer_element *)&buf->buffer->element[0]; | 
					
						
							|  |  |  | 	offset = 0; | 
					
						
							|  |  |  | 	if (card->options.performance_stats) | 
					
						
							|  |  |  | 		card->perf_stats.bufs_rec++; | 
					
						
							|  |  |  | 	while ((skb = qeth_core_get_next_skb(card, buf->buffer, &element, | 
					
						
							|  |  |  | 				       &offset, &hdr))) { | 
					
						
							|  |  |  | 		skb->dev = card->dev; | 
					
						
							|  |  |  | 		/* is device UP ? */ | 
					
						
							|  |  |  | 		if (!(card->dev->flags & IFF_UP)) { | 
					
						
							|  |  |  | 			dev_kfree_skb_any(skb); | 
					
						
							|  |  |  | 			continue; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		switch (hdr->hdr.l3.id) { | 
					
						
							|  |  |  | 		case QETH_HEADER_TYPE_LAYER3: | 
					
						
							|  |  |  | 			vlan_tag = qeth_l3_rebuild_skb(card, skb, hdr); | 
					
						
							|  |  |  | 			len = skb->len; | 
					
						
							| 
									
										
										
										
											2010-01-11 02:50:50 +00:00
										 |  |  | 			if (vlan_tag && !card->options.sniffer) | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 				if (card->vlangrp) | 
					
						
							|  |  |  | 					vlan_hwaccel_rx(skb, card->vlangrp, | 
					
						
							|  |  |  | 						vlan_tag); | 
					
						
							|  |  |  | 				else { | 
					
						
							|  |  |  | 					dev_kfree_skb_any(skb); | 
					
						
							|  |  |  | 					continue; | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 			else | 
					
						
							|  |  |  | 				netif_rx(skb); | 
					
						
							|  |  |  | 			break; | 
					
						
							| 
									
										
										
										
											2010-01-11 02:50:50 +00:00
										 |  |  | 		case QETH_HEADER_TYPE_LAYER2: /* for HiperSockets sniffer */ | 
					
						
							|  |  |  | 			skb->pkt_type = PACKET_HOST; | 
					
						
							|  |  |  | 			skb->protocol = eth_type_trans(skb, skb->dev); | 
					
						
							|  |  |  | 			if (card->options.checksum_type == NO_CHECKSUMMING) | 
					
						
							|  |  |  | 				skb->ip_summed = CHECKSUM_UNNECESSARY; | 
					
						
							|  |  |  | 			else | 
					
						
							|  |  |  | 				skb->ip_summed = CHECKSUM_NONE; | 
					
						
							|  |  |  | 			len = skb->len; | 
					
						
							|  |  |  | 			netif_receive_skb(skb); | 
					
						
							|  |  |  | 			break; | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 		default: | 
					
						
							|  |  |  | 			dev_kfree_skb_any(skb); | 
					
						
							| 
									
										
										
										
											2010-06-21 22:57:05 +00:00
										 |  |  | 			QETH_CARD_TEXT(card, 3, "inbunkno"); | 
					
						
							| 
									
										
										
										
											2008-04-01 10:26:58 +02:00
										 |  |  | 			QETH_DBF_HEX(CTRL, 3, hdr, QETH_DBF_CTRL_LEN); | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 			continue; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		card->stats.rx_packets++; | 
					
						
							|  |  |  | 		card->stats.rx_bytes += len; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int qeth_l3_verify_vlan_dev(struct net_device *dev, | 
					
						
							|  |  |  | 			struct qeth_card *card) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	int rc = 0; | 
					
						
							|  |  |  | 	struct vlan_group *vg; | 
					
						
							|  |  |  | 	int i; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	vg = card->vlangrp; | 
					
						
							|  |  |  | 	if (!vg) | 
					
						
							|  |  |  | 		return rc; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	for (i = 0; i < VLAN_GROUP_ARRAY_LEN; i++) { | 
					
						
							|  |  |  | 		if (vlan_group_get_device(vg, i) == dev) { | 
					
						
							|  |  |  | 			rc = QETH_VLAN_CARD; | 
					
						
							|  |  |  | 			break; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-07-26 02:24:10 -07:00
										 |  |  | 	if (rc && !(vlan_dev_real_dev(dev)->ml_priv == (void *)card)) | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 		return 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return rc; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int qeth_l3_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; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		rc = qeth_l3_verify_vlan_dev(dev, card); | 
					
						
							|  |  |  | 		if (rc) | 
					
						
							|  |  |  | 			break; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	read_unlock_irqrestore(&qeth_core_card_list.rwlock, flags); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return rc; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static struct qeth_card *qeth_l3_get_card_from_dev(struct net_device *dev) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct qeth_card *card = NULL; | 
					
						
							|  |  |  | 	int rc; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	rc = qeth_l3_verify_dev(dev); | 
					
						
							|  |  |  | 	if (rc == QETH_REAL_CARD) | 
					
						
							| 
									
										
										
										
											2008-07-26 02:24:10 -07:00
										 |  |  | 		card = dev->ml_priv; | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 	else if (rc == QETH_VLAN_CARD) | 
					
						
							| 
									
										
										
										
											2008-07-26 02:24:10 -07:00
										 |  |  | 		card = vlan_dev_real_dev(dev)->ml_priv; | 
					
						
							| 
									
										
										
										
											2008-06-06 12:37:44 +02:00
										 |  |  | 	if (card && card->options.layer2) | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 		card = NULL; | 
					
						
							| 
									
										
										
										
											2010-06-21 22:57:05 +00:00
										 |  |  | 	if (card) | 
					
						
							|  |  |  | 		QETH_CARD_TEXT_(card, 4, "%d", rc); | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 	return card ; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int qeth_l3_stop_card(struct qeth_card *card, int recovery_mode) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	int rc = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-04-01 10:26:58 +02:00
										 |  |  | 	QETH_DBF_TEXT(SETUP, 2, "stopcard"); | 
					
						
							|  |  |  | 	QETH_DBF_HEX(SETUP, 2, &card, sizeof(void *)); | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	qeth_set_allowed_threads(card, 0, 1); | 
					
						
							| 
									
										
										
										
											2010-01-11 02:50:50 +00:00
										 |  |  | 	if (card->options.sniffer && | 
					
						
							|  |  |  | 	    (card->info.promisc_mode == SET_PROMISC_MODE_ON)) | 
					
						
							|  |  |  | 		qeth_diags_trace(card, QETH_DIAGS_CMD_TRACE_DISABLE); | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 	if (card->read.state == CH_STATE_UP && | 
					
						
							|  |  |  | 	    card->write.state == CH_STATE_UP && | 
					
						
							|  |  |  | 	    (card->state == CARD_STATE_UP)) { | 
					
						
							|  |  |  | 		if (recovery_mode) | 
					
						
							|  |  |  | 			qeth_l3_stop(card->dev); | 
					
						
							| 
									
										
										
										
											2008-04-24 10:15:25 +02:00
										 |  |  | 		else { | 
					
						
							| 
									
										
										
										
											2010-03-08 20:36:56 +00:00
										 |  |  | 			rtnl_lock(); | 
					
						
							|  |  |  | 			dev_close(card->dev); | 
					
						
							|  |  |  | 			rtnl_unlock(); | 
					
						
							| 
									
										
										
										
											2008-04-24 10:15:25 +02:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 		if (!card->use_hard_stop) { | 
					
						
							|  |  |  | 			rc = qeth_send_stoplan(card); | 
					
						
							|  |  |  | 			if (rc) | 
					
						
							| 
									
										
										
										
											2008-04-01 10:26:58 +02:00
										 |  |  | 				QETH_DBF_TEXT_(SETUP, 2, "1err%d", rc); | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 		} | 
					
						
							|  |  |  | 		card->state = CARD_STATE_SOFTSETUP; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	if (card->state == CARD_STATE_SOFTSETUP) { | 
					
						
							|  |  |  | 		qeth_l3_clear_ip_list(card, !card->use_hard_stop, 1); | 
					
						
							|  |  |  | 		qeth_clear_ipacmd_list(card); | 
					
						
							|  |  |  | 		card->state = CARD_STATE_HARDSETUP; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	if (card->state == CARD_STATE_HARDSETUP) { | 
					
						
							|  |  |  | 		if (!card->use_hard_stop && | 
					
						
							|  |  |  | 		    (card->info.type != QETH_CARD_TYPE_IQD)) { | 
					
						
							|  |  |  | 			rc = qeth_l3_put_unique_id(card); | 
					
						
							|  |  |  | 			if (rc) | 
					
						
							| 
									
										
										
										
											2008-04-01 10:26:58 +02:00
										 |  |  | 				QETH_DBF_TEXT_(SETUP, 2, "2err%d", rc); | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 		} | 
					
						
							|  |  |  | 		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); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	card->use_hard_stop = 0; | 
					
						
							|  |  |  | 	return rc; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-01-11 02:50:50 +00:00
										 |  |  | /*
 | 
					
						
							|  |  |  |  * test for and Switch promiscuous mode (on or off) | 
					
						
							|  |  |  |  *  either for guestlan or HiperSocket Sniffer | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | static void | 
					
						
							|  |  |  | qeth_l3_handle_promisc_mode(struct qeth_card *card) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct net_device *dev = card->dev; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (((dev->flags & IFF_PROMISC) && | 
					
						
							|  |  |  | 	     (card->info.promisc_mode == SET_PROMISC_MODE_ON)) || | 
					
						
							|  |  |  | 	    (!(dev->flags & IFF_PROMISC) && | 
					
						
							|  |  |  | 	     (card->info.promisc_mode == SET_PROMISC_MODE_OFF))) | 
					
						
							|  |  |  | 		return; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (card->info.guestlan) {		/* Guestlan trace */ | 
					
						
							|  |  |  | 		if (qeth_adp_supported(card, IPA_SETADP_SET_PROMISC_MODE)) | 
					
						
							|  |  |  | 			qeth_setadp_promisc_mode(card); | 
					
						
							|  |  |  | 	} else if (card->options.sniffer &&	/* HiperSockets trace */ | 
					
						
							|  |  |  | 		   qeth_adp_supported(card, IPA_SETADP_SET_DIAG_ASSIST)) { | 
					
						
							|  |  |  | 		if (dev->flags & IFF_PROMISC) { | 
					
						
							| 
									
										
										
										
											2010-06-21 22:57:05 +00:00
										 |  |  | 			QETH_CARD_TEXT(card, 3, "+promisc"); | 
					
						
							| 
									
										
										
										
											2010-01-11 02:50:50 +00:00
										 |  |  | 			qeth_diags_trace(card, QETH_DIAGS_CMD_TRACE_ENABLE); | 
					
						
							|  |  |  | 		} else { | 
					
						
							| 
									
										
										
										
											2010-06-21 22:57:05 +00:00
										 |  |  | 			QETH_CARD_TEXT(card, 3, "-promisc"); | 
					
						
							| 
									
										
										
										
											2010-01-11 02:50:50 +00:00
										 |  |  | 			qeth_diags_trace(card, QETH_DIAGS_CMD_TRACE_DISABLE); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | static void qeth_l3_set_multicast_list(struct net_device *dev) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2008-07-26 02:24:10 -07:00
										 |  |  | 	struct qeth_card *card = dev->ml_priv; | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-06-21 22:57:05 +00:00
										 |  |  | 	QETH_CARD_TEXT(card, 3, "setmulti"); | 
					
						
							| 
									
										
										
										
											2009-03-24 20:57:18 +00:00
										 |  |  | 	if (qeth_threads_running(card, QETH_RECOVER_THREAD) && | 
					
						
							|  |  |  | 	    (card->state != CARD_STATE_UP)) | 
					
						
							|  |  |  | 		return; | 
					
						
							| 
									
										
										
										
											2010-01-11 02:50:50 +00:00
										 |  |  | 	if (!card->options.sniffer) { | 
					
						
							|  |  |  | 		qeth_l3_delete_mc_addresses(card); | 
					
						
							|  |  |  | 		qeth_l3_add_multicast_ipv4(card); | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | #ifdef CONFIG_QETH_IPV6
 | 
					
						
							| 
									
										
										
										
											2010-01-11 02:50:50 +00:00
										 |  |  | 		qeth_l3_add_multicast_ipv6(card); | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2010-01-11 02:50:50 +00:00
										 |  |  | 		qeth_l3_set_ip_addr_list(card); | 
					
						
							|  |  |  | 		if (!qeth_adp_supported(card, IPA_SETADP_SET_PROMISC_MODE)) | 
					
						
							|  |  |  | 			return; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	qeth_l3_handle_promisc_mode(card); | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static const char *qeth_l3_arp_get_error_cause(int *rc) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	switch (*rc) { | 
					
						
							|  |  |  | 	case QETH_IPA_ARP_RC_FAILED: | 
					
						
							|  |  |  | 		*rc = -EIO; | 
					
						
							|  |  |  | 		return "operation failed"; | 
					
						
							|  |  |  | 	case QETH_IPA_ARP_RC_NOTSUPP: | 
					
						
							|  |  |  | 		*rc = -EOPNOTSUPP; | 
					
						
							|  |  |  | 		return "operation not supported"; | 
					
						
							|  |  |  | 	case QETH_IPA_ARP_RC_OUT_OF_RANGE: | 
					
						
							|  |  |  | 		*rc = -EINVAL; | 
					
						
							|  |  |  | 		return "argument out of range"; | 
					
						
							|  |  |  | 	case QETH_IPA_ARP_RC_Q_NOTSUPP: | 
					
						
							|  |  |  | 		*rc = -EOPNOTSUPP; | 
					
						
							|  |  |  | 		return "query operation not supported"; | 
					
						
							|  |  |  | 	case QETH_IPA_ARP_RC_Q_NO_DATA: | 
					
						
							|  |  |  | 		*rc = -ENOENT; | 
					
						
							|  |  |  | 		return "no query data available"; | 
					
						
							|  |  |  | 	default: | 
					
						
							|  |  |  | 		return "unknown error"; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int qeth_l3_arp_set_no_entries(struct qeth_card *card, int no_entries) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	int tmp; | 
					
						
							|  |  |  | 	int rc; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-06-21 22:57:05 +00:00
										 |  |  | 	QETH_CARD_TEXT(card, 3, "arpstnoe"); | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	/*
 | 
					
						
							|  |  |  | 	 * currently GuestLAN only supports the ARP assist function | 
					
						
							|  |  |  | 	 * IPA_CMD_ASS_ARP_QUERY_INFO, but not IPA_CMD_ASS_ARP_SET_NO_ENTRIES; | 
					
						
							|  |  |  | 	 * thus we say EOPNOTSUPP for this ARP function | 
					
						
							|  |  |  | 	 */ | 
					
						
							|  |  |  | 	if (card->info.guestlan) | 
					
						
							|  |  |  | 		return -EOPNOTSUPP; | 
					
						
							|  |  |  | 	if (!qeth_is_supported(card, IPA_ARP_PROCESSING)) { | 
					
						
							|  |  |  | 		return -EOPNOTSUPP; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	rc = qeth_l3_send_simple_setassparms(card, IPA_ARP_PROCESSING, | 
					
						
							|  |  |  | 					  IPA_CMD_ASS_ARP_SET_NO_ENTRIES, | 
					
						
							|  |  |  | 					  no_entries); | 
					
						
							|  |  |  | 	if (rc) { | 
					
						
							|  |  |  | 		tmp = rc; | 
					
						
							| 
									
										
										
										
											2008-06-06 12:37:46 +02:00
										 |  |  | 		QETH_DBF_MESSAGE(2, "Could not set number of ARP entries on " | 
					
						
							|  |  |  | 			"%s: %s (0x%x/%d)\n", QETH_CARD_IFNAME(card), | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 			qeth_l3_arp_get_error_cause(&rc), tmp, tmp); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return rc; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void qeth_l3_copy_arp_entries_stripped(struct qeth_arp_query_info *qinfo, | 
					
						
							|  |  |  | 		struct qeth_arp_query_data *qdata, int entry_size, | 
					
						
							|  |  |  | 		int uentry_size) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	char *entry_ptr; | 
					
						
							|  |  |  | 	char *uentry_ptr; | 
					
						
							|  |  |  | 	int i; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	entry_ptr = (char *)&qdata->data; | 
					
						
							|  |  |  | 	uentry_ptr = (char *)(qinfo->udata + qinfo->udata_offset); | 
					
						
							|  |  |  | 	for (i = 0; i < qdata->no_entries; ++i) { | 
					
						
							|  |  |  | 		/* strip off 32 bytes "media specific information" */ | 
					
						
							|  |  |  | 		memcpy(uentry_ptr, (entry_ptr + 32), entry_size - 32); | 
					
						
							|  |  |  | 		entry_ptr += entry_size; | 
					
						
							|  |  |  | 		uentry_ptr += uentry_size; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int qeth_l3_arp_query_cb(struct qeth_card *card, | 
					
						
							|  |  |  | 		struct qeth_reply *reply, unsigned long data) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct qeth_ipa_cmd *cmd; | 
					
						
							|  |  |  | 	struct qeth_arp_query_data *qdata; | 
					
						
							|  |  |  | 	struct qeth_arp_query_info *qinfo; | 
					
						
							|  |  |  | 	int entry_size; | 
					
						
							|  |  |  | 	int uentry_size; | 
					
						
							|  |  |  | 	int i; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-06-21 22:57:05 +00:00
										 |  |  | 	QETH_CARD_TEXT(card, 4, "arpquecb"); | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	qinfo = (struct qeth_arp_query_info *) reply->param; | 
					
						
							|  |  |  | 	cmd = (struct qeth_ipa_cmd *) data; | 
					
						
							|  |  |  | 	if (cmd->hdr.return_code) { | 
					
						
							| 
									
										
										
										
											2010-06-21 22:57:05 +00:00
										 |  |  | 		QETH_CARD_TEXT_(card, 4, "qaer1%i", cmd->hdr.return_code); | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 		return 0; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	if (cmd->data.setassparms.hdr.return_code) { | 
					
						
							|  |  |  | 		cmd->hdr.return_code = cmd->data.setassparms.hdr.return_code; | 
					
						
							| 
									
										
										
										
											2010-06-21 22:57:05 +00:00
										 |  |  | 		QETH_CARD_TEXT_(card, 4, "qaer2%i", cmd->hdr.return_code); | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 		return 0; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	qdata = &cmd->data.setassparms.data.query_arp; | 
					
						
							|  |  |  | 	switch (qdata->reply_bits) { | 
					
						
							|  |  |  | 	case 5: | 
					
						
							|  |  |  | 		uentry_size = entry_size = sizeof(struct qeth_arp_qi_entry5); | 
					
						
							|  |  |  | 		if (qinfo->mask_bits & QETH_QARP_STRIP_ENTRIES) | 
					
						
							|  |  |  | 			uentry_size = sizeof(struct qeth_arp_qi_entry5_short); | 
					
						
							|  |  |  | 		break; | 
					
						
							|  |  |  | 	case 7: | 
					
						
							|  |  |  | 		/* fall through to default */ | 
					
						
							|  |  |  | 	default: | 
					
						
							|  |  |  | 		/* tr is the same as eth -> entry7 */ | 
					
						
							|  |  |  | 		uentry_size = entry_size = sizeof(struct qeth_arp_qi_entry7); | 
					
						
							|  |  |  | 		if (qinfo->mask_bits & QETH_QARP_STRIP_ENTRIES) | 
					
						
							|  |  |  | 			uentry_size = sizeof(struct qeth_arp_qi_entry7_short); | 
					
						
							|  |  |  | 		break; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	/* check if there is enough room in userspace */ | 
					
						
							|  |  |  | 	if ((qinfo->udata_len - qinfo->udata_offset) < | 
					
						
							|  |  |  | 			qdata->no_entries * uentry_size){ | 
					
						
							| 
									
										
										
										
											2010-06-21 22:57:05 +00:00
										 |  |  | 		QETH_CARD_TEXT_(card, 4, "qaer3%i", -ENOMEM); | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 		cmd->hdr.return_code = -ENOMEM; | 
					
						
							|  |  |  | 		goto out_error; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2010-06-21 22:57:05 +00:00
										 |  |  | 	QETH_CARD_TEXT_(card, 4, "anore%i", | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 		       cmd->data.setassparms.hdr.number_of_replies); | 
					
						
							| 
									
										
										
										
											2010-06-21 22:57:05 +00:00
										 |  |  | 	QETH_CARD_TEXT_(card, 4, "aseqn%i", cmd->data.setassparms.hdr.seq_no); | 
					
						
							|  |  |  | 	QETH_CARD_TEXT_(card, 4, "anoen%i", qdata->no_entries); | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	if (qinfo->mask_bits & QETH_QARP_STRIP_ENTRIES) { | 
					
						
							|  |  |  | 		/* strip off "media specific information" */ | 
					
						
							|  |  |  | 		qeth_l3_copy_arp_entries_stripped(qinfo, qdata, entry_size, | 
					
						
							|  |  |  | 					       uentry_size); | 
					
						
							|  |  |  | 	} else | 
					
						
							|  |  |  | 		/*copy entries to user buffer*/ | 
					
						
							|  |  |  | 		memcpy(qinfo->udata + qinfo->udata_offset, | 
					
						
							|  |  |  | 		       (char *)&qdata->data, qdata->no_entries*uentry_size); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	qinfo->no_entries += qdata->no_entries; | 
					
						
							|  |  |  | 	qinfo->udata_offset += (qdata->no_entries*uentry_size); | 
					
						
							|  |  |  | 	/* check if all replies received ... */ | 
					
						
							|  |  |  | 	if (cmd->data.setassparms.hdr.seq_no < | 
					
						
							|  |  |  | 	    cmd->data.setassparms.hdr.number_of_replies) | 
					
						
							|  |  |  | 		return 1; | 
					
						
							|  |  |  | 	memcpy(qinfo->udata, &qinfo->no_entries, 4); | 
					
						
							|  |  |  | 	/* keep STRIP_ENTRIES flag so the user program can distinguish
 | 
					
						
							|  |  |  | 	 * stripped entries from normal ones */ | 
					
						
							|  |  |  | 	if (qinfo->mask_bits & QETH_QARP_STRIP_ENTRIES) | 
					
						
							|  |  |  | 		qdata->reply_bits |= QETH_QARP_STRIP_ENTRIES; | 
					
						
							|  |  |  | 	memcpy(qinfo->udata + QETH_QARP_MASK_OFFSET, &qdata->reply_bits, 2); | 
					
						
							|  |  |  | 	return 0; | 
					
						
							|  |  |  | out_error: | 
					
						
							|  |  |  | 	i = 0; | 
					
						
							|  |  |  | 	memcpy(qinfo->udata, &i, 4); | 
					
						
							|  |  |  | 	return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int qeth_l3_send_ipa_arp_cmd(struct qeth_card *card, | 
					
						
							|  |  |  | 		struct qeth_cmd_buffer *iob, int len, | 
					
						
							|  |  |  | 		int (*reply_cb)(struct qeth_card *, struct qeth_reply *, | 
					
						
							|  |  |  | 			unsigned long), | 
					
						
							|  |  |  | 		void *reply_param) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2010-06-21 22:57:05 +00:00
										 |  |  | 	QETH_CARD_TEXT(card, 4, "sendarp"); | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	memcpy(iob->data, IPA_PDU_HEADER, IPA_PDU_HEADER_SIZE); | 
					
						
							|  |  |  | 	memcpy(QETH_IPA_CMD_DEST_ADDR(iob->data), | 
					
						
							|  |  |  | 	       &card->token.ulp_connection_r, QETH_MPC_TOKEN_LENGTH); | 
					
						
							|  |  |  | 	return qeth_send_control_data(card, IPA_PDU_HEADER_SIZE + len, iob, | 
					
						
							|  |  |  | 				      reply_cb, reply_param); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int qeth_l3_arp_query(struct qeth_card *card, char __user *udata) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct qeth_cmd_buffer *iob; | 
					
						
							|  |  |  | 	struct qeth_arp_query_info qinfo = {0, }; | 
					
						
							|  |  |  | 	int tmp; | 
					
						
							|  |  |  | 	int rc; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-06-21 22:57:05 +00:00
										 |  |  | 	QETH_CARD_TEXT(card, 3, "arpquery"); | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	if (!qeth_is_supported(card,/*IPA_QUERY_ARP_ADDR_INFO*/ | 
					
						
							|  |  |  | 			       IPA_ARP_PROCESSING)) { | 
					
						
							|  |  |  | 		return -EOPNOTSUPP; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	/* get size of userspace buffer and mask_bits -> 6 bytes */ | 
					
						
							|  |  |  | 	if (copy_from_user(&qinfo, udata, 6)) | 
					
						
							|  |  |  | 		return -EFAULT; | 
					
						
							|  |  |  | 	qinfo.udata = kzalloc(qinfo.udata_len, GFP_KERNEL); | 
					
						
							|  |  |  | 	if (!qinfo.udata) | 
					
						
							|  |  |  | 		return -ENOMEM; | 
					
						
							|  |  |  | 	qinfo.udata_offset = QETH_QARP_ENTRIES_OFFSET; | 
					
						
							|  |  |  | 	iob = qeth_l3_get_setassparms_cmd(card, IPA_ARP_PROCESSING, | 
					
						
							|  |  |  | 				       IPA_CMD_ASS_ARP_QUERY_INFO, | 
					
						
							|  |  |  | 				       sizeof(int), QETH_PROT_IPV4); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	rc = qeth_l3_send_ipa_arp_cmd(card, iob, | 
					
						
							|  |  |  | 				   QETH_SETASS_BASE_LEN+QETH_ARP_CMD_LEN, | 
					
						
							|  |  |  | 				   qeth_l3_arp_query_cb, (void *)&qinfo); | 
					
						
							|  |  |  | 	if (rc) { | 
					
						
							|  |  |  | 		tmp = rc; | 
					
						
							| 
									
										
										
										
											2008-06-06 12:37:46 +02:00
										 |  |  | 		QETH_DBF_MESSAGE(2, "Error while querying ARP cache on %s: %s " | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 			"(0x%x/%d)\n", QETH_CARD_IFNAME(card), | 
					
						
							|  |  |  | 			qeth_l3_arp_get_error_cause(&rc), tmp, tmp); | 
					
						
							|  |  |  | 		if (copy_to_user(udata, qinfo.udata, 4)) | 
					
						
							|  |  |  | 			rc = -EFAULT; | 
					
						
							|  |  |  | 	} else { | 
					
						
							|  |  |  | 		if (copy_to_user(udata, qinfo.udata, qinfo.udata_len)) | 
					
						
							|  |  |  | 			rc = -EFAULT; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	kfree(qinfo.udata); | 
					
						
							|  |  |  | 	return rc; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int qeth_l3_arp_add_entry(struct qeth_card *card, | 
					
						
							|  |  |  | 				struct qeth_arp_cache_entry *entry) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct qeth_cmd_buffer *iob; | 
					
						
							|  |  |  | 	char buf[16]; | 
					
						
							|  |  |  | 	int tmp; | 
					
						
							|  |  |  | 	int rc; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-06-21 22:57:05 +00:00
										 |  |  | 	QETH_CARD_TEXT(card, 3, "arpadent"); | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	/*
 | 
					
						
							|  |  |  | 	 * currently GuestLAN only supports the ARP assist function | 
					
						
							|  |  |  | 	 * IPA_CMD_ASS_ARP_QUERY_INFO, but not IPA_CMD_ASS_ARP_ADD_ENTRY; | 
					
						
							|  |  |  | 	 * thus we say EOPNOTSUPP for this ARP function | 
					
						
							|  |  |  | 	 */ | 
					
						
							|  |  |  | 	if (card->info.guestlan) | 
					
						
							|  |  |  | 		return -EOPNOTSUPP; | 
					
						
							|  |  |  | 	if (!qeth_is_supported(card, IPA_ARP_PROCESSING)) { | 
					
						
							|  |  |  | 		return -EOPNOTSUPP; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	iob = qeth_l3_get_setassparms_cmd(card, IPA_ARP_PROCESSING, | 
					
						
							|  |  |  | 				       IPA_CMD_ASS_ARP_ADD_ENTRY, | 
					
						
							|  |  |  | 				       sizeof(struct qeth_arp_cache_entry), | 
					
						
							|  |  |  | 				       QETH_PROT_IPV4); | 
					
						
							|  |  |  | 	rc = qeth_l3_send_setassparms(card, iob, | 
					
						
							|  |  |  | 				   sizeof(struct qeth_arp_cache_entry), | 
					
						
							|  |  |  | 				   (unsigned long) entry, | 
					
						
							|  |  |  | 				   qeth_l3_default_setassparms_cb, NULL); | 
					
						
							|  |  |  | 	if (rc) { | 
					
						
							|  |  |  | 		tmp = rc; | 
					
						
							|  |  |  | 		qeth_l3_ipaddr4_to_string((u8 *)entry->ipaddr, buf); | 
					
						
							| 
									
										
										
										
											2008-06-06 12:37:46 +02:00
										 |  |  | 		QETH_DBF_MESSAGE(2, "Could not add ARP entry for address %s " | 
					
						
							|  |  |  | 			"on %s: %s (0x%x/%d)\n", buf, QETH_CARD_IFNAME(card), | 
					
						
							|  |  |  | 			qeth_l3_arp_get_error_cause(&rc), tmp, tmp); | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 	} | 
					
						
							|  |  |  | 	return rc; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int qeth_l3_arp_remove_entry(struct qeth_card *card, | 
					
						
							|  |  |  | 				struct qeth_arp_cache_entry *entry) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct qeth_cmd_buffer *iob; | 
					
						
							|  |  |  | 	char buf[16] = {0, }; | 
					
						
							|  |  |  | 	int tmp; | 
					
						
							|  |  |  | 	int rc; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-06-21 22:57:05 +00:00
										 |  |  | 	QETH_CARD_TEXT(card, 3, "arprment"); | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	/*
 | 
					
						
							|  |  |  | 	 * currently GuestLAN only supports the ARP assist function | 
					
						
							|  |  |  | 	 * IPA_CMD_ASS_ARP_QUERY_INFO, but not IPA_CMD_ASS_ARP_REMOVE_ENTRY; | 
					
						
							|  |  |  | 	 * thus we say EOPNOTSUPP for this ARP function | 
					
						
							|  |  |  | 	 */ | 
					
						
							|  |  |  | 	if (card->info.guestlan) | 
					
						
							|  |  |  | 		return -EOPNOTSUPP; | 
					
						
							|  |  |  | 	if (!qeth_is_supported(card, IPA_ARP_PROCESSING)) { | 
					
						
							|  |  |  | 		return -EOPNOTSUPP; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	memcpy(buf, entry, 12); | 
					
						
							|  |  |  | 	iob = qeth_l3_get_setassparms_cmd(card, IPA_ARP_PROCESSING, | 
					
						
							|  |  |  | 				       IPA_CMD_ASS_ARP_REMOVE_ENTRY, | 
					
						
							|  |  |  | 				       12, | 
					
						
							|  |  |  | 				       QETH_PROT_IPV4); | 
					
						
							|  |  |  | 	rc = qeth_l3_send_setassparms(card, iob, | 
					
						
							|  |  |  | 				   12, (unsigned long)buf, | 
					
						
							|  |  |  | 				   qeth_l3_default_setassparms_cb, NULL); | 
					
						
							|  |  |  | 	if (rc) { | 
					
						
							|  |  |  | 		tmp = rc; | 
					
						
							|  |  |  | 		memset(buf, 0, 16); | 
					
						
							|  |  |  | 		qeth_l3_ipaddr4_to_string((u8 *)entry->ipaddr, buf); | 
					
						
							| 
									
										
										
										
											2008-06-06 12:37:46 +02:00
										 |  |  | 		QETH_DBF_MESSAGE(2, "Could not delete ARP entry for address %s" | 
					
						
							|  |  |  | 			" on %s: %s (0x%x/%d)\n", buf, QETH_CARD_IFNAME(card), | 
					
						
							|  |  |  | 			qeth_l3_arp_get_error_cause(&rc), tmp, tmp); | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 	} | 
					
						
							|  |  |  | 	return rc; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int qeth_l3_arp_flush_cache(struct qeth_card *card) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	int rc; | 
					
						
							|  |  |  | 	int tmp; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-06-21 22:57:05 +00:00
										 |  |  | 	QETH_CARD_TEXT(card, 3, "arpflush"); | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	/*
 | 
					
						
							|  |  |  | 	 * currently GuestLAN only supports the ARP assist function | 
					
						
							|  |  |  | 	 * IPA_CMD_ASS_ARP_QUERY_INFO, but not IPA_CMD_ASS_ARP_FLUSH_CACHE; | 
					
						
							|  |  |  | 	 * thus we say EOPNOTSUPP for this ARP function | 
					
						
							|  |  |  | 	*/ | 
					
						
							|  |  |  | 	if (card->info.guestlan || (card->info.type == QETH_CARD_TYPE_IQD)) | 
					
						
							|  |  |  | 		return -EOPNOTSUPP; | 
					
						
							|  |  |  | 	if (!qeth_is_supported(card, IPA_ARP_PROCESSING)) { | 
					
						
							|  |  |  | 		return -EOPNOTSUPP; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	rc = qeth_l3_send_simple_setassparms(card, IPA_ARP_PROCESSING, | 
					
						
							|  |  |  | 					  IPA_CMD_ASS_ARP_FLUSH_CACHE, 0); | 
					
						
							|  |  |  | 	if (rc) { | 
					
						
							|  |  |  | 		tmp = rc; | 
					
						
							| 
									
										
										
										
											2008-06-06 12:37:46 +02:00
										 |  |  | 		QETH_DBF_MESSAGE(2, "Could not flush ARP cache on %s: %s " | 
					
						
							|  |  |  | 			"(0x%x/%d)\n", QETH_CARD_IFNAME(card), | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 			qeth_l3_arp_get_error_cause(&rc), tmp, tmp); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return rc; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int qeth_l3_do_ioctl(struct net_device *dev, struct ifreq *rq, int cmd) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2008-07-26 02:24:10 -07:00
										 |  |  | 	struct qeth_card *card = dev->ml_priv; | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 	struct qeth_arp_cache_entry arp_entry; | 
					
						
							|  |  |  | 	struct mii_ioctl_data *mii_data; | 
					
						
							|  |  |  | 	int rc = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (!card) | 
					
						
							|  |  |  | 		return -ENODEV; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if ((card->state != CARD_STATE_UP) && | 
					
						
							|  |  |  | 		(card->state != CARD_STATE_SOFTSETUP)) | 
					
						
							|  |  |  | 		return -ENODEV; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	switch (cmd) { | 
					
						
							|  |  |  | 	case SIOC_QETH_ARP_SET_NO_ENTRIES: | 
					
						
							|  |  |  | 		if (!capable(CAP_NET_ADMIN)) { | 
					
						
							|  |  |  | 			rc = -EPERM; | 
					
						
							|  |  |  | 			break; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		rc = qeth_l3_arp_set_no_entries(card, rq->ifr_ifru.ifru_ivalue); | 
					
						
							|  |  |  | 		break; | 
					
						
							|  |  |  | 	case SIOC_QETH_ARP_QUERY_INFO: | 
					
						
							|  |  |  | 		if (!capable(CAP_NET_ADMIN)) { | 
					
						
							|  |  |  | 			rc = -EPERM; | 
					
						
							|  |  |  | 			break; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		rc = qeth_l3_arp_query(card, rq->ifr_ifru.ifru_data); | 
					
						
							|  |  |  | 		break; | 
					
						
							|  |  |  | 	case SIOC_QETH_ARP_ADD_ENTRY: | 
					
						
							|  |  |  | 		if (!capable(CAP_NET_ADMIN)) { | 
					
						
							|  |  |  | 			rc = -EPERM; | 
					
						
							|  |  |  | 			break; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		if (copy_from_user(&arp_entry, rq->ifr_ifru.ifru_data, | 
					
						
							|  |  |  | 				   sizeof(struct qeth_arp_cache_entry))) | 
					
						
							|  |  |  | 			rc = -EFAULT; | 
					
						
							|  |  |  | 		else | 
					
						
							|  |  |  | 			rc = qeth_l3_arp_add_entry(card, &arp_entry); | 
					
						
							|  |  |  | 		break; | 
					
						
							|  |  |  | 	case SIOC_QETH_ARP_REMOVE_ENTRY: | 
					
						
							|  |  |  | 		if (!capable(CAP_NET_ADMIN)) { | 
					
						
							|  |  |  | 			rc = -EPERM; | 
					
						
							|  |  |  | 			break; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		if (copy_from_user(&arp_entry, rq->ifr_ifru.ifru_data, | 
					
						
							|  |  |  | 				   sizeof(struct qeth_arp_cache_entry))) | 
					
						
							|  |  |  | 			rc = -EFAULT; | 
					
						
							|  |  |  | 		else | 
					
						
							|  |  |  | 			rc = qeth_l3_arp_remove_entry(card, &arp_entry); | 
					
						
							|  |  |  | 		break; | 
					
						
							|  |  |  | 	case SIOC_QETH_ARP_FLUSH_CACHE: | 
					
						
							|  |  |  | 		if (!capable(CAP_NET_ADMIN)) { | 
					
						
							|  |  |  | 			rc = -EPERM; | 
					
						
							|  |  |  | 			break; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		rc = qeth_l3_arp_flush_cache(card); | 
					
						
							|  |  |  | 		break; | 
					
						
							|  |  |  | 	case SIOC_QETH_ADP_SET_SNMP_CONTROL: | 
					
						
							|  |  |  | 		rc = qeth_snmp_command(card, rq->ifr_ifru.ifru_data); | 
					
						
							|  |  |  | 		break; | 
					
						
							|  |  |  | 	case SIOC_QETH_GET_CARD_TYPE: | 
					
						
							| 
									
										
										
										
											2010-05-16 21:15:14 +00:00
										 |  |  | 		if ((card->info.type == QETH_CARD_TYPE_OSD || | 
					
						
							|  |  |  | 		     card->info.type == QETH_CARD_TYPE_OSX) && | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 		    !card->info.guestlan) | 
					
						
							|  |  |  | 			return 1; | 
					
						
							|  |  |  | 		return 0; | 
					
						
							|  |  |  | 		break; | 
					
						
							|  |  |  | 	case SIOCGMIIPHY: | 
					
						
							|  |  |  | 		mii_data = if_mii(rq); | 
					
						
							|  |  |  | 		mii_data->phy_id = 0; | 
					
						
							|  |  |  | 		break; | 
					
						
							|  |  |  | 	case SIOCGMIIREG: | 
					
						
							|  |  |  | 		mii_data = if_mii(rq); | 
					
						
							|  |  |  | 		if (mii_data->phy_id != 0) | 
					
						
							|  |  |  | 			rc = -EINVAL; | 
					
						
							|  |  |  | 		else | 
					
						
							|  |  |  | 			mii_data->val_out = qeth_mdio_read(dev, | 
					
						
							|  |  |  | 							mii_data->phy_id, | 
					
						
							|  |  |  | 							mii_data->reg_num); | 
					
						
							|  |  |  | 		break; | 
					
						
							|  |  |  | 	default: | 
					
						
							|  |  |  | 		rc = -EOPNOTSUPP; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	if (rc) | 
					
						
							| 
									
										
										
										
											2010-06-21 22:57:05 +00:00
										 |  |  | 		QETH_CARD_TEXT_(card, 2, "ioce%d", rc); | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 	return rc; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-08-26 02:01:08 +00:00
										 |  |  | int inline qeth_l3_get_cast_type(struct qeth_card *card, struct sk_buff *skb) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	int cast_type = RTN_UNSPEC; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (skb_dst(skb) && skb_dst(skb)->neighbour) { | 
					
						
							|  |  |  | 		cast_type = skb_dst(skb)->neighbour->type; | 
					
						
							|  |  |  | 		if ((cast_type == RTN_BROADCAST) || | 
					
						
							|  |  |  | 		    (cast_type == RTN_MULTICAST) || | 
					
						
							|  |  |  | 		    (cast_type == RTN_ANYCAST)) | 
					
						
							|  |  |  | 			return cast_type; | 
					
						
							|  |  |  | 		else | 
					
						
							|  |  |  | 			return RTN_UNSPEC; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	/* try something else */ | 
					
						
							|  |  |  | 	if (skb->protocol == ETH_P_IPV6) | 
					
						
							|  |  |  | 		return (skb_network_header(skb)[24] == 0xff) ? | 
					
						
							|  |  |  | 				RTN_MULTICAST : 0; | 
					
						
							|  |  |  | 	else if (skb->protocol == ETH_P_IP) | 
					
						
							|  |  |  | 		return ((skb_network_header(skb)[16] & 0xf0) == 0xe0) ? | 
					
						
							|  |  |  | 				RTN_MULTICAST : 0; | 
					
						
							|  |  |  | 	/* ... */ | 
					
						
							|  |  |  | 	if (!memcmp(skb->data, skb->dev->broadcast, 6)) | 
					
						
							|  |  |  | 		return RTN_BROADCAST; | 
					
						
							|  |  |  | 	else { | 
					
						
							|  |  |  | 		u16 hdr_mac; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		hdr_mac = *((u16 *)skb->data); | 
					
						
							|  |  |  | 		/* tr multicast? */ | 
					
						
							|  |  |  | 		switch (card->info.link_type) { | 
					
						
							|  |  |  | 		case QETH_LINK_TYPE_HSTR: | 
					
						
							|  |  |  | 		case QETH_LINK_TYPE_LANE_TR: | 
					
						
							|  |  |  | 			if ((hdr_mac == QETH_TR_MAC_NC) || | 
					
						
							|  |  |  | 			    (hdr_mac == QETH_TR_MAC_C)) | 
					
						
							|  |  |  | 				return RTN_MULTICAST; | 
					
						
							|  |  |  | 			break; | 
					
						
							|  |  |  | 		/* eth or so multicast? */ | 
					
						
							|  |  |  | 		default: | 
					
						
							|  |  |  | 		if ((hdr_mac == QETH_ETH_MAC_V4) || | 
					
						
							|  |  |  | 			    (hdr_mac == QETH_ETH_MAC_V6)) | 
					
						
							|  |  |  | 				return RTN_MULTICAST; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return cast_type; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | static void qeth_l3_fill_header(struct qeth_card *card, struct qeth_hdr *hdr, | 
					
						
							|  |  |  | 		struct sk_buff *skb, int ipv, int cast_type) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	memset(hdr, 0, sizeof(struct qeth_hdr)); | 
					
						
							|  |  |  | 	hdr->hdr.l3.id = QETH_HEADER_TYPE_LAYER3; | 
					
						
							|  |  |  | 	hdr->hdr.l3.ext_flags = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/*
 | 
					
						
							|  |  |  | 	 * before we're going to overwrite this location with next hop ip. | 
					
						
							|  |  |  | 	 * v6 uses passthrough, v4 sets the tag in the QDIO header. | 
					
						
							|  |  |  | 	 */ | 
					
						
							|  |  |  | 	if (card->vlangrp && vlan_tx_tag_present(skb)) { | 
					
						
							| 
									
										
										
										
											2008-04-24 10:15:22 +02:00
										 |  |  | 		if ((ipv == 4) || (card->info.type == QETH_CARD_TYPE_IQD)) | 
					
						
							|  |  |  | 			hdr->hdr.l3.ext_flags = QETH_HDR_EXT_VLAN_FRAME; | 
					
						
							|  |  |  | 		else | 
					
						
							|  |  |  | 			hdr->hdr.l3.ext_flags = QETH_HDR_EXT_INCLUDE_VLAN_TAG; | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 		hdr->hdr.l3.vlan_id = vlan_tx_tag_get(skb); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	hdr->hdr.l3.length = skb->len - sizeof(struct qeth_hdr); | 
					
						
							|  |  |  | 	if (ipv == 4) { | 
					
						
							|  |  |  | 		/* IPv4 */ | 
					
						
							|  |  |  | 		hdr->hdr.l3.flags = qeth_l3_get_qeth_hdr_flags4(cast_type); | 
					
						
							|  |  |  | 		memset(hdr->hdr.l3.dest_addr, 0, 12); | 
					
						
							| 
									
										
										
										
											2009-06-02 05:19:30 +00:00
										 |  |  | 		if ((skb_dst(skb)) && (skb_dst(skb)->neighbour)) { | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 			*((u32 *) (&hdr->hdr.l3.dest_addr[12])) = | 
					
						
							| 
									
										
										
										
											2009-06-02 05:19:30 +00:00
										 |  |  | 			    *((u32 *) skb_dst(skb)->neighbour->primary_key); | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 		} else { | 
					
						
							|  |  |  | 			/* fill in destination address used in ip header */ | 
					
						
							|  |  |  | 			*((u32 *) (&hdr->hdr.l3.dest_addr[12])) = | 
					
						
							|  |  |  | 							ip_hdr(skb)->daddr; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} else if (ipv == 6) { | 
					
						
							|  |  |  | 		/* IPv6 */ | 
					
						
							|  |  |  | 		hdr->hdr.l3.flags = qeth_l3_get_qeth_hdr_flags6(cast_type); | 
					
						
							|  |  |  | 		if (card->info.type == QETH_CARD_TYPE_IQD) | 
					
						
							|  |  |  | 			hdr->hdr.l3.flags &= ~QETH_HDR_PASSTHRU; | 
					
						
							| 
									
										
										
										
											2009-06-02 05:19:30 +00:00
										 |  |  | 		if ((skb_dst(skb)) && (skb_dst(skb)->neighbour)) { | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 			memcpy(hdr->hdr.l3.dest_addr, | 
					
						
							| 
									
										
										
										
											2009-06-02 05:19:30 +00:00
										 |  |  | 			       skb_dst(skb)->neighbour->primary_key, 16); | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 		} else { | 
					
						
							|  |  |  | 			/* fill in destination address used in ip header */ | 
					
						
							|  |  |  | 			memcpy(hdr->hdr.l3.dest_addr, | 
					
						
							|  |  |  | 			       &ipv6_hdr(skb)->daddr, 16); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} else { | 
					
						
							|  |  |  | 		/* passthrough */ | 
					
						
							|  |  |  | 		if ((skb->dev->type == ARPHRD_IEEE802_TR) && | 
					
						
							|  |  |  | 			!memcmp(skb->data + sizeof(struct qeth_hdr) + | 
					
						
							|  |  |  | 			sizeof(__u16), skb->dev->broadcast, 6)) { | 
					
						
							|  |  |  | 			hdr->hdr.l3.flags = QETH_CAST_BROADCAST | | 
					
						
							|  |  |  | 						QETH_HDR_PASSTHRU; | 
					
						
							|  |  |  | 		} else if (!memcmp(skb->data + sizeof(struct qeth_hdr), | 
					
						
							|  |  |  | 			    skb->dev->broadcast, 6)) { | 
					
						
							|  |  |  | 			/* broadcast? */ | 
					
						
							|  |  |  | 			hdr->hdr.l3.flags = QETH_CAST_BROADCAST | | 
					
						
							|  |  |  | 						QETH_HDR_PASSTHRU; | 
					
						
							|  |  |  | 		} else { | 
					
						
							|  |  |  | 			hdr->hdr.l3.flags = (cast_type == RTN_MULTICAST) ? | 
					
						
							|  |  |  | 				QETH_CAST_MULTICAST | QETH_HDR_PASSTHRU : | 
					
						
							|  |  |  | 				QETH_CAST_UNICAST | QETH_HDR_PASSTHRU; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-05-11 19:34:45 +00:00
										 |  |  | static inline void qeth_l3_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.l3.ext_flags |= QETH_HDR_EXT_UDP; | 
					
						
							|  |  |  | 	hdr->hdr.l3.ext_flags |= QETH_HDR_EXT_CSUM_TRANSP_REQ; | 
					
						
							|  |  |  | 	if (card->options.performance_stats) | 
					
						
							|  |  |  | 		card->perf_stats.tx_csum++; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-03-24 20:57:16 +00:00
										 |  |  | static void qeth_tso_fill_header(struct qeth_card *card, | 
					
						
							|  |  |  | 		struct qeth_hdr *qhdr, struct sk_buff *skb) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct qeth_hdr_tso *hdr = (struct qeth_hdr_tso *)qhdr; | 
					
						
							|  |  |  | 	struct tcphdr *tcph = tcp_hdr(skb); | 
					
						
							|  |  |  | 	struct iphdr *iph = ip_hdr(skb); | 
					
						
							|  |  |  | 	struct ipv6hdr *ip6h = ipv6_hdr(skb); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/*fix header to TSO values ...*/ | 
					
						
							|  |  |  | 	hdr->hdr.hdr.l3.id = QETH_HEADER_TYPE_TSO; | 
					
						
							|  |  |  | 	/*set values which are fix for the first approach ...*/ | 
					
						
							|  |  |  | 	hdr->ext.hdr_tot_len = (__u16) sizeof(struct qeth_hdr_ext_tso); | 
					
						
							|  |  |  | 	hdr->ext.imb_hdr_no  = 1; | 
					
						
							|  |  |  | 	hdr->ext.hdr_type    = 1; | 
					
						
							|  |  |  | 	hdr->ext.hdr_version = 1; | 
					
						
							|  |  |  | 	hdr->ext.hdr_len     = 28; | 
					
						
							|  |  |  | 	/*insert non-fix values */ | 
					
						
							|  |  |  | 	hdr->ext.mss = skb_shinfo(skb)->gso_size; | 
					
						
							|  |  |  | 	hdr->ext.dg_hdr_len = (__u16)(iph->ihl*4 + tcph->doff*4); | 
					
						
							|  |  |  | 	hdr->ext.payload_len = (__u16)(skb->len - hdr->ext.dg_hdr_len - | 
					
						
							|  |  |  | 				       sizeof(struct qeth_hdr_tso)); | 
					
						
							|  |  |  | 	tcph->check = 0; | 
					
						
							|  |  |  | 	if (skb->protocol == ETH_P_IPV6) { | 
					
						
							|  |  |  | 		ip6h->payload_len = 0; | 
					
						
							|  |  |  | 		tcph->check = ~csum_ipv6_magic(&ip6h->saddr, &ip6h->daddr, | 
					
						
							|  |  |  | 					       0, IPPROTO_TCP, 0); | 
					
						
							|  |  |  | 	} else { | 
					
						
							|  |  |  | 		/*OSA want us to set these values ...*/ | 
					
						
							|  |  |  | 		tcph->check = ~csum_tcpudp_magic(iph->saddr, iph->daddr, | 
					
						
							|  |  |  | 					 0, IPPROTO_TCP, 0); | 
					
						
							|  |  |  | 		iph->tot_len = 0; | 
					
						
							|  |  |  | 		iph->check = 0; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-11-12 00:11:44 +00:00
										 |  |  | static inline int qeth_l3_tso_elements(struct sk_buff *skb) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	unsigned long tcpd = (unsigned long)tcp_hdr(skb) + | 
					
						
							|  |  |  | 		tcp_hdr(skb)->doff * 4; | 
					
						
							|  |  |  | 	int tcpd_len = skb->len - (tcpd - (unsigned long)skb->data); | 
					
						
							| 
									
										
										
										
											2010-06-21 22:57:10 +00:00
										 |  |  | 	int elements = PFN_UP(tcpd + tcpd_len - 1) - PFN_DOWN(tcpd); | 
					
						
							| 
									
										
										
										
											2009-11-12 00:11:44 +00:00
										 |  |  | 	elements += skb_shinfo(skb)->nr_frags; | 
					
						
							|  |  |  | 	return elements; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | static int qeth_l3_hard_start_xmit(struct sk_buff *skb, struct net_device *dev) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	int rc; | 
					
						
							|  |  |  | 	u16 *tag; | 
					
						
							|  |  |  | 	struct qeth_hdr *hdr = NULL; | 
					
						
							|  |  |  | 	int elements_needed = 0; | 
					
						
							| 
									
										
										
										
											2009-03-24 20:57:16 +00:00
										 |  |  | 	int elems; | 
					
						
							| 
									
										
										
										
											2008-07-26 02:24:10 -07:00
										 |  |  | 	struct qeth_card *card = dev->ml_priv; | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 	struct sk_buff *new_skb = NULL; | 
					
						
							|  |  |  | 	int ipv = qeth_get_ip_version(skb); | 
					
						
							| 
									
										
										
										
											2009-08-26 02:01:08 +00:00
										 |  |  | 	int cast_type = qeth_l3_get_cast_type(card, skb); | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 	struct qeth_qdio_out_q *queue = card->qdio.out_qs | 
					
						
							|  |  |  | 		[qeth_get_priority_queue(card, skb, ipv, cast_type)]; | 
					
						
							|  |  |  | 	int tx_bytes = skb->len; | 
					
						
							|  |  |  | 	enum qeth_large_send_types large_send = QETH_LARGE_SEND_NO; | 
					
						
							| 
									
										
										
										
											2008-08-01 16:39:13 +02:00
										 |  |  | 	int data_offset = -1; | 
					
						
							| 
									
										
										
										
											2009-03-24 20:57:16 +00:00
										 |  |  | 	int nr_frags; | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-03-08 20:36:54 +00:00
										 |  |  | 	if (((card->info.type == QETH_CARD_TYPE_IQD) && (!ipv)) || | 
					
						
							|  |  |  | 	     card->options.sniffer) | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 			goto tx_drop; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if ((card->state != CARD_STATE_UP) || !card->lan_online) { | 
					
						
							|  |  |  | 		card->stats.tx_carrier_errors++; | 
					
						
							|  |  |  | 		goto tx_drop; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if ((cast_type == RTN_BROADCAST) && | 
					
						
							|  |  |  | 	    (card->info.broadcast_capable == 0)) | 
					
						
							|  |  |  | 		goto tx_drop; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (card->options.performance_stats) { | 
					
						
							|  |  |  | 		card->perf_stats.outbound_cnt++; | 
					
						
							|  |  |  | 		card->perf_stats.outbound_start_time = qeth_get_micros(); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-08-01 16:39:13 +02:00
										 |  |  | 	if (skb_is_gso(skb)) | 
					
						
							|  |  |  | 		large_send = card->options.large_send; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if ((card->info.type == QETH_CARD_TYPE_IQD) && (!large_send) && | 
					
						
							|  |  |  | 	    (skb_shinfo(skb)->nr_frags == 0)) { | 
					
						
							|  |  |  | 		new_skb = skb; | 
					
						
							|  |  |  | 		data_offset = ETH_HLEN; | 
					
						
							|  |  |  | 		hdr = kmem_cache_alloc(qeth_core_header_cache, GFP_ATOMIC); | 
					
						
							|  |  |  | 		if (!hdr) | 
					
						
							|  |  |  | 			goto tx_drop; | 
					
						
							|  |  |  | 		elements_needed++; | 
					
						
							|  |  |  | 	} else { | 
					
						
							|  |  |  | 		/* create a clone with writeable headroom */ | 
					
						
							|  |  |  | 		new_skb = skb_realloc_headroom(skb, sizeof(struct qeth_hdr_tso) | 
					
						
							|  |  |  | 					+ VLAN_HLEN); | 
					
						
							|  |  |  | 		if (!new_skb) | 
					
						
							|  |  |  | 			goto tx_drop; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	if (card->info.type == QETH_CARD_TYPE_IQD) { | 
					
						
							| 
									
										
										
										
											2008-08-01 16:39:13 +02:00
										 |  |  | 		if (data_offset < 0) | 
					
						
							|  |  |  | 			skb_pull(new_skb, ETH_HLEN); | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 	} else { | 
					
						
							| 
									
										
										
										
											2010-03-08 20:36:54 +00:00
										 |  |  | 		if (ipv == 4) { | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 			if (card->dev->type == ARPHRD_IEEE802_TR) | 
					
						
							|  |  |  | 				skb_pull(new_skb, TR_HLEN); | 
					
						
							|  |  |  | 			else | 
					
						
							|  |  |  | 				skb_pull(new_skb, ETH_HLEN); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-03-08 20:36:54 +00:00
										 |  |  | 		if (ipv == 6 && card->vlangrp && | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 				vlan_tx_tag_present(new_skb)) { | 
					
						
							|  |  |  | 			skb_push(new_skb, VLAN_HLEN); | 
					
						
							|  |  |  | 			skb_copy_to_linear_data(new_skb, new_skb->data + 4, 4); | 
					
						
							|  |  |  | 			skb_copy_to_linear_data_offset(new_skb, 4, | 
					
						
							|  |  |  | 				new_skb->data + 8, 4); | 
					
						
							|  |  |  | 			skb_copy_to_linear_data_offset(new_skb, 8, | 
					
						
							|  |  |  | 				new_skb->data + 12, 4); | 
					
						
							|  |  |  | 			tag = (u16 *)(new_skb->data + 12); | 
					
						
							|  |  |  | 			*tag = __constant_htons(ETH_P_8021Q); | 
					
						
							|  |  |  | 			*(tag + 1) = htons(vlan_tx_tag_get(new_skb)); | 
					
						
							| 
									
										
										
										
											2008-07-22 09:39:25 +02:00
										 |  |  | 			new_skb->vlan_tci = 0; | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	netif_stop_queue(dev); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* fix hardware limitation: as long as we do not have sbal
 | 
					
						
							| 
									
										
										
										
											2009-03-24 20:57:16 +00:00
										 |  |  | 	 * chaining we can not send long frag lists | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 	 */ | 
					
						
							| 
									
										
										
										
											2009-11-12 00:11:44 +00:00
										 |  |  | 	if (large_send == QETH_LARGE_SEND_TSO) { | 
					
						
							|  |  |  | 		if (qeth_l3_tso_elements(new_skb) + 1 > 16) { | 
					
						
							|  |  |  | 			if (skb_linearize(new_skb)) | 
					
						
							|  |  |  | 				goto tx_drop; | 
					
						
							|  |  |  | 			if (card->options.performance_stats) | 
					
						
							|  |  |  | 				card->perf_stats.tx_lin++; | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2009-03-24 20:57:16 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	if ((large_send == QETH_LARGE_SEND_TSO) && | 
					
						
							|  |  |  | 	    (cast_type == RTN_UNSPEC)) { | 
					
						
							|  |  |  | 		hdr = (struct qeth_hdr *)skb_push(new_skb, | 
					
						
							|  |  |  | 						sizeof(struct qeth_hdr_tso)); | 
					
						
							|  |  |  | 		memset(hdr, 0, sizeof(struct qeth_hdr_tso)); | 
					
						
							|  |  |  | 		qeth_l3_fill_header(card, hdr, new_skb, ipv, cast_type); | 
					
						
							|  |  |  | 		qeth_tso_fill_header(card, hdr, new_skb); | 
					
						
							|  |  |  | 		elements_needed++; | 
					
						
							|  |  |  | 	} else { | 
					
						
							| 
									
										
										
										
											2008-08-01 16:39:13 +02:00
										 |  |  | 		if (data_offset < 0) { | 
					
						
							|  |  |  | 			hdr = (struct qeth_hdr *)skb_push(new_skb, | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 						sizeof(struct qeth_hdr)); | 
					
						
							| 
									
										
										
										
											2008-08-01 16:39:13 +02:00
										 |  |  | 			qeth_l3_fill_header(card, hdr, new_skb, ipv, | 
					
						
							|  |  |  | 						cast_type); | 
					
						
							|  |  |  | 		} else { | 
					
						
							|  |  |  | 			qeth_l3_fill_header(card, hdr, new_skb, ipv, | 
					
						
							|  |  |  | 						cast_type); | 
					
						
							|  |  |  | 			hdr->hdr.l3.length = new_skb->len - data_offset; | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2010-05-11 19:34:45 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 		if (skb->ip_summed == CHECKSUM_PARTIAL) | 
					
						
							|  |  |  | 			qeth_l3_hdr_csum(card, hdr, new_skb); | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-03-24 20:57:16 +00:00
										 |  |  | 	elems = qeth_get_elements_no(card, (void *)hdr, new_skb, | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 						 elements_needed); | 
					
						
							| 
									
										
										
										
											2009-03-24 20:57:16 +00:00
										 |  |  | 	if (!elems) { | 
					
						
							|  |  |  | 		if (data_offset >= 0) | 
					
						
							|  |  |  | 			kmem_cache_free(qeth_core_header_cache, hdr); | 
					
						
							|  |  |  | 		goto tx_drop; | 
					
						
							| 
									
										
										
										
											2009-03-24 20:57:15 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2009-03-24 20:57:16 +00:00
										 |  |  | 	elements_needed += elems; | 
					
						
							|  |  |  | 	nr_frags = skb_shinfo(new_skb)->nr_frags; | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-06-21 22:57:10 +00:00
										 |  |  | 	if (card->info.type != QETH_CARD_TYPE_IQD) { | 
					
						
							|  |  |  | 		int len; | 
					
						
							|  |  |  | 		if (large_send == QETH_LARGE_SEND_TSO) | 
					
						
							|  |  |  | 			len = ((unsigned long)tcp_hdr(new_skb) + | 
					
						
							|  |  |  | 				tcp_hdr(new_skb)->doff * 4) - | 
					
						
							|  |  |  | 				(unsigned long)new_skb->data; | 
					
						
							|  |  |  | 		else | 
					
						
							|  |  |  | 			len = sizeof(struct qeth_hdr_layer3); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		if (qeth_hdr_chk_and_bounce(new_skb, len)) | 
					
						
							|  |  |  | 			goto tx_drop; | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 		rc = qeth_do_send_packet(card, queue, new_skb, hdr, | 
					
						
							| 
									
										
										
										
											2009-03-24 20:57:16 +00:00
										 |  |  | 					 elements_needed); | 
					
						
							| 
									
										
										
										
											2010-06-21 22:57:10 +00:00
										 |  |  | 	} else | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 		rc = qeth_do_send_packet_fast(card, queue, new_skb, hdr, | 
					
						
							| 
									
										
										
										
											2009-03-24 20:57:16 +00:00
										 |  |  | 					elements_needed, data_offset, 0); | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	if (!rc) { | 
					
						
							|  |  |  | 		card->stats.tx_packets++; | 
					
						
							|  |  |  | 		card->stats.tx_bytes += tx_bytes; | 
					
						
							|  |  |  | 		if (new_skb != skb) | 
					
						
							|  |  |  | 			dev_kfree_skb_any(skb); | 
					
						
							|  |  |  | 		if (card->options.performance_stats) { | 
					
						
							|  |  |  | 			if (large_send != QETH_LARGE_SEND_NO) { | 
					
						
							|  |  |  | 				card->perf_stats.large_send_bytes += tx_bytes; | 
					
						
							|  |  |  | 				card->perf_stats.large_send_cnt++; | 
					
						
							|  |  |  | 			} | 
					
						
							| 
									
										
										
										
											2009-03-24 20:57:16 +00:00
										 |  |  | 			if (nr_frags) { | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 				card->perf_stats.sg_skbs_sent++; | 
					
						
							|  |  |  | 				/* nr_frags + skb->data */ | 
					
						
							| 
									
										
										
										
											2009-03-24 20:57:16 +00:00
										 |  |  | 				card->perf_stats.sg_frags_sent += nr_frags + 1; | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2009-07-05 19:23:38 -07:00
										 |  |  | 		rc = NETDEV_TX_OK; | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 	} else { | 
					
						
							| 
									
										
										
										
											2008-08-01 16:39:13 +02:00
										 |  |  | 		if (data_offset >= 0) | 
					
						
							|  |  |  | 			kmem_cache_free(qeth_core_header_cache, hdr); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01: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 12:37:48 +02:00
										 |  |  | 	netif_wake_queue(dev); | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 	return NETDEV_TX_OK; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int qeth_l3_open(struct net_device *dev) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2008-07-26 02:24:10 -07:00
										 |  |  | 	struct qeth_card *card = dev->ml_priv; | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-06-21 22:57:05 +00:00
										 |  |  | 	QETH_CARD_TEXT(card, 4, "qethopen"); | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 	if (card->state != CARD_STATE_SOFTSETUP) | 
					
						
							|  |  |  | 		return -ENODEV; | 
					
						
							|  |  |  | 	card->data.state = CH_STATE_UP; | 
					
						
							|  |  |  | 	card->state = CARD_STATE_UP; | 
					
						
							|  |  |  | 	netif_start_queue(dev); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (!card->lan_online && netif_carrier_ok(dev)) | 
					
						
							|  |  |  | 		netif_carrier_off(dev); | 
					
						
							|  |  |  | 	return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int qeth_l3_stop(struct net_device *dev) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2008-07-26 02:24:10 -07:00
										 |  |  | 	struct qeth_card *card = dev->ml_priv; | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-06-21 22:57:05 +00:00
										 |  |  | 	QETH_CARD_TEXT(card, 4, "qethstop"); | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 	netif_tx_disable(dev); | 
					
						
							|  |  |  | 	if (card->state == CARD_STATE_UP) | 
					
						
							|  |  |  | 		card->state = CARD_STATE_SOFTSETUP; | 
					
						
							|  |  |  | 	return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static u32 qeth_l3_ethtool_get_rx_csum(struct net_device *dev) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2008-07-26 02:24:10 -07:00
										 |  |  | 	struct qeth_card *card = dev->ml_priv; | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	return (card->options.checksum_type == HW_CHECKSUMMING); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int qeth_l3_ethtool_set_rx_csum(struct net_device *dev, u32 data) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2008-07-26 02:24:10 -07:00
										 |  |  | 	struct qeth_card *card = dev->ml_priv; | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 	enum qeth_checksum_types csum_type; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (data) | 
					
						
							|  |  |  | 		csum_type = HW_CHECKSUMMING; | 
					
						
							|  |  |  | 	else | 
					
						
							|  |  |  | 		csum_type = SW_CHECKSUMMING; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-11-12 00:11:45 +00:00
										 |  |  | 	return qeth_l3_set_rx_csum(card, csum_type); | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int qeth_l3_ethtool_set_tso(struct net_device *dev, u32 data) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2008-07-26 02:24:10 -07:00
										 |  |  | 	struct qeth_card *card = dev->ml_priv; | 
					
						
							| 
									
										
										
										
											2009-11-12 00:11:44 +00:00
										 |  |  | 	int rc = 0; | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	if (data) { | 
					
						
							| 
									
										
										
										
											2009-11-12 00:11:44 +00:00
										 |  |  | 		rc = qeth_l3_set_large_send(card, QETH_LARGE_SEND_TSO); | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 	} else { | 
					
						
							|  |  |  | 		dev->features &= ~NETIF_F_TSO; | 
					
						
							|  |  |  | 		card->options.large_send = QETH_LARGE_SEND_NO; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2009-11-12 00:11:44 +00:00
										 |  |  | 	return rc; | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-05-11 19:34:45 +00:00
										 |  |  | static int qeth_l3_ethtool_set_tx_csum(struct net_device *dev, u32 data) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct qeth_card *card = dev->ml_priv; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (data) { | 
					
						
							|  |  |  | 		if (qeth_is_supported(card, IPA_OUTBOUND_CHECKSUM)) | 
					
						
							|  |  |  | 			dev->features |= NETIF_F_IP_CSUM; | 
					
						
							|  |  |  | 		else | 
					
						
							|  |  |  | 			return -EPERM; | 
					
						
							|  |  |  | 	} else | 
					
						
							|  |  |  | 		dev->features &= ~NETIF_F_IP_CSUM; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-09-02 01:03:33 -07:00
										 |  |  | static const struct ethtool_ops qeth_l3_ethtool_ops = { | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 	.get_link = ethtool_op_get_link, | 
					
						
							|  |  |  | 	.get_tx_csum = ethtool_op_get_tx_csum, | 
					
						
							| 
									
										
										
										
											2010-05-11 19:34:45 +00:00
										 |  |  | 	.set_tx_csum = qeth_l3_ethtool_set_tx_csum, | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 	.get_rx_csum = qeth_l3_ethtool_get_rx_csum, | 
					
						
							|  |  |  | 	.set_rx_csum = qeth_l3_ethtool_set_rx_csum, | 
					
						
							|  |  |  | 	.get_sg      = ethtool_op_get_sg, | 
					
						
							|  |  |  | 	.set_sg      = ethtool_op_set_sg, | 
					
						
							|  |  |  | 	.get_tso     = ethtool_op_get_tso, | 
					
						
							|  |  |  | 	.set_tso     = qeth_l3_ethtool_set_tso, | 
					
						
							|  |  |  | 	.get_strings = qeth_core_get_strings, | 
					
						
							|  |  |  | 	.get_ethtool_stats = qeth_core_get_ethtool_stats, | 
					
						
							| 
									
										
										
										
											2009-10-01 11:24:32 +00:00
										 |  |  | 	.get_sset_count = qeth_core_get_sset_count, | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 	.get_drvinfo = qeth_core_get_drvinfo, | 
					
						
							| 
									
										
										
										
											2008-04-24 10:15:23 +02:00
										 |  |  | 	.get_settings = qeth_core_ethtool_get_settings, | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /*
 | 
					
						
							|  |  |  |  * we need NOARP for IPv4 but we want neighbor solicitation for IPv6. Setting | 
					
						
							|  |  |  |  * NOARP on the netdevice is no option because it also turns off neighbor | 
					
						
							|  |  |  |  * solicitation. For IPv4 we install a neighbor_setup function. We don't want | 
					
						
							|  |  |  |  * arp resolution but we want the hard header (packet socket will work | 
					
						
							|  |  |  |  * e.g. tcpdump) | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | static int qeth_l3_neigh_setup_noarp(struct neighbour *n) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	n->nud_state = NUD_NOARP; | 
					
						
							|  |  |  | 	memcpy(n->ha, "FAKELL", 6); | 
					
						
							|  |  |  | 	n->output = n->ops->connected_output; | 
					
						
							|  |  |  | 	return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int | 
					
						
							|  |  |  | qeth_l3_neigh_setup(struct net_device *dev, struct neigh_parms *np) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	if (np->tbl->family == AF_INET) | 
					
						
							|  |  |  | 		np->neigh_setup = qeth_l3_neigh_setup_noarp; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-01-09 03:44:00 +00:00
										 |  |  | static const struct net_device_ops qeth_l3_netdev_ops = { | 
					
						
							| 
									
										
										
										
											2009-01-08 10:50:55 -08:00
										 |  |  | 	.ndo_open		= qeth_l3_open, | 
					
						
							|  |  |  | 	.ndo_stop		= qeth_l3_stop, | 
					
						
							|  |  |  | 	.ndo_get_stats		= qeth_get_stats, | 
					
						
							|  |  |  | 	.ndo_start_xmit		= qeth_l3_hard_start_xmit, | 
					
						
							|  |  |  | 	.ndo_validate_addr	= eth_validate_addr, | 
					
						
							|  |  |  | 	.ndo_set_multicast_list = qeth_l3_set_multicast_list, | 
					
						
							|  |  |  | 	.ndo_do_ioctl	   	= qeth_l3_do_ioctl, | 
					
						
							|  |  |  | 	.ndo_change_mtu	   	= qeth_change_mtu, | 
					
						
							|  |  |  | 	.ndo_vlan_rx_register	= qeth_l3_vlan_rx_register, | 
					
						
							|  |  |  | 	.ndo_vlan_rx_add_vid	= qeth_l3_vlan_rx_add_vid, | 
					
						
							|  |  |  | 	.ndo_vlan_rx_kill_vid   = qeth_l3_vlan_rx_kill_vid, | 
					
						
							|  |  |  | 	.ndo_tx_timeout	   	= qeth_tx_timeout, | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-01-09 03:44:00 +00:00
										 |  |  | static const struct net_device_ops qeth_l3_osa_netdev_ops = { | 
					
						
							|  |  |  | 	.ndo_open		= qeth_l3_open, | 
					
						
							|  |  |  | 	.ndo_stop		= qeth_l3_stop, | 
					
						
							|  |  |  | 	.ndo_get_stats		= qeth_get_stats, | 
					
						
							|  |  |  | 	.ndo_start_xmit		= qeth_l3_hard_start_xmit, | 
					
						
							|  |  |  | 	.ndo_validate_addr	= eth_validate_addr, | 
					
						
							|  |  |  | 	.ndo_set_multicast_list = qeth_l3_set_multicast_list, | 
					
						
							|  |  |  | 	.ndo_do_ioctl	   	= qeth_l3_do_ioctl, | 
					
						
							|  |  |  | 	.ndo_change_mtu	   	= qeth_change_mtu, | 
					
						
							|  |  |  | 	.ndo_vlan_rx_register	= qeth_l3_vlan_rx_register, | 
					
						
							|  |  |  | 	.ndo_vlan_rx_add_vid	= qeth_l3_vlan_rx_add_vid, | 
					
						
							|  |  |  | 	.ndo_vlan_rx_kill_vid   = qeth_l3_vlan_rx_kill_vid, | 
					
						
							|  |  |  | 	.ndo_tx_timeout	   	= qeth_tx_timeout, | 
					
						
							|  |  |  | 	.ndo_neigh_setup	= qeth_l3_neigh_setup, | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | static int qeth_l3_setup_netdev(struct qeth_card *card) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2010-05-16 21:15:14 +00:00
										 |  |  | 	if (card->info.type == QETH_CARD_TYPE_OSD || | 
					
						
							|  |  |  | 	    card->info.type == QETH_CARD_TYPE_OSX) { | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 		if ((card->info.link_type == QETH_LINK_TYPE_LANE_TR) || | 
					
						
							|  |  |  | 		    (card->info.link_type == QETH_LINK_TYPE_HSTR)) { | 
					
						
							|  |  |  | #ifdef CONFIG_TR
 | 
					
						
							|  |  |  | 			card->dev = alloc_trdev(0); | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 			if (!card->dev) | 
					
						
							|  |  |  | 				return -ENODEV; | 
					
						
							| 
									
										
										
										
											2009-01-09 03:44:00 +00:00
										 |  |  | 			card->dev->netdev_ops = &qeth_l3_netdev_ops; | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 		} else { | 
					
						
							|  |  |  | 			card->dev = alloc_etherdev(0); | 
					
						
							|  |  |  | 			if (!card->dev) | 
					
						
							|  |  |  | 				return -ENODEV; | 
					
						
							| 
									
										
										
										
											2009-01-09 03:44:00 +00:00
										 |  |  | 			card->dev->netdev_ops = &qeth_l3_osa_netdev_ops; | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 			/*IPv6 address autoconfiguration stuff*/ | 
					
						
							|  |  |  | 			qeth_l3_get_unique_id(card); | 
					
						
							|  |  |  | 			if (!(card->info.unique_id & UNIQUE_ID_NOT_BY_CARD)) | 
					
						
							|  |  |  | 				card->dev->dev_id = card->info.unique_id & | 
					
						
							|  |  |  | 							 0xffff; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} else if (card->info.type == QETH_CARD_TYPE_IQD) { | 
					
						
							|  |  |  | 		card->dev = alloc_netdev(0, "hsi%d", ether_setup); | 
					
						
							|  |  |  | 		if (!card->dev) | 
					
						
							|  |  |  | 			return -ENODEV; | 
					
						
							|  |  |  | 		card->dev->flags |= IFF_NOARP; | 
					
						
							| 
									
										
										
										
											2009-01-09 03:44:00 +00:00
										 |  |  | 		card->dev->netdev_ops = &qeth_l3_netdev_ops; | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 		qeth_l3_iqd_read_initial_mac(card); | 
					
						
							|  |  |  | 	} else | 
					
						
							|  |  |  | 		return -ENODEV; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-07-26 02:24:10 -07:00
										 |  |  | 	card->dev->ml_priv = card; | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 	card->dev->watchdog_timeo = QETH_TX_TIMEOUT; | 
					
						
							|  |  |  | 	card->dev->mtu = card->info.initial_mtu; | 
					
						
							|  |  |  | 	SET_ETHTOOL_OPS(card->dev, &qeth_l3_ethtool_ops); | 
					
						
							|  |  |  | 	card->dev->features |=	NETIF_F_HW_VLAN_TX | | 
					
						
							|  |  |  | 				NETIF_F_HW_VLAN_RX | | 
					
						
							|  |  |  | 				NETIF_F_HW_VLAN_FILTER; | 
					
						
							| 
									
										
										
										
											2009-05-30 23:04:46 -07:00
										 |  |  | 	card->dev->priv_flags &= ~IFF_XMIT_DST_RELEASE; | 
					
						
							| 
									
										
										
										
											2009-11-12 00:11:44 +00:00
										 |  |  | 	card->dev->gso_max_size = 15 * PAGE_SIZE; | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	SET_NETDEV_DEV(card->dev, &card->gdev->dev); | 
					
						
							|  |  |  | 	return register_netdev(card->dev); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void qeth_l3_qdio_input_handler(struct ccw_device *ccwdev, | 
					
						
							| 
									
										
										
										
											2008-07-17 17:16:48 +02:00
										 |  |  | 		unsigned int qdio_err, unsigned int queue, int first_element, | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 		int count, unsigned long card_ptr) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct net_device *net_dev; | 
					
						
							|  |  |  | 	struct qeth_card *card; | 
					
						
							|  |  |  | 	struct qeth_qdio_buffer *buffer; | 
					
						
							|  |  |  | 	int index; | 
					
						
							|  |  |  | 	int i; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	card = (struct qeth_card *) card_ptr; | 
					
						
							|  |  |  | 	net_dev = card->dev; | 
					
						
							|  |  |  | 	if (card->options.performance_stats) { | 
					
						
							|  |  |  | 		card->perf_stats.inbound_cnt++; | 
					
						
							|  |  |  | 		card->perf_stats.inbound_start_time = qeth_get_micros(); | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2008-07-17 17:16:48 +02:00
										 |  |  | 	if (qdio_err & QDIO_ERROR_ACTIVATE_CHECK_CONDITION) { | 
					
						
							| 
									
										
										
										
											2010-06-21 22:57:05 +00:00
										 |  |  | 		QETH_CARD_TEXT(card, 1, "qdinchk"); | 
					
						
							|  |  |  | 		QETH_CARD_TEXT_(card, 1, "%04X%04X", | 
					
						
							| 
									
										
										
										
											2008-07-17 17:16:48 +02:00
										 |  |  | 				first_element, count); | 
					
						
							| 
									
										
										
										
											2010-06-21 22:57:05 +00:00
										 |  |  | 		QETH_CARD_TEXT_(card, 1, "%04X", queue); | 
					
						
							| 
									
										
										
										
											2008-07-17 17:16:48 +02:00
										 |  |  | 		qeth_schedule_recovery(card); | 
					
						
							|  |  |  | 		return; | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 	} | 
					
						
							|  |  |  | 	for (i = first_element; i < (first_element + count); ++i) { | 
					
						
							|  |  |  | 		index = i % QDIO_MAX_BUFFERS_PER_Q; | 
					
						
							|  |  |  | 		buffer = &card->qdio.in_q->bufs[index]; | 
					
						
							| 
									
										
										
										
											2008-07-17 17:16:48 +02:00
										 |  |  | 		if (!(qdio_err && | 
					
						
							| 
									
										
										
										
											2010-01-11 02:50:50 +00:00
										 |  |  | 		      qeth_check_qdio_errors(card, buffer->buffer, | 
					
						
							| 
									
										
										
										
											2008-07-17 17:16:48 +02:00
										 |  |  | 					     qdio_err, "qinerr"))) | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 			qeth_l3_process_inbound_buffer(card, buffer, index); | 
					
						
							|  |  |  | 		/* clear buffer and give back to hardware */ | 
					
						
							|  |  |  | 		qeth_put_buffer_pool_entry(card, buffer->pool_entry); | 
					
						
							|  |  |  | 		qeth_queue_input_buffer(card, index); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	if (card->options.performance_stats) | 
					
						
							|  |  |  | 		card->perf_stats.inbound_time += qeth_get_micros() - | 
					
						
							|  |  |  | 			card->perf_stats.inbound_start_time; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int qeth_l3_probe_device(struct ccwgroup_device *gdev) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct qeth_card *card = dev_get_drvdata(&gdev->dev); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	qeth_l3_create_device_attributes(&gdev->dev); | 
					
						
							|  |  |  | 	card->options.layer2 = 0; | 
					
						
							|  |  |  | 	card->discipline.input_handler = (qdio_handler_t *) | 
					
						
							|  |  |  | 		qeth_l3_qdio_input_handler; | 
					
						
							|  |  |  | 	card->discipline.output_handler = (qdio_handler_t *) | 
					
						
							|  |  |  | 		qeth_qdio_output_handler; | 
					
						
							|  |  |  | 	card->discipline.recover = qeth_l3_recover; | 
					
						
							|  |  |  | 	return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void qeth_l3_remove_device(struct ccwgroup_device *cgdev) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct qeth_card *card = dev_get_drvdata(&cgdev->dev); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-07-22 23:15:05 +00:00
										 |  |  | 	qeth_l3_remove_device_attributes(&cgdev->dev); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-05-19 21:38:37 +00:00
										 |  |  | 	qeth_set_allowed_threads(card, 0, 1); | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 	wait_event(card->wait_q, qeth_threads_running(card, 0xffffffff) == 0); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (cgdev->state == CCWGROUP_ONLINE) { | 
					
						
							|  |  |  | 		card->use_hard_stop = 1; | 
					
						
							|  |  |  | 		qeth_l3_set_offline(cgdev); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (card->dev) { | 
					
						
							|  |  |  | 		unregister_netdev(card->dev); | 
					
						
							|  |  |  | 		card->dev = NULL; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	qeth_l3_clear_ip_list(card, 0, 0); | 
					
						
							|  |  |  | 	qeth_l3_clear_ipato_list(card); | 
					
						
							|  |  |  | 	return; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int __qeth_l3_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; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	BUG_ON(!card); | 
					
						
							| 
									
										
										
										
											2010-07-22 23:15:05 +00:00
										 |  |  | 	mutex_lock(&card->discipline_mutex); | 
					
						
							| 
									
										
										
										
											2010-05-11 19:34:47 +00:00
										 |  |  | 	mutex_lock(&card->conf_mutex); | 
					
						
							| 
									
										
										
										
											2008-04-01 10:26:58 +02:00
										 |  |  | 	QETH_DBF_TEXT(SETUP, 2, "setonlin"); | 
					
						
							|  |  |  | 	QETH_DBF_HEX(SETUP, 2, &card, sizeof(void *)); | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	recover_flag = card->state; | 
					
						
							|  |  |  | 	rc = qeth_core_hardsetup_card(card); | 
					
						
							|  |  |  | 	if (rc) { | 
					
						
							| 
									
										
										
										
											2008-04-01 10:26:58 +02:00
										 |  |  | 		QETH_DBF_TEXT_(SETUP, 2, "2err%d", rc); | 
					
						
							| 
									
										
										
										
											2009-11-12 00:11:43 +00:00
										 |  |  | 		rc = -ENODEV; | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 		goto out_remove; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	qeth_l3_query_ipassists(card, QETH_PROT_IPV4); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-11-12 00:11:43 +00:00
										 |  |  | 	if (!card->dev && qeth_l3_setup_netdev(card)) { | 
					
						
							|  |  |  | 		rc = -ENODEV; | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 		goto out_remove; | 
					
						
							| 
									
										
										
										
											2009-11-12 00:11:43 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	card->state = CARD_STATE_HARDSETUP; | 
					
						
							|  |  |  | 	qeth_print_status_message(card); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* softsetup */ | 
					
						
							| 
									
										
										
										
											2008-04-01 10:26:58 +02:00
										 |  |  | 	QETH_DBF_TEXT(SETUP, 2, "softsetp"); | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	rc = qeth_send_startlan(card); | 
					
						
							|  |  |  | 	if (rc) { | 
					
						
							| 
									
										
										
										
											2008-04-01 10:26:58 +02:00
										 |  |  | 		QETH_DBF_TEXT_(SETUP, 2, "1err%d", rc); | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 		if (rc == 0xe080) { | 
					
						
							| 
									
										
										
										
											2008-12-25 13:39:49 +01:00
										 |  |  | 			dev_warn(&card->gdev->dev, | 
					
						
							|  |  |  | 				"The LAN is offline\n"); | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 			card->lan_online = 0; | 
					
						
							| 
									
										
										
										
											2010-05-11 19:34:47 +00:00
										 |  |  | 			goto out; | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2009-11-12 00:11:43 +00:00
										 |  |  | 		rc = -ENODEV; | 
					
						
							| 
									
										
										
										
											2009-05-19 21:38:37 +00:00
										 |  |  | 		goto out_remove; | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 	} else | 
					
						
							|  |  |  | 		card->lan_online = 1; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	rc = qeth_l3_setadapter_parms(card); | 
					
						
							|  |  |  | 	if (rc) | 
					
						
							| 
									
										
										
										
											2008-04-01 10:26:58 +02:00
										 |  |  | 		QETH_DBF_TEXT_(SETUP, 2, "2err%d", rc); | 
					
						
							| 
									
										
										
										
											2010-01-11 02:50:50 +00:00
										 |  |  | 	if (!card->options.sniffer) { | 
					
						
							|  |  |  | 		rc = qeth_l3_start_ipassists(card); | 
					
						
							|  |  |  | 		if (rc) | 
					
						
							|  |  |  | 			QETH_DBF_TEXT_(SETUP, 2, "3err%d", rc); | 
					
						
							|  |  |  | 		qeth_l3_set_large_send(card, card->options.large_send); | 
					
						
							|  |  |  | 		rc = qeth_l3_setrouting_v4(card); | 
					
						
							|  |  |  | 		if (rc) | 
					
						
							|  |  |  | 			QETH_DBF_TEXT_(SETUP, 2, "4err%d", rc); | 
					
						
							|  |  |  | 		rc = qeth_l3_setrouting_v6(card); | 
					
						
							|  |  |  | 		if (rc) | 
					
						
							|  |  |  | 			QETH_DBF_TEXT_(SETUP, 2, "5err%d", rc); | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 	netif_tx_disable(card->dev); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	rc = qeth_init_qdio_queues(card); | 
					
						
							|  |  |  | 	if (rc) { | 
					
						
							| 
									
										
										
										
											2008-04-01 10:26:58 +02:00
										 |  |  | 		QETH_DBF_TEXT_(SETUP, 2, "6err%d", rc); | 
					
						
							| 
									
										
										
										
											2009-11-12 00:11:43 +00:00
										 |  |  | 		rc = -ENODEV; | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 		goto out_remove; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	card->state = CARD_STATE_SOFTSETUP; | 
					
						
							|  |  |  | 	netif_carrier_on(card->dev); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	qeth_set_allowed_threads(card, 0xffffffff, 0); | 
					
						
							| 
									
										
										
										
											2009-08-26 02:01:07 +00:00
										 |  |  | 	qeth_l3_set_ip_addr_list(card); | 
					
						
							| 
									
										
										
										
											2008-04-24 10:15:25 +02:00
										 |  |  | 	if (recover_flag == CARD_STATE_RECOVER) { | 
					
						
							|  |  |  | 		if (recovery_mode) | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 			qeth_l3_open(card->dev); | 
					
						
							| 
									
										
										
										
											2008-04-24 10:15:25 +02:00
										 |  |  | 		else { | 
					
						
							|  |  |  | 			rtnl_lock(); | 
					
						
							|  |  |  | 			dev_open(card->dev); | 
					
						
							|  |  |  | 			rtnl_unlock(); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		qeth_l3_set_multicast_list(card->dev); | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 	} | 
					
						
							|  |  |  | 	/* let user_space know that device is online */ | 
					
						
							|  |  |  | 	kobject_uevent(&gdev->dev.kobj, KOBJ_CHANGE); | 
					
						
							| 
									
										
										
										
											2010-05-11 19:34:47 +00:00
										 |  |  | out: | 
					
						
							|  |  |  | 	mutex_unlock(&card->conf_mutex); | 
					
						
							| 
									
										
										
										
											2010-07-22 23:15:05 +00:00
										 |  |  | 	mutex_unlock(&card->discipline_mutex); | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 	return 0; | 
					
						
							|  |  |  | out_remove: | 
					
						
							|  |  |  | 	card->use_hard_stop = 1; | 
					
						
							|  |  |  | 	qeth_l3_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)); | 
					
						
							|  |  |  | 	if (recover_flag == CARD_STATE_RECOVER) | 
					
						
							|  |  |  | 		card->state = CARD_STATE_RECOVER; | 
					
						
							|  |  |  | 	else | 
					
						
							|  |  |  | 		card->state = CARD_STATE_DOWN; | 
					
						
							| 
									
										
										
										
											2010-05-11 19:34:47 +00:00
										 |  |  | 	mutex_unlock(&card->conf_mutex); | 
					
						
							| 
									
										
										
										
											2010-07-22 23:15:05 +00:00
										 |  |  | 	mutex_unlock(&card->discipline_mutex); | 
					
						
							| 
									
										
										
										
											2009-11-12 00:11:43 +00:00
										 |  |  | 	return rc; | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int qeth_l3_set_online(struct ccwgroup_device *gdev) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	return __qeth_l3_set_online(gdev, 0); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int __qeth_l3_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-22 23:15:05 +00:00
										 |  |  | 	mutex_lock(&card->discipline_mutex); | 
					
						
							| 
									
										
										
										
											2010-05-11 19:34:47 +00:00
										 |  |  | 	mutex_lock(&card->conf_mutex); | 
					
						
							| 
									
										
										
										
											2008-04-01 10:26:58 +02:00
										 |  |  | 	QETH_DBF_TEXT(SETUP, 3, "setoffl"); | 
					
						
							|  |  |  | 	QETH_DBF_HEX(SETUP, 3, &card, sizeof(void *)); | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	if (card->dev && netif_carrier_ok(card->dev)) | 
					
						
							|  |  |  | 		netif_carrier_off(card->dev); | 
					
						
							|  |  |  | 	recover_flag = card->state; | 
					
						
							| 
									
										
										
										
											2008-10-24 11:16:52 +02:00
										 |  |  | 	qeth_l3_stop_card(card, recovery_mode); | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01: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 10:26:58 +02:00
										 |  |  | 		QETH_DBF_TEXT_(SETUP, 2, "1err%d", rc); | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01: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-11 19:34:47 +00:00
										 |  |  | 	mutex_unlock(&card->conf_mutex); | 
					
						
							| 
									
										
										
										
											2010-07-22 23:15:05 +00:00
										 |  |  | 	mutex_unlock(&card->discipline_mutex); | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 	return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int qeth_l3_set_offline(struct ccwgroup_device *cgdev) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	return __qeth_l3_set_offline(cgdev, 0); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int qeth_l3_recover(void *ptr) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct qeth_card *card; | 
					
						
							|  |  |  | 	int rc = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	card = (struct qeth_card *) ptr; | 
					
						
							| 
									
										
										
										
											2010-06-21 22:57:05 +00:00
										 |  |  | 	QETH_CARD_TEXT(card, 2, "recover1"); | 
					
						
							|  |  |  | 	QETH_CARD_HEX(card, 2, &card, sizeof(void *)); | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 	if (!qeth_do_run_thread(card, QETH_RECOVER_THREAD)) | 
					
						
							|  |  |  | 		return 0; | 
					
						
							| 
									
										
										
										
											2010-06-21 22:57:05 +00:00
										 |  |  | 	QETH_CARD_TEXT(card, 2, "recover2"); | 
					
						
							| 
									
										
										
										
											2008-12-25 13:39:49 +01:00
										 |  |  | 	dev_warn(&card->gdev->dev, | 
					
						
							|  |  |  | 		"A recovery process has been started for the device\n"); | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 	card->use_hard_stop = 1; | 
					
						
							|  |  |  | 	__qeth_l3_set_offline(card->gdev, 1); | 
					
						
							|  |  |  | 	rc = __qeth_l3_set_online(card->gdev, 1); | 
					
						
							|  |  |  | 	/* don't run another scheduled recovery */ | 
					
						
							|  |  |  | 	qeth_clear_thread_start_bit(card, QETH_RECOVER_THREAD); | 
					
						
							|  |  |  | 	qeth_clear_thread_running_bit(card, QETH_RECOVER_THREAD); | 
					
						
							|  |  |  | 	if (!rc) | 
					
						
							| 
									
										
										
										
											2008-12-25 13:39:49 +01:00
										 |  |  | 		dev_info(&card->gdev->dev, | 
					
						
							|  |  |  | 			"Device successfully recovered!\n"); | 
					
						
							| 
									
										
										
										
											2008-09-19 12:56:03 +02:00
										 |  |  | 	else { | 
					
						
							|  |  |  | 		rtnl_lock(); | 
					
						
							|  |  |  | 		dev_close(card->dev); | 
					
						
							|  |  |  | 		rtnl_unlock(); | 
					
						
							| 
									
										
										
										
											2008-12-25 13:39:49 +01:00
										 |  |  | 		dev_warn(&card->gdev->dev, "The qeth device driver " | 
					
						
							|  |  |  | 			"failed to recover an error on the device\n"); | 
					
						
							| 
									
										
										
										
											2008-09-19 12:56:03 +02:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 	return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void qeth_l3_shutdown(struct ccwgroup_device *gdev) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct qeth_card *card = dev_get_drvdata(&gdev->dev); | 
					
						
							|  |  |  | 	qeth_l3_clear_ip_list(card, 0, 0); | 
					
						
							|  |  |  | 	qeth_qdio_clear_card(card, 0); | 
					
						
							|  |  |  | 	qeth_clear_qdio_buffers(card); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-06-16 10:30:31 +02:00
										 |  |  | static int qeth_l3_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) { | 
					
						
							|  |  |  | 		card->use_hard_stop = 1; | 
					
						
							|  |  |  | 		__qeth_l3_set_offline(card->gdev, 1); | 
					
						
							|  |  |  | 	} else | 
					
						
							|  |  |  | 		__qeth_l3_set_offline(card->gdev, 0); | 
					
						
							|  |  |  | 	return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int qeth_l3_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_l3_set_online(card->gdev, 1); | 
					
						
							|  |  |  | 		if (rc) { | 
					
						
							| 
									
										
										
										
											2010-03-08 20:36:56 +00:00
										 |  |  | 			rtnl_lock(); | 
					
						
							|  |  |  | 			dev_close(card->dev); | 
					
						
							|  |  |  | 			rtnl_unlock(); | 
					
						
							| 
									
										
										
										
											2009-06-16 10:30:31 +02:00
										 |  |  | 		} | 
					
						
							|  |  |  | 	} else | 
					
						
							|  |  |  | 		rc = __qeth_l3_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; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | struct ccwgroup_driver qeth_l3_ccwgroup_driver = { | 
					
						
							|  |  |  | 	.probe = qeth_l3_probe_device, | 
					
						
							|  |  |  | 	.remove = qeth_l3_remove_device, | 
					
						
							|  |  |  | 	.set_online = qeth_l3_set_online, | 
					
						
							|  |  |  | 	.set_offline = qeth_l3_set_offline, | 
					
						
							|  |  |  | 	.shutdown = qeth_l3_shutdown, | 
					
						
							| 
									
										
										
										
											2009-06-16 10:30:31 +02:00
										 |  |  | 	.freeze = qeth_l3_pm_suspend, | 
					
						
							|  |  |  | 	.thaw = qeth_l3_pm_resume, | 
					
						
							|  |  |  | 	.restore = qeth_l3_pm_resume, | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | }; | 
					
						
							|  |  |  | EXPORT_SYMBOL_GPL(qeth_l3_ccwgroup_driver); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int qeth_l3_ip_event(struct notifier_block *this, | 
					
						
							| 
									
										
										
										
											2008-03-22 18:22:42 -07:00
										 |  |  | 			    unsigned long event, void *ptr) | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | { | 
					
						
							|  |  |  | 	struct in_ifaddr *ifa = (struct in_ifaddr *)ptr; | 
					
						
							|  |  |  | 	struct net_device *dev = (struct net_device *)ifa->ifa_dev->dev; | 
					
						
							|  |  |  | 	struct qeth_ipaddr *addr; | 
					
						
							|  |  |  | 	struct qeth_card *card; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-03-25 21:47:49 +09:00
										 |  |  | 	if (dev_net(dev) != &init_net) | 
					
						
							| 
									
										
										
										
											2008-03-22 18:22:42 -07:00
										 |  |  | 		return NOTIFY_DONE; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 	card = qeth_l3_get_card_from_dev(dev); | 
					
						
							| 
									
										
										
										
											2010-06-21 22:57:05 +00:00
										 |  |  | 	QETH_CARD_TEXT(card, 3, "ipevent"); | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 	if (!card) | 
					
						
							|  |  |  | 		return NOTIFY_DONE; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	addr = qeth_l3_get_addr_buffer(QETH_PROT_IPV4); | 
					
						
							|  |  |  | 	if (addr != NULL) { | 
					
						
							|  |  |  | 		addr->u.a4.addr = ifa->ifa_address; | 
					
						
							|  |  |  | 		addr->u.a4.mask = ifa->ifa_mask; | 
					
						
							|  |  |  | 		addr->type = QETH_IP_TYPE_NORMAL; | 
					
						
							|  |  |  | 	} else | 
					
						
							|  |  |  | 		goto out; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	switch (event) { | 
					
						
							|  |  |  | 	case NETDEV_UP: | 
					
						
							|  |  |  | 		if (!qeth_l3_add_ip(card, addr)) | 
					
						
							|  |  |  | 			kfree(addr); | 
					
						
							|  |  |  | 		break; | 
					
						
							|  |  |  | 	case NETDEV_DOWN: | 
					
						
							|  |  |  | 		if (!qeth_l3_delete_ip(card, addr)) | 
					
						
							|  |  |  | 			kfree(addr); | 
					
						
							|  |  |  | 		break; | 
					
						
							|  |  |  | 	default: | 
					
						
							|  |  |  | 		break; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	qeth_l3_set_ip_addr_list(card); | 
					
						
							|  |  |  | out: | 
					
						
							|  |  |  | 	return NOTIFY_DONE; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static struct notifier_block qeth_l3_ip_notifier = { | 
					
						
							|  |  |  | 	qeth_l3_ip_event, | 
					
						
							|  |  |  | 	NULL, | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #ifdef CONFIG_QETH_IPV6
 | 
					
						
							|  |  |  | /**
 | 
					
						
							|  |  |  |  * IPv6 event handler | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | static int qeth_l3_ip6_event(struct notifier_block *this, | 
					
						
							| 
									
										
										
										
											2008-03-22 18:22:42 -07:00
										 |  |  | 			     unsigned long event, void *ptr) | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | { | 
					
						
							|  |  |  | 	struct inet6_ifaddr *ifa = (struct inet6_ifaddr *)ptr; | 
					
						
							|  |  |  | 	struct net_device *dev = (struct net_device *)ifa->idev->dev; | 
					
						
							|  |  |  | 	struct qeth_ipaddr *addr; | 
					
						
							|  |  |  | 	struct qeth_card *card; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	card = qeth_l3_get_card_from_dev(dev); | 
					
						
							|  |  |  | 	if (!card) | 
					
						
							|  |  |  | 		return NOTIFY_DONE; | 
					
						
							| 
									
										
										
										
											2010-06-21 22:57:05 +00:00
										 |  |  | 	QETH_CARD_TEXT(card, 3, "ip6event"); | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 	if (!qeth_is_supported(card, IPA_IPV6)) | 
					
						
							|  |  |  | 		return NOTIFY_DONE; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	addr = qeth_l3_get_addr_buffer(QETH_PROT_IPV6); | 
					
						
							|  |  |  | 	if (addr != NULL) { | 
					
						
							|  |  |  | 		memcpy(&addr->u.a6.addr, &ifa->addr, sizeof(struct in6_addr)); | 
					
						
							|  |  |  | 		addr->u.a6.pfxlen = ifa->prefix_len; | 
					
						
							|  |  |  | 		addr->type = QETH_IP_TYPE_NORMAL; | 
					
						
							|  |  |  | 	} else | 
					
						
							|  |  |  | 		goto out; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	switch (event) { | 
					
						
							|  |  |  | 	case NETDEV_UP: | 
					
						
							|  |  |  | 		if (!qeth_l3_add_ip(card, addr)) | 
					
						
							|  |  |  | 			kfree(addr); | 
					
						
							|  |  |  | 		break; | 
					
						
							|  |  |  | 	case NETDEV_DOWN: | 
					
						
							|  |  |  | 		if (!qeth_l3_delete_ip(card, addr)) | 
					
						
							|  |  |  | 			kfree(addr); | 
					
						
							|  |  |  | 		break; | 
					
						
							|  |  |  | 	default: | 
					
						
							|  |  |  | 		break; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	qeth_l3_set_ip_addr_list(card); | 
					
						
							|  |  |  | out: | 
					
						
							|  |  |  | 	return NOTIFY_DONE; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static struct notifier_block qeth_l3_ip6_notifier = { | 
					
						
							|  |  |  | 	qeth_l3_ip6_event, | 
					
						
							|  |  |  | 	NULL, | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int qeth_l3_register_notifiers(void) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	int rc; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-06-21 22:57:05 +00:00
										 |  |  | 	QETH_DBF_TEXT(SETUP, 5, "regnotif"); | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 	rc = register_inetaddr_notifier(&qeth_l3_ip_notifier); | 
					
						
							|  |  |  | 	if (rc) | 
					
						
							|  |  |  | 		return rc; | 
					
						
							|  |  |  | #ifdef CONFIG_QETH_IPV6
 | 
					
						
							|  |  |  | 	rc = register_inet6addr_notifier(&qeth_l3_ip6_notifier); | 
					
						
							|  |  |  | 	if (rc) { | 
					
						
							|  |  |  | 		unregister_inetaddr_notifier(&qeth_l3_ip_notifier); | 
					
						
							|  |  |  | 		return rc; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | #else
 | 
					
						
							| 
									
										
										
										
											2008-12-25 13:39:49 +01:00
										 |  |  | 	pr_warning("There is no IPv6 support for the layer 3 discipline\n"); | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | #endif
 | 
					
						
							|  |  |  | 	return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void qeth_l3_unregister_notifiers(void) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-06-21 22:57:05 +00:00
										 |  |  | 	QETH_DBF_TEXT(SETUP, 5, "unregnot"); | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 	BUG_ON(unregister_inetaddr_notifier(&qeth_l3_ip_notifier)); | 
					
						
							|  |  |  | #ifdef CONFIG_QETH_IPV6
 | 
					
						
							|  |  |  | 	BUG_ON(unregister_inet6addr_notifier(&qeth_l3_ip6_notifier)); | 
					
						
							|  |  |  | #endif /* QETH_IPV6 */
 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int __init qeth_l3_init(void) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	int rc = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-25 13:39:49 +01:00
										 |  |  | 	pr_info("register layer 3 discipline\n"); | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | 	rc = qeth_l3_register_notifiers(); | 
					
						
							|  |  |  | 	return rc; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void __exit qeth_l3_exit(void) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	qeth_l3_unregister_notifiers(); | 
					
						
							| 
									
										
										
										
											2008-12-25 13:39:49 +01:00
										 |  |  | 	pr_info("unregister layer 3 discipline\n"); | 
					
						
							| 
									
										
										
										
											2008-02-15 09:19:42 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | module_init(qeth_l3_init); | 
					
						
							|  |  |  | module_exit(qeth_l3_exit); | 
					
						
							|  |  |  | MODULE_AUTHOR("Frank Blaschka <frank.blaschka@de.ibm.com>"); | 
					
						
							|  |  |  | MODULE_DESCRIPTION("qeth layer 3 discipline"); | 
					
						
							|  |  |  | MODULE_LICENSE("GPL"); |