dhd_linux.c 23.1 KB
Newer Older
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16
/*
 * Copyright (c) 2010 Broadcom Corporation
 *
 * Permission to use, copy, modify, and/or distribute this software for any
 * purpose with or without fee is hereby granted, provided that the above
 * copyright notice and this permission notice appear in all copies.
 *
 * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
 * WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
 * MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY
 * SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
 * WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN ACTION
 * OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF OR IN
 * CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
 */

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

19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35
#include <linux/init.h>
#include <linux/kernel.h>
#include <linux/kthread.h>
#include <linux/slab.h>
#include <linux/skbuff.h>
#include <linux/netdevice.h>
#include <linux/etherdevice.h>
#include <linux/mmc/sdio_func.h>
#include <linux/random.h>
#include <linux/spinlock.h>
#include <linux/ethtool.h>
#include <linux/fcntl.h>
#include <linux/fs.h>
#include <linux/uaccess.h>
#include <linux/hardirq.h>
#include <linux/mutex.h>
#include <linux/wait.h>
36
#include <linux/module.h>
37 38 39 40 41 42 43 44 45 46 47
#include <net/cfg80211.h>
#include <net/rtnetlink.h>
#include <defs.h>
#include <brcmu_utils.h>
#include <brcmu_wifi.h>

#include "dhd.h"
#include "dhd_bus.h"
#include "dhd_proto.h"
#include "dhd_dbg.h"
#include "wl_cfg80211.h"
48
#include "fwil.h"
49 50 51 52 53 54

MODULE_AUTHOR("Broadcom Corporation");
MODULE_DESCRIPTION("Broadcom 802.11n wireless LAN fullmac driver.");
MODULE_SUPPORTED_DEVICE("Broadcom 802.11n WLAN fullmac cards");
MODULE_LICENSE("Dual BSD/GPL");

55
#define MAX_WAIT_FOR_8021X_TX		50	/* msecs */
56 57

/* Error bits */
58
int brcmf_msg_level = BRCMF_ERROR_VAL;
59 60
module_param(brcmf_msg_level, int, 0);

61
int brcmf_ifname2idx(struct brcmf_pub *drvr, char *name)
62 63 64 65 66 67 68 69
{
	int i = BRCMF_MAX_IFS;
	struct brcmf_if *ifp;

	if (name == NULL || *name == '\0')
		return 0;

	while (--i > 0) {
70
		ifp = drvr->iflist[i];
71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86
		if (ifp && !strncmp(ifp->ndev->name, name, IFNAMSIZ))
			break;
	}

	brcmf_dbg(TRACE, "return idx %d for \"%s\"\n", i, name);

	return i;		/* default - the primary interface */
}

char *brcmf_ifname(struct brcmf_pub *drvr, int ifidx)
{
	if (ifidx < 0 || ifidx >= BRCMF_MAX_IFS) {
		brcmf_dbg(ERROR, "ifidx %d out of range\n", ifidx);
		return "<if_bad>";
	}

87
	if (drvr->iflist[ifidx] == NULL) {
88 89 90 91
		brcmf_dbg(ERROR, "null i/f %d\n", ifidx);
		return "<if_null>";
	}

92 93
	if (drvr->iflist[ifidx]->ndev)
		return drvr->iflist[ifidx]->ndev->name;
94 95 96 97 98 99

	return "<if_none>";
}

static void _brcmf_set_multicast_list(struct work_struct *work)
{
100
	struct brcmf_if *ifp;
101 102
	struct net_device *ndev;
	struct netdev_hw_addr *ha;
103
	u32 cmd_value, cnt;
104 105
	__le32 cnt_le;
	char *buf, *bufp;
106 107
	u32 buflen;
	s32 err;
108
	struct brcmf_pub *drvr = container_of(work, struct brcmf_pub,
109 110
						    multicast_work);

111 112 113 114
	brcmf_dbg(TRACE, "enter\n");

	ifp = drvr->iflist[0];
	ndev = ifp->ndev;
115 116

	/* Determine initial value of allmulti flag */
117
	cmd_value = (ndev->flags & IFF_ALLMULTI) ? true : false;
118 119

	/* Send down the multicast list first. */
120 121 122 123
	cnt = netdev_mc_count(ndev);
	buflen = sizeof(cnt) + (cnt * ETH_ALEN);
	buf = kmalloc(buflen, GFP_ATOMIC);
	if (!buf)
124
		return;
125
	bufp = buf;
126 127

	cnt_le = cpu_to_le32(cnt);
128
	memcpy(bufp, &cnt_le, sizeof(cnt_le));
129 130 131 132 133 134 135 136 137 138
	bufp += sizeof(cnt_le);

	netdev_for_each_mc_addr(ha, ndev) {
		if (!cnt)
			break;
		memcpy(bufp, ha->addr, ETH_ALEN);
		bufp += ETH_ALEN;
		cnt--;
	}

139 140 141 142
	err = brcmf_fil_iovar_data_set(ifp, "mcast_list", buf, buflen);
	if (err < 0) {
		brcmf_dbg(ERROR, "Setting mcast_list failed, %d\n", err);
		cmd_value = cnt ? true : cmd_value;
143 144 145 146
	}

	kfree(buf);

147 148
	/*
	 * Now send the allmulti setting.  This is based on the setting in the
149 150 151
	 * net_device flags, but might be modified above to be turned on if we
	 * were trying to set some addresses and dongle rejected it...
	 */
152 153 154 155 156 157 158 159 160 161
	err = brcmf_fil_iovar_int_set(ifp, "allmulti", cmd_value);
	if (err < 0)
		brcmf_dbg(ERROR, "Setting allmulti failed, %d\n", err);

	/*Finally, pick up the PROMISC flag */
	cmd_value = (ndev->flags & IFF_PROMISC) ? true : false;
	err = brcmf_fil_cmd_int_set(ifp, BRCMF_C_SET_PROMISC, cmd_value);
	if (err < 0)
		brcmf_dbg(ERROR, "Setting BRCMF_C_SET_PROMISC failed, %d\n",
			  err);
162 163 164 165 166
}

