/**
  * nfp_net_write_mac_addr() - Write mac address to the device control BAR
  * @nn:      NFP Net device to reconfigure
+ * @addr:    MAC address to write
  *
  * Writes the MAC address from the netdev to the device control BAR.  Does not
  * perform the required reconfig.  We do a bit of byte swapping dance because
  * firmware is LE.
  */
-static void nfp_net_write_mac_addr(struct nfp_net *nn)
+static void nfp_net_write_mac_addr(struct nfp_net *nn, const u8 *addr)
 {
-       nn_writel(nn, NFP_NET_CFG_MACADDR + 0,
-                 get_unaligned_be32(nn->dp.netdev->dev_addr));
-       nn_writew(nn, NFP_NET_CFG_MACADDR + 6,
-                 get_unaligned_be16(nn->dp.netdev->dev_addr + 4));
+       nn_writel(nn, NFP_NET_CFG_MACADDR + 0, get_unaligned_be32(addr));
+       nn_writew(nn, NFP_NET_CFG_MACADDR + 6, get_unaligned_be16(addr + 4));
 }
 
 static void nfp_net_vec_clear_ring_data(struct nfp_net *nn, unsigned int idx)
        nn_writeq(nn, NFP_NET_CFG_RXRS_ENABLE, nn->dp.num_rx_rings == 64 ?
                  0xffffffffffffffffULL : ((u64)1 << nn->dp.num_rx_rings) - 1);
 
-       nfp_net_write_mac_addr(nn);
+       nfp_net_write_mac_addr(nn, nn->dp.netdev->dev_addr);
 
        nn_writel(nn, NFP_NET_CFG_MTU, nn->dp.netdev->mtu);
 
        }
 }
 
+static int nfp_net_set_mac_address(struct net_device *netdev, void *addr)
+{
+       struct nfp_net *nn = netdev_priv(netdev);
+       struct sockaddr *saddr = addr;
+       int err;
+
+       err = eth_prepare_mac_addr_change(netdev, addr);
+       if (err)
+               return err;
+
+       nfp_net_write_mac_addr(nn, saddr->sa_data);
+
+       err = nfp_net_reconfig(nn, NFP_NET_CFG_UPDATE_MACADDR);
+       if (err)
+               return err;
+
+       eth_commit_mac_addr_change(netdev, addr);
+
+       return 0;
+}
+
 const struct net_device_ops nfp_net_netdev_ops = {
        .ndo_open               = nfp_net_netdev_open,
        .ndo_stop               = nfp_net_netdev_close,
        .ndo_tx_timeout         = nfp_net_tx_timeout,
        .ndo_set_rx_mode        = nfp_net_set_rx_mode,
        .ndo_change_mtu         = nfp_net_change_mtu,
-       .ndo_set_mac_address    = eth_mac_addr,
+       .ndo_set_mac_address    = nfp_net_set_mac_address,
        .ndo_set_features       = nfp_net_set_features,
        .ndo_features_check     = nfp_net_features_check,
        .ndo_get_phys_port_name = nfp_port_get_phys_port_name,
                nn->fw_ver.resv, nn->fw_ver.class,
                nn->fw_ver.major, nn->fw_ver.minor,
                nn->max_mtu);
-       nn_info(nn, "CAP: %#x %s%s%s%s%s%s%s%s%s%s%s%s%s%s%s%s%s%s%s%s\n",
+       nn_info(nn, "CAP: %#x %s%s%s%s%s%s%s%s%s%s%s%s%s%s%s%s%s%s%s%s%s\n",
                nn->cap,
                nn->cap & NFP_NET_CFG_CTRL_PROMISC  ? "PROMISC "  : "",
                nn->cap & NFP_NET_CFG_CTRL_L2BC     ? "L2BCFILT " : "",
                nn->cap & NFP_NET_CFG_CTRL_NVGRE    ? "NVGRE "    : "",
                nfp_net_ebpf_capable(nn)            ? "BPF "      : "",
                nn->cap & NFP_NET_CFG_CTRL_CSUM_COMPLETE ?
-                                                     "RXCSUM_COMPLETE " : "");
+                                                     "RXCSUM_COMPLETE " : "",
+               nn->cap & NFP_NET_CFG_CTRL_LIVE_ADDR ? "LIVE_ADDR " : "");
 }
 
 /**
        if (nn->dp.chained_metadata_format && nn->fw_ver.major != 4)
                nn->cap &= ~NFP_NET_CFG_CTRL_RSS;
 
-       nfp_net_write_mac_addr(nn);
+       nfp_net_write_mac_addr(nn, nn->dp.netdev->dev_addr);
 
        /* Determine RX packet/metadata boundary offset */
        if (nn->fw_ver.major >= 2) {
         * and netdev->hw_features advertises which features are
         * supported.  By default we enable most features.
         */
+       if (nn->cap & NFP_NET_CFG_CTRL_LIVE_ADDR)
+               netdev->priv_flags |= IFF_LIVE_ADDR_CHANGE;
+
        netdev->hw_features = NETIF_F_HIGHDMA;
        if (nn->cap & NFP_NET_CFG_CTRL_RXCSUM_ANY) {
                netdev->hw_features |= NETIF_F_RXCSUM;