⭐ 欢迎来到虫虫下载站! | 📦 资源下载 📁 资源专辑 ℹ️ 关于我们
⭐ 虫虫下载站

📄 ieee80211_rx.c

📁 linux 内核源代码
💻 C
📖 第 1 页 / 共 4 页
字号:
			      crypt->ops->decrypt_mpdu == NULL))			crypt = NULL;		if (!crypt && (fc & IEEE80211_FCTL_PROTECTED)) {			/* This seems to be triggered by some (multicast?)			 * frames from other than current BSS, so just drop the			 * frames silently instead of filling system log with			 * these reports. */			IEEE80211_DEBUG_DROP("Decryption failed (not set)"					     " (SA=%s)\n",					     print_mac(mac, hdr->addr2));			ieee->ieee_stats.rx_discards_undecryptable++;			goto rx_dropped;		}	}#ifdef NOT_YET	if (type != WLAN_FC_TYPE_DATA) {		if (type == WLAN_FC_TYPE_MGMT && stype == WLAN_FC_STYPE_AUTH &&		    fc & IEEE80211_FCTL_PROTECTED && ieee->host_decrypt &&		    (keyidx = hostap_rx_frame_decrypt(ieee, skb, crypt)) < 0) {			printk(KERN_DEBUG "%s: failed to decrypt mgmt::auth "			       "from %s\n", dev->name,			       print_mac(mac, hdr->addr2));			/* TODO: could inform hostapd about this so that it			 * could send auth failure report */			goto rx_dropped;		}		if (ieee80211_rx_frame_mgmt(ieee, skb, rx_stats, type, stype))			goto rx_dropped;		else			goto rx_exit;	}#endif	/* drop duplicate 802.11 retransmissions (IEEE 802.11 Chap. 9.29) */	if (sc == ieee->prev_seq_ctl)		goto rx_dropped;	else		ieee->prev_seq_ctl = sc;	/* Data frame - extract src/dst addresses */	if (skb->len < IEEE80211_3ADDR_LEN)		goto rx_dropped;	switch (fc & (IEEE80211_FCTL_FROMDS | IEEE80211_FCTL_TODS)) {	case IEEE80211_FCTL_FROMDS:		memcpy(dst, hdr->addr1, ETH_ALEN);		memcpy(src, hdr->addr3, ETH_ALEN);		break;	case IEEE80211_FCTL_TODS:		memcpy(dst, hdr->addr3, ETH_ALEN);		memcpy(src, hdr->addr2, ETH_ALEN);		break;	case IEEE80211_FCTL_FROMDS | IEEE80211_FCTL_TODS:		if (skb->len < IEEE80211_4ADDR_LEN)			goto rx_dropped;		memcpy(dst, hdr->addr3, ETH_ALEN);		memcpy(src, hdr->addr4, ETH_ALEN);		break;	case 0:		memcpy(dst, hdr->addr1, ETH_ALEN);		memcpy(src, hdr->addr2, ETH_ALEN);		break;	}#ifdef NOT_YET	if (hostap_rx_frame_wds(ieee, hdr, fc, &wds))		goto rx_dropped;	if (wds) {		skb->dev = dev = wds;		stats = hostap_get_stats(dev);	}	if (ieee->iw_mode == IW_MODE_MASTER && !wds &&	    (fc & (IEEE80211_FCTL_TODS | IEEE80211_FCTL_FROMDS)) ==	    IEEE80211_FCTL_FROMDS && ieee->stadev	    && !compare_ether_addr(hdr->addr2, ieee->assoc_ap_addr)) {		/* Frame from BSSID of the AP for which we are a client */		skb->dev = dev = ieee->stadev;		stats = hostap_get_stats(dev);		from_assoc_ap = 1;	}#endif	dev->last_rx = jiffies;#ifdef NOT_YET	if ((ieee->iw_mode == IW_MODE_MASTER ||	     ieee->iw_mode == IW_MODE_REPEAT) && !from_assoc_ap) {		switch (hostap_handle_sta_rx(ieee, dev, skb, rx_stats,					     wds != NULL)) {		case AP_RX_CONTINUE_NOT_AUTHORIZED:			frame_authorized = 0;			break;		case AP_RX_CONTINUE:			frame_authorized = 1;			break;		case AP_RX_DROP:			goto rx_dropped;		case AP_RX_EXIT:			goto rx_exit;		}	}#endif	/* Nullfunc frames may have PS-bit set, so they must be passed to	 * hostap_handle_sta_rx() before being dropped here. */	stype &= ~IEEE80211_STYPE_QOS_DATA;	if (stype != IEEE80211_STYPE_DATA &&	    stype != IEEE80211_STYPE_DATA_CFACK &&	    stype != IEEE80211_STYPE_DATA_CFPOLL &&	    stype != IEEE80211_STYPE_DATA_CFACKPOLL) {		if (stype != IEEE80211_STYPE_NULLFUNC)			IEEE80211_DEBUG_DROP("RX: dropped data frame "					     "with no data (type=0x%02x, "					     "subtype=0x%02x, len=%d)\n",					     type, stype, skb->len);		goto rx_dropped;	}	/* skb: hdr + (possibly fragmented, possibly encrypted) payload */	if ((fc & IEEE80211_FCTL_PROTECTED) && can_be_decrypted &&	    (keyidx = ieee80211_rx_frame_decrypt(ieee, skb, crypt)) < 0)		goto rx_dropped;	hdr = (struct ieee80211_hdr_4addr *)skb->data;	/* skb: hdr + (possibly fragmented) plaintext payload */	// PR: FIXME: hostap has additional conditions in the "if" below:	// ieee->host_decrypt && (fc & IEEE80211_FCTL_PROTECTED) &&	if ((frag != 0) || (fc & IEEE80211_FCTL_MOREFRAGS)) {		int flen;		struct sk_buff *frag_skb = ieee80211_frag_cache_get(ieee, hdr);		IEEE80211_DEBUG_FRAG("Rx Fragment received (%u)\n", frag);		if (!frag_skb) {			IEEE80211_DEBUG(IEEE80211_DL_RX | IEEE80211_DL_FRAG,					"Rx cannot get skb from fragment "					"cache (morefrag=%d seq=%u frag=%u)\n",					(fc & IEEE80211_FCTL_MOREFRAGS) != 0,					WLAN_GET_SEQ_SEQ(sc), frag);			goto rx_dropped;		}		flen = skb->len;		if (frag != 0)			flen -= hdrlen;		if (frag_skb->tail + flen > frag_skb->end) {			printk(KERN_WARNING "%s: host decrypted and "			       "reassembled frame did not fit skb\n",			       dev->name);			ieee80211_frag_cache_invalidate(ieee, hdr);			goto rx_dropped;		}		if (frag == 0) {			/* copy first fragment (including full headers) into			 * beginning of the fragment cache skb */			skb_copy_from_linear_data(skb, skb_put(frag_skb, flen), flen);		} else {			/* append frame payload to the end of the fragment			 * cache skb */			skb_copy_from_linear_data_offset(skb, hdrlen,				      skb_put(frag_skb, flen), flen);		}		dev_kfree_skb_any(skb);		skb = NULL;		if (fc & IEEE80211_FCTL_MOREFRAGS) {			/* more fragments expected - leave the skb in fragment			 * cache for now; it will be delivered to upper layers			 * after all fragments have been received */			goto rx_exit;		}		/* this was the last fragment and the frame will be		 * delivered, so remove skb from fragment cache */		skb = frag_skb;		hdr = (struct ieee80211_hdr_4addr *)skb->data;		ieee80211_frag_cache_invalidate(ieee, hdr);	}	/* skb: hdr + (possible reassembled) full MSDU payload; possibly still	 * encrypted/authenticated */	if ((fc & IEEE80211_FCTL_PROTECTED) && can_be_decrypted &&	    ieee80211_rx_frame_decrypt_msdu(ieee, skb, keyidx, crypt))		goto rx_dropped;	hdr = (struct ieee80211_hdr_4addr *)skb->data;	if (crypt && !(fc & IEEE80211_FCTL_PROTECTED) && !ieee->open_wep) {		if (		/*ieee->ieee802_1x && */			   ieee80211_is_eapol_frame(ieee, skb)) {			/* pass unencrypted EAPOL frames even if encryption is			 * configured */		} else {			IEEE80211_DEBUG_DROP("encryption configured, but RX "					     "frame not encrypted (SA=%s"					     ")\n", print_mac(mac, hdr->addr2));			goto rx_dropped;		}	}	if (crypt && !(fc & IEEE80211_FCTL_PROTECTED) && !ieee->open_wep &&	    !ieee80211_is_eapol_frame(ieee, skb)) {		IEEE80211_DEBUG_DROP("dropped unencrypted RX data "				     "frame from %s"				     " (drop_unencrypted=1)\n",				     print_mac(mac, hdr->addr2));		goto rx_dropped;	}	/* If the frame was decrypted in hardware, we may need to strip off	 * any security data (IV, ICV, etc) that was left behind */	if (!can_be_decrypted && (fc & IEEE80211_FCTL_PROTECTED) &&	    ieee->host_strip_iv_icv) {		int trimlen = 0;		/* Top two-bits of byte 3 are the key index */		if (skb->len >= hdrlen + 3)			keyidx = skb->data[hdrlen + 3] >> 6;		/* To strip off any security data which appears before the		 * payload, we simply increase hdrlen (as the header gets		 * chopped off immediately below). For the security data which		 * appears after the payload, we use skb_trim. */		switch (ieee->sec.encode_alg[keyidx]) {		case SEC_ALG_WEP:			/* 4 byte IV */			hdrlen += 4;			/* 4 byte ICV */			trimlen = 4;			break;		case SEC_ALG_TKIP:			/* 4 byte IV, 4 byte ExtIV */			hdrlen += 8;			/* 8 byte MIC, 4 byte ICV */			trimlen = 12;			break;		case SEC_ALG_CCMP:			/* 8 byte CCMP header */			hdrlen += 8;			/* 8 byte MIC */			trimlen = 8;			break;		}		if (skb->len < trimlen)			goto rx_dropped;		__skb_trim(skb, skb->len - trimlen);		if (skb->len < hdrlen)			goto rx_dropped;	}	/* skb: hdr + (possible reassembled) full plaintext payload */	payload = skb->data + hdrlen;	ethertype = (payload[6] << 8) | payload[7];#ifdef NOT_YET	/* If IEEE 802.1X is used, check whether the port is authorized to send	 * the received frame. */	if (ieee->ieee802_1x && ieee->iw_mode == IW_MODE_MASTER) {		if (ethertype == ETH_P_PAE) {			printk(KERN_DEBUG "%s: RX: IEEE 802.1X frame\n",			       dev->name);			if (ieee->hostapd && ieee->apdev) {				/* Send IEEE 802.1X frames to the user				 * space daemon for processing */				prism2_rx_80211(ieee->apdev, skb, rx_stats,						PRISM2_RX_MGMT);				ieee->apdevstats.rx_packets++;				ieee->apdevstats.rx_bytes += skb->len;				goto rx_exit;			}		} else if (!frame_authorized) {			printk(KERN_DEBUG "%s: dropped frame from "			       "unauthorized port (IEEE 802.1X): "			       "ethertype=0x%04x\n", dev->name, ethertype);			goto rx_dropped;		}	}#endif	/* convert hdr + possible LLC headers into Ethernet header */	if (skb->len - hdrlen >= 8 &&	    ((memcmp(payload, rfc1042_header, SNAP_SIZE) == 0 &&	      ethertype != ETH_P_AARP && ethertype != ETH_P_IPX) ||	     memcmp(payload, bridge_tunnel_header, SNAP_SIZE) == 0)) {		/* remove RFC1042 or Bridge-Tunnel encapsulation and		 * replace EtherType */		skb_pull(skb, hdrlen + SNAP_SIZE);		memcpy(skb_push(skb, ETH_ALEN), src, ETH_ALEN);		memcpy(skb_push(skb, ETH_ALEN), dst, ETH_ALEN);	} else {		u16 len;		/* Leave Ethernet header part of hdr and full payload */		skb_pull(skb, hdrlen);		len = htons(skb->len);		memcpy(skb_push(skb, 2), &len, 2);		memcpy(skb_push(skb, ETH_ALEN), src, ETH_ALEN);		memcpy(skb_push(skb, ETH_ALEN), dst, ETH_ALEN);	}#ifdef NOT_YET	if (wds && ((fc & (IEEE80211_FCTL_TODS | IEEE80211_FCTL_FROMDS)) ==		    IEEE80211_FCTL_TODS) && skb->len >= ETH_HLEN + ETH_ALEN) {		/* Non-standard frame: get addr4 from its bogus location after		 * the payload */		skb_copy_to_linear_data_offset(skb, ETH_ALEN,					       skb->data + skb->len - ETH_ALEN,					       ETH_ALEN);		skb_trim(skb, skb->len - ETH_ALEN);	}#endif	stats->rx_packets++;	stats->rx_bytes += skb->len;#ifdef NOT_YET	if (ieee->iw_mode == IW_MODE_MASTER && !wds && ieee->ap->bridge_packets) {		if (dst[0] & 0x01) {			/* copy multicast frame both to the higher layers and			 * to the wireless media */			ieee->ap->bridged_multicast++;			skb2 = skb_clone(skb, GFP_ATOMIC);			if (skb2 == NULL)				printk(KERN_DEBUG "%s: skb_clone failed for "				       "multicast frame\n", dev->name);		} else if (hostap_is_sta_assoc(ieee->ap, dst)) {			/* send frame directly to the associated STA using			 * wireless media and not passing to higher layers */			ieee->ap->bridged_unicast++;			skb2 = skb;			skb = NULL;		}	}	if (skb2 != NULL) {		/* send to wireless media */		skb2->dev = dev;		skb2->protocol = __constant_htons(ETH_P_802_3);		skb_reset_mac_header(skb2);		skb_reset_network_header(skb2);		/* skb2->network_header += ETH_HLEN; */		dev_queue_xmit(skb2);	}#endif	if (skb) {		skb->protocol = eth_type_trans(skb, dev);		memset(skb->cb, 0, sizeof(skb->cb));		skb->ip_summed = CHECKSUM_NONE;	/* 802.11 crc not sufficient */		if (netif_rx(skb) == NET_RX_DROP) {			/* netif_rx always succeeds, but it might drop			 * the packet.  If it drops the packet, we log that			 * in our stats. */			IEEE80211_DEBUG_DROP			    ("RX: netif_rx dropped the packet\n");			stats->rx_dropped++;		}	}      rx_exit:#ifdef NOT_YET	if (sta)		hostap_handle_sta_release(sta);#endif	return 1;      rx_dropped:	stats->rx_dropped++;	/* Returning 0 indicates to caller that we have not handled the SKB--	 * so it is still allocated and can be used again by underlying	 * hardware as a DMA target */	return 0;}/* Filter out unrelated packets, call ieee80211_rx[_mgt] * This function takes over the skb, it should not be used again after calling * this function. */void ieee80211_rx_any(struct ieee80211_device *ieee,		     struct sk_buff *skb, struct ieee80211_rx_stats *stats){	struct ieee80211_hdr_4addr *hdr;	int is_packet_for_us;	u16 fc;	if (ieee->iw_mode == IW_MODE_MONITOR) {		if (!ieee80211_rx(ieee, skb, stats))			dev_kfree_skb_irq(skb);		return;	}	if (skb->len < sizeof(struct ieee80211_hdr))		goto drop_free;	hdr = (struct ieee80211_hdr_4addr *)skb->data;	fc = le16_to_cpu(hdr->frame_ctl);	if ((fc & IEEE80211_FCTL_VERS) != 0)		goto drop_free;	switch (fc & IEEE80211_FCTL_FTYPE) {	case IEEE80211_FTYPE_MGMT:		if (skb->len < sizeof(struct ieee80211_hdr_3addr))			goto drop_free;		ieee80211_rx_mgt(ieee, hdr, stats);		dev_kfree_skb_irq(skb);		return;	case IEEE80211_FTYPE_DATA:		break;	case IEEE80211_FTYPE_CTL:		return;	default:		return;	}	is_packet_for_us = 0;	switch (ieee->iw_mode) {	case IW_MODE_ADHOC:		/* our BSS and not from/to DS */		if (memcmp(hdr->addr3, ieee->bssid, ETH_ALEN) == 0)		if ((fc & (IEEE80211_FCTL_TODS+IEEE80211_FCTL_FROMDS)) == 0) {			/* promisc: get all */			if (ieee->dev->flags & IFF_PROMISC)				is_packet_for_us = 1;			/* to us */			else if (memcmp(hdr->addr1, ieee->dev->dev_addr, ETH_ALEN) == 0)				is_packet_for_us = 1;			/* mcast */			else if (is_multicast_ether_addr(hdr->addr1))				is_packet_for_us = 1;		}		break;	case IW_MODE_INFRA:		/* our BSS (== from our AP) and from DS */		if (memcmp(hdr->addr2, ieee->bssid, ETH_ALEN) == 0)		if ((fc & (IEEE80211_FCTL_TODS+IEEE80211_FCTL_FROMDS)) == IEEE80211_FCTL_FROMDS) {			/* promisc: get all */			if (ieee->dev->flags & IFF_PROMISC)				is_packet_for_us = 1;			/* to us */			else if (memcmp(hdr->addr1, ieee->dev->dev_addr, ETH_ALEN) == 0)				is_packet_for_us = 1;			/* mcast */			else if (is_multicast_ether_addr(hdr->addr1)) {				/* not our own packet bcasted from AP */

⌨️ 快捷键说明

复制代码 Ctrl + C
搜索代码 Ctrl + F
全屏模式 F11
切换主题 Ctrl + Shift + D
显示快捷键 ?
增大字号 Ctrl + =
减小字号 Ctrl + -