blob: 76a5cb04e4fe42ef2f67b6fc2301269765a7dc8f [file] [log] [blame]
Jesse Brandeburg56a62fc2013-09-11 08:40:12 +00001/*******************************************************************************
2 *
3 * Intel Ethernet Controller XL710 Family Linux Driver
Greg Rosedc641b72013-12-18 13:45:51 +00004 * Copyright(c) 2013 - 2014 Intel Corporation.
Jesse Brandeburg56a62fc2013-09-11 08:40:12 +00005 *
6 * This program is free software; you can redistribute it and/or modify it
7 * under the terms and conditions of the GNU General Public License,
8 * version 2, as published by the Free Software Foundation.
9 *
10 * This program is distributed in the hope it will be useful, but WITHOUT
11 * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
12 * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for
13 * more details.
14 *
Greg Rosedc641b72013-12-18 13:45:51 +000015 * You should have received a copy of the GNU General Public License along
16 * with this program. If not, see <http://www.gnu.org/licenses/>.
Jesse Brandeburg56a62fc2013-09-11 08:40:12 +000017 *
18 * The full GNU General Public License is included in this distribution in
19 * the file called "COPYING".
20 *
21 * Contact Information:
22 * e1000-devel Mailing List <e1000-devel@lists.sourceforge.net>
23 * Intel Corporation, 5200 N.E. Elam Young Parkway, Hillsboro, OR 97124-6497
24 *
25 ******************************************************************************/
26
27#include "i40e_prototype.h"
28
29/**
Shannon Nelson3e261862014-02-06 05:51:06 +000030 * i40e_init_nvm_ops - Initialize NVM function pointers
31 * @hw: pointer to the HW structure
Jesse Brandeburg56a62fc2013-09-11 08:40:12 +000032 *
Shannon Nelson3e261862014-02-06 05:51:06 +000033 * Setup the function pointers and the NVM info structure. Should be called
34 * once per NVM initialization, e.g. inside the i40e_init_shared_code().
35 * Please notice that the NVM term is used here (& in all methods covered
36 * in this file) as an equivalent of the FLASH part mapped into the SR.
37 * We are accessing FLASH always thru the Shadow RAM.
Jesse Brandeburg56a62fc2013-09-11 08:40:12 +000038 **/
39i40e_status i40e_init_nvm(struct i40e_hw *hw)
40{
41 struct i40e_nvm_info *nvm = &hw->nvm;
42 i40e_status ret_code = 0;
43 u32 fla, gens;
44 u8 sr_size;
45
46 /* The SR size is stored regardless of the nvm programming mode
47 * as the blank mode may be used in the factory line.
48 */
49 gens = rd32(hw, I40E_GLNVM_GENS);
50 sr_size = ((gens & I40E_GLNVM_GENS_SR_SIZE_MASK) >>
51 I40E_GLNVM_GENS_SR_SIZE_SHIFT);
Shannon Nelson3e261862014-02-06 05:51:06 +000052 /* Switching to words (sr_size contains power of 2KB) */
Jesse Brandeburg41a1d042015-06-04 16:24:02 -040053 nvm->sr_size = BIT(sr_size) * I40E_SR_WORDS_IN_1KB;
Jesse Brandeburg56a62fc2013-09-11 08:40:12 +000054
Shannon Nelson3e261862014-02-06 05:51:06 +000055 /* Check if we are in the normal or blank NVM programming mode */
Jesse Brandeburg56a62fc2013-09-11 08:40:12 +000056 fla = rd32(hw, I40E_GLNVM_FLA);
Shannon Nelson3e261862014-02-06 05:51:06 +000057 if (fla & I40E_GLNVM_FLA_LOCKED_MASK) { /* Normal programming mode */
58 /* Max NVM timeout */
Jesse Brandeburg56a62fc2013-09-11 08:40:12 +000059 nvm->timeout = I40E_MAX_NVM_TIMEOUT;
60 nvm->blank_nvm_mode = false;
Shannon Nelson3e261862014-02-06 05:51:06 +000061 } else { /* Blank programming mode */
Jesse Brandeburg56a62fc2013-09-11 08:40:12 +000062 nvm->blank_nvm_mode = true;
63 ret_code = I40E_ERR_NVM_BLANK_MODE;
Shannon Nelson74d0d0e2014-11-13 08:23:15 +000064 i40e_debug(hw, I40E_DEBUG_NVM, "NVM init error: unsupported blank mode.\n");
Jesse Brandeburg56a62fc2013-09-11 08:40:12 +000065 }
66
67 return ret_code;
68}
69
70/**
Shannon Nelson3e261862014-02-06 05:51:06 +000071 * i40e_acquire_nvm - Generic request for acquiring the NVM ownership
72 * @hw: pointer to the HW structure
73 * @access: NVM access type (read or write)
Jesse Brandeburg56a62fc2013-09-11 08:40:12 +000074 *
Shannon Nelson3e261862014-02-06 05:51:06 +000075 * This function will request NVM ownership for reading
76 * via the proper Admin Command.
Jesse Brandeburg56a62fc2013-09-11 08:40:12 +000077 **/
78i40e_status i40e_acquire_nvm(struct i40e_hw *hw,
79 enum i40e_aq_resource_access_type access)
80{
81 i40e_status ret_code = 0;
82 u64 gtime, timeout;
Shannon Nelsonc509c1d2014-11-13 08:23:19 +000083 u64 time_left = 0;
Jesse Brandeburg56a62fc2013-09-11 08:40:12 +000084
85 if (hw->nvm.blank_nvm_mode)
86 goto i40e_i40e_acquire_nvm_exit;
87
88 ret_code = i40e_aq_request_resource(hw, I40E_NVM_RESOURCE_ID, access,
Shannon Nelsonc509c1d2014-11-13 08:23:19 +000089 0, &time_left, NULL);
Shannon Nelson3e261862014-02-06 05:51:06 +000090 /* Reading the Global Device Timer */
Jesse Brandeburg56a62fc2013-09-11 08:40:12 +000091 gtime = rd32(hw, I40E_GLVFGEN_TIMER);
92
Shannon Nelson3e261862014-02-06 05:51:06 +000093 /* Store the timeout */
Shannon Nelsonc509c1d2014-11-13 08:23:19 +000094 hw->nvm.hw_semaphore_timeout = I40E_MS_TO_GTIME(time_left) + gtime;
Jesse Brandeburg56a62fc2013-09-11 08:40:12 +000095
Shannon Nelsona3f0b382014-11-13 08:23:21 +000096 if (ret_code)
97 i40e_debug(hw, I40E_DEBUG_NVM,
98 "NVM acquire type %d failed time_left=%llu ret=%d aq_err=%d\n",
99 access, time_left, ret_code, hw->aq.asq_last_status);
100
101 if (ret_code && time_left) {
Shannon Nelson3e261862014-02-06 05:51:06 +0000102 /* Poll until the current NVM owner timeouts */
Shannon Nelsonc509c1d2014-11-13 08:23:19 +0000103 timeout = I40E_MS_TO_GTIME(I40E_MAX_NVM_TIMEOUT) + gtime;
Shannon Nelsona3f0b382014-11-13 08:23:21 +0000104 while ((gtime < timeout) && time_left) {
Jesse Brandeburg56a62fc2013-09-11 08:40:12 +0000105 usleep_range(10000, 20000);
Shannon Nelsonc509c1d2014-11-13 08:23:19 +0000106 gtime = rd32(hw, I40E_GLVFGEN_TIMER);
Jesse Brandeburg56a62fc2013-09-11 08:40:12 +0000107 ret_code = i40e_aq_request_resource(hw,
108 I40E_NVM_RESOURCE_ID,
Shannon Nelsonc509c1d2014-11-13 08:23:19 +0000109 access, 0, &time_left,
Jesse Brandeburg56a62fc2013-09-11 08:40:12 +0000110 NULL);
111 if (!ret_code) {
112 hw->nvm.hw_semaphore_timeout =
Shannon Nelsonc509c1d2014-11-13 08:23:19 +0000113 I40E_MS_TO_GTIME(time_left) + gtime;
Jesse Brandeburg56a62fc2013-09-11 08:40:12 +0000114 break;
115 }
Jesse Brandeburg56a62fc2013-09-11 08:40:12 +0000116 }
117 if (ret_code) {
118 hw->nvm.hw_semaphore_timeout = 0;
Shannon Nelson74d0d0e2014-11-13 08:23:15 +0000119 i40e_debug(hw, I40E_DEBUG_NVM,
Shannon Nelsona3f0b382014-11-13 08:23:21 +0000120 "NVM acquire timed out, wait %llu ms before trying again. status=%d aq_err=%d\n",
121 time_left, ret_code, hw->aq.asq_last_status);
Jesse Brandeburg56a62fc2013-09-11 08:40:12 +0000122 }
123 }
124
125i40e_i40e_acquire_nvm_exit:
126 return ret_code;
127}
128
129/**
Shannon Nelson3e261862014-02-06 05:51:06 +0000130 * i40e_release_nvm - Generic request for releasing the NVM ownership
131 * @hw: pointer to the HW structure
Jesse Brandeburg56a62fc2013-09-11 08:40:12 +0000132 *
Shannon Nelson3e261862014-02-06 05:51:06 +0000133 * This function will release NVM resource via the proper Admin Command.
Jesse Brandeburg56a62fc2013-09-11 08:40:12 +0000134 **/
135void i40e_release_nvm(struct i40e_hw *hw)
136{
Paul M Stillwell Jr981e25c2017-06-20 15:16:55 -0700137 i40e_status ret_code = I40E_SUCCESS;
138 u32 total_delay = 0;
139
140 if (hw->nvm.blank_nvm_mode)
141 return;
142
143 ret_code = i40e_aq_release_resource(hw, I40E_NVM_RESOURCE_ID, 0, NULL);
144
145 /* there are some rare cases when trying to release the resource
146 * results in an admin Q timeout, so handle them correctly
147 */
148 while ((ret_code == I40E_ERR_ADMIN_QUEUE_TIMEOUT) &&
149 (total_delay < hw->aq.asq_cmd_timeout)) {
150 usleep_range(1000, 2000);
151 ret_code = i40e_aq_release_resource(hw,
152 I40E_NVM_RESOURCE_ID,
153 0, NULL);
154 total_delay++;
155 }
Jesse Brandeburg56a62fc2013-09-11 08:40:12 +0000156}
157
158/**
Shannon Nelson3e261862014-02-06 05:51:06 +0000159 * i40e_poll_sr_srctl_done_bit - Polls the GLNVM_SRCTL done bit
160 * @hw: pointer to the HW structure
Jesse Brandeburg56a62fc2013-09-11 08:40:12 +0000161 *
Shannon Nelson3e261862014-02-06 05:51:06 +0000162 * Polls the SRCTL Shadow RAM register done bit.
Jesse Brandeburg56a62fc2013-09-11 08:40:12 +0000163 **/
164static i40e_status i40e_poll_sr_srctl_done_bit(struct i40e_hw *hw)
165{
166 i40e_status ret_code = I40E_ERR_TIMEOUT;
167 u32 srctl, wait_cnt;
168
Shannon Nelson3e261862014-02-06 05:51:06 +0000169 /* Poll the I40E_GLNVM_SRCTL until the done bit is set */
Jesse Brandeburg56a62fc2013-09-11 08:40:12 +0000170 for (wait_cnt = 0; wait_cnt < I40E_SRRD_SRCTL_ATTEMPTS; wait_cnt++) {
171 srctl = rd32(hw, I40E_GLNVM_SRCTL);
172 if (srctl & I40E_GLNVM_SRCTL_DONE_MASK) {
173 ret_code = 0;
174 break;
175 }
176 udelay(5);
177 }
178 if (ret_code == I40E_ERR_TIMEOUT)
Shannon Nelson74d0d0e2014-11-13 08:23:15 +0000179 i40e_debug(hw, I40E_DEBUG_NVM, "Done bit in GLNVM_SRCTL not set");
Jesse Brandeburg56a62fc2013-09-11 08:40:12 +0000180 return ret_code;
181}
182
183/**
Kamil Krawczykd1bbe0e2015-01-24 09:58:33 +0000184 * i40e_read_nvm_word_srctl - Reads Shadow RAM via SRCTL register
Shannon Nelson3e261862014-02-06 05:51:06 +0000185 * @hw: pointer to the HW structure
186 * @offset: offset of the Shadow RAM word to read (0x000000 - 0x001FFF)
187 * @data: word read from the Shadow RAM
Jesse Brandeburg56a62fc2013-09-11 08:40:12 +0000188 *
Shannon Nelson3e261862014-02-06 05:51:06 +0000189 * Reads one 16 bit word from the Shadow RAM using the GLNVM_SRCTL register.
Jesse Brandeburg56a62fc2013-09-11 08:40:12 +0000190 **/
Shannon Nelson37a29732015-02-27 09:15:19 +0000191static i40e_status i40e_read_nvm_word_srctl(struct i40e_hw *hw, u16 offset,
192 u16 *data)
Jesse Brandeburg56a62fc2013-09-11 08:40:12 +0000193{
194 i40e_status ret_code = I40E_ERR_TIMEOUT;
195 u32 sr_reg;
196
197 if (offset >= hw->nvm.sr_size) {
Shannon Nelson74d0d0e2014-11-13 08:23:15 +0000198 i40e_debug(hw, I40E_DEBUG_NVM,
199 "NVM read error: offset %d beyond Shadow RAM limit %d\n",
200 offset, hw->nvm.sr_size);
Jesse Brandeburg56a62fc2013-09-11 08:40:12 +0000201 ret_code = I40E_ERR_PARAM;
202 goto read_nvm_exit;
203 }
204
Shannon Nelson3e261862014-02-06 05:51:06 +0000205 /* Poll the done bit first */
Jesse Brandeburg56a62fc2013-09-11 08:40:12 +0000206 ret_code = i40e_poll_sr_srctl_done_bit(hw);
207 if (!ret_code) {
Shannon Nelson3e261862014-02-06 05:51:06 +0000208 /* Write the address and start reading */
Jesse Brandeburg41a1d042015-06-04 16:24:02 -0400209 sr_reg = ((u32)offset << I40E_GLNVM_SRCTL_ADDR_SHIFT) |
210 BIT(I40E_GLNVM_SRCTL_START_SHIFT);
Jesse Brandeburg56a62fc2013-09-11 08:40:12 +0000211 wr32(hw, I40E_GLNVM_SRCTL, sr_reg);
212
Shannon Nelson3e261862014-02-06 05:51:06 +0000213 /* Poll I40E_GLNVM_SRCTL until the done bit is set */
Jesse Brandeburg56a62fc2013-09-11 08:40:12 +0000214 ret_code = i40e_poll_sr_srctl_done_bit(hw);
215 if (!ret_code) {
216 sr_reg = rd32(hw, I40E_GLNVM_SRDATA);
217 *data = (u16)((sr_reg &
218 I40E_GLNVM_SRDATA_RDDATA_MASK)
219 >> I40E_GLNVM_SRDATA_RDDATA_SHIFT);
220 }
221 }
222 if (ret_code)
Shannon Nelson74d0d0e2014-11-13 08:23:15 +0000223 i40e_debug(hw, I40E_DEBUG_NVM,
224 "NVM read error: Couldn't access Shadow RAM address: 0x%x\n",
225 offset);
Jesse Brandeburg56a62fc2013-09-11 08:40:12 +0000226
227read_nvm_exit:
228 return ret_code;
229}
230
231/**
Shannon Nelson7073f462015-06-05 12:20:34 -0400232 * i40e_read_nvm_aq - Read Shadow RAM.
233 * @hw: pointer to the HW structure.
234 * @module_pointer: module pointer location in words from the NVM beginning
235 * @offset: offset in words from module start
236 * @words: number of words to write
237 * @data: buffer with words to write to the Shadow RAM
238 * @last_command: tells the AdminQ that this is the last command
239 *
240 * Writes a 16 bit words buffer to the Shadow RAM using the admin command.
241 **/
Pawel Jablonskie3a5d6e2017-12-18 05:14:44 -0500242static i40e_status i40e_read_nvm_aq(struct i40e_hw *hw,
243 u8 module_pointer, u32 offset,
244 u16 words, void *data,
Shannon Nelson7073f462015-06-05 12:20:34 -0400245 bool last_command)
246{
247 i40e_status ret_code = I40E_ERR_NVM;
248 struct i40e_asq_cmd_details cmd_details;
249
250 memset(&cmd_details, 0, sizeof(cmd_details));
Jacob Keller3c8f3e92017-09-01 13:43:08 -0700251 cmd_details.wb_desc = &hw->nvm_wb_desc;
Shannon Nelson7073f462015-06-05 12:20:34 -0400252
253 /* Here we are checking the SR limit only for the flat memory model.
254 * We cannot do it for the module-based model, as we did not acquire
255 * the NVM resource yet (we cannot get the module pointer value).
256 * Firmware will check the module-based model.
257 */
258 if ((offset + words) > hw->nvm.sr_size)
259 i40e_debug(hw, I40E_DEBUG_NVM,
260 "NVM write error: offset %d beyond Shadow RAM limit %d\n",
261 (offset + words), hw->nvm.sr_size);
262 else if (words > I40E_SR_SECTOR_SIZE_IN_WORDS)
263 /* We can write only up to 4KB (one sector), in one AQ write */
264 i40e_debug(hw, I40E_DEBUG_NVM,
265 "NVM write fail error: tried to write %d words, limit is %d.\n",
266 words, I40E_SR_SECTOR_SIZE_IN_WORDS);
267 else if (((offset + (words - 1)) / I40E_SR_SECTOR_SIZE_IN_WORDS)
268 != (offset / I40E_SR_SECTOR_SIZE_IN_WORDS))
269 /* A single write cannot spread over two sectors */
270 i40e_debug(hw, I40E_DEBUG_NVM,
271 "NVM write error: cannot spread over two sectors in a single write offset=%d words=%d\n",
272 offset, words);
273 else
274 ret_code = i40e_aq_read_nvm(hw, module_pointer,
275 2 * offset, /*bytes*/
276 2 * words, /*bytes*/
277 data, last_command, &cmd_details);
278
279 return ret_code;
280}
281
282/**
283 * i40e_read_nvm_word_aq - Reads Shadow RAM via AQ
284 * @hw: pointer to the HW structure
285 * @offset: offset of the Shadow RAM word to read (0x000000 - 0x001FFF)
286 * @data: word read from the Shadow RAM
287 *
Anjali Singhai Jain09f79fd2017-09-01 13:42:49 -0700288 * Reads one 16 bit word from the Shadow RAM using the AdminQ
Shannon Nelson7073f462015-06-05 12:20:34 -0400289 **/
290static i40e_status i40e_read_nvm_word_aq(struct i40e_hw *hw, u16 offset,
291 u16 *data)
292{
293 i40e_status ret_code = I40E_ERR_TIMEOUT;
294
295 ret_code = i40e_read_nvm_aq(hw, 0x0, offset, 1, data, true);
296 *data = le16_to_cpu(*(__le16 *)data);
297
298 return ret_code;
299}
300
301/**
Stefano Brivioe836e322017-09-06 10:11:38 +0200302 * __i40e_read_nvm_word - Reads nvm word, assumes caller does the locking
Kamil Krawczykd1bbe0e2015-01-24 09:58:33 +0000303 * @hw: pointer to the HW structure
304 * @offset: offset of the Shadow RAM word to read (0x000000 - 0x001FFF)
305 * @data: word read from the Shadow RAM
306 *
Anjali Singhai Jain09f79fd2017-09-01 13:42:49 -0700307 * Reads one 16 bit word from the Shadow RAM.
308 *
309 * Do not use this function except in cases where the nvm lock is already
310 * taken via i40e_acquire_nvm().
311 **/
312static i40e_status __i40e_read_nvm_word(struct i40e_hw *hw,
313 u16 offset, u16 *data)
314{
Anjali Singhai Jain09f79fd2017-09-01 13:42:49 -0700315 if (hw->flags & I40E_HW_FLAG_AQ_SRCTL_ACCESS_ENABLE)
Stefano Brivio2c4d36b2017-09-06 10:11:39 +0200316 return i40e_read_nvm_word_aq(hw, offset, data);
317
318 return i40e_read_nvm_word_srctl(hw, offset, data);
Anjali Singhai Jain09f79fd2017-09-01 13:42:49 -0700319}
320
321/**
322 * i40e_read_nvm_word - Reads nvm word and acquire lock if necessary
323 * @hw: pointer to the HW structure
324 * @offset: offset of the Shadow RAM word to read (0x000000 - 0x001FFF)
325 * @data: word read from the Shadow RAM
326 *
327 * Reads one 16 bit word from the Shadow RAM.
Kamil Krawczykd1bbe0e2015-01-24 09:58:33 +0000328 **/
329i40e_status i40e_read_nvm_word(struct i40e_hw *hw, u16 offset,
330 u16 *data)
331{
Jacob Keller3d72aeb2017-10-27 11:06:55 -0400332 i40e_status ret_code = 0;
Anjali Singhai07f89be2015-09-24 15:26:32 -0700333
Jacob Keller3d72aeb2017-10-27 11:06:55 -0400334 if (hw->flags & I40E_HW_FLAG_NVM_READ_REQUIRES_LOCK)
335 ret_code = i40e_acquire_nvm(hw, I40E_RESOURCE_READ);
Anjali Singhai Jain09f79fd2017-09-01 13:42:49 -0700336 if (ret_code)
337 return ret_code;
338
339 ret_code = __i40e_read_nvm_word(hw, offset, data);
340
Jacob Keller3d72aeb2017-10-27 11:06:55 -0400341 if (hw->flags & I40E_HW_FLAG_NVM_READ_REQUIRES_LOCK)
342 i40e_release_nvm(hw);
Anjali Singhai Jain09f79fd2017-09-01 13:42:49 -0700343
Anjali Singhai07f89be2015-09-24 15:26:32 -0700344 return ret_code;
Kamil Krawczykd1bbe0e2015-01-24 09:58:33 +0000345}
346
347/**
348 * i40e_read_nvm_buffer_srctl - Reads Shadow RAM buffer via SRCTL register
349 * @hw: pointer to the HW structure
350 * @offset: offset of the Shadow RAM word to read (0x000000 - 0x001FFF).
351 * @words: (in) number of words to read; (out) number of words actually read
352 * @data: words read from the Shadow RAM
353 *
354 * Reads 16 bit words (data buffer) from the SR using the i40e_read_nvm_srrd()
355 * method. The buffer read is preceded by the NVM ownership take
356 * and followed by the release.
357 **/
Shannon Nelson37a29732015-02-27 09:15:19 +0000358static i40e_status i40e_read_nvm_buffer_srctl(struct i40e_hw *hw, u16 offset,
359 u16 *words, u16 *data)
Kamil Krawczykd1bbe0e2015-01-24 09:58:33 +0000360{
361 i40e_status ret_code = 0;
362 u16 index, word;
363
364 /* Loop thru the selected region */
365 for (word = 0; word < *words; word++) {
366 index = offset + word;
367 ret_code = i40e_read_nvm_word_srctl(hw, index, &data[word]);
368 if (ret_code)
369 break;
370 }
371
372 /* Update the number of words read from the Shadow RAM */
373 *words = word;
374
375 return ret_code;
376}
377
378/**
Shannon Nelson7073f462015-06-05 12:20:34 -0400379 * i40e_read_nvm_buffer_aq - Reads Shadow RAM buffer via AQ
380 * @hw: pointer to the HW structure
381 * @offset: offset of the Shadow RAM word to read (0x000000 - 0x001FFF).
382 * @words: (in) number of words to read; (out) number of words actually read
383 * @data: words read from the Shadow RAM
384 *
385 * Reads 16 bit words (data buffer) from the SR using the i40e_read_nvm_aq()
386 * method. The buffer read is preceded by the NVM ownership take
387 * and followed by the release.
388 **/
389static i40e_status i40e_read_nvm_buffer_aq(struct i40e_hw *hw, u16 offset,
390 u16 *words, u16 *data)
391{
392 i40e_status ret_code;
Colin Ian King793c6f82017-11-05 13:04:29 +0000393 u16 read_size;
Shannon Nelson7073f462015-06-05 12:20:34 -0400394 bool last_cmd = false;
395 u16 words_read = 0;
396 u16 i = 0;
397
398 do {
399 /* Calculate number of bytes we should read in this step.
400 * FVL AQ do not allow to read more than one page at a time or
401 * to cross page boundaries.
402 */
403 if (offset % I40E_SR_SECTOR_SIZE_IN_WORDS)
404 read_size = min(*words,
405 (u16)(I40E_SR_SECTOR_SIZE_IN_WORDS -
406 (offset % I40E_SR_SECTOR_SIZE_IN_WORDS)));
407 else
408 read_size = min((*words - words_read),
409 I40E_SR_SECTOR_SIZE_IN_WORDS);
410
411 /* Check if this is last command, if so set proper flag */
412 if ((words_read + read_size) >= *words)
413 last_cmd = true;
414
415 ret_code = i40e_read_nvm_aq(hw, 0x0, offset, read_size,
416 data + words_read, last_cmd);
417 if (ret_code)
418 goto read_nvm_buffer_aq_exit;
419
420 /* Increment counter for words already read and move offset to
421 * new read location
422 */
423 words_read += read_size;
424 offset += read_size;
425 } while (words_read < *words);
426
427 for (i = 0; i < *words; i++)
428 data[i] = le16_to_cpu(((__le16 *)data)[i]);
429
430read_nvm_buffer_aq_exit:
431 *words = words_read;
432 return ret_code;
433}
434
435/**
Anjali Singhai Jain09f79fd2017-09-01 13:42:49 -0700436 * __i40e_read_nvm_buffer - Reads nvm buffer, caller must acquire lock
Shannon Nelson3e261862014-02-06 05:51:06 +0000437 * @hw: pointer to the HW structure
438 * @offset: offset of the Shadow RAM word to read (0x000000 - 0x001FFF).
439 * @words: (in) number of words to read; (out) number of words actually read
440 * @data: words read from the Shadow RAM
Jesse Brandeburg56a62fc2013-09-11 08:40:12 +0000441 *
Shannon Nelson3e261862014-02-06 05:51:06 +0000442 * Reads 16 bit words (data buffer) from the SR using the i40e_read_nvm_srrd()
Anjali Singhai Jain09f79fd2017-09-01 13:42:49 -0700443 * method.
Jesse Brandeburg56a62fc2013-09-11 08:40:12 +0000444 **/
Anjali Singhai Jain09f79fd2017-09-01 13:42:49 -0700445static i40e_status __i40e_read_nvm_buffer(struct i40e_hw *hw,
446 u16 offset, u16 *words,
447 u16 *data)
Jesse Brandeburg56a62fc2013-09-11 08:40:12 +0000448{
Anjali Singhai Jain09f79fd2017-09-01 13:42:49 -0700449 if (hw->flags & I40E_HW_FLAG_AQ_SRCTL_ACCESS_ENABLE)
Stefano Brivio2c4d36b2017-09-06 10:11:39 +0200450 return i40e_read_nvm_buffer_aq(hw, offset, words, data);
451
452 return i40e_read_nvm_buffer_srctl(hw, offset, words, data);
Jesse Brandeburg56a62fc2013-09-11 08:40:12 +0000453}
454
455/**
Shannon Nelsoncd552cb2014-07-09 07:46:09 +0000456 * i40e_write_nvm_aq - Writes Shadow RAM.
457 * @hw: pointer to the HW structure.
458 * @module_pointer: module pointer location in words from the NVM beginning
459 * @offset: offset in words from module start
460 * @words: number of words to write
461 * @data: buffer with words to write to the Shadow RAM
462 * @last_command: tells the AdminQ that this is the last command
463 *
464 * Writes a 16 bit words buffer to the Shadow RAM using the admin command.
465 **/
Wei Yongjun952d9632014-07-30 09:02:53 +0000466static i40e_status i40e_write_nvm_aq(struct i40e_hw *hw, u8 module_pointer,
467 u32 offset, u16 words, void *data,
468 bool last_command)
Shannon Nelsoncd552cb2014-07-09 07:46:09 +0000469{
470 i40e_status ret_code = I40E_ERR_NVM;
Shannon Nelson6b5c1b82015-08-28 17:55:47 -0400471 struct i40e_asq_cmd_details cmd_details;
472
473 memset(&cmd_details, 0, sizeof(cmd_details));
474 cmd_details.wb_desc = &hw->nvm_wb_desc;
Shannon Nelsoncd552cb2014-07-09 07:46:09 +0000475
476 /* Here we are checking the SR limit only for the flat memory model.
477 * We cannot do it for the module-based model, as we did not acquire
478 * the NVM resource yet (we cannot get the module pointer value).
479 * Firmware will check the module-based model.
480 */
481 if ((offset + words) > hw->nvm.sr_size)
Shannon Nelson74d0d0e2014-11-13 08:23:15 +0000482 i40e_debug(hw, I40E_DEBUG_NVM,
483 "NVM write error: offset %d beyond Shadow RAM limit %d\n",
484 (offset + words), hw->nvm.sr_size);
Shannon Nelsoncd552cb2014-07-09 07:46:09 +0000485 else if (words > I40E_SR_SECTOR_SIZE_IN_WORDS)
486 /* We can write only up to 4KB (one sector), in one AQ write */
Shannon Nelson74d0d0e2014-11-13 08:23:15 +0000487 i40e_debug(hw, I40E_DEBUG_NVM,
488 "NVM write fail error: tried to write %d words, limit is %d.\n",
489 words, I40E_SR_SECTOR_SIZE_IN_WORDS);
Shannon Nelsoncd552cb2014-07-09 07:46:09 +0000490 else if (((offset + (words - 1)) / I40E_SR_SECTOR_SIZE_IN_WORDS)
491 != (offset / I40E_SR_SECTOR_SIZE_IN_WORDS))
492 /* A single write cannot spread over two sectors */
Shannon Nelson74d0d0e2014-11-13 08:23:15 +0000493 i40e_debug(hw, I40E_DEBUG_NVM,
494 "NVM write error: cannot spread over two sectors in a single write offset=%d words=%d\n",
495 offset, words);
Shannon Nelsoncd552cb2014-07-09 07:46:09 +0000496 else
497 ret_code = i40e_aq_update_nvm(hw, module_pointer,
498 2 * offset, /*bytes*/
499 2 * words, /*bytes*/
Pawel Jablonskie3a5d6e2017-12-18 05:14:44 -0500500 data, last_command, 0,
501 &cmd_details);
Shannon Nelsoncd552cb2014-07-09 07:46:09 +0000502
503 return ret_code;
504}
505
506/**
Shannon Nelson3e261862014-02-06 05:51:06 +0000507 * i40e_calc_nvm_checksum - Calculates and returns the checksum
508 * @hw: pointer to hardware structure
509 * @checksum: pointer to the checksum
Jesse Brandeburg56a62fc2013-09-11 08:40:12 +0000510 *
Shannon Nelson3e261862014-02-06 05:51:06 +0000511 * This function calculates SW Checksum that covers the whole 64kB shadow RAM
512 * except the VPD and PCIe ALT Auto-load modules. The structure and size of VPD
513 * is customer specific and unknown. Therefore, this function skips all maximum
514 * possible size of VPD (1kB).
Jesse Brandeburg56a62fc2013-09-11 08:40:12 +0000515 **/
516static i40e_status i40e_calc_nvm_checksum(struct i40e_hw *hw,
517 u16 *checksum)
518{
Jean Sacren0e5229c2015-10-13 01:06:31 -0600519 i40e_status ret_code;
Kamil Krawczykd1bbe0e2015-01-24 09:58:33 +0000520 struct i40e_virt_mem vmem;
Jesse Brandeburg56a62fc2013-09-11 08:40:12 +0000521 u16 pcie_alt_module = 0;
522 u16 checksum_local = 0;
523 u16 vpd_module = 0;
Kamil Krawczykd1bbe0e2015-01-24 09:58:33 +0000524 u16 *data;
525 u16 i = 0;
526
527 ret_code = i40e_allocate_virt_mem(hw, &vmem,
528 I40E_SR_SECTOR_SIZE_IN_WORDS * sizeof(u16));
529 if (ret_code)
530 goto i40e_calc_nvm_checksum_exit;
531 data = (u16 *)vmem.va;
Jesse Brandeburg56a62fc2013-09-11 08:40:12 +0000532
533 /* read pointer to VPD area */
Anjali Singhai Jain09f79fd2017-09-01 13:42:49 -0700534 ret_code = __i40e_read_nvm_word(hw, I40E_SR_VPD_PTR, &vpd_module);
Jesse Brandeburg56a62fc2013-09-11 08:40:12 +0000535 if (ret_code) {
536 ret_code = I40E_ERR_NVM_CHECKSUM;
537 goto i40e_calc_nvm_checksum_exit;
538 }
539
540 /* read pointer to PCIe Alt Auto-load module */
Anjali Singhai Jain09f79fd2017-09-01 13:42:49 -0700541 ret_code = __i40e_read_nvm_word(hw, I40E_SR_PCIE_ALT_AUTO_LOAD_PTR,
542 &pcie_alt_module);
Jesse Brandeburg56a62fc2013-09-11 08:40:12 +0000543 if (ret_code) {
544 ret_code = I40E_ERR_NVM_CHECKSUM;
545 goto i40e_calc_nvm_checksum_exit;
546 }
547
548 /* Calculate SW checksum that covers the whole 64kB shadow RAM
549 * except the VPD and PCIe ALT Auto-load modules
550 */
551 for (i = 0; i < hw->nvm.sr_size; i++) {
Kamil Krawczykd1bbe0e2015-01-24 09:58:33 +0000552 /* Read SR page */
553 if ((i % I40E_SR_SECTOR_SIZE_IN_WORDS) == 0) {
554 u16 words = I40E_SR_SECTOR_SIZE_IN_WORDS;
555
Anjali Singhai Jain09f79fd2017-09-01 13:42:49 -0700556 ret_code = __i40e_read_nvm_buffer(hw, i, &words, data);
Kamil Krawczykd1bbe0e2015-01-24 09:58:33 +0000557 if (ret_code) {
558 ret_code = I40E_ERR_NVM_CHECKSUM;
559 goto i40e_calc_nvm_checksum_exit;
560 }
Jesse Brandeburg56a62fc2013-09-11 08:40:12 +0000561 }
562
Kamil Krawczykd1bbe0e2015-01-24 09:58:33 +0000563 /* Skip Checksum word */
564 if (i == I40E_SR_SW_CHECKSUM_WORD)
565 continue;
566 /* Skip VPD module (convert byte size to word count) */
567 if ((i >= (u32)vpd_module) &&
568 (i < ((u32)vpd_module +
569 (I40E_SR_VPD_MODULE_MAX_SIZE / 2)))) {
570 continue;
Jesse Brandeburg56a62fc2013-09-11 08:40:12 +0000571 }
Kamil Krawczykd1bbe0e2015-01-24 09:58:33 +0000572 /* Skip PCIe ALT module (convert byte size to word count) */
573 if ((i >= (u32)pcie_alt_module) &&
574 (i < ((u32)pcie_alt_module +
575 (I40E_SR_PCIE_ALT_MODULE_MAX_SIZE / 2)))) {
576 continue;
577 }
578
579 checksum_local += data[i % I40E_SR_SECTOR_SIZE_IN_WORDS];
Jesse Brandeburg56a62fc2013-09-11 08:40:12 +0000580 }
581
582 *checksum = (u16)I40E_SR_SW_CHECKSUM_BASE - checksum_local;
583
584i40e_calc_nvm_checksum_exit:
Kamil Krawczykd1bbe0e2015-01-24 09:58:33 +0000585 i40e_free_virt_mem(hw, &vmem);
Jesse Brandeburg56a62fc2013-09-11 08:40:12 +0000586 return ret_code;
587}
588
589/**
Shannon Nelsoncd552cb2014-07-09 07:46:09 +0000590 * i40e_update_nvm_checksum - Updates the NVM checksum
591 * @hw: pointer to hardware structure
592 *
593 * NVM ownership must be acquired before calling this function and released
594 * on ARQ completion event reception by caller.
595 * This function will commit SR to NVM.
596 **/
597i40e_status i40e_update_nvm_checksum(struct i40e_hw *hw)
598{
Jean Sacren0e5229c2015-10-13 01:06:31 -0600599 i40e_status ret_code;
Shannon Nelsoncd552cb2014-07-09 07:46:09 +0000600 u16 checksum;
Jesse Brandeburgdd38c582015-08-26 15:14:18 -0400601 __le16 le_sum;
Shannon Nelsoncd552cb2014-07-09 07:46:09 +0000602
603 ret_code = i40e_calc_nvm_checksum(hw, &checksum);
Jean Sacren2fc4cd52015-10-13 01:06:32 -0600604 if (!ret_code) {
605 le_sum = cpu_to_le16(checksum);
Shannon Nelsoncd552cb2014-07-09 07:46:09 +0000606 ret_code = i40e_write_nvm_aq(hw, 0x00, I40E_SR_SW_CHECKSUM_WORD,
Jesse Brandeburgdd38c582015-08-26 15:14:18 -0400607 1, &le_sum, true);
Jean Sacren2fc4cd52015-10-13 01:06:32 -0600608 }
Shannon Nelsoncd552cb2014-07-09 07:46:09 +0000609
610 return ret_code;
611}
612
613/**
Shannon Nelson3e261862014-02-06 05:51:06 +0000614 * i40e_validate_nvm_checksum - Validate EEPROM checksum
615 * @hw: pointer to hardware structure
616 * @checksum: calculated checksum
Jesse Brandeburg56a62fc2013-09-11 08:40:12 +0000617 *
Shannon Nelson3e261862014-02-06 05:51:06 +0000618 * Performs checksum calculation and validates the NVM SW checksum. If the
619 * caller does not need checksum, the value can be NULL.
Jesse Brandeburg56a62fc2013-09-11 08:40:12 +0000620 **/
621i40e_status i40e_validate_nvm_checksum(struct i40e_hw *hw,
622 u16 *checksum)
623{
624 i40e_status ret_code = 0;
625 u16 checksum_sr = 0;
Jesse Brandeburge15c9fa2014-01-17 15:36:31 -0800626 u16 checksum_local = 0;
Jesse Brandeburg56a62fc2013-09-11 08:40:12 +0000627
Anjali Singhai Jain09f79fd2017-09-01 13:42:49 -0700628 /* We must acquire the NVM lock in order to correctly synchronize the
629 * NVM accesses across multiple PFs. Without doing so it is possible
630 * for one of the PFs to read invalid data potentially indicating that
631 * the checksum is invalid.
Jesse Brandeburg56a62fc2013-09-11 08:40:12 +0000632 */
Anjali Singhai Jain09f79fd2017-09-01 13:42:49 -0700633 ret_code = i40e_acquire_nvm(hw, I40E_RESOURCE_READ);
634 if (ret_code)
635 return ret_code;
636 ret_code = i40e_calc_nvm_checksum(hw, &checksum_local);
637 __i40e_read_nvm_word(hw, I40E_SR_SW_CHECKSUM_WORD, &checksum_sr);
638 i40e_release_nvm(hw);
639 if (ret_code)
640 return ret_code;
Jesse Brandeburg56a62fc2013-09-11 08:40:12 +0000641
642 /* Verify read checksum from EEPROM is the same as
643 * calculated checksum
644 */
645 if (checksum_local != checksum_sr)
646 ret_code = I40E_ERR_NVM_CHECKSUM;
647
648 /* If the user cares, return the calculated checksum */
649 if (checksum)
650 *checksum = checksum_local;
651
Jesse Brandeburg56a62fc2013-09-11 08:40:12 +0000652 return ret_code;
653}
Shannon Nelsoncd552cb2014-07-09 07:46:09 +0000654
655static i40e_status i40e_nvmupd_state_init(struct i40e_hw *hw,
656 struct i40e_nvm_access *cmd,
Shannon Nelson79afe832015-07-23 16:54:33 -0400657 u8 *bytes, int *perrno);
Shannon Nelsoncd552cb2014-07-09 07:46:09 +0000658static i40e_status i40e_nvmupd_state_reading(struct i40e_hw *hw,
659 struct i40e_nvm_access *cmd,
Shannon Nelson79afe832015-07-23 16:54:33 -0400660 u8 *bytes, int *perrno);
Shannon Nelsoncd552cb2014-07-09 07:46:09 +0000661static i40e_status i40e_nvmupd_state_writing(struct i40e_hw *hw,
662 struct i40e_nvm_access *cmd,
663 u8 *bytes, int *errno);
664static enum i40e_nvmupd_cmd i40e_nvmupd_validate_command(struct i40e_hw *hw,
665 struct i40e_nvm_access *cmd,
Shannon Nelson79afe832015-07-23 16:54:33 -0400666 int *perrno);
Shannon Nelsoncd552cb2014-07-09 07:46:09 +0000667static i40e_status i40e_nvmupd_nvm_erase(struct i40e_hw *hw,
668 struct i40e_nvm_access *cmd,
Shannon Nelson79afe832015-07-23 16:54:33 -0400669 int *perrno);
Shannon Nelsoncd552cb2014-07-09 07:46:09 +0000670static i40e_status i40e_nvmupd_nvm_write(struct i40e_hw *hw,
671 struct i40e_nvm_access *cmd,
Shannon Nelson79afe832015-07-23 16:54:33 -0400672 u8 *bytes, int *perrno);
Shannon Nelsoncd552cb2014-07-09 07:46:09 +0000673static i40e_status i40e_nvmupd_nvm_read(struct i40e_hw *hw,
674 struct i40e_nvm_access *cmd,
Shannon Nelson79afe832015-07-23 16:54:33 -0400675 u8 *bytes, int *perrno);
Shannon Nelsone4c83c22015-08-28 17:55:50 -0400676static i40e_status i40e_nvmupd_exec_aq(struct i40e_hw *hw,
677 struct i40e_nvm_access *cmd,
678 u8 *bytes, int *perrno);
Shannon Nelsonb72dc7b2015-08-28 17:55:51 -0400679static i40e_status i40e_nvmupd_get_aq_result(struct i40e_hw *hw,
680 struct i40e_nvm_access *cmd,
681 u8 *bytes, int *perrno);
Pawel Jablonskie3a5d6e2017-12-18 05:14:44 -0500682static i40e_status i40e_nvmupd_get_aq_event(struct i40e_hw *hw,
683 struct i40e_nvm_access *cmd,
684 u8 *bytes, int *perrno);
Shannon Nelsoncd552cb2014-07-09 07:46:09 +0000685static inline u8 i40e_nvmupd_get_module(u32 val)
686{
687 return (u8)(val & I40E_NVM_MOD_PNT_MASK);
688}
689static inline u8 i40e_nvmupd_get_transaction(u32 val)
690{
691 return (u8)((val & I40E_NVM_TRANS_MASK) >> I40E_NVM_TRANS_SHIFT);
692}
693
Pawel Jablonskie3a5d6e2017-12-18 05:14:44 -0500694static inline u8 i40e_nvmupd_get_preservation_flags(u32 val)
695{
696 return (u8)((val & I40E_NVM_PRESERVATION_FLAGS_MASK) >>
697 I40E_NVM_PRESERVATION_FLAGS_SHIFT);
698}
699
Jingjing Wu4e68adfe2015-09-28 14:12:31 -0400700static const char * const i40e_nvm_update_state_str[] = {
Shannon Nelson74d0d0e2014-11-13 08:23:15 +0000701 "I40E_NVMUPD_INVALID",
702 "I40E_NVMUPD_READ_CON",
703 "I40E_NVMUPD_READ_SNT",
704 "I40E_NVMUPD_READ_LCB",
705 "I40E_NVMUPD_READ_SA",
706 "I40E_NVMUPD_WRITE_ERA",
707 "I40E_NVMUPD_WRITE_CON",
708 "I40E_NVMUPD_WRITE_SNT",
709 "I40E_NVMUPD_WRITE_LCB",
710 "I40E_NVMUPD_WRITE_SA",
711 "I40E_NVMUPD_CSUM_CON",
712 "I40E_NVMUPD_CSUM_SA",
713 "I40E_NVMUPD_CSUM_LCB",
Shannon Nelson0af8e9d2015-08-28 17:55:48 -0400714 "I40E_NVMUPD_STATUS",
Shannon Nelsone4c83c22015-08-28 17:55:50 -0400715 "I40E_NVMUPD_EXEC_AQ",
Shannon Nelsonb72dc7b2015-08-28 17:55:51 -0400716 "I40E_NVMUPD_GET_AQ_RESULT",
Pawel Jablonskie3a5d6e2017-12-18 05:14:44 -0500717 "I40E_NVMUPD_GET_AQ_EVENT",
Shannon Nelson74d0d0e2014-11-13 08:23:15 +0000718};
719
Shannon Nelsoncd552cb2014-07-09 07:46:09 +0000720/**
721 * i40e_nvmupd_command - Process an NVM update command
722 * @hw: pointer to hardware structure
723 * @cmd: pointer to nvm update command
724 * @bytes: pointer to the data buffer
Shannon Nelson79afe832015-07-23 16:54:33 -0400725 * @perrno: pointer to return error code
Shannon Nelsoncd552cb2014-07-09 07:46:09 +0000726 *
727 * Dispatches command depending on what update state is current
728 **/
729i40e_status i40e_nvmupd_command(struct i40e_hw *hw,
730 struct i40e_nvm_access *cmd,
Shannon Nelson79afe832015-07-23 16:54:33 -0400731 u8 *bytes, int *perrno)
Shannon Nelsoncd552cb2014-07-09 07:46:09 +0000732{
733 i40e_status status;
Shannon Nelson0af8e9d2015-08-28 17:55:48 -0400734 enum i40e_nvmupd_cmd upd_cmd;
Shannon Nelsoncd552cb2014-07-09 07:46:09 +0000735
736 /* assume success */
Shannon Nelson79afe832015-07-23 16:54:33 -0400737 *perrno = 0;
Shannon Nelsoncd552cb2014-07-09 07:46:09 +0000738
Shannon Nelson0af8e9d2015-08-28 17:55:48 -0400739 /* early check for status command and debug msgs */
740 upd_cmd = i40e_nvmupd_validate_command(hw, cmd, perrno);
741
Shannon Nelsonfed2db92016-04-12 08:30:43 -0700742 i40e_debug(hw, I40E_DEBUG_NVM, "%s state %d nvm_release_on_hold %d opc 0x%04x cmd 0x%08x config 0x%08x offset 0x%08x data_size 0x%08x\n",
Shannon Nelson0af8e9d2015-08-28 17:55:48 -0400743 i40e_nvm_update_state_str[upd_cmd],
744 hw->nvmupd_state,
Shannon Nelsonfed2db92016-04-12 08:30:43 -0700745 hw->nvm_release_on_done, hw->nvm_wait_opcode,
Shannon Nelson1d73b2d2015-12-23 12:05:51 -0800746 cmd->command, cmd->config, cmd->offset, cmd->data_size);
Shannon Nelson0af8e9d2015-08-28 17:55:48 -0400747
748 if (upd_cmd == I40E_NVMUPD_INVALID) {
749 *perrno = -EFAULT;
750 i40e_debug(hw, I40E_DEBUG_NVM,
751 "i40e_nvmupd_validate_command returns %d errno %d\n",
752 upd_cmd, *perrno);
753 }
754
755 /* a status request returns immediately rather than
756 * going into the state machine
757 */
758 if (upd_cmd == I40E_NVMUPD_STATUS) {
Shannon Nelsonfed2db92016-04-12 08:30:43 -0700759 if (!cmd->data_size) {
760 *perrno = -EFAULT;
761 return I40E_ERR_BUF_TOO_SHORT;
762 }
763
Shannon Nelson0af8e9d2015-08-28 17:55:48 -0400764 bytes[0] = hw->nvmupd_state;
Shannon Nelsonfed2db92016-04-12 08:30:43 -0700765
766 if (cmd->data_size >= 4) {
767 bytes[1] = 0;
768 *((u16 *)&bytes[2]) = hw->nvm_wait_opcode;
769 }
770
Maciej Sosin81fa7c92016-10-11 15:26:57 -0700771 /* Clear error status on read */
772 if (hw->nvmupd_state == I40E_NVMUPD_STATE_ERROR)
773 hw->nvmupd_state = I40E_NVMUPD_STATE_INIT;
774
Shannon Nelson0af8e9d2015-08-28 17:55:48 -0400775 return 0;
776 }
777
Maciej Sosin81fa7c92016-10-11 15:26:57 -0700778 /* Clear status even it is not read and log */
779 if (hw->nvmupd_state == I40E_NVMUPD_STATE_ERROR) {
780 i40e_debug(hw, I40E_DEBUG_NVM,
781 "Clearing I40E_NVMUPD_STATE_ERROR state without reading\n");
782 hw->nvmupd_state = I40E_NVMUPD_STATE_INIT;
783 }
784
Sudheer Mogilappagari2bf01932017-07-12 05:46:07 -0400785 /* Acquire lock to prevent race condition where adminq_task
786 * can execute after i40e_nvmupd_nvm_read/write but before state
Sudheer Mogilappagari167d52e2017-08-27 15:07:47 -0700787 * variables (nvm_wait_opcode, nvm_release_on_done) are updated.
788 *
789 * During NVMUpdate, it is observed that lock could be held for
790 * ~5ms for most commands. However lock is held for ~60ms for
791 * NVMUPD_CSUM_LCB command.
Sudheer Mogilappagari2bf01932017-07-12 05:46:07 -0400792 */
793 mutex_lock(&hw->aq.arq_mutex);
Shannon Nelsoncd552cb2014-07-09 07:46:09 +0000794 switch (hw->nvmupd_state) {
795 case I40E_NVMUPD_STATE_INIT:
Shannon Nelson79afe832015-07-23 16:54:33 -0400796 status = i40e_nvmupd_state_init(hw, cmd, bytes, perrno);
Shannon Nelsoncd552cb2014-07-09 07:46:09 +0000797 break;
798
799 case I40E_NVMUPD_STATE_READING:
Shannon Nelson79afe832015-07-23 16:54:33 -0400800 status = i40e_nvmupd_state_reading(hw, cmd, bytes, perrno);
Shannon Nelsoncd552cb2014-07-09 07:46:09 +0000801 break;
802
803 case I40E_NVMUPD_STATE_WRITING:
Shannon Nelson79afe832015-07-23 16:54:33 -0400804 status = i40e_nvmupd_state_writing(hw, cmd, bytes, perrno);
Shannon Nelsoncd552cb2014-07-09 07:46:09 +0000805 break;
806
Shannon Nelson2f1b5bc2015-08-28 17:55:49 -0400807 case I40E_NVMUPD_STATE_INIT_WAIT:
808 case I40E_NVMUPD_STATE_WRITE_WAIT:
Shannon Nelsonfed2db92016-04-12 08:30:43 -0700809 /* if we need to stop waiting for an event, clear
810 * the wait info and return before doing anything else
811 */
812 if (cmd->offset == 0xffff) {
Pawel Jablonskie3a5d6e2017-12-18 05:14:44 -0500813 i40e_nvmupd_clear_wait_state(hw);
Sudheer Mogilappagari167d52e2017-08-27 15:07:47 -0700814 status = 0;
Pawel Jablonskie3a5d6e2017-12-18 05:14:44 -0500815 break;
Shannon Nelsonfed2db92016-04-12 08:30:43 -0700816 }
817
Shannon Nelson2f1b5bc2015-08-28 17:55:49 -0400818 status = I40E_ERR_NOT_READY;
819 *perrno = -EBUSY;
820 break;
821
Shannon Nelsoncd552cb2014-07-09 07:46:09 +0000822 default:
823 /* invalid state, should never happen */
Shannon Nelson74d0d0e2014-11-13 08:23:15 +0000824 i40e_debug(hw, I40E_DEBUG_NVM,
825 "NVMUPD: no such state %d\n", hw->nvmupd_state);
Shannon Nelsoncd552cb2014-07-09 07:46:09 +0000826 status = I40E_NOT_SUPPORTED;
Shannon Nelson79afe832015-07-23 16:54:33 -0400827 *perrno = -ESRCH;
Shannon Nelsoncd552cb2014-07-09 07:46:09 +0000828 break;
829 }
Pawel Jablonskie3a5d6e2017-12-18 05:14:44 -0500830
Sudheer Mogilappagari2bf01932017-07-12 05:46:07 -0400831 mutex_unlock(&hw->aq.arq_mutex);
Shannon Nelsoncd552cb2014-07-09 07:46:09 +0000832 return status;
833}
834
835/**
836 * i40e_nvmupd_state_init - Handle NVM update state Init
837 * @hw: pointer to hardware structure
838 * @cmd: pointer to nvm update command buffer
839 * @bytes: pointer to the data buffer
Shannon Nelson79afe832015-07-23 16:54:33 -0400840 * @perrno: pointer to return error code
Shannon Nelsoncd552cb2014-07-09 07:46:09 +0000841 *
842 * Process legitimate commands of the Init state and conditionally set next
843 * state. Reject all other commands.
844 **/
845static i40e_status i40e_nvmupd_state_init(struct i40e_hw *hw,
846 struct i40e_nvm_access *cmd,
Shannon Nelson79afe832015-07-23 16:54:33 -0400847 u8 *bytes, int *perrno)
Shannon Nelsoncd552cb2014-07-09 07:46:09 +0000848{
849 i40e_status status = 0;
850 enum i40e_nvmupd_cmd upd_cmd;
851
Shannon Nelson79afe832015-07-23 16:54:33 -0400852 upd_cmd = i40e_nvmupd_validate_command(hw, cmd, perrno);
Shannon Nelsoncd552cb2014-07-09 07:46:09 +0000853
854 switch (upd_cmd) {
855 case I40E_NVMUPD_READ_SA:
856 status = i40e_acquire_nvm(hw, I40E_RESOURCE_READ);
857 if (status) {
Shannon Nelson79afe832015-07-23 16:54:33 -0400858 *perrno = i40e_aq_rc_to_posix(status,
Shannon Nelsonbf848f32014-11-13 08:23:22 +0000859 hw->aq.asq_last_status);
Shannon Nelsoncd552cb2014-07-09 07:46:09 +0000860 } else {
Shannon Nelson79afe832015-07-23 16:54:33 -0400861 status = i40e_nvmupd_nvm_read(hw, cmd, bytes, perrno);
Shannon Nelsoncd552cb2014-07-09 07:46:09 +0000862 i40e_release_nvm(hw);
863 }
864 break;
865
866 case I40E_NVMUPD_READ_SNT:
867 status = i40e_acquire_nvm(hw, I40E_RESOURCE_READ);
868 if (status) {
Shannon Nelson79afe832015-07-23 16:54:33 -0400869 *perrno = i40e_aq_rc_to_posix(status,
Shannon Nelsonbf848f32014-11-13 08:23:22 +0000870 hw->aq.asq_last_status);
Shannon Nelsoncd552cb2014-07-09 07:46:09 +0000871 } else {
Shannon Nelson79afe832015-07-23 16:54:33 -0400872 status = i40e_nvmupd_nvm_read(hw, cmd, bytes, perrno);
Shannon Nelson0fdd0522014-11-13 08:23:20 +0000873 if (status)
874 i40e_release_nvm(hw);
875 else
876 hw->nvmupd_state = I40E_NVMUPD_STATE_READING;
Shannon Nelsoncd552cb2014-07-09 07:46:09 +0000877 }
878 break;
879
880 case I40E_NVMUPD_WRITE_ERA:
881 status = i40e_acquire_nvm(hw, I40E_RESOURCE_WRITE);
882 if (status) {
Shannon Nelson79afe832015-07-23 16:54:33 -0400883 *perrno = i40e_aq_rc_to_posix(status,
Shannon Nelsonbf848f32014-11-13 08:23:22 +0000884 hw->aq.asq_last_status);
Shannon Nelsoncd552cb2014-07-09 07:46:09 +0000885 } else {
Shannon Nelson79afe832015-07-23 16:54:33 -0400886 status = i40e_nvmupd_nvm_erase(hw, cmd, perrno);
Shannon Nelson2f1b5bc2015-08-28 17:55:49 -0400887 if (status) {
Shannon Nelsoncd552cb2014-07-09 07:46:09 +0000888 i40e_release_nvm(hw);
Shannon Nelson2f1b5bc2015-08-28 17:55:49 -0400889 } else {
Shannon Nelson437f82a2016-04-01 03:56:09 -0700890 hw->nvm_release_on_done = true;
Shannon Nelsonfed2db92016-04-12 08:30:43 -0700891 hw->nvm_wait_opcode = i40e_aqc_opc_nvm_erase;
Shannon Nelson2f1b5bc2015-08-28 17:55:49 -0400892 hw->nvmupd_state = I40E_NVMUPD_STATE_INIT_WAIT;
893 }
Shannon Nelsoncd552cb2014-07-09 07:46:09 +0000894 }
895 break;
896
897 case I40E_NVMUPD_WRITE_SA:
898 status = i40e_acquire_nvm(hw, I40E_RESOURCE_WRITE);
899 if (status) {
Shannon Nelson79afe832015-07-23 16:54:33 -0400900 *perrno = i40e_aq_rc_to_posix(status,
Shannon Nelsonbf848f32014-11-13 08:23:22 +0000901 hw->aq.asq_last_status);
Shannon Nelsoncd552cb2014-07-09 07:46:09 +0000902 } else {
Shannon Nelson79afe832015-07-23 16:54:33 -0400903 status = i40e_nvmupd_nvm_write(hw, cmd, bytes, perrno);
Shannon Nelson2f1b5bc2015-08-28 17:55:49 -0400904 if (status) {
Shannon Nelsoncd552cb2014-07-09 07:46:09 +0000905 i40e_release_nvm(hw);
Shannon Nelson2f1b5bc2015-08-28 17:55:49 -0400906 } else {
Shannon Nelson437f82a2016-04-01 03:56:09 -0700907 hw->nvm_release_on_done = true;
Shannon Nelsonfed2db92016-04-12 08:30:43 -0700908 hw->nvm_wait_opcode = i40e_aqc_opc_nvm_update;
Shannon Nelson2f1b5bc2015-08-28 17:55:49 -0400909 hw->nvmupd_state = I40E_NVMUPD_STATE_INIT_WAIT;
910 }
Shannon Nelsoncd552cb2014-07-09 07:46:09 +0000911 }
912 break;
913
914 case I40E_NVMUPD_WRITE_SNT:
915 status = i40e_acquire_nvm(hw, I40E_RESOURCE_WRITE);
916 if (status) {
Shannon Nelson79afe832015-07-23 16:54:33 -0400917 *perrno = i40e_aq_rc_to_posix(status,
Shannon Nelsonbf848f32014-11-13 08:23:22 +0000918 hw->aq.asq_last_status);
Shannon Nelsoncd552cb2014-07-09 07:46:09 +0000919 } else {
Shannon Nelson79afe832015-07-23 16:54:33 -0400920 status = i40e_nvmupd_nvm_write(hw, cmd, bytes, perrno);
Shannon Nelsonfed2db92016-04-12 08:30:43 -0700921 if (status) {
Shannon Nelson0fdd0522014-11-13 08:23:20 +0000922 i40e_release_nvm(hw);
Shannon Nelsonfed2db92016-04-12 08:30:43 -0700923 } else {
924 hw->nvm_wait_opcode = i40e_aqc_opc_nvm_update;
Shannon Nelson2f1b5bc2015-08-28 17:55:49 -0400925 hw->nvmupd_state = I40E_NVMUPD_STATE_WRITE_WAIT;
Shannon Nelsonfed2db92016-04-12 08:30:43 -0700926 }
Shannon Nelsoncd552cb2014-07-09 07:46:09 +0000927 }
928 break;
929
930 case I40E_NVMUPD_CSUM_SA:
931 status = i40e_acquire_nvm(hw, I40E_RESOURCE_WRITE);
932 if (status) {
Shannon Nelson79afe832015-07-23 16:54:33 -0400933 *perrno = i40e_aq_rc_to_posix(status,
Shannon Nelsonbf848f32014-11-13 08:23:22 +0000934 hw->aq.asq_last_status);
Shannon Nelsoncd552cb2014-07-09 07:46:09 +0000935 } else {
936 status = i40e_update_nvm_checksum(hw);
937 if (status) {
Shannon Nelson79afe832015-07-23 16:54:33 -0400938 *perrno = hw->aq.asq_last_status ?
Shannon Nelsonbf848f32014-11-13 08:23:22 +0000939 i40e_aq_rc_to_posix(status,
940 hw->aq.asq_last_status) :
Shannon Nelsoncd552cb2014-07-09 07:46:09 +0000941 -EIO;
942 i40e_release_nvm(hw);
943 } else {
Shannon Nelson437f82a2016-04-01 03:56:09 -0700944 hw->nvm_release_on_done = true;
Shannon Nelsonfed2db92016-04-12 08:30:43 -0700945 hw->nvm_wait_opcode = i40e_aqc_opc_nvm_update;
Shannon Nelson2f1b5bc2015-08-28 17:55:49 -0400946 hw->nvmupd_state = I40E_NVMUPD_STATE_INIT_WAIT;
Shannon Nelsoncd552cb2014-07-09 07:46:09 +0000947 }
948 }
949 break;
950
Shannon Nelsone4c83c22015-08-28 17:55:50 -0400951 case I40E_NVMUPD_EXEC_AQ:
952 status = i40e_nvmupd_exec_aq(hw, cmd, bytes, perrno);
953 break;
954
Shannon Nelsonb72dc7b2015-08-28 17:55:51 -0400955 case I40E_NVMUPD_GET_AQ_RESULT:
956 status = i40e_nvmupd_get_aq_result(hw, cmd, bytes, perrno);
957 break;
958
Pawel Jablonskie3a5d6e2017-12-18 05:14:44 -0500959 case I40E_NVMUPD_GET_AQ_EVENT:
960 status = i40e_nvmupd_get_aq_event(hw, cmd, bytes, perrno);
961 break;
962
Shannon Nelsoncd552cb2014-07-09 07:46:09 +0000963 default:
Shannon Nelson74d0d0e2014-11-13 08:23:15 +0000964 i40e_debug(hw, I40E_DEBUG_NVM,
965 "NVMUPD: bad cmd %s in init state\n",
966 i40e_nvm_update_state_str[upd_cmd]);
Shannon Nelsoncd552cb2014-07-09 07:46:09 +0000967 status = I40E_ERR_NVM;
Shannon Nelson79afe832015-07-23 16:54:33 -0400968 *perrno = -ESRCH;
Shannon Nelsoncd552cb2014-07-09 07:46:09 +0000969 break;
970 }
971 return status;
972}
973
974/**
975 * i40e_nvmupd_state_reading - Handle NVM update state Reading
976 * @hw: pointer to hardware structure
977 * @cmd: pointer to nvm update command buffer
978 * @bytes: pointer to the data buffer
Shannon Nelson79afe832015-07-23 16:54:33 -0400979 * @perrno: pointer to return error code
Shannon Nelsoncd552cb2014-07-09 07:46:09 +0000980 *
981 * NVM ownership is already held. Process legitimate commands and set any
982 * change in state; reject all other commands.
983 **/
984static i40e_status i40e_nvmupd_state_reading(struct i40e_hw *hw,
985 struct i40e_nvm_access *cmd,
Shannon Nelson79afe832015-07-23 16:54:33 -0400986 u8 *bytes, int *perrno)
Shannon Nelsoncd552cb2014-07-09 07:46:09 +0000987{
Shannon Nelson2f1b5bc2015-08-28 17:55:49 -0400988 i40e_status status = 0;
Shannon Nelsoncd552cb2014-07-09 07:46:09 +0000989 enum i40e_nvmupd_cmd upd_cmd;
990
Shannon Nelson79afe832015-07-23 16:54:33 -0400991 upd_cmd = i40e_nvmupd_validate_command(hw, cmd, perrno);
Shannon Nelsoncd552cb2014-07-09 07:46:09 +0000992
993 switch (upd_cmd) {
994 case I40E_NVMUPD_READ_SA:
995 case I40E_NVMUPD_READ_CON:
Shannon Nelson79afe832015-07-23 16:54:33 -0400996 status = i40e_nvmupd_nvm_read(hw, cmd, bytes, perrno);
Shannon Nelsoncd552cb2014-07-09 07:46:09 +0000997 break;
998
999 case I40E_NVMUPD_READ_LCB:
Shannon Nelson79afe832015-07-23 16:54:33 -04001000 status = i40e_nvmupd_nvm_read(hw, cmd, bytes, perrno);
Shannon Nelsoncd552cb2014-07-09 07:46:09 +00001001 i40e_release_nvm(hw);
1002 hw->nvmupd_state = I40E_NVMUPD_STATE_INIT;
1003 break;
1004
1005 default:
Shannon Nelson74d0d0e2014-11-13 08:23:15 +00001006 i40e_debug(hw, I40E_DEBUG_NVM,
1007 "NVMUPD: bad cmd %s in reading state.\n",
1008 i40e_nvm_update_state_str[upd_cmd]);
Shannon Nelsoncd552cb2014-07-09 07:46:09 +00001009 status = I40E_NOT_SUPPORTED;
Shannon Nelson79afe832015-07-23 16:54:33 -04001010 *perrno = -ESRCH;
Shannon Nelsoncd552cb2014-07-09 07:46:09 +00001011 break;
1012 }
1013 return status;
1014}
1015
1016/**
1017 * i40e_nvmupd_state_writing - Handle NVM update state Writing
1018 * @hw: pointer to hardware structure
1019 * @cmd: pointer to nvm update command buffer
1020 * @bytes: pointer to the data buffer
Shannon Nelson79afe832015-07-23 16:54:33 -04001021 * @perrno: pointer to return error code
Shannon Nelsoncd552cb2014-07-09 07:46:09 +00001022 *
1023 * NVM ownership is already held. Process legitimate commands and set any
1024 * change in state; reject all other commands
1025 **/
1026static i40e_status i40e_nvmupd_state_writing(struct i40e_hw *hw,
1027 struct i40e_nvm_access *cmd,
Shannon Nelson79afe832015-07-23 16:54:33 -04001028 u8 *bytes, int *perrno)
Shannon Nelsoncd552cb2014-07-09 07:46:09 +00001029{
Shannon Nelson2f1b5bc2015-08-28 17:55:49 -04001030 i40e_status status = 0;
Shannon Nelsoncd552cb2014-07-09 07:46:09 +00001031 enum i40e_nvmupd_cmd upd_cmd;
Shannon Nelson2c47e352015-02-21 06:45:10 +00001032 bool retry_attempt = false;
Shannon Nelsoncd552cb2014-07-09 07:46:09 +00001033
Shannon Nelson79afe832015-07-23 16:54:33 -04001034 upd_cmd = i40e_nvmupd_validate_command(hw, cmd, perrno);
Shannon Nelsoncd552cb2014-07-09 07:46:09 +00001035
Shannon Nelson2c47e352015-02-21 06:45:10 +00001036retry:
Shannon Nelsoncd552cb2014-07-09 07:46:09 +00001037 switch (upd_cmd) {
1038 case I40E_NVMUPD_WRITE_CON:
Shannon Nelson79afe832015-07-23 16:54:33 -04001039 status = i40e_nvmupd_nvm_write(hw, cmd, bytes, perrno);
Shannon Nelsonfed2db92016-04-12 08:30:43 -07001040 if (!status) {
1041 hw->nvm_wait_opcode = i40e_aqc_opc_nvm_update;
Shannon Nelson2f1b5bc2015-08-28 17:55:49 -04001042 hw->nvmupd_state = I40E_NVMUPD_STATE_WRITE_WAIT;
Shannon Nelsonfed2db92016-04-12 08:30:43 -07001043 }
Shannon Nelsoncd552cb2014-07-09 07:46:09 +00001044 break;
1045
1046 case I40E_NVMUPD_WRITE_LCB:
Shannon Nelson79afe832015-07-23 16:54:33 -04001047 status = i40e_nvmupd_nvm_write(hw, cmd, bytes, perrno);
Shannon Nelson2f1b5bc2015-08-28 17:55:49 -04001048 if (status) {
1049 *perrno = hw->aq.asq_last_status ?
1050 i40e_aq_rc_to_posix(status,
1051 hw->aq.asq_last_status) :
1052 -EIO;
1053 hw->nvmupd_state = I40E_NVMUPD_STATE_INIT;
1054 } else {
Shannon Nelson437f82a2016-04-01 03:56:09 -07001055 hw->nvm_release_on_done = true;
Shannon Nelsonfed2db92016-04-12 08:30:43 -07001056 hw->nvm_wait_opcode = i40e_aqc_opc_nvm_update;
Shannon Nelson2f1b5bc2015-08-28 17:55:49 -04001057 hw->nvmupd_state = I40E_NVMUPD_STATE_INIT_WAIT;
1058 }
Shannon Nelsoncd552cb2014-07-09 07:46:09 +00001059 break;
1060
1061 case I40E_NVMUPD_CSUM_CON:
Anjali Singhai Jain09f79fd2017-09-01 13:42:49 -07001062 /* Assumes the caller has acquired the nvm */
Shannon Nelsoncd552cb2014-07-09 07:46:09 +00001063 status = i40e_update_nvm_checksum(hw);
Shannon Nelsoncd552cb2014-07-09 07:46:09 +00001064 if (status) {
Shannon Nelson79afe832015-07-23 16:54:33 -04001065 *perrno = hw->aq.asq_last_status ?
Shannon Nelsonbf848f32014-11-13 08:23:22 +00001066 i40e_aq_rc_to_posix(status,
1067 hw->aq.asq_last_status) :
Shannon Nelsoncd552cb2014-07-09 07:46:09 +00001068 -EIO;
Shannon Nelsoncd552cb2014-07-09 07:46:09 +00001069 hw->nvmupd_state = I40E_NVMUPD_STATE_INIT;
Shannon Nelson2f1b5bc2015-08-28 17:55:49 -04001070 } else {
Shannon Nelsonfed2db92016-04-12 08:30:43 -07001071 hw->nvm_wait_opcode = i40e_aqc_opc_nvm_update;
Shannon Nelson2f1b5bc2015-08-28 17:55:49 -04001072 hw->nvmupd_state = I40E_NVMUPD_STATE_WRITE_WAIT;
Shannon Nelsoncd552cb2014-07-09 07:46:09 +00001073 }
1074 break;
1075
Shannon Nelson0fdd0522014-11-13 08:23:20 +00001076 case I40E_NVMUPD_CSUM_LCB:
Anjali Singhai Jain09f79fd2017-09-01 13:42:49 -07001077 /* Assumes the caller has acquired the nvm */
Shannon Nelson0fdd0522014-11-13 08:23:20 +00001078 status = i40e_update_nvm_checksum(hw);
Shannon Nelson2f1b5bc2015-08-28 17:55:49 -04001079 if (status) {
Shannon Nelson79afe832015-07-23 16:54:33 -04001080 *perrno = hw->aq.asq_last_status ?
Shannon Nelsonbf848f32014-11-13 08:23:22 +00001081 i40e_aq_rc_to_posix(status,
1082 hw->aq.asq_last_status) :
Shannon Nelson0fdd0522014-11-13 08:23:20 +00001083 -EIO;
Shannon Nelson2f1b5bc2015-08-28 17:55:49 -04001084 hw->nvmupd_state = I40E_NVMUPD_STATE_INIT;
1085 } else {
Shannon Nelson437f82a2016-04-01 03:56:09 -07001086 hw->nvm_release_on_done = true;
Shannon Nelsonfed2db92016-04-12 08:30:43 -07001087 hw->nvm_wait_opcode = i40e_aqc_opc_nvm_update;
Shannon Nelson2f1b5bc2015-08-28 17:55:49 -04001088 hw->nvmupd_state = I40E_NVMUPD_STATE_INIT_WAIT;
1089 }
Shannon Nelson0fdd0522014-11-13 08:23:20 +00001090 break;
1091
Shannon Nelsoncd552cb2014-07-09 07:46:09 +00001092 default:
Shannon Nelson74d0d0e2014-11-13 08:23:15 +00001093 i40e_debug(hw, I40E_DEBUG_NVM,
1094 "NVMUPD: bad cmd %s in writing state.\n",
1095 i40e_nvm_update_state_str[upd_cmd]);
Shannon Nelsoncd552cb2014-07-09 07:46:09 +00001096 status = I40E_NOT_SUPPORTED;
Shannon Nelson79afe832015-07-23 16:54:33 -04001097 *perrno = -ESRCH;
Shannon Nelsoncd552cb2014-07-09 07:46:09 +00001098 break;
1099 }
Shannon Nelson2c47e352015-02-21 06:45:10 +00001100
1101 /* In some circumstances, a multi-write transaction takes longer
1102 * than the default 3 minute timeout on the write semaphore. If
1103 * the write failed with an EBUSY status, this is likely the problem,
1104 * so here we try to reacquire the semaphore then retry the write.
1105 * We only do one retry, then give up.
1106 */
1107 if (status && (hw->aq.asq_last_status == I40E_AQ_RC_EBUSY) &&
1108 !retry_attempt) {
1109 i40e_status old_status = status;
1110 u32 old_asq_status = hw->aq.asq_last_status;
1111 u32 gtime;
1112
1113 gtime = rd32(hw, I40E_GLVFGEN_TIMER);
1114 if (gtime >= hw->nvm.hw_semaphore_timeout) {
1115 i40e_debug(hw, I40E_DEBUG_ALL,
1116 "NVMUPD: write semaphore expired (%d >= %lld), retrying\n",
1117 gtime, hw->nvm.hw_semaphore_timeout);
1118 i40e_release_nvm(hw);
1119 status = i40e_acquire_nvm(hw, I40E_RESOURCE_WRITE);
1120 if (status) {
1121 i40e_debug(hw, I40E_DEBUG_ALL,
1122 "NVMUPD: write semaphore reacquire failed aq_err = %d\n",
1123 hw->aq.asq_last_status);
1124 status = old_status;
1125 hw->aq.asq_last_status = old_asq_status;
1126 } else {
1127 retry_attempt = true;
1128 goto retry;
1129 }
1130 }
1131 }
1132
Shannon Nelsoncd552cb2014-07-09 07:46:09 +00001133 return status;
1134}
1135
1136/**
Pawel Jablonskie3a5d6e2017-12-18 05:14:44 -05001137 * i40e_nvmupd_clear_wait_state - clear wait state on hw
1138 * @hw: pointer to the hardware structure
1139 **/
1140void i40e_nvmupd_clear_wait_state(struct i40e_hw *hw)
1141{
1142 i40e_debug(hw, I40E_DEBUG_NVM,
1143 "NVMUPD: clearing wait on opcode 0x%04x\n",
1144 hw->nvm_wait_opcode);
1145
1146 if (hw->nvm_release_on_done) {
1147 i40e_release_nvm(hw);
1148 hw->nvm_release_on_done = false;
1149 }
1150 hw->nvm_wait_opcode = 0;
1151
1152 if (hw->aq.arq_last_status) {
1153 hw->nvmupd_state = I40E_NVMUPD_STATE_ERROR;
1154 return;
1155 }
1156
1157 switch (hw->nvmupd_state) {
1158 case I40E_NVMUPD_STATE_INIT_WAIT:
1159 hw->nvmupd_state = I40E_NVMUPD_STATE_INIT;
1160 break;
1161
1162 case I40E_NVMUPD_STATE_WRITE_WAIT:
1163 hw->nvmupd_state = I40E_NVMUPD_STATE_WRITING;
1164 break;
1165
1166 default:
1167 break;
1168 }
1169}
1170
1171/**
Shannon Nelsonbab2fb62016-04-01 03:56:11 -07001172 * i40e_nvmupd_check_wait_event - handle NVM update operation events
1173 * @hw: pointer to the hardware structure
1174 * @opcode: the event that just happened
1175 **/
Pawel Jablonskie3a5d6e2017-12-18 05:14:44 -05001176void i40e_nvmupd_check_wait_event(struct i40e_hw *hw, u16 opcode,
1177 struct i40e_aq_desc *desc)
Shannon Nelsonbab2fb62016-04-01 03:56:11 -07001178{
Pawel Jablonskie3a5d6e2017-12-18 05:14:44 -05001179 u32 aq_desc_len = sizeof(struct i40e_aq_desc);
1180
Shannon Nelsonfed2db92016-04-12 08:30:43 -07001181 if (opcode == hw->nvm_wait_opcode) {
Pawel Jablonskie3a5d6e2017-12-18 05:14:44 -05001182 memcpy(&hw->nvm_aq_event_desc, desc, aq_desc_len);
1183 i40e_nvmupd_clear_wait_state(hw);
Shannon Nelsonbab2fb62016-04-01 03:56:11 -07001184 }
1185}
1186
1187/**
Shannon Nelsoncd552cb2014-07-09 07:46:09 +00001188 * i40e_nvmupd_validate_command - Validate given command
1189 * @hw: pointer to hardware structure
1190 * @cmd: pointer to nvm update command buffer
Shannon Nelson79afe832015-07-23 16:54:33 -04001191 * @perrno: pointer to return error code
Shannon Nelsoncd552cb2014-07-09 07:46:09 +00001192 *
1193 * Return one of the valid command types or I40E_NVMUPD_INVALID
1194 **/
1195static enum i40e_nvmupd_cmd i40e_nvmupd_validate_command(struct i40e_hw *hw,
1196 struct i40e_nvm_access *cmd,
Shannon Nelson79afe832015-07-23 16:54:33 -04001197 int *perrno)
Shannon Nelsoncd552cb2014-07-09 07:46:09 +00001198{
1199 enum i40e_nvmupd_cmd upd_cmd;
Shannon Nelson0af8e9d2015-08-28 17:55:48 -04001200 u8 module, transaction;
Shannon Nelsoncd552cb2014-07-09 07:46:09 +00001201
1202 /* anything that doesn't match a recognized case is an error */
1203 upd_cmd = I40E_NVMUPD_INVALID;
1204
1205 transaction = i40e_nvmupd_get_transaction(cmd->config);
Shannon Nelson0af8e9d2015-08-28 17:55:48 -04001206 module = i40e_nvmupd_get_module(cmd->config);
Shannon Nelsoncd552cb2014-07-09 07:46:09 +00001207
1208 /* limits on data size */
1209 if ((cmd->data_size < 1) ||
1210 (cmd->data_size > I40E_NVMUPD_MAX_DATA)) {
Shannon Nelson74d0d0e2014-11-13 08:23:15 +00001211 i40e_debug(hw, I40E_DEBUG_NVM,
1212 "i40e_nvmupd_validate_command data_size %d\n",
1213 cmd->data_size);
Shannon Nelson79afe832015-07-23 16:54:33 -04001214 *perrno = -EFAULT;
Shannon Nelsoncd552cb2014-07-09 07:46:09 +00001215 return I40E_NVMUPD_INVALID;
1216 }
1217
1218 switch (cmd->command) {
1219 case I40E_NVM_READ:
1220 switch (transaction) {
1221 case I40E_NVM_CON:
1222 upd_cmd = I40E_NVMUPD_READ_CON;
1223 break;
1224 case I40E_NVM_SNT:
1225 upd_cmd = I40E_NVMUPD_READ_SNT;
1226 break;
1227 case I40E_NVM_LCB:
1228 upd_cmd = I40E_NVMUPD_READ_LCB;
1229 break;
1230 case I40E_NVM_SA:
1231 upd_cmd = I40E_NVMUPD_READ_SA;
1232 break;
Shannon Nelson0af8e9d2015-08-28 17:55:48 -04001233 case I40E_NVM_EXEC:
1234 if (module == 0xf)
1235 upd_cmd = I40E_NVMUPD_STATUS;
Shannon Nelsonb72dc7b2015-08-28 17:55:51 -04001236 else if (module == 0)
1237 upd_cmd = I40E_NVMUPD_GET_AQ_RESULT;
Shannon Nelson0af8e9d2015-08-28 17:55:48 -04001238 break;
Pawel Jablonskie3a5d6e2017-12-18 05:14:44 -05001239 case I40E_NVM_AQE:
1240 upd_cmd = I40E_NVMUPD_GET_AQ_EVENT;
1241 break;
Shannon Nelsoncd552cb2014-07-09 07:46:09 +00001242 }
1243 break;
1244
1245 case I40E_NVM_WRITE:
1246 switch (transaction) {
1247 case I40E_NVM_CON:
1248 upd_cmd = I40E_NVMUPD_WRITE_CON;
1249 break;
1250 case I40E_NVM_SNT:
1251 upd_cmd = I40E_NVMUPD_WRITE_SNT;
1252 break;
1253 case I40E_NVM_LCB:
1254 upd_cmd = I40E_NVMUPD_WRITE_LCB;
1255 break;
1256 case I40E_NVM_SA:
1257 upd_cmd = I40E_NVMUPD_WRITE_SA;
1258 break;
1259 case I40E_NVM_ERA:
1260 upd_cmd = I40E_NVMUPD_WRITE_ERA;
1261 break;
1262 case I40E_NVM_CSUM:
1263 upd_cmd = I40E_NVMUPD_CSUM_CON;
1264 break;
1265 case (I40E_NVM_CSUM|I40E_NVM_SA):
1266 upd_cmd = I40E_NVMUPD_CSUM_SA;
1267 break;
1268 case (I40E_NVM_CSUM|I40E_NVM_LCB):
1269 upd_cmd = I40E_NVMUPD_CSUM_LCB;
1270 break;
Shannon Nelsone4c83c22015-08-28 17:55:50 -04001271 case I40E_NVM_EXEC:
1272 if (module == 0)
1273 upd_cmd = I40E_NVMUPD_EXEC_AQ;
1274 break;
Shannon Nelsoncd552cb2014-07-09 07:46:09 +00001275 }
1276 break;
1277 }
1278
Shannon Nelsoncd552cb2014-07-09 07:46:09 +00001279 return upd_cmd;
1280}
1281
1282/**
Shannon Nelsone4c83c22015-08-28 17:55:50 -04001283 * i40e_nvmupd_exec_aq - Run an AQ command
1284 * @hw: pointer to hardware structure
1285 * @cmd: pointer to nvm update command buffer
1286 * @bytes: pointer to the data buffer
1287 * @perrno: pointer to return error code
1288 *
1289 * cmd structure contains identifiers and data buffer
1290 **/
1291static i40e_status i40e_nvmupd_exec_aq(struct i40e_hw *hw,
1292 struct i40e_nvm_access *cmd,
1293 u8 *bytes, int *perrno)
1294{
1295 struct i40e_asq_cmd_details cmd_details;
1296 i40e_status status;
1297 struct i40e_aq_desc *aq_desc;
1298 u32 buff_size = 0;
1299 u8 *buff = NULL;
1300 u32 aq_desc_len;
1301 u32 aq_data_len;
1302
1303 i40e_debug(hw, I40E_DEBUG_NVM, "NVMUPD: %s\n", __func__);
Pawel Jablonskie3a5d6e2017-12-18 05:14:44 -05001304 if (cmd->offset == 0xffff)
1305 return 0;
1306
Shannon Nelsone4c83c22015-08-28 17:55:50 -04001307 memset(&cmd_details, 0, sizeof(cmd_details));
1308 cmd_details.wb_desc = &hw->nvm_wb_desc;
1309
1310 aq_desc_len = sizeof(struct i40e_aq_desc);
1311 memset(&hw->nvm_wb_desc, 0, aq_desc_len);
1312
1313 /* get the aq descriptor */
1314 if (cmd->data_size < aq_desc_len) {
1315 i40e_debug(hw, I40E_DEBUG_NVM,
1316 "NVMUPD: not enough aq desc bytes for exec, size %d < %d\n",
1317 cmd->data_size, aq_desc_len);
1318 *perrno = -EINVAL;
1319 return I40E_ERR_PARAM;
1320 }
1321 aq_desc = (struct i40e_aq_desc *)bytes;
1322
1323 /* if data buffer needed, make sure it's ready */
1324 aq_data_len = cmd->data_size - aq_desc_len;
1325 buff_size = max_t(u32, aq_data_len, le16_to_cpu(aq_desc->datalen));
1326 if (buff_size) {
1327 if (!hw->nvm_buff.va) {
1328 status = i40e_allocate_virt_mem(hw, &hw->nvm_buff,
1329 hw->aq.asq_buf_size);
1330 if (status)
1331 i40e_debug(hw, I40E_DEBUG_NVM,
1332 "NVMUPD: i40e_allocate_virt_mem for exec buff failed, %d\n",
1333 status);
1334 }
1335
1336 if (hw->nvm_buff.va) {
1337 buff = hw->nvm_buff.va;
1338 memcpy(buff, &bytes[aq_desc_len], aq_data_len);
1339 }
1340 }
1341
Pawel Jablonskie3a5d6e2017-12-18 05:14:44 -05001342 if (cmd->offset)
1343 memset(&hw->nvm_aq_event_desc, 0, aq_desc_len);
1344
Shannon Nelsone4c83c22015-08-28 17:55:50 -04001345 /* and away we go! */
1346 status = i40e_asq_send_command(hw, aq_desc, buff,
1347 buff_size, &cmd_details);
1348 if (status) {
1349 i40e_debug(hw, I40E_DEBUG_NVM,
1350 "i40e_nvmupd_exec_aq err %s aq_err %s\n",
1351 i40e_stat_str(hw, status),
1352 i40e_aq_str(hw, hw->aq.asq_last_status));
1353 *perrno = i40e_aq_rc_to_posix(status, hw->aq.asq_last_status);
Pawel Jablonskie3a5d6e2017-12-18 05:14:44 -05001354 return status;
Shannon Nelsone4c83c22015-08-28 17:55:50 -04001355 }
1356
Shannon Nelsonfed2db92016-04-12 08:30:43 -07001357 /* should we wait for a followup event? */
1358 if (cmd->offset) {
1359 hw->nvm_wait_opcode = cmd->offset;
1360 hw->nvmupd_state = I40E_NVMUPD_STATE_INIT_WAIT;
1361 }
1362
Shannon Nelsone4c83c22015-08-28 17:55:50 -04001363 return status;
1364}
1365
1366/**
Shannon Nelsonb72dc7b2015-08-28 17:55:51 -04001367 * i40e_nvmupd_get_aq_result - Get the results from the previous exec_aq
1368 * @hw: pointer to hardware structure
1369 * @cmd: pointer to nvm update command buffer
1370 * @bytes: pointer to the data buffer
1371 * @perrno: pointer to return error code
1372 *
1373 * cmd structure contains identifiers and data buffer
1374 **/
1375static i40e_status i40e_nvmupd_get_aq_result(struct i40e_hw *hw,
1376 struct i40e_nvm_access *cmd,
1377 u8 *bytes, int *perrno)
1378{
1379 u32 aq_total_len;
1380 u32 aq_desc_len;
1381 int remainder;
1382 u8 *buff;
1383
1384 i40e_debug(hw, I40E_DEBUG_NVM, "NVMUPD: %s\n", __func__);
1385
1386 aq_desc_len = sizeof(struct i40e_aq_desc);
1387 aq_total_len = aq_desc_len + le16_to_cpu(hw->nvm_wb_desc.datalen);
1388
1389 /* check offset range */
1390 if (cmd->offset > aq_total_len) {
1391 i40e_debug(hw, I40E_DEBUG_NVM, "%s: offset too big %d > %d\n",
1392 __func__, cmd->offset, aq_total_len);
1393 *perrno = -EINVAL;
1394 return I40E_ERR_PARAM;
1395 }
1396
1397 /* check copylength range */
1398 if (cmd->data_size > (aq_total_len - cmd->offset)) {
1399 int new_len = aq_total_len - cmd->offset;
1400
1401 i40e_debug(hw, I40E_DEBUG_NVM, "%s: copy length %d too big, trimming to %d\n",
1402 __func__, cmd->data_size, new_len);
1403 cmd->data_size = new_len;
1404 }
1405
1406 remainder = cmd->data_size;
1407 if (cmd->offset < aq_desc_len) {
1408 u32 len = aq_desc_len - cmd->offset;
1409
1410 len = min(len, cmd->data_size);
1411 i40e_debug(hw, I40E_DEBUG_NVM, "%s: aq_desc bytes %d to %d\n",
1412 __func__, cmd->offset, cmd->offset + len);
1413
1414 buff = ((u8 *)&hw->nvm_wb_desc) + cmd->offset;
1415 memcpy(bytes, buff, len);
1416
1417 bytes += len;
1418 remainder -= len;
1419 buff = hw->nvm_buff.va;
1420 } else {
1421 buff = hw->nvm_buff.va + (cmd->offset - aq_desc_len);
1422 }
1423
1424 if (remainder > 0) {
1425 int start_byte = buff - (u8 *)hw->nvm_buff.va;
1426
1427 i40e_debug(hw, I40E_DEBUG_NVM, "%s: databuf bytes %d to %d\n",
1428 __func__, start_byte, start_byte + remainder);
1429 memcpy(bytes, buff, remainder);
1430 }
1431
1432 return 0;
1433}
1434
1435/**
Pawel Jablonskie3a5d6e2017-12-18 05:14:44 -05001436 * i40e_nvmupd_get_aq_event - Get the Admin Queue event from previous exec_aq
1437 * @hw: pointer to hardware structure
1438 * @cmd: pointer to nvm update command buffer
1439 * @bytes: pointer to the data buffer
1440 * @perrno: pointer to return error code
1441 *
1442 * cmd structure contains identifiers and data buffer
1443 **/
1444static i40e_status i40e_nvmupd_get_aq_event(struct i40e_hw *hw,
1445 struct i40e_nvm_access *cmd,
1446 u8 *bytes, int *perrno)
1447{
1448 u32 aq_total_len;
1449 u32 aq_desc_len;
1450
1451 i40e_debug(hw, I40E_DEBUG_NVM, "NVMUPD: %s\n", __func__);
1452
1453 aq_desc_len = sizeof(struct i40e_aq_desc);
1454 aq_total_len = aq_desc_len + le16_to_cpu(hw->nvm_aq_event_desc.datalen);
1455
1456 /* check copylength range */
1457 if (cmd->data_size > aq_total_len) {
1458 i40e_debug(hw, I40E_DEBUG_NVM,
1459 "%s: copy length %d too big, trimming to %d\n",
1460 __func__, cmd->data_size, aq_total_len);
1461 cmd->data_size = aq_total_len;
1462 }
1463
1464 memcpy(bytes, &hw->nvm_aq_event_desc, cmd->data_size);
1465
1466 return 0;
1467}
1468
1469/**
Shannon Nelsoncd552cb2014-07-09 07:46:09 +00001470 * i40e_nvmupd_nvm_read - Read NVM
1471 * @hw: pointer to hardware structure
1472 * @cmd: pointer to nvm update command buffer
1473 * @bytes: pointer to the data buffer
Shannon Nelson79afe832015-07-23 16:54:33 -04001474 * @perrno: pointer to return error code
Shannon Nelsoncd552cb2014-07-09 07:46:09 +00001475 *
1476 * cmd structure contains identifiers and data buffer
1477 **/
1478static i40e_status i40e_nvmupd_nvm_read(struct i40e_hw *hw,
1479 struct i40e_nvm_access *cmd,
Shannon Nelson79afe832015-07-23 16:54:33 -04001480 u8 *bytes, int *perrno)
Shannon Nelsoncd552cb2014-07-09 07:46:09 +00001481{
Shannon Nelson6b5c1b82015-08-28 17:55:47 -04001482 struct i40e_asq_cmd_details cmd_details;
Shannon Nelsoncd552cb2014-07-09 07:46:09 +00001483 i40e_status status;
1484 u8 module, transaction;
1485 bool last;
1486
1487 transaction = i40e_nvmupd_get_transaction(cmd->config);
1488 module = i40e_nvmupd_get_module(cmd->config);
1489 last = (transaction == I40E_NVM_LCB) || (transaction == I40E_NVM_SA);
Shannon Nelsoncd552cb2014-07-09 07:46:09 +00001490
Shannon Nelson6b5c1b82015-08-28 17:55:47 -04001491 memset(&cmd_details, 0, sizeof(cmd_details));
1492 cmd_details.wb_desc = &hw->nvm_wb_desc;
1493
Shannon Nelsoncd552cb2014-07-09 07:46:09 +00001494 status = i40e_aq_read_nvm(hw, module, cmd->offset, (u16)cmd->data_size,
Shannon Nelson6b5c1b82015-08-28 17:55:47 -04001495 bytes, last, &cmd_details);
Shannon Nelson74d0d0e2014-11-13 08:23:15 +00001496 if (status) {
1497 i40e_debug(hw, I40E_DEBUG_NVM,
1498 "i40e_nvmupd_nvm_read mod 0x%x off 0x%x len 0x%x\n",
1499 module, cmd->offset, cmd->data_size);
1500 i40e_debug(hw, I40E_DEBUG_NVM,
1501 "i40e_nvmupd_nvm_read status %d aq %d\n",
1502 status, hw->aq.asq_last_status);
Shannon Nelson79afe832015-07-23 16:54:33 -04001503 *perrno = i40e_aq_rc_to_posix(status, hw->aq.asq_last_status);
Shannon Nelson74d0d0e2014-11-13 08:23:15 +00001504 }
Shannon Nelsoncd552cb2014-07-09 07:46:09 +00001505
1506 return status;
1507}
1508
1509/**
1510 * i40e_nvmupd_nvm_erase - Erase an NVM module
1511 * @hw: pointer to hardware structure
1512 * @cmd: pointer to nvm update command buffer
Shannon Nelson79afe832015-07-23 16:54:33 -04001513 * @perrno: pointer to return error code
Shannon Nelsoncd552cb2014-07-09 07:46:09 +00001514 *
1515 * module, offset, data_size and data are in cmd structure
1516 **/
1517static i40e_status i40e_nvmupd_nvm_erase(struct i40e_hw *hw,
1518 struct i40e_nvm_access *cmd,
Shannon Nelson79afe832015-07-23 16:54:33 -04001519 int *perrno)
Shannon Nelsoncd552cb2014-07-09 07:46:09 +00001520{
1521 i40e_status status = 0;
Shannon Nelson6b5c1b82015-08-28 17:55:47 -04001522 struct i40e_asq_cmd_details cmd_details;
Shannon Nelsoncd552cb2014-07-09 07:46:09 +00001523 u8 module, transaction;
1524 bool last;
1525
1526 transaction = i40e_nvmupd_get_transaction(cmd->config);
1527 module = i40e_nvmupd_get_module(cmd->config);
1528 last = (transaction & I40E_NVM_LCB);
Shannon Nelson6b5c1b82015-08-28 17:55:47 -04001529
1530 memset(&cmd_details, 0, sizeof(cmd_details));
1531 cmd_details.wb_desc = &hw->nvm_wb_desc;
1532
Shannon Nelsoncd552cb2014-07-09 07:46:09 +00001533 status = i40e_aq_erase_nvm(hw, module, cmd->offset, (u16)cmd->data_size,
Shannon Nelson6b5c1b82015-08-28 17:55:47 -04001534 last, &cmd_details);
Shannon Nelson74d0d0e2014-11-13 08:23:15 +00001535 if (status) {
1536 i40e_debug(hw, I40E_DEBUG_NVM,
1537 "i40e_nvmupd_nvm_erase mod 0x%x off 0x%x len 0x%x\n",
1538 module, cmd->offset, cmd->data_size);
1539 i40e_debug(hw, I40E_DEBUG_NVM,
1540 "i40e_nvmupd_nvm_erase status %d aq %d\n",
1541 status, hw->aq.asq_last_status);
Shannon Nelson79afe832015-07-23 16:54:33 -04001542 *perrno = i40e_aq_rc_to_posix(status, hw->aq.asq_last_status);
Shannon Nelson74d0d0e2014-11-13 08:23:15 +00001543 }
Shannon Nelsoncd552cb2014-07-09 07:46:09 +00001544
1545 return status;
1546}
1547
1548/**
1549 * i40e_nvmupd_nvm_write - Write NVM
1550 * @hw: pointer to hardware structure
1551 * @cmd: pointer to nvm update command buffer
1552 * @bytes: pointer to the data buffer
Shannon Nelson79afe832015-07-23 16:54:33 -04001553 * @perrno: pointer to return error code
Shannon Nelsoncd552cb2014-07-09 07:46:09 +00001554 *
1555 * module, offset, data_size and data are in cmd structure
1556 **/
1557static i40e_status i40e_nvmupd_nvm_write(struct i40e_hw *hw,
1558 struct i40e_nvm_access *cmd,
Shannon Nelson79afe832015-07-23 16:54:33 -04001559 u8 *bytes, int *perrno)
Shannon Nelsoncd552cb2014-07-09 07:46:09 +00001560{
1561 i40e_status status = 0;
Shannon Nelson6b5c1b82015-08-28 17:55:47 -04001562 struct i40e_asq_cmd_details cmd_details;
Shannon Nelsoncd552cb2014-07-09 07:46:09 +00001563 u8 module, transaction;
Pawel Jablonskie3a5d6e2017-12-18 05:14:44 -05001564 u8 preservation_flags;
Shannon Nelsoncd552cb2014-07-09 07:46:09 +00001565 bool last;
1566
1567 transaction = i40e_nvmupd_get_transaction(cmd->config);
1568 module = i40e_nvmupd_get_module(cmd->config);
1569 last = (transaction & I40E_NVM_LCB);
Pawel Jablonskie3a5d6e2017-12-18 05:14:44 -05001570 preservation_flags = i40e_nvmupd_get_preservation_flags(cmd->config);
Shannon Nelson74d0d0e2014-11-13 08:23:15 +00001571
Shannon Nelson6b5c1b82015-08-28 17:55:47 -04001572 memset(&cmd_details, 0, sizeof(cmd_details));
1573 cmd_details.wb_desc = &hw->nvm_wb_desc;
1574
Shannon Nelsoncd552cb2014-07-09 07:46:09 +00001575 status = i40e_aq_update_nvm(hw, module, cmd->offset,
Shannon Nelson6b5c1b82015-08-28 17:55:47 -04001576 (u16)cmd->data_size, bytes, last,
Pawel Jablonskie3a5d6e2017-12-18 05:14:44 -05001577 preservation_flags, &cmd_details);
Shannon Nelson74d0d0e2014-11-13 08:23:15 +00001578 if (status) {
1579 i40e_debug(hw, I40E_DEBUG_NVM,
1580 "i40e_nvmupd_nvm_write mod 0x%x off 0x%x len 0x%x\n",
1581 module, cmd->offset, cmd->data_size);
1582 i40e_debug(hw, I40E_DEBUG_NVM,
1583 "i40e_nvmupd_nvm_write status %d aq %d\n",
1584 status, hw->aq.asq_last_status);
Shannon Nelson79afe832015-07-23 16:54:33 -04001585 *perrno = i40e_aq_rc_to_posix(status, hw->aq.asq_last_status);
Shannon Nelson74d0d0e2014-11-13 08:23:15 +00001586 }
Shannon Nelsoncd552cb2014-07-09 07:46:09 +00001587
1588 return status;
1589}