dhd_linux.c 23.2 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 55 56

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


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

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

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

	while (--i > 0) {
69
		ifp = drvr->iflist[i];
70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85
		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>";
	}

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

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

	return "<if_none>";
}

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

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

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

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

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

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

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

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

	kfree(buf);

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

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

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

	brcmf_dbg(TRACE, "enter\n");

175 176 177 178 179 180 181 182 183 184 185 186
	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);
	}
187 188 189 190
}

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

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

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

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

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

	brcmf_dbg(TRACE, "Enter\n");

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

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

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

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

250 251 252 253 254 255 256 257 258 259 260 261 262 263 264 265
	/* 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);
266 267 268

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

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

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

	brcmf_dbg(TRACE, "Enter\n");

286 287 288 289 290 291 292 293
	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);
		}
294 295
}

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

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

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

	return bcmerror;
}

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

	brcmf_dbg(TRACE, "Enter\n");

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

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

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

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

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

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

		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)
369
			brcmf_host_event(drvr, &ifidx,
370 371 372
					  skb_mac_header(skb),
					  &event, &data);

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

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

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

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

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

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

	if (type == ETH_P_PAE)
408
		atomic_dec(&drvr->pend_8021x_cnt);
409 410 411 412 413

}

static struct net_device_stats *brcmf_netdev_get_stats(struct net_device *ndev)
{
414
	struct brcmf_if *ifp = netdev_priv(ndev);
415
	struct brcmf_bus *bus_if = ifp->drvr->bus_if;
416 417 418 419

	brcmf_dbg(TRACE, "Enter\n");

	/* Copy dongle stats to net device stats */
420 421 422 423 424 425 426 427 428
	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;
429 430 431 432

	return &ifp->stats;
}

433 434 435 436 437
/*
 * 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)
438
{
439
	s32 err;
440

441 442 443 444
	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;
445 446
	}

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

451
	return err;
452 453 454 455 456 457

}

static void brcmf_ethtool_get_drvinfo(struct net_device *ndev,
				    struct ethtool_drvinfo *info)
{
458
	struct brcmf_if *ifp = netdev_priv(ndev);
459
	struct brcmf_pub *drvr = ifp->drvr;
460 461

	sprintf(info->driver, KBUILD_MODNAME);
462 463
	sprintf(info->version, "%lu", drvr->drv_version);
	sprintf(info->bus_info, "%s", dev_name(drvr->dev));
464 465
}

466 467
static const struct ethtool_ops brcmf_ethtool_ops = {
	.get_drvinfo = brcmf_ethtool_get_drvinfo,
468 469
};

470
static int brcmf_ethtool(struct brcmf_if *ifp, void __user *uaddr)
471
{
472
	struct brcmf_pub *drvr = ifp->drvr;
473 474 475 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
	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 */
505
		else if (!drvr->bus_if->drvr_up) {
506 507 508 509 510
			brcmf_dbg(ERROR, "dongle is not up\n");
			return -ENODEV;
		}
		/* finally, report dongle driver type */
		else
511
			sprintf(info.driver, "wl");
512

513
		sprintf(info.version, "%lu", drvr->drv_version);
514 515 516 517 518 519 520 521 522
		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:
523
		ret = brcmf_fil_iovar_int_get(ifp, "toe_ol", &toe_cmpnt);
524 525 526 527 528 529 530 531 532 533 534 535 536 537 538 539 540 541 542 543
		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 */
544
		ret = brcmf_fil_iovar_int_get(ifp, "toe_ol", &toe_cmpnt);
545 546 547 548 549 550 551 552 553 554 555
		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;

556
		ret = brcmf_toe_set(ifp, toe_cmpnt);
557 558 559 560 561 562
		if (ret < 0)
			return ret;

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

		break;

	default:
		return -EOPNOTSUPP;
	}

	return 0;
}

