dhd_linux.c 22.7 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

109 110
	brcmf_dbg(TRACE, "enter\n");

111
	ifp = container_of(work, struct brcmf_if, multicast_work);
112
	ndev = ifp->ndev;
113 114

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

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

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

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

137 138 139 140
	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;
141 142 143 144
	}

	kfree(buf);

145 146
	/*
	 * Now send the allmulti setting.  This is based on the setting in the
147 148 149
	 * net_device flags, but might be modified above to be turned on if we
	 * were trying to set some addresses and dongle rejected it...
	 */
150 151 152 153 154 155 156 157 158 159
	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);
160 161 162 163 164
}

static void
_brcmf_set_mac_address(struct work_struct *work)
{
165 166
	struct brcmf_if *ifp;
	s32 err;
167 168 169

	brcmf_dbg(TRACE, "enter\n");

170 171
	ifp = container_of(work, struct brcmf_if, setmacaddr_work);
	err = brcmf_fil_iovar_data_set(ifp, "cur_etheraddr", ifp->mac_addr,
172 173 174 175 176
				       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",
177 178
			  ifp->mac_addr);
		memcpy(ifp->ndev->dev_addr, ifp->mac_addr, ETH_ALEN);
179
	}
180 181 182 183
}

static int brcmf_netdev_set_mac_address(struct net_device *ndev, void *addr)
{
184
	struct brcmf_if *ifp = netdev_priv(ndev);
185 186
	struct sockaddr *sa = (struct sockaddr *)addr;

187 188
	memcpy(&ifp->mac_addr, sa->sa_data, ETH_ALEN);
	schedule_work(&ifp->setmacaddr_work);
189 190 191 192 193
	return 0;
}

static void brcmf_netdev_set_multicast_list(struct net_device *ndev)
{
194
	struct brcmf_if *ifp = netdev_priv(ndev);
195

196
	schedule_work(&ifp->multicast_work);
197 198 199 200 201
}