static void
_brcmf_set_mac_address(struct work_struct *work)
{
167 168 169
	struct brcmf_if *ifp;
	struct net_device *ndev;
	s32 err;
170

171
	struct brcmf_pub *drvr = container_of(work, struct brcmf_pub,
172 173 174 175
						    setmacaddr_work);

	brcmf_dbg(TRACE, "enter\n");

176 177 178 179 180 181 182 183 184 185 186 187
	ifp = drvr->iflist[0];
	ndev = ifp->ndev;

	err = brcmf_fil_iovar_data_set(ifp, "cur_etheraddr", drvr->macvalue,
				       ETH_ALEN);
	if (err < 0) {
		brcmf_dbg(ERROR, "Setting cur_etheraddr failed, %d\n", err);
	} else {
		brcmf_dbg(TRACE, "MAC address updated to %pM\n",
			  drvr->macvalue);
		memcpy(ndev->dev_addr, drvr->macvalue, ETH_ALEN);
	}
188 189 190 191
}

static int brcmf_netdev_set_mac_address(struct net_device *ndev, void *addr)
{
192
	struct brcmf_if *ifp = netdev_priv(ndev);
193
	struct brcmf_pub *drvr = ifp->drvr;
194 195
	struct sockaddr *sa = (struct sockaddr *)addr;

196 197
	memcpy(&drvr->macvalue, sa->sa_data, ETH_ALEN);
	schedule_work(&drvr->setmacaddr_work);
198 199 200 201 202
	return 0;
}

static void brcmf_netdev_set_multicast_list(struct net_device *ndev)
{
203
	struct brcmf_if *ifp = netdev_priv(ndev);
204
	struct brcmf_pub *drvr = ifp->drvr;
205

206
	schedule_work(&drvr->multicast_work);
207 208 209 210 211
}

static int brcmf_netdev_start_xmit(struct sk_buff *skb, struct net_device *ndev)
{
	int ret;
212
	struct brcmf_if *ifp = netdev_priv(ndev);
213
	struct brcmf_pub *drvr = ifp->drvr;
214 215 216 217

	brcmf_dbg(TRACE, "Enter\n");

	/* Reject if down */
218
	if (!drvr->bus_if->drvr_up ||
219
	    (drvr->bus_if->state != BRCMF_BUS_DATA)) {
220 221
		brcmf_dbg(ERROR, "xmit rejected drvup=%d state=%d\n",
			  drvr->bus_if->drvr_up,
222
			  drvr->bus_if->state);
223 224 225 226
		netif_stop_queue(ndev);
		return -ENODEV;
	}

227
	if (!drvr->iflist[ifp->idx]) {
228
		brcmf_dbg(ERROR, "bad ifidx %d\n", ifp->idx);
229 230 231 232 233
		netif_stop_queue(ndev);
		return -ENODEV;
	}

	/* Make sure there's enough room for any header */
234
	if (skb_headroom(skb) < drvr->hdrlen) {
235 236 237
		struct sk_buff *skb2;

		brcmf_dbg(INFO, "%s: insufficient headroom\n",
238
			  brcmf_ifname(drvr, ifp->idx));
239
		drvr->bus_if->tx_realloc++;
240
		skb2 = skb_realloc_headroom(skb, drvr->hdrlen);
241 242 243 244
		dev_kfree_skb(skb);
		skb = skb2;
		if (skb == NULL) {
			brcmf_dbg(ERROR, "%s: skb_realloc_headroom failed\n",
245
				  brcmf_ifname(drvr, ifp->idx));
246 247 248 249 250
			ret = -ENOMEM;
			goto done;
		}
	}

251 252 253 254 255 256 257 258 259 260 261 262 263 264 265 266
	/* Update multicast statistic */
	if (skb->len >= ETH_ALEN) {
		u8 *pktdata = (u8 *)(skb->data);
		struct ethhdr *eh = (struct ethhdr *)pktdata;

		if (is_multicast_ether_addr(eh->h_dest))
			drvr->tx_multicast++;
		if (ntohs(eh->h_proto) == ETH_P_PAE)
			atomic_inc(&drvr->pend_8021x_cnt);
	}

	/* If the protocol uses a data header, apply it */
	brcmf_proto_hdrpush(drvr, ifp->idx, skb);

	/* Use bus module to send data frame */
	ret =  drvr->bus_if->brcmf_bus_txdata(drvr->dev, skb);
267 268 269

done:
	if (ret)
270
		drvr->bus_if->dstats.tx_dropped++;
271
	else
272
		drvr->bus_if->dstats.tx_packets++;
273 274 275 276 277

	/* Return ok: we always eat the packet */
	return 0;
}