static int brcmf_netdev_ioctl_entry(struct net_device *ndev, struct ifreq *ifr,
				    int cmd)
{
580
	struct brcmf_if *ifp = netdev_priv(ndev);
581
	struct brcmf_pub *drvr = ifp->drvr;
582

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

585
	if (!drvr->iflist[ifp->idx])
586 587 588
		return -1;

	if (cmd == SIOCETHTOOL)
589
		return brcmf_ethtool(ifp, ifr->ifr_data);
590 591 592 593 594 595

	return -EOPNOTSUPP;
}

static int brcmf_netdev_stop(struct net_device *ndev)
{
596
	struct brcmf_if *ifp = netdev_priv(ndev);
597
	struct brcmf_pub *drvr = ifp->drvr;
598 599 600

	brcmf_dbg(TRACE, "Enter\n");
	brcmf_cfg80211_down(drvr->config);
601
	if (drvr->bus_if->drvr_up == 0)
602 603 604
		return 0;

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

	return 0;
}

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

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

621
	if (ifp->idx == 0) {	/* do it only for primary eth0 */
622 623 624 625
		/* 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;
626
		}
627

628
		atomic_set(&drvr->pend_8021x_cnt, 0);
629

630
		memcpy(ndev->dev_addr, drvr->mac, ETH_ALEN);
631 632

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

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

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

	return ret;
}

656 657 658 659 660 661 662 663 664 665
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
};

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

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

674
	ndev = drvr->iflist[ifp->idx]->ndev;
675 676 677
	ndev->netdev_ops = &brcmf_netdev_ops_pri;

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

685
	if (ifp->idx == 1) {
686 687 688 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
		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;
}

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

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

724
	ifp = drvr->iflist[ifidx];
725 726 727 728 729 730 731 732 733 734
	/*
	 * 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);
735
		drvr->iflist[ifidx] = NULL;
736
	}
737

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

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

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

757
	return ifp;
758 759
}

760
void brcmf_del_if(struct brcmf_pub *drvr, int ifidx)
761 762 763 764 765
{
	struct brcmf_if *ifp;

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

766
	ifp = drvr->iflist[ifidx];
767 768 769 770
	if (!ifp) {
		brcmf_dbg(ERROR, "Null interface\n");
		return;
	}
771 772 773 774 775 776 777 778 779 780 781
	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);
		}

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

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

	brcmf_dbg(TRACE, "Enter\n");

	/* Allocate primary brcmf_info */
798 799
	drvr = kzalloc(sizeof(struct brcmf_pub), GFP_ATOMIC);
	if (!drvr)
800
		return -ENOMEM;
801

802
	mutex_init(&drvr->proto_block);
803 804

	/* Link to bus module */
805 806
	drvr->hdrlen = bus_hdrlen;
	drvr->bus_if = dev_get_drvdata(dev);
807
	drvr->bus_if->drvr = drvr;
808
	drvr->dev = dev;
809

810 811 812
	/* create device debugfs folder */
	brcmf_debugfs_attach(drvr);

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

820 821
	INIT_WORK(&drvr->setmacaddr_work, _brcmf_set_mac_address);
	INIT_WORK(&drvr->multicast_work, _brcmf_set_multicast_list);
822

823 824
	INIT_LIST_HEAD(&drvr->bus_if->dcmd_list);

825
	return ret;
826 827

fail:
828
	brcmf_detach(dev);
829

830
	return ret;
831 832
}