static int brcmf_netdev_start_xmit(struct sk_buff *skb, struct net_device *ndev)
{
	int ret;
202
	struct brcmf_if *ifp = netdev_priv(ndev);
203
	struct brcmf_pub *drvr = ifp->drvr;
204 205 206 207

	brcmf_dbg(TRACE, "Enter\n");

	/* Reject if down */
208
	if (!drvr->bus_if->drvr_up ||
209
	    (drvr->bus_if->state != BRCMF_BUS_DATA)) {
210 211
		brcmf_dbg(ERROR, "xmit rejected drvup=%d state=%d\n",
			  drvr->bus_if->drvr_up,
212
			  drvr->bus_if->state);
213 214 215 216
		netif_stop_queue(ndev);
		return -ENODEV;
	}

217
	if (!drvr->iflist[ifp->idx]) {
218
		brcmf_dbg(ERROR, "bad ifidx %d\n", ifp->idx);
219 220 221 222 223
		netif_stop_queue(ndev);
		return -ENODEV;
	}

	/* Make sure there's enough room for any header */
224
	if (skb_headroom(skb) < drvr->hdrlen) {
225 226 227
		struct sk_buff *skb2;

		brcmf_dbg(INFO, "%s: insufficient headroom\n",
228
			  brcmf_ifname(drvr, ifp->idx));
229
		drvr->bus_if->tx_realloc++;
230
		skb2 = skb_realloc_headroom(skb, drvr->hdrlen);
231 232 233 234
		dev_kfree_skb(skb);
		skb = skb2;
		if (skb == NULL) {
			brcmf_dbg(ERROR, "%s: skb_realloc_headroom failed\n",
235
				  brcmf_ifname(drvr, ifp->idx));
236 237 238 239 240
			ret = -ENOMEM;
			goto done;
		}
	}

241 242 243 244 245 246 247 248 249 250 251 252 253 254 255 256
	/* 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);
257 258 259

done:
	if (ret)
260
		drvr->bus_if->dstats.tx_dropped++;
261
	else
262
		drvr->bus_if->dstats.tx_packets++;
263 264 265 266 267

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

268
void brcmf_txflowblock(struct device *dev, bool state)
269 270
{
	struct net_device *ndev;
271 272
	struct brcmf_bus *bus_if = dev_get_drvdata(dev);
	struct brcmf_pub *drvr = bus_if->drvr;
273
	int i;
274 275 276

	brcmf_dbg(TRACE, "Enter\n");

277 278 279 280 281 282 283 284
	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);
		}
285 286
}

287
void brcmf_rx_frame(struct device *dev, u8 ifidx,
288
		    struct sk_buff_head *skb_list)
289 290 291
{
	unsigned char *eth;
	uint len;
292
	struct sk_buff *skb, *pnext;
293
	struct brcmf_if *ifp;
294 295
	struct brcmf_bus *bus_if = dev_get_drvdata(dev);
	struct brcmf_pub *drvr = bus_if->drvr;
296 297 298

	brcmf_dbg(TRACE, "Enter\n");

299 300
	skb_queue_walk_safe(skb_list, skb, pnext) {
		skb_unlink(skb, skb_list);
301 302 303 304 305 306 307 308 309 310 311 312 313 314 315 316

		/* 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;

317
		ifp = drvr->iflist[ifidx];
318
		if (ifp == NULL)
319
			ifp = drvr->iflist[0];
320

321 322 323 324 325 326
		if (!ifp || !ifp->ndev ||
		    ifp->ndev->reg_state != NETREG_REGISTERED) {
			brcmu_pkt_buf_free_skb(skb);
			continue;
		}

327 328 329 330
		skb->dev = ifp->ndev;
		skb->protocol = eth_type_trans(skb, skb->dev);

		if (skb->pkt_type == PACKET_MULTICAST)
331
			bus_if->dstats.multicast++;
332 333 334 335 336 337 338 339

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

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

		/* Process special event packets and then discard them */
340
		brcmf_fweh_process_skb(drvr, skb, &ifidx);
341

342 343
		if (drvr->iflist[ifidx]) {
			ifp = drvr->iflist[ifidx];
344
			ifp->ndev->last_rx = jiffies;
345
		}
346

347 348
		bus_if->dstats.rx_bytes += skb->len;
		bus_if->dstats.rx_packets++;	/* Local count */
349 350 351 352 353 354 355 356 357 358 359 360 361 362

		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);
	}
}

363
void brcmf_txcomplete(struct device *dev, struct sk_buff *txp, bool success)
364 365 366 367
{
	uint ifidx;
	struct ethhdr *eh;
	u16 type;
368 369
	struct brcmf_bus *bus_if = dev_get_drvdata(dev);
	struct brcmf_pub *drvr = bus_if->drvr;
370

371
	brcmf_proto_hdrpull(dev, &ifidx, txp);
372 373 374 375

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

376
	if (type == ETH_P_PAE) {
377
		atomic_dec(&drvr->pend_8021x_cnt);
378 379 380
		if (waitqueue_active(&drvr->pend_8021x_wait))
			wake_up(&drvr->pend_8021x_wait);
	}
381 382 383 384
}

static struct net_device_stats *brcmf_netdev_get_stats(struct net_device *ndev)
{
385
	struct brcmf_if *ifp = netdev_priv(ndev);
386
	struct brcmf_bus *bus_if = ifp->drvr->bus_if;
387 388 389 390

	brcmf_dbg(TRACE, "Enter\n");

	/* Copy dongle stats to net device stats */
391 392 393 394 395 396 397 398 399
	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;
400 401 402 403

	return &ifp->stats;
}