278
void brcmf_txflowblock(struct device *dev, bool state)
279 280
{
	struct net_device *ndev;
281 282
	struct brcmf_bus *bus_if = dev_get_drvdata(dev);
	struct brcmf_pub *drvr = bus_if->drvr;
283
	int i;
284 285 286

	brcmf_dbg(TRACE, "Enter\n");

287 288 289 290 291 292 293 294
	for (i = 0; i < BRCMF_MAX_IFS; i++)
		if (drvr->iflist[i]) {
			ndev = drvr->iflist[i]->ndev;
			if (state)
				netif_stop_queue(ndev);
			else
				netif_wake_queue(ndev);
		}
295 296
}

297
static int brcmf_host_event(struct brcmf_pub *drvr, int *ifidx,
298 299 300 301 302
			    void *pktdata, struct brcmf_event_msg *event,
			    void **data)
{
	int bcmerror = 0;

303
	bcmerror = brcmf_c_host_event(drvr, ifidx, pktdata, event, data);
304 305 306
	if (bcmerror != 0)
		return bcmerror;

307 308
	if (drvr->iflist[*ifidx]->ndev)
		brcmf_cfg80211_event(drvr->iflist[*ifidx]->ndev,
309 310 311 312 313
				     event, *data);

	return bcmerror;
}

314
void brcmf_rx_frame(struct device *dev, int ifidx,
315
		    struct sk_buff_head *skb_list)
316 317 318 319
{
	unsigned char *eth;
	uint len;
	void *data;
320
	struct sk_buff *skb, *pnext;
321 322
	struct brcmf_if *ifp;
	struct brcmf_event_msg event;
323 324
	struct brcmf_bus *bus_if = dev_get_drvdata(dev);
	struct brcmf_pub *drvr = bus_if->drvr;
325 326 327

	brcmf_dbg(TRACE, "Enter\n");

328 329
	skb_queue_walk_safe(skb_list, skb, pnext) {
		skb_unlink(skb, skb_list);
330 331 332 333 334 335 336 337 338 339 340 341 342 343 344 345

		/* Get the protocol, maintain skb around eth_type_trans()
		 * The main reason for this hack is for the limitation of
		 * Linux 2.4 where 'eth_type_trans' uses the
		 * 'net->hard_header_len'
		 * to perform skb_pull inside vs ETH_HLEN. Since to avoid
		 * coping of the packet coming from the network stack to add
		 * BDC, Hardware header etc, during network interface
		 * registration
		 * we set the 'net->hard_header_len' to ETH_HLEN + extra space
		 * required
		 * for BDC, Hardware header etc. and not just the ETH_HLEN
		 */
		eth = skb->data;
		len = skb->len;

346
		ifp = drvr->iflist[ifidx];
347
		if (ifp == NULL)
348
			ifp = drvr->iflist[0];
349

350 351 352 353 354 355
		if (!ifp || !ifp->ndev ||
		    ifp->ndev->reg_state != NETREG_REGISTERED) {
			brcmu_pkt_buf_free_skb(skb);
			continue;
		}

356 357 358 359
		skb->dev = ifp->ndev;
		skb->protocol = eth_type_trans(skb, skb->dev);

		if (skb->pkt_type == PACKET_MULTICAST)
360
			bus_if->dstats.multicast++;
361 362 363 364 365 366 367 368 369

		skb->data = eth;
		skb->len = len;

		/* Strip header, count, deliver upward */
		skb_pull(skb, ETH_HLEN);

		/* Process special event packets and then discard them */
		if (ntohs(skb->protocol) == ETH_P_LINK_CTL)
370
			brcmf_host_event(drvr, &ifidx,
371 372 373
					  skb_mac_header(skb),
					  &event, &data);

374 375
		if (drvr->iflist[ifidx]) {
			ifp = drvr->iflist[ifidx];
376
			ifp->ndev->last_rx = jiffies;
377
		}
378

379 380
		bus_if->dstats.rx_bytes += skb->len;
		bus_if->dstats.rx_packets++;	/* Local count */
381 382 383 384 385 386 387 388 389 390 391 392 393 394

		if (in_interrupt())
			netif_rx(skb);
		else
			/* If the receive is not processed inside an ISR,
			 * the softirqd must be woken explicitly to service
			 * the NET_RX_SOFTIRQ.  In 2.6 kernels, this is handled
			 * by netif_rx_ni(), but in earlier kernels, we need
			 * to do it manually.
			 */
			netif_rx_ni(skb);
	}
}

