Merge git://git.kernel.org/pub/scm/linux/kernel/git/netdev/net
Pull networking fixes from David Miller: 1) BPF sample build fixes from Björn Töpel 2) Fix powerpc bpf tail call implementation, from Eric Dumazet. 3) DCCP leaks jiffies on the wire, fix also from Eric Dumazet. 4) Fix crash in ebtables when using dnat target, from Florian Westphal. 5) Fix port disable handling whne removing bcm_sf2 driver, from Florian Fainelli. 6) Fix kTLS sk_msg trim on fallback to copy mode, from Jakub Kicinski. 7) Various KCSAN fixes all over the networking, from Eric Dumazet. 8) Memory leaks in mlx5 driver, from Alex Vesker. 9) SMC interface refcounting fix, from Ursula Braun. 10) TSO descriptor handling fixes in stmmac driver, from Jose Abreu. 11) Add a TX lock to synchonize the kTLS TX path properly with crypto operations. From Jakub Kicinski. 12) Sock refcount during shutdown fix in vsock/virtio code, from Stefano Garzarella. 13) Infinite loop in Intel ice driver, from Colin Ian King. * git://git.kernel.org/pub/scm/linux/kernel/git/netdev/net: (108 commits) ixgbe: need_wakeup flag might not be set for Tx i40e: need_wakeup flag might not be set for Tx igb/igc: use ktime accessors for skb->tstamp i40e: Fix for ethtool -m issue on X722 NIC iavf: initialize ITRN registers with correct values ice: fix potential infinite loop because loop counter being too small qede: fix NULL pointer deref in __qede_remove() net: fix data-race in neigh_event_send() vsock/virtio: fix sock refcnt holding during the shutdown net: ethernet: octeon_mgmt: Account for second possible VLAN header mac80211: fix station inactive_time shortly after boot net/fq_impl: Switch to kvmalloc() for memory allocation mac80211: fix ieee80211_txq_setup_flows() failure path ipv4: Fix table id reference in fib_sync_down_addr ipv6: fixes rt6_probe() and fib6_nh->last_probe init net: hns: Fix the stray netpoll locks causing deadlock in NAPI path net: usb: qmi_wwan: add support for DW5821e with eSIM support CDC-NCM: handle incomplete transfer of MTU nfc: netlink: fix double device reference drop NFC: st21nfca: fix double free ...
This commit is contained in:
commit
0058b0a506
@ -436,6 +436,10 @@ by the driver:
|
|||||||
encryption.
|
encryption.
|
||||||
* ``tx_tls_ooo`` - number of TX packets which were part of a TLS stream
|
* ``tx_tls_ooo`` - number of TX packets which were part of a TLS stream
|
||||||
but did not arrive in the expected order.
|
but did not arrive in the expected order.
|
||||||
|
* ``tx_tls_skip_no_sync_data`` - number of TX packets which were part of
|
||||||
|
a TLS stream and arrived out-of-order, but skipped the HW offload routine
|
||||||
|
and went to the regular transmit flow as they were retransmissions of the
|
||||||
|
connection handshake.
|
||||||
* ``tx_tls_drop_no_sync_data`` - number of TX packets which were part of
|
* ``tx_tls_drop_no_sync_data`` - number of TX packets which were part of
|
||||||
a TLS stream dropped, because they arrived out of order and associated
|
a TLS stream dropped, because they arrived out of order and associated
|
||||||
record could not be found.
|
record could not be found.
|
||||||
|
@ -3053,6 +3053,7 @@ M: Daniel Borkmann <daniel@iogearbox.net>
|
|||||||
R: Martin KaFai Lau <kafai@fb.com>
|
R: Martin KaFai Lau <kafai@fb.com>
|
||||||
R: Song Liu <songliubraving@fb.com>
|
R: Song Liu <songliubraving@fb.com>
|
||||||
R: Yonghong Song <yhs@fb.com>
|
R: Yonghong Song <yhs@fb.com>
|
||||||
|
R: Andrii Nakryiko <andriin@fb.com>
|
||||||
L: netdev@vger.kernel.org
|
L: netdev@vger.kernel.org
|
||||||
L: bpf@vger.kernel.org
|
L: bpf@vger.kernel.org
|
||||||
T: git git://git.kernel.org/pub/scm/linux/kernel/git/bpf/bpf.git
|
T: git git://git.kernel.org/pub/scm/linux/kernel/git/bpf/bpf.git
|
||||||
|
@ -1141,6 +1141,19 @@ struct bpf_prog *bpf_int_jit_compile(struct bpf_prog *fp)
|
|||||||
goto out_addrs;
|
goto out_addrs;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* If we have seen a tail call, we need a second pass.
|
||||||
|
* This is because bpf_jit_emit_common_epilogue() is called
|
||||||
|
* from bpf_jit_emit_tail_call() with a not yet stable ctx->seen.
|
||||||
|
*/
|
||||||
|
if (cgctx.seen & SEEN_TAILCALL) {
|
||||||
|
cgctx.idx = 0;
|
||||||
|
if (bpf_jit_build_body(fp, 0, &cgctx, addrs, false)) {
|
||||||
|
fp = org_fp;
|
||||||
|
goto out_addrs;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Pretend to build prologue, given the features we've seen. This will
|
* Pretend to build prologue, given the features we've seen. This will
|
||||||
* update ctgtx.idx as it pretends to output instructions, then we can
|
* update ctgtx.idx as it pretends to output instructions, then we can
|
||||||
|
@ -2083,8 +2083,7 @@ static int bond_miimon_inspect(struct bonding *bond)
|
|||||||
ignore_updelay = !rcu_dereference(bond->curr_active_slave);
|
ignore_updelay = !rcu_dereference(bond->curr_active_slave);
|
||||||
|
|
||||||
bond_for_each_slave_rcu(bond, slave, iter) {
|
bond_for_each_slave_rcu(bond, slave, iter) {
|
||||||
slave->new_link = BOND_LINK_NOCHANGE;
|
bond_propose_link_state(slave, BOND_LINK_NOCHANGE);
|
||||||
slave->link_new_state = slave->link;
|
|
||||||
|
|
||||||
link_state = bond_check_dev_link(bond, slave->dev, 0);
|
link_state = bond_check_dev_link(bond, slave->dev, 0);
|
||||||
|
|
||||||
@ -2118,7 +2117,7 @@ static int bond_miimon_inspect(struct bonding *bond)
|
|||||||
}
|
}
|
||||||
|
|
||||||
if (slave->delay <= 0) {
|
if (slave->delay <= 0) {
|
||||||
slave->new_link = BOND_LINK_DOWN;
|
bond_propose_link_state(slave, BOND_LINK_DOWN);
|
||||||
commit++;
|
commit++;
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
@ -2155,7 +2154,7 @@ static int bond_miimon_inspect(struct bonding *bond)
|
|||||||
slave->delay = 0;
|
slave->delay = 0;
|
||||||
|
|
||||||
if (slave->delay <= 0) {
|
if (slave->delay <= 0) {
|
||||||
slave->new_link = BOND_LINK_UP;
|
bond_propose_link_state(slave, BOND_LINK_UP);
|
||||||
commit++;
|
commit++;
|
||||||
ignore_updelay = false;
|
ignore_updelay = false;
|
||||||
continue;
|
continue;
|
||||||
@ -2193,7 +2192,7 @@ static void bond_miimon_commit(struct bonding *bond)
|
|||||||
struct slave *slave, *primary;
|
struct slave *slave, *primary;
|
||||||
|
|
||||||
bond_for_each_slave(bond, slave, iter) {
|
bond_for_each_slave(bond, slave, iter) {
|
||||||
switch (slave->new_link) {
|
switch (slave->link_new_state) {
|
||||||
case BOND_LINK_NOCHANGE:
|
case BOND_LINK_NOCHANGE:
|
||||||
/* For 802.3ad mode, check current slave speed and
|
/* For 802.3ad mode, check current slave speed and
|
||||||
* duplex again in case its port was disabled after
|
* duplex again in case its port was disabled after
|
||||||
@ -2265,8 +2264,8 @@ static void bond_miimon_commit(struct bonding *bond)
|
|||||||
|
|
||||||
default:
|
default:
|
||||||
slave_err(bond->dev, slave->dev, "invalid new link %d on slave\n",
|
slave_err(bond->dev, slave->dev, "invalid new link %d on slave\n",
|
||||||
slave->new_link);
|
slave->link_new_state);
|
||||||
slave->new_link = BOND_LINK_NOCHANGE;
|
bond_propose_link_state(slave, BOND_LINK_NOCHANGE);
|
||||||
|
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
@ -2674,13 +2673,13 @@ static void bond_loadbalance_arp_mon(struct bonding *bond)
|
|||||||
bond_for_each_slave_rcu(bond, slave, iter) {
|
bond_for_each_slave_rcu(bond, slave, iter) {
|
||||||
unsigned long trans_start = dev_trans_start(slave->dev);
|
unsigned long trans_start = dev_trans_start(slave->dev);
|
||||||
|
|
||||||
slave->new_link = BOND_LINK_NOCHANGE;
|
bond_propose_link_state(slave, BOND_LINK_NOCHANGE);
|
||||||
|
|
||||||
if (slave->link != BOND_LINK_UP) {
|
if (slave->link != BOND_LINK_UP) {
|
||||||
if (bond_time_in_interval(bond, trans_start, 1) &&
|
if (bond_time_in_interval(bond, trans_start, 1) &&
|
||||||
bond_time_in_interval(bond, slave->last_rx, 1)) {
|
bond_time_in_interval(bond, slave->last_rx, 1)) {
|
||||||
|
|
||||||
slave->new_link = BOND_LINK_UP;
|
bond_propose_link_state(slave, BOND_LINK_UP);
|
||||||
slave_state_changed = 1;
|
slave_state_changed = 1;
|
||||||
|
|
||||||
/* primary_slave has no meaning in round-robin
|
/* primary_slave has no meaning in round-robin
|
||||||
@ -2705,7 +2704,7 @@ static void bond_loadbalance_arp_mon(struct bonding *bond)
|
|||||||
if (!bond_time_in_interval(bond, trans_start, 2) ||
|
if (!bond_time_in_interval(bond, trans_start, 2) ||
|
||||||
!bond_time_in_interval(bond, slave->last_rx, 2)) {
|
!bond_time_in_interval(bond, slave->last_rx, 2)) {
|
||||||
|
|
||||||
slave->new_link = BOND_LINK_DOWN;
|
bond_propose_link_state(slave, BOND_LINK_DOWN);
|
||||||
slave_state_changed = 1;
|
slave_state_changed = 1;
|
||||||
|
|
||||||
if (slave->link_failure_count < UINT_MAX)
|
if (slave->link_failure_count < UINT_MAX)
|
||||||
@ -2736,8 +2735,8 @@ static void bond_loadbalance_arp_mon(struct bonding *bond)
|
|||||||
goto re_arm;
|
goto re_arm;
|
||||||
|
|
||||||
bond_for_each_slave(bond, slave, iter) {
|
bond_for_each_slave(bond, slave, iter) {
|
||||||
if (slave->new_link != BOND_LINK_NOCHANGE)
|
if (slave->link_new_state != BOND_LINK_NOCHANGE)
|
||||||
slave->link = slave->new_link;
|
slave->link = slave->link_new_state;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (slave_state_changed) {
|
if (slave_state_changed) {
|
||||||
@ -2760,9 +2759,9 @@ re_arm:
|
|||||||
}
|
}
|
||||||
|
|
||||||
/* Called to inspect slaves for active-backup mode ARP monitor link state
|
/* Called to inspect slaves for active-backup mode ARP monitor link state
|
||||||
* changes. Sets new_link in slaves to specify what action should take
|
* changes. Sets proposed link state in slaves to specify what action
|
||||||
* place for the slave. Returns 0 if no changes are found, >0 if changes
|
* should take place for the slave. Returns 0 if no changes are found, >0
|
||||||
* to link states must be committed.
|
* if changes to link states must be committed.
|
||||||
*
|
*
|
||||||
* Called with rcu_read_lock held.
|
* Called with rcu_read_lock held.
|
||||||
*/
|
*/
|
||||||
@ -2774,12 +2773,12 @@ static int bond_ab_arp_inspect(struct bonding *bond)
|
|||||||
int commit = 0;
|
int commit = 0;
|
||||||
|
|
||||||
bond_for_each_slave_rcu(bond, slave, iter) {
|
bond_for_each_slave_rcu(bond, slave, iter) {
|
||||||
slave->new_link = BOND_LINK_NOCHANGE;
|
bond_propose_link_state(slave, BOND_LINK_NOCHANGE);
|
||||||
last_rx = slave_last_rx(bond, slave);
|
last_rx = slave_last_rx(bond, slave);
|
||||||
|
|
||||||
if (slave->link != BOND_LINK_UP) {
|
if (slave->link != BOND_LINK_UP) {
|
||||||
if (bond_time_in_interval(bond, last_rx, 1)) {
|
if (bond_time_in_interval(bond, last_rx, 1)) {
|
||||||
slave->new_link = BOND_LINK_UP;
|
bond_propose_link_state(slave, BOND_LINK_UP);
|
||||||
commit++;
|
commit++;
|
||||||
}
|
}
|
||||||
continue;
|
continue;
|
||||||
@ -2807,7 +2806,7 @@ static int bond_ab_arp_inspect(struct bonding *bond)
|
|||||||
if (!bond_is_active_slave(slave) &&
|
if (!bond_is_active_slave(slave) &&
|
||||||
!rcu_access_pointer(bond->current_arp_slave) &&
|
!rcu_access_pointer(bond->current_arp_slave) &&
|
||||||
!bond_time_in_interval(bond, last_rx, 3)) {
|
!bond_time_in_interval(bond, last_rx, 3)) {
|
||||||
slave->new_link = BOND_LINK_DOWN;
|
bond_propose_link_state(slave, BOND_LINK_DOWN);
|
||||||
commit++;
|
commit++;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -2820,7 +2819,7 @@ static int bond_ab_arp_inspect(struct bonding *bond)
|
|||||||
if (bond_is_active_slave(slave) &&
|
if (bond_is_active_slave(slave) &&
|
||||||
(!bond_time_in_interval(bond, trans_start, 2) ||
|
(!bond_time_in_interval(bond, trans_start, 2) ||
|
||||||
!bond_time_in_interval(bond, last_rx, 2))) {
|
!bond_time_in_interval(bond, last_rx, 2))) {
|
||||||
slave->new_link = BOND_LINK_DOWN;
|
bond_propose_link_state(slave, BOND_LINK_DOWN);
|
||||||
commit++;
|
commit++;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -2840,7 +2839,7 @@ static void bond_ab_arp_commit(struct bonding *bond)
|
|||||||
struct slave *slave;
|
struct slave *slave;
|
||||||
|
|
||||||
bond_for_each_slave(bond, slave, iter) {
|
bond_for_each_slave(bond, slave, iter) {
|
||||||
switch (slave->new_link) {
|
switch (slave->link_new_state) {
|
||||||
case BOND_LINK_NOCHANGE:
|
case BOND_LINK_NOCHANGE:
|
||||||
continue;
|
continue;
|
||||||
|
|
||||||
@ -2890,8 +2889,9 @@ static void bond_ab_arp_commit(struct bonding *bond)
|
|||||||
continue;
|
continue;
|
||||||
|
|
||||||
default:
|
default:
|
||||||
slave_err(bond->dev, slave->dev, "impossible: new_link %d on slave\n",
|
slave_err(bond->dev, slave->dev,
|
||||||
slave->new_link);
|
"impossible: link_new_state %d on slave\n",
|
||||||
|
slave->link_new_state);
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -52,6 +52,7 @@
|
|||||||
#define CONTROL_EX_PDR BIT(8)
|
#define CONTROL_EX_PDR BIT(8)
|
||||||
|
|
||||||
/* control register */
|
/* control register */
|
||||||
|
#define CONTROL_SWR BIT(15)
|
||||||
#define CONTROL_TEST BIT(7)
|
#define CONTROL_TEST BIT(7)
|
||||||
#define CONTROL_CCE BIT(6)
|
#define CONTROL_CCE BIT(6)
|
||||||
#define CONTROL_DISABLE_AR BIT(5)
|
#define CONTROL_DISABLE_AR BIT(5)
|
||||||
@ -97,6 +98,9 @@
|
|||||||
#define BTR_TSEG2_SHIFT 12
|
#define BTR_TSEG2_SHIFT 12
|
||||||
#define BTR_TSEG2_MASK (0x7 << BTR_TSEG2_SHIFT)
|
#define BTR_TSEG2_MASK (0x7 << BTR_TSEG2_SHIFT)
|
||||||
|
|
||||||
|
/* interrupt register */
|
||||||
|
#define INT_STS_PENDING 0x8000
|
||||||
|
|
||||||
/* brp extension register */
|
/* brp extension register */
|
||||||
#define BRP_EXT_BRPE_MASK 0x0f
|
#define BRP_EXT_BRPE_MASK 0x0f
|
||||||
#define BRP_EXT_BRPE_SHIFT 0
|
#define BRP_EXT_BRPE_SHIFT 0
|
||||||
@ -569,6 +573,26 @@ static void c_can_configure_msg_objects(struct net_device *dev)
|
|||||||
IF_MCONT_RCV_EOB);
|
IF_MCONT_RCV_EOB);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static int c_can_software_reset(struct net_device *dev)
|
||||||
|
{
|
||||||
|
struct c_can_priv *priv = netdev_priv(dev);
|
||||||
|
int retry = 0;
|
||||||
|
|
||||||
|
if (priv->type != BOSCH_D_CAN)
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
priv->write_reg(priv, C_CAN_CTRL_REG, CONTROL_SWR | CONTROL_INIT);
|
||||||
|
while (priv->read_reg(priv, C_CAN_CTRL_REG) & CONTROL_SWR) {
|
||||||
|
msleep(20);
|
||||||
|
if (retry++ > 100) {
|
||||||
|
netdev_err(dev, "CCTRL: software reset failed\n");
|
||||||
|
return -EIO;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Configure C_CAN chip:
|
* Configure C_CAN chip:
|
||||||
* - enable/disable auto-retransmission
|
* - enable/disable auto-retransmission
|
||||||
@ -578,6 +602,11 @@ static void c_can_configure_msg_objects(struct net_device *dev)
|
|||||||
static int c_can_chip_config(struct net_device *dev)
|
static int c_can_chip_config(struct net_device *dev)
|
||||||
{
|
{
|
||||||
struct c_can_priv *priv = netdev_priv(dev);
|
struct c_can_priv *priv = netdev_priv(dev);
|
||||||
|
int err;
|
||||||
|
|
||||||
|
err = c_can_software_reset(dev);
|
||||||
|
if (err)
|
||||||
|
return err;
|
||||||
|
|
||||||
/* enable automatic retransmission */
|
/* enable automatic retransmission */
|
||||||
priv->write_reg(priv, C_CAN_CTRL_REG, CONTROL_ENABLE_AR);
|
priv->write_reg(priv, C_CAN_CTRL_REG, CONTROL_ENABLE_AR);
|
||||||
@ -886,6 +915,9 @@ static int c_can_handle_state_change(struct net_device *dev,
|
|||||||
struct can_berr_counter bec;
|
struct can_berr_counter bec;
|
||||||
|
|
||||||
switch (error_type) {
|
switch (error_type) {
|
||||||
|
case C_CAN_NO_ERROR:
|
||||||
|
priv->can.state = CAN_STATE_ERROR_ACTIVE;
|
||||||
|
break;
|
||||||
case C_CAN_ERROR_WARNING:
|
case C_CAN_ERROR_WARNING:
|
||||||
/* error warning state */
|
/* error warning state */
|
||||||
priv->can.can_stats.error_warning++;
|
priv->can.can_stats.error_warning++;
|
||||||
@ -916,6 +948,13 @@ static int c_can_handle_state_change(struct net_device *dev,
|
|||||||
ERR_CNT_RP_SHIFT;
|
ERR_CNT_RP_SHIFT;
|
||||||
|
|
||||||
switch (error_type) {
|
switch (error_type) {
|
||||||
|
case C_CAN_NO_ERROR:
|
||||||
|
/* error warning state */
|
||||||
|
cf->can_id |= CAN_ERR_CRTL;
|
||||||
|
cf->data[1] = CAN_ERR_CRTL_ACTIVE;
|
||||||
|
cf->data[6] = bec.txerr;
|
||||||
|
cf->data[7] = bec.rxerr;
|
||||||
|
break;
|
||||||
case C_CAN_ERROR_WARNING:
|
case C_CAN_ERROR_WARNING:
|
||||||
/* error warning state */
|
/* error warning state */
|
||||||
cf->can_id |= CAN_ERR_CRTL;
|
cf->can_id |= CAN_ERR_CRTL;
|
||||||
@ -1029,10 +1068,16 @@ static int c_can_poll(struct napi_struct *napi, int quota)
|
|||||||
u16 curr, last = priv->last_status;
|
u16 curr, last = priv->last_status;
|
||||||
int work_done = 0;
|
int work_done = 0;
|
||||||
|
|
||||||
priv->last_status = curr = priv->read_reg(priv, C_CAN_STS_REG);
|
/* Only read the status register if a status interrupt was pending */
|
||||||
/* Ack status on C_CAN. D_CAN is self clearing */
|
if (atomic_xchg(&priv->sie_pending, 0)) {
|
||||||
if (priv->type != BOSCH_D_CAN)
|
priv->last_status = curr = priv->read_reg(priv, C_CAN_STS_REG);
|
||||||
priv->write_reg(priv, C_CAN_STS_REG, LEC_UNUSED);
|
/* Ack status on C_CAN. D_CAN is self clearing */
|
||||||
|
if (priv->type != BOSCH_D_CAN)
|
||||||
|
priv->write_reg(priv, C_CAN_STS_REG, LEC_UNUSED);
|
||||||
|
} else {
|
||||||
|
/* no change detected ... */
|
||||||
|
curr = last;
|
||||||
|
}
|
||||||
|
|
||||||
/* handle state changes */
|
/* handle state changes */
|
||||||
if ((curr & STATUS_EWARN) && (!(last & STATUS_EWARN))) {
|
if ((curr & STATUS_EWARN) && (!(last & STATUS_EWARN))) {
|
||||||
@ -1054,11 +1099,17 @@ static int c_can_poll(struct napi_struct *napi, int quota)
|
|||||||
/* handle bus recovery events */
|
/* handle bus recovery events */
|
||||||
if ((!(curr & STATUS_BOFF)) && (last & STATUS_BOFF)) {
|
if ((!(curr & STATUS_BOFF)) && (last & STATUS_BOFF)) {
|
||||||
netdev_dbg(dev, "left bus off state\n");
|
netdev_dbg(dev, "left bus off state\n");
|
||||||
priv->can.state = CAN_STATE_ERROR_ACTIVE;
|
work_done += c_can_handle_state_change(dev, C_CAN_ERROR_PASSIVE);
|
||||||
}
|
}
|
||||||
|
|
||||||
if ((!(curr & STATUS_EPASS)) && (last & STATUS_EPASS)) {
|
if ((!(curr & STATUS_EPASS)) && (last & STATUS_EPASS)) {
|
||||||
netdev_dbg(dev, "left error passive state\n");
|
netdev_dbg(dev, "left error passive state\n");
|
||||||
priv->can.state = CAN_STATE_ERROR_ACTIVE;
|
work_done += c_can_handle_state_change(dev, C_CAN_ERROR_WARNING);
|
||||||
|
}
|
||||||
|
|
||||||
|
if ((!(curr & STATUS_EWARN)) && (last & STATUS_EWARN)) {
|
||||||
|
netdev_dbg(dev, "left error warning state\n");
|
||||||
|
work_done += c_can_handle_state_change(dev, C_CAN_NO_ERROR);
|
||||||
}
|
}
|
||||||
|
|
||||||
/* handle lec errors on the bus */
|
/* handle lec errors on the bus */
|
||||||
@ -1083,10 +1134,16 @@ static irqreturn_t c_can_isr(int irq, void *dev_id)
|
|||||||
{
|
{
|
||||||
struct net_device *dev = (struct net_device *)dev_id;
|
struct net_device *dev = (struct net_device *)dev_id;
|
||||||
struct c_can_priv *priv = netdev_priv(dev);
|
struct c_can_priv *priv = netdev_priv(dev);
|
||||||
|
int reg_int;
|
||||||
|
|
||||||
if (!priv->read_reg(priv, C_CAN_INT_REG))
|
reg_int = priv->read_reg(priv, C_CAN_INT_REG);
|
||||||
|
if (!reg_int)
|
||||||
return IRQ_NONE;
|
return IRQ_NONE;
|
||||||
|
|
||||||
|
/* save for later use */
|
||||||
|
if (reg_int & INT_STS_PENDING)
|
||||||
|
atomic_set(&priv->sie_pending, 1);
|
||||||
|
|
||||||
/* disable all interrupts and schedule the NAPI */
|
/* disable all interrupts and schedule the NAPI */
|
||||||
c_can_irq_control(priv, false);
|
c_can_irq_control(priv, false);
|
||||||
napi_schedule(&priv->napi);
|
napi_schedule(&priv->napi);
|
||||||
|
@ -198,6 +198,7 @@ struct c_can_priv {
|
|||||||
struct net_device *dev;
|
struct net_device *dev;
|
||||||
struct device *device;
|
struct device *device;
|
||||||
atomic_t tx_active;
|
atomic_t tx_active;
|
||||||
|
atomic_t sie_pending;
|
||||||
unsigned long tx_dir;
|
unsigned long tx_dir;
|
||||||
int last_status;
|
int last_status;
|
||||||
u16 (*read_reg) (const struct c_can_priv *priv, enum reg index);
|
u16 (*read_reg) (const struct c_can_priv *priv, enum reg index);
|
||||||
|
@ -848,6 +848,7 @@ void of_can_transceiver(struct net_device *dev)
|
|||||||
return;
|
return;
|
||||||
|
|
||||||
ret = of_property_read_u32(dn, "max-bitrate", &priv->bitrate_max);
|
ret = of_property_read_u32(dn, "max-bitrate", &priv->bitrate_max);
|
||||||
|
of_node_put(dn);
|
||||||
if ((ret && ret != -EINVAL) || (!ret && !priv->bitrate_max))
|
if ((ret && ret != -EINVAL) || (!ret && !priv->bitrate_max))
|
||||||
netdev_warn(dev, "Invalid value for transceiver max bitrate. Ignoring bitrate limit.\n");
|
netdev_warn(dev, "Invalid value for transceiver max bitrate. Ignoring bitrate limit.\n");
|
||||||
}
|
}
|
||||||
|
@ -677,6 +677,7 @@ static void flexcan_irq_bus_err(struct net_device *dev, u32 reg_esr)
|
|||||||
struct can_frame *cf;
|
struct can_frame *cf;
|
||||||
bool rx_errors = false, tx_errors = false;
|
bool rx_errors = false, tx_errors = false;
|
||||||
u32 timestamp;
|
u32 timestamp;
|
||||||
|
int err;
|
||||||
|
|
||||||
timestamp = priv->read(®s->timer) << 16;
|
timestamp = priv->read(®s->timer) << 16;
|
||||||
|
|
||||||
@ -725,7 +726,9 @@ static void flexcan_irq_bus_err(struct net_device *dev, u32 reg_esr)
|
|||||||
if (tx_errors)
|
if (tx_errors)
|
||||||
dev->stats.tx_errors++;
|
dev->stats.tx_errors++;
|
||||||
|
|
||||||
can_rx_offload_queue_sorted(&priv->offload, skb, timestamp);
|
err = can_rx_offload_queue_sorted(&priv->offload, skb, timestamp);
|
||||||
|
if (err)
|
||||||
|
dev->stats.rx_fifo_errors++;
|
||||||
}
|
}
|
||||||
|
|
||||||
static void flexcan_irq_state(struct net_device *dev, u32 reg_esr)
|
static void flexcan_irq_state(struct net_device *dev, u32 reg_esr)
|
||||||
@ -738,6 +741,7 @@ static void flexcan_irq_state(struct net_device *dev, u32 reg_esr)
|
|||||||
int flt;
|
int flt;
|
||||||
struct can_berr_counter bec;
|
struct can_berr_counter bec;
|
||||||
u32 timestamp;
|
u32 timestamp;
|
||||||
|
int err;
|
||||||
|
|
||||||
timestamp = priv->read(®s->timer) << 16;
|
timestamp = priv->read(®s->timer) << 16;
|
||||||
|
|
||||||
@ -769,7 +773,9 @@ static void flexcan_irq_state(struct net_device *dev, u32 reg_esr)
|
|||||||
if (unlikely(new_state == CAN_STATE_BUS_OFF))
|
if (unlikely(new_state == CAN_STATE_BUS_OFF))
|
||||||
can_bus_off(dev);
|
can_bus_off(dev);
|
||||||
|
|
||||||
can_rx_offload_queue_sorted(&priv->offload, skb, timestamp);
|
err = can_rx_offload_queue_sorted(&priv->offload, skb, timestamp);
|
||||||
|
if (err)
|
||||||
|
dev->stats.rx_fifo_errors++;
|
||||||
}
|
}
|
||||||
|
|
||||||
static inline struct flexcan_priv *rx_offload_to_priv(struct can_rx_offload *offload)
|
static inline struct flexcan_priv *rx_offload_to_priv(struct can_rx_offload *offload)
|
||||||
@ -1188,6 +1194,7 @@ static int flexcan_chip_start(struct net_device *dev)
|
|||||||
reg_mecr = priv->read(®s->mecr);
|
reg_mecr = priv->read(®s->mecr);
|
||||||
reg_mecr &= ~FLEXCAN_MECR_ECRWRDIS;
|
reg_mecr &= ~FLEXCAN_MECR_ECRWRDIS;
|
||||||
priv->write(reg_mecr, ®s->mecr);
|
priv->write(reg_mecr, ®s->mecr);
|
||||||
|
reg_mecr |= FLEXCAN_MECR_ECCDIS;
|
||||||
reg_mecr &= ~(FLEXCAN_MECR_NCEFAFRZ | FLEXCAN_MECR_HANCEI_MSK |
|
reg_mecr &= ~(FLEXCAN_MECR_NCEFAFRZ | FLEXCAN_MECR_HANCEI_MSK |
|
||||||
FLEXCAN_MECR_FANCEI_MSK);
|
FLEXCAN_MECR_FANCEI_MSK);
|
||||||
priv->write(reg_mecr, ®s->mecr);
|
priv->write(reg_mecr, ®s->mecr);
|
||||||
|
@ -107,37 +107,95 @@ static int can_rx_offload_compare(struct sk_buff *a, struct sk_buff *b)
|
|||||||
return cb_b->timestamp - cb_a->timestamp;
|
return cb_b->timestamp - cb_a->timestamp;
|
||||||
}
|
}
|
||||||
|
|
||||||
static struct sk_buff *can_rx_offload_offload_one(struct can_rx_offload *offload, unsigned int n)
|
/**
|
||||||
|
* can_rx_offload_offload_one() - Read one CAN frame from HW
|
||||||
|
* @offload: pointer to rx_offload context
|
||||||
|
* @n: number of mailbox to read
|
||||||
|
*
|
||||||
|
* The task of this function is to read a CAN frame from mailbox @n
|
||||||
|
* from the device and return the mailbox's content as a struct
|
||||||
|
* sk_buff.
|
||||||
|
*
|
||||||
|
* If the struct can_rx_offload::skb_queue exceeds the maximal queue
|
||||||
|
* length (struct can_rx_offload::skb_queue_len_max) or no skb can be
|
||||||
|
* allocated, the mailbox contents is discarded by reading it into an
|
||||||
|
* overflow buffer. This way the mailbox is marked as free by the
|
||||||
|
* driver.
|
||||||
|
*
|
||||||
|
* Return: A pointer to skb containing the CAN frame on success.
|
||||||
|
*
|
||||||
|
* NULL if the mailbox @n is empty.
|
||||||
|
*
|
||||||
|
* ERR_PTR() in case of an error
|
||||||
|
*/
|
||||||
|
static struct sk_buff *
|
||||||
|
can_rx_offload_offload_one(struct can_rx_offload *offload, unsigned int n)
|
||||||
{
|
{
|
||||||
struct sk_buff *skb = NULL;
|
struct sk_buff *skb = NULL, *skb_error = NULL;
|
||||||
struct can_rx_offload_cb *cb;
|
struct can_rx_offload_cb *cb;
|
||||||
struct can_frame *cf;
|
struct can_frame *cf;
|
||||||
int ret;
|
int ret;
|
||||||
|
|
||||||
/* If queue is full or skb not available, read to discard mailbox */
|
if (likely(skb_queue_len(&offload->skb_queue) <
|
||||||
if (likely(skb_queue_len(&offload->skb_queue) <=
|
offload->skb_queue_len_max)) {
|
||||||
offload->skb_queue_len_max))
|
|
||||||
skb = alloc_can_skb(offload->dev, &cf);
|
skb = alloc_can_skb(offload->dev, &cf);
|
||||||
|
if (unlikely(!skb))
|
||||||
|
skb_error = ERR_PTR(-ENOMEM); /* skb alloc failed */
|
||||||
|
} else {
|
||||||
|
skb_error = ERR_PTR(-ENOBUFS); /* skb_queue is full */
|
||||||
|
}
|
||||||
|
|
||||||
if (!skb) {
|
/* If queue is full or skb not available, drop by reading into
|
||||||
|
* overflow buffer.
|
||||||
|
*/
|
||||||
|
if (unlikely(skb_error)) {
|
||||||
struct can_frame cf_overflow;
|
struct can_frame cf_overflow;
|
||||||
u32 timestamp;
|
u32 timestamp;
|
||||||
|
|
||||||
ret = offload->mailbox_read(offload, &cf_overflow,
|
ret = offload->mailbox_read(offload, &cf_overflow,
|
||||||
×tamp, n);
|
×tamp, n);
|
||||||
if (ret)
|
|
||||||
offload->dev->stats.rx_dropped++;
|
|
||||||
|
|
||||||
return NULL;
|
/* Mailbox was empty. */
|
||||||
|
if (unlikely(!ret))
|
||||||
|
return NULL;
|
||||||
|
|
||||||
|
/* Mailbox has been read and we're dropping it or
|
||||||
|
* there was a problem reading the mailbox.
|
||||||
|
*
|
||||||
|
* Increment error counters in any case.
|
||||||
|
*/
|
||||||
|
offload->dev->stats.rx_dropped++;
|
||||||
|
offload->dev->stats.rx_fifo_errors++;
|
||||||
|
|
||||||
|
/* There was a problem reading the mailbox, propagate
|
||||||
|
* error value.
|
||||||
|
*/
|
||||||
|
if (unlikely(ret < 0))
|
||||||
|
return ERR_PTR(ret);
|
||||||
|
|
||||||
|
return skb_error;
|
||||||
}
|
}
|
||||||
|
|
||||||
cb = can_rx_offload_get_cb(skb);
|
cb = can_rx_offload_get_cb(skb);
|
||||||
ret = offload->mailbox_read(offload, cf, &cb->timestamp, n);
|
ret = offload->mailbox_read(offload, cf, &cb->timestamp, n);
|
||||||
if (!ret) {
|
|
||||||
|
/* Mailbox was empty. */
|
||||||
|
if (unlikely(!ret)) {
|
||||||
kfree_skb(skb);
|
kfree_skb(skb);
|
||||||
return NULL;
|
return NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/* There was a problem reading the mailbox, propagate error value. */
|
||||||
|
if (unlikely(ret < 0)) {
|
||||||
|
kfree_skb(skb);
|
||||||
|
|
||||||
|
offload->dev->stats.rx_dropped++;
|
||||||
|
offload->dev->stats.rx_fifo_errors++;
|
||||||
|
|
||||||
|
return ERR_PTR(ret);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Mailbox was read. */
|
||||||
return skb;
|
return skb;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -157,8 +215,8 @@ int can_rx_offload_irq_offload_timestamp(struct can_rx_offload *offload, u64 pen
|
|||||||
continue;
|
continue;
|
||||||
|
|
||||||
skb = can_rx_offload_offload_one(offload, i);
|
skb = can_rx_offload_offload_one(offload, i);
|
||||||
if (!skb)
|
if (IS_ERR_OR_NULL(skb))
|
||||||
break;
|
continue;
|
||||||
|
|
||||||
__skb_queue_add_sort(&skb_queue, skb, can_rx_offload_compare);
|
__skb_queue_add_sort(&skb_queue, skb, can_rx_offload_compare);
|
||||||
}
|
}
|
||||||
@ -188,7 +246,13 @@ int can_rx_offload_irq_offload_fifo(struct can_rx_offload *offload)
|
|||||||
struct sk_buff *skb;
|
struct sk_buff *skb;
|
||||||
int received = 0;
|
int received = 0;
|
||||||
|
|
||||||
while ((skb = can_rx_offload_offload_one(offload, 0))) {
|
while (1) {
|
||||||
|
skb = can_rx_offload_offload_one(offload, 0);
|
||||||
|
if (IS_ERR(skb))
|
||||||
|
continue;
|
||||||
|
if (!skb)
|
||||||
|
break;
|
||||||
|
|
||||||
skb_queue_tail(&offload->skb_queue, skb);
|
skb_queue_tail(&offload->skb_queue, skb);
|
||||||
received++;
|
received++;
|
||||||
}
|
}
|
||||||
@ -207,8 +271,10 @@ int can_rx_offload_queue_sorted(struct can_rx_offload *offload,
|
|||||||
unsigned long flags;
|
unsigned long flags;
|
||||||
|
|
||||||
if (skb_queue_len(&offload->skb_queue) >
|
if (skb_queue_len(&offload->skb_queue) >
|
||||||
offload->skb_queue_len_max)
|
offload->skb_queue_len_max) {
|
||||||
return -ENOMEM;
|
kfree_skb(skb);
|
||||||
|
return -ENOBUFS;
|
||||||
|
}
|
||||||
|
|
||||||
cb = can_rx_offload_get_cb(skb);
|
cb = can_rx_offload_get_cb(skb);
|
||||||
cb->timestamp = timestamp;
|
cb->timestamp = timestamp;
|
||||||
@ -250,8 +316,10 @@ int can_rx_offload_queue_tail(struct can_rx_offload *offload,
|
|||||||
struct sk_buff *skb)
|
struct sk_buff *skb)
|
||||||
{
|
{
|
||||||
if (skb_queue_len(&offload->skb_queue) >
|
if (skb_queue_len(&offload->skb_queue) >
|
||||||
offload->skb_queue_len_max)
|
offload->skb_queue_len_max) {
|
||||||
return -ENOMEM;
|
kfree_skb(skb);
|
||||||
|
return -ENOBUFS;
|
||||||
|
}
|
||||||
|
|
||||||
skb_queue_tail(&offload->skb_queue, skb);
|
skb_queue_tail(&offload->skb_queue, skb);
|
||||||
can_rx_offload_schedule(offload);
|
can_rx_offload_schedule(offload);
|
||||||
|
@ -717,6 +717,7 @@ static void mcp251x_restart_work_handler(struct work_struct *ws)
|
|||||||
if (priv->after_suspend) {
|
if (priv->after_suspend) {
|
||||||
mcp251x_hw_reset(spi);
|
mcp251x_hw_reset(spi);
|
||||||
mcp251x_setup(net, spi);
|
mcp251x_setup(net, spi);
|
||||||
|
priv->force_quit = 0;
|
||||||
if (priv->after_suspend & AFTER_SUSPEND_RESTART) {
|
if (priv->after_suspend & AFTER_SUSPEND_RESTART) {
|
||||||
mcp251x_set_normal_mode(spi);
|
mcp251x_set_normal_mode(spi);
|
||||||
} else if (priv->after_suspend & AFTER_SUSPEND_UP) {
|
} else if (priv->after_suspend & AFTER_SUSPEND_UP) {
|
||||||
@ -728,7 +729,6 @@ static void mcp251x_restart_work_handler(struct work_struct *ws)
|
|||||||
mcp251x_hw_sleep(spi);
|
mcp251x_hw_sleep(spi);
|
||||||
}
|
}
|
||||||
priv->after_suspend = 0;
|
priv->after_suspend = 0;
|
||||||
priv->force_quit = 0;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
if (priv->restart_tx) {
|
if (priv->restart_tx) {
|
||||||
|
@ -73,6 +73,7 @@ MODULE_VERSION(HECC_MODULE_VERSION);
|
|||||||
*/
|
*/
|
||||||
#define HECC_MAX_RX_MBOX (HECC_MAX_MAILBOXES - HECC_MAX_TX_MBOX)
|
#define HECC_MAX_RX_MBOX (HECC_MAX_MAILBOXES - HECC_MAX_TX_MBOX)
|
||||||
#define HECC_RX_FIRST_MBOX (HECC_MAX_MAILBOXES - 1)
|
#define HECC_RX_FIRST_MBOX (HECC_MAX_MAILBOXES - 1)
|
||||||
|
#define HECC_RX_LAST_MBOX (HECC_MAX_TX_MBOX)
|
||||||
|
|
||||||
/* TI HECC module registers */
|
/* TI HECC module registers */
|
||||||
#define HECC_CANME 0x0 /* Mailbox enable */
|
#define HECC_CANME 0x0 /* Mailbox enable */
|
||||||
@ -82,7 +83,7 @@ MODULE_VERSION(HECC_MODULE_VERSION);
|
|||||||
#define HECC_CANTA 0x10 /* Transmission acknowledge */
|
#define HECC_CANTA 0x10 /* Transmission acknowledge */
|
||||||
#define HECC_CANAA 0x14 /* Abort acknowledge */
|
#define HECC_CANAA 0x14 /* Abort acknowledge */
|
||||||
#define HECC_CANRMP 0x18 /* Receive message pending */
|
#define HECC_CANRMP 0x18 /* Receive message pending */
|
||||||
#define HECC_CANRML 0x1C /* Remote message lost */
|
#define HECC_CANRML 0x1C /* Receive message lost */
|
||||||
#define HECC_CANRFP 0x20 /* Remote frame pending */
|
#define HECC_CANRFP 0x20 /* Remote frame pending */
|
||||||
#define HECC_CANGAM 0x24 /* SECC only:Global acceptance mask */
|
#define HECC_CANGAM 0x24 /* SECC only:Global acceptance mask */
|
||||||
#define HECC_CANMC 0x28 /* Master control */
|
#define HECC_CANMC 0x28 /* Master control */
|
||||||
@ -149,6 +150,8 @@ MODULE_VERSION(HECC_MODULE_VERSION);
|
|||||||
#define HECC_BUS_ERROR (HECC_CANES_FE | HECC_CANES_BE |\
|
#define HECC_BUS_ERROR (HECC_CANES_FE | HECC_CANES_BE |\
|
||||||
HECC_CANES_CRCE | HECC_CANES_SE |\
|
HECC_CANES_CRCE | HECC_CANES_SE |\
|
||||||
HECC_CANES_ACKE)
|
HECC_CANES_ACKE)
|
||||||
|
#define HECC_CANES_FLAGS (HECC_BUS_ERROR | HECC_CANES_BO |\
|
||||||
|
HECC_CANES_EP | HECC_CANES_EW)
|
||||||
|
|
||||||
#define HECC_CANMCF_RTR BIT(4) /* Remote transmit request */
|
#define HECC_CANMCF_RTR BIT(4) /* Remote transmit request */
|
||||||
|
|
||||||
@ -382,8 +385,18 @@ static void ti_hecc_start(struct net_device *ndev)
|
|||||||
hecc_set_bit(priv, HECC_CANMIM, mbx_mask);
|
hecc_set_bit(priv, HECC_CANMIM, mbx_mask);
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Prevent message over-write & Enable interrupts */
|
/* Enable tx interrupts */
|
||||||
hecc_write(priv, HECC_CANOPC, HECC_SET_REG);
|
hecc_set_bit(priv, HECC_CANMIM, BIT(HECC_MAX_TX_MBOX) - 1);
|
||||||
|
|
||||||
|
/* Prevent message over-write to create a rx fifo, but not for
|
||||||
|
* the lowest priority mailbox, since that allows detecting
|
||||||
|
* overflows instead of the hardware silently dropping the
|
||||||
|
* messages.
|
||||||
|
*/
|
||||||
|
mbx_mask = ~BIT(HECC_RX_LAST_MBOX);
|
||||||
|
hecc_write(priv, HECC_CANOPC, mbx_mask);
|
||||||
|
|
||||||
|
/* Enable interrupts */
|
||||||
if (priv->use_hecc1int) {
|
if (priv->use_hecc1int) {
|
||||||
hecc_write(priv, HECC_CANMIL, HECC_SET_REG);
|
hecc_write(priv, HECC_CANMIL, HECC_SET_REG);
|
||||||
hecc_write(priv, HECC_CANGIM, HECC_CANGIM_DEF_MASK |
|
hecc_write(priv, HECC_CANGIM, HECC_CANGIM_DEF_MASK |
|
||||||
@ -400,6 +413,9 @@ static void ti_hecc_stop(struct net_device *ndev)
|
|||||||
{
|
{
|
||||||
struct ti_hecc_priv *priv = netdev_priv(ndev);
|
struct ti_hecc_priv *priv = netdev_priv(ndev);
|
||||||
|
|
||||||
|
/* Disable the CPK; stop sending, erroring and acking */
|
||||||
|
hecc_set_bit(priv, HECC_CANMC, HECC_CANMC_CCR);
|
||||||
|
|
||||||
/* Disable interrupts and disable mailboxes */
|
/* Disable interrupts and disable mailboxes */
|
||||||
hecc_write(priv, HECC_CANGIM, 0);
|
hecc_write(priv, HECC_CANGIM, 0);
|
||||||
hecc_write(priv, HECC_CANMIM, 0);
|
hecc_write(priv, HECC_CANMIM, 0);
|
||||||
@ -508,8 +524,6 @@ static netdev_tx_t ti_hecc_xmit(struct sk_buff *skb, struct net_device *ndev)
|
|||||||
hecc_set_bit(priv, HECC_CANME, mbx_mask);
|
hecc_set_bit(priv, HECC_CANME, mbx_mask);
|
||||||
spin_unlock_irqrestore(&priv->mbx_lock, flags);
|
spin_unlock_irqrestore(&priv->mbx_lock, flags);
|
||||||
|
|
||||||
hecc_clear_bit(priv, HECC_CANMD, mbx_mask);
|
|
||||||
hecc_set_bit(priv, HECC_CANMIM, mbx_mask);
|
|
||||||
hecc_write(priv, HECC_CANTRS, mbx_mask);
|
hecc_write(priv, HECC_CANTRS, mbx_mask);
|
||||||
|
|
||||||
return NETDEV_TX_OK;
|
return NETDEV_TX_OK;
|
||||||
@ -526,8 +540,10 @@ static unsigned int ti_hecc_mailbox_read(struct can_rx_offload *offload,
|
|||||||
u32 *timestamp, unsigned int mbxno)
|
u32 *timestamp, unsigned int mbxno)
|
||||||
{
|
{
|
||||||
struct ti_hecc_priv *priv = rx_offload_to_priv(offload);
|
struct ti_hecc_priv *priv = rx_offload_to_priv(offload);
|
||||||
u32 data;
|
u32 data, mbx_mask;
|
||||||
|
int ret = 1;
|
||||||
|
|
||||||
|
mbx_mask = BIT(mbxno);
|
||||||
data = hecc_read_mbx(priv, mbxno, HECC_CANMID);
|
data = hecc_read_mbx(priv, mbxno, HECC_CANMID);
|
||||||
if (data & HECC_CANMID_IDE)
|
if (data & HECC_CANMID_IDE)
|
||||||
cf->can_id = (data & CAN_EFF_MASK) | CAN_EFF_FLAG;
|
cf->can_id = (data & CAN_EFF_MASK) | CAN_EFF_FLAG;
|
||||||
@ -548,7 +564,25 @@ static unsigned int ti_hecc_mailbox_read(struct can_rx_offload *offload,
|
|||||||
|
|
||||||
*timestamp = hecc_read_stamp(priv, mbxno);
|
*timestamp = hecc_read_stamp(priv, mbxno);
|
||||||
|
|
||||||
return 1;
|
/* Check for FIFO overrun.
|
||||||
|
*
|
||||||
|
* All but the last RX mailbox have activated overwrite
|
||||||
|
* protection. So skip check for overrun, if we're not
|
||||||
|
* handling the last RX mailbox.
|
||||||
|
*
|
||||||
|
* As the overwrite protection for the last RX mailbox is
|
||||||
|
* disabled, the CAN core might update while we're reading
|
||||||
|
* it. This means the skb might be inconsistent.
|
||||||
|
*
|
||||||
|
* Return an error to let rx-offload discard this CAN frame.
|
||||||
|
*/
|
||||||
|
if (unlikely(mbxno == HECC_RX_LAST_MBOX &&
|
||||||
|
hecc_read(priv, HECC_CANRML) & mbx_mask))
|
||||||
|
ret = -ENOBUFS;
|
||||||
|
|
||||||
|
hecc_write(priv, HECC_CANRMP, mbx_mask);
|
||||||
|
|
||||||
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
static int ti_hecc_error(struct net_device *ndev, int int_status,
|
static int ti_hecc_error(struct net_device *ndev, int int_status,
|
||||||
@ -558,90 +592,71 @@ static int ti_hecc_error(struct net_device *ndev, int int_status,
|
|||||||
struct can_frame *cf;
|
struct can_frame *cf;
|
||||||
struct sk_buff *skb;
|
struct sk_buff *skb;
|
||||||
u32 timestamp;
|
u32 timestamp;
|
||||||
|
int err;
|
||||||
/* propagate the error condition to the can stack */
|
|
||||||
skb = alloc_can_err_skb(ndev, &cf);
|
|
||||||
if (!skb) {
|
|
||||||
if (printk_ratelimit())
|
|
||||||
netdev_err(priv->ndev,
|
|
||||||
"%s: alloc_can_err_skb() failed\n",
|
|
||||||
__func__);
|
|
||||||
return -ENOMEM;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (int_status & HECC_CANGIF_WLIF) { /* warning level int */
|
|
||||||
if ((int_status & HECC_CANGIF_BOIF) == 0) {
|
|
||||||
priv->can.state = CAN_STATE_ERROR_WARNING;
|
|
||||||
++priv->can.can_stats.error_warning;
|
|
||||||
cf->can_id |= CAN_ERR_CRTL;
|
|
||||||
if (hecc_read(priv, HECC_CANTEC) > 96)
|
|
||||||
cf->data[1] |= CAN_ERR_CRTL_TX_WARNING;
|
|
||||||
if (hecc_read(priv, HECC_CANREC) > 96)
|
|
||||||
cf->data[1] |= CAN_ERR_CRTL_RX_WARNING;
|
|
||||||
}
|
|
||||||
hecc_set_bit(priv, HECC_CANES, HECC_CANES_EW);
|
|
||||||
netdev_dbg(priv->ndev, "Error Warning interrupt\n");
|
|
||||||
hecc_clear_bit(priv, HECC_CANMC, HECC_CANMC_CCR);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (int_status & HECC_CANGIF_EPIF) { /* error passive int */
|
|
||||||
if ((int_status & HECC_CANGIF_BOIF) == 0) {
|
|
||||||
priv->can.state = CAN_STATE_ERROR_PASSIVE;
|
|
||||||
++priv->can.can_stats.error_passive;
|
|
||||||
cf->can_id |= CAN_ERR_CRTL;
|
|
||||||
if (hecc_read(priv, HECC_CANTEC) > 127)
|
|
||||||
cf->data[1] |= CAN_ERR_CRTL_TX_PASSIVE;
|
|
||||||
if (hecc_read(priv, HECC_CANREC) > 127)
|
|
||||||
cf->data[1] |= CAN_ERR_CRTL_RX_PASSIVE;
|
|
||||||
}
|
|
||||||
hecc_set_bit(priv, HECC_CANES, HECC_CANES_EP);
|
|
||||||
netdev_dbg(priv->ndev, "Error passive interrupt\n");
|
|
||||||
hecc_clear_bit(priv, HECC_CANMC, HECC_CANMC_CCR);
|
|
||||||
}
|
|
||||||
|
|
||||||
/* Need to check busoff condition in error status register too to
|
|
||||||
* ensure warning interrupts don't hog the system
|
|
||||||
*/
|
|
||||||
if ((int_status & HECC_CANGIF_BOIF) || (err_status & HECC_CANES_BO)) {
|
|
||||||
priv->can.state = CAN_STATE_BUS_OFF;
|
|
||||||
cf->can_id |= CAN_ERR_BUSOFF;
|
|
||||||
hecc_set_bit(priv, HECC_CANES, HECC_CANES_BO);
|
|
||||||
hecc_clear_bit(priv, HECC_CANMC, HECC_CANMC_CCR);
|
|
||||||
/* Disable all interrupts in bus-off to avoid int hog */
|
|
||||||
hecc_write(priv, HECC_CANGIM, 0);
|
|
||||||
++priv->can.can_stats.bus_off;
|
|
||||||
can_bus_off(ndev);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (err_status & HECC_BUS_ERROR) {
|
if (err_status & HECC_BUS_ERROR) {
|
||||||
|
/* propagate the error condition to the can stack */
|
||||||
|
skb = alloc_can_err_skb(ndev, &cf);
|
||||||
|
if (!skb) {
|
||||||
|
if (net_ratelimit())
|
||||||
|
netdev_err(priv->ndev,
|
||||||
|
"%s: alloc_can_err_skb() failed\n",
|
||||||
|
__func__);
|
||||||
|
return -ENOMEM;
|
||||||
|
}
|
||||||
|
|
||||||
++priv->can.can_stats.bus_error;
|
++priv->can.can_stats.bus_error;
|
||||||
cf->can_id |= CAN_ERR_BUSERROR | CAN_ERR_PROT;
|
cf->can_id |= CAN_ERR_BUSERROR | CAN_ERR_PROT;
|
||||||
if (err_status & HECC_CANES_FE) {
|
if (err_status & HECC_CANES_FE)
|
||||||
hecc_set_bit(priv, HECC_CANES, HECC_CANES_FE);
|
|
||||||
cf->data[2] |= CAN_ERR_PROT_FORM;
|
cf->data[2] |= CAN_ERR_PROT_FORM;
|
||||||
}
|
if (err_status & HECC_CANES_BE)
|
||||||
if (err_status & HECC_CANES_BE) {
|
|
||||||
hecc_set_bit(priv, HECC_CANES, HECC_CANES_BE);
|
|
||||||
cf->data[2] |= CAN_ERR_PROT_BIT;
|
cf->data[2] |= CAN_ERR_PROT_BIT;
|
||||||
}
|
if (err_status & HECC_CANES_SE)
|
||||||
if (err_status & HECC_CANES_SE) {
|
|
||||||
hecc_set_bit(priv, HECC_CANES, HECC_CANES_SE);
|
|
||||||
cf->data[2] |= CAN_ERR_PROT_STUFF;
|
cf->data[2] |= CAN_ERR_PROT_STUFF;
|
||||||
}
|
if (err_status & HECC_CANES_CRCE)
|
||||||
if (err_status & HECC_CANES_CRCE) {
|
|
||||||
hecc_set_bit(priv, HECC_CANES, HECC_CANES_CRCE);
|
|
||||||
cf->data[3] = CAN_ERR_PROT_LOC_CRC_SEQ;
|
cf->data[3] = CAN_ERR_PROT_LOC_CRC_SEQ;
|
||||||
}
|
if (err_status & HECC_CANES_ACKE)
|
||||||
if (err_status & HECC_CANES_ACKE) {
|
|
||||||
hecc_set_bit(priv, HECC_CANES, HECC_CANES_ACKE);
|
|
||||||
cf->data[3] = CAN_ERR_PROT_LOC_ACK;
|
cf->data[3] = CAN_ERR_PROT_LOC_ACK;
|
||||||
}
|
|
||||||
|
timestamp = hecc_read(priv, HECC_CANLNT);
|
||||||
|
err = can_rx_offload_queue_sorted(&priv->offload, skb,
|
||||||
|
timestamp);
|
||||||
|
if (err)
|
||||||
|
ndev->stats.rx_fifo_errors++;
|
||||||
|
}
|
||||||
|
|
||||||
|
hecc_write(priv, HECC_CANES, HECC_CANES_FLAGS);
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
static void ti_hecc_change_state(struct net_device *ndev,
|
||||||
|
enum can_state rx_state,
|
||||||
|
enum can_state tx_state)
|
||||||
|
{
|
||||||
|
struct ti_hecc_priv *priv = netdev_priv(ndev);
|
||||||
|
struct can_frame *cf;
|
||||||
|
struct sk_buff *skb;
|
||||||
|
u32 timestamp;
|
||||||
|
int err;
|
||||||
|
|
||||||
|
skb = alloc_can_err_skb(priv->ndev, &cf);
|
||||||
|
if (unlikely(!skb)) {
|
||||||
|
priv->can.state = max(tx_state, rx_state);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
can_change_state(priv->ndev, cf, tx_state, rx_state);
|
||||||
|
|
||||||
|
if (max(tx_state, rx_state) != CAN_STATE_BUS_OFF) {
|
||||||
|
cf->data[6] = hecc_read(priv, HECC_CANTEC);
|
||||||
|
cf->data[7] = hecc_read(priv, HECC_CANREC);
|
||||||
}
|
}
|
||||||
|
|
||||||
timestamp = hecc_read(priv, HECC_CANLNT);
|
timestamp = hecc_read(priv, HECC_CANLNT);
|
||||||
can_rx_offload_queue_sorted(&priv->offload, skb, timestamp);
|
err = can_rx_offload_queue_sorted(&priv->offload, skb, timestamp);
|
||||||
|
if (err)
|
||||||
return 0;
|
ndev->stats.rx_fifo_errors++;
|
||||||
}
|
}
|
||||||
|
|
||||||
static irqreturn_t ti_hecc_interrupt(int irq, void *dev_id)
|
static irqreturn_t ti_hecc_interrupt(int irq, void *dev_id)
|
||||||
@ -651,6 +666,7 @@ static irqreturn_t ti_hecc_interrupt(int irq, void *dev_id)
|
|||||||
struct net_device_stats *stats = &ndev->stats;
|
struct net_device_stats *stats = &ndev->stats;
|
||||||
u32 mbxno, mbx_mask, int_status, err_status, stamp;
|
u32 mbxno, mbx_mask, int_status, err_status, stamp;
|
||||||
unsigned long flags, rx_pending;
|
unsigned long flags, rx_pending;
|
||||||
|
u32 handled = 0;
|
||||||
|
|
||||||
int_status = hecc_read(priv,
|
int_status = hecc_read(priv,
|
||||||
priv->use_hecc1int ?
|
priv->use_hecc1int ?
|
||||||
@ -660,17 +676,66 @@ static irqreturn_t ti_hecc_interrupt(int irq, void *dev_id)
|
|||||||
return IRQ_NONE;
|
return IRQ_NONE;
|
||||||
|
|
||||||
err_status = hecc_read(priv, HECC_CANES);
|
err_status = hecc_read(priv, HECC_CANES);
|
||||||
if (err_status & (HECC_BUS_ERROR | HECC_CANES_BO |
|
if (unlikely(err_status & HECC_CANES_FLAGS))
|
||||||
HECC_CANES_EP | HECC_CANES_EW))
|
|
||||||
ti_hecc_error(ndev, int_status, err_status);
|
ti_hecc_error(ndev, int_status, err_status);
|
||||||
|
|
||||||
|
if (unlikely(int_status & HECC_CANGIM_DEF_MASK)) {
|
||||||
|
enum can_state rx_state, tx_state;
|
||||||
|
u32 rec = hecc_read(priv, HECC_CANREC);
|
||||||
|
u32 tec = hecc_read(priv, HECC_CANTEC);
|
||||||
|
|
||||||
|
if (int_status & HECC_CANGIF_WLIF) {
|
||||||
|
handled |= HECC_CANGIF_WLIF;
|
||||||
|
rx_state = rec >= tec ? CAN_STATE_ERROR_WARNING : 0;
|
||||||
|
tx_state = rec <= tec ? CAN_STATE_ERROR_WARNING : 0;
|
||||||
|
netdev_dbg(priv->ndev, "Error Warning interrupt\n");
|
||||||
|
ti_hecc_change_state(ndev, rx_state, tx_state);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (int_status & HECC_CANGIF_EPIF) {
|
||||||
|
handled |= HECC_CANGIF_EPIF;
|
||||||
|
rx_state = rec >= tec ? CAN_STATE_ERROR_PASSIVE : 0;
|
||||||
|
tx_state = rec <= tec ? CAN_STATE_ERROR_PASSIVE : 0;
|
||||||
|
netdev_dbg(priv->ndev, "Error passive interrupt\n");
|
||||||
|
ti_hecc_change_state(ndev, rx_state, tx_state);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (int_status & HECC_CANGIF_BOIF) {
|
||||||
|
handled |= HECC_CANGIF_BOIF;
|
||||||
|
rx_state = CAN_STATE_BUS_OFF;
|
||||||
|
tx_state = CAN_STATE_BUS_OFF;
|
||||||
|
netdev_dbg(priv->ndev, "Bus off interrupt\n");
|
||||||
|
|
||||||
|
/* Disable all interrupts */
|
||||||
|
hecc_write(priv, HECC_CANGIM, 0);
|
||||||
|
can_bus_off(ndev);
|
||||||
|
ti_hecc_change_state(ndev, rx_state, tx_state);
|
||||||
|
}
|
||||||
|
} else if (unlikely(priv->can.state != CAN_STATE_ERROR_ACTIVE)) {
|
||||||
|
enum can_state new_state, tx_state, rx_state;
|
||||||
|
u32 rec = hecc_read(priv, HECC_CANREC);
|
||||||
|
u32 tec = hecc_read(priv, HECC_CANTEC);
|
||||||
|
|
||||||
|
if (rec >= 128 || tec >= 128)
|
||||||
|
new_state = CAN_STATE_ERROR_PASSIVE;
|
||||||
|
else if (rec >= 96 || tec >= 96)
|
||||||
|
new_state = CAN_STATE_ERROR_WARNING;
|
||||||
|
else
|
||||||
|
new_state = CAN_STATE_ERROR_ACTIVE;
|
||||||
|
|
||||||
|
if (new_state < priv->can.state) {
|
||||||
|
rx_state = rec >= tec ? new_state : 0;
|
||||||
|
tx_state = rec <= tec ? new_state : 0;
|
||||||
|
ti_hecc_change_state(ndev, rx_state, tx_state);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
if (int_status & HECC_CANGIF_GMIF) {
|
if (int_status & HECC_CANGIF_GMIF) {
|
||||||
while (priv->tx_tail - priv->tx_head > 0) {
|
while (priv->tx_tail - priv->tx_head > 0) {
|
||||||
mbxno = get_tx_tail_mb(priv);
|
mbxno = get_tx_tail_mb(priv);
|
||||||
mbx_mask = BIT(mbxno);
|
mbx_mask = BIT(mbxno);
|
||||||
if (!(mbx_mask & hecc_read(priv, HECC_CANTA)))
|
if (!(mbx_mask & hecc_read(priv, HECC_CANTA)))
|
||||||
break;
|
break;
|
||||||
hecc_clear_bit(priv, HECC_CANMIM, mbx_mask);
|
|
||||||
hecc_write(priv, HECC_CANTA, mbx_mask);
|
hecc_write(priv, HECC_CANTA, mbx_mask);
|
||||||
spin_lock_irqsave(&priv->mbx_lock, flags);
|
spin_lock_irqsave(&priv->mbx_lock, flags);
|
||||||
hecc_clear_bit(priv, HECC_CANME, mbx_mask);
|
hecc_clear_bit(priv, HECC_CANME, mbx_mask);
|
||||||
@ -695,16 +760,15 @@ static irqreturn_t ti_hecc_interrupt(int irq, void *dev_id)
|
|||||||
while ((rx_pending = hecc_read(priv, HECC_CANRMP))) {
|
while ((rx_pending = hecc_read(priv, HECC_CANRMP))) {
|
||||||
can_rx_offload_irq_offload_timestamp(&priv->offload,
|
can_rx_offload_irq_offload_timestamp(&priv->offload,
|
||||||
rx_pending);
|
rx_pending);
|
||||||
hecc_write(priv, HECC_CANRMP, rx_pending);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/* clear all interrupt conditions - read back to avoid spurious ints */
|
/* clear all interrupt conditions - read back to avoid spurious ints */
|
||||||
if (priv->use_hecc1int) {
|
if (priv->use_hecc1int) {
|
||||||
hecc_write(priv, HECC_CANGIF1, HECC_SET_REG);
|
hecc_write(priv, HECC_CANGIF1, handled);
|
||||||
int_status = hecc_read(priv, HECC_CANGIF1);
|
int_status = hecc_read(priv, HECC_CANGIF1);
|
||||||
} else {
|
} else {
|
||||||
hecc_write(priv, HECC_CANGIF0, HECC_SET_REG);
|
hecc_write(priv, HECC_CANGIF0, handled);
|
||||||
int_status = hecc_read(priv, HECC_CANGIF0);
|
int_status = hecc_read(priv, HECC_CANGIF0);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -877,7 +941,7 @@ static int ti_hecc_probe(struct platform_device *pdev)
|
|||||||
|
|
||||||
priv->offload.mailbox_read = ti_hecc_mailbox_read;
|
priv->offload.mailbox_read = ti_hecc_mailbox_read;
|
||||||
priv->offload.mb_first = HECC_RX_FIRST_MBOX;
|
priv->offload.mb_first = HECC_RX_FIRST_MBOX;
|
||||||
priv->offload.mb_last = HECC_MAX_TX_MBOX;
|
priv->offload.mb_last = HECC_RX_LAST_MBOX;
|
||||||
err = can_rx_offload_add_timestamp(ndev, &priv->offload);
|
err = can_rx_offload_add_timestamp(ndev, &priv->offload);
|
||||||
if (err) {
|
if (err) {
|
||||||
dev_err(&pdev->dev, "can_rx_offload_add_timestamp() failed\n");
|
dev_err(&pdev->dev, "can_rx_offload_add_timestamp() failed\n");
|
||||||
|
@ -623,6 +623,7 @@ static int gs_can_open(struct net_device *netdev)
|
|||||||
rc);
|
rc);
|
||||||
|
|
||||||
usb_unanchor_urb(urb);
|
usb_unanchor_urb(urb);
|
||||||
|
usb_free_urb(urb);
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -876,9 +876,8 @@ static void mcba_usb_disconnect(struct usb_interface *intf)
|
|||||||
netdev_info(priv->netdev, "device disconnected\n");
|
netdev_info(priv->netdev, "device disconnected\n");
|
||||||
|
|
||||||
unregister_candev(priv->netdev);
|
unregister_candev(priv->netdev);
|
||||||
free_candev(priv->netdev);
|
|
||||||
|
|
||||||
mcba_urb_unlink(priv);
|
mcba_urb_unlink(priv);
|
||||||
|
free_candev(priv->netdev);
|
||||||
}
|
}
|
||||||
|
|
||||||
static struct usb_driver mcba_usb_driver = {
|
static struct usb_driver mcba_usb_driver = {
|
||||||
|
@ -100,7 +100,7 @@ struct pcan_usb_msg_context {
|
|||||||
u8 *end;
|
u8 *end;
|
||||||
u8 rec_cnt;
|
u8 rec_cnt;
|
||||||
u8 rec_idx;
|
u8 rec_idx;
|
||||||
u8 rec_data_idx;
|
u8 rec_ts_idx;
|
||||||
struct net_device *netdev;
|
struct net_device *netdev;
|
||||||
struct pcan_usb *pdev;
|
struct pcan_usb *pdev;
|
||||||
};
|
};
|
||||||
@ -436,8 +436,8 @@ static int pcan_usb_decode_error(struct pcan_usb_msg_context *mc, u8 n,
|
|||||||
}
|
}
|
||||||
if ((n & PCAN_USB_ERROR_BUS_LIGHT) == 0) {
|
if ((n & PCAN_USB_ERROR_BUS_LIGHT) == 0) {
|
||||||
/* no error (back to active state) */
|
/* no error (back to active state) */
|
||||||
mc->pdev->dev.can.state = CAN_STATE_ERROR_ACTIVE;
|
new_state = CAN_STATE_ERROR_ACTIVE;
|
||||||
return 0;
|
break;
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
|
|
||||||
@ -460,9 +460,9 @@ static int pcan_usb_decode_error(struct pcan_usb_msg_context *mc, u8 n,
|
|||||||
}
|
}
|
||||||
|
|
||||||
if ((n & PCAN_USB_ERROR_BUS_HEAVY) == 0) {
|
if ((n & PCAN_USB_ERROR_BUS_HEAVY) == 0) {
|
||||||
/* no error (back to active state) */
|
/* no error (back to warning state) */
|
||||||
mc->pdev->dev.can.state = CAN_STATE_ERROR_ACTIVE;
|
new_state = CAN_STATE_ERROR_WARNING;
|
||||||
return 0;
|
break;
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
|
|
||||||
@ -501,6 +501,11 @@ static int pcan_usb_decode_error(struct pcan_usb_msg_context *mc, u8 n,
|
|||||||
mc->pdev->dev.can.can_stats.error_warning++;
|
mc->pdev->dev.can.can_stats.error_warning++;
|
||||||
break;
|
break;
|
||||||
|
|
||||||
|
case CAN_STATE_ERROR_ACTIVE:
|
||||||
|
cf->can_id |= CAN_ERR_CRTL;
|
||||||
|
cf->data[1] = CAN_ERR_CRTL_ACTIVE;
|
||||||
|
break;
|
||||||
|
|
||||||
default:
|
default:
|
||||||
/* CAN_STATE_MAX (trick to handle other errors) */
|
/* CAN_STATE_MAX (trick to handle other errors) */
|
||||||
cf->can_id |= CAN_ERR_CRTL;
|
cf->can_id |= CAN_ERR_CRTL;
|
||||||
@ -547,10 +552,15 @@ static int pcan_usb_decode_status(struct pcan_usb_msg_context *mc,
|
|||||||
mc->ptr += PCAN_USB_CMD_ARGS;
|
mc->ptr += PCAN_USB_CMD_ARGS;
|
||||||
|
|
||||||
if (status_len & PCAN_USB_STATUSLEN_TIMESTAMP) {
|
if (status_len & PCAN_USB_STATUSLEN_TIMESTAMP) {
|
||||||
int err = pcan_usb_decode_ts(mc, !mc->rec_idx);
|
int err = pcan_usb_decode_ts(mc, !mc->rec_ts_idx);
|
||||||
|
|
||||||
if (err)
|
if (err)
|
||||||
return err;
|
return err;
|
||||||
|
|
||||||
|
/* Next packet in the buffer will have a timestamp on a single
|
||||||
|
* byte
|
||||||
|
*/
|
||||||
|
mc->rec_ts_idx++;
|
||||||
}
|
}
|
||||||
|
|
||||||
switch (f) {
|
switch (f) {
|
||||||
@ -632,10 +642,13 @@ static int pcan_usb_decode_data(struct pcan_usb_msg_context *mc, u8 status_len)
|
|||||||
|
|
||||||
cf->can_dlc = get_can_dlc(rec_len);
|
cf->can_dlc = get_can_dlc(rec_len);
|
||||||
|
|
||||||
/* first data packet timestamp is a word */
|
/* Only first packet timestamp is a word */
|
||||||
if (pcan_usb_decode_ts(mc, !mc->rec_data_idx))
|
if (pcan_usb_decode_ts(mc, !mc->rec_ts_idx))
|
||||||
goto decode_failed;
|
goto decode_failed;
|
||||||
|
|
||||||
|
/* Next packet in the buffer will have a timestamp on a single byte */
|
||||||
|
mc->rec_ts_idx++;
|
||||||
|
|
||||||
/* read data */
|
/* read data */
|
||||||
memset(cf->data, 0x0, sizeof(cf->data));
|
memset(cf->data, 0x0, sizeof(cf->data));
|
||||||
if (status_len & PCAN_USB_STATUSLEN_RTR) {
|
if (status_len & PCAN_USB_STATUSLEN_RTR) {
|
||||||
@ -688,7 +701,6 @@ static int pcan_usb_decode_msg(struct peak_usb_device *dev, u8 *ibuf, u32 lbuf)
|
|||||||
/* handle normal can frames here */
|
/* handle normal can frames here */
|
||||||
} else {
|
} else {
|
||||||
err = pcan_usb_decode_data(&mc, sl);
|
err = pcan_usb_decode_data(&mc, sl);
|
||||||
mc.rec_data_idx++;
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -750,7 +750,7 @@ static int peak_usb_create_dev(const struct peak_usb_adapter *peak_usb_adapter,
|
|||||||
dev = netdev_priv(netdev);
|
dev = netdev_priv(netdev);
|
||||||
|
|
||||||
/* allocate a buffer large enough to send commands */
|
/* allocate a buffer large enough to send commands */
|
||||||
dev->cmd_buf = kmalloc(PCAN_USB_MAX_CMD_LEN, GFP_KERNEL);
|
dev->cmd_buf = kzalloc(PCAN_USB_MAX_CMD_LEN, GFP_KERNEL);
|
||||||
if (!dev->cmd_buf) {
|
if (!dev->cmd_buf) {
|
||||||
err = -ENOMEM;
|
err = -ENOMEM;
|
||||||
goto lbl_free_candev;
|
goto lbl_free_candev;
|
||||||
|
@ -996,9 +996,8 @@ static void usb_8dev_disconnect(struct usb_interface *intf)
|
|||||||
netdev_info(priv->netdev, "device disconnected\n");
|
netdev_info(priv->netdev, "device disconnected\n");
|
||||||
|
|
||||||
unregister_netdev(priv->netdev);
|
unregister_netdev(priv->netdev);
|
||||||
free_candev(priv->netdev);
|
|
||||||
|
|
||||||
unlink_all_urbs(priv);
|
unlink_all_urbs(priv);
|
||||||
|
free_candev(priv->netdev);
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
@ -1599,7 +1599,6 @@ static const struct xcan_devtype_data xcan_zynq_data = {
|
|||||||
|
|
||||||
static const struct xcan_devtype_data xcan_axi_data = {
|
static const struct xcan_devtype_data xcan_axi_data = {
|
||||||
.cantype = XAXI_CAN,
|
.cantype = XAXI_CAN,
|
||||||
.flags = XCAN_FLAG_TXFEMP,
|
|
||||||
.bittiming_const = &xcan_bittiming_const,
|
.bittiming_const = &xcan_bittiming_const,
|
||||||
.btr_ts2_shift = XCAN_BTR_TS2_SHIFT,
|
.btr_ts2_shift = XCAN_BTR_TS2_SHIFT,
|
||||||
.btr_sjw_shift = XCAN_BTR_SJW_SHIFT,
|
.btr_sjw_shift = XCAN_BTR_SJW_SHIFT,
|
||||||
|
@ -1215,10 +1215,10 @@ static int bcm_sf2_sw_remove(struct platform_device *pdev)
|
|||||||
struct bcm_sf2_priv *priv = platform_get_drvdata(pdev);
|
struct bcm_sf2_priv *priv = platform_get_drvdata(pdev);
|
||||||
|
|
||||||
priv->wol_ports_mask = 0;
|
priv->wol_ports_mask = 0;
|
||||||
|
/* Disable interrupts */
|
||||||
|
bcm_sf2_intr_disable(priv);
|
||||||
dsa_unregister_switch(priv->dev->ds);
|
dsa_unregister_switch(priv->dev->ds);
|
||||||
bcm_sf2_cfp_exit(priv->dev->ds);
|
bcm_sf2_cfp_exit(priv->dev->ds);
|
||||||
/* Disable all ports and interrupts */
|
|
||||||
bcm_sf2_sw_suspend(priv->dev->ds);
|
|
||||||
bcm_sf2_mdio_unregister(priv);
|
bcm_sf2_mdio_unregister(priv);
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
|
@ -1996,8 +1996,6 @@ static void reset_umac(struct bcmgenet_priv *priv)
|
|||||||
|
|
||||||
/* issue soft reset with (rg)mii loopback to ensure a stable rxclk */
|
/* issue soft reset with (rg)mii loopback to ensure a stable rxclk */
|
||||||
bcmgenet_umac_writel(priv, CMD_SW_RESET | CMD_LCL_LOOP_EN, UMAC_CMD);
|
bcmgenet_umac_writel(priv, CMD_SW_RESET | CMD_LCL_LOOP_EN, UMAC_CMD);
|
||||||
udelay(2);
|
|
||||||
bcmgenet_umac_writel(priv, 0, UMAC_CMD);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
static void bcmgenet_intr_disable(struct bcmgenet_priv *priv)
|
static void bcmgenet_intr_disable(struct bcmgenet_priv *priv)
|
||||||
@ -2614,8 +2612,10 @@ static void bcmgenet_irq_task(struct work_struct *work)
|
|||||||
spin_unlock_irq(&priv->lock);
|
spin_unlock_irq(&priv->lock);
|
||||||
|
|
||||||
if (status & UMAC_IRQ_PHY_DET_R &&
|
if (status & UMAC_IRQ_PHY_DET_R &&
|
||||||
priv->dev->phydev->autoneg != AUTONEG_ENABLE)
|
priv->dev->phydev->autoneg != AUTONEG_ENABLE) {
|
||||||
phy_init_hw(priv->dev->phydev);
|
phy_init_hw(priv->dev->phydev);
|
||||||
|
genphy_config_aneg(priv->dev->phydev);
|
||||||
|
}
|
||||||
|
|
||||||
/* Link UP/DOWN event */
|
/* Link UP/DOWN event */
|
||||||
if (status & UMAC_IRQ_LINK_EVENT)
|
if (status & UMAC_IRQ_LINK_EVENT)
|
||||||
@ -2879,12 +2879,6 @@ static int bcmgenet_open(struct net_device *dev)
|
|||||||
if (priv->internal_phy)
|
if (priv->internal_phy)
|
||||||
bcmgenet_power_up(priv, GENET_POWER_PASSIVE);
|
bcmgenet_power_up(priv, GENET_POWER_PASSIVE);
|
||||||
|
|
||||||
ret = bcmgenet_mii_connect(dev);
|
|
||||||
if (ret) {
|
|
||||||
netdev_err(dev, "failed to connect to PHY\n");
|
|
||||||
goto err_clk_disable;
|
|
||||||
}
|
|
||||||
|
|
||||||
/* take MAC out of reset */
|
/* take MAC out of reset */
|
||||||
bcmgenet_umac_reset(priv);
|
bcmgenet_umac_reset(priv);
|
||||||
|
|
||||||
@ -2894,12 +2888,6 @@ static int bcmgenet_open(struct net_device *dev)
|
|||||||
reg = bcmgenet_umac_readl(priv, UMAC_CMD);
|
reg = bcmgenet_umac_readl(priv, UMAC_CMD);
|
||||||
priv->crc_fwd_en = !!(reg & CMD_CRC_FWD);
|
priv->crc_fwd_en = !!(reg & CMD_CRC_FWD);
|
||||||
|
|
||||||
ret = bcmgenet_mii_config(dev, true);
|
|
||||||
if (ret) {
|
|
||||||
netdev_err(dev, "unsupported PHY\n");
|
|
||||||
goto err_disconnect_phy;
|
|
||||||
}
|
|
||||||
|
|
||||||
bcmgenet_set_hw_addr(priv, dev->dev_addr);
|
bcmgenet_set_hw_addr(priv, dev->dev_addr);
|
||||||
|
|
||||||
if (priv->internal_phy) {
|
if (priv->internal_phy) {
|
||||||
@ -2915,7 +2903,7 @@ static int bcmgenet_open(struct net_device *dev)
|
|||||||
ret = bcmgenet_init_dma(priv);
|
ret = bcmgenet_init_dma(priv);
|
||||||
if (ret) {
|
if (ret) {
|
||||||
netdev_err(dev, "failed to initialize DMA\n");
|
netdev_err(dev, "failed to initialize DMA\n");
|
||||||
goto err_disconnect_phy;
|
goto err_clk_disable;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Always enable ring 16 - descriptor ring */
|
/* Always enable ring 16 - descriptor ring */
|
||||||
@ -2938,19 +2926,25 @@ static int bcmgenet_open(struct net_device *dev)
|
|||||||
goto err_irq0;
|
goto err_irq0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
ret = bcmgenet_mii_probe(dev);
|
||||||
|
if (ret) {
|
||||||
|
netdev_err(dev, "failed to connect to PHY\n");
|
||||||
|
goto err_irq1;
|
||||||
|
}
|
||||||
|
|
||||||
bcmgenet_netif_start(dev);
|
bcmgenet_netif_start(dev);
|
||||||
|
|
||||||
netif_tx_start_all_queues(dev);
|
netif_tx_start_all_queues(dev);
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
|
|
||||||
|
err_irq1:
|
||||||
|
free_irq(priv->irq1, priv);
|
||||||
err_irq0:
|
err_irq0:
|
||||||
free_irq(priv->irq0, priv);
|
free_irq(priv->irq0, priv);
|
||||||
err_fini_dma:
|
err_fini_dma:
|
||||||
bcmgenet_dma_teardown(priv);
|
bcmgenet_dma_teardown(priv);
|
||||||
bcmgenet_fini_dma(priv);
|
bcmgenet_fini_dma(priv);
|
||||||
err_disconnect_phy:
|
|
||||||
phy_disconnect(dev->phydev);
|
|
||||||
err_clk_disable:
|
err_clk_disable:
|
||||||
if (priv->internal_phy)
|
if (priv->internal_phy)
|
||||||
bcmgenet_power_down(priv, GENET_POWER_PASSIVE);
|
bcmgenet_power_down(priv, GENET_POWER_PASSIVE);
|
||||||
@ -3631,8 +3625,6 @@ static int bcmgenet_resume(struct device *d)
|
|||||||
if (priv->internal_phy)
|
if (priv->internal_phy)
|
||||||
bcmgenet_power_up(priv, GENET_POWER_PASSIVE);
|
bcmgenet_power_up(priv, GENET_POWER_PASSIVE);
|
||||||
|
|
||||||
phy_init_hw(dev->phydev);
|
|
||||||
|
|
||||||
bcmgenet_umac_reset(priv);
|
bcmgenet_umac_reset(priv);
|
||||||
|
|
||||||
init_umac(priv);
|
init_umac(priv);
|
||||||
@ -3641,7 +3633,10 @@ static int bcmgenet_resume(struct device *d)
|
|||||||
if (priv->wolopts)
|
if (priv->wolopts)
|
||||||
clk_disable_unprepare(priv->clk_wol);
|
clk_disable_unprepare(priv->clk_wol);
|
||||||
|
|
||||||
|
phy_init_hw(dev->phydev);
|
||||||
|
|
||||||
/* Speed settings must be restored */
|
/* Speed settings must be restored */
|
||||||
|
genphy_config_aneg(dev->phydev);
|
||||||
bcmgenet_mii_config(priv->dev, false);
|
bcmgenet_mii_config(priv->dev, false);
|
||||||
|
|
||||||
bcmgenet_set_hw_addr(priv, dev->dev_addr);
|
bcmgenet_set_hw_addr(priv, dev->dev_addr);
|
||||||
|
@ -720,8 +720,8 @@ GENET_IO_MACRO(rbuf, GENET_RBUF_OFF);
|
|||||||
|
|
||||||
/* MDIO routines */
|
/* MDIO routines */
|
||||||
int bcmgenet_mii_init(struct net_device *dev);
|
int bcmgenet_mii_init(struct net_device *dev);
|
||||||
int bcmgenet_mii_connect(struct net_device *dev);
|
|
||||||
int bcmgenet_mii_config(struct net_device *dev, bool init);
|
int bcmgenet_mii_config(struct net_device *dev, bool init);
|
||||||
|
int bcmgenet_mii_probe(struct net_device *dev);
|
||||||
void bcmgenet_mii_exit(struct net_device *dev);
|
void bcmgenet_mii_exit(struct net_device *dev);
|
||||||
void bcmgenet_phy_power_set(struct net_device *dev, bool enable);
|
void bcmgenet_phy_power_set(struct net_device *dev, bool enable);
|
||||||
void bcmgenet_mii_setup(struct net_device *dev);
|
void bcmgenet_mii_setup(struct net_device *dev);
|
||||||
|
@ -173,46 +173,6 @@ static void bcmgenet_moca_phy_setup(struct bcmgenet_priv *priv)
|
|||||||
bcmgenet_fixed_phy_link_update);
|
bcmgenet_fixed_phy_link_update);
|
||||||
}
|
}
|
||||||
|
|
||||||
int bcmgenet_mii_connect(struct net_device *dev)
|
|
||||||
{
|
|
||||||
struct bcmgenet_priv *priv = netdev_priv(dev);
|
|
||||||
struct device_node *dn = priv->pdev->dev.of_node;
|
|
||||||
struct phy_device *phydev;
|
|
||||||
u32 phy_flags = 0;
|
|
||||||
int ret;
|
|
||||||
|
|
||||||
/* Communicate the integrated PHY revision */
|
|
||||||
if (priv->internal_phy)
|
|
||||||
phy_flags = priv->gphy_rev;
|
|
||||||
|
|
||||||
/* Initialize link state variables that bcmgenet_mii_setup() uses */
|
|
||||||
priv->old_link = -1;
|
|
||||||
priv->old_speed = -1;
|
|
||||||
priv->old_duplex = -1;
|
|
||||||
priv->old_pause = -1;
|
|
||||||
|
|
||||||
if (dn) {
|
|
||||||
phydev = of_phy_connect(dev, priv->phy_dn, bcmgenet_mii_setup,
|
|
||||||
phy_flags, priv->phy_interface);
|
|
||||||
if (!phydev) {
|
|
||||||
pr_err("could not attach to PHY\n");
|
|
||||||
return -ENODEV;
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
phydev = dev->phydev;
|
|
||||||
phydev->dev_flags = phy_flags;
|
|
||||||
|
|
||||||
ret = phy_connect_direct(dev, phydev, bcmgenet_mii_setup,
|
|
||||||
priv->phy_interface);
|
|
||||||
if (ret) {
|
|
||||||
pr_err("could not attach to PHY\n");
|
|
||||||
return -ENODEV;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
int bcmgenet_mii_config(struct net_device *dev, bool init)
|
int bcmgenet_mii_config(struct net_device *dev, bool init)
|
||||||
{
|
{
|
||||||
struct bcmgenet_priv *priv = netdev_priv(dev);
|
struct bcmgenet_priv *priv = netdev_priv(dev);
|
||||||
@ -221,8 +181,38 @@ int bcmgenet_mii_config(struct net_device *dev, bool init)
|
|||||||
const char *phy_name = NULL;
|
const char *phy_name = NULL;
|
||||||
u32 id_mode_dis = 0;
|
u32 id_mode_dis = 0;
|
||||||
u32 port_ctrl;
|
u32 port_ctrl;
|
||||||
|
int bmcr = -1;
|
||||||
|
int ret;
|
||||||
u32 reg;
|
u32 reg;
|
||||||
|
|
||||||
|
/* MAC clocking workaround during reset of umac state machines */
|
||||||
|
reg = bcmgenet_umac_readl(priv, UMAC_CMD);
|
||||||
|
if (reg & CMD_SW_RESET) {
|
||||||
|
/* An MII PHY must be isolated to prevent TXC contention */
|
||||||
|
if (priv->phy_interface == PHY_INTERFACE_MODE_MII) {
|
||||||
|
ret = phy_read(phydev, MII_BMCR);
|
||||||
|
if (ret >= 0) {
|
||||||
|
bmcr = ret;
|
||||||
|
ret = phy_write(phydev, MII_BMCR,
|
||||||
|
bmcr | BMCR_ISOLATE);
|
||||||
|
}
|
||||||
|
if (ret) {
|
||||||
|
netdev_err(dev, "failed to isolate PHY\n");
|
||||||
|
return ret;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
/* Switch MAC clocking to RGMII generated clock */
|
||||||
|
bcmgenet_sys_writel(priv, PORT_MODE_EXT_GPHY, SYS_PORT_CTRL);
|
||||||
|
/* Ensure 5 clks with Rx disabled
|
||||||
|
* followed by 5 clks with Reset asserted
|
||||||
|
*/
|
||||||
|
udelay(4);
|
||||||
|
reg &= ~(CMD_SW_RESET | CMD_LCL_LOOP_EN);
|
||||||
|
bcmgenet_umac_writel(priv, reg, UMAC_CMD);
|
||||||
|
/* Ensure 5 more clocks before Rx is enabled */
|
||||||
|
udelay(2);
|
||||||
|
}
|
||||||
|
|
||||||
priv->ext_phy = !priv->internal_phy &&
|
priv->ext_phy = !priv->internal_phy &&
|
||||||
(priv->phy_interface != PHY_INTERFACE_MODE_MOCA);
|
(priv->phy_interface != PHY_INTERFACE_MODE_MOCA);
|
||||||
|
|
||||||
@ -254,6 +244,9 @@ int bcmgenet_mii_config(struct net_device *dev, bool init)
|
|||||||
phy_set_max_speed(phydev, SPEED_100);
|
phy_set_max_speed(phydev, SPEED_100);
|
||||||
bcmgenet_sys_writel(priv,
|
bcmgenet_sys_writel(priv,
|
||||||
PORT_MODE_EXT_EPHY, SYS_PORT_CTRL);
|
PORT_MODE_EXT_EPHY, SYS_PORT_CTRL);
|
||||||
|
/* Restore the MII PHY after isolation */
|
||||||
|
if (bmcr >= 0)
|
||||||
|
phy_write(phydev, MII_BMCR, bmcr);
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case PHY_INTERFACE_MODE_REVMII:
|
case PHY_INTERFACE_MODE_REVMII:
|
||||||
@ -306,21 +299,71 @@ int bcmgenet_mii_config(struct net_device *dev, bool init)
|
|||||||
bcmgenet_ext_writel(priv, reg, EXT_RGMII_OOB_CTRL);
|
bcmgenet_ext_writel(priv, reg, EXT_RGMII_OOB_CTRL);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (init) {
|
if (init)
|
||||||
linkmode_copy(phydev->advertising, phydev->supported);
|
|
||||||
|
|
||||||
/* The internal PHY has its link interrupts routed to the
|
|
||||||
* Ethernet MAC ISRs. On GENETv5 there is a hardware issue
|
|
||||||
* that prevents the signaling of link UP interrupts when
|
|
||||||
* the link operates at 10Mbps, so fallback to polling for
|
|
||||||
* those versions of GENET.
|
|
||||||
*/
|
|
||||||
if (priv->internal_phy && !GENET_IS_V5(priv))
|
|
||||||
phydev->irq = PHY_IGNORE_INTERRUPT;
|
|
||||||
|
|
||||||
dev_info(kdev, "configuring instance for %s\n", phy_name);
|
dev_info(kdev, "configuring instance for %s\n", phy_name);
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
int bcmgenet_mii_probe(struct net_device *dev)
|
||||||
|
{
|
||||||
|
struct bcmgenet_priv *priv = netdev_priv(dev);
|
||||||
|
struct device_node *dn = priv->pdev->dev.of_node;
|
||||||
|
struct phy_device *phydev;
|
||||||
|
u32 phy_flags = 0;
|
||||||
|
int ret;
|
||||||
|
|
||||||
|
/* Communicate the integrated PHY revision */
|
||||||
|
if (priv->internal_phy)
|
||||||
|
phy_flags = priv->gphy_rev;
|
||||||
|
|
||||||
|
/* Initialize link state variables that bcmgenet_mii_setup() uses */
|
||||||
|
priv->old_link = -1;
|
||||||
|
priv->old_speed = -1;
|
||||||
|
priv->old_duplex = -1;
|
||||||
|
priv->old_pause = -1;
|
||||||
|
|
||||||
|
if (dn) {
|
||||||
|
phydev = of_phy_connect(dev, priv->phy_dn, bcmgenet_mii_setup,
|
||||||
|
phy_flags, priv->phy_interface);
|
||||||
|
if (!phydev) {
|
||||||
|
pr_err("could not attach to PHY\n");
|
||||||
|
return -ENODEV;
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
phydev = dev->phydev;
|
||||||
|
phydev->dev_flags = phy_flags;
|
||||||
|
|
||||||
|
ret = phy_connect_direct(dev, phydev, bcmgenet_mii_setup,
|
||||||
|
priv->phy_interface);
|
||||||
|
if (ret) {
|
||||||
|
pr_err("could not attach to PHY\n");
|
||||||
|
return -ENODEV;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/* Configure port multiplexer based on what the probed PHY device since
|
||||||
|
* reading the 'max-speed' property determines the maximum supported
|
||||||
|
* PHY speed which is needed for bcmgenet_mii_config() to configure
|
||||||
|
* things appropriately.
|
||||||
|
*/
|
||||||
|
ret = bcmgenet_mii_config(dev, true);
|
||||||
|
if (ret) {
|
||||||
|
phy_disconnect(dev->phydev);
|
||||||
|
return ret;
|
||||||
|
}
|
||||||
|
|
||||||
|
linkmode_copy(phydev->advertising, phydev->supported);
|
||||||
|
|
||||||
|
/* The internal PHY has its link interrupts routed to the
|
||||||
|
* Ethernet MAC ISRs. On GENETv5 there is a hardware issue
|
||||||
|
* that prevents the signaling of link UP interrupts when
|
||||||
|
* the link operates at 10Mbps, so fallback to polling for
|
||||||
|
* those versions of GENET.
|
||||||
|
*/
|
||||||
|
if (priv->internal_phy && !GENET_IS_V5(priv))
|
||||||
|
dev->phydev->irq = PHY_IGNORE_INTERRUPT;
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1499,7 +1499,7 @@ static int octeon_mgmt_probe(struct platform_device *pdev)
|
|||||||
netdev->ethtool_ops = &octeon_mgmt_ethtool_ops;
|
netdev->ethtool_ops = &octeon_mgmt_ethtool_ops;
|
||||||
|
|
||||||
netdev->min_mtu = 64 - OCTEON_MGMT_RX_HEADROOM;
|
netdev->min_mtu = 64 - OCTEON_MGMT_RX_HEADROOM;
|
||||||
netdev->max_mtu = 16383 - OCTEON_MGMT_RX_HEADROOM;
|
netdev->max_mtu = 16383 - OCTEON_MGMT_RX_HEADROOM - VLAN_HLEN;
|
||||||
|
|
||||||
mac = of_get_mac_address(pdev->dev.of_node);
|
mac = of_get_mac_address(pdev->dev.of_node);
|
||||||
|
|
||||||
|
@ -3645,6 +3645,8 @@ fec_drv_remove(struct platform_device *pdev)
|
|||||||
regulator_disable(fep->reg_phy);
|
regulator_disable(fep->reg_phy);
|
||||||
pm_runtime_put(&pdev->dev);
|
pm_runtime_put(&pdev->dev);
|
||||||
pm_runtime_disable(&pdev->dev);
|
pm_runtime_disable(&pdev->dev);
|
||||||
|
clk_disable_unprepare(fep->clk_ahb);
|
||||||
|
clk_disable_unprepare(fep->clk_ipg);
|
||||||
if (of_phy_is_fixed_link(np))
|
if (of_phy_is_fixed_link(np))
|
||||||
of_phy_deregister_fixed_link(np);
|
of_phy_deregister_fixed_link(np);
|
||||||
of_node_put(fep->phy_node);
|
of_node_put(fep->phy_node);
|
||||||
|
@ -199,7 +199,6 @@ hnae_init_ring(struct hnae_queue *q, struct hnae_ring *ring, int flags)
|
|||||||
|
|
||||||
ring->q = q;
|
ring->q = q;
|
||||||
ring->flags = flags;
|
ring->flags = flags;
|
||||||
spin_lock_init(&ring->lock);
|
|
||||||
ring->coal_param = q->handle->coal_param;
|
ring->coal_param = q->handle->coal_param;
|
||||||
assert(!ring->desc && !ring->desc_cb && !ring->desc_dma_addr);
|
assert(!ring->desc && !ring->desc_cb && !ring->desc_dma_addr);
|
||||||
|
|
||||||
|
@ -274,9 +274,6 @@ struct hnae_ring {
|
|||||||
/* statistic */
|
/* statistic */
|
||||||
struct ring_stats stats;
|
struct ring_stats stats;
|
||||||
|
|
||||||
/* ring lock for poll one */
|
|
||||||
spinlock_t lock;
|
|
||||||
|
|
||||||
dma_addr_t desc_dma_addr;
|
dma_addr_t desc_dma_addr;
|
||||||
u32 buf_size; /* size for hnae_desc->addr, preset by AE */
|
u32 buf_size; /* size for hnae_desc->addr, preset by AE */
|
||||||
u16 desc_num; /* total number of desc */
|
u16 desc_num; /* total number of desc */
|
||||||
|
@ -943,15 +943,6 @@ static int is_valid_clean_head(struct hnae_ring *ring, int h)
|
|||||||
return u > c ? (h > c && h <= u) : (h > c || h <= u);
|
return u > c ? (h > c && h <= u) : (h > c || h <= u);
|
||||||
}
|
}
|
||||||
|
|
||||||
/* netif_tx_lock will turn down the performance, set only when necessary */
|
|
||||||
#ifdef CONFIG_NET_POLL_CONTROLLER
|
|
||||||
#define NETIF_TX_LOCK(ring) spin_lock(&(ring)->lock)
|
|
||||||
#define NETIF_TX_UNLOCK(ring) spin_unlock(&(ring)->lock)
|
|
||||||
#else
|
|
||||||
#define NETIF_TX_LOCK(ring)
|
|
||||||
#define NETIF_TX_UNLOCK(ring)
|
|
||||||
#endif
|
|
||||||
|
|
||||||
/* reclaim all desc in one budget
|
/* reclaim all desc in one budget
|
||||||
* return error or number of desc left
|
* return error or number of desc left
|
||||||
*/
|
*/
|
||||||
@ -965,21 +956,16 @@ static int hns_nic_tx_poll_one(struct hns_nic_ring_data *ring_data,
|
|||||||
int head;
|
int head;
|
||||||
int bytes, pkts;
|
int bytes, pkts;
|
||||||
|
|
||||||
NETIF_TX_LOCK(ring);
|
|
||||||
|
|
||||||
head = readl_relaxed(ring->io_base + RCB_REG_HEAD);
|
head = readl_relaxed(ring->io_base + RCB_REG_HEAD);
|
||||||
rmb(); /* make sure head is ready before touch any data */
|
rmb(); /* make sure head is ready before touch any data */
|
||||||
|
|
||||||
if (is_ring_empty(ring) || head == ring->next_to_clean) {
|
if (is_ring_empty(ring) || head == ring->next_to_clean)
|
||||||
NETIF_TX_UNLOCK(ring);
|
|
||||||
return 0; /* no data to poll */
|
return 0; /* no data to poll */
|
||||||
}
|
|
||||||
|
|
||||||
if (!is_valid_clean_head(ring, head)) {
|
if (!is_valid_clean_head(ring, head)) {
|
||||||
netdev_err(ndev, "wrong head (%d, %d-%d)\n", head,
|
netdev_err(ndev, "wrong head (%d, %d-%d)\n", head,
|
||||||
ring->next_to_use, ring->next_to_clean);
|
ring->next_to_use, ring->next_to_clean);
|
||||||
ring->stats.io_err_cnt++;
|
ring->stats.io_err_cnt++;
|
||||||
NETIF_TX_UNLOCK(ring);
|
|
||||||
return -EIO;
|
return -EIO;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -994,8 +980,6 @@ static int hns_nic_tx_poll_one(struct hns_nic_ring_data *ring_data,
|
|||||||
ring->stats.tx_pkts += pkts;
|
ring->stats.tx_pkts += pkts;
|
||||||
ring->stats.tx_bytes += bytes;
|
ring->stats.tx_bytes += bytes;
|
||||||
|
|
||||||
NETIF_TX_UNLOCK(ring);
|
|
||||||
|
|
||||||
dev_queue = netdev_get_tx_queue(ndev, ring_data->queue_index);
|
dev_queue = netdev_get_tx_queue(ndev, ring_data->queue_index);
|
||||||
netdev_tx_completed_queue(dev_queue, pkts, bytes);
|
netdev_tx_completed_queue(dev_queue, pkts, bytes);
|
||||||
|
|
||||||
@ -1055,16 +1039,12 @@ static void hns_nic_tx_clr_all_bufs(struct hns_nic_ring_data *ring_data)
|
|||||||
int head;
|
int head;
|
||||||
int bytes, pkts;
|
int bytes, pkts;
|
||||||
|
|
||||||
NETIF_TX_LOCK(ring);
|
|
||||||
|
|
||||||
head = ring->next_to_use; /* ntu :soft setted ring position*/
|
head = ring->next_to_use; /* ntu :soft setted ring position*/
|
||||||
bytes = 0;
|
bytes = 0;
|
||||||
pkts = 0;
|
pkts = 0;
|
||||||
while (head != ring->next_to_clean)
|
while (head != ring->next_to_clean)
|
||||||
hns_nic_reclaim_one_desc(ring, &bytes, &pkts);
|
hns_nic_reclaim_one_desc(ring, &bytes, &pkts);
|
||||||
|
|
||||||
NETIF_TX_UNLOCK(ring);
|
|
||||||
|
|
||||||
dev_queue = netdev_get_tx_queue(ndev, ring_data->queue_index);
|
dev_queue = netdev_get_tx_queue(ndev, ring_data->queue_index);
|
||||||
netdev_tx_reset_queue(dev_queue);
|
netdev_tx_reset_queue(dev_queue);
|
||||||
}
|
}
|
||||||
|
@ -1,4 +1,4 @@
|
|||||||
// SPDX-License-Identifier: GPL-2.0+
|
/* SPDX-License-Identifier: GPL-2.0+ */
|
||||||
// Copyright (c) 2016-2017 Hisilicon Limited.
|
// Copyright (c) 2016-2017 Hisilicon Limited.
|
||||||
|
|
||||||
#ifndef __HNAE3_H
|
#ifndef __HNAE3_H
|
||||||
|
@ -1,4 +1,4 @@
|
|||||||
// SPDX-License-Identifier: GPL-2.0+
|
/* SPDX-License-Identifier: GPL-2.0+ */
|
||||||
// Copyright (c) 2016-2017 Hisilicon Limited.
|
// Copyright (c) 2016-2017 Hisilicon Limited.
|
||||||
|
|
||||||
#ifndef __HNS3_ENET_H
|
#ifndef __HNS3_ENET_H
|
||||||
|
@ -1,4 +1,4 @@
|
|||||||
// SPDX-License-Identifier: GPL-2.0+
|
/* SPDX-License-Identifier: GPL-2.0+ */
|
||||||
// Copyright (c) 2016-2017 Hisilicon Limited.
|
// Copyright (c) 2016-2017 Hisilicon Limited.
|
||||||
|
|
||||||
#ifndef __HCLGE_CMD_H
|
#ifndef __HCLGE_CMD_H
|
||||||
|
@ -1,4 +1,4 @@
|
|||||||
// SPDX-License-Identifier: GPL-2.0+
|
/* SPDX-License-Identifier: GPL-2.0+ */
|
||||||
// Copyright (c) 2016-2017 Hisilicon Limited.
|
// Copyright (c) 2016-2017 Hisilicon Limited.
|
||||||
|
|
||||||
#ifndef __HCLGE_DCB_H__
|
#ifndef __HCLGE_DCB_H__
|
||||||
|
@ -3587,12 +3587,28 @@ static int hclge_set_rst_done(struct hclge_dev *hdev)
|
|||||||
{
|
{
|
||||||
struct hclge_pf_rst_done_cmd *req;
|
struct hclge_pf_rst_done_cmd *req;
|
||||||
struct hclge_desc desc;
|
struct hclge_desc desc;
|
||||||
|
int ret;
|
||||||
|
|
||||||
req = (struct hclge_pf_rst_done_cmd *)desc.data;
|
req = (struct hclge_pf_rst_done_cmd *)desc.data;
|
||||||
hclge_cmd_setup_basic_desc(&desc, HCLGE_OPC_PF_RST_DONE, false);
|
hclge_cmd_setup_basic_desc(&desc, HCLGE_OPC_PF_RST_DONE, false);
|
||||||
req->pf_rst_done |= HCLGE_PF_RESET_DONE_BIT;
|
req->pf_rst_done |= HCLGE_PF_RESET_DONE_BIT;
|
||||||
|
|
||||||
return hclge_cmd_send(&hdev->hw, &desc, 1);
|
ret = hclge_cmd_send(&hdev->hw, &desc, 1);
|
||||||
|
/* To be compatible with the old firmware, which does not support
|
||||||
|
* command HCLGE_OPC_PF_RST_DONE, just print a warning and
|
||||||
|
* return success
|
||||||
|
*/
|
||||||
|
if (ret == -EOPNOTSUPP) {
|
||||||
|
dev_warn(&hdev->pdev->dev,
|
||||||
|
"current firmware does not support command(0x%x)!\n",
|
||||||
|
HCLGE_OPC_PF_RST_DONE);
|
||||||
|
return 0;
|
||||||
|
} else if (ret) {
|
||||||
|
dev_err(&hdev->pdev->dev, "assert PF reset done fail %d!\n",
|
||||||
|
ret);
|
||||||
|
}
|
||||||
|
|
||||||
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
static int hclge_reset_prepare_up(struct hclge_dev *hdev)
|
static int hclge_reset_prepare_up(struct hclge_dev *hdev)
|
||||||
|
@ -1,4 +1,4 @@
|
|||||||
// SPDX-License-Identifier: GPL-2.0+
|
/* SPDX-License-Identifier: GPL-2.0+ */
|
||||||
// Copyright (c) 2016-2017 Hisilicon Limited.
|
// Copyright (c) 2016-2017 Hisilicon Limited.
|
||||||
|
|
||||||
#ifndef __HCLGE_MAIN_H
|
#ifndef __HCLGE_MAIN_H
|
||||||
|
@ -1,4 +1,4 @@
|
|||||||
// SPDX-License-Identifier: GPL-2.0+
|
/* SPDX-License-Identifier: GPL-2.0+ */
|
||||||
// Copyright (c) 2016-2017 Hisilicon Limited.
|
// Copyright (c) 2016-2017 Hisilicon Limited.
|
||||||
|
|
||||||
#ifndef __HCLGE_MDIO_H
|
#ifndef __HCLGE_MDIO_H
|
||||||
|
@ -1,4 +1,4 @@
|
|||||||
// SPDX-License-Identifier: GPL-2.0+
|
/* SPDX-License-Identifier: GPL-2.0+ */
|
||||||
// Copyright (c) 2016-2017 Hisilicon Limited.
|
// Copyright (c) 2016-2017 Hisilicon Limited.
|
||||||
|
|
||||||
#ifndef __HCLGE_TM_H
|
#ifndef __HCLGE_TM_H
|
||||||
|
@ -20,6 +20,8 @@
|
|||||||
|
|
||||||
/* API version 1.7 implements additional link and PHY-specific APIs */
|
/* API version 1.7 implements additional link and PHY-specific APIs */
|
||||||
#define I40E_MINOR_VER_GET_LINK_INFO_XL710 0x0007
|
#define I40E_MINOR_VER_GET_LINK_INFO_XL710 0x0007
|
||||||
|
/* API version 1.9 for X722 implements additional link and PHY-specific APIs */
|
||||||
|
#define I40E_MINOR_VER_GET_LINK_INFO_X722 0x0009
|
||||||
/* API version 1.6 for X722 devices adds ability to stop FW LLDP agent */
|
/* API version 1.6 for X722 devices adds ability to stop FW LLDP agent */
|
||||||
#define I40E_MINOR_VER_FW_LLDP_STOPPABLE_X722 0x0006
|
#define I40E_MINOR_VER_FW_LLDP_STOPPABLE_X722 0x0006
|
||||||
|
|
||||||
|
@ -1876,7 +1876,8 @@ i40e_status i40e_aq_get_link_info(struct i40e_hw *hw,
|
|||||||
hw->aq.fw_min_ver < 40)) && hw_link_info->phy_type == 0xE)
|
hw->aq.fw_min_ver < 40)) && hw_link_info->phy_type == 0xE)
|
||||||
hw_link_info->phy_type = I40E_PHY_TYPE_10GBASE_SFPP_CU;
|
hw_link_info->phy_type = I40E_PHY_TYPE_10GBASE_SFPP_CU;
|
||||||
|
|
||||||
if (hw->flags & I40E_HW_FLAG_AQ_PHY_ACCESS_CAPABLE) {
|
if (hw->flags & I40E_HW_FLAG_AQ_PHY_ACCESS_CAPABLE &&
|
||||||
|
hw->mac.type != I40E_MAC_X722) {
|
||||||
__le32 tmp;
|
__le32 tmp;
|
||||||
|
|
||||||
memcpy(&tmp, resp->link_type, sizeof(tmp));
|
memcpy(&tmp, resp->link_type, sizeof(tmp));
|
||||||
|
@ -689,8 +689,6 @@ static bool i40e_xmit_zc(struct i40e_ring *xdp_ring, unsigned int budget)
|
|||||||
i40e_xdp_ring_update_tail(xdp_ring);
|
i40e_xdp_ring_update_tail(xdp_ring);
|
||||||
|
|
||||||
xsk_umem_consume_tx_done(xdp_ring->xsk_umem);
|
xsk_umem_consume_tx_done(xdp_ring->xsk_umem);
|
||||||
if (xsk_umem_uses_need_wakeup(xdp_ring->xsk_umem))
|
|
||||||
xsk_clear_tx_need_wakeup(xdp_ring->xsk_umem);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
return !!budget && work_done;
|
return !!budget && work_done;
|
||||||
@ -769,12 +767,8 @@ bool i40e_clean_xdp_tx_irq(struct i40e_vsi *vsi,
|
|||||||
i40e_update_tx_stats(tx_ring, completed_frames, total_bytes);
|
i40e_update_tx_stats(tx_ring, completed_frames, total_bytes);
|
||||||
|
|
||||||
out_xmit:
|
out_xmit:
|
||||||
if (xsk_umem_uses_need_wakeup(tx_ring->xsk_umem)) {
|
if (xsk_umem_uses_need_wakeup(tx_ring->xsk_umem))
|
||||||
if (tx_ring->next_to_clean == tx_ring->next_to_use)
|
xsk_set_tx_need_wakeup(tx_ring->xsk_umem);
|
||||||
xsk_set_tx_need_wakeup(tx_ring->xsk_umem);
|
|
||||||
else
|
|
||||||
xsk_clear_tx_need_wakeup(tx_ring->xsk_umem);
|
|
||||||
}
|
|
||||||
|
|
||||||
xmit_done = i40e_xmit_zc(tx_ring, budget);
|
xmit_done = i40e_xmit_zc(tx_ring, budget);
|
||||||
|
|
||||||
|
@ -314,7 +314,7 @@ iavf_map_vector_to_rxq(struct iavf_adapter *adapter, int v_idx, int r_idx)
|
|||||||
q_vector->rx.target_itr = ITR_TO_REG(rx_ring->itr_setting);
|
q_vector->rx.target_itr = ITR_TO_REG(rx_ring->itr_setting);
|
||||||
q_vector->ring_mask |= BIT(r_idx);
|
q_vector->ring_mask |= BIT(r_idx);
|
||||||
wr32(hw, IAVF_VFINT_ITRN1(IAVF_RX_ITR, q_vector->reg_idx),
|
wr32(hw, IAVF_VFINT_ITRN1(IAVF_RX_ITR, q_vector->reg_idx),
|
||||||
q_vector->rx.current_itr);
|
q_vector->rx.current_itr >> 1);
|
||||||
q_vector->rx.current_itr = q_vector->rx.target_itr;
|
q_vector->rx.current_itr = q_vector->rx.target_itr;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -340,7 +340,7 @@ iavf_map_vector_to_txq(struct iavf_adapter *adapter, int v_idx, int t_idx)
|
|||||||
q_vector->tx.target_itr = ITR_TO_REG(tx_ring->itr_setting);
|
q_vector->tx.target_itr = ITR_TO_REG(tx_ring->itr_setting);
|
||||||
q_vector->num_ringpairs++;
|
q_vector->num_ringpairs++;
|
||||||
wr32(hw, IAVF_VFINT_ITRN1(IAVF_TX_ITR, q_vector->reg_idx),
|
wr32(hw, IAVF_VFINT_ITRN1(IAVF_TX_ITR, q_vector->reg_idx),
|
||||||
q_vector->tx.target_itr);
|
q_vector->tx.target_itr >> 1);
|
||||||
q_vector->tx.current_itr = q_vector->tx.target_itr;
|
q_vector->tx.current_itr = q_vector->tx.target_itr;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1036,7 +1036,7 @@ enum ice_status ice_sched_query_res_alloc(struct ice_hw *hw)
|
|||||||
struct ice_aqc_query_txsched_res_resp *buf;
|
struct ice_aqc_query_txsched_res_resp *buf;
|
||||||
enum ice_status status = 0;
|
enum ice_status status = 0;
|
||||||
__le16 max_sibl;
|
__le16 max_sibl;
|
||||||
u8 i;
|
u16 i;
|
||||||
|
|
||||||
if (hw->layer_info)
|
if (hw->layer_info)
|
||||||
return status;
|
return status;
|
||||||
|
@ -5675,8 +5675,8 @@ static void igb_tx_ctxtdesc(struct igb_ring *tx_ring,
|
|||||||
* should have been handled by the upper layers.
|
* should have been handled by the upper layers.
|
||||||
*/
|
*/
|
||||||
if (tx_ring->launchtime_enable) {
|
if (tx_ring->launchtime_enable) {
|
||||||
ts = ns_to_timespec64(first->skb->tstamp);
|
ts = ktime_to_timespec64(first->skb->tstamp);
|
||||||
first->skb->tstamp = 0;
|
first->skb->tstamp = ktime_set(0, 0);
|
||||||
context_desc->seqnum_seed = cpu_to_le32(ts.tv_nsec / 32);
|
context_desc->seqnum_seed = cpu_to_le32(ts.tv_nsec / 32);
|
||||||
} else {
|
} else {
|
||||||
context_desc->seqnum_seed = 0;
|
context_desc->seqnum_seed = 0;
|
||||||
|
@ -824,8 +824,8 @@ static void igc_tx_ctxtdesc(struct igc_ring *tx_ring,
|
|||||||
* should have been handled by the upper layers.
|
* should have been handled by the upper layers.
|
||||||
*/
|
*/
|
||||||
if (tx_ring->launchtime_enable) {
|
if (tx_ring->launchtime_enable) {
|
||||||
ts = ns_to_timespec64(first->skb->tstamp);
|
ts = ktime_to_timespec64(first->skb->tstamp);
|
||||||
first->skb->tstamp = 0;
|
first->skb->tstamp = ktime_set(0, 0);
|
||||||
context_desc->launch_time = cpu_to_le32(ts.tv_nsec / 32);
|
context_desc->launch_time = cpu_to_le32(ts.tv_nsec / 32);
|
||||||
} else {
|
} else {
|
||||||
context_desc->launch_time = 0;
|
context_desc->launch_time = 0;
|
||||||
|
@ -622,8 +622,6 @@ static bool ixgbe_xmit_zc(struct ixgbe_ring *xdp_ring, unsigned int budget)
|
|||||||
if (tx_desc) {
|
if (tx_desc) {
|
||||||
ixgbe_xdp_ring_update_tail(xdp_ring);
|
ixgbe_xdp_ring_update_tail(xdp_ring);
|
||||||
xsk_umem_consume_tx_done(xdp_ring->xsk_umem);
|
xsk_umem_consume_tx_done(xdp_ring->xsk_umem);
|
||||||
if (xsk_umem_uses_need_wakeup(xdp_ring->xsk_umem))
|
|
||||||
xsk_clear_tx_need_wakeup(xdp_ring->xsk_umem);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
return !!budget && work_done;
|
return !!budget && work_done;
|
||||||
@ -691,12 +689,8 @@ bool ixgbe_clean_xdp_tx_irq(struct ixgbe_q_vector *q_vector,
|
|||||||
if (xsk_frames)
|
if (xsk_frames)
|
||||||
xsk_umem_complete_tx(umem, xsk_frames);
|
xsk_umem_complete_tx(umem, xsk_frames);
|
||||||
|
|
||||||
if (xsk_umem_uses_need_wakeup(tx_ring->xsk_umem)) {
|
if (xsk_umem_uses_need_wakeup(tx_ring->xsk_umem))
|
||||||
if (tx_ring->next_to_clean == tx_ring->next_to_use)
|
xsk_set_tx_need_wakeup(tx_ring->xsk_umem);
|
||||||
xsk_set_tx_need_wakeup(tx_ring->xsk_umem);
|
|
||||||
else
|
|
||||||
xsk_clear_tx_need_wakeup(tx_ring->xsk_umem);
|
|
||||||
}
|
|
||||||
|
|
||||||
return ixgbe_xmit_zc(tx_ring, q_vector->tx.work_limit);
|
return ixgbe_xmit_zc(tx_ring, q_vector->tx.work_limit);
|
||||||
}
|
}
|
||||||
|
@ -514,8 +514,7 @@ static int mlx4_dev_cap(struct mlx4_dev *dev, struct mlx4_dev_cap *dev_cap)
|
|||||||
dev->caps.max_rq_desc_sz = dev_cap->max_rq_desc_sz;
|
dev->caps.max_rq_desc_sz = dev_cap->max_rq_desc_sz;
|
||||||
/*
|
/*
|
||||||
* Subtract 1 from the limit because we need to allocate a
|
* Subtract 1 from the limit because we need to allocate a
|
||||||
* spare CQE so the HCA HW can tell the difference between an
|
* spare CQE to enable resizing the CQ.
|
||||||
* empty CQ and a full CQ.
|
|
||||||
*/
|
*/
|
||||||
dev->caps.max_cqes = dev_cap->max_cq_sz - 1;
|
dev->caps.max_cqes = dev_cap->max_cq_sz - 1;
|
||||||
dev->caps.reserved_cqs = dev_cap->reserved_cqs;
|
dev->caps.reserved_cqs = dev_cap->reserved_cqs;
|
||||||
|
@ -1079,7 +1079,7 @@ static int esw_create_offloads_fdb_tables(struct mlx5_eswitch *esw, int nvports)
|
|||||||
MLX5_CAP_GEN(dev, max_flow_counter_15_0);
|
MLX5_CAP_GEN(dev, max_flow_counter_15_0);
|
||||||
fdb_max = 1 << MLX5_CAP_ESW_FLOWTABLE_FDB(dev, log_max_ft_size);
|
fdb_max = 1 << MLX5_CAP_ESW_FLOWTABLE_FDB(dev, log_max_ft_size);
|
||||||
|
|
||||||
esw_debug(dev, "Create offloads FDB table, min (max esw size(2^%d), max counters(%d), groups(%d), max flow table size(2^%d))\n",
|
esw_debug(dev, "Create offloads FDB table, min (max esw size(2^%d), max counters(%d), groups(%d), max flow table size(%d))\n",
|
||||||
MLX5_CAP_ESW_FLOWTABLE_FDB(dev, log_max_ft_size),
|
MLX5_CAP_ESW_FLOWTABLE_FDB(dev, log_max_ft_size),
|
||||||
max_flow_counter, ESW_OFFLOADS_NUM_GROUPS,
|
max_flow_counter, ESW_OFFLOADS_NUM_GROUPS,
|
||||||
fdb_max);
|
fdb_max);
|
||||||
|
@ -183,7 +183,8 @@ static bool mlx5_eswitch_offload_is_uplink_port(const struct mlx5_eswitch *esw,
|
|||||||
u32 port_mask, port_value;
|
u32 port_mask, port_value;
|
||||||
|
|
||||||
if (MLX5_CAP_ESW_FLOWTABLE(esw->dev, flow_source))
|
if (MLX5_CAP_ESW_FLOWTABLE(esw->dev, flow_source))
|
||||||
return spec->flow_context.flow_source == MLX5_VPORT_UPLINK;
|
return spec->flow_context.flow_source ==
|
||||||
|
MLX5_FLOW_CONTEXT_FLOW_SOURCE_UPLINK;
|
||||||
|
|
||||||
port_mask = MLX5_GET(fte_match_param, spec->match_criteria,
|
port_mask = MLX5_GET(fte_match_param, spec->match_criteria,
|
||||||
misc_parameters.source_port);
|
misc_parameters.source_port);
|
||||||
|
@ -1577,6 +1577,7 @@ int mlx5dr_action_destroy(struct mlx5dr_action *action)
|
|||||||
break;
|
break;
|
||||||
case DR_ACTION_TYP_MODIFY_HDR:
|
case DR_ACTION_TYP_MODIFY_HDR:
|
||||||
mlx5dr_icm_free_chunk(action->rewrite.chunk);
|
mlx5dr_icm_free_chunk(action->rewrite.chunk);
|
||||||
|
kfree(action->rewrite.data);
|
||||||
refcount_dec(&action->rewrite.dmn->refcount);
|
refcount_dec(&action->rewrite.dmn->refcount);
|
||||||
break;
|
break;
|
||||||
default:
|
default:
|
||||||
|
@ -1096,6 +1096,8 @@ dr_rule_create_rule_nic(struct mlx5dr_rule *rule,
|
|||||||
if (htbl)
|
if (htbl)
|
||||||
mlx5dr_htbl_put(htbl);
|
mlx5dr_htbl_put(htbl);
|
||||||
|
|
||||||
|
kfree(hw_ste_arr);
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
|
|
||||||
free_ste:
|
free_ste:
|
||||||
|
@ -1680,9 +1680,6 @@ static int ocelot_netdevice_port_event(struct net_device *dev,
|
|||||||
struct ocelot_port *ocelot_port = netdev_priv(dev);
|
struct ocelot_port *ocelot_port = netdev_priv(dev);
|
||||||
int err = 0;
|
int err = 0;
|
||||||
|
|
||||||
if (!ocelot_netdevice_dev_check(dev))
|
|
||||||
return 0;
|
|
||||||
|
|
||||||
switch (event) {
|
switch (event) {
|
||||||
case NETDEV_CHANGEUPPER:
|
case NETDEV_CHANGEUPPER:
|
||||||
if (netif_is_bridge_master(info->upper_dev)) {
|
if (netif_is_bridge_master(info->upper_dev)) {
|
||||||
@ -1719,12 +1716,16 @@ static int ocelot_netdevice_event(struct notifier_block *unused,
|
|||||||
struct net_device *dev = netdev_notifier_info_to_dev(ptr);
|
struct net_device *dev = netdev_notifier_info_to_dev(ptr);
|
||||||
int ret = 0;
|
int ret = 0;
|
||||||
|
|
||||||
|
if (!ocelot_netdevice_dev_check(dev))
|
||||||
|
return 0;
|
||||||
|
|
||||||
if (event == NETDEV_PRECHANGEUPPER &&
|
if (event == NETDEV_PRECHANGEUPPER &&
|
||||||
netif_is_lag_master(info->upper_dev)) {
|
netif_is_lag_master(info->upper_dev)) {
|
||||||
struct netdev_lag_upper_info *lag_upper_info = info->upper_info;
|
struct netdev_lag_upper_info *lag_upper_info = info->upper_info;
|
||||||
struct netlink_ext_ack *extack;
|
struct netlink_ext_ack *extack;
|
||||||
|
|
||||||
if (lag_upper_info->tx_type != NETDEV_LAG_TX_TYPE_HASH) {
|
if (lag_upper_info &&
|
||||||
|
lag_upper_info->tx_type != NETDEV_LAG_TX_TYPE_HASH) {
|
||||||
extack = netdev_notifier_info_to_extack(&info->info);
|
extack = netdev_notifier_info_to_extack(&info->info);
|
||||||
NL_SET_ERR_MSG_MOD(extack, "LAG device using unsupported Tx type");
|
NL_SET_ERR_MSG_MOD(extack, "LAG device using unsupported Tx type");
|
||||||
|
|
||||||
|
@ -523,7 +523,7 @@ void __ocelot_write_ix(struct ocelot *ocelot, u32 val, u32 reg, u32 offset);
|
|||||||
#define ocelot_write_rix(ocelot, val, reg, ri) __ocelot_write_ix(ocelot, val, reg, reg##_RSZ * (ri))
|
#define ocelot_write_rix(ocelot, val, reg, ri) __ocelot_write_ix(ocelot, val, reg, reg##_RSZ * (ri))
|
||||||
#define ocelot_write(ocelot, val, reg) __ocelot_write_ix(ocelot, val, reg, 0)
|
#define ocelot_write(ocelot, val, reg) __ocelot_write_ix(ocelot, val, reg, 0)
|
||||||
|
|
||||||
void __ocelot_rmw_ix(struct ocelot *ocelot, u32 val, u32 reg, u32 mask,
|
void __ocelot_rmw_ix(struct ocelot *ocelot, u32 val, u32 mask, u32 reg,
|
||||||
u32 offset);
|
u32 offset);
|
||||||
#define ocelot_rmw_ix(ocelot, val, m, reg, gi, ri) __ocelot_rmw_ix(ocelot, val, m, reg, reg##_GSZ * (gi) + reg##_RSZ * (ri))
|
#define ocelot_rmw_ix(ocelot, val, m, reg, gi, ri) __ocelot_rmw_ix(ocelot, val, m, reg, reg##_GSZ * (gi) + reg##_RSZ * (ri))
|
||||||
#define ocelot_rmw_gix(ocelot, val, m, reg, gi) __ocelot_rmw_ix(ocelot, val, m, reg, reg##_GSZ * (gi))
|
#define ocelot_rmw_gix(ocelot, val, m, reg, gi) __ocelot_rmw_ix(ocelot, val, m, reg, reg##_GSZ * (gi))
|
||||||
|
@ -1208,8 +1208,16 @@ enum qede_remove_mode {
|
|||||||
static void __qede_remove(struct pci_dev *pdev, enum qede_remove_mode mode)
|
static void __qede_remove(struct pci_dev *pdev, enum qede_remove_mode mode)
|
||||||
{
|
{
|
||||||
struct net_device *ndev = pci_get_drvdata(pdev);
|
struct net_device *ndev = pci_get_drvdata(pdev);
|
||||||
struct qede_dev *edev = netdev_priv(ndev);
|
struct qede_dev *edev;
|
||||||
struct qed_dev *cdev = edev->cdev;
|
struct qed_dev *cdev;
|
||||||
|
|
||||||
|
if (!ndev) {
|
||||||
|
dev_info(&pdev->dev, "Device has already been removed\n");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
edev = netdev_priv(ndev);
|
||||||
|
cdev = edev->cdev;
|
||||||
|
|
||||||
DP_INFO(edev, "Starting qede_remove\n");
|
DP_INFO(edev, "Starting qede_remove\n");
|
||||||
|
|
||||||
|
@ -57,10 +57,10 @@ static int rmnet_unregister_real_device(struct net_device *real_dev,
|
|||||||
if (port->nr_rmnet_devs)
|
if (port->nr_rmnet_devs)
|
||||||
return -EINVAL;
|
return -EINVAL;
|
||||||
|
|
||||||
kfree(port);
|
|
||||||
|
|
||||||
netdev_rx_handler_unregister(real_dev);
|
netdev_rx_handler_unregister(real_dev);
|
||||||
|
|
||||||
|
kfree(port);
|
||||||
|
|
||||||
/* release reference on real_dev */
|
/* release reference on real_dev */
|
||||||
dev_put(real_dev);
|
dev_put(real_dev);
|
||||||
|
|
||||||
|
@ -916,6 +916,9 @@ static void r8168g_mdio_write(struct rtl8169_private *tp, int reg, int value)
|
|||||||
|
|
||||||
static int r8168g_mdio_read(struct rtl8169_private *tp, int reg)
|
static int r8168g_mdio_read(struct rtl8169_private *tp, int reg)
|
||||||
{
|
{
|
||||||
|
if (reg == 0x1f)
|
||||||
|
return tp->ocp_base == OCP_STD_PHY_BASE ? 0 : tp->ocp_base >> 4;
|
||||||
|
|
||||||
if (tp->ocp_base != OCP_STD_PHY_BASE)
|
if (tp->ocp_base != OCP_STD_PHY_BASE)
|
||||||
reg -= 0x10;
|
reg -= 0x10;
|
||||||
|
|
||||||
|
@ -432,7 +432,7 @@ static void dwmac4_set_filter(struct mac_device_info *hw,
|
|||||||
* bits used depends on the hardware configuration
|
* bits used depends on the hardware configuration
|
||||||
* selected at core configuration time.
|
* selected at core configuration time.
|
||||||
*/
|
*/
|
||||||
int bit_nr = bitrev32(~crc32_le(~0, ha->addr,
|
u32 bit_nr = bitrev32(~crc32_le(~0, ha->addr,
|
||||||
ETH_ALEN)) >> (32 - mcbitslog2);
|
ETH_ALEN)) >> (32 - mcbitslog2);
|
||||||
/* The most significant bit determines the register to
|
/* The most significant bit determines the register to
|
||||||
* use (H/L) while the other 5 bits determine the bit
|
* use (H/L) while the other 5 bits determine the bit
|
||||||
|
@ -224,6 +224,7 @@ static void dwxgmac2_config_cbs(struct mac_device_info *hw,
|
|||||||
writel(low_credit, ioaddr + XGMAC_MTL_TCx_LOCREDIT(queue));
|
writel(low_credit, ioaddr + XGMAC_MTL_TCx_LOCREDIT(queue));
|
||||||
|
|
||||||
value = readl(ioaddr + XGMAC_MTL_TCx_ETS_CONTROL(queue));
|
value = readl(ioaddr + XGMAC_MTL_TCx_ETS_CONTROL(queue));
|
||||||
|
value &= ~XGMAC_TSA;
|
||||||
value |= XGMAC_CC | XGMAC_CBS;
|
value |= XGMAC_CC | XGMAC_CBS;
|
||||||
writel(value, ioaddr + XGMAC_MTL_TCx_ETS_CONTROL(queue));
|
writel(value, ioaddr + XGMAC_MTL_TCx_ETS_CONTROL(queue));
|
||||||
}
|
}
|
||||||
@ -463,7 +464,7 @@ static void dwxgmac2_set_filter(struct mac_device_info *hw,
|
|||||||
value |= XGMAC_FILTER_HMC;
|
value |= XGMAC_FILTER_HMC;
|
||||||
|
|
||||||
netdev_for_each_mc_addr(ha, dev) {
|
netdev_for_each_mc_addr(ha, dev) {
|
||||||
int nr = (bitrev32(~crc32_le(~0, ha->addr, 6)) >>
|
u32 nr = (bitrev32(~crc32_le(~0, ha->addr, 6)) >>
|
||||||
(32 - mcbitslog2));
|
(32 - mcbitslog2));
|
||||||
mc_filter[nr >> 5] |= (1 << (nr & 0x1F));
|
mc_filter[nr >> 5] |= (1 << (nr & 0x1F));
|
||||||
}
|
}
|
||||||
|
@ -288,7 +288,8 @@ static int dwxgmac2_get_rx_hash(struct dma_desc *p, u32 *hash,
|
|||||||
|
|
||||||
static int dwxgmac2_get_rx_header_len(struct dma_desc *p, unsigned int *len)
|
static int dwxgmac2_get_rx_header_len(struct dma_desc *p, unsigned int *len)
|
||||||
{
|
{
|
||||||
*len = le32_to_cpu(p->des2) & XGMAC_RDES2_HL;
|
if (le32_to_cpu(p->des3) & XGMAC_RDES3_L34T)
|
||||||
|
*len = le32_to_cpu(p->des2) & XGMAC_RDES2_HL;
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -369,7 +369,7 @@ static void dwxgmac2_get_hw_feature(void __iomem *ioaddr,
|
|||||||
dma_cap->eee = (hw_cap & XGMAC_HWFEAT_EEESEL) >> 13;
|
dma_cap->eee = (hw_cap & XGMAC_HWFEAT_EEESEL) >> 13;
|
||||||
dma_cap->atime_stamp = (hw_cap & XGMAC_HWFEAT_TSSEL) >> 12;
|
dma_cap->atime_stamp = (hw_cap & XGMAC_HWFEAT_TSSEL) >> 12;
|
||||||
dma_cap->av = (hw_cap & XGMAC_HWFEAT_AVSEL) >> 11;
|
dma_cap->av = (hw_cap & XGMAC_HWFEAT_AVSEL) >> 11;
|
||||||
dma_cap->av &= !(hw_cap & XGMAC_HWFEAT_RAVSEL) >> 10;
|
dma_cap->av &= !((hw_cap & XGMAC_HWFEAT_RAVSEL) >> 10);
|
||||||
dma_cap->arpoffsel = (hw_cap & XGMAC_HWFEAT_ARPOFFSEL) >> 9;
|
dma_cap->arpoffsel = (hw_cap & XGMAC_HWFEAT_ARPOFFSEL) >> 9;
|
||||||
dma_cap->rmon = (hw_cap & XGMAC_HWFEAT_MMCSEL) >> 8;
|
dma_cap->rmon = (hw_cap & XGMAC_HWFEAT_MMCSEL) >> 8;
|
||||||
dma_cap->pmt_magic_frame = (hw_cap & XGMAC_HWFEAT_MGKSEL) >> 7;
|
dma_cap->pmt_magic_frame = (hw_cap & XGMAC_HWFEAT_MGKSEL) >> 7;
|
||||||
@ -470,6 +470,7 @@ static void dwxgmac2_enable_tso(void __iomem *ioaddr, bool en, u32 chan)
|
|||||||
static void dwxgmac2_qmode(void __iomem *ioaddr, u32 channel, u8 qmode)
|
static void dwxgmac2_qmode(void __iomem *ioaddr, u32 channel, u8 qmode)
|
||||||
{
|
{
|
||||||
u32 value = readl(ioaddr + XGMAC_MTL_TXQ_OPMODE(channel));
|
u32 value = readl(ioaddr + XGMAC_MTL_TXQ_OPMODE(channel));
|
||||||
|
u32 flow = readl(ioaddr + XGMAC_RX_FLOW_CTRL);
|
||||||
|
|
||||||
value &= ~XGMAC_TXQEN;
|
value &= ~XGMAC_TXQEN;
|
||||||
if (qmode != MTL_QUEUE_AVB) {
|
if (qmode != MTL_QUEUE_AVB) {
|
||||||
@ -477,6 +478,7 @@ static void dwxgmac2_qmode(void __iomem *ioaddr, u32 channel, u8 qmode)
|
|||||||
writel(0, ioaddr + XGMAC_MTL_TCx_ETS_CONTROL(channel));
|
writel(0, ioaddr + XGMAC_MTL_TCx_ETS_CONTROL(channel));
|
||||||
} else {
|
} else {
|
||||||
value |= 0x1 << XGMAC_TXQEN_SHIFT;
|
value |= 0x1 << XGMAC_TXQEN_SHIFT;
|
||||||
|
writel(flow & (~XGMAC_RFE), ioaddr + XGMAC_RX_FLOW_CTRL);
|
||||||
}
|
}
|
||||||
|
|
||||||
writel(value, ioaddr + XGMAC_MTL_TXQ_OPMODE(channel));
|
writel(value, ioaddr + XGMAC_MTL_TXQ_OPMODE(channel));
|
||||||
|
@ -176,6 +176,7 @@
|
|||||||
#define MMC_XGMAC_RX_PKT_SMD_ERR 0x22c
|
#define MMC_XGMAC_RX_PKT_SMD_ERR 0x22c
|
||||||
#define MMC_XGMAC_RX_PKT_ASSEMBLY_OK 0x230
|
#define MMC_XGMAC_RX_PKT_ASSEMBLY_OK 0x230
|
||||||
#define MMC_XGMAC_RX_FPE_FRAG 0x234
|
#define MMC_XGMAC_RX_FPE_FRAG 0x234
|
||||||
|
#define MMC_XGMAC_RX_IPC_INTR_MASK 0x25c
|
||||||
|
|
||||||
static void dwmac_mmc_ctrl(void __iomem *mmcaddr, unsigned int mode)
|
static void dwmac_mmc_ctrl(void __iomem *mmcaddr, unsigned int mode)
|
||||||
{
|
{
|
||||||
@ -333,8 +334,9 @@ static void dwxgmac_mmc_ctrl(void __iomem *mmcaddr, unsigned int mode)
|
|||||||
|
|
||||||
static void dwxgmac_mmc_intr_all_mask(void __iomem *mmcaddr)
|
static void dwxgmac_mmc_intr_all_mask(void __iomem *mmcaddr)
|
||||||
{
|
{
|
||||||
writel(MMC_DEFAULT_MASK, mmcaddr + MMC_RX_INTR_MASK);
|
writel(0x0, mmcaddr + MMC_RX_INTR_MASK);
|
||||||
writel(MMC_DEFAULT_MASK, mmcaddr + MMC_TX_INTR_MASK);
|
writel(0x0, mmcaddr + MMC_TX_INTR_MASK);
|
||||||
|
writel(MMC_DEFAULT_MASK, mmcaddr + MMC_XGMAC_RX_IPC_INTR_MASK);
|
||||||
}
|
}
|
||||||
|
|
||||||
static void dwxgmac_read_mmc_reg(void __iomem *addr, u32 reg, u32 *dest)
|
static void dwxgmac_read_mmc_reg(void __iomem *addr, u32 reg, u32 *dest)
|
||||||
|
@ -2996,6 +2996,7 @@ static netdev_tx_t stmmac_tso_xmit(struct sk_buff *skb, struct net_device *dev)
|
|||||||
stmmac_set_desc_addr(priv, first, des);
|
stmmac_set_desc_addr(priv, first, des);
|
||||||
tmp_pay_len = pay_len;
|
tmp_pay_len = pay_len;
|
||||||
des += proto_hdr_len;
|
des += proto_hdr_len;
|
||||||
|
pay_len = 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
stmmac_tso_allocator(priv, des, tmp_pay_len, (nfrags == 0), queue);
|
stmmac_tso_allocator(priv, des, tmp_pay_len, (nfrags == 0), queue);
|
||||||
@ -3023,6 +3024,19 @@ static netdev_tx_t stmmac_tso_xmit(struct sk_buff *skb, struct net_device *dev)
|
|||||||
/* Only the last descriptor gets to point to the skb. */
|
/* Only the last descriptor gets to point to the skb. */
|
||||||
tx_q->tx_skbuff[tx_q->cur_tx] = skb;
|
tx_q->tx_skbuff[tx_q->cur_tx] = skb;
|
||||||
|
|
||||||
|
/* Manage tx mitigation */
|
||||||
|
tx_q->tx_count_frames += nfrags + 1;
|
||||||
|
if (likely(priv->tx_coal_frames > tx_q->tx_count_frames) &&
|
||||||
|
!((skb_shinfo(skb)->tx_flags & SKBTX_HW_TSTAMP) &&
|
||||||
|
priv->hwts_tx_en)) {
|
||||||
|
stmmac_tx_timer_arm(priv, queue);
|
||||||
|
} else {
|
||||||
|
desc = &tx_q->dma_tx[tx_q->cur_tx];
|
||||||
|
tx_q->tx_count_frames = 0;
|
||||||
|
stmmac_set_tx_ic(priv, desc);
|
||||||
|
priv->xstats.tx_set_ic_bit++;
|
||||||
|
}
|
||||||
|
|
||||||
/* We've used all descriptors we need for this skb, however,
|
/* We've used all descriptors we need for this skb, however,
|
||||||
* advance cur_tx so that it references a fresh descriptor.
|
* advance cur_tx so that it references a fresh descriptor.
|
||||||
* ndo_start_xmit will fill this descriptor the next time it's
|
* ndo_start_xmit will fill this descriptor the next time it's
|
||||||
@ -3040,19 +3054,6 @@ static netdev_tx_t stmmac_tso_xmit(struct sk_buff *skb, struct net_device *dev)
|
|||||||
priv->xstats.tx_tso_frames++;
|
priv->xstats.tx_tso_frames++;
|
||||||
priv->xstats.tx_tso_nfrags += nfrags;
|
priv->xstats.tx_tso_nfrags += nfrags;
|
||||||
|
|
||||||
/* Manage tx mitigation */
|
|
||||||
tx_q->tx_count_frames += nfrags + 1;
|
|
||||||
if (likely(priv->tx_coal_frames > tx_q->tx_count_frames) &&
|
|
||||||
!(priv->synopsys_id >= DWMAC_CORE_4_00 &&
|
|
||||||
(skb_shinfo(skb)->tx_flags & SKBTX_HW_TSTAMP) &&
|
|
||||||
priv->hwts_tx_en)) {
|
|
||||||
stmmac_tx_timer_arm(priv, queue);
|
|
||||||
} else {
|
|
||||||
tx_q->tx_count_frames = 0;
|
|
||||||
stmmac_set_tx_ic(priv, desc);
|
|
||||||
priv->xstats.tx_set_ic_bit++;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (priv->sarc_type)
|
if (priv->sarc_type)
|
||||||
stmmac_set_desc_sarc(priv, first, priv->sarc_type);
|
stmmac_set_desc_sarc(priv, first, priv->sarc_type);
|
||||||
|
|
||||||
@ -3224,6 +3225,27 @@ static netdev_tx_t stmmac_xmit(struct sk_buff *skb, struct net_device *dev)
|
|||||||
/* Only the last descriptor gets to point to the skb. */
|
/* Only the last descriptor gets to point to the skb. */
|
||||||
tx_q->tx_skbuff[entry] = skb;
|
tx_q->tx_skbuff[entry] = skb;
|
||||||
|
|
||||||
|
/* According to the coalesce parameter the IC bit for the latest
|
||||||
|
* segment is reset and the timer re-started to clean the tx status.
|
||||||
|
* This approach takes care about the fragments: desc is the first
|
||||||
|
* element in case of no SG.
|
||||||
|
*/
|
||||||
|
tx_q->tx_count_frames += nfrags + 1;
|
||||||
|
if (likely(priv->tx_coal_frames > tx_q->tx_count_frames) &&
|
||||||
|
!((skb_shinfo(skb)->tx_flags & SKBTX_HW_TSTAMP) &&
|
||||||
|
priv->hwts_tx_en)) {
|
||||||
|
stmmac_tx_timer_arm(priv, queue);
|
||||||
|
} else {
|
||||||
|
if (likely(priv->extend_desc))
|
||||||
|
desc = &tx_q->dma_etx[entry].basic;
|
||||||
|
else
|
||||||
|
desc = &tx_q->dma_tx[entry];
|
||||||
|
|
||||||
|
tx_q->tx_count_frames = 0;
|
||||||
|
stmmac_set_tx_ic(priv, desc);
|
||||||
|
priv->xstats.tx_set_ic_bit++;
|
||||||
|
}
|
||||||
|
|
||||||
/* We've used all descriptors we need for this skb, however,
|
/* We've used all descriptors we need for this skb, however,
|
||||||
* advance cur_tx so that it references a fresh descriptor.
|
* advance cur_tx so that it references a fresh descriptor.
|
||||||
* ndo_start_xmit will fill this descriptor the next time it's
|
* ndo_start_xmit will fill this descriptor the next time it's
|
||||||
@ -3259,23 +3281,6 @@ static netdev_tx_t stmmac_xmit(struct sk_buff *skb, struct net_device *dev)
|
|||||||
|
|
||||||
dev->stats.tx_bytes += skb->len;
|
dev->stats.tx_bytes += skb->len;
|
||||||
|
|
||||||
/* According to the coalesce parameter the IC bit for the latest
|
|
||||||
* segment is reset and the timer re-started to clean the tx status.
|
|
||||||
* This approach takes care about the fragments: desc is the first
|
|
||||||
* element in case of no SG.
|
|
||||||
*/
|
|
||||||
tx_q->tx_count_frames += nfrags + 1;
|
|
||||||
if (likely(priv->tx_coal_frames > tx_q->tx_count_frames) &&
|
|
||||||
!(priv->synopsys_id >= DWMAC_CORE_4_00 &&
|
|
||||||
(skb_shinfo(skb)->tx_flags & SKBTX_HW_TSTAMP) &&
|
|
||||||
priv->hwts_tx_en)) {
|
|
||||||
stmmac_tx_timer_arm(priv, queue);
|
|
||||||
} else {
|
|
||||||
tx_q->tx_count_frames = 0;
|
|
||||||
stmmac_set_tx_ic(priv, desc);
|
|
||||||
priv->xstats.tx_set_ic_bit++;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (priv->sarc_type)
|
if (priv->sarc_type)
|
||||||
stmmac_set_desc_sarc(priv, first, priv->sarc_type);
|
stmmac_set_desc_sarc(priv, first, priv->sarc_type);
|
||||||
|
|
||||||
@ -3506,8 +3511,6 @@ read_again:
|
|||||||
if (unlikely(status & dma_own))
|
if (unlikely(status & dma_own))
|
||||||
break;
|
break;
|
||||||
|
|
||||||
count++;
|
|
||||||
|
|
||||||
rx_q->cur_rx = STMMAC_GET_ENTRY(rx_q->cur_rx, DMA_RX_SIZE);
|
rx_q->cur_rx = STMMAC_GET_ENTRY(rx_q->cur_rx, DMA_RX_SIZE);
|
||||||
next_entry = rx_q->cur_rx;
|
next_entry = rx_q->cur_rx;
|
||||||
|
|
||||||
@ -3534,6 +3537,7 @@ read_again:
|
|||||||
goto read_again;
|
goto read_again;
|
||||||
if (unlikely(error)) {
|
if (unlikely(error)) {
|
||||||
dev_kfree_skb(skb);
|
dev_kfree_skb(skb);
|
||||||
|
count++;
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -3573,6 +3577,7 @@ read_again:
|
|||||||
skb = napi_alloc_skb(&ch->rx_napi, len);
|
skb = napi_alloc_skb(&ch->rx_napi, len);
|
||||||
if (!skb) {
|
if (!skb) {
|
||||||
priv->dev->stats.rx_dropped++;
|
priv->dev->stats.rx_dropped++;
|
||||||
|
count++;
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -3638,6 +3643,7 @@ read_again:
|
|||||||
|
|
||||||
priv->dev->stats.rx_packets++;
|
priv->dev->stats.rx_packets++;
|
||||||
priv->dev->stats.rx_bytes += len;
|
priv->dev->stats.rx_bytes += len;
|
||||||
|
count++;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (status & rx_not_ls) {
|
if (status & rx_not_ls) {
|
||||||
|
@ -6,7 +6,9 @@
|
|||||||
* Author: Jose Abreu <joabreu@synopsys.com>
|
* Author: Jose Abreu <joabreu@synopsys.com>
|
||||||
*/
|
*/
|
||||||
|
|
||||||
|
#include <linux/bitrev.h>
|
||||||
#include <linux/completion.h>
|
#include <linux/completion.h>
|
||||||
|
#include <linux/crc32.h>
|
||||||
#include <linux/ethtool.h>
|
#include <linux/ethtool.h>
|
||||||
#include <linux/ip.h>
|
#include <linux/ip.h>
|
||||||
#include <linux/phy.h>
|
#include <linux/phy.h>
|
||||||
@ -485,12 +487,48 @@ static int stmmac_filter_check(struct stmmac_priv *priv)
|
|||||||
return -EOPNOTSUPP;
|
return -EOPNOTSUPP;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static bool stmmac_hash_check(struct stmmac_priv *priv, unsigned char *addr)
|
||||||
|
{
|
||||||
|
int mc_offset = 32 - priv->hw->mcast_bits_log2;
|
||||||
|
struct netdev_hw_addr *ha;
|
||||||
|
u32 hash, hash_nr;
|
||||||
|
|
||||||
|
/* First compute the hash for desired addr */
|
||||||
|
hash = bitrev32(~crc32_le(~0, addr, 6)) >> mc_offset;
|
||||||
|
hash_nr = hash >> 5;
|
||||||
|
hash = 1 << (hash & 0x1f);
|
||||||
|
|
||||||
|
/* Now, check if it collides with any existing one */
|
||||||
|
netdev_for_each_mc_addr(ha, priv->dev) {
|
||||||
|
u32 nr = bitrev32(~crc32_le(~0, ha->addr, ETH_ALEN)) >> mc_offset;
|
||||||
|
if (((nr >> 5) == hash_nr) && ((1 << (nr & 0x1f)) == hash))
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* No collisions, address is good to go */
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
static bool stmmac_perfect_check(struct stmmac_priv *priv, unsigned char *addr)
|
||||||
|
{
|
||||||
|
struct netdev_hw_addr *ha;
|
||||||
|
|
||||||
|
/* Check if it collides with any existing one */
|
||||||
|
netdev_for_each_uc_addr(ha, priv->dev) {
|
||||||
|
if (!memcmp(ha->addr, addr, ETH_ALEN))
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* No collisions, address is good to go */
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
static int stmmac_test_hfilt(struct stmmac_priv *priv)
|
static int stmmac_test_hfilt(struct stmmac_priv *priv)
|
||||||
{
|
{
|
||||||
unsigned char gd_addr[ETH_ALEN] = {0x01, 0xee, 0xdd, 0xcc, 0xbb, 0xaa};
|
unsigned char gd_addr[ETH_ALEN] = {0xf1, 0xee, 0xdd, 0xcc, 0xbb, 0xaa};
|
||||||
unsigned char bd_addr[ETH_ALEN] = {0x01, 0x01, 0x02, 0x03, 0x04, 0x05};
|
unsigned char bd_addr[ETH_ALEN] = {0xf1, 0xff, 0xff, 0xff, 0xff, 0xff};
|
||||||
struct stmmac_packet_attrs attr = { };
|
struct stmmac_packet_attrs attr = { };
|
||||||
int ret;
|
int ret, tries = 256;
|
||||||
|
|
||||||
ret = stmmac_filter_check(priv);
|
ret = stmmac_filter_check(priv);
|
||||||
if (ret)
|
if (ret)
|
||||||
@ -499,6 +537,16 @@ static int stmmac_test_hfilt(struct stmmac_priv *priv)
|
|||||||
if (netdev_mc_count(priv->dev) >= priv->hw->multicast_filter_bins)
|
if (netdev_mc_count(priv->dev) >= priv->hw->multicast_filter_bins)
|
||||||
return -EOPNOTSUPP;
|
return -EOPNOTSUPP;
|
||||||
|
|
||||||
|
while (--tries) {
|
||||||
|
/* We only need to check the bd_addr for collisions */
|
||||||
|
bd_addr[ETH_ALEN - 1] = tries;
|
||||||
|
if (stmmac_hash_check(priv, bd_addr))
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!tries)
|
||||||
|
return -EOPNOTSUPP;
|
||||||
|
|
||||||
ret = dev_mc_add(priv->dev, gd_addr);
|
ret = dev_mc_add(priv->dev, gd_addr);
|
||||||
if (ret)
|
if (ret)
|
||||||
return ret;
|
return ret;
|
||||||
@ -523,13 +571,25 @@ cleanup:
|
|||||||
|
|
||||||
static int stmmac_test_pfilt(struct stmmac_priv *priv)
|
static int stmmac_test_pfilt(struct stmmac_priv *priv)
|
||||||
{
|
{
|
||||||
unsigned char gd_addr[ETH_ALEN] = {0x00, 0x01, 0x44, 0x55, 0x66, 0x77};
|
unsigned char gd_addr[ETH_ALEN] = {0xf0, 0x01, 0x44, 0x55, 0x66, 0x77};
|
||||||
unsigned char bd_addr[ETH_ALEN] = {0x08, 0x00, 0x22, 0x33, 0x44, 0x55};
|
unsigned char bd_addr[ETH_ALEN] = {0xf0, 0xff, 0xff, 0xff, 0xff, 0xff};
|
||||||
struct stmmac_packet_attrs attr = { };
|
struct stmmac_packet_attrs attr = { };
|
||||||
int ret;
|
int ret, tries = 256;
|
||||||
|
|
||||||
if (stmmac_filter_check(priv))
|
if (stmmac_filter_check(priv))
|
||||||
return -EOPNOTSUPP;
|
return -EOPNOTSUPP;
|
||||||
|
if (netdev_uc_count(priv->dev) >= priv->hw->unicast_filter_entries)
|
||||||
|
return -EOPNOTSUPP;
|
||||||
|
|
||||||
|
while (--tries) {
|
||||||
|
/* We only need to check the bd_addr for collisions */
|
||||||
|
bd_addr[ETH_ALEN - 1] = tries;
|
||||||
|
if (stmmac_perfect_check(priv, bd_addr))
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!tries)
|
||||||
|
return -EOPNOTSUPP;
|
||||||
|
|
||||||
ret = dev_uc_add(priv->dev, gd_addr);
|
ret = dev_uc_add(priv->dev, gd_addr);
|
||||||
if (ret)
|
if (ret)
|
||||||
@ -553,39 +613,31 @@ cleanup:
|
|||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
static int stmmac_dummy_sync(struct net_device *netdev, const u8 *addr)
|
|
||||||
{
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
static void stmmac_test_set_rx_mode(struct net_device *netdev)
|
|
||||||
{
|
|
||||||
/* As we are in test mode of ethtool we already own the rtnl lock
|
|
||||||
* so no address will change from user. We can just call the
|
|
||||||
* ndo_set_rx_mode() callback directly */
|
|
||||||
if (netdev->netdev_ops->ndo_set_rx_mode)
|
|
||||||
netdev->netdev_ops->ndo_set_rx_mode(netdev);
|
|
||||||
}
|
|
||||||
|
|
||||||
static int stmmac_test_mcfilt(struct stmmac_priv *priv)
|
static int stmmac_test_mcfilt(struct stmmac_priv *priv)
|
||||||
{
|
{
|
||||||
unsigned char uc_addr[ETH_ALEN] = {0x00, 0x01, 0x44, 0x55, 0x66, 0x77};
|
unsigned char uc_addr[ETH_ALEN] = {0xf0, 0xff, 0xff, 0xff, 0xff, 0xff};
|
||||||
unsigned char mc_addr[ETH_ALEN] = {0x01, 0x01, 0x44, 0x55, 0x66, 0x77};
|
unsigned char mc_addr[ETH_ALEN] = {0xf1, 0xff, 0xff, 0xff, 0xff, 0xff};
|
||||||
struct stmmac_packet_attrs attr = { };
|
struct stmmac_packet_attrs attr = { };
|
||||||
int ret;
|
int ret, tries = 256;
|
||||||
|
|
||||||
if (stmmac_filter_check(priv))
|
if (stmmac_filter_check(priv))
|
||||||
return -EOPNOTSUPP;
|
return -EOPNOTSUPP;
|
||||||
if (!priv->hw->multicast_filter_bins)
|
if (netdev_uc_count(priv->dev) >= priv->hw->unicast_filter_entries)
|
||||||
return -EOPNOTSUPP;
|
return -EOPNOTSUPP;
|
||||||
|
|
||||||
/* Remove all MC addresses */
|
while (--tries) {
|
||||||
__dev_mc_unsync(priv->dev, NULL);
|
/* We only need to check the mc_addr for collisions */
|
||||||
stmmac_test_set_rx_mode(priv->dev);
|
mc_addr[ETH_ALEN - 1] = tries;
|
||||||
|
if (stmmac_hash_check(priv, mc_addr))
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!tries)
|
||||||
|
return -EOPNOTSUPP;
|
||||||
|
|
||||||
ret = dev_uc_add(priv->dev, uc_addr);
|
ret = dev_uc_add(priv->dev, uc_addr);
|
||||||
if (ret)
|
if (ret)
|
||||||
goto cleanup;
|
return ret;
|
||||||
|
|
||||||
attr.dst = uc_addr;
|
attr.dst = uc_addr;
|
||||||
|
|
||||||
@ -602,30 +654,34 @@ static int stmmac_test_mcfilt(struct stmmac_priv *priv)
|
|||||||
|
|
||||||
cleanup:
|
cleanup:
|
||||||
dev_uc_del(priv->dev, uc_addr);
|
dev_uc_del(priv->dev, uc_addr);
|
||||||
__dev_mc_sync(priv->dev, stmmac_dummy_sync, NULL);
|
|
||||||
stmmac_test_set_rx_mode(priv->dev);
|
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
static int stmmac_test_ucfilt(struct stmmac_priv *priv)
|
static int stmmac_test_ucfilt(struct stmmac_priv *priv)
|
||||||
{
|
{
|
||||||
unsigned char uc_addr[ETH_ALEN] = {0x00, 0x01, 0x44, 0x55, 0x66, 0x77};
|
unsigned char uc_addr[ETH_ALEN] = {0xf0, 0xff, 0xff, 0xff, 0xff, 0xff};
|
||||||
unsigned char mc_addr[ETH_ALEN] = {0x01, 0x01, 0x44, 0x55, 0x66, 0x77};
|
unsigned char mc_addr[ETH_ALEN] = {0xf1, 0xff, 0xff, 0xff, 0xff, 0xff};
|
||||||
struct stmmac_packet_attrs attr = { };
|
struct stmmac_packet_attrs attr = { };
|
||||||
int ret;
|
int ret, tries = 256;
|
||||||
|
|
||||||
if (stmmac_filter_check(priv))
|
if (stmmac_filter_check(priv))
|
||||||
return -EOPNOTSUPP;
|
return -EOPNOTSUPP;
|
||||||
if (!priv->hw->multicast_filter_bins)
|
if (netdev_mc_count(priv->dev) >= priv->hw->multicast_filter_bins)
|
||||||
return -EOPNOTSUPP;
|
return -EOPNOTSUPP;
|
||||||
|
|
||||||
/* Remove all UC addresses */
|
while (--tries) {
|
||||||
__dev_uc_unsync(priv->dev, NULL);
|
/* We only need to check the uc_addr for collisions */
|
||||||
stmmac_test_set_rx_mode(priv->dev);
|
uc_addr[ETH_ALEN - 1] = tries;
|
||||||
|
if (stmmac_perfect_check(priv, uc_addr))
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!tries)
|
||||||
|
return -EOPNOTSUPP;
|
||||||
|
|
||||||
ret = dev_mc_add(priv->dev, mc_addr);
|
ret = dev_mc_add(priv->dev, mc_addr);
|
||||||
if (ret)
|
if (ret)
|
||||||
goto cleanup;
|
return ret;
|
||||||
|
|
||||||
attr.dst = mc_addr;
|
attr.dst = mc_addr;
|
||||||
|
|
||||||
@ -642,8 +698,6 @@ static int stmmac_test_ucfilt(struct stmmac_priv *priv)
|
|||||||
|
|
||||||
cleanup:
|
cleanup:
|
||||||
dev_mc_del(priv->dev, mc_addr);
|
dev_mc_del(priv->dev, mc_addr);
|
||||||
__dev_uc_sync(priv->dev, stmmac_dummy_sync, NULL);
|
|
||||||
stmmac_test_set_rx_mode(priv->dev);
|
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -578,8 +578,8 @@ static void cdc_ncm_set_dgram_size(struct usbnet *dev, int new_size)
|
|||||||
/* read current mtu value from device */
|
/* read current mtu value from device */
|
||||||
err = usbnet_read_cmd(dev, USB_CDC_GET_MAX_DATAGRAM_SIZE,
|
err = usbnet_read_cmd(dev, USB_CDC_GET_MAX_DATAGRAM_SIZE,
|
||||||
USB_TYPE_CLASS | USB_DIR_IN | USB_RECIP_INTERFACE,
|
USB_TYPE_CLASS | USB_DIR_IN | USB_RECIP_INTERFACE,
|
||||||
0, iface_no, &max_datagram_size, 2);
|
0, iface_no, &max_datagram_size, sizeof(max_datagram_size));
|
||||||
if (err < 0) {
|
if (err < sizeof(max_datagram_size)) {
|
||||||
dev_dbg(&dev->intf->dev, "GET_MAX_DATAGRAM_SIZE failed\n");
|
dev_dbg(&dev->intf->dev, "GET_MAX_DATAGRAM_SIZE failed\n");
|
||||||
goto out;
|
goto out;
|
||||||
}
|
}
|
||||||
@ -590,7 +590,7 @@ static void cdc_ncm_set_dgram_size(struct usbnet *dev, int new_size)
|
|||||||
max_datagram_size = cpu_to_le16(ctx->max_datagram_size);
|
max_datagram_size = cpu_to_le16(ctx->max_datagram_size);
|
||||||
err = usbnet_write_cmd(dev, USB_CDC_SET_MAX_DATAGRAM_SIZE,
|
err = usbnet_write_cmd(dev, USB_CDC_SET_MAX_DATAGRAM_SIZE,
|
||||||
USB_TYPE_CLASS | USB_DIR_OUT | USB_RECIP_INTERFACE,
|
USB_TYPE_CLASS | USB_DIR_OUT | USB_RECIP_INTERFACE,
|
||||||
0, iface_no, &max_datagram_size, 2);
|
0, iface_no, &max_datagram_size, sizeof(max_datagram_size));
|
||||||
if (err < 0)
|
if (err < 0)
|
||||||
dev_dbg(&dev->intf->dev, "SET_MAX_DATAGRAM_SIZE failed\n");
|
dev_dbg(&dev->intf->dev, "SET_MAX_DATAGRAM_SIZE failed\n");
|
||||||
|
|
||||||
|
@ -1362,6 +1362,7 @@ static const struct usb_device_id products[] = {
|
|||||||
{QMI_FIXED_INTF(0x413c, 0x81b6, 8)}, /* Dell Wireless 5811e */
|
{QMI_FIXED_INTF(0x413c, 0x81b6, 8)}, /* Dell Wireless 5811e */
|
||||||
{QMI_FIXED_INTF(0x413c, 0x81b6, 10)}, /* Dell Wireless 5811e */
|
{QMI_FIXED_INTF(0x413c, 0x81b6, 10)}, /* Dell Wireless 5811e */
|
||||||
{QMI_FIXED_INTF(0x413c, 0x81d7, 0)}, /* Dell Wireless 5821e */
|
{QMI_FIXED_INTF(0x413c, 0x81d7, 0)}, /* Dell Wireless 5821e */
|
||||||
|
{QMI_FIXED_INTF(0x413c, 0x81e0, 0)}, /* Dell Wireless 5821e with eSIM support*/
|
||||||
{QMI_FIXED_INTF(0x03f0, 0x4e1d, 8)}, /* HP lt4111 LTE/EV-DO/HSPA+ Gobi 4G Module */
|
{QMI_FIXED_INTF(0x03f0, 0x4e1d, 8)}, /* HP lt4111 LTE/EV-DO/HSPA+ Gobi 4G Module */
|
||||||
{QMI_FIXED_INTF(0x03f0, 0x9d1d, 1)}, /* HP lt4120 Snapdragon X5 LTE */
|
{QMI_FIXED_INTF(0x03f0, 0x9d1d, 1)}, /* HP lt4120 Snapdragon X5 LTE */
|
||||||
{QMI_FIXED_INTF(0x22de, 0x9061, 3)}, /* WeTelecom WPD-600N */
|
{QMI_FIXED_INTF(0x22de, 0x9061, 3)}, /* WeTelecom WPD-600N */
|
||||||
|
@ -259,7 +259,7 @@ static void fdp_nci_i2c_read_device_properties(struct device *dev,
|
|||||||
*fw_vsc_cfg, len);
|
*fw_vsc_cfg, len);
|
||||||
|
|
||||||
if (r) {
|
if (r) {
|
||||||
devm_kfree(dev, fw_vsc_cfg);
|
devm_kfree(dev, *fw_vsc_cfg);
|
||||||
goto vsc_read_err;
|
goto vsc_read_err;
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
|
@ -708,6 +708,7 @@ static int st21nfca_hci_complete_target_discovered(struct nfc_hci_dev *hdev,
|
|||||||
NFC_PROTO_FELICA_MASK;
|
NFC_PROTO_FELICA_MASK;
|
||||||
} else {
|
} else {
|
||||||
kfree_skb(nfcid_skb);
|
kfree_skb(nfcid_skb);
|
||||||
|
nfcid_skb = NULL;
|
||||||
/* P2P in type A */
|
/* P2P in type A */
|
||||||
r = nfc_hci_get_param(hdev, ST21NFCA_RF_READER_F_GATE,
|
r = nfc_hci_get_param(hdev, ST21NFCA_RF_READER_F_GATE,
|
||||||
ST21NFCA_RF_READER_F_NFCID1,
|
ST21NFCA_RF_READER_F_NFCID1,
|
||||||
|
@ -656,11 +656,11 @@ void bpf_map_put_with_uref(struct bpf_map *map);
|
|||||||
void bpf_map_put(struct bpf_map *map);
|
void bpf_map_put(struct bpf_map *map);
|
||||||
int bpf_map_charge_memlock(struct bpf_map *map, u32 pages);
|
int bpf_map_charge_memlock(struct bpf_map *map, u32 pages);
|
||||||
void bpf_map_uncharge_memlock(struct bpf_map *map, u32 pages);
|
void bpf_map_uncharge_memlock(struct bpf_map *map, u32 pages);
|
||||||
int bpf_map_charge_init(struct bpf_map_memory *mem, size_t size);
|
int bpf_map_charge_init(struct bpf_map_memory *mem, u64 size);
|
||||||
void bpf_map_charge_finish(struct bpf_map_memory *mem);
|
void bpf_map_charge_finish(struct bpf_map_memory *mem);
|
||||||
void bpf_map_charge_move(struct bpf_map_memory *dst,
|
void bpf_map_charge_move(struct bpf_map_memory *dst,
|
||||||
struct bpf_map_memory *src);
|
struct bpf_map_memory *src);
|
||||||
void *bpf_map_area_alloc(size_t size, int numa_node);
|
void *bpf_map_area_alloc(u64 size, int numa_node);
|
||||||
void bpf_map_area_free(void *base);
|
void bpf_map_area_free(void *base);
|
||||||
void bpf_map_init_from_attr(struct bpf_map *map, union bpf_attr *attr);
|
void bpf_map_init_from_attr(struct bpf_map *map, union bpf_attr *attr);
|
||||||
|
|
||||||
|
@ -139,6 +139,11 @@ static inline void sk_msg_apply_bytes(struct sk_psock *psock, u32 bytes)
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static inline u32 sk_msg_iter_dist(u32 start, u32 end)
|
||||||
|
{
|
||||||
|
return end >= start ? end - start : end + (MAX_MSG_FRAGS - start);
|
||||||
|
}
|
||||||
|
|
||||||
#define sk_msg_iter_var_prev(var) \
|
#define sk_msg_iter_var_prev(var) \
|
||||||
do { \
|
do { \
|
||||||
if (var == 0) \
|
if (var == 0) \
|
||||||
@ -198,9 +203,7 @@ static inline u32 sk_msg_elem_used(const struct sk_msg *msg)
|
|||||||
if (sk_msg_full(msg))
|
if (sk_msg_full(msg))
|
||||||
return MAX_MSG_FRAGS;
|
return MAX_MSG_FRAGS;
|
||||||
|
|
||||||
return msg->sg.end >= msg->sg.start ?
|
return sk_msg_iter_dist(msg->sg.start, msg->sg.end);
|
||||||
msg->sg.end - msg->sg.start :
|
|
||||||
msg->sg.end + (MAX_MSG_FRAGS - msg->sg.start);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
static inline struct scatterlist *sk_msg_elem(struct sk_msg *msg, int which)
|
static inline struct scatterlist *sk_msg_elem(struct sk_msg *msg, int which)
|
||||||
|
@ -159,7 +159,6 @@ struct slave {
|
|||||||
unsigned long target_last_arp_rx[BOND_MAX_ARP_TARGETS];
|
unsigned long target_last_arp_rx[BOND_MAX_ARP_TARGETS];
|
||||||
s8 link; /* one of BOND_LINK_XXXX */
|
s8 link; /* one of BOND_LINK_XXXX */
|
||||||
s8 link_new_state; /* one of BOND_LINK_XXXX */
|
s8 link_new_state; /* one of BOND_LINK_XXXX */
|
||||||
s8 new_link;
|
|
||||||
u8 backup:1, /* indicates backup slave. Value corresponds with
|
u8 backup:1, /* indicates backup slave. Value corresponds with
|
||||||
BOND_STATE_ACTIVE and BOND_STATE_BACKUP */
|
BOND_STATE_ACTIVE and BOND_STATE_BACKUP */
|
||||||
inactive:1, /* indicates inactive slave */
|
inactive:1, /* indicates inactive slave */
|
||||||
@ -549,7 +548,7 @@ static inline void bond_propose_link_state(struct slave *slave, int state)
|
|||||||
|
|
||||||
static inline void bond_commit_link_state(struct slave *slave, bool notify)
|
static inline void bond_commit_link_state(struct slave *slave, bool notify)
|
||||||
{
|
{
|
||||||
if (slave->link == slave->link_new_state)
|
if (slave->link_new_state == BOND_LINK_NOCHANGE)
|
||||||
return;
|
return;
|
||||||
|
|
||||||
slave->link = slave->link_new_state;
|
slave->link = slave->link_new_state;
|
||||||
|
@ -313,7 +313,7 @@ static int fq_init(struct fq *fq, int flows_cnt)
|
|||||||
fq->limit = 8192;
|
fq->limit = 8192;
|
||||||
fq->memory_limit = 16 << 20; /* 16 MBytes */
|
fq->memory_limit = 16 << 20; /* 16 MBytes */
|
||||||
|
|
||||||
fq->flows = kcalloc(fq->flows_cnt, sizeof(fq->flows[0]), GFP_KERNEL);
|
fq->flows = kvcalloc(fq->flows_cnt, sizeof(fq->flows[0]), GFP_KERNEL);
|
||||||
if (!fq->flows)
|
if (!fq->flows)
|
||||||
return -ENOMEM;
|
return -ENOMEM;
|
||||||
|
|
||||||
@ -331,7 +331,7 @@ static void fq_reset(struct fq *fq,
|
|||||||
for (i = 0; i < fq->flows_cnt; i++)
|
for (i = 0; i < fq->flows_cnt; i++)
|
||||||
fq_flow_reset(fq, &fq->flows[i], free_func);
|
fq_flow_reset(fq, &fq->flows[i], free_func);
|
||||||
|
|
||||||
kfree(fq->flows);
|
kvfree(fq->flows);
|
||||||
fq->flows = NULL;
|
fq->flows = NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -439,8 +439,8 @@ static inline int neigh_event_send(struct neighbour *neigh, struct sk_buff *skb)
|
|||||||
{
|
{
|
||||||
unsigned long now = jiffies;
|
unsigned long now = jiffies;
|
||||||
|
|
||||||
if (neigh->used != now)
|
if (READ_ONCE(neigh->used) != now)
|
||||||
neigh->used = now;
|
WRITE_ONCE(neigh->used, now);
|
||||||
if (!(neigh->nud_state&(NUD_CONNECTED|NUD_DELAY|NUD_PROBE)))
|
if (!(neigh->nud_state&(NUD_CONNECTED|NUD_DELAY|NUD_PROBE)))
|
||||||
return __neigh_event_send(neigh, skb);
|
return __neigh_event_send(neigh, skb);
|
||||||
return 0;
|
return 0;
|
||||||
|
@ -820,7 +820,8 @@ struct nft_expr_ops {
|
|||||||
*/
|
*/
|
||||||
struct nft_expr {
|
struct nft_expr {
|
||||||
const struct nft_expr_ops *ops;
|
const struct nft_expr_ops *ops;
|
||||||
unsigned char data[];
|
unsigned char data[]
|
||||||
|
__attribute__((aligned(__alignof__(u64))));
|
||||||
};
|
};
|
||||||
|
|
||||||
static inline void *nft_expr_priv(const struct nft_expr *expr)
|
static inline void *nft_expr_priv(const struct nft_expr *expr)
|
||||||
|
@ -15,6 +15,7 @@
|
|||||||
#include <linux/mutex.h>
|
#include <linux/mutex.h>
|
||||||
#include <linux/rwsem.h>
|
#include <linux/rwsem.h>
|
||||||
#include <linux/atomic.h>
|
#include <linux/atomic.h>
|
||||||
|
#include <linux/hashtable.h>
|
||||||
#include <net/gen_stats.h>
|
#include <net/gen_stats.h>
|
||||||
#include <net/rtnetlink.h>
|
#include <net/rtnetlink.h>
|
||||||
#include <net/flow_offload.h>
|
#include <net/flow_offload.h>
|
||||||
@ -362,6 +363,7 @@ struct tcf_proto {
|
|||||||
bool deleting;
|
bool deleting;
|
||||||
refcount_t refcnt;
|
refcount_t refcnt;
|
||||||
struct rcu_head rcu;
|
struct rcu_head rcu;
|
||||||
|
struct hlist_node destroy_ht_node;
|
||||||
};
|
};
|
||||||
|
|
||||||
struct qdisc_skb_cb {
|
struct qdisc_skb_cb {
|
||||||
@ -414,6 +416,8 @@ struct tcf_block {
|
|||||||
struct list_head filter_chain_list;
|
struct list_head filter_chain_list;
|
||||||
} chain0;
|
} chain0;
|
||||||
struct rcu_head rcu;
|
struct rcu_head rcu;
|
||||||
|
DECLARE_HASHTABLE(proto_destroy_ht, 7);
|
||||||
|
struct mutex proto_destroy_lock; /* Lock for proto_destroy hashtable. */
|
||||||
};
|
};
|
||||||
|
|
||||||
#ifdef CONFIG_PROVE_LOCKING
|
#ifdef CONFIG_PROVE_LOCKING
|
||||||
|
@ -2342,7 +2342,7 @@ static inline ktime_t sock_read_timestamp(struct sock *sk)
|
|||||||
|
|
||||||
return kt;
|
return kt;
|
||||||
#else
|
#else
|
||||||
return sk->sk_stamp;
|
return READ_ONCE(sk->sk_stamp);
|
||||||
#endif
|
#endif
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -2353,7 +2353,7 @@ static inline void sock_write_timestamp(struct sock *sk, ktime_t kt)
|
|||||||
sk->sk_stamp = kt;
|
sk->sk_stamp = kt;
|
||||||
write_sequnlock(&sk->sk_stamp_seq);
|
write_sequnlock(&sk->sk_stamp_seq);
|
||||||
#else
|
#else
|
||||||
sk->sk_stamp = kt;
|
WRITE_ONCE(sk->sk_stamp, kt);
|
||||||
#endif
|
#endif
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -40,6 +40,7 @@
|
|||||||
#include <linux/socket.h>
|
#include <linux/socket.h>
|
||||||
#include <linux/tcp.h>
|
#include <linux/tcp.h>
|
||||||
#include <linux/skmsg.h>
|
#include <linux/skmsg.h>
|
||||||
|
#include <linux/mutex.h>
|
||||||
#include <linux/netdevice.h>
|
#include <linux/netdevice.h>
|
||||||
#include <linux/rcupdate.h>
|
#include <linux/rcupdate.h>
|
||||||
|
|
||||||
@ -269,6 +270,10 @@ struct tls_context {
|
|||||||
|
|
||||||
bool in_tcp_sendpages;
|
bool in_tcp_sendpages;
|
||||||
bool pending_open_record_frags;
|
bool pending_open_record_frags;
|
||||||
|
|
||||||
|
struct mutex tx_lock; /* protects partially_sent_* fields and
|
||||||
|
* per-type TX fields
|
||||||
|
*/
|
||||||
unsigned long flags;
|
unsigned long flags;
|
||||||
|
|
||||||
/* cache cold stuff */
|
/* cache cold stuff */
|
||||||
|
@ -1,4 +1,4 @@
|
|||||||
/* SPDX-License-Identifier: ((GPL-2.0 WITH Linux-syscall-note) OR BSD-3-Clause) */
|
/* SPDX-License-Identifier: ((GPL-2.0-only WITH Linux-syscall-note) OR BSD-3-Clause) */
|
||||||
/*
|
/*
|
||||||
* linux/can.h
|
* linux/can.h
|
||||||
*
|
*
|
||||||
|
@ -1,4 +1,4 @@
|
|||||||
/* SPDX-License-Identifier: ((GPL-2.0 WITH Linux-syscall-note) OR BSD-3-Clause) */
|
/* SPDX-License-Identifier: ((GPL-2.0-only WITH Linux-syscall-note) OR BSD-3-Clause) */
|
||||||
/*
|
/*
|
||||||
* linux/can/bcm.h
|
* linux/can/bcm.h
|
||||||
*
|
*
|
||||||
|
@ -1,4 +1,4 @@
|
|||||||
/* SPDX-License-Identifier: ((GPL-2.0 WITH Linux-syscall-note) OR BSD-3-Clause) */
|
/* SPDX-License-Identifier: ((GPL-2.0-only WITH Linux-syscall-note) OR BSD-3-Clause) */
|
||||||
/*
|
/*
|
||||||
* linux/can/error.h
|
* linux/can/error.h
|
||||||
*
|
*
|
||||||
|
@ -1,4 +1,4 @@
|
|||||||
/* SPDX-License-Identifier: ((GPL-2.0 WITH Linux-syscall-note) OR BSD-3-Clause) */
|
/* SPDX-License-Identifier: ((GPL-2.0-only WITH Linux-syscall-note) OR BSD-3-Clause) */
|
||||||
/*
|
/*
|
||||||
* linux/can/gw.h
|
* linux/can/gw.h
|
||||||
*
|
*
|
||||||
|
@ -1,4 +1,4 @@
|
|||||||
/* SPDX-License-Identifier: GPL-2.0 WITH Linux-syscall-note */
|
/* SPDX-License-Identifier: GPL-2.0-only WITH Linux-syscall-note */
|
||||||
/*
|
/*
|
||||||
* j1939.h
|
* j1939.h
|
||||||
*
|
*
|
||||||
|
@ -1,4 +1,4 @@
|
|||||||
/* SPDX-License-Identifier: GPL-2.0 WITH Linux-syscall-note */
|
/* SPDX-License-Identifier: GPL-2.0-only WITH Linux-syscall-note */
|
||||||
/*
|
/*
|
||||||
* linux/can/netlink.h
|
* linux/can/netlink.h
|
||||||
*
|
*
|
||||||
|
@ -1,4 +1,4 @@
|
|||||||
/* SPDX-License-Identifier: ((GPL-2.0 WITH Linux-syscall-note) OR BSD-3-Clause) */
|
/* SPDX-License-Identifier: ((GPL-2.0-only WITH Linux-syscall-note) OR BSD-3-Clause) */
|
||||||
/*
|
/*
|
||||||
* linux/can/raw.h
|
* linux/can/raw.h
|
||||||
*
|
*
|
||||||
|
@ -1,4 +1,4 @@
|
|||||||
/* SPDX-License-Identifier: GPL-2.0 WITH Linux-syscall-note */
|
/* SPDX-License-Identifier: GPL-2.0-only WITH Linux-syscall-note */
|
||||||
#ifndef _UAPI_CAN_VXCAN_H
|
#ifndef _UAPI_CAN_VXCAN_H
|
||||||
#define _UAPI_CAN_VXCAN_H
|
#define _UAPI_CAN_VXCAN_H
|
||||||
|
|
||||||
|
@ -1311,12 +1311,12 @@ static bool sysctl_is_valid_access(int off, int size, enum bpf_access_type type,
|
|||||||
return false;
|
return false;
|
||||||
|
|
||||||
switch (off) {
|
switch (off) {
|
||||||
case offsetof(struct bpf_sysctl, write):
|
case bpf_ctx_range(struct bpf_sysctl, write):
|
||||||
if (type != BPF_READ)
|
if (type != BPF_READ)
|
||||||
return false;
|
return false;
|
||||||
bpf_ctx_record_field_size(info, size_default);
|
bpf_ctx_record_field_size(info, size_default);
|
||||||
return bpf_ctx_narrow_access_ok(off, size, size_default);
|
return bpf_ctx_narrow_access_ok(off, size, size_default);
|
||||||
case offsetof(struct bpf_sysctl, file_pos):
|
case bpf_ctx_range(struct bpf_sysctl, file_pos):
|
||||||
if (type == BPF_READ) {
|
if (type == BPF_READ) {
|
||||||
bpf_ctx_record_field_size(info, size_default);
|
bpf_ctx_record_field_size(info, size_default);
|
||||||
return bpf_ctx_narrow_access_ok(off, size, size_default);
|
return bpf_ctx_narrow_access_ok(off, size, size_default);
|
||||||
|
@ -126,7 +126,7 @@ static struct bpf_map *find_and_alloc_map(union bpf_attr *attr)
|
|||||||
return map;
|
return map;
|
||||||
}
|
}
|
||||||
|
|
||||||
void *bpf_map_area_alloc(size_t size, int numa_node)
|
void *bpf_map_area_alloc(u64 size, int numa_node)
|
||||||
{
|
{
|
||||||
/* We really just want to fail instead of triggering OOM killer
|
/* We really just want to fail instead of triggering OOM killer
|
||||||
* under memory pressure, therefore we set __GFP_NORETRY to kmalloc,
|
* under memory pressure, therefore we set __GFP_NORETRY to kmalloc,
|
||||||
@ -141,6 +141,9 @@ void *bpf_map_area_alloc(size_t size, int numa_node)
|
|||||||
const gfp_t flags = __GFP_NOWARN | __GFP_ZERO;
|
const gfp_t flags = __GFP_NOWARN | __GFP_ZERO;
|
||||||
void *area;
|
void *area;
|
||||||
|
|
||||||
|
if (size >= SIZE_MAX)
|
||||||
|
return NULL;
|
||||||
|
|
||||||
if (size <= (PAGE_SIZE << PAGE_ALLOC_COSTLY_ORDER)) {
|
if (size <= (PAGE_SIZE << PAGE_ALLOC_COSTLY_ORDER)) {
|
||||||
area = kmalloc_node(size, GFP_USER | __GFP_NORETRY | flags,
|
area = kmalloc_node(size, GFP_USER | __GFP_NORETRY | flags,
|
||||||
numa_node);
|
numa_node);
|
||||||
@ -197,7 +200,7 @@ static void bpf_uncharge_memlock(struct user_struct *user, u32 pages)
|
|||||||
atomic_long_sub(pages, &user->locked_vm);
|
atomic_long_sub(pages, &user->locked_vm);
|
||||||
}
|
}
|
||||||
|
|
||||||
int bpf_map_charge_init(struct bpf_map_memory *mem, size_t size)
|
int bpf_map_charge_init(struct bpf_map_memory *mem, u64 size)
|
||||||
{
|
{
|
||||||
u32 pages = round_up(size, PAGE_SIZE) >> PAGE_SHIFT;
|
u32 pages = round_up(size, PAGE_SIZE) >> PAGE_SHIFT;
|
||||||
struct user_struct *user;
|
struct user_struct *user;
|
||||||
|
@ -20,7 +20,6 @@ static unsigned int
|
|||||||
ebt_dnat_tg(struct sk_buff *skb, const struct xt_action_param *par)
|
ebt_dnat_tg(struct sk_buff *skb, const struct xt_action_param *par)
|
||||||
{
|
{
|
||||||
const struct ebt_nat_info *info = par->targinfo;
|
const struct ebt_nat_info *info = par->targinfo;
|
||||||
struct net_device *dev;
|
|
||||||
|
|
||||||
if (skb_ensure_writable(skb, ETH_ALEN))
|
if (skb_ensure_writable(skb, ETH_ALEN))
|
||||||
return EBT_DROP;
|
return EBT_DROP;
|
||||||
@ -33,10 +32,22 @@ ebt_dnat_tg(struct sk_buff *skb, const struct xt_action_param *par)
|
|||||||
else
|
else
|
||||||
skb->pkt_type = PACKET_MULTICAST;
|
skb->pkt_type = PACKET_MULTICAST;
|
||||||
} else {
|
} else {
|
||||||
if (xt_hooknum(par) != NF_BR_BROUTING)
|
const struct net_device *dev;
|
||||||
dev = br_port_get_rcu(xt_in(par))->br->dev;
|
|
||||||
else
|
switch (xt_hooknum(par)) {
|
||||||
|
case NF_BR_BROUTING:
|
||||||
dev = xt_in(par);
|
dev = xt_in(par);
|
||||||
|
break;
|
||||||
|
case NF_BR_PRE_ROUTING:
|
||||||
|
dev = br_port_get_rcu(xt_in(par))->br->dev;
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
dev = NULL;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!dev) /* NF_BR_LOCAL_OUT */
|
||||||
|
return info->target;
|
||||||
|
|
||||||
if (ether_addr_equal(info->mac, dev->dev_addr))
|
if (ether_addr_equal(info->mac, dev->dev_addr))
|
||||||
skb->pkt_type = PACKET_HOST;
|
skb->pkt_type = PACKET_HOST;
|
||||||
|
@ -580,6 +580,7 @@ static int j1939_sk_release(struct socket *sock)
|
|||||||
j1939_netdev_stop(priv);
|
j1939_netdev_stop(priv);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
kfree(jsk->filters);
|
||||||
sock_orphan(sk);
|
sock_orphan(sk);
|
||||||
sock->sk = NULL;
|
sock->sk = NULL;
|
||||||
|
|
||||||
@ -909,8 +910,10 @@ void j1939_sk_errqueue(struct j1939_session *session,
|
|||||||
memset(serr, 0, sizeof(*serr));
|
memset(serr, 0, sizeof(*serr));
|
||||||
switch (type) {
|
switch (type) {
|
||||||
case J1939_ERRQUEUE_ACK:
|
case J1939_ERRQUEUE_ACK:
|
||||||
if (!(sk->sk_tsflags & SOF_TIMESTAMPING_TX_ACK))
|
if (!(sk->sk_tsflags & SOF_TIMESTAMPING_TX_ACK)) {
|
||||||
|
kfree_skb(skb);
|
||||||
return;
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
serr->ee.ee_errno = ENOMSG;
|
serr->ee.ee_errno = ENOMSG;
|
||||||
serr->ee.ee_origin = SO_EE_ORIGIN_TIMESTAMPING;
|
serr->ee.ee_origin = SO_EE_ORIGIN_TIMESTAMPING;
|
||||||
@ -918,8 +921,10 @@ void j1939_sk_errqueue(struct j1939_session *session,
|
|||||||
state = "ACK";
|
state = "ACK";
|
||||||
break;
|
break;
|
||||||
case J1939_ERRQUEUE_SCHED:
|
case J1939_ERRQUEUE_SCHED:
|
||||||
if (!(sk->sk_tsflags & SOF_TIMESTAMPING_TX_SCHED))
|
if (!(sk->sk_tsflags & SOF_TIMESTAMPING_TX_SCHED)) {
|
||||||
|
kfree_skb(skb);
|
||||||
return;
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
serr->ee.ee_errno = ENOMSG;
|
serr->ee.ee_errno = ENOMSG;
|
||||||
serr->ee.ee_origin = SO_EE_ORIGIN_TIMESTAMPING;
|
serr->ee.ee_origin = SO_EE_ORIGIN_TIMESTAMPING;
|
||||||
|
@ -1273,9 +1273,27 @@ j1939_xtp_rx_abort(struct j1939_priv *priv, struct sk_buff *skb,
|
|||||||
static void
|
static void
|
||||||
j1939_xtp_rx_eoma_one(struct j1939_session *session, struct sk_buff *skb)
|
j1939_xtp_rx_eoma_one(struct j1939_session *session, struct sk_buff *skb)
|
||||||
{
|
{
|
||||||
|
struct j1939_sk_buff_cb *skcb = j1939_skb_to_cb(skb);
|
||||||
|
const u8 *dat;
|
||||||
|
int len;
|
||||||
|
|
||||||
if (j1939_xtp_rx_cmd_bad_pgn(session, skb))
|
if (j1939_xtp_rx_cmd_bad_pgn(session, skb))
|
||||||
return;
|
return;
|
||||||
|
|
||||||
|
dat = skb->data;
|
||||||
|
|
||||||
|
if (skcb->addr.type == J1939_ETP)
|
||||||
|
len = j1939_etp_ctl_to_size(dat);
|
||||||
|
else
|
||||||
|
len = j1939_tp_ctl_to_size(dat);
|
||||||
|
|
||||||
|
if (session->total_message_size != len) {
|
||||||
|
netdev_warn_once(session->priv->ndev,
|
||||||
|
"%s: 0x%p: Incorrect size. Expected: %i; got: %i.\n",
|
||||||
|
__func__, session, session->total_message_size,
|
||||||
|
len);
|
||||||
|
}
|
||||||
|
|
||||||
netdev_dbg(session->priv->ndev, "%s: 0x%p\n", __func__, session);
|
netdev_dbg(session->priv->ndev, "%s: 0x%p\n", __func__, session);
|
||||||
|
|
||||||
session->pkt.tx_acked = session->pkt.total;
|
session->pkt.tx_acked = session->pkt.total;
|
||||||
@ -1432,7 +1450,7 @@ j1939_session *j1939_session_fresh_new(struct j1939_priv *priv,
|
|||||||
skcb = j1939_skb_to_cb(skb);
|
skcb = j1939_skb_to_cb(skb);
|
||||||
memcpy(skcb, rel_skcb, sizeof(*skcb));
|
memcpy(skcb, rel_skcb, sizeof(*skcb));
|
||||||
|
|
||||||
session = j1939_session_new(priv, skb, skb->len);
|
session = j1939_session_new(priv, skb, size);
|
||||||
if (!session) {
|
if (!session) {
|
||||||
kfree_skb(skb);
|
kfree_skb(skb);
|
||||||
return NULL;
|
return NULL;
|
||||||
|
@ -270,18 +270,28 @@ void sk_msg_trim(struct sock *sk, struct sk_msg *msg, int len)
|
|||||||
|
|
||||||
msg->sg.data[i].length -= trim;
|
msg->sg.data[i].length -= trim;
|
||||||
sk_mem_uncharge(sk, trim);
|
sk_mem_uncharge(sk, trim);
|
||||||
|
/* Adjust copybreak if it falls into the trimmed part of last buf */
|
||||||
|
if (msg->sg.curr == i && msg->sg.copybreak > msg->sg.data[i].length)
|
||||||
|
msg->sg.copybreak = msg->sg.data[i].length;
|
||||||
out:
|
out:
|
||||||
/* If we trim data before curr pointer update copybreak and current
|
sk_msg_iter_var_next(i);
|
||||||
* so that any future copy operations start at new copy location.
|
msg->sg.end = i;
|
||||||
|
|
||||||
|
/* If we trim data a full sg elem before curr pointer update
|
||||||
|
* copybreak and current so that any future copy operations
|
||||||
|
* start at new copy location.
|
||||||
* However trimed data that has not yet been used in a copy op
|
* However trimed data that has not yet been used in a copy op
|
||||||
* does not require an update.
|
* does not require an update.
|
||||||
*/
|
*/
|
||||||
if (msg->sg.curr >= i) {
|
if (!msg->sg.size) {
|
||||||
|
msg->sg.curr = msg->sg.start;
|
||||||
|
msg->sg.copybreak = 0;
|
||||||
|
} else if (sk_msg_iter_dist(msg->sg.start, msg->sg.curr) >=
|
||||||
|
sk_msg_iter_dist(msg->sg.start, msg->sg.end)) {
|
||||||
|
sk_msg_iter_var_prev(i);
|
||||||
msg->sg.curr = i;
|
msg->sg.curr = i;
|
||||||
msg->sg.copybreak = msg->sg.data[i].length;
|
msg->sg.copybreak = msg->sg.data[i].length;
|
||||||
}
|
}
|
||||||
sk_msg_iter_var_next(i);
|
|
||||||
msg->sg.end = i;
|
|
||||||
}
|
}
|
||||||
EXPORT_SYMBOL_GPL(sk_msg_trim);
|
EXPORT_SYMBOL_GPL(sk_msg_trim);
|
||||||
|
|
||||||
|
@ -416,7 +416,7 @@ struct sock *dccp_v4_request_recv_sock(const struct sock *sk,
|
|||||||
RCU_INIT_POINTER(newinet->inet_opt, rcu_dereference(ireq->ireq_opt));
|
RCU_INIT_POINTER(newinet->inet_opt, rcu_dereference(ireq->ireq_opt));
|
||||||
newinet->mc_index = inet_iif(skb);
|
newinet->mc_index = inet_iif(skb);
|
||||||
newinet->mc_ttl = ip_hdr(skb)->ttl;
|
newinet->mc_ttl = ip_hdr(skb)->ttl;
|
||||||
newinet->inet_id = jiffies;
|
newinet->inet_id = prandom_u32();
|
||||||
|
|
||||||
if (dst == NULL && (dst = inet_csk_route_child_sock(sk, newsk, req)) == NULL)
|
if (dst == NULL && (dst = inet_csk_route_child_sock(sk, newsk, req)) == NULL)
|
||||||
goto put_and_exit;
|
goto put_and_exit;
|
||||||
|
@ -1814,8 +1814,8 @@ int fib_sync_down_addr(struct net_device *dev, __be32 local)
|
|||||||
int ret = 0;
|
int ret = 0;
|
||||||
unsigned int hash = fib_laddr_hashfn(local);
|
unsigned int hash = fib_laddr_hashfn(local);
|
||||||
struct hlist_head *head = &fib_info_laddrhash[hash];
|
struct hlist_head *head = &fib_info_laddrhash[hash];
|
||||||
|
int tb_id = l3mdev_fib_table(dev) ? : RT_TABLE_MAIN;
|
||||||
struct net *net = dev_net(dev);
|
struct net *net = dev_net(dev);
|
||||||
int tb_id = l3mdev_fib_table(dev);
|
|
||||||
struct fib_info *fi;
|
struct fib_info *fi;
|
||||||
|
|
||||||
if (!fib_info_laddrhash || local == 0)
|
if (!fib_info_laddrhash || local == 0)
|
||||||
|
@ -621,6 +621,7 @@ static void rt6_probe(struct fib6_nh *fib6_nh)
|
|||||||
{
|
{
|
||||||
struct __rt6_probe_work *work = NULL;
|
struct __rt6_probe_work *work = NULL;
|
||||||
const struct in6_addr *nh_gw;
|
const struct in6_addr *nh_gw;
|
||||||
|
unsigned long last_probe;
|
||||||
struct neighbour *neigh;
|
struct neighbour *neigh;
|
||||||
struct net_device *dev;
|
struct net_device *dev;
|
||||||
struct inet6_dev *idev;
|
struct inet6_dev *idev;
|
||||||
@ -639,6 +640,7 @@ static void rt6_probe(struct fib6_nh *fib6_nh)
|
|||||||
nh_gw = &fib6_nh->fib_nh_gw6;
|
nh_gw = &fib6_nh->fib_nh_gw6;
|
||||||
dev = fib6_nh->fib_nh_dev;
|
dev = fib6_nh->fib_nh_dev;
|
||||||
rcu_read_lock_bh();
|
rcu_read_lock_bh();
|
||||||
|
last_probe = READ_ONCE(fib6_nh->last_probe);
|
||||||
idev = __in6_dev_get(dev);
|
idev = __in6_dev_get(dev);
|
||||||
neigh = __ipv6_neigh_lookup_noref(dev, nh_gw);
|
neigh = __ipv6_neigh_lookup_noref(dev, nh_gw);
|
||||||
if (neigh) {
|
if (neigh) {
|
||||||
@ -654,13 +656,15 @@ static void rt6_probe(struct fib6_nh *fib6_nh)
|
|||||||
__neigh_set_probe_once(neigh);
|
__neigh_set_probe_once(neigh);
|
||||||
}
|
}
|
||||||
write_unlock(&neigh->lock);
|
write_unlock(&neigh->lock);
|
||||||
} else if (time_after(jiffies, fib6_nh->last_probe +
|
} else if (time_after(jiffies, last_probe +
|
||||||
idev->cnf.rtr_probe_interval)) {
|
idev->cnf.rtr_probe_interval)) {
|
||||||
work = kmalloc(sizeof(*work), GFP_ATOMIC);
|
work = kmalloc(sizeof(*work), GFP_ATOMIC);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (work) {
|
if (!work || cmpxchg(&fib6_nh->last_probe,
|
||||||
fib6_nh->last_probe = jiffies;
|
last_probe, jiffies) != last_probe) {
|
||||||
|
kfree(work);
|
||||||
|
} else {
|
||||||
INIT_WORK(&work->work, rt6_probe_deferred);
|
INIT_WORK(&work->work, rt6_probe_deferred);
|
||||||
work->target = *nh_gw;
|
work->target = *nh_gw;
|
||||||
dev_hold(dev);
|
dev_hold(dev);
|
||||||
@ -3383,6 +3387,9 @@ int fib6_nh_init(struct net *net, struct fib6_nh *fib6_nh,
|
|||||||
int err;
|
int err;
|
||||||
|
|
||||||
fib6_nh->fib_nh_family = AF_INET6;
|
fib6_nh->fib_nh_family = AF_INET6;
|
||||||
|
#ifdef CONFIG_IPV6_ROUTER_PREF
|
||||||
|
fib6_nh->last_probe = jiffies;
|
||||||
|
#endif
|
||||||
|
|
||||||
err = -ENODEV;
|
err = -ENODEV;
|
||||||
if (cfg->fc_ifindex) {
|
if (cfg->fc_ifindex) {
|
||||||
|
@ -1292,8 +1292,8 @@ int ieee80211_register_hw(struct ieee80211_hw *hw)
|
|||||||
ieee80211_remove_interfaces(local);
|
ieee80211_remove_interfaces(local);
|
||||||
fail_rate:
|
fail_rate:
|
||||||
rtnl_unlock();
|
rtnl_unlock();
|
||||||
ieee80211_led_exit(local);
|
|
||||||
fail_flows:
|
fail_flows:
|
||||||
|
ieee80211_led_exit(local);
|
||||||
destroy_workqueue(local->workqueue);
|
destroy_workqueue(local->workqueue);
|
||||||
fail_workqueue:
|
fail_workqueue:
|
||||||
wiphy_unregister(local->hw.wiphy);
|
wiphy_unregister(local->hw.wiphy);
|
||||||
|
@ -2457,7 +2457,8 @@ unsigned long ieee80211_sta_last_active(struct sta_info *sta)
|
|||||||
{
|
{
|
||||||
struct ieee80211_sta_rx_stats *stats = sta_get_last_rx_stats(sta);
|
struct ieee80211_sta_rx_stats *stats = sta_get_last_rx_stats(sta);
|
||||||
|
|
||||||
if (time_after(stats->last_rx, sta->status_stats.last_ack))
|
if (!sta->status_stats.last_ack ||
|
||||||
|
time_after(stats->last_rx, sta->status_stats.last_ack))
|
||||||
return stats->last_rx;
|
return stats->last_rx;
|
||||||
return sta->status_stats.last_ack;
|
return sta->status_stats.last_ack;
|
||||||
}
|
}
|
||||||
|
@ -296,7 +296,8 @@ ip_set_get_ipaddr4(struct nlattr *nla, __be32 *ipaddr)
|
|||||||
|
|
||||||
if (unlikely(!flag_nested(nla)))
|
if (unlikely(!flag_nested(nla)))
|
||||||
return -IPSET_ERR_PROTOCOL;
|
return -IPSET_ERR_PROTOCOL;
|
||||||
if (nla_parse_nested_deprecated(tb, IPSET_ATTR_IPADDR_MAX, nla, ipaddr_policy, NULL))
|
if (nla_parse_nested(tb, IPSET_ATTR_IPADDR_MAX, nla,
|
||||||
|
ipaddr_policy, NULL))
|
||||||
return -IPSET_ERR_PROTOCOL;
|
return -IPSET_ERR_PROTOCOL;
|
||||||
if (unlikely(!ip_set_attr_netorder(tb, IPSET_ATTR_IPADDR_IPV4)))
|
if (unlikely(!ip_set_attr_netorder(tb, IPSET_ATTR_IPADDR_IPV4)))
|
||||||
return -IPSET_ERR_PROTOCOL;
|
return -IPSET_ERR_PROTOCOL;
|
||||||
@ -314,7 +315,8 @@ ip_set_get_ipaddr6(struct nlattr *nla, union nf_inet_addr *ipaddr)
|
|||||||
if (unlikely(!flag_nested(nla)))
|
if (unlikely(!flag_nested(nla)))
|
||||||
return -IPSET_ERR_PROTOCOL;
|
return -IPSET_ERR_PROTOCOL;
|
||||||
|
|
||||||
if (nla_parse_nested_deprecated(tb, IPSET_ATTR_IPADDR_MAX, nla, ipaddr_policy, NULL))
|
if (nla_parse_nested(tb, IPSET_ATTR_IPADDR_MAX, nla,
|
||||||
|
ipaddr_policy, NULL))
|
||||||
return -IPSET_ERR_PROTOCOL;
|
return -IPSET_ERR_PROTOCOL;
|
||||||
if (unlikely(!ip_set_attr_netorder(tb, IPSET_ATTR_IPADDR_IPV6)))
|
if (unlikely(!ip_set_attr_netorder(tb, IPSET_ATTR_IPADDR_IPV6)))
|
||||||
return -IPSET_ERR_PROTOCOL;
|
return -IPSET_ERR_PROTOCOL;
|
||||||
@ -934,7 +936,8 @@ static int ip_set_create(struct net *net, struct sock *ctnl,
|
|||||||
|
|
||||||
/* Without holding any locks, create private part. */
|
/* Without holding any locks, create private part. */
|
||||||
if (attr[IPSET_ATTR_DATA] &&
|
if (attr[IPSET_ATTR_DATA] &&
|
||||||
nla_parse_nested_deprecated(tb, IPSET_ATTR_CREATE_MAX, attr[IPSET_ATTR_DATA], set->type->create_policy, NULL)) {
|
nla_parse_nested(tb, IPSET_ATTR_CREATE_MAX, attr[IPSET_ATTR_DATA],
|
||||||
|
set->type->create_policy, NULL)) {
|
||||||
ret = -IPSET_ERR_PROTOCOL;
|
ret = -IPSET_ERR_PROTOCOL;
|
||||||
goto put_out;
|
goto put_out;
|
||||||
}
|
}
|
||||||
@ -1281,6 +1284,14 @@ dump_attrs(struct nlmsghdr *nlh)
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static const struct nla_policy
|
||||||
|
ip_set_dump_policy[IPSET_ATTR_CMD_MAX + 1] = {
|
||||||
|
[IPSET_ATTR_PROTOCOL] = { .type = NLA_U8 },
|
||||||
|
[IPSET_ATTR_SETNAME] = { .type = NLA_NUL_STRING,
|
||||||
|
.len = IPSET_MAXNAMELEN - 1 },
|
||||||
|
[IPSET_ATTR_FLAGS] = { .type = NLA_U32 },
|
||||||
|
};
|
||||||
|
|
||||||
static int
|
static int
|
||||||
dump_init(struct netlink_callback *cb, struct ip_set_net *inst)
|
dump_init(struct netlink_callback *cb, struct ip_set_net *inst)
|
||||||
{
|
{
|
||||||
@ -1292,9 +1303,9 @@ dump_init(struct netlink_callback *cb, struct ip_set_net *inst)
|
|||||||
ip_set_id_t index;
|
ip_set_id_t index;
|
||||||
int ret;
|
int ret;
|
||||||
|
|
||||||
ret = nla_parse_deprecated(cda, IPSET_ATTR_CMD_MAX, attr,
|
ret = nla_parse(cda, IPSET_ATTR_CMD_MAX, attr,
|
||||||
nlh->nlmsg_len - min_len,
|
nlh->nlmsg_len - min_len,
|
||||||
ip_set_setname_policy, NULL);
|
ip_set_dump_policy, NULL);
|
||||||
if (ret)
|
if (ret)
|
||||||
return ret;
|
return ret;
|
||||||
|
|
||||||
@ -1543,9 +1554,9 @@ call_ad(struct sock *ctnl, struct sk_buff *skb, struct ip_set *set,
|
|||||||
memcpy(&errmsg->msg, nlh, nlh->nlmsg_len);
|
memcpy(&errmsg->msg, nlh, nlh->nlmsg_len);
|
||||||
cmdattr = (void *)&errmsg->msg + min_len;
|
cmdattr = (void *)&errmsg->msg + min_len;
|
||||||
|
|
||||||
ret = nla_parse_deprecated(cda, IPSET_ATTR_CMD_MAX, cmdattr,
|
ret = nla_parse(cda, IPSET_ATTR_CMD_MAX, cmdattr,
|
||||||
nlh->nlmsg_len - min_len,
|
nlh->nlmsg_len - min_len, ip_set_adt_policy,
|
||||||
ip_set_adt_policy, NULL);
|
NULL);
|
||||||
|
|
||||||
if (ret) {
|
if (ret) {
|
||||||
nlmsg_free(skb2);
|
nlmsg_free(skb2);
|
||||||
@ -1596,7 +1607,9 @@ static int ip_set_ad(struct net *net, struct sock *ctnl,
|
|||||||
|
|
||||||
use_lineno = !!attr[IPSET_ATTR_LINENO];
|
use_lineno = !!attr[IPSET_ATTR_LINENO];
|
||||||
if (attr[IPSET_ATTR_DATA]) {
|
if (attr[IPSET_ATTR_DATA]) {
|
||||||
if (nla_parse_nested_deprecated(tb, IPSET_ATTR_ADT_MAX, attr[IPSET_ATTR_DATA], set->type->adt_policy, NULL))
|
if (nla_parse_nested(tb, IPSET_ATTR_ADT_MAX,
|
||||||
|
attr[IPSET_ATTR_DATA],
|
||||||
|
set->type->adt_policy, NULL))
|
||||||
return -IPSET_ERR_PROTOCOL;
|
return -IPSET_ERR_PROTOCOL;
|
||||||
ret = call_ad(ctnl, skb, set, tb, adt, flags,
|
ret = call_ad(ctnl, skb, set, tb, adt, flags,
|
||||||
use_lineno);
|
use_lineno);
|
||||||
@ -1606,7 +1619,8 @@ static int ip_set_ad(struct net *net, struct sock *ctnl,
|
|||||||
nla_for_each_nested(nla, attr[IPSET_ATTR_ADT], nla_rem) {
|
nla_for_each_nested(nla, attr[IPSET_ATTR_ADT], nla_rem) {
|
||||||
if (nla_type(nla) != IPSET_ATTR_DATA ||
|
if (nla_type(nla) != IPSET_ATTR_DATA ||
|
||||||
!flag_nested(nla) ||
|
!flag_nested(nla) ||
|
||||||
nla_parse_nested_deprecated(tb, IPSET_ATTR_ADT_MAX, nla, set->type->adt_policy, NULL))
|
nla_parse_nested(tb, IPSET_ATTR_ADT_MAX, nla,
|
||||||
|
set->type->adt_policy, NULL))
|
||||||
return -IPSET_ERR_PROTOCOL;
|
return -IPSET_ERR_PROTOCOL;
|
||||||
ret = call_ad(ctnl, skb, set, tb, adt,
|
ret = call_ad(ctnl, skb, set, tb, adt,
|
||||||
flags, use_lineno);
|
flags, use_lineno);
|
||||||
@ -1655,7 +1669,8 @@ static int ip_set_utest(struct net *net, struct sock *ctnl, struct sk_buff *skb,
|
|||||||
if (!set)
|
if (!set)
|
||||||
return -ENOENT;
|
return -ENOENT;
|
||||||
|
|
||||||
if (nla_parse_nested_deprecated(tb, IPSET_ATTR_ADT_MAX, attr[IPSET_ATTR_DATA], set->type->adt_policy, NULL))
|
if (nla_parse_nested(tb, IPSET_ATTR_ADT_MAX, attr[IPSET_ATTR_DATA],
|
||||||
|
set->type->adt_policy, NULL))
|
||||||
return -IPSET_ERR_PROTOCOL;
|
return -IPSET_ERR_PROTOCOL;
|
||||||
|
|
||||||
rcu_read_lock_bh();
|
rcu_read_lock_bh();
|
||||||
@ -1961,7 +1976,7 @@ static const struct nfnl_callback ip_set_netlink_subsys_cb[IPSET_MSG_MAX] = {
|
|||||||
[IPSET_CMD_LIST] = {
|
[IPSET_CMD_LIST] = {
|
||||||
.call = ip_set_dump,
|
.call = ip_set_dump,
|
||||||
.attr_count = IPSET_ATTR_CMD_MAX,
|
.attr_count = IPSET_ATTR_CMD_MAX,
|
||||||
.policy = ip_set_setname_policy,
|
.policy = ip_set_dump_policy,
|
||||||
},
|
},
|
||||||
[IPSET_CMD_SAVE] = {
|
[IPSET_CMD_SAVE] = {
|
||||||
.call = ip_set_dump,
|
.call = ip_set_dump,
|
||||||
@ -2069,8 +2084,9 @@ ip_set_sockfn_get(struct sock *sk, int optval, void __user *user, int *len)
|
|||||||
}
|
}
|
||||||
|
|
||||||
req_version->version = IPSET_PROTOCOL;
|
req_version->version = IPSET_PROTOCOL;
|
||||||
ret = copy_to_user(user, req_version,
|
if (copy_to_user(user, req_version,
|
||||||
sizeof(struct ip_set_req_version));
|
sizeof(struct ip_set_req_version)))
|
||||||
|
ret = -EFAULT;
|
||||||
goto done;
|
goto done;
|
||||||
}
|
}
|
||||||
case IP_SET_OP_GET_BYNAME: {
|
case IP_SET_OP_GET_BYNAME: {
|
||||||
@ -2129,7 +2145,8 @@ ip_set_sockfn_get(struct sock *sk, int optval, void __user *user, int *len)
|
|||||||
} /* end of switch(op) */
|
} /* end of switch(op) */
|
||||||
|
|
||||||
copy:
|
copy:
|
||||||
ret = copy_to_user(user, data, copylen);
|
if (copy_to_user(user, data, copylen))
|
||||||
|
ret = -EFAULT;
|
||||||
|
|
||||||
done:
|
done:
|
||||||
vfree(data);
|
vfree(data);
|
||||||
|
@ -209,7 +209,7 @@ hash_ipmac6_kadt(struct ip_set *set, const struct sk_buff *skb,
|
|||||||
(skb_mac_header(skb) + ETH_HLEN) > skb->data)
|
(skb_mac_header(skb) + ETH_HLEN) > skb->data)
|
||||||
return -EINVAL;
|
return -EINVAL;
|
||||||
|
|
||||||
if (opt->flags & IPSET_DIM_ONE_SRC)
|
if (opt->flags & IPSET_DIM_TWO_SRC)
|
||||||
ether_addr_copy(e.ether, eth_hdr(skb)->h_source);
|
ether_addr_copy(e.ether, eth_hdr(skb)->h_source);
|
||||||
else
|
else
|
||||||
ether_addr_copy(e.ether, eth_hdr(skb)->h_dest);
|
ether_addr_copy(e.ether, eth_hdr(skb)->h_dest);
|
||||||
|
@ -368,6 +368,7 @@ static struct ip_set_type hash_net_type __read_mostly = {
|
|||||||
[IPSET_ATTR_IP_TO] = { .type = NLA_NESTED },
|
[IPSET_ATTR_IP_TO] = { .type = NLA_NESTED },
|
||||||
[IPSET_ATTR_CIDR] = { .type = NLA_U8 },
|
[IPSET_ATTR_CIDR] = { .type = NLA_U8 },
|
||||||
[IPSET_ATTR_TIMEOUT] = { .type = NLA_U32 },
|
[IPSET_ATTR_TIMEOUT] = { .type = NLA_U32 },
|
||||||
|
[IPSET_ATTR_LINENO] = { .type = NLA_U32 },
|
||||||
[IPSET_ATTR_CADT_FLAGS] = { .type = NLA_U32 },
|
[IPSET_ATTR_CADT_FLAGS] = { .type = NLA_U32 },
|
||||||
[IPSET_ATTR_BYTES] = { .type = NLA_U64 },
|
[IPSET_ATTR_BYTES] = { .type = NLA_U64 },
|
||||||
[IPSET_ATTR_PACKETS] = { .type = NLA_U64 },
|
[IPSET_ATTR_PACKETS] = { .type = NLA_U64 },
|
||||||
|
@ -476,6 +476,7 @@ static struct ip_set_type hash_netnet_type __read_mostly = {
|
|||||||
[IPSET_ATTR_CIDR] = { .type = NLA_U8 },
|
[IPSET_ATTR_CIDR] = { .type = NLA_U8 },
|
||||||
[IPSET_ATTR_CIDR2] = { .type = NLA_U8 },
|
[IPSET_ATTR_CIDR2] = { .type = NLA_U8 },
|
||||||
[IPSET_ATTR_TIMEOUT] = { .type = NLA_U32 },
|
[IPSET_ATTR_TIMEOUT] = { .type = NLA_U32 },
|
||||||
|
[IPSET_ATTR_LINENO] = { .type = NLA_U32 },
|
||||||
[IPSET_ATTR_CADT_FLAGS] = { .type = NLA_U32 },
|
[IPSET_ATTR_CADT_FLAGS] = { .type = NLA_U32 },
|
||||||
[IPSET_ATTR_BYTES] = { .type = NLA_U64 },
|
[IPSET_ATTR_BYTES] = { .type = NLA_U64 },
|
||||||
[IPSET_ATTR_PACKETS] = { .type = NLA_U64 },
|
[IPSET_ATTR_PACKETS] = { .type = NLA_U64 },
|
||||||
|
@ -1922,6 +1922,7 @@ static int nf_tables_newchain(struct net *net, struct sock *nlsk,
|
|||||||
if (nlh->nlmsg_flags & NLM_F_REPLACE)
|
if (nlh->nlmsg_flags & NLM_F_REPLACE)
|
||||||
return -EOPNOTSUPP;
|
return -EOPNOTSUPP;
|
||||||
|
|
||||||
|
flags |= chain->flags & NFT_BASE_CHAIN;
|
||||||
return nf_tables_updchain(&ctx, genmask, policy, flags);
|
return nf_tables_updchain(&ctx, genmask, policy, flags);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -5143,9 +5144,6 @@ static int nf_tables_updobj(const struct nft_ctx *ctx,
|
|||||||
struct nft_trans *trans;
|
struct nft_trans *trans;
|
||||||
int err;
|
int err;
|
||||||
|
|
||||||
if (!obj->ops->update)
|
|
||||||
return -EOPNOTSUPP;
|
|
||||||
|
|
||||||
trans = nft_trans_alloc(ctx, NFT_MSG_NEWOBJ,
|
trans = nft_trans_alloc(ctx, NFT_MSG_NEWOBJ,
|
||||||
sizeof(struct nft_trans_obj));
|
sizeof(struct nft_trans_obj));
|
||||||
if (!trans)
|
if (!trans)
|
||||||
@ -6499,7 +6497,8 @@ static void nft_obj_commit_update(struct nft_trans *trans)
|
|||||||
obj = nft_trans_obj(trans);
|
obj = nft_trans_obj(trans);
|
||||||
newobj = nft_trans_obj_newobj(trans);
|
newobj = nft_trans_obj_newobj(trans);
|
||||||
|
|
||||||
obj->ops->update(obj, newobj);
|
if (obj->ops->update)
|
||||||
|
obj->ops->update(obj, newobj);
|
||||||
|
|
||||||
kfree(newobj);
|
kfree(newobj);
|
||||||
}
|
}
|
||||||
|
@ -334,7 +334,8 @@ int nft_flow_rule_offload_commit(struct net *net)
|
|||||||
|
|
||||||
switch (trans->msg_type) {
|
switch (trans->msg_type) {
|
||||||
case NFT_MSG_NEWCHAIN:
|
case NFT_MSG_NEWCHAIN:
|
||||||
if (!(trans->ctx.chain->flags & NFT_CHAIN_HW_OFFLOAD))
|
if (!(trans->ctx.chain->flags & NFT_CHAIN_HW_OFFLOAD) ||
|
||||||
|
nft_trans_chain_update(trans))
|
||||||
continue;
|
continue;
|
||||||
|
|
||||||
policy = nft_trans_chain_policy(trans);
|
policy = nft_trans_chain_policy(trans);
|
||||||
|
@ -134,12 +134,13 @@ static int nft_bitwise_offload(struct nft_offload_ctx *ctx,
|
|||||||
const struct nft_expr *expr)
|
const struct nft_expr *expr)
|
||||||
{
|
{
|
||||||
const struct nft_bitwise *priv = nft_expr_priv(expr);
|
const struct nft_bitwise *priv = nft_expr_priv(expr);
|
||||||
|
struct nft_offload_reg *reg = &ctx->regs[priv->dreg];
|
||||||
|
|
||||||
if (memcmp(&priv->xor, &zero, sizeof(priv->xor)) ||
|
if (memcmp(&priv->xor, &zero, sizeof(priv->xor)) ||
|
||||||
priv->sreg != priv->dreg)
|
priv->sreg != priv->dreg || priv->len != reg->len)
|
||||||
return -EOPNOTSUPP;
|
return -EOPNOTSUPP;
|
||||||
|
|
||||||
memcpy(&ctx->regs[priv->dreg].mask, &priv->mask, sizeof(priv->mask));
|
memcpy(®->mask, &priv->mask, sizeof(priv->mask));
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
@ -116,7 +116,7 @@ static int __nft_cmp_offload(struct nft_offload_ctx *ctx,
|
|||||||
u8 *mask = (u8 *)&flow->match.mask;
|
u8 *mask = (u8 *)&flow->match.mask;
|
||||||
u8 *key = (u8 *)&flow->match.key;
|
u8 *key = (u8 *)&flow->match.key;
|
||||||
|
|
||||||
if (priv->op != NFT_CMP_EQ)
|
if (priv->op != NFT_CMP_EQ || reg->len != priv->len)
|
||||||
return -EOPNOTSUPP;
|
return -EOPNOTSUPP;
|
||||||
|
|
||||||
memcpy(key + reg->offset, &priv->data, priv->len);
|
memcpy(key + reg->offset, &priv->data, priv->len);
|
||||||
|
@ -1099,7 +1099,6 @@ static int nfc_genl_llc_set_params(struct sk_buff *skb, struct genl_info *info)
|
|||||||
|
|
||||||
local = nfc_llcp_find_local(dev);
|
local = nfc_llcp_find_local(dev);
|
||||||
if (!local) {
|
if (!local) {
|
||||||
nfc_put_device(dev);
|
|
||||||
rc = -ENODEV;
|
rc = -ENODEV;
|
||||||
goto exit;
|
goto exit;
|
||||||
}
|
}
|
||||||
@ -1159,7 +1158,6 @@ static int nfc_genl_llc_sdreq(struct sk_buff *skb, struct genl_info *info)
|
|||||||
|
|
||||||
local = nfc_llcp_find_local(dev);
|
local = nfc_llcp_find_local(dev);
|
||||||
if (!local) {
|
if (!local) {
|
||||||
nfc_put_device(dev);
|
|
||||||
rc = -ENODEV;
|
rc = -ENODEV;
|
||||||
goto exit;
|
goto exit;
|
||||||
}
|
}
|
||||||
|
Some files were not shown because too many files have changed in this diff Show More
Loading…
Reference in New Issue
Block a user