833
int brcmf_bus_start(struct device *dev)
834 835
{
	int ret = -1;
836 837
	struct brcmf_bus *bus_if = dev_get_drvdata(dev);
	struct brcmf_pub *drvr = bus_if->drvr;
838
	struct brcmf_if *ifp;
839 840 841 842

	brcmf_dbg(TRACE, "\n");

	/* Bring up the bus */
843
	ret = bus_if->brcmf_bus_init(dev);
844 845 846 847 848
	if (ret != 0) {
		brcmf_dbg(ERROR, "brcmf_sdbrcm_bus_init failed %d\n", ret);
		return ret;
	}

849
	/* add primary networking interface */
850
	ifp = brcmf_add_if(dev, 0, 0, "wlan%d", NULL);
851 852 853
	if (IS_ERR(ifp))
		return PTR_ERR(ifp);

854 855 856 857 858
	/* signal bus ready */
	bus_if->state = BRCMF_BUS_DATA;

	/* Bus is ready, do any initialization */
	ret = brcmf_c_preinit_dcmds(ifp);
859
	if (ret < 0)
860
		goto fail;
861

862
	drvr->config = brcmf_cfg80211_attach(drvr);
863 864 865 866
	if (drvr->config == NULL) {
		ret = -ENOMEM;
		goto fail;
	}
867 868

	ret = brcmf_net_attach(ifp);
869
fail:
870 871
	if (ret < 0) {
		brcmf_dbg(ERROR, "brcmf_net_attach failed");
872 873 874
		if (drvr->config)
			brcmf_cfg80211_detach(drvr->config);
		free_netdev(drvr->iflist[0]->ndev);
875
		drvr->iflist[0] = NULL;
876
		return ret;
877 878
	}

879 880 881 882 883 884 885 886
	return 0;
}

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

	if (drvr) {
887 888
		/* Stop the protocol module */
		brcmf_proto_stop(drvr);
889

890
		/* Stop the bus module */
891
		drvr->bus_if->brcmf_bus_stop(drvr->dev);
892 893 894
	}
}

895
void brcmf_detach(struct device *dev)
896
{
897 898 899
	int i;
	struct brcmf_bus *bus_if = dev_get_drvdata(dev);
	struct brcmf_pub *drvr = bus_if->drvr;
900 901 902 903

	brcmf_dbg(TRACE, "Enter\n");


904 905 906 907
	/* make sure primary interface removed last */
	for (i = BRCMF_MAX_IFS-1; i > -1; i--)
		if (drvr->iflist[i])
			brcmf_del_if(drvr, i);
908

909
	brcmf_bus_detach(drvr);
910

911 912 913
	if (drvr->prot) {
		cancel_work_sync(&drvr->setmacaddr_work);
		cancel_work_sync(&drvr->multicast_work);
914
		brcmf_proto_detach(drvr);
915
	}
916

917
	brcmf_debugfs_detach(drvr);
918 919
	bus_if->drvr = NULL;
	kfree(drvr);
920 921
}

922
static int brcmf_get_pend_8021x_cnt(struct brcmf_pub *drvr)
923
{
924
	return atomic_read(&drvr->pend_8021x_cnt);
925 926 927 928 929 930
}

#define MAX_WAIT_FOR_8021X_TX	10

int brcmf_netdev_wait_pend8021x(struct net_device *ndev)
{
931
	struct brcmf_if *ifp = netdev_priv(ndev);
932
	struct brcmf_pub *drvr = ifp->drvr;
933 934
	int timeout = 10 * HZ / 1000;
	int ntimes = MAX_WAIT_FOR_8021X_TX;
935
	int pend = brcmf_get_pend_8021x_cnt(drvr);
936 937 938 939 940 941 942 943

	while (ntimes && pend) {
		if (pend) {
			set_current_state(TASK_INTERRUPTIBLE);
			schedule_timeout(timeout);
			set_current_state(TASK_RUNNING);
			ntimes--;
		}
944
		pend = brcmf_get_pend_8021x_cnt(drvr);
945 946 947 948
	}
	return pend;
}

949
static void brcmf_driver_init(struct work_struct *work)
950
{
951 952
	brcmf_debugfs_init();

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

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

967
	return 0;
968 969
}

970
static void __exit brcmfmac_module_exit(void)
971
{
972 973
	cancel_work_sync(&brcmf_driver_work);

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

983 984
module_init(brcmfmac_module_init);
module_exit(brcmfmac_module_exit);