/* * Marvell Wireless LAN device driver: ethtool * * Copyright (C) 2013-2014, Marvell International Ltd. * * This software file (the "File") is distributed by Marvell International * Ltd. under the terms of the GNU General Public License Version 2, June 1991 * (the "License"). You may use, redistribute and/or modify this File in * accordance with the terms and conditions of the License, a copy of which * is available by writing to the Free Software Foundation, Inc., * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA or on the * worldwide web at http://www.gnu.org/licenses/old-licenses/gpl-2.0.txt. * * THE FILE IS DISTRIBUTED AS-IS, WITHOUT WARRANTY OF ANY KIND, AND THE * IMPLIED WARRANTIES OF MERCHANTABILITY OR FITNESS FOR A PARTICULAR PURPOSE * ARE EXPRESSLY DISCLAIMED. The License provides additional details about * this warranty disclaimer. */ #include "main.h" static void mwifiex_ethtool_get_wol(struct net_device *dev, struct ethtool_wolinfo *wol) { struct mwifiex_private *priv = mwifiex_netdev_get_priv(dev); u32 conditions = le32_to_cpu(priv->adapter->hs_cfg.conditions); wol->supported = WAKE_UCAST|WAKE_MCAST|WAKE_BCAST|WAKE_PHY; if (conditions == HS_CFG_COND_DEF) return; if (conditions & HS_CFG_COND_UNICAST_DATA) wol->wolopts |= WAKE_UCAST; if (conditions & HS_CFG_COND_MULTICAST_DATA) wol->wolopts |= WAKE_MCAST; if (conditions & HS_CFG_COND_BROADCAST_DATA) wol->wolopts |= WAKE_BCAST; if (conditions & HS_CFG_COND_MAC_EVENT) wol->wolopts |= WAKE_PHY; } static int mwifiex_ethtool_set_wol(struct net_device *dev, struct ethtool_wolinfo *wol) { struct mwifiex_private *priv = mwifiex_netdev_get_priv(dev); u32 conditions = 0; if (wol->wolopts & ~(WAKE_UCAST|WAKE_MCAST|WAKE_BCAST|WAKE_PHY)) return -EOPNOTSUPP; if (wol->wolopts & WAKE_UCAST) conditions |= HS_CFG_COND_UNICAST_DATA; if (wol->wolopts & WAKE_MCAST) conditions |= HS_CFG_COND_MULTICAST_DATA; if (wol->wolopts & WAKE_BCAST) conditions |= HS_CFG_COND_BROADCAST_DATA; if (wol->wolopts & WAKE_PHY) conditions |= HS_CFG_COND_MAC_EVENT; if (wol->wolopts == 0) conditions |= HS_CFG_COND_DEF; priv->adapter->hs_cfg.conditions = cpu_to_le32(conditions); return 0; } static int mwifiex_get_dump_flag(struct net_device *dev, struct ethtool_dump *dump) { struct mwifiex_private *priv = mwifiex_netdev_get_priv(dev); struct mwifiex_adapter *adapter = priv->adapter; struct memory_type_mapping *entry; if (!adapter->if_ops.fw_dump) return -ENOTSUPP; dump->flag = adapter->curr_mem_idx; dump->version = 1; if (adapter->curr_mem_idx == MWIFIEX_DRV_INFO_IDX) { dump->len = adapter->drv_info_size; } else if (adapter->curr_mem_idx != MWIFIEX_FW_DUMP_IDX) { entry = &adapter->mem_type_mapping_tbl[adapter->curr_mem_idx]; dump->len = entry->mem_size; } else { dump->len = 0; } return 0; } static int mwifiex_get_dump_data(struct net_device *dev, struct ethtool_dump *dump, void *buffer) { u8 *p = buffer; struct mwifiex_private *priv = mwifiex_netdev_get_priv(dev); struct mwifiex_adapter *adapter = priv->adapter; struct memory_type_mapping *entry; if (!adapter->if_ops.fw_dump) return -ENOTSUPP; if (adapter->curr_mem_idx == MWIFIEX_DRV_INFO_IDX) { if (!adapter->drv_info_dump) return -EFAULT; memcpy(p, adapter->drv_info_dump, adapter->drv_info_size); return 0; } if (adapter->curr_mem_idx == MWIFIEX_FW_DUMP_IDX) { dev_err(adapter->dev, "firmware dump in progress!!\n"); return -EBUSY; } entry = &adapter->mem_type_mapping_tbl[adapter->curr_mem_idx]; if (!entry->mem_ptr) return -EFAULT; memcpy(p, entry->mem_ptr, entry->mem_size); entry->mem_size = 0; vfree(entry->mem_ptr); entry->mem_ptr = NULL; return 0; } static int mwifiex_set_dump(struct net_device *dev, struct ethtool_dump *val) { struct mwifiex_private *priv = mwifiex_netdev_get_priv(dev); struct mwifiex_adapter *adapter = priv->adapter; if (!adapter->if_ops.fw_dump) return -ENOTSUPP; if (val->flag == MWIFIEX_DRV_INFO_IDX) { adapter->curr_mem_idx = MWIFIEX_DRV_INFO_IDX; return 0; } if (adapter->curr_mem_idx == MWIFIEX_FW_DUMP_IDX) { dev_err(adapter->dev, "firmware dump in progress!!\n"); return -EBUSY; } if (val->flag == MWIFIEX_FW_DUMP_IDX) { adapter->curr_mem_idx = val->flag; adapter->if_ops.fw_dump(adapter); return 0; } if (val->flag < 0 || val->flag >= adapter->num_mem_types) return -EINVAL; adapter->curr_mem_idx = val->flag; return 0; } const struct ethtool_ops mwifiex_ethtool_ops = { .get_wol = mwifiex_ethtool_get_wol, .set_wol = mwifiex_ethtool_set_wol, .get_dump_flag = mwifiex_get_dump_flag, .get_dump_data = mwifiex_get_dump_data, .set_dump = mwifiex_set_dump, };