Loading...
Note: File does not exist in v6.8.
1/******************************************************************************
2 *
3 * This file is provided under a dual BSD/GPLv2 license. When using or
4 * redistributing this file, you may do so under either license.
5 *
6 * GPL LICENSE SUMMARY
7 *
8 * Copyright(c) 2012 - 2014 Intel Corporation. All rights reserved.
9 * Copyright(c) 2013 - 2015 Intel Mobile Communications GmbH
10 * Copyright(c) 2016 - 2017 Intel Deutschland GmbH
11 *
12 * This program is free software; you can redistribute it and/or modify
13 * it under the terms of version 2 of the GNU General Public License as
14 * published by the Free Software Foundation.
15 *
16 * This program is distributed in the hope that it will be useful, but
17 * WITHOUT ANY WARRANTY; without even the implied warranty of
18 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
19 * General Public License for more details.
20 *
21 * You should have received a copy of the GNU General Public License
22 * along with this program; if not, write to the Free Software
23 * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110,
24 * USA
25 *
26 * The full GNU General Public License is included in this distribution
27 * in the file called COPYING.
28 *
29 * Contact Information:
30 * Intel Linux Wireless <linuxwifi@intel.com>
31 * Intel Corporation, 5200 N.E. Elam Young Parkway, Hillsboro, OR 97124-6497
32 *
33 * BSD LICENSE
34 *
35 * Copyright(c) 2012 - 2014 Intel Corporation. All rights reserved.
36 * Copyright(c) 2013 - 2015 Intel Mobile Communications GmbH
37 * Copyright(c) 2016 - 2017 Intel Deutschland GmbH
38 * All rights reserved.
39 *
40 * Redistribution and use in source and binary forms, with or without
41 * modification, are permitted provided that the following conditions
42 * are met:
43 *
44 * * Redistributions of source code must retain the above copyright
45 * notice, this list of conditions and the following disclaimer.
46 * * Redistributions in binary form must reproduce the above copyright
47 * notice, this list of conditions and the following disclaimer in
48 * the documentation and/or other materials provided with the
49 * distribution.
50 * * Neither the name Intel Corporation nor the names of its
51 * contributors may be used to endorse or promote products derived
52 * from this software without specific prior written permission.
53 *
54 * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
55 * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
56 * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
57 * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
58 * OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
59 * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
60 * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
61 * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
62 * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
63 * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
64 * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
65 *
66 *****************************************************************************/
67#include "iwl-drv.h"
68#include "runtime.h"
69#include "fw/api/nvm-reg.h"
70#include "fw/api/commands.h"
71#include "iwl-nvm-parse.h"
72
73struct iwl_nvm_data *iwl_fw_get_nvm(struct iwl_fw_runtime *fwrt)
74{
75 struct iwl_nvm_get_info cmd = {};
76 struct iwl_nvm_get_info_rsp *rsp;
77 struct iwl_trans *trans = fwrt->trans;
78 struct iwl_nvm_data *nvm;
79 struct iwl_host_cmd hcmd = {
80 .flags = CMD_WANT_SKB | CMD_SEND_IN_RFKILL,
81 .data = { &cmd, },
82 .len = { sizeof(cmd) },
83 .id = WIDE_ID(REGULATORY_AND_NVM_GROUP, NVM_GET_INFO)
84 };
85 int ret;
86 bool lar_fw_supported = !iwlwifi_mod_params.lar_disable &&
87 fw_has_capa(&fwrt->fw->ucode_capa,
88 IWL_UCODE_TLV_CAPA_LAR_SUPPORT);
89
90 ret = iwl_trans_send_cmd(trans, &hcmd);
91 if (ret)
92 return ERR_PTR(ret);
93
94 if (WARN(iwl_rx_packet_payload_len(hcmd.resp_pkt) != sizeof(*rsp),
95 "Invalid payload len in NVM response from FW %d",
96 iwl_rx_packet_payload_len(hcmd.resp_pkt))) {
97 ret = -EINVAL;
98 goto out;
99 }
100
101 rsp = (void *)hcmd.resp_pkt->data;
102 if (le32_to_cpu(rsp->general.flags) & NVM_GENERAL_FLAGS_EMPTY_OTP)
103 IWL_INFO(fwrt, "OTP is empty\n");
104
105 nvm = kzalloc(sizeof(*nvm) +
106 sizeof(struct ieee80211_channel) * IWL_NUM_CHANNELS,
107 GFP_KERNEL);
108 if (!nvm) {
109 ret = -ENOMEM;
110 goto out;
111 }
112
113 iwl_set_hw_address_from_csr(trans, nvm);
114 /* TODO: if platform NVM has MAC address - override it here */
115
116 if (!is_valid_ether_addr(nvm->hw_addr)) {
117 IWL_ERR(fwrt, "no valid mac address was found\n");
118 ret = -EINVAL;
119 goto err_free;
120 }
121
122 IWL_INFO(trans, "base HW address: %pM\n", nvm->hw_addr);
123
124 /* Initialize general data */
125 nvm->nvm_version = le16_to_cpu(rsp->general.nvm_version);
126
127 /* Initialize MAC sku data */
128 nvm->sku_cap_11ac_enable =
129 le32_to_cpu(rsp->mac_sku.enable_11ac);
130 nvm->sku_cap_11n_enable =
131 le32_to_cpu(rsp->mac_sku.enable_11n);
132 nvm->sku_cap_band_24GHz_enable =
133 le32_to_cpu(rsp->mac_sku.enable_24g);
134 nvm->sku_cap_band_52GHz_enable =
135 le32_to_cpu(rsp->mac_sku.enable_5g);
136 nvm->sku_cap_mimo_disabled =
137 le32_to_cpu(rsp->mac_sku.mimo_disable);
138
139 /* Initialize PHY sku data */
140 nvm->valid_tx_ant = (u8)le32_to_cpu(rsp->phy_sku.tx_chains);
141 nvm->valid_rx_ant = (u8)le32_to_cpu(rsp->phy_sku.rx_chains);
142
143 /* Initialize regulatory data */
144 nvm->lar_enabled =
145 le32_to_cpu(rsp->regulatory.lar_enabled) && lar_fw_supported;
146
147 iwl_init_sbands(trans->dev, trans->cfg, nvm,
148 rsp->regulatory.channel_profile,
149 nvm->valid_tx_ant & fwrt->fw->valid_tx_ant,
150 nvm->valid_rx_ant & fwrt->fw->valid_rx_ant,
151 nvm->lar_enabled, false);
152
153 iwl_free_resp(&hcmd);
154 return nvm;
155
156err_free:
157 kfree(nvm);
158out:
159 iwl_free_resp(&hcmd);
160 return ERR_PTR(ret);
161}
162IWL_EXPORT_SYMBOL(iwl_fw_get_nvm);