dhd_linux.c 22.8 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
	struct net_device *ndev;
651
	u8 temp_addr[ETH_ALEN];
652

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

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

	/*
663
	 * determine mac address to use
664
	 */
665 666 667
	if (is_valid_ether_addr(ifp->mac_addr))
		memcpy(temp_addr, ifp->mac_addr, ETH_ALEN);
	else
668 669 670 671 672 673 674 675 676 677 678 679 680 681 682 683 684 685 686 687 688 689 690 691
		memcpy(temp_addr, drvr->mac, ETH_ALEN);

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

692
struct brcmf_if *brcmf_add_if(struct brcmf_pub *drvr, int ifidx, s32 bssidx,
693
			      char *name, u8 *addr_mask)
694 695
{
	struct brcmf_if *ifp;
696
	struct net_device *ndev;
697
	int i;
698

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

701
	ifp = drvr->iflist[ifidx];
702 703 704 705 706 707 708 709 710 711
	/*
	 * 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);
712
		drvr->iflist[ifidx] = NULL;
713
	}
714

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

722 723
	ifp = netdev_priv(ndev);
	ifp->ndev = ndev;
724 725
	ifp->drvr = drvr;
	drvr->iflist[ifidx] = ifp;
726
	ifp->idx = ifidx;
727
	ifp->bssidx = bssidx;
728 729 730 731

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

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

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

739
	return ifp;
740 741
}

742
void brcmf_del_if(struct brcmf_pub *drvr, int ifidx)
743 744 745 746 747
{
	struct brcmf_if *ifp;

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

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

764 765 766
		cancel_work_sync(&ifp->setmacaddr_work);
		cancel_work_sync(&ifp->multicast_work);

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

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

	brcmf_dbg(TRACE, "Enter\n");

	/* Allocate primary brcmf_info */
783 784
	drvr = kzalloc(sizeof(struct brcmf_pub), GFP_ATOMIC);
	if (!drvr)
785
		return -ENOMEM;
786

787
	mutex_init(&drvr->proto_block);
788 789

	/* Link to bus module */
790 791
	drvr->hdrlen = bus_hdrlen;
	drvr->bus_if = dev_get_drvdata(dev);
792
	drvr->bus_if->drvr = drvr;
793
	drvr->dev = dev;
794

795 796 797
	/* create device debugfs folder */
	brcmf_debugfs_attach(drvr);

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

805 806 807
	/* attach firmware event handler */
	brcmf_fweh_attach(drvr);

808 809
	INIT_LIST_HEAD(&drvr->bus_if->dcmd_list);

810 811
	init_waitqueue_head(&drvr->pend_8021x_wait);

812
	return ret;
813 814

fail:
815
	brcmf_detach(dev);
816

817
	return ret;
818 819
}

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

	brcmf_dbg(TRACE, "\n");

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

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

841 842 843 844 845
	/* signal bus ready */
	bus_if->state = BRCMF_BUS_DATA;

	/* Bus is ready, do any initialization */
	ret = brcmf_c_preinit_dcmds(ifp);
846
	if (ret < 0)
847
		goto fail;
848

849
	drvr->config = brcmf_cfg80211_attach(drvr);
850 851 852 853
	if (drvr->config == NULL) {
		ret = -ENOMEM;
		goto fail;
	}
854

855 856 857 858
	ret = brcmf_fweh_activate_events(ifp);
	if (ret < 0)
		goto fail;

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

870 871 872 873 874 875 876 877
	return 0;
}

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

	if (drvr) {
878 879
		/* Stop the protocol module */
		brcmf_proto_stop(drvr);
880

881
		/* Stop the bus module */
882
		drvr->bus_if->brcmf_bus_stop(drvr->dev);
883 884 885
	}
}

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

	brcmf_dbg(TRACE, "Enter\n");

894 895
	if (drvr == NULL)
		return;
896

897 898 899
	/* stop firmware event handling */
	brcmf_fweh_detach(drvr);

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

905
	brcmf_bus_detach(drvr);
906

907
	if (drvr->prot) {
908
		brcmf_proto_detach(drvr);
909
	}
910

911
	brcmf_debugfs_detach(drvr);
912 913
	bus_if->drvr = NULL;
	kfree(drvr);
914 915
}

916
static int brcmf_get_pend_8021x_cnt(struct brcmf_pub *drvr)
917
{
918
	return atomic_read(&drvr->pend_8021x_cnt);
919 920 921 922
}

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

936
static void brcmf_driver_init(struct work_struct *work)
937
{
938 939
	brcmf_debugfs_init();

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

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

954
	return 0;
955 956
}

957
static void __exit brcmfmac_module_exit(void)
958
{
959 960
	cancel_work_sync(&brcmf_driver_work);

961 962 963
#ifdef CONFIG_BRCMFMAC_SDIO
	brcmf_sdio_exit();
#endif
964 965 966
#ifdef CONFIG_BRCMFMAC_USB
	brcmf_usb_exit();
#endif
967
	brcmf_debugfs_exit();
968 969
}

970 971
module_init(brcmfmac_module_init);
module_exit(brcmfmac_module_exit);