404 405 406 407 408
/*
 * 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)
409
{
410
	s32 err;
411

412 413 414 415
	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;
416 417
	}

418 419 420
	err = brcmf_fil_iovar_int_set(ifp, "toe", (toe_ol != 0));
	if (err < 0)
		brcmf_dbg(ERROR, "Setting toe failed, %d\n", err);
421

422
	return err;
423 424 425 426 427 428

}

static void brcmf_ethtool_get_drvinfo(struct net_device *ndev,
				    struct ethtool_drvinfo *info)
{
429
	struct brcmf_if *ifp = netdev_priv(ndev);
430
	struct brcmf_pub *drvr = ifp->drvr;
431 432

	sprintf(info->driver, KBUILD_MODNAME);
433 434
	sprintf(info->version, "%lu", drvr->drv_version);
	sprintf(info->bus_info, "%s", dev_name(drvr->dev));
435 436
}

437 438
static const struct ethtool_ops brcmf_ethtool_ops = {
	.get_drvinfo = brcmf_ethtool_get_drvinfo,
439 440
};

441
static int brcmf_ethtool(struct brcmf_if *ifp, void __user *uaddr)
442
{
443
	struct brcmf_pub *drvr = ifp->drvr;
444 445 446 447 448 449 450 451 452 453 454 455 456 457 458 459 460 461 462 463 464 465 466 467 468 469 470 471 472 473 474 475
	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 */
476
		else if (!drvr->bus_if->drvr_up) {
477 478 479 480 481
			brcmf_dbg(ERROR, "dongle is not up\n");
			return -ENODEV;
		}
		/* finally, report dongle driver type */
		else
482
			sprintf(info.driver, "wl");
483

484
		sprintf(info.version, "%lu", drvr->drv_version);
485 486 487 488 489 490 491 492 493
		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:
494
		ret = brcmf_fil_iovar_int_get(ifp, "toe_ol", &toe_cmpnt);
495 496 497 498 499 500 501 502 503 504 505 506 507 508 509 510 511 512 513 514
		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 */
515
		ret = brcmf_fil_iovar_int_get(ifp, "toe_ol", &toe_cmpnt);
516 517 518 519 520 521 522 523 524 525 526
		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;

527
		ret = brcmf_toe_set(ifp, toe_cmpnt);
528 529 530 531 532 533
		if (ret < 0)
			return ret;

		/* If setting TX checksum mode, tell Linux the new mode */
		if (cmd == ETHTOOL_STXCSUM) {
			if (edata.data)
534
				ifp->ndev->features |= NETIF_F_IP_CSUM;
535
			else
536
				ifp->ndev->features &= ~NETIF_F_IP_CSUM;
537 538 539 540 541 542 543 544 545 546 547 548 549 550
		}

		break;

	default:
		return -EOPNOTSUPP;
	}

	return 0;
}

static int brcmf_netdev_ioctl_entry(struct net_device *ndev, struct ifreq *ifr,
				    int cmd)
{
551
	struct brcmf_if *ifp = netdev_priv(ndev);
552
	struct brcmf_pub *drvr = ifp->drvr;
553

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

556
	if (!drvr->iflist[ifp->idx])
557 558 559
		return -1;

	if (cmd == SIOCETHTOOL)
560
		return brcmf_ethtool(ifp, ifr->ifr_data);
561 562 563 564 565 566

	return -EOPNOTSUPP;
}

static int brcmf_netdev_stop(struct net_device *ndev)
{
567
	struct brcmf_if *ifp = netdev_priv(ndev);
568
	struct brcmf_pub *drvr = ifp->drvr;
569 570

	brcmf_dbg(TRACE, "Enter\n");
571

572
	if (drvr->bus_if->drvr_up == 0)
573 574
		return 0;

575
	brcmf_cfg80211_down(ndev);
576

577
	/* Set state and stop OS transmissions */
J
John W. Linville 已提交
578
	drvr->bus_if->drvr_up = false;
579 580 581 582 583 584 585
	netif_stop_queue(ndev);

	return 0;
}