395
void brcmf_txcomplete(struct device *dev, struct sk_buff *txp, bool success)
396 397 398 399
{
	uint ifidx;
	struct ethhdr *eh;
	u16 type;
400 401
	struct brcmf_bus *bus_if = dev_get_drvdata(dev);
	struct brcmf_pub *drvr = bus_if->drvr;
402

403
	brcmf_proto_hdrpull(dev, &ifidx, txp);
404 405 406 407

	eh = (struct ethhdr *)(txp->data);
	type = ntohs(eh->h_proto);

408
	if (type == ETH_P_PAE) {
409
		atomic_dec(&drvr->pend_8021x_cnt);
410 411 412
		if (waitqueue_active(&drvr->pend_8021x_wait))
			wake_up(&drvr->pend_8021x_wait);
	}
413 414 415 416
}

static struct net_device_stats *brcmf_netdev_get_stats(struct net_device *ndev)
{
417
	struct brcmf_if *ifp = netdev_priv(ndev);
418
	struct brcmf_bus *bus_if = ifp->drvr->bus_if;
419 420 421 422

	brcmf_dbg(TRACE, "Enter\n");

	/* Copy dongle stats to net device stats */
423 424 425 426 427 428 429 430 431
	ifp->stats.rx_packets = bus_if->dstats.rx_packets;
	ifp->stats.tx_packets = bus_if->dstats.tx_packets;
	ifp->stats.rx_bytes = bus_if->dstats.rx_bytes;
	ifp->stats.tx_bytes = bus_if->dstats.tx_bytes;
	ifp->stats.rx_errors = bus_if->dstats.rx_errors;
	ifp->stats.tx_errors = bus_if->dstats.tx_errors;
	ifp->stats.rx_dropped = bus_if->dstats.rx_dropped;
	ifp->stats.tx_dropped = bus_if->dstats.tx_dropped;
	ifp->stats.multicast = bus_if->dstats.multicast;
432 433 434 435

	return &ifp->stats;
}

436 437 438 439 440
/*
 * Set current toe component enables in toe_ol iovar,
 * and set toe global enable iovar
 */
static int brcmf_toe_set(struct brcmf_if *ifp, u32 toe_ol)
441
{
442
	s32 err;
443

444 445 446 447
	err = brcmf_fil_iovar_int_set(ifp, "toe_ol", toe_ol);
	if (err < 0) {
		brcmf_dbg(ERROR, "Setting toe_ol failed, %d\n", err);
		return err;
448 449
	}

450 451 452
	err = brcmf_fil_iovar_int_set(ifp, "toe", (toe_ol != 0));
	if (err < 0)
		brcmf_dbg(ERROR, "Setting toe failed, %d\n", err);
453

454
	return err;
455 456 457 458 459 460

}

static void brcmf_ethtool_get_drvinfo(struct net_device *ndev,
				    struct ethtool_drvinfo *info)
{
461
	struct brcmf_if *ifp = netdev_priv(ndev);
462
	struct brcmf_pub *drvr = ifp->drvr;
463 464

	sprintf(info->driver, KBUILD_MODNAME);
465 466
	sprintf(info->version, "%lu", drvr->drv_version);
	sprintf(info->bus_info, "%s", dev_name(drvr->dev));
467 468
}

469 470
static const struct ethtool_ops brcmf_ethtool_ops = {
	.get_drvinfo = brcmf_ethtool_get_drvinfo,
471 472
};

473
static int brcmf_ethtool(struct brcmf_if *ifp, void __user *uaddr)
474
{
475
	struct brcmf_pub *drvr = ifp->drvr;
476 477 478 479 480 481 482 483 484 485 486 487 488 489 490 491 492 493 494 495 496 497 498 499 500 501 502 503 504 505 506 507
	struct ethtool_drvinfo info;
	char drvname[sizeof(info.driver)];
	u32 cmd;
	struct ethtool_value edata;
	u32 toe_cmpnt, csum_dir;
	int ret;

	brcmf_dbg(TRACE, "Enter\n");

	/* all ethtool calls start with a cmd word */
	if (copy_from_user(&cmd, uaddr, sizeof(u32)))
		return -EFAULT;

	switch (cmd) {
	case ETHTOOL_GDRVINFO:
		/* Copy out any request driver name */
		if (copy_from_user(&info, uaddr, sizeof(info)))
			return -EFAULT;
		strncpy(drvname, info.driver, sizeof(info.driver));
		drvname[sizeof(info.driver) - 1] = '\0';

		/* clear struct for return */
		memset(&info, 0, sizeof(info));
		info.cmd = cmd;

		/* if requested, identify ourselves */
		if (strcmp(drvname, "?dhd") == 0) {
			sprintf(info.driver, "dhd");
			strcpy(info.version, BRCMF_VERSION_STR);
		}

		/* otherwise, require dongle to be up */
508
		else if (!drvr->bus_if->drvr_up) {
509 510 511 512 513
			brcmf_dbg(ERROR, "dongle is not up\n");
			return -ENODEV;
		}
		/* finally, report dongle driver type */
		else
514
			sprintf(info.driver, "wl");
515

516
		sprintf(info.version, "%lu", drvr->drv_version);
517 518 519 520 521 522 523 524 525
		if (copy_to_user(uaddr, &info, sizeof(info)))
			return -EFAULT;
		brcmf_dbg(CTL, "given %*s, returning %s\n",
			  (int)sizeof(drvname), drvname, info.driver);
		break;

		/* Get toe offload components from dongle */
	case ETHTOOL_GRXCSUM:
	case ETHTOOL_GTXCSUM:
526
		ret = brcmf_fil_iovar_int_get(ifp, "toe_ol", &toe_cmpnt);
527 528 529 530 531 532 533 534 535 536 537 538 539 540 541 542 543 544 545 546
		if (ret < 0)
			return ret;

		csum_dir =
		    (cmd == ETHTOOL_GTXCSUM) ? TOE_TX_CSUM_OL : TOE_RX_CSUM_OL;

		edata.cmd = cmd;
		edata.data = (toe_cmpnt & csum_dir) ? 1 : 0;

		if (copy_to_user(uaddr, &edata, sizeof(edata)))
			return -EFAULT;
		break;

		/* Set toe offload components in dongle */
	case ETHTOOL_SRXCSUM:
	case ETHTOOL_STXCSUM:
		if (copy_from_user(&edata, uaddr, sizeof(edata)))
			return -EFAULT;

		/* Read the current settings, update and write back */
547
		ret = brcmf_fil_iovar_int_get(ifp, "toe_ol", &toe_cmpnt);
548 549 550 551 552 553 554 555 556 557 558
		if (ret < 0)
			return ret;

		csum_dir =
		    (cmd == ETHTOOL_STXCSUM) ? TOE_TX_CSUM_OL : TOE_RX_CSUM_OL;

		if (edata.data != 0)
			toe_cmpnt |= csum_dir;
		else
			toe_cmpnt &= ~csum_dir;

559
		ret = brcmf_toe_set(ifp, toe_cmpnt);
560 561 562 563 564 565
		if (ret < 0)
			return ret;

		/* If setting TX checksum mode, tell Linux the new mode */
		if (cmd == ETHTOOL_STXCSUM) {
			if (edata.data)
566
				ifp->ndev->features |= NETIF_F_IP_CSUM;
567
			else
568
				ifp->ndev->features &= ~NETIF_F_IP_CSUM;
569 570 571 572 573 574 575 576 577 578 579 580 581 582
		}

		break;

	default:
		return -EOPNOTSUPP;
	}

	return 0;
}

static int brcmf_netdev_ioctl_entry(struct net_device *ndev, struct ifreq *ifr,
				    int cmd)
{
583
	struct brcmf_if *ifp = netdev_priv(ndev);
584
	struct brcmf_pub *drvr = ifp->drvr;
585

586
	brcmf_dbg(TRACE, "ifidx %d, cmd 0x%04x\n", ifp->idx, cmd);
587

588
	if (!drvr->iflist[ifp->idx])
589 590 591
		return -1;

	if (cmd == SIOCETHTOOL)
592
		return brcmf_ethtool(ifp, ifr->ifr_data);
593 594 595 596 597 598

	return -EOPNOTSUPP;
}

static int brcmf_netdev_stop(struct net_device *ndev)
{
599
	struct brcmf_if *ifp = netdev_priv(ndev);
600
	struct brcmf_pub *drvr = ifp->drvr;
601 602 603

	brcmf_dbg(TRACE, "Enter\n");
	brcmf_cfg80211_down(drvr->config);
604
	if (drvr->bus_if->drvr_up == 0)
605 606 607
		return 0;

	/* Set state and stop OS transmissions */
J
John W. Linville 已提交
608
	drvr->bus_if->drvr_up = false;
609 610 611 612 613 614 615
	netif_stop_queue(ndev);

	return 0;
}

static int brcmf_netdev_open(struct net_device *ndev)
{
616
	struct brcmf_if *ifp = netdev_priv(ndev);
617
	struct brcmf_pub *drvr = ifp->drvr;
618
	struct brcmf_bus *bus_if = drvr->bus_if;
619 620 621
	u32 toe_ol;
	s32 ret = 0;

622
	brcmf_dbg(TRACE, "ifidx %d\n", ifp->idx);
623

624
	if (ifp->idx == 0) {	/* do it only for primary eth0 */
625 626 627 628
		/* If bus is not ready, can't continue */
		if (bus_if->state != BRCMF_BUS_DATA) {
			brcmf_dbg(ERROR, "failed bus is not ready\n");
			return -EAGAIN;
629
		}
630

631
		atomic_set(&drvr->pend_8021x_cnt, 0);
632

633
		memcpy(ndev->dev_addr, drvr->mac, ETH_ALEN);
634 635

		/* Get current TOE mode from dongle */
636
		if (brcmf_fil_iovar_int_get(ifp, "toe_ol", &toe_ol) >= 0
637
		    && (toe_ol & TOE_TX_CSUM_OL) != 0)
638
			drvr->iflist[ifp->idx]->ndev->features |=
639 640
				NETIF_F_IP_CSUM;
		else
641
			drvr->iflist[ifp->idx]->ndev->features &=
642 643
				~NETIF_F_IP_CSUM;
	}
644 645

	/* make sure RF is ready for work */
646
	brcmf_fil_cmd_int_set(ifp, BRCMF_C_UP, 0);
647

648 649
	/* Allow transmit calls */
	netif_start_queue(ndev);
J
John W. Linville 已提交
650
	drvr->bus_if->drvr_up = true;
651
	if (brcmf_cfg80211_up(drvr->config)) {
652 653 654 655 656 657 658
		brcmf_dbg(ERROR, "failed to bring up cfg80211\n");
		return -1;
	}

	return ret;
}

659 660 661 662 663 664 665 666 667 668
static const struct net_device_ops brcmf_netdev_ops_pri = {
	.ndo_open = brcmf_netdev_open,
	.ndo_stop = brcmf_netdev_stop,
	.ndo_get_stats = brcmf_netdev_get_stats,
	.ndo_do_ioctl = brcmf_netdev_ioctl_entry,
	.ndo_start_xmit = brcmf_netdev_start_xmit,
	.ndo_set_mac_address = brcmf_netdev_set_mac_address,
	.ndo_set_rx_mode = brcmf_netdev_set_multicast_list
};

669
int brcmf_net_attach(struct brcmf_if *ifp)
670
{
671
	struct brcmf_pub *drvr = ifp->drvr;
672
	struct net_device *ndev;
673
	u8 temp_addr[ETH_ALEN];
674

675
	brcmf_dbg(TRACE, "ifidx %d\n", ifp->idx);
676

677
	ndev = drvr->iflist[ifp->idx]->ndev;
678 679 680
	ndev->netdev_ops = &brcmf_netdev_ops_pri;

	/*
681
	 * determine mac address to use
682
	 */
683 684 685
	if (is_valid_ether_addr(ifp->mac_addr))
		memcpy(temp_addr, ifp->mac_addr, ETH_ALEN);
	else
686 687
		memcpy(temp_addr, drvr->mac, ETH_ALEN);

688
	if (ifp->idx == 1) {
689 690 691 692 693 694 695 696 697 698 699 700 701 702 703 704 705 706 707 708 709 710 711 712 713 714 715 716
		brcmf_dbg(TRACE, "ACCESS POINT MAC:\n");
		/*  ACCESSPOINT INTERFACE CASE */
		temp_addr[0] |= 0X02;	/* set bit 2 ,
			 - Locally Administered address  */

	}
	ndev->hard_header_len = ETH_HLEN + drvr->hdrlen;
	ndev->ethtool_ops = &brcmf_ethtool_ops;

	drvr->rxsz = ndev->mtu + ndev->hard_header_len +
			      drvr->hdrlen;

	memcpy(ndev->dev_addr, temp_addr, ETH_ALEN);

	if (register_netdev(ndev) != 0) {
		brcmf_dbg(ERROR, "couldn't register the net device\n");
		goto fail;
	}

	brcmf_dbg(INFO, "%s: Broadcom Dongle Host Driver\n", ndev->name);

	return 0;

fail:
	ndev->netdev_ops = NULL;
	return -EBADE;
}

717 718
struct brcmf_if *brcmf_add_if(struct device *dev, int ifidx, s32 bssidx,
			      char *name, u8 *mac_addr)
719 720
{
	struct brcmf_if *ifp;
721
	struct net_device *ndev;
722 723
	struct brcmf_bus *bus_if = dev_get_drvdata(dev);
	struct brcmf_pub *drvr = bus_if->drvr;
724

725
	brcmf_dbg(TRACE, "idx %d\n", ifidx);
726

727
	ifp = drvr->iflist[ifidx];
728 729 730 731 732 733 734 735 736 737
	/*
	 * Delete the existing interface before overwriting it
	 * in case we missed the BRCMF_E_IF_DEL event.
	 */
	if (ifp) {
		brcmf_dbg(ERROR, "ERROR: netdev:%s already exists, try free & unregister\n",
			  ifp->ndev->name);
		netif_stop_queue(ifp->ndev);
		unregister_netdev(ifp->ndev);
		free_netdev(ifp->ndev);
738
		drvr->iflist[ifidx] = NULL;
739
	}
740

741
	/* Allocate netdev, including space for private structure */
742 743
	ndev = alloc_netdev(sizeof(struct brcmf_if), name, ether_setup);
	if (!ndev) {
744
		brcmf_dbg(ERROR, "OOM - alloc_netdev\n");
745
		return ERR_PTR(-ENOMEM);
746
	}
747

748 749
	ifp = netdev_priv(ndev);
	ifp->ndev = ndev;
750 751
	ifp->drvr = drvr;
	drvr->iflist[ifidx] = ifp;
752
	ifp->idx = ifidx;
753
	ifp->bssidx = bssidx;
754 755
	if (mac_addr != NULL)
		memcpy(&ifp->mac_addr, mac_addr, ETH_ALEN);
756

757 758
	brcmf_dbg(TRACE, " ==== pid:%x, net_device for if:%s created ===\n",
		  current->pid, ifp->ndev->name);
759

760
	return ifp;
761 762
}

763
void brcmf_del_if(struct brcmf_pub *drvr, int ifidx)
764 765 766 767 768
{
	struct brcmf_if *ifp;

	brcmf_dbg(TRACE, "idx %d\n", ifidx);

769
	ifp = drvr->iflist[ifidx];
770 771 772 773
	if (!ifp) {
		brcmf_dbg(ERROR, "Null interface\n");
		return;
	}
774 775 776 777 778 779 780 781 782 783 784
	if (ifp->ndev) {
		if (ifidx == 0) {
			if (ifp->ndev->netdev_ops == &brcmf_netdev_ops_pri) {
				rtnl_lock();
				brcmf_netdev_stop(ifp->ndev);
				rtnl_unlock();
			}
		} else {
			netif_stop_queue(ifp->ndev);
		}

785
		unregister_netdev(ifp->ndev);
786
		drvr->iflist[ifidx] = NULL;
787
		if (ifidx == 0)
788
			brcmf_cfg80211_detach(drvr->config);
789
		free_netdev(ifp->ndev);
790 791 792
	}
}

793
int brcmf_attach(uint bus_hdrlen, struct device *dev)
794
{
795
	struct brcmf_pub *drvr = NULL;
796
	int ret = 0;
797 798 799 800

	brcmf_dbg(TRACE, "Enter\n");

	/* Allocate primary brcmf_info */
801 802
	drvr = kzalloc(sizeof(struct brcmf_pub), GFP_ATOMIC);
	if (!drvr)
803
		return -ENOMEM;
804

805
	mutex_init(&drvr->proto_block);
806 807

	/* Link to bus module */
808 809
	drvr->hdrlen = bus_hdrlen;
	drvr->bus_if = dev_get_drvdata(dev);
810
	drvr->bus_if->drvr = drvr;
811
	drvr->dev = dev;
812

813 814 815
	/* create device debugfs folder */
	brcmf_debugfs_attach(drvr);

816
	/* Attach and link in the protocol */
817 818
	ret = brcmf_proto_attach(drvr);
	if (ret != 0) {
819 820 821 822
		brcmf_dbg(ERROR, "brcmf_prot_attach failed\n");
		goto fail;
	}

823 824
	INIT_WORK(&drvr->setmacaddr_work, _brcmf_set_mac_address);
	INIT_WORK(&drvr->multicast_work, _brcmf_set_multicast_list);
825

826 827
	INIT_LIST_HEAD(&drvr->bus_if->dcmd_list);

828 829
	init_waitqueue_head(&drvr->pend_8021x_wait);

830
	return ret;
831 832

fail:
833
	brcmf_detach(dev);
834

835
	return ret;
836 837
}

