Donate to e Foundation | Murena handsets with /e/OS | Own a part of Murena! Learn more

Commit bd049a90 authored by Pavel Fedin's avatar Pavel Fedin Committed by David S. Miller
Browse files

net: Cavium: Fix MAC address setting in shutdown state



This bug pops up with NetworkManager on Fedora 21. NetworkManager tends to
stop the interface (nicvf_stop() is called) before changing settings. In
stopped state MAC cannot be sent to a PF. However, when the interface is
restarted (nicvf_open() is called), we ping the PF using NIC_MBOX_MSG_READY
message, and the PF replies back with old MAC address, overriding what we
had after MAC setting from userspace. As a result, we cannot set MAC
address using NetworkManager.

This patch introduces special tracking of MAC change in stopped state so
that the correct new MAC address is sent to a PF when interface is reopen.

Signed-off-by: default avatarPavel Fedin <p.fedin@samsung.com>
Signed-off-by: default avatarDavid S. Miller <davem@davemloft.net>
parent 8beeef8d
Loading
Loading
Loading
Loading
+1 −0
Original line number Original line Diff line number Diff line
@@ -263,6 +263,7 @@ struct nicvf {
	bool			pf_acked;
	bool			pf_acked;
	bool			pf_nacked;
	bool			pf_nacked;
	bool			bgx_stats_acked;
	bool			bgx_stats_acked;
	bool			set_mac_pending;
} ____cacheline_aligned_in_smp;
} ____cacheline_aligned_in_smp;


/* PF <--> VF Mailbox communication
/* PF <--> VF Mailbox communication
+12 −2
Original line number Original line Diff line number Diff line
@@ -201,7 +201,9 @@ static void nicvf_handle_mbx_intr(struct nicvf *nic)
		nic->vf_id = mbx.nic_cfg.vf_id & 0x7F;
		nic->vf_id = mbx.nic_cfg.vf_id & 0x7F;
		nic->tns_mode = mbx.nic_cfg.tns_mode & 0x7F;
		nic->tns_mode = mbx.nic_cfg.tns_mode & 0x7F;
		nic->node = mbx.nic_cfg.node_id;
		nic->node = mbx.nic_cfg.node_id;
		ether_addr_copy(nic->netdev->dev_addr, mbx.nic_cfg.mac_addr);
		if (!nic->set_mac_pending)
			ether_addr_copy(nic->netdev->dev_addr,
					mbx.nic_cfg.mac_addr);
		nic->link_up = false;
		nic->link_up = false;
		nic->duplex = 0;
		nic->duplex = 0;
		nic->speed = 0;
		nic->speed = 0;
@@ -941,6 +943,11 @@ int nicvf_open(struct net_device *netdev)
		nicvf_hw_set_mac_addr(nic, netdev);
		nicvf_hw_set_mac_addr(nic, netdev);
	}
	}


	if (nic->set_mac_pending) {
		nic->set_mac_pending = false;
		nicvf_hw_set_mac_addr(nic, netdev);
	}

	/* Init tasklet for handling Qset err interrupt */
	/* Init tasklet for handling Qset err interrupt */
	tasklet_init(&nic->qs_err_task, nicvf_handle_qs_err,
	tasklet_init(&nic->qs_err_task, nicvf_handle_qs_err,
		     (unsigned long)nic);
		     (unsigned long)nic);
@@ -1040,9 +1047,12 @@ static int nicvf_set_mac_address(struct net_device *netdev, void *p)


	memcpy(netdev->dev_addr, addr->sa_data, netdev->addr_len);
	memcpy(netdev->dev_addr, addr->sa_data, netdev->addr_len);


	if (nic->msix_enabled)
	if (nic->msix_enabled) {
		if (nicvf_hw_set_mac_addr(nic, netdev))
		if (nicvf_hw_set_mac_addr(nic, netdev))
			return -EBUSY;
			return -EBUSY;
	} else {
		nic->set_mac_pending = true;
	}


	return 0;
	return 0;
}
}