static int brcmf_netdev_open(struct net_device *ndev)
{
586
	struct brcmf_if *ifp = netdev_priv(ndev);
587
	struct brcmf_pub *drvr = ifp->drvr;
588
	struct brcmf_bus *bus_if = drvr->bus_if;
589 590 591
	u32 toe_ol;
	s32 ret = 0;

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

594 595 596 597 598
	/* 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;
	}
599

600
	atomic_set(&drvr->pend_8021x_cnt, 0);
601

602
	memcpy(ndev->dev_addr, drvr->mac, ETH_ALEN);
603

604 605 606 607 608 609 610 611
	/* Get current TOE mode from dongle */
	if (brcmf_fil_iovar_int_get(ifp, "toe_ol", &toe_ol) >= 0
	    && (toe_ol & TOE_TX_CSUM_OL) != 0)
		drvr->iflist[ifp->idx]->ndev->features |=
			NETIF_F_IP_CSUM;
	else
		drvr->iflist[ifp->idx]->ndev->features &=
			~NETIF_F_IP_CSUM;
612 613

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

616 617
	/* Allow transmit calls */
	netif_start_queue(ndev);
J
John W. Linville 已提交
618
	drvr->bus_if->drvr_up = true;
619
	if (brcmf_cfg80211_up(ndev)) {
620 621 622 623 624 625 626
		brcmf_dbg(ERROR, "failed to bring up cfg80211\n");
		return -1;
	}

	return ret;
}

627 628 629 630 631 632 633 634 635 636
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
};

637 638 639 640 641 642 643 644 645 646
static const struct net_device_ops brcmf_netdev_ops_virt = {
	.ndo_open = brcmf_cfg80211_up,
	.ndo_stop = brcmf_cfg80211_down,
	.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
};

647
int brcmf_net_attach(struct brcmf_if *ifp)
648
{
649
	struct brcmf_pub *drvr = ifp->drvr;
650 651
	struct net_device *ndev;

652 653
	brcmf_dbg(TRACE, "ifidx %d mac %pM\n", ifp->idx, ifp->mac_addr);
	ndev = ifp->ndev;
654

655 656 657 658 659
	/* set appropriate operations */
	if (!ifp->idx)
		ndev->netdev_ops = &brcmf_netdev_ops_pri;
	else
		ndev->netdev_ops = &brcmf_netdev_ops_virt;
660 661 662 663 664 665 666

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

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

667 668
	/* set the mac address */
	memcpy(ndev->dev_addr, ifp->mac_addr, ETH_ALEN);
669 670 671 672 673 674 675 676 677 678 679 680 681 682 683

	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;
}

684
struct brcmf_if *brcmf_add_if(struct brcmf_pub *drvr, int ifidx, s32 bssidx,
685
			      char *name, u8 *addr_mask)
686 687
{
	struct brcmf_if *ifp;
688
	struct net_device *ndev;
689
	int i;
690

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

693
	ifp = drvr->iflist[ifidx];
694 695 696 697 698
	/*
	 * Delete the existing interface before overwriting it
	 * in case we missed the BRCMF_E_IF_DEL event.
	 */
	if (ifp) {
699
		brcmf_dbg(ERROR, "ERROR: netdev:%s already exists\n",
700
			  ifp->ndev->name);
701 702 703 704 705 706 707 708 709
		if (ifidx) {
			netif_stop_queue(ifp->ndev);
			unregister_netdev(ifp->ndev);
			free_netdev(ifp->ndev);
			drvr->iflist[ifidx] = NULL;
		} else {
			brcmf_dbg(ERROR, "ignore IF event\n");
			return ERR_PTR(-EINVAL);
		}
710
	}
711

712
	/* Allocate netdev, including space for private structure */
713 714
	ndev = alloc_netdev(sizeof(struct brcmf_if), name, ether_setup);
	if (!ndev) {
715
		brcmf_dbg(ERROR, "OOM - alloc_netdev\n");
716
		return ERR_PTR(-ENOMEM);
717
	}
718

719 720
	ifp = netdev_priv(ndev);
	ifp->ndev = ndev;
721 722
	ifp->drvr = drvr;
	drvr->iflist[ifidx] = ifp;
723
	ifp->idx = ifidx;
724
	ifp->bssidx = bssidx;
725 726 727 728

	INIT_WORK(&ifp->setmacaddr_work, _brcmf_set_mac_address);
	INIT_WORK(&ifp->multicast_work, _brcmf_set_multicast_list);

729 730 731
	if (addr_mask != NULL)
		for (i = 0; i < ETH_ALEN; i++)
			ifp->mac_addr[i] = drvr->mac[i] ^ addr_mask[i];
732

733 734
	brcmf_dbg(TRACE, " ==== pid:%x, if:%s (%pM) created ===\n",
		  current->pid, ifp->ndev->name, ifp->mac_addr);
735

736
	return ifp;
737 738
}