838
int brcmf_bus_start(struct device *dev)
839 840
{
	int ret = -1;
841 842
	struct brcmf_bus *bus_if = dev_get_drvdata(dev);
	struct brcmf_pub *drvr = bus_if->drvr;
843
	struct brcmf_if *ifp;
844 845 846 847

	brcmf_dbg(TRACE, "\n");

	/* Bring up the bus */
848
	ret = bus_if->brcmf_bus_init(dev);
849 850 851 852 853
	if (ret != 0) {
		brcmf_dbg(ERROR, "brcmf_sdbrcm_bus_init failed %d\n", ret);
		return ret;
	}

854
	/* add primary networking interface */
855
	ifp = brcmf_add_if(dev, 0, 0, "wlan%d", NULL);
856 857 858
	if (IS_ERR(ifp))
		return PTR_ERR(ifp);

859 860 861 862 863
	/* signal bus ready */
	bus_if->state = BRCMF_BUS_DATA;

	/* Bus is ready, do any initialization */
	ret = brcmf_c_preinit_dcmds(ifp);
864
	if (ret < 0)
865
		goto fail;
866

867
	drvr->config = brcmf_cfg80211_attach(drvr);
868 869 870 871
	if (drvr->config == NULL) {
		ret = -ENOMEM;
		goto fail;
	}
872 873

	ret = brcmf_net_attach(ifp);
874
fail:
875 876
	if (ret < 0) {
		brcmf_dbg(ERROR, "brcmf_net_attach failed");
877 878 879
		if (drvr->config)
			brcmf_cfg80211_detach(drvr->config);
		free_netdev(drvr->iflist[0]->ndev);
880
		drvr->iflist[0] = NULL;
881
		return ret;
882 883
	}

884 885 886 887 888 889 890 891
	return 0;
}

static void brcmf_bus_detach(struct brcmf_pub *drvr)
{
	brcmf_dbg(TRACE, "Enter\n");

	if (drvr) {
892 893
		/* Stop the protocol module */
		brcmf_proto_stop(drvr);
894

895
		/* Stop the bus module */
896
		drvr->bus_if->brcmf_bus_stop(drvr->dev);
897 898 899
	}
}

900
void brcmf_detach(struct device *dev)
901
{
902 903 904
	int i;
	struct brcmf_bus *bus_if = dev_get_drvdata(dev);
	struct brcmf_pub *drvr = bus_if->drvr;
905 906 907 908

	brcmf_dbg(TRACE, "Enter\n");


909 910 911 912
	/* make sure primary interface removed last */
	for (i = BRCMF_MAX_IFS-1; i > -1; i--)
		if (drvr->iflist[i])
			brcmf_del_if(drvr, i);
913

914
	brcmf_bus_detach(drvr);
915

916 917 918
	if (drvr->prot) {
		cancel_work_sync(&drvr->setmacaddr_work);
		cancel_work_sync(&drvr->multicast_work);
919
		brcmf_proto_detach(drvr);
920
	}
921

922
	brcmf_debugfs_detach(drvr);
923 924
	bus_if->drvr = NULL;
	kfree(drvr);
925 926
}

927
static int brcmf_get_pend_8021x_cnt(struct brcmf_pub *drvr)
928
{
929
	return atomic_read(&drvr->pend_8021x_cnt);
930 931 932 933
}

int brcmf_netdev_wait_pend8021x(struct net_device *ndev)
{
934
	struct brcmf_if *ifp = netdev_priv(ndev);
935
	struct brcmf_pub *drvr = ifp->drvr;
936 937 938 939 940 941 942 943 944
	int err;

	err = wait_event_timeout(drvr->pend_8021x_wait,
				 !brcmf_get_pend_8021x_cnt(drvr),
				 msecs_to_jiffies(MAX_WAIT_FOR_8021X_TX));

	WARN_ON(!err);

	return !err;
945 946
}

947
static void brcmf_driver_init(struct work_struct *work)
948
{
949 950
	brcmf_debugfs_init();

951
#ifdef CONFIG_BRCMFMAC_SDIO
952
	brcmf_sdio_init();
953
#endif
954
#ifdef CONFIG_BRCMFMAC_USB
955
	brcmf_usb_init();
956
#endif
957 958 959 960 961 962 963 964
}
static DECLARE_WORK(brcmf_driver_work, brcmf_driver_init);

static int __init brcmfmac_module_init(void)
{
	if (!schedule_work(&brcmf_driver_work))
		return -EBUSY;

965
	return 0;
966 967
}

968
static void __exit brcmfmac_module_exit(void)
969
{
970 971
	cancel_work_sync(&brcmf_driver_work);

972 973 974
#ifdef CONFIG_BRCMFMAC_SDIO
	brcmf_sdio_exit();
#endif
975 976 977
#ifdef CONFIG_BRCMFMAC_USB
	brcmf_usb_exit();
#endif
978
	brcmf_debugfs_exit();
979 980
}

981 982
module_init(brcmfmac_module_init);
module_exit(brcmfmac_module_exit);