Merge git://git.kernel.org/pub/scm/linux/kernel/git/davem/net
Pull networking updates from David Miller: 1) ax88796 does 64-bit divides which causes link errors on ARM, fix from Arnd Bergmann. 2) Once an improper offload setting is detected on an SKB we don't rate limit the log message so we can very easily live lock. From Ben Greear. 3) Openvswitch cannot report vport configuration changes reliably because it didn't preallocate the netlink notification message before changing state. From Jesse Gross. 4) The effective UID/GID SCM credentials fix, from Linus. 5) When a user explicitly asks for wireless authentication, cfg80211 isn't told about the AP detachment leaving inconsistent state. Fix from Johannes Berg. 6) Fix self-MAC checks in batman-adv on multi-mesh nodes, from Antonio Quartulli. 7) Revert build_skb() change sin IGB driver, can result in memory corruption. From Alexander Duyck. 8) Fix setting VLANs on virtual functions in IXGBE, from Greg Rose. 9) Fix TSO races in qlcnic driver, from Sritej Velaga. 10) In bnx2x the kernel driver and UNDI firmware can try to program the chip at the same time, resulting in corruption. Add proper synchronization. From Dmitry Kravkov. 11) Fix corruption of status block in firmware ram in bxn2x, from Ariel Elior. 12) Fix load balancing hash regression of bonding driver in forwarding configurations, from Eric Dumazet. 13) Fix TS ECR regression in TCP by calling tcp_replace_ts_recent() in all the right spots, from Eric Dumazet. 14) Fix several bonding bugs having to do with address manintainence, including not removing address when configuration operations encounter errors, missed locking on the address lists, missing refcounting on VLAN objects, etc. All from Nikolay Aleksandrov. 15) Add workarounds for firmware bugs in LTE qmi_wwan devices, wherein the devices fail to add a proper ethernet header while on LTE networks but otherwise properly do so on 2G and 3G ones. From Bjørn Mork. * git://git.kernel.org/pub/scm/linux/kernel/git/davem/net: (38 commits) net: fix incorrect credentials passing net: rate-limit warn-bad-offload splats. net: ax88796: avoid 64 bit arithmetic qlge: Update version to 1.00.00.32. qlge: Fix ethtool autoneg advertising. qlge: Fix receive path to drop error frames net: qmi_wwan: prevent duplicate mac address on link (firmware bug workaround) net: qmi_wwan: fixup destination address (firmware bug workaround) net: qmi_wwan: fixup missing ethernet header (firmware bug workaround) bonding: in bond_mc_swap() bond's mc addr list is walked without lock bonding: disable netpoll on enslave failure bonding: primary_slave & curr_active_slave are not cleaned on enslave failure bonding: vlans don't get deleted on enslave failure bonding: mc addresses don't get deleted on enslave failure pkt_sched: fix error return code in fw_change_attrs() irda: small read past the end of array in debug code tcp: call tcp_replace_ts_recent() from tcp_ack() netfilter: xt_rpfilter: skip locally generated broadcast/multicast, too netfilter: ipset: bitmap:ip,mac: fix listing with timeout bonding: fix l23 and l34 load balancing in forwarding path ...
This commit is contained in:
commit
c437d888c7
@ -846,8 +846,10 @@ static void bond_mc_swap(struct bonding *bond, struct slave *new_active,
|
||||
if (bond->dev->flags & IFF_ALLMULTI)
|
||||
dev_set_allmulti(old_active->dev, -1);
|
||||
|
||||
netif_addr_lock_bh(bond->dev);
|
||||
netdev_for_each_mc_addr(ha, bond->dev)
|
||||
dev_mc_del(old_active->dev, ha->addr);
|
||||
netif_addr_unlock_bh(bond->dev);
|
||||
}
|
||||
|
||||
if (new_active) {
|
||||
@ -858,8 +860,10 @@ static void bond_mc_swap(struct bonding *bond, struct slave *new_active,
|
||||
if (bond->dev->flags & IFF_ALLMULTI)
|
||||
dev_set_allmulti(new_active->dev, 1);
|
||||
|
||||
netif_addr_lock_bh(bond->dev);
|
||||
netdev_for_each_mc_addr(ha, bond->dev)
|
||||
dev_mc_add(new_active->dev, ha->addr);
|
||||
netif_addr_unlock_bh(bond->dev);
|
||||
}
|
||||
}
|
||||
|
||||
@ -1901,9 +1905,26 @@ err_dest_symlinks:
|
||||
bond_destroy_slave_symlinks(bond_dev, slave_dev);
|
||||
|
||||
err_detach:
|
||||
if (!USES_PRIMARY(bond->params.mode)) {
|
||||
netif_addr_lock_bh(bond_dev);
|
||||
bond_mc_list_flush(bond_dev, slave_dev);
|
||||
netif_addr_unlock_bh(bond_dev);
|
||||
}
|
||||
bond_del_vlans_from_slave(bond, slave_dev);
|
||||
write_lock_bh(&bond->lock);
|
||||
bond_detach_slave(bond, new_slave);
|
||||
if (bond->primary_slave == new_slave)
|
||||
bond->primary_slave = NULL;
|
||||
write_unlock_bh(&bond->lock);
|
||||
if (bond->curr_active_slave == new_slave) {
|
||||
read_lock(&bond->lock);
|
||||
write_lock_bh(&bond->curr_slave_lock);
|
||||
bond_change_active_slave(bond, NULL);
|
||||
bond_select_active_slave(bond);
|
||||
write_unlock_bh(&bond->curr_slave_lock);
|
||||
read_unlock(&bond->lock);
|
||||
}
|
||||
slave_disable_netpoll(new_slave);
|
||||
|
||||
err_close:
|
||||
slave_dev->priv_flags &= ~IFF_BONDING;
|
||||
@ -3296,20 +3317,22 @@ static int bond_xmit_hash_policy_l2(struct sk_buff *skb, int count)
|
||||
*/
|
||||
static int bond_xmit_hash_policy_l23(struct sk_buff *skb, int count)
|
||||
{
|
||||
struct ethhdr *data = (struct ethhdr *)skb->data;
|
||||
struct iphdr *iph;
|
||||
struct ipv6hdr *ipv6h;
|
||||
const struct ethhdr *data;
|
||||
const struct iphdr *iph;
|
||||
const struct ipv6hdr *ipv6h;
|
||||
u32 v6hash;
|
||||
__be32 *s, *d;
|
||||
const __be32 *s, *d;
|
||||
|
||||
if (skb->protocol == htons(ETH_P_IP) &&
|
||||
skb_network_header_len(skb) >= sizeof(*iph)) {
|
||||
pskb_network_may_pull(skb, sizeof(*iph))) {
|
||||
iph = ip_hdr(skb);
|
||||
data = (struct ethhdr *)skb->data;
|
||||
return ((ntohl(iph->saddr ^ iph->daddr) & 0xffff) ^
|
||||
(data->h_dest[5] ^ data->h_source[5])) % count;
|
||||
} else if (skb->protocol == htons(ETH_P_IPV6) &&
|
||||
skb_network_header_len(skb) >= sizeof(*ipv6h)) {
|
||||
pskb_network_may_pull(skb, sizeof(*ipv6h))) {
|
||||
ipv6h = ipv6_hdr(skb);
|
||||
data = (struct ethhdr *)skb->data;
|
||||
s = &ipv6h->saddr.s6_addr32[0];
|
||||
d = &ipv6h->daddr.s6_addr32[0];
|
||||
v6hash = (s[1] ^ d[1]) ^ (s[2] ^ d[2]) ^ (s[3] ^ d[3]);
|
||||
@ -3328,33 +3351,36 @@ static int bond_xmit_hash_policy_l23(struct sk_buff *skb, int count)
|
||||
static int bond_xmit_hash_policy_l34(struct sk_buff *skb, int count)
|
||||
{
|
||||
u32 layer4_xor = 0;
|
||||
struct iphdr *iph;
|
||||
struct ipv6hdr *ipv6h;
|
||||
__be32 *s, *d;
|
||||
__be16 *layer4hdr;
|
||||
const struct iphdr *iph;
|
||||
const struct ipv6hdr *ipv6h;
|
||||
const __be32 *s, *d;
|
||||
const __be16 *l4 = NULL;
|
||||
__be16 _l4[2];
|
||||
int noff = skb_network_offset(skb);
|
||||
int poff;
|
||||
|
||||
if (skb->protocol == htons(ETH_P_IP) &&
|
||||
skb_network_header_len(skb) >= sizeof(*iph)) {
|
||||
pskb_may_pull(skb, noff + sizeof(*iph))) {
|
||||
iph = ip_hdr(skb);
|
||||
if (!ip_is_fragment(iph) &&
|
||||
(iph->protocol == IPPROTO_TCP ||
|
||||
iph->protocol == IPPROTO_UDP) &&
|
||||
(skb_headlen(skb) - skb_network_offset(skb) >=
|
||||
iph->ihl * sizeof(u32) + sizeof(*layer4hdr) * 2)) {
|
||||
layer4hdr = (__be16 *)((u32 *)iph + iph->ihl);
|
||||
layer4_xor = ntohs(*layer4hdr ^ *(layer4hdr + 1));
|
||||
poff = proto_ports_offset(iph->protocol);
|
||||
|
||||
if (!ip_is_fragment(iph) && poff >= 0) {
|
||||
l4 = skb_header_pointer(skb, noff + (iph->ihl << 2) + poff,
|
||||
sizeof(_l4), &_l4);
|
||||
if (l4)
|
||||
layer4_xor = ntohs(l4[0] ^ l4[1]);
|
||||
}
|
||||
return (layer4_xor ^
|
||||
((ntohl(iph->saddr ^ iph->daddr)) & 0xffff)) % count;
|
||||
} else if (skb->protocol == htons(ETH_P_IPV6) &&
|
||||
skb_network_header_len(skb) >= sizeof(*ipv6h)) {
|
||||
pskb_may_pull(skb, noff + sizeof(*ipv6h))) {
|
||||
ipv6h = ipv6_hdr(skb);
|
||||
if ((ipv6h->nexthdr == IPPROTO_TCP ||
|
||||
ipv6h->nexthdr == IPPROTO_UDP) &&
|
||||
(skb_headlen(skb) - skb_network_offset(skb) >=
|
||||
sizeof(*ipv6h) + sizeof(*layer4hdr) * 2)) {
|
||||
layer4hdr = (__be16 *)(ipv6h + 1);
|
||||
layer4_xor = ntohs(*layer4hdr ^ *(layer4hdr + 1));
|
||||
poff = proto_ports_offset(ipv6h->nexthdr);
|
||||
if (poff >= 0) {
|
||||
l4 = skb_header_pointer(skb, noff + sizeof(*ipv6h) + poff,
|
||||
sizeof(_l4), &_l4);
|
||||
if (l4)
|
||||
layer4_xor = ntohs(l4[0] ^ l4[1]);
|
||||
}
|
||||
s = &ipv6h->saddr.s6_addr32[0];
|
||||
d = &ipv6h->daddr.s6_addr32[0];
|
||||
|
@ -828,7 +828,7 @@ static int ax_probe(struct platform_device *pdev)
|
||||
struct ei_device *ei_local;
|
||||
struct ax_device *ax;
|
||||
struct resource *irq, *mem, *mem2;
|
||||
resource_size_t mem_size, mem2_size = 0;
|
||||
unsigned long mem_size, mem2_size = 0;
|
||||
int ret = 0;
|
||||
|
||||
dev = ax__alloc_ei_netdev(sizeof(struct ax_device));
|
||||
|
@ -2614,6 +2614,9 @@ int bnx2x_nic_load(struct bnx2x *bp, int load_mode)
|
||||
}
|
||||
}
|
||||
|
||||
/* initialize FW coalescing state machines in RAM */
|
||||
bnx2x_update_coalesce(bp);
|
||||
|
||||
/* setup the leading queue */
|
||||
rc = bnx2x_setup_leading(bp);
|
||||
if (rc) {
|
||||
@ -4580,11 +4583,11 @@ static void storm_memset_hc_disable(struct bnx2x *bp, u8 port,
|
||||
u32 enable_flag = disable ? 0 : (1 << HC_INDEX_DATA_HC_ENABLED_SHIFT);
|
||||
u32 addr = BAR_CSTRORM_INTMEM +
|
||||
CSTORM_STATUS_BLOCK_DATA_FLAGS_OFFSET(fw_sb_id, sb_index);
|
||||
u16 flags = REG_RD16(bp, addr);
|
||||
u8 flags = REG_RD8(bp, addr);
|
||||
/* clear and set */
|
||||
flags &= ~HC_INDEX_DATA_HC_ENABLED;
|
||||
flags |= enable_flag;
|
||||
REG_WR16(bp, addr, flags);
|
||||
REG_WR8(bp, addr, flags);
|
||||
DP(NETIF_MSG_IFUP,
|
||||
"port %x fw_sb_id %d sb_index %d disable %d\n",
|
||||
port, fw_sb_id, sb_index, disable);
|
||||
|
@ -9878,6 +9878,10 @@ static int bnx2x_prev_unload_common(struct bnx2x *bp)
|
||||
REG_RD(bp, NIG_REG_NIG_INT_STS_CLR_0);
|
||||
}
|
||||
}
|
||||
if (!CHIP_IS_E1x(bp))
|
||||
/* block FW from writing to host */
|
||||
REG_WR(bp, PGLUE_B_REG_INTERNAL_PFID_ENABLE_MASTER, 0);
|
||||
|
||||
/* wait until BRB is empty */
|
||||
tmp_reg = REG_RD(bp, BRB1_REG_NUM_OF_FULL_BLOCKS);
|
||||
while (timer_count) {
|
||||
|
@ -284,18 +284,10 @@ struct igb_q_vector {
|
||||
enum e1000_ring_flags_t {
|
||||
IGB_RING_FLAG_RX_SCTP_CSUM,
|
||||
IGB_RING_FLAG_RX_LB_VLAN_BSWAP,
|
||||
IGB_RING_FLAG_RX_BUILD_SKB_ENABLED,
|
||||
IGB_RING_FLAG_TX_CTX_IDX,
|
||||
IGB_RING_FLAG_TX_DETECT_HANG
|
||||
};
|
||||
|
||||
#define ring_uses_build_skb(ring) \
|
||||
test_bit(IGB_RING_FLAG_RX_BUILD_SKB_ENABLED, &(ring)->flags)
|
||||
#define set_ring_build_skb_enabled(ring) \
|
||||
set_bit(IGB_RING_FLAG_RX_BUILD_SKB_ENABLED, &(ring)->flags)
|
||||
#define clear_ring_build_skb_enabled(ring) \
|
||||
clear_bit(IGB_RING_FLAG_RX_BUILD_SKB_ENABLED, &(ring)->flags)
|
||||
|
||||
#define IGB_TXD_DCMD (E1000_ADVTXD_DCMD_EOP | E1000_ADVTXD_DCMD_RS)
|
||||
|
||||
#define IGB_RX_DESC(R, i) \
|
||||
|
@ -3350,20 +3350,6 @@ void igb_configure_rx_ring(struct igb_adapter *adapter,
|
||||
wr32(E1000_RXDCTL(reg_idx), rxdctl);
|
||||
}
|
||||
|
||||
static void igb_set_rx_buffer_len(struct igb_adapter *adapter,
|
||||
struct igb_ring *rx_ring)
|
||||
{
|
||||
#define IGB_MAX_BUILD_SKB_SIZE \
|
||||
(SKB_WITH_OVERHEAD(IGB_RX_BUFSZ) - \
|
||||
(NET_SKB_PAD + NET_IP_ALIGN + IGB_TS_HDR_LEN))
|
||||
|
||||
/* set build_skb flag */
|
||||
if (adapter->max_frame_size <= IGB_MAX_BUILD_SKB_SIZE)
|
||||
set_ring_build_skb_enabled(rx_ring);
|
||||
else
|
||||
clear_ring_build_skb_enabled(rx_ring);
|
||||
}
|
||||
|
||||
/**
|
||||
* igb_configure_rx - Configure receive Unit after Reset
|
||||
* @adapter: board private structure
|
||||
@ -3383,11 +3369,8 @@ static void igb_configure_rx(struct igb_adapter *adapter)
|
||||
|
||||
/* Setup the HW Rx Head and Tail Descriptor Pointers and
|
||||
* the Base and Length of the Rx Descriptor Ring */
|
||||
for (i = 0; i < adapter->num_rx_queues; i++) {
|
||||
struct igb_ring *rx_ring = adapter->rx_ring[i];
|
||||
igb_set_rx_buffer_len(adapter, rx_ring);
|
||||
igb_configure_rx_ring(adapter, rx_ring);
|
||||
}
|
||||
for (i = 0; i < adapter->num_rx_queues; i++)
|
||||
igb_configure_rx_ring(adapter, adapter->rx_ring[i]);
|
||||
}
|
||||
|
||||
/**
|
||||
@ -6203,78 +6186,6 @@ static bool igb_add_rx_frag(struct igb_ring *rx_ring,
|
||||
return igb_can_reuse_rx_page(rx_buffer, page, truesize);
|
||||
}
|
||||
|
||||
static struct sk_buff *igb_build_rx_buffer(struct igb_ring *rx_ring,
|
||||
union e1000_adv_rx_desc *rx_desc)
|
||||
{
|
||||
struct igb_rx_buffer *rx_buffer;
|
||||
struct sk_buff *skb;
|
||||
struct page *page;
|
||||
void *page_addr;
|
||||
unsigned int size = le16_to_cpu(rx_desc->wb.upper.length);
|
||||
#if (PAGE_SIZE < 8192)
|
||||
unsigned int truesize = IGB_RX_BUFSZ;
|
||||
#else
|
||||
unsigned int truesize = SKB_DATA_ALIGN(sizeof(struct skb_shared_info)) +
|
||||
SKB_DATA_ALIGN(NET_SKB_PAD +
|
||||
NET_IP_ALIGN +
|
||||
size);
|
||||
#endif
|
||||
|
||||
/* If we spanned a buffer we have a huge mess so test for it */
|
||||
BUG_ON(unlikely(!igb_test_staterr(rx_desc, E1000_RXD_STAT_EOP)));
|
||||
|
||||
rx_buffer = &rx_ring->rx_buffer_info[rx_ring->next_to_clean];
|
||||
page = rx_buffer->page;
|
||||
prefetchw(page);
|
||||
|
||||
page_addr = page_address(page) + rx_buffer->page_offset;
|
||||
|
||||
/* prefetch first cache line of first page */
|
||||
prefetch(page_addr + NET_SKB_PAD + NET_IP_ALIGN);
|
||||
#if L1_CACHE_BYTES < 128
|
||||
prefetch(page_addr + L1_CACHE_BYTES + NET_SKB_PAD + NET_IP_ALIGN);
|
||||
#endif
|
||||
|
||||
/* build an skb to around the page buffer */
|
||||
skb = build_skb(page_addr, truesize);
|
||||
if (unlikely(!skb)) {
|
||||
rx_ring->rx_stats.alloc_failed++;
|
||||
return NULL;
|
||||
}
|
||||
|
||||
/* we are reusing so sync this buffer for CPU use */
|
||||
dma_sync_single_range_for_cpu(rx_ring->dev,
|
||||
rx_buffer->dma,
|
||||
rx_buffer->page_offset,
|
||||
IGB_RX_BUFSZ,
|
||||
DMA_FROM_DEVICE);
|
||||
|
||||
/* update pointers within the skb to store the data */
|
||||
skb_reserve(skb, NET_IP_ALIGN + NET_SKB_PAD);
|
||||
__skb_put(skb, size);
|
||||
|
||||
/* pull timestamp out of packet data */
|
||||
if (igb_test_staterr(rx_desc, E1000_RXDADV_STAT_TSIP)) {
|
||||
igb_ptp_rx_pktstamp(rx_ring->q_vector, skb->data, skb);
|
||||
__skb_pull(skb, IGB_TS_HDR_LEN);
|
||||
}
|
||||
|
||||
if (igb_can_reuse_rx_page(rx_buffer, page, truesize)) {
|
||||
/* hand second half of page back to the ring */
|
||||
igb_reuse_rx_page(rx_ring, rx_buffer);
|
||||
} else {
|
||||
/* we are not reusing the buffer so unmap it */
|
||||
dma_unmap_page(rx_ring->dev, rx_buffer->dma,
|
||||
PAGE_SIZE, DMA_FROM_DEVICE);
|
||||
}
|
||||
|
||||
/* clear contents of buffer_info */
|
||||
rx_buffer->dma = 0;
|
||||
rx_buffer->page = NULL;
|
||||
|
||||
return skb;
|
||||
}
|
||||
|
||||
static struct sk_buff *igb_fetch_rx_buffer(struct igb_ring *rx_ring,
|
||||
union e1000_adv_rx_desc *rx_desc,
|
||||
struct sk_buff *skb)
|
||||
@ -6690,10 +6601,7 @@ static bool igb_clean_rx_irq(struct igb_q_vector *q_vector, const int budget)
|
||||
rmb();
|
||||
|
||||
/* retrieve a buffer from the ring */
|
||||
if (ring_uses_build_skb(rx_ring))
|
||||
skb = igb_build_rx_buffer(rx_ring, rx_desc);
|
||||
else
|
||||
skb = igb_fetch_rx_buffer(rx_ring, rx_desc, skb);
|
||||
skb = igb_fetch_rx_buffer(rx_ring, rx_desc, skb);
|
||||
|
||||
/* exit if we failed to retrieve a buffer */
|
||||
if (!skb)
|
||||
@ -6780,14 +6688,6 @@ static bool igb_alloc_mapped_page(struct igb_ring *rx_ring,
|
||||
return true;
|
||||
}
|
||||
|
||||
static inline unsigned int igb_rx_offset(struct igb_ring *rx_ring)
|
||||
{
|
||||
if (ring_uses_build_skb(rx_ring))
|
||||
return NET_SKB_PAD + NET_IP_ALIGN;
|
||||
else
|
||||
return 0;
|
||||
}
|
||||
|
||||
/**
|
||||
* igb_alloc_rx_buffers - Replace used receive buffers; packet split
|
||||
* @adapter: address of board private structure
|
||||
@ -6814,9 +6714,7 @@ void igb_alloc_rx_buffers(struct igb_ring *rx_ring, u16 cleaned_count)
|
||||
* Refresh the desc even if buffer_addrs didn't change
|
||||
* because each write-back erases this info.
|
||||
*/
|
||||
rx_desc->read.pkt_addr = cpu_to_le64(bi->dma +
|
||||
bi->page_offset +
|
||||
igb_rx_offset(rx_ring));
|
||||
rx_desc->read.pkt_addr = cpu_to_le64(bi->dma + bi->page_offset);
|
||||
|
||||
rx_desc++;
|
||||
bi++;
|
||||
|
@ -1049,6 +1049,12 @@ int ixgbe_ndo_set_vf_vlan(struct net_device *netdev, int vf, u16 vlan, u8 qos)
|
||||
if ((vf >= adapter->num_vfs) || (vlan > 4095) || (qos > 7))
|
||||
return -EINVAL;
|
||||
if (vlan || qos) {
|
||||
if (adapter->vfinfo[vf].pf_vlan)
|
||||
err = ixgbe_set_vf_vlan(adapter, false,
|
||||
adapter->vfinfo[vf].pf_vlan,
|
||||
vf);
|
||||
if (err)
|
||||
goto out;
|
||||
err = ixgbe_set_vf_vlan(adapter, true, vlan, vf);
|
||||
if (err)
|
||||
goto out;
|
||||
|
@ -1500,6 +1500,12 @@ int qlcnic_83xx_loopback_test(struct net_device *netdev, u8 mode)
|
||||
}
|
||||
} while ((adapter->ahw->linkup && ahw->has_link_events) != 1);
|
||||
|
||||
/* Make sure carrier is off and queue is stopped during loopback */
|
||||
if (netif_running(netdev)) {
|
||||
netif_carrier_off(netdev);
|
||||
netif_stop_queue(netdev);
|
||||
}
|
||||
|
||||
ret = qlcnic_do_lb_test(adapter, mode);
|
||||
|
||||
qlcnic_83xx_clear_lb_mode(adapter, mode);
|
||||
@ -2780,6 +2786,7 @@ static u64 *qlcnic_83xx_fill_stats(struct qlcnic_adapter *adapter,
|
||||
void qlcnic_83xx_get_stats(struct qlcnic_adapter *adapter, u64 *data)
|
||||
{
|
||||
struct qlcnic_cmd_args cmd;
|
||||
struct net_device *netdev = adapter->netdev;
|
||||
int ret = 0;
|
||||
|
||||
qlcnic_alloc_mbx_args(&cmd, adapter, QLCNIC_CMD_GET_STATISTICS);
|
||||
@ -2789,7 +2796,7 @@ void qlcnic_83xx_get_stats(struct qlcnic_adapter *adapter, u64 *data)
|
||||
data = qlcnic_83xx_fill_stats(adapter, &cmd, data,
|
||||
QLC_83XX_STAT_TX, &ret);
|
||||
if (ret) {
|
||||
dev_info(&adapter->pdev->dev, "Error getting MAC stats\n");
|
||||
netdev_err(netdev, "Error getting Tx stats\n");
|
||||
goto out;
|
||||
}
|
||||
/* Get MAC stats */
|
||||
@ -2799,8 +2806,7 @@ void qlcnic_83xx_get_stats(struct qlcnic_adapter *adapter, u64 *data)
|
||||
data = qlcnic_83xx_fill_stats(adapter, &cmd, data,
|
||||
QLC_83XX_STAT_MAC, &ret);
|
||||
if (ret) {
|
||||
dev_info(&adapter->pdev->dev,
|
||||
"Error getting Rx stats\n");
|
||||
netdev_err(netdev, "Error getting MAC stats\n");
|
||||
goto out;
|
||||
}
|
||||
/* Get Rx stats */
|
||||
@ -2810,8 +2816,7 @@ void qlcnic_83xx_get_stats(struct qlcnic_adapter *adapter, u64 *data)
|
||||
data = qlcnic_83xx_fill_stats(adapter, &cmd, data,
|
||||
QLC_83XX_STAT_RX, &ret);
|
||||
if (ret)
|
||||
dev_info(&adapter->pdev->dev,
|
||||
"Error getting Tx stats\n");
|
||||
netdev_err(netdev, "Error getting Rx stats\n");
|
||||
out:
|
||||
qlcnic_free_mbx_args(&cmd);
|
||||
}
|
||||
|
@ -358,8 +358,7 @@ set_flags:
|
||||
memcpy(&first_desc->eth_addr, skb->data, ETH_ALEN);
|
||||
}
|
||||
opcode = TX_ETHER_PKT;
|
||||
if ((adapter->netdev->features & (NETIF_F_TSO | NETIF_F_TSO6)) &&
|
||||
skb_shinfo(skb)->gso_size > 0) {
|
||||
if (skb_is_gso(skb)) {
|
||||
hdr_len = skb_transport_offset(skb) + tcp_hdrlen(skb);
|
||||
first_desc->mss = cpu_to_le16(skb_shinfo(skb)->gso_size);
|
||||
first_desc->total_hdr_length = hdr_len;
|
||||
|
@ -18,7 +18,7 @@
|
||||
*/
|
||||
#define DRV_NAME "qlge"
|
||||
#define DRV_STRING "QLogic 10 Gigabit PCI-E Ethernet Driver "
|
||||
#define DRV_VERSION "v1.00.00.31"
|
||||
#define DRV_VERSION "v1.00.00.32"
|
||||
|
||||
#define WQ_ADDR_ALIGN 0x3 /* 4 byte alignment */
|
||||
|
||||
|
@ -379,13 +379,13 @@ static int ql_get_settings(struct net_device *ndev,
|
||||
|
||||
ecmd->supported = SUPPORTED_10000baseT_Full;
|
||||
ecmd->advertising = ADVERTISED_10000baseT_Full;
|
||||
ecmd->autoneg = AUTONEG_ENABLE;
|
||||
ecmd->transceiver = XCVR_EXTERNAL;
|
||||
if ((qdev->link_status & STS_LINK_TYPE_MASK) ==
|
||||
STS_LINK_TYPE_10GBASET) {
|
||||
ecmd->supported |= (SUPPORTED_TP | SUPPORTED_Autoneg);
|
||||
ecmd->advertising |= (ADVERTISED_TP | ADVERTISED_Autoneg);
|
||||
ecmd->port = PORT_TP;
|
||||
ecmd->autoneg = AUTONEG_ENABLE;
|
||||
} else {
|
||||
ecmd->supported |= SUPPORTED_FIBRE;
|
||||
ecmd->advertising |= ADVERTISED_FIBRE;
|
||||
|
@ -1434,11 +1434,13 @@ map_error:
|
||||
}
|
||||
|
||||
/* Categorizing receive firmware frame errors */
|
||||
static void ql_categorize_rx_err(struct ql_adapter *qdev, u8 rx_err)
|
||||
static void ql_categorize_rx_err(struct ql_adapter *qdev, u8 rx_err,
|
||||
struct rx_ring *rx_ring)
|
||||
{
|
||||
struct nic_stats *stats = &qdev->nic_stats;
|
||||
|
||||
stats->rx_err_count++;
|
||||
rx_ring->rx_errors++;
|
||||
|
||||
switch (rx_err & IB_MAC_IOCB_RSP_ERR_MASK) {
|
||||
case IB_MAC_IOCB_RSP_ERR_CODE_ERR:
|
||||
@ -1474,6 +1476,12 @@ static void ql_process_mac_rx_gro_page(struct ql_adapter *qdev,
|
||||
struct bq_desc *lbq_desc = ql_get_curr_lchunk(qdev, rx_ring);
|
||||
struct napi_struct *napi = &rx_ring->napi;
|
||||
|
||||
/* Frame error, so drop the packet. */
|
||||
if (ib_mac_rsp->flags2 & IB_MAC_IOCB_RSP_ERR_MASK) {
|
||||
ql_categorize_rx_err(qdev, ib_mac_rsp->flags2, rx_ring);
|
||||
put_page(lbq_desc->p.pg_chunk.page);
|
||||
return;
|
||||
}
|
||||
napi->dev = qdev->ndev;
|
||||
|
||||
skb = napi_get_frags(napi);
|
||||
@ -1529,6 +1537,12 @@ static void ql_process_mac_rx_page(struct ql_adapter *qdev,
|
||||
addr = lbq_desc->p.pg_chunk.va;
|
||||
prefetch(addr);
|
||||
|
||||
/* Frame error, so drop the packet. */
|
||||
if (ib_mac_rsp->flags2 & IB_MAC_IOCB_RSP_ERR_MASK) {
|
||||
ql_categorize_rx_err(qdev, ib_mac_rsp->flags2, rx_ring);
|
||||
goto err_out;
|
||||
}
|
||||
|
||||
/* The max framesize filter on this chip is set higher than
|
||||
* MTU since FCoE uses 2k frames.
|
||||
*/
|
||||
@ -1614,6 +1628,13 @@ static void ql_process_mac_rx_skb(struct ql_adapter *qdev,
|
||||
memcpy(skb_put(new_skb, length), skb->data, length);
|
||||
skb = new_skb;
|
||||
|
||||
/* Frame error, so drop the packet. */
|
||||
if (ib_mac_rsp->flags2 & IB_MAC_IOCB_RSP_ERR_MASK) {
|
||||
ql_categorize_rx_err(qdev, ib_mac_rsp->flags2, rx_ring);
|
||||
dev_kfree_skb_any(skb);
|
||||
return;
|
||||
}
|
||||
|
||||
/* loopback self test for ethtool */
|
||||
if (test_bit(QL_SELFTEST, &qdev->flags)) {
|
||||
ql_check_lb_frame(qdev, skb);
|
||||
@ -1919,6 +1940,13 @@ static void ql_process_mac_split_rx_intr(struct ql_adapter *qdev,
|
||||
return;
|
||||
}
|
||||
|
||||
/* Frame error, so drop the packet. */
|
||||
if (ib_mac_rsp->flags2 & IB_MAC_IOCB_RSP_ERR_MASK) {
|
||||
ql_categorize_rx_err(qdev, ib_mac_rsp->flags2, rx_ring);
|
||||
dev_kfree_skb_any(skb);
|
||||
return;
|
||||
}
|
||||
|
||||
/* The max framesize filter on this chip is set higher than
|
||||
* MTU since FCoE uses 2k frames.
|
||||
*/
|
||||
@ -2000,12 +2028,6 @@ static unsigned long ql_process_mac_rx_intr(struct ql_adapter *qdev,
|
||||
|
||||
QL_DUMP_IB_MAC_RSP(ib_mac_rsp);
|
||||
|
||||
/* Frame error, so drop the packet. */
|
||||
if (ib_mac_rsp->flags2 & IB_MAC_IOCB_RSP_ERR_MASK) {
|
||||
ql_categorize_rx_err(qdev, ib_mac_rsp->flags2);
|
||||
return (unsigned long)length;
|
||||
}
|
||||
|
||||
if (ib_mac_rsp->flags4 & IB_MAC_IOCB_RSP_HV) {
|
||||
/* The data and headers are split into
|
||||
* separate buffers.
|
||||
|
@ -13,6 +13,7 @@
|
||||
#include <linux/module.h>
|
||||
#include <linux/netdevice.h>
|
||||
#include <linux/ethtool.h>
|
||||
#include <linux/etherdevice.h>
|
||||
#include <linux/mii.h>
|
||||
#include <linux/usb.h>
|
||||
#include <linux/usb/cdc.h>
|
||||
@ -52,6 +53,96 @@ struct qmi_wwan_state {
|
||||
struct usb_interface *data;
|
||||
};
|
||||
|
||||
/* default ethernet address used by the modem */
|
||||
static const u8 default_modem_addr[ETH_ALEN] = {0x02, 0x50, 0xf3};
|
||||
|
||||
/* Make up an ethernet header if the packet doesn't have one.
|
||||
*
|
||||
* A firmware bug common among several devices cause them to send raw
|
||||
* IP packets under some circumstances. There is no way for the
|
||||
* driver/host to know when this will happen. And even when the bug
|
||||
* hits, some packets will still arrive with an intact header.
|
||||
*
|
||||
* The supported devices are only capably of sending IPv4, IPv6 and
|
||||
* ARP packets on a point-to-point link. Any packet with an ethernet
|
||||
* header will have either our address or a broadcast/multicast
|
||||
* address as destination. ARP packets will always have a header.
|
||||
*
|
||||
* This means that this function will reliably add the appropriate
|
||||
* header iff necessary, provided our hardware address does not start
|
||||
* with 4 or 6.
|
||||
*
|
||||
* Another common firmware bug results in all packets being addressed
|
||||
* to 00:a0:c6:00:00:00 despite the host address being different.
|
||||
* This function will also fixup such packets.
|
||||
*/
|
||||
static int qmi_wwan_rx_fixup(struct usbnet *dev, struct sk_buff *skb)
|
||||
{
|
||||
__be16 proto;
|
||||
|
||||
/* usbnet rx_complete guarantees that skb->len is at least
|
||||
* hard_header_len, so we can inspect the dest address without
|
||||
* checking skb->len
|
||||
*/
|
||||
switch (skb->data[0] & 0xf0) {
|
||||
case 0x40:
|
||||
proto = htons(ETH_P_IP);
|
||||
break;
|
||||
case 0x60:
|
||||
proto = htons(ETH_P_IPV6);
|
||||
break;
|
||||
case 0x00:
|
||||
if (is_multicast_ether_addr(skb->data))
|
||||
return 1;
|
||||
/* possibly bogus destination - rewrite just in case */
|
||||
skb_reset_mac_header(skb);
|
||||
goto fix_dest;
|
||||
default:
|
||||
/* pass along other packets without modifications */
|
||||
return 1;
|
||||
}
|
||||
if (skb_headroom(skb) < ETH_HLEN)
|
||||
return 0;
|
||||
skb_push(skb, ETH_HLEN);
|
||||
skb_reset_mac_header(skb);
|
||||
eth_hdr(skb)->h_proto = proto;
|
||||
memset(eth_hdr(skb)->h_source, 0, ETH_ALEN);
|
||||
fix_dest:
|
||||
memcpy(eth_hdr(skb)->h_dest, dev->net->dev_addr, ETH_ALEN);
|
||||
return 1;
|
||||
}
|
||||
|
||||
/* very simplistic detection of IPv4 or IPv6 headers */
|
||||
static bool possibly_iphdr(const char *data)
|
||||
{
|
||||
return (data[0] & 0xd0) == 0x40;
|
||||
}
|
||||
|
||||
/* disallow addresses which may be confused with IP headers */
|
||||
static int qmi_wwan_mac_addr(struct net_device *dev, void *p)
|
||||
{
|
||||
int ret;
|
||||
struct sockaddr *addr = p;
|
||||
|
||||
ret = eth_prepare_mac_addr_change(dev, p);
|
||||
if (ret < 0)
|
||||
return ret;
|
||||
if (possibly_iphdr(addr->sa_data))
|
||||
return -EADDRNOTAVAIL;
|
||||
eth_commit_mac_addr_change(dev, p);
|
||||
return 0;
|
||||
}
|
||||
|
||||
static const struct net_device_ops qmi_wwan_netdev_ops = {
|
||||
.ndo_open = usbnet_open,
|
||||
.ndo_stop = usbnet_stop,
|
||||
.ndo_start_xmit = usbnet_start_xmit,
|
||||
.ndo_tx_timeout = usbnet_tx_timeout,
|
||||
.ndo_change_mtu = usbnet_change_mtu,
|
||||
.ndo_set_mac_address = qmi_wwan_mac_addr,
|
||||
.ndo_validate_addr = eth_validate_addr,
|
||||
};
|
||||
|
||||
/* using a counter to merge subdriver requests with our own into a combined state */
|
||||
static int qmi_wwan_manage_power(struct usbnet *dev, int on)
|
||||
{
|
||||
@ -229,6 +320,18 @@ next_desc:
|
||||
usb_driver_release_interface(driver, info->data);
|
||||
}
|
||||
|
||||
/* Never use the same address on both ends of the link, even
|
||||
* if the buggy firmware told us to.
|
||||
*/
|
||||
if (!compare_ether_addr(dev->net->dev_addr, default_modem_addr))
|
||||
eth_hw_addr_random(dev->net);
|
||||
|
||||
/* make MAC addr easily distinguishable from an IP header */
|
||||
if (possibly_iphdr(dev->net->dev_addr)) {
|
||||
dev->net->dev_addr[0] |= 0x02; /* set local assignment bit */
|
||||
dev->net->dev_addr[0] &= 0xbf; /* clear "IP" bit */
|
||||
}
|
||||
dev->net->netdev_ops = &qmi_wwan_netdev_ops;
|
||||
err:
|
||||
return status;
|
||||
}
|
||||
@ -307,6 +410,7 @@ static const struct driver_info qmi_wwan_info = {
|
||||
.bind = qmi_wwan_bind,
|
||||
.unbind = qmi_wwan_unbind,
|
||||
.manage_power = qmi_wwan_manage_power,
|
||||
.rx_fixup = qmi_wwan_rx_fixup,
|
||||
};
|
||||
|
||||
#define HUAWEI_VENDOR_ID 0x12D1
|
||||
|
@ -519,7 +519,7 @@ static const u32 ar9580_1p0_mac_core[][2] = {
|
||||
{0x00008258, 0x00000000},
|
||||
{0x0000825c, 0x40000000},
|
||||
{0x00008260, 0x00080922},
|
||||
{0x00008264, 0x9bc00010},
|
||||
{0x00008264, 0x9d400010},
|
||||
{0x00008268, 0xffffffff},
|
||||
{0x0000826c, 0x0000ffff},
|
||||
{0x00008270, 0x00000000},
|
||||
|
@ -143,14 +143,14 @@ channel_detector_create(struct dfs_pattern_detector *dpd, u16 freq)
|
||||
u32 sz, i;
|
||||
struct channel_detector *cd;
|
||||
|
||||
cd = kmalloc(sizeof(*cd), GFP_KERNEL);
|
||||
cd = kmalloc(sizeof(*cd), GFP_ATOMIC);
|
||||
if (cd == NULL)
|
||||
goto fail;
|
||||
|
||||
INIT_LIST_HEAD(&cd->head);
|
||||
cd->freq = freq;
|
||||
sz = sizeof(cd->detectors) * dpd->num_radar_types;
|
||||
cd->detectors = kzalloc(sz, GFP_KERNEL);
|
||||
cd->detectors = kzalloc(sz, GFP_ATOMIC);
|
||||
if (cd->detectors == NULL)
|
||||
goto fail;
|
||||
|
||||
|
@ -218,7 +218,7 @@ static bool pulse_queue_enqueue(struct pri_detector *pde, u64 ts)
|
||||
{
|
||||
struct pulse_elem *p = pool_get_pulse_elem();
|
||||
if (p == NULL) {
|
||||
p = kmalloc(sizeof(*p), GFP_KERNEL);
|
||||
p = kmalloc(sizeof(*p), GFP_ATOMIC);
|
||||
if (p == NULL) {
|
||||
DFS_POOL_STAT_INC(pulse_alloc_error);
|
||||
return false;
|
||||
@ -299,7 +299,7 @@ static bool pseq_handler_create_sequences(struct pri_detector *pde,
|
||||
ps.deadline_ts = ps.first_ts + ps.dur;
|
||||
new_ps = pool_get_pseq_elem();
|
||||
if (new_ps == NULL) {
|
||||
new_ps = kmalloc(sizeof(*new_ps), GFP_KERNEL);
|
||||
new_ps = kmalloc(sizeof(*new_ps), GFP_ATOMIC);
|
||||
if (new_ps == NULL) {
|
||||
DFS_POOL_STAT_INC(pseq_alloc_error);
|
||||
return false;
|
||||
|
@ -796,7 +796,7 @@ static int ath9k_init_firmware_version(struct ath9k_htc_priv *priv)
|
||||
* required version.
|
||||
*/
|
||||
if (priv->fw_version_major != MAJOR_VERSION_REQ ||
|
||||
priv->fw_version_minor != MINOR_VERSION_REQ) {
|
||||
priv->fw_version_minor < MINOR_VERSION_REQ) {
|
||||
dev_err(priv->dev, "ath9k_htc: Please upgrade to FW version %d.%d\n",
|
||||
MAJOR_VERSION_REQ, MINOR_VERSION_REQ);
|
||||
return -EINVAL;
|
||||
|
@ -5165,7 +5165,8 @@ static void b43_nphy_pmu_spur_avoid(struct b43_wldev *dev, bool avoid)
|
||||
#endif
|
||||
#ifdef CONFIG_B43_SSB
|
||||
case B43_BUS_SSB:
|
||||
/* FIXME */
|
||||
ssb_pmu_spuravoid_pllupdate(&dev->dev->sdev->bus->chipco,
|
||||
avoid);
|
||||
break;
|
||||
#endif
|
||||
}
|
||||
|
@ -4126,10 +4126,6 @@ static const struct ieee80211_iface_limit brcmf_iface_limits[] = {
|
||||
BIT(NL80211_IFTYPE_ADHOC) |
|
||||
BIT(NL80211_IFTYPE_AP)
|
||||
},
|
||||
{
|
||||
.max = 1,
|
||||
.types = BIT(NL80211_IFTYPE_P2P_DEVICE)
|
||||
},
|
||||
{
|
||||
.max = 1,
|
||||
.types = BIT(NL80211_IFTYPE_P2P_CLIENT) |
|
||||
@ -4187,8 +4183,7 @@ static struct wiphy *brcmf_setup_wiphy(struct device *phydev)
|
||||
BIT(NL80211_IFTYPE_ADHOC) |
|
||||
BIT(NL80211_IFTYPE_AP) |
|
||||
BIT(NL80211_IFTYPE_P2P_CLIENT) |
|
||||
BIT(NL80211_IFTYPE_P2P_GO) |
|
||||
BIT(NL80211_IFTYPE_P2P_DEVICE);
|
||||
BIT(NL80211_IFTYPE_P2P_GO);
|
||||
wiphy->iface_combinations = brcmf_iface_combos;
|
||||
wiphy->n_iface_combinations = ARRAY_SIZE(brcmf_iface_combos);
|
||||
wiphy->bands[IEEE80211_BAND_2GHZ] = &__wl_band_2ghz;
|
||||
|
@ -274,6 +274,130 @@ static void brcms_set_basic_rate(struct brcm_rateset *rs, u16 rate, bool is_br)
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* This function frees the WL per-device resources.
|
||||
*
|
||||
* This function frees resources owned by the WL device pointed to
|
||||
* by the wl parameter.
|
||||
*
|
||||
* precondition: can both be called locked and unlocked
|
||||
*
|
||||
*/
|
||||
static void brcms_free(struct brcms_info *wl)
|
||||
{
|
||||
struct brcms_timer *t, *next;
|
||||
|
||||
/* free ucode data */
|
||||
if (wl->fw.fw_cnt)
|
||||
brcms_ucode_data_free(&wl->ucode);
|
||||
if (wl->irq)
|
||||
free_irq(wl->irq, wl);
|
||||
|
||||
/* kill dpc */
|
||||
tasklet_kill(&wl->tasklet);
|
||||
|
||||
if (wl->pub) {
|
||||
brcms_debugfs_detach(wl->pub);
|
||||
brcms_c_module_unregister(wl->pub, "linux", wl);
|
||||
}
|
||||
|
||||
/* free common resources */
|
||||
if (wl->wlc) {
|
||||
brcms_c_detach(wl->wlc);
|
||||
wl->wlc = NULL;
|
||||
wl->pub = NULL;
|
||||
}
|
||||
|
||||
/* virtual interface deletion is deferred so we cannot spinwait */
|
||||
|
||||
/* wait for all pending callbacks to complete */
|
||||
while (atomic_read(&wl->callbacks) > 0)
|
||||
schedule();
|
||||
|
||||
/* free timers */
|
||||
for (t = wl->timers; t; t = next) {
|
||||
next = t->next;
|
||||
#ifdef DEBUG
|
||||
kfree(t->name);
|
||||
#endif
|
||||
kfree(t);
|
||||
}
|
||||
}
|
||||
|
||||
/*
|
||||
* called from both kernel as from this kernel module (error flow on attach)
|
||||
* precondition: perimeter lock is not acquired.
|
||||
*/
|
||||
static void brcms_remove(struct bcma_device *pdev)
|
||||
{
|
||||
struct ieee80211_hw *hw = bcma_get_drvdata(pdev);
|
||||
struct brcms_info *wl = hw->priv;
|
||||
|
||||
if (wl->wlc) {
|
||||
wiphy_rfkill_set_hw_state(wl->pub->ieee_hw->wiphy, false);
|
||||
wiphy_rfkill_stop_polling(wl->pub->ieee_hw->wiphy);
|
||||
ieee80211_unregister_hw(hw);
|
||||
}
|
||||
|
||||
brcms_free(wl);
|
||||
|
||||
bcma_set_drvdata(pdev, NULL);
|
||||
ieee80211_free_hw(hw);
|
||||
}
|
||||
|
||||
/*
|
||||
* Precondition: Since this function is called in brcms_pci_probe() context,
|
||||
* no locking is required.
|
||||
*/
|
||||
static void brcms_release_fw(struct brcms_info *wl)
|
||||
{
|
||||
int i;
|
||||
for (i = 0; i < MAX_FW_IMAGES; i++) {
|
||||
release_firmware(wl->fw.fw_bin[i]);
|
||||
release_firmware(wl->fw.fw_hdr[i]);
|
||||
}
|
||||
}
|
||||
|
||||
/*
|
||||
* Precondition: Since this function is called in brcms_pci_probe() context,
|
||||
* no locking is required.
|
||||
*/
|
||||
static int brcms_request_fw(struct brcms_info *wl, struct bcma_device *pdev)
|
||||
{
|
||||
int status;
|
||||
struct device *device = &pdev->dev;
|
||||
char fw_name[100];
|
||||
int i;
|
||||
|
||||
memset(&wl->fw, 0, sizeof(struct brcms_firmware));
|
||||
for (i = 0; i < MAX_FW_IMAGES; i++) {
|
||||
if (brcms_firmwares[i] == NULL)
|
||||
break;
|
||||
sprintf(fw_name, "%s-%d.fw", brcms_firmwares[i],
|
||||
UCODE_LOADER_API_VER);
|
||||
status = request_firmware(&wl->fw.fw_bin[i], fw_name, device);
|
||||
if (status) {
|
||||
wiphy_err(wl->wiphy, "%s: fail to load firmware %s\n",
|
||||
KBUILD_MODNAME, fw_name);
|
||||
return status;
|
||||
}
|
||||
sprintf(fw_name, "%s_hdr-%d.fw", brcms_firmwares[i],
|
||||
UCODE_LOADER_API_VER);
|
||||
status = request_firmware(&wl->fw.fw_hdr[i], fw_name, device);
|
||||
if (status) {
|
||||
wiphy_err(wl->wiphy, "%s: fail to load firmware %s\n",
|
||||
KBUILD_MODNAME, fw_name);
|
||||
return status;
|
||||
}
|
||||
wl->fw.hdr_num_entries[i] =
|
||||
wl->fw.fw_hdr[i]->size / (sizeof(struct firmware_hdr));
|
||||
}
|
||||
wl->fw.fw_cnt = i;
|
||||
status = brcms_ucode_data_init(wl, &wl->ucode);
|
||||
brcms_release_fw(wl);
|
||||
return status;
|
||||
}
|
||||
|
||||
static void brcms_ops_tx(struct ieee80211_hw *hw,
|
||||
struct ieee80211_tx_control *control,
|
||||
struct sk_buff *skb)
|
||||
@ -306,6 +430,14 @@ static int brcms_ops_start(struct ieee80211_hw *hw)
|
||||
if (!blocked)
|
||||
wiphy_rfkill_stop_polling(wl->pub->ieee_hw->wiphy);
|
||||
|
||||
if (!wl->ucode.bcm43xx_bomminor) {
|
||||
err = brcms_request_fw(wl, wl->wlc->hw->d11core);
|
||||
if (err) {
|
||||
brcms_remove(wl->wlc->hw->d11core);
|
||||
return -ENOENT;
|
||||
}
|
||||
}
|
||||
|
||||
spin_lock_bh(&wl->lock);
|
||||
/* avoid acknowledging frames before a non-monitor device is added */
|
||||
wl->mute_tx = true;
|
||||
@ -793,128 +925,6 @@ void brcms_dpc(unsigned long data)
|
||||
wake_up(&wl->tx_flush_wq);
|
||||
}
|
||||
|
||||
/*
|
||||
* Precondition: Since this function is called in brcms_pci_probe() context,
|
||||
* no locking is required.
|
||||
*/
|
||||
static int brcms_request_fw(struct brcms_info *wl, struct bcma_device *pdev)
|
||||
{
|
||||
int status;
|
||||
struct device *device = &pdev->dev;
|
||||
char fw_name[100];
|
||||
int i;
|
||||
|
||||
memset(&wl->fw, 0, sizeof(struct brcms_firmware));
|
||||
for (i = 0; i < MAX_FW_IMAGES; i++) {
|
||||
if (brcms_firmwares[i] == NULL)
|
||||
break;
|
||||
sprintf(fw_name, "%s-%d.fw", brcms_firmwares[i],
|
||||
UCODE_LOADER_API_VER);
|
||||
status = request_firmware(&wl->fw.fw_bin[i], fw_name, device);
|
||||
if (status) {
|
||||
wiphy_err(wl->wiphy, "%s: fail to load firmware %s\n",
|
||||
KBUILD_MODNAME, fw_name);
|
||||
return status;
|
||||
}
|
||||
sprintf(fw_name, "%s_hdr-%d.fw", brcms_firmwares[i],
|
||||
UCODE_LOADER_API_VER);
|
||||
status = request_firmware(&wl->fw.fw_hdr[i], fw_name, device);
|
||||
if (status) {
|
||||
wiphy_err(wl->wiphy, "%s: fail to load firmware %s\n",
|
||||
KBUILD_MODNAME, fw_name);
|
||||
return status;
|
||||
}
|
||||
wl->fw.hdr_num_entries[i] =
|
||||
wl->fw.fw_hdr[i]->size / (sizeof(struct firmware_hdr));
|
||||
}
|
||||
wl->fw.fw_cnt = i;
|
||||
return brcms_ucode_data_init(wl, &wl->ucode);
|
||||
}
|
||||
|
||||
/*
|
||||
* Precondition: Since this function is called in brcms_pci_probe() context,
|
||||
* no locking is required.
|
||||
*/
|
||||
static void brcms_release_fw(struct brcms_info *wl)
|
||||
{
|
||||
int i;
|
||||
for (i = 0; i < MAX_FW_IMAGES; i++) {
|
||||
release_firmware(wl->fw.fw_bin[i]);
|
||||
release_firmware(wl->fw.fw_hdr[i]);
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* This function frees the WL per-device resources.
|
||||
*
|
||||
* This function frees resources owned by the WL device pointed to
|
||||
* by the wl parameter.
|
||||
*
|
||||
* precondition: can both be called locked and unlocked
|
||||
*
|
||||
*/
|
||||
static void brcms_free(struct brcms_info *wl)
|
||||
{
|
||||
struct brcms_timer *t, *next;
|
||||
|
||||
/* free ucode data */
|
||||
if (wl->fw.fw_cnt)
|
||||
brcms_ucode_data_free(&wl->ucode);
|
||||
if (wl->irq)
|
||||
free_irq(wl->irq, wl);
|
||||
|
||||
/* kill dpc */
|
||||
tasklet_kill(&wl->tasklet);
|
||||
|
||||
if (wl->pub) {
|
||||
brcms_debugfs_detach(wl->pub);
|
||||
brcms_c_module_unregister(wl->pub, "linux", wl);
|
||||
}
|
||||
|
||||
/* free common resources */
|
||||
if (wl->wlc) {
|
||||
brcms_c_detach(wl->wlc);
|
||||
wl->wlc = NULL;
|
||||
wl->pub = NULL;
|
||||
}
|
||||
|
||||
/* virtual interface deletion is deferred so we cannot spinwait */
|
||||
|
||||
/* wait for all pending callbacks to complete */
|
||||
while (atomic_read(&wl->callbacks) > 0)
|
||||
schedule();
|
||||
|
||||
/* free timers */
|
||||
for (t = wl->timers; t; t = next) {
|
||||
next = t->next;
|
||||
#ifdef DEBUG
|
||||
kfree(t->name);
|
||||
#endif
|
||||
kfree(t);
|
||||
}
|
||||
}
|
||||
|
||||
/*
|
||||
* called from both kernel as from this kernel module (error flow on attach)
|
||||
* precondition: perimeter lock is not acquired.
|
||||
*/
|
||||
static void brcms_remove(struct bcma_device *pdev)
|
||||
{
|
||||
struct ieee80211_hw *hw = bcma_get_drvdata(pdev);
|
||||
struct brcms_info *wl = hw->priv;
|
||||
|
||||
if (wl->wlc) {
|
||||
wiphy_rfkill_set_hw_state(wl->pub->ieee_hw->wiphy, false);
|
||||
wiphy_rfkill_stop_polling(wl->pub->ieee_hw->wiphy);
|
||||
ieee80211_unregister_hw(hw);
|
||||
}
|
||||
|
||||
brcms_free(wl);
|
||||
|
||||
bcma_set_drvdata(pdev, NULL);
|
||||
ieee80211_free_hw(hw);
|
||||
}
|
||||
|
||||
static irqreturn_t brcms_isr(int irq, void *dev_id)
|
||||
{
|
||||
struct brcms_info *wl;
|
||||
@ -1047,18 +1057,8 @@ static struct brcms_info *brcms_attach(struct bcma_device *pdev)
|
||||
spin_lock_init(&wl->lock);
|
||||
spin_lock_init(&wl->isr_lock);
|
||||
|
||||
/* prepare ucode */
|
||||
if (brcms_request_fw(wl, pdev) < 0) {
|
||||
wiphy_err(wl->wiphy, "%s: Failed to find firmware usually in "
|
||||
"%s\n", KBUILD_MODNAME, "/lib/firmware/brcm");
|
||||
brcms_release_fw(wl);
|
||||
brcms_remove(pdev);
|
||||
return NULL;
|
||||
}
|
||||
|
||||
/* common load-time initialization */
|
||||
wl->wlc = brcms_c_attach((void *)wl, pdev, unit, false, &err);
|
||||
brcms_release_fw(wl);
|
||||
if (!wl->wlc) {
|
||||
wiphy_err(wl->wiphy, "%s: attach() failed with code %d\n",
|
||||
KBUILD_MODNAME, err);
|
||||
|
@ -675,3 +675,32 @@ u32 ssb_pmu_get_controlclock(struct ssb_chipcommon *cc)
|
||||
return 0;
|
||||
}
|
||||
}
|
||||
|
||||
void ssb_pmu_spuravoid_pllupdate(struct ssb_chipcommon *cc, int spuravoid)
|
||||
{
|
||||
u32 pmu_ctl = 0;
|
||||
|
||||
switch (cc->dev->bus->chip_id) {
|
||||
case 0x4322:
|
||||
ssb_chipco_pll_write(cc, SSB_PMU1_PLLCTL0, 0x11100070);
|
||||
ssb_chipco_pll_write(cc, SSB_PMU1_PLLCTL1, 0x1014140a);
|
||||
ssb_chipco_pll_write(cc, SSB_PMU1_PLLCTL5, 0x88888854);
|
||||
if (spuravoid == 1)
|
||||
ssb_chipco_pll_write(cc, SSB_PMU1_PLLCTL2, 0x05201828);
|
||||
else
|
||||
ssb_chipco_pll_write(cc, SSB_PMU1_PLLCTL2, 0x05001828);
|
||||
pmu_ctl = SSB_CHIPCO_PMU_CTL_PLL_UPD;
|
||||
break;
|
||||
case 43222:
|
||||
/* TODO: BCM43222 requires updating PLLs too */
|
||||
return;
|
||||
default:
|
||||
ssb_printk(KERN_ERR PFX
|
||||
"Unknown spuravoidance settings for chip 0x%04X, not changing PLL\n",
|
||||
cc->dev->bus->chip_id);
|
||||
return;
|
||||
}
|
||||
|
||||
chipco_set32(cc, SSB_CHIPCO_PMU_CTL, pmu_ctl);
|
||||
}
|
||||
EXPORT_SYMBOL_GPL(ssb_pmu_spuravoid_pllupdate);
|
||||
|
@ -219,6 +219,7 @@
|
||||
#define SSB_CHIPCO_PMU_CTL 0x0600 /* PMU control */
|
||||
#define SSB_CHIPCO_PMU_CTL_ILP_DIV 0xFFFF0000 /* ILP div mask */
|
||||
#define SSB_CHIPCO_PMU_CTL_ILP_DIV_SHIFT 16
|
||||
#define SSB_CHIPCO_PMU_CTL_PLL_UPD 0x00000400
|
||||
#define SSB_CHIPCO_PMU_CTL_NOILPONW 0x00000200 /* No ILP on wait */
|
||||
#define SSB_CHIPCO_PMU_CTL_HTREQEN 0x00000100 /* HT req enable */
|
||||
#define SSB_CHIPCO_PMU_CTL_ALPREQEN 0x00000080 /* ALP req enable */
|
||||
@ -667,5 +668,6 @@ enum ssb_pmu_ldo_volt_id {
|
||||
void ssb_pmu_set_ldo_voltage(struct ssb_chipcommon *cc,
|
||||
enum ssb_pmu_ldo_volt_id id, u32 voltage);
|
||||
void ssb_pmu_set_ldo_paref(struct ssb_chipcommon *cc, bool on);
|
||||
void ssb_pmu_spuravoid_pllupdate(struct ssb_chipcommon *cc, int spuravoid);
|
||||
|
||||
#endif /* LINUX_SSB_CHIPCO_H_ */
|
||||
|
@ -256,7 +256,8 @@ static inline __u32 irlmp_get_daddr(const struct lsap_cb *self)
|
||||
return (self && self->lap) ? self->lap->daddr : 0;
|
||||
}
|
||||
|
||||
extern const char *irlmp_reasons[];
|
||||
const char *irlmp_reason_str(LM_REASON reason);
|
||||
|
||||
extern int sysctl_discovery_timeout;
|
||||
extern int sysctl_discovery_slots;
|
||||
extern int sysctl_discovery;
|
||||
|
@ -56,8 +56,8 @@ static __inline__ void scm_set_cred(struct scm_cookie *scm,
|
||||
scm->pid = get_pid(pid);
|
||||
scm->cred = cred ? get_cred(cred) : NULL;
|
||||
scm->creds.pid = pid_vnr(pid);
|
||||
scm->creds.uid = cred ? cred->euid : INVALID_UID;
|
||||
scm->creds.gid = cred ? cred->egid : INVALID_GID;
|
||||
scm->creds.uid = cred ? cred->uid : INVALID_UID;
|
||||
scm->creds.gid = cred ? cred->gid : INVALID_GID;
|
||||
}
|
||||
|
||||
static __inline__ void scm_destroy_cred(struct scm_cookie *scm)
|
||||
|
@ -169,7 +169,7 @@ void batadv_mesh_free(struct net_device *soft_iface)
|
||||
atomic_set(&bat_priv->mesh_state, BATADV_MESH_INACTIVE);
|
||||
}
|
||||
|
||||
int batadv_is_my_mac(const uint8_t *addr)
|
||||
int batadv_is_my_mac(struct batadv_priv *bat_priv, const uint8_t *addr)
|
||||
{
|
||||
const struct batadv_hard_iface *hard_iface;
|
||||
|
||||
@ -178,6 +178,9 @@ int batadv_is_my_mac(const uint8_t *addr)
|
||||
if (hard_iface->if_status != BATADV_IF_ACTIVE)
|
||||
continue;
|
||||
|
||||
if (hard_iface->soft_iface != bat_priv->soft_iface)
|
||||
continue;
|
||||
|
||||
if (batadv_compare_eth(hard_iface->net_dev->dev_addr, addr)) {
|
||||
rcu_read_unlock();
|
||||
return 1;
|
||||
|
@ -162,7 +162,7 @@ extern struct workqueue_struct *batadv_event_workqueue;
|
||||
|
||||
int batadv_mesh_init(struct net_device *soft_iface);
|
||||
void batadv_mesh_free(struct net_device *soft_iface);
|
||||
int batadv_is_my_mac(const uint8_t *addr);
|
||||
int batadv_is_my_mac(struct batadv_priv *bat_priv, const uint8_t *addr);
|
||||
struct batadv_hard_iface *
|
||||
batadv_seq_print_text_primary_if_get(struct seq_file *seq);
|
||||
int batadv_batman_skb_recv(struct sk_buff *skb, struct net_device *dev,
|
||||
|
@ -402,7 +402,7 @@ int batadv_recv_icmp_packet(struct sk_buff *skb,
|
||||
goto out;
|
||||
|
||||
/* not for me */
|
||||
if (!batadv_is_my_mac(ethhdr->h_dest))
|
||||
if (!batadv_is_my_mac(bat_priv, ethhdr->h_dest))
|
||||
goto out;
|
||||
|
||||
icmp_packet = (struct batadv_icmp_packet_rr *)skb->data;
|
||||
@ -416,7 +416,7 @@ int batadv_recv_icmp_packet(struct sk_buff *skb,
|
||||
}
|
||||
|
||||
/* packet for me */
|
||||
if (batadv_is_my_mac(icmp_packet->dst))
|
||||
if (batadv_is_my_mac(bat_priv, icmp_packet->dst))
|
||||
return batadv_recv_my_icmp_packet(bat_priv, skb, hdr_size);
|
||||
|
||||
/* TTL exceeded */
|
||||
@ -548,7 +548,8 @@ batadv_find_ifalter_router(struct batadv_orig_node *primary_orig,
|
||||
return router;
|
||||
}
|
||||
|
||||
static int batadv_check_unicast_packet(struct sk_buff *skb, int hdr_size)
|
||||
static int batadv_check_unicast_packet(struct batadv_priv *bat_priv,
|
||||
struct sk_buff *skb, int hdr_size)
|
||||
{
|
||||
struct ethhdr *ethhdr;
|
||||
|
||||
@ -567,7 +568,7 @@ static int batadv_check_unicast_packet(struct sk_buff *skb, int hdr_size)
|
||||
return -1;
|
||||
|
||||
/* not for me */
|
||||
if (!batadv_is_my_mac(ethhdr->h_dest))
|
||||
if (!batadv_is_my_mac(bat_priv, ethhdr->h_dest))
|
||||
return -1;
|
||||
|
||||
return 0;
|
||||
@ -582,7 +583,7 @@ int batadv_recv_tt_query(struct sk_buff *skb, struct batadv_hard_iface *recv_if)
|
||||
char tt_flag;
|
||||
size_t packet_size;
|
||||
|
||||
if (batadv_check_unicast_packet(skb, hdr_size) < 0)
|
||||
if (batadv_check_unicast_packet(bat_priv, skb, hdr_size) < 0)
|
||||
return NET_RX_DROP;
|
||||
|
||||
/* I could need to modify it */
|
||||
@ -614,7 +615,7 @@ int batadv_recv_tt_query(struct sk_buff *skb, struct batadv_hard_iface *recv_if)
|
||||
case BATADV_TT_RESPONSE:
|
||||
batadv_inc_counter(bat_priv, BATADV_CNT_TT_RESPONSE_RX);
|
||||
|
||||
if (batadv_is_my_mac(tt_query->dst)) {
|
||||
if (batadv_is_my_mac(bat_priv, tt_query->dst)) {
|
||||
/* packet needs to be linearized to access the TT
|
||||
* changes
|
||||
*/
|
||||
@ -657,14 +658,15 @@ int batadv_recv_roam_adv(struct sk_buff *skb, struct batadv_hard_iface *recv_if)
|
||||
struct batadv_roam_adv_packet *roam_adv_packet;
|
||||
struct batadv_orig_node *orig_node;
|
||||
|
||||
if (batadv_check_unicast_packet(skb, sizeof(*roam_adv_packet)) < 0)
|
||||
if (batadv_check_unicast_packet(bat_priv, skb,
|
||||
sizeof(*roam_adv_packet)) < 0)
|
||||
goto out;
|
||||
|
||||
batadv_inc_counter(bat_priv, BATADV_CNT_TT_ROAM_ADV_RX);
|
||||
|
||||
roam_adv_packet = (struct batadv_roam_adv_packet *)skb->data;
|
||||
|
||||
if (!batadv_is_my_mac(roam_adv_packet->dst))
|
||||
if (!batadv_is_my_mac(bat_priv, roam_adv_packet->dst))
|
||||
return batadv_route_unicast_packet(skb, recv_if);
|
||||
|
||||
/* check if it is a backbone gateway. we don't accept
|
||||
@ -967,7 +969,7 @@ static int batadv_check_unicast_ttvn(struct batadv_priv *bat_priv,
|
||||
* last time) the packet had an updated information or not
|
||||
*/
|
||||
curr_ttvn = (uint8_t)atomic_read(&bat_priv->tt.vn);
|
||||
if (!batadv_is_my_mac(unicast_packet->dest)) {
|
||||
if (!batadv_is_my_mac(bat_priv, unicast_packet->dest)) {
|
||||
orig_node = batadv_orig_hash_find(bat_priv,
|
||||
unicast_packet->dest);
|
||||
/* if it is not possible to find the orig_node representing the
|
||||
@ -1044,14 +1046,14 @@ int batadv_recv_unicast_packet(struct sk_buff *skb,
|
||||
if (is4addr)
|
||||
hdr_size = sizeof(*unicast_4addr_packet);
|
||||
|
||||
if (batadv_check_unicast_packet(skb, hdr_size) < 0)
|
||||
if (batadv_check_unicast_packet(bat_priv, skb, hdr_size) < 0)
|
||||
return NET_RX_DROP;
|
||||
|
||||
if (!batadv_check_unicast_ttvn(bat_priv, skb))
|
||||
return NET_RX_DROP;
|
||||
|
||||
/* packet for me */
|
||||
if (batadv_is_my_mac(unicast_packet->dest)) {
|
||||
if (batadv_is_my_mac(bat_priv, unicast_packet->dest)) {
|
||||
if (is4addr) {
|
||||
batadv_dat_inc_counter(bat_priv,
|
||||
unicast_4addr_packet->subtype);
|
||||
@ -1088,7 +1090,7 @@ int batadv_recv_ucast_frag_packet(struct sk_buff *skb,
|
||||
struct sk_buff *new_skb = NULL;
|
||||
int ret;
|
||||
|
||||
if (batadv_check_unicast_packet(skb, hdr_size) < 0)
|
||||
if (batadv_check_unicast_packet(bat_priv, skb, hdr_size) < 0)
|
||||
return NET_RX_DROP;
|
||||
|
||||
if (!batadv_check_unicast_ttvn(bat_priv, skb))
|
||||
@ -1097,7 +1099,7 @@ int batadv_recv_ucast_frag_packet(struct sk_buff *skb,
|
||||
unicast_packet = (struct batadv_unicast_frag_packet *)skb->data;
|
||||
|
||||
/* packet for me */
|
||||
if (batadv_is_my_mac(unicast_packet->dest)) {
|
||||
if (batadv_is_my_mac(bat_priv, unicast_packet->dest)) {
|
||||
ret = batadv_frag_reassemble_skb(skb, bat_priv, &new_skb);
|
||||
|
||||
if (ret == NET_RX_DROP)
|
||||
@ -1151,13 +1153,13 @@ int batadv_recv_bcast_packet(struct sk_buff *skb,
|
||||
goto out;
|
||||
|
||||
/* ignore broadcasts sent by myself */
|
||||
if (batadv_is_my_mac(ethhdr->h_source))
|
||||
if (batadv_is_my_mac(bat_priv, ethhdr->h_source))
|
||||
goto out;
|
||||
|
||||
bcast_packet = (struct batadv_bcast_packet *)skb->data;
|
||||
|
||||
/* ignore broadcasts originated by myself */
|
||||
if (batadv_is_my_mac(bcast_packet->orig))
|
||||
if (batadv_is_my_mac(bat_priv, bcast_packet->orig))
|
||||
goto out;
|
||||
|
||||
if (bcast_packet->header.ttl < 2)
|
||||
@ -1243,14 +1245,14 @@ int batadv_recv_vis_packet(struct sk_buff *skb,
|
||||
ethhdr = (struct ethhdr *)skb_mac_header(skb);
|
||||
|
||||
/* not for me */
|
||||
if (!batadv_is_my_mac(ethhdr->h_dest))
|
||||
if (!batadv_is_my_mac(bat_priv, ethhdr->h_dest))
|
||||
return NET_RX_DROP;
|
||||
|
||||
/* ignore own packets */
|
||||
if (batadv_is_my_mac(vis_packet->vis_orig))
|
||||
if (batadv_is_my_mac(bat_priv, vis_packet->vis_orig))
|
||||
return NET_RX_DROP;
|
||||
|
||||
if (batadv_is_my_mac(vis_packet->sender_orig))
|
||||
if (batadv_is_my_mac(bat_priv, vis_packet->sender_orig))
|
||||
return NET_RX_DROP;
|
||||
|
||||
switch (vis_packet->vis_type) {
|
||||
|
@ -1953,7 +1953,7 @@ out:
|
||||
bool batadv_send_tt_response(struct batadv_priv *bat_priv,
|
||||
struct batadv_tt_query_packet *tt_request)
|
||||
{
|
||||
if (batadv_is_my_mac(tt_request->dst)) {
|
||||
if (batadv_is_my_mac(bat_priv, tt_request->dst)) {
|
||||
/* don't answer backbone gws! */
|
||||
if (batadv_bla_is_backbone_gw_orig(bat_priv, tt_request->src))
|
||||
return true;
|
||||
|
@ -477,7 +477,7 @@ void batadv_receive_client_update_packet(struct batadv_priv *bat_priv,
|
||||
|
||||
/* Are we the target for this VIS packet? */
|
||||
if (vis_server == BATADV_VIS_TYPE_SERVER_SYNC &&
|
||||
batadv_is_my_mac(vis_packet->target_orig))
|
||||
batadv_is_my_mac(bat_priv, vis_packet->target_orig))
|
||||
are_target = 1;
|
||||
|
||||
spin_lock_bh(&bat_priv->vis.hash_lock);
|
||||
@ -496,7 +496,7 @@ void batadv_receive_client_update_packet(struct batadv_priv *bat_priv,
|
||||
batadv_send_list_add(bat_priv, info);
|
||||
|
||||
/* ... we're not the recipient (and thus need to forward). */
|
||||
} else if (!batadv_is_my_mac(packet->target_orig)) {
|
||||
} else if (!batadv_is_my_mac(bat_priv, packet->target_orig)) {
|
||||
batadv_send_list_add(bat_priv, info);
|
||||
}
|
||||
|
||||
|
@ -2148,6 +2148,9 @@ static void skb_warn_bad_offload(const struct sk_buff *skb)
|
||||
struct net_device *dev = skb->dev;
|
||||
const char *driver = "";
|
||||
|
||||
if (!net_ratelimit())
|
||||
return;
|
||||
|
||||
if (dev && dev->dev.parent)
|
||||
driver = dev_driver_string(dev->dev.parent);
|
||||
|
||||
|
@ -66,6 +66,12 @@ static bool rpfilter_lookup_reverse(struct flowi4 *fl4,
|
||||
return dev_match;
|
||||
}
|
||||
|
||||
static bool rpfilter_is_local(const struct sk_buff *skb)
|
||||
{
|
||||
const struct rtable *rt = skb_rtable(skb);
|
||||
return rt && (rt->rt_flags & RTCF_LOCAL);
|
||||
}
|
||||
|
||||
static bool rpfilter_mt(const struct sk_buff *skb, struct xt_action_param *par)
|
||||
{
|
||||
const struct xt_rpfilter_info *info;
|
||||
@ -76,7 +82,7 @@ static bool rpfilter_mt(const struct sk_buff *skb, struct xt_action_param *par)
|
||||
info = par->matchinfo;
|
||||
invert = info->flags & XT_RPFILTER_INVERT;
|
||||
|
||||
if (par->in->flags & IFF_LOOPBACK)
|
||||
if (rpfilter_is_local(skb))
|
||||
return true ^ invert;
|
||||
|
||||
iph = ip_hdr(skb);
|
||||
|
@ -113,6 +113,7 @@ int sysctl_tcp_early_retrans __read_mostly = 2;
|
||||
#define FLAG_DSACKING_ACK 0x800 /* SACK blocks contained D-SACK info */
|
||||
#define FLAG_NONHEAD_RETRANS_ACKED 0x1000 /* Non-head rexmitted data was ACKed */
|
||||
#define FLAG_SACK_RENEGING 0x2000 /* snd_una advanced to a sacked seq */
|
||||
#define FLAG_UPDATE_TS_RECENT 0x4000 /* tcp_replace_ts_recent() */
|
||||
|
||||
#define FLAG_ACKED (FLAG_DATA_ACKED|FLAG_SYN_ACKED)
|
||||
#define FLAG_NOT_DUP (FLAG_DATA|FLAG_WIN_UPDATE|FLAG_ACKED)
|
||||
@ -3564,6 +3565,27 @@ static void tcp_send_challenge_ack(struct sock *sk)
|
||||
}
|
||||
}
|
||||
|
||||
static void tcp_store_ts_recent(struct tcp_sock *tp)
|
||||
{
|
||||
tp->rx_opt.ts_recent = tp->rx_opt.rcv_tsval;
|
||||
tp->rx_opt.ts_recent_stamp = get_seconds();
|
||||
}
|
||||
|
||||
static void tcp_replace_ts_recent(struct tcp_sock *tp, u32 seq)
|
||||
{
|
||||
if (tp->rx_opt.saw_tstamp && !after(seq, tp->rcv_wup)) {
|
||||
/* PAWS bug workaround wrt. ACK frames, the PAWS discard
|
||||
* extra check below makes sure this can only happen
|
||||
* for pure ACK frames. -DaveM
|
||||
*
|
||||
* Not only, also it occurs for expired timestamps.
|
||||
*/
|
||||
|
||||
if (tcp_paws_check(&tp->rx_opt, 0))
|
||||
tcp_store_ts_recent(tp);
|
||||
}
|
||||
}
|
||||
|
||||
/* This routine deals with incoming acks, but not outgoing ones. */
|
||||
static int tcp_ack(struct sock *sk, const struct sk_buff *skb, int flag)
|
||||
{
|
||||
@ -3607,6 +3629,12 @@ static int tcp_ack(struct sock *sk, const struct sk_buff *skb, int flag)
|
||||
prior_fackets = tp->fackets_out;
|
||||
prior_in_flight = tcp_packets_in_flight(tp);
|
||||
|
||||
/* ts_recent update must be made after we are sure that the packet
|
||||
* is in window.
|
||||
*/
|
||||
if (flag & FLAG_UPDATE_TS_RECENT)
|
||||
tcp_replace_ts_recent(tp, TCP_SKB_CB(skb)->seq);
|
||||
|
||||
if (!(flag & FLAG_SLOWPATH) && after(ack, prior_snd_una)) {
|
||||
/* Window is constant, pure forward advance.
|
||||
* No more checks are required.
|
||||
@ -3927,27 +3955,6 @@ const u8 *tcp_parse_md5sig_option(const struct tcphdr *th)
|
||||
EXPORT_SYMBOL(tcp_parse_md5sig_option);
|
||||
#endif
|
||||
|
||||
static inline void tcp_store_ts_recent(struct tcp_sock *tp)
|
||||
{
|
||||
tp->rx_opt.ts_recent = tp->rx_opt.rcv_tsval;
|
||||
tp->rx_opt.ts_recent_stamp = get_seconds();
|
||||
}
|
||||
|
||||
static inline void tcp_replace_ts_recent(struct tcp_sock *tp, u32 seq)
|
||||
{
|
||||
if (tp->rx_opt.saw_tstamp && !after(seq, tp->rcv_wup)) {
|
||||
/* PAWS bug workaround wrt. ACK frames, the PAWS discard
|
||||
* extra check below makes sure this can only happen
|
||||
* for pure ACK frames. -DaveM
|
||||
*
|
||||
* Not only, also it occurs for expired timestamps.
|
||||
*/
|
||||
|
||||
if (tcp_paws_check(&tp->rx_opt, 0))
|
||||
tcp_store_ts_recent(tp);
|
||||
}
|
||||
}
|
||||
|
||||
/* Sorry, PAWS as specified is broken wrt. pure-ACKs -DaveM
|
||||
*
|
||||
* It is not fatal. If this ACK does _not_ change critical state (seqs, window)
|
||||
@ -5543,14 +5550,9 @@ slow_path:
|
||||
return 0;
|
||||
|
||||
step5:
|
||||
if (tcp_ack(sk, skb, FLAG_SLOWPATH) < 0)
|
||||
if (tcp_ack(sk, skb, FLAG_SLOWPATH | FLAG_UPDATE_TS_RECENT) < 0)
|
||||
goto discard;
|
||||
|
||||
/* ts_recent update must be made after we are sure that the packet
|
||||
* is in window.
|
||||
*/
|
||||
tcp_replace_ts_recent(tp, TCP_SKB_CB(skb)->seq);
|
||||
|
||||
tcp_rcv_rtt_measure_ts(sk, skb);
|
||||
|
||||
/* Process urgent data. */
|
||||
@ -5986,7 +5988,8 @@ int tcp_rcv_state_process(struct sock *sk, struct sk_buff *skb,
|
||||
|
||||
/* step 5: check the ACK field */
|
||||
if (true) {
|
||||
int acceptable = tcp_ack(sk, skb, FLAG_SLOWPATH) > 0;
|
||||
int acceptable = tcp_ack(sk, skb, FLAG_SLOWPATH |
|
||||
FLAG_UPDATE_TS_RECENT) > 0;
|
||||
|
||||
switch (sk->sk_state) {
|
||||
case TCP_SYN_RECV:
|
||||
@ -6137,11 +6140,6 @@ int tcp_rcv_state_process(struct sock *sk, struct sk_buff *skb,
|
||||
}
|
||||
}
|
||||
|
||||
/* ts_recent update must be made after we are sure that the packet
|
||||
* is in window.
|
||||
*/
|
||||
tcp_replace_ts_recent(tp, TCP_SKB_CB(skb)->seq);
|
||||
|
||||
/* step 6: check the URG bit */
|
||||
tcp_urg(sk, skb, th);
|
||||
|
||||
|
@ -71,6 +71,12 @@ static bool rpfilter_lookup_reverse6(const struct sk_buff *skb,
|
||||
return ret;
|
||||
}
|
||||
|
||||
static bool rpfilter_is_local(const struct sk_buff *skb)
|
||||
{
|
||||
const struct rt6_info *rt = (const void *) skb_dst(skb);
|
||||
return rt && (rt->rt6i_flags & RTF_LOCAL);
|
||||
}
|
||||
|
||||
static bool rpfilter_mt(const struct sk_buff *skb, struct xt_action_param *par)
|
||||
{
|
||||
const struct xt_rpfilter_info *info = par->matchinfo;
|
||||
@ -78,7 +84,7 @@ static bool rpfilter_mt(const struct sk_buff *skb, struct xt_action_param *par)
|
||||
struct ipv6hdr *iph;
|
||||
bool invert = info->flags & XT_RPFILTER_INVERT;
|
||||
|
||||
if (par->in->flags & IFF_LOOPBACK)
|
||||
if (rpfilter_is_local(skb))
|
||||
return true ^ invert;
|
||||
|
||||
iph = ipv6_hdr(skb);
|
||||
|
@ -303,7 +303,8 @@ static void iriap_disconnect_indication(void *instance, void *sap,
|
||||
{
|
||||
struct iriap_cb *self;
|
||||
|
||||
IRDA_DEBUG(4, "%s(), reason=%s\n", __func__, irlmp_reasons[reason]);
|
||||
IRDA_DEBUG(4, "%s(), reason=%s [%d]\n", __func__,
|
||||
irlmp_reason_str(reason), reason);
|
||||
|
||||
self = instance;
|
||||
|
||||
|
@ -66,8 +66,15 @@ const char *irlmp_reasons[] = {
|
||||
"LM_LAP_RESET",
|
||||
"LM_INIT_DISCONNECT",
|
||||
"ERROR, NOT USED",
|
||||
"UNKNOWN",
|
||||
};
|
||||
|
||||
const char *irlmp_reason_str(LM_REASON reason)
|
||||
{
|
||||
reason = min_t(size_t, reason, ARRAY_SIZE(irlmp_reasons) - 1);
|
||||
return irlmp_reasons[reason];
|
||||
}
|
||||
|
||||
/*
|
||||
* Function irlmp_init (void)
|
||||
*
|
||||
@ -747,7 +754,8 @@ void irlmp_disconnect_indication(struct lsap_cb *self, LM_REASON reason,
|
||||
{
|
||||
struct lsap_cb *lsap;
|
||||
|
||||
IRDA_DEBUG(1, "%s(), reason=%s\n", __func__, irlmp_reasons[reason]);
|
||||
IRDA_DEBUG(1, "%s(), reason=%s [%d]\n", __func__,
|
||||
irlmp_reason_str(reason), reason);
|
||||
IRDA_ASSERT(self != NULL, return;);
|
||||
IRDA_ASSERT(self->magic == LMP_LSAP_MAGIC, return;);
|
||||
|
||||
|
@ -78,7 +78,7 @@ void ieee80211_recalc_txpower(struct ieee80211_sub_if_data *sdata)
|
||||
ieee80211_bss_info_change_notify(sdata, BSS_CHANGED_TXPOWER);
|
||||
}
|
||||
|
||||
u32 ieee80211_idle_off(struct ieee80211_local *local)
|
||||
static u32 __ieee80211_idle_off(struct ieee80211_local *local)
|
||||
{
|
||||
if (!(local->hw.conf.flags & IEEE80211_CONF_IDLE))
|
||||
return 0;
|
||||
@ -87,7 +87,7 @@ u32 ieee80211_idle_off(struct ieee80211_local *local)
|
||||
return IEEE80211_CONF_CHANGE_IDLE;
|
||||
}
|
||||
|
||||
static u32 ieee80211_idle_on(struct ieee80211_local *local)
|
||||
static u32 __ieee80211_idle_on(struct ieee80211_local *local)
|
||||
{
|
||||
if (local->hw.conf.flags & IEEE80211_CONF_IDLE)
|
||||
return 0;
|
||||
@ -98,16 +98,18 @@ static u32 ieee80211_idle_on(struct ieee80211_local *local)
|
||||
return IEEE80211_CONF_CHANGE_IDLE;
|
||||
}
|
||||
|
||||
void ieee80211_recalc_idle(struct ieee80211_local *local)
|
||||
static u32 __ieee80211_recalc_idle(struct ieee80211_local *local,
|
||||
bool force_active)
|
||||
{
|
||||
bool working = false, scanning, active;
|
||||
unsigned int led_trig_start = 0, led_trig_stop = 0;
|
||||
struct ieee80211_roc_work *roc;
|
||||
u32 change;
|
||||
|
||||
lockdep_assert_held(&local->mtx);
|
||||
|
||||
active = !list_empty(&local->chanctx_list) || local->monitors;
|
||||
active = force_active ||
|
||||
!list_empty(&local->chanctx_list) ||
|
||||
local->monitors;
|
||||
|
||||
if (!local->ops->remain_on_channel) {
|
||||
list_for_each_entry(roc, &local->roc_list, list) {
|
||||
@ -132,9 +134,18 @@ void ieee80211_recalc_idle(struct ieee80211_local *local)
|
||||
ieee80211_mod_tpt_led_trig(local, led_trig_start, led_trig_stop);
|
||||
|
||||
if (working || scanning || active)
|
||||
change = ieee80211_idle_off(local);
|
||||
else
|
||||
change = ieee80211_idle_on(local);
|
||||
return __ieee80211_idle_off(local);
|
||||
return __ieee80211_idle_on(local);
|
||||
}
|
||||
|
||||
u32 ieee80211_idle_off(struct ieee80211_local *local)
|
||||
{
|
||||
return __ieee80211_recalc_idle(local, true);
|
||||
}
|
||||
|
||||
void ieee80211_recalc_idle(struct ieee80211_local *local)
|
||||
{
|
||||
u32 change = __ieee80211_recalc_idle(local, false);
|
||||
if (change)
|
||||
ieee80211_hw_config(local, change);
|
||||
}
|
||||
|
@ -3964,8 +3964,16 @@ int ieee80211_mgd_auth(struct ieee80211_sub_if_data *sdata,
|
||||
/* prep auth_data so we don't go into idle on disassoc */
|
||||
ifmgd->auth_data = auth_data;
|
||||
|
||||
if (ifmgd->associated)
|
||||
ieee80211_set_disassoc(sdata, 0, 0, false, NULL);
|
||||
if (ifmgd->associated) {
|
||||
u8 frame_buf[IEEE80211_DEAUTH_FRAME_LEN];
|
||||
|
||||
ieee80211_set_disassoc(sdata, IEEE80211_STYPE_DEAUTH,
|
||||
WLAN_REASON_UNSPECIFIED,
|
||||
false, frame_buf);
|
||||
|
||||
__cfg80211_send_deauth(sdata->dev, frame_buf,
|
||||
sizeof(frame_buf));
|
||||
}
|
||||
|
||||
sdata_info(sdata, "authenticate with %pM\n", req->bss->bssid);
|
||||
|
||||
@ -4025,8 +4033,16 @@ int ieee80211_mgd_assoc(struct ieee80211_sub_if_data *sdata,
|
||||
|
||||
mutex_lock(&ifmgd->mtx);
|
||||
|
||||
if (ifmgd->associated)
|
||||
ieee80211_set_disassoc(sdata, 0, 0, false, NULL);
|
||||
if (ifmgd->associated) {
|
||||
u8 frame_buf[IEEE80211_DEAUTH_FRAME_LEN];
|
||||
|
||||
ieee80211_set_disassoc(sdata, IEEE80211_STYPE_DEAUTH,
|
||||
WLAN_REASON_UNSPECIFIED,
|
||||
false, frame_buf);
|
||||
|
||||
__cfg80211_send_deauth(sdata->dev, frame_buf,
|
||||
sizeof(frame_buf));
|
||||
}
|
||||
|
||||
if (ifmgd->auth_data && !ifmgd->auth_data->done) {
|
||||
err = -EBUSY;
|
||||
|
@ -339,7 +339,11 @@ bitmap_ipmac_tlist(const struct ip_set *set,
|
||||
nla_put_failure:
|
||||
nla_nest_cancel(skb, nested);
|
||||
ipset_nest_end(skb, atd);
|
||||
return -EMSGSIZE;
|
||||
if (unlikely(id == first)) {
|
||||
cb->args[2] = 0;
|
||||
return -EMSGSIZE;
|
||||
}
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int
|
||||
|
@ -1593,10 +1593,8 @@ struct sk_buff *ovs_vport_cmd_build_info(struct vport *vport, u32 portid,
|
||||
return ERR_PTR(-ENOMEM);
|
||||
|
||||
retval = ovs_vport_cmd_fill_info(vport, skb, portid, seq, 0, cmd);
|
||||
if (retval < 0) {
|
||||
kfree_skb(skb);
|
||||
return ERR_PTR(retval);
|
||||
}
|
||||
BUG_ON(retval < 0);
|
||||
|
||||
return skb;
|
||||
}
|
||||
|
||||
@ -1726,24 +1724,32 @@ static int ovs_vport_cmd_set(struct sk_buff *skb, struct genl_info *info)
|
||||
nla_get_u32(a[OVS_VPORT_ATTR_TYPE]) != vport->ops->type)
|
||||
err = -EINVAL;
|
||||
|
||||
reply = nlmsg_new(NLMSG_DEFAULT_SIZE, GFP_KERNEL);
|
||||
if (!reply) {
|
||||
err = -ENOMEM;
|
||||
goto exit_unlock;
|
||||
}
|
||||
|
||||
if (!err && a[OVS_VPORT_ATTR_OPTIONS])
|
||||
err = ovs_vport_set_options(vport, a[OVS_VPORT_ATTR_OPTIONS]);
|
||||
if (err)
|
||||
goto exit_unlock;
|
||||
goto exit_free;
|
||||
|
||||
if (a[OVS_VPORT_ATTR_UPCALL_PID])
|
||||
vport->upcall_portid = nla_get_u32(a[OVS_VPORT_ATTR_UPCALL_PID]);
|
||||
|
||||
reply = ovs_vport_cmd_build_info(vport, info->snd_portid, info->snd_seq,
|
||||
OVS_VPORT_CMD_NEW);
|
||||
if (IS_ERR(reply)) {
|
||||
netlink_set_err(sock_net(skb->sk)->genl_sock, 0,
|
||||
ovs_dp_vport_multicast_group.id, PTR_ERR(reply));
|
||||
goto exit_unlock;
|
||||
}
|
||||
err = ovs_vport_cmd_fill_info(vport, reply, info->snd_portid,
|
||||
info->snd_seq, 0, OVS_VPORT_CMD_NEW);
|
||||
BUG_ON(err < 0);
|
||||
|
||||
genl_notify(reply, genl_info_net(info), info->snd_portid,
|
||||
ovs_dp_vport_multicast_group.id, info->nlhdr, GFP_KERNEL);
|
||||
|
||||
rtnl_unlock();
|
||||
return 0;
|
||||
|
||||
exit_free:
|
||||
kfree_skb(reply);
|
||||
exit_unlock:
|
||||
rtnl_unlock();
|
||||
return err;
|
||||
|
@ -795,9 +795,9 @@ void ovs_flow_tbl_insert(struct flow_table *table, struct sw_flow *flow)
|
||||
|
||||
void ovs_flow_tbl_remove(struct flow_table *table, struct sw_flow *flow)
|
||||
{
|
||||
BUG_ON(table->count == 0);
|
||||
hlist_del_rcu(&flow->hash_node[table->node_ver]);
|
||||
table->count--;
|
||||
BUG_ON(table->count < 0);
|
||||
}
|
||||
|
||||
/* The size of the argument for each %OVS_KEY_ATTR_* Netlink attribute. */
|
||||
|
@ -204,7 +204,6 @@ fw_change_attrs(struct net *net, struct tcf_proto *tp, struct fw_filter *f,
|
||||
if (err < 0)
|
||||
return err;
|
||||
|
||||
err = -EINVAL;
|
||||
if (tb[TCA_FW_CLASSID]) {
|
||||
f->res.classid = nla_get_u32(tb[TCA_FW_CLASSID]);
|
||||
tcf_bind_filter(tp, &f->res, base);
|
||||
@ -218,6 +217,7 @@ fw_change_attrs(struct net *net, struct tcf_proto *tp, struct fw_filter *f,
|
||||
}
|
||||
#endif /* CONFIG_NET_CLS_IND */
|
||||
|
||||
err = -EINVAL;
|
||||
if (tb[TCA_FW_MASK]) {
|
||||
mask = nla_get_u32(tb[TCA_FW_MASK]);
|
||||
if (mask != head->mask)
|
||||
|
Loading…
Reference in New Issue
Block a user