739
void brcmf_del_if(struct brcmf_pub *drvr, int ifidx)
740 741 742 743 744
{
	struct brcmf_if *ifp;

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

745
	ifp = drvr->iflist[ifidx];
746 747 748 749
	if (!ifp) {
		brcmf_dbg(ERROR, "Null interface\n");
		return;
	}
750 751 752 753 754 755 756 757 758 759 760
	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);
		}

761 762 763
		cancel_work_sync(&ifp->setmacaddr_work);
		cancel_work_sync(&ifp->multicast_work);

764
		unregister_netdev(ifp->ndev);
765
		drvr->iflist[ifidx] = NULL;
766
		if (ifidx == 0)
767
			brcmf_cfg80211_detach(drvr->config);
768
		free_netdev(ifp->ndev);
769 770 771
	}
}

772
int brcmf_attach(uint bus_hdrlen, struct device *dev)
773
{
774
	struct brcmf_pub *drvr = NULL;
775
	int ret = 0;
776 777 778 779

	brcmf_dbg(TRACE, "Enter\n");

	/* Allocate primary brcmf_info */
780 781
	drvr = kzalloc(sizeof(struct brcmf_pub), GFP_ATOMIC);
	if (!drvr)
782
		return -ENOMEM;
783

784
	mutex_init(&drvr->proto_block);
785 786

	/* Link to bus module */
787 788
	drvr->hdrlen = bus_hdrlen;
	drvr->bus_if = dev_get_drvdata(dev);
789
	drvr->bus_if->drvr = drvr;
790
	drvr->dev = dev;
791

792 793 794
	/* create device debugfs folder */
	brcmf_debugfs_attach(drvr);

795
	/* Attach and link in the protocol */
796 797
	ret = brcmf_proto_attach(drvr);
	if (ret != 0) {
798 799 800 801
		brcmf_dbg(ERROR, "brcmf_prot_attach failed\n");
		goto fail;
	}

802 803 804
	/* attach firmware event handler */
	brcmf_fweh_attach(drvr);

805 806
	INIT_LIST_HEAD(&drvr->bus_if->dcmd_list);

807 808
	init_waitqueue_head(&drvr->pend_8021x_wait);

809
	return ret;
810 811

fail:
812
	brcmf_detach(dev);
813

814
	return ret;
815 816
}

817
int brcmf_bus_start(struct device *dev)
818 819
{
	int ret = -1;
820 821
	struct brcmf_bus *bus_if = dev_get_drvdata(dev);
	struct brcmf_pub *drvr = bus_if->drvr;
822
	struct brcmf_if *ifp;
823 824 825 826

	brcmf_dbg(TRACE, "\n");

	/* Bring up the bus */
827
	ret = bus_if->brcmf_bus_init(dev);
828 829 830 831 832
	if (ret != 0) {
		brcmf_dbg(ERROR, "brcmf_sdbrcm_bus_init failed %d\n", ret);
		return ret;
	}

833
	/* add primary networking interface */
834
	ifp = brcmf_add_if(drvr, 0, 0, "wlan%d", NULL);
835 836 837
	if (IS_ERR(ifp))
		return PTR_ERR(ifp);

838 839 840 841 842
	/* signal bus ready */
	bus_if->state = BRCMF_BUS_DATA;

	/* Bus is ready, do any initialization */
	ret = brcmf_c_preinit_dcmds(ifp);
843
	if (ret < 0)
844
		goto fail;
845

846
	drvr->config = brcmf_cfg80211_attach(drvr);
847 848 849 850
	if (drvr->config == NULL) {
		ret = -ENOMEM;
		goto fail;
	}
851

852 853 854 855
	ret = brcmf_fweh_activate_events(ifp);
	if (ret < 0)
		goto fail;

856
	ret = brcmf_net_attach(ifp);
857
fail:
858
	if (ret < 0) {
859
		brcmf_dbg(ERROR, "failed: %d\n", ret);
860 861 862
		if (drvr->config)
			brcmf_cfg80211_detach(drvr->config);
		free_netdev(drvr->iflist[0]->ndev);
863
		drvr->iflist[0] = NULL;
864
		return ret;
865 866
	}

867 868 869 870 871 872 873 874
	return 0;
}

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

	if (drvr) {
875 876
		/* Stop the protocol module */
		brcmf_proto_stop(drvr);
877

878
		/* Stop the bus module */
879
		drvr->bus_if->brcmf_bus_stop(drvr->dev);
880 881 882
	}
}

883
void brcmf_detach(struct device *dev)
884
{
885 886 887
	int i;
	struct brcmf_bus *bus_if = dev_get_drvdata(dev);
	struct brcmf_pub *drvr = bus_if->drvr;
888 889 890

	brcmf_dbg(TRACE, "Enter\n");

891 892
	if (drvr == NULL)
		return;
893

894 895 896
	/* stop firmware event handling */
	brcmf_fweh_detach(drvr);

897 898 899 900
	/* make sure primary interface removed last */
	for (i = BRCMF_MAX_IFS-1; i > -1; i--)
		if (drvr->iflist[i])
			brcmf_del_if(drvr, i);
901

902
	brcmf_bus_detach(drvr);
903

904
	if (drvr->prot) {
905
		brcmf_proto_detach(drvr);
906
	}
907

908
	brcmf_debugfs_detach(drvr);
909 910
	bus_if->drvr = NULL;
	kfree(drvr);
911 912
}

913
static int brcmf_get_pend_8021x_cnt(struct brcmf_pub *drvr)
914
{
915
	return atomic_read(&drvr->pend_8021x_cnt);
916 917 918 919
}

int brcmf_netdev_wait_pend8021x(struct net_device *ndev)
{
920
	struct brcmf_if *ifp = netdev_priv(ndev);
921
	struct brcmf_pub *drvr = ifp->drvr;
922 923 924 925 926 927 928 929 930
	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;
931 932
}

933
static void brcmf_driver_init(struct work_struct *work)
934
{
935 936
	brcmf_debugfs_init();

937
#ifdef CONFIG_BRCMFMAC_SDIO
938
	brcmf_sdio_init();
939
#endif
940
#ifdef CONFIG_BRCMFMAC_USB
941
	brcmf_usb_init();
942
#endif
943 944 945 946 947 948 949 950
}
static DECLARE_WORK(brcmf_driver_work, brcmf_driver_init);

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

951
	return 0;
952 953
}

954
static void __exit brcmfmac_module_exit(void)
955
{
956 957
	cancel_work_sync(&brcmf_driver_work);

958 959 960
#ifdef CONFIG_BRCMFMAC_SDIO
	brcmf_sdio_exit();
#endif
961 962 963
#ifdef CONFIG_BRCMFMAC_USB
	brcmf_usb_exit();
#endif
964
	brcmf_debugfs_exit();
965 966
}

967 968
module_init(brcmfmac_module_init);
module_exit(brcmfmac_module_exit);