blob: dd62701256142b04f5d3e4989017fc9848cd95ec [file] [log] [blame]
Sreekanth Reddyf92363d2012-11-30 07:44:21 +05301/*
2 * This module provides common API for accessing firmware configuration pages
3 *
4 * This code is based on drivers/scsi/mpt3sas/mpt3sas_base.c
Sreekanth Reddya4ffce02014-09-12 15:35:29 +05305 * Copyright (C) 2012-2014 LSI Corporation
Sreekanth Reddya03bd152015-01-12 11:39:02 +05306 * Copyright (C) 2013-2014 Avago Technologies
7 * (mailto: MPT-FusionLinux.pdl@avagotech.com)
Sreekanth Reddyf92363d2012-11-30 07:44:21 +05308 *
9 * This program is free software; you can redistribute it and/or
10 * modify it under the terms of the GNU General Public License
11 * as published by the Free Software Foundation; either version 2
12 * of the License, or (at your option) any later version.
13 *
14 * This program is distributed in the hope that it will be useful,
15 * but WITHOUT ANY WARRANTY; without even the implied warranty of
16 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
17 * GNU General Public License for more details.
18 *
19 * NO WARRANTY
20 * THE PROGRAM IS PROVIDED ON AN "AS IS" BASIS, WITHOUT WARRANTIES OR
21 * CONDITIONS OF ANY KIND, EITHER EXPRESS OR IMPLIED INCLUDING, WITHOUT
22 * LIMITATION, ANY WARRANTIES OR CONDITIONS OF TITLE, NON-INFRINGEMENT,
23 * MERCHANTABILITY OR FITNESS FOR A PARTICULAR PURPOSE. Each Recipient is
24 * solely responsible for determining the appropriateness of using and
25 * distributing the Program and assumes all risks associated with its
26 * exercise of rights under this Agreement, including but not limited to
27 * the risks and costs of program errors, damage to or loss of data,
28 * programs or equipment, and unavailability or interruption of operations.
29
30 * DISCLAIMER OF LIABILITY
31 * NEITHER RECIPIENT NOR ANY CONTRIBUTORS SHALL HAVE ANY LIABILITY FOR ANY
32 * DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
33 * DAMAGES (INCLUDING WITHOUT LIMITATION LOST PROFITS), HOWEVER CAUSED AND
34 * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR
35 * TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE
36 * USE OR DISTRIBUTION OF THE PROGRAM OR THE EXERCISE OF ANY RIGHTS GRANTED
37 * HEREUNDER, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGES
38
39 * You should have received a copy of the GNU General Public License
40 * along with this program; if not, write to the Free Software
41 * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301,
42 * USA.
43 */
44
Sreekanth Reddyf92363d2012-11-30 07:44:21 +053045#include <linux/module.h>
46#include <linux/kernel.h>
47#include <linux/init.h>
48#include <linux/errno.h>
49#include <linux/blkdev.h>
50#include <linux/sched.h>
51#include <linux/workqueue.h>
52#include <linux/delay.h>
53#include <linux/pci.h>
54
55#include "mpt3sas_base.h"
56
57/* local definitions */
58
59/* Timeout for config page request (in seconds) */
60#define MPT3_CONFIG_PAGE_DEFAULT_TIMEOUT 15
61
62/* Common sgl flags for READING a config page. */
63#define MPT3_CONFIG_COMMON_SGLFLAGS ((MPI2_SGE_FLAGS_SIMPLE_ELEMENT | \
64 MPI2_SGE_FLAGS_LAST_ELEMENT | MPI2_SGE_FLAGS_END_OF_BUFFER \
65 | MPI2_SGE_FLAGS_END_OF_LIST) << MPI2_SGE_FLAGS_SHIFT)
66
67/* Common sgl flags for WRITING a config page. */
68#define MPT3_CONFIG_COMMON_WRITE_SGLFLAGS ((MPI2_SGE_FLAGS_SIMPLE_ELEMENT | \
69 MPI2_SGE_FLAGS_LAST_ELEMENT | MPI2_SGE_FLAGS_END_OF_BUFFER \
70 | MPI2_SGE_FLAGS_END_OF_LIST | MPI2_SGE_FLAGS_HOST_TO_IOC) \
71 << MPI2_SGE_FLAGS_SHIFT)
72
73/**
74 * struct config_request - obtain dma memory via routine
75 * @sz: size
76 * @page: virt pointer
77 * @page_dma: phys pointer
78 *
79 */
80struct config_request {
81 u16 sz;
82 void *page;
83 dma_addr_t page_dma;
84};
85
Sreekanth Reddyf92363d2012-11-30 07:44:21 +053086/**
87 * _config_display_some_debug - debug routine
88 * @ioc: per adapter object
89 * @smid: system request message index
90 * @calling_function_name: string pass from calling function
91 * @mpi_reply: reply message frame
92 * Context: none.
93 *
94 * Function for displaying debug info helpful when debugging issues
95 * in this module.
96 */
97static void
98_config_display_some_debug(struct MPT3SAS_ADAPTER *ioc, u16 smid,
99 char *calling_function_name, MPI2DefaultReply_t *mpi_reply)
100{
101 Mpi2ConfigRequest_t *mpi_request;
102 char *desc = NULL;
103
104 if (!(ioc->logging_level & MPT_DEBUG_CONFIG))
105 return;
106
107 mpi_request = mpt3sas_base_get_msg_frame(ioc, smid);
108 switch (mpi_request->Header.PageType & MPI2_CONFIG_PAGETYPE_MASK) {
109 case MPI2_CONFIG_PAGETYPE_IO_UNIT:
110 desc = "io_unit";
111 break;
112 case MPI2_CONFIG_PAGETYPE_IOC:
113 desc = "ioc";
114 break;
115 case MPI2_CONFIG_PAGETYPE_BIOS:
116 desc = "bios";
117 break;
118 case MPI2_CONFIG_PAGETYPE_RAID_VOLUME:
119 desc = "raid_volume";
120 break;
121 case MPI2_CONFIG_PAGETYPE_MANUFACTURING:
122 desc = "manufaucturing";
123 break;
124 case MPI2_CONFIG_PAGETYPE_RAID_PHYSDISK:
125 desc = "physdisk";
126 break;
127 case MPI2_CONFIG_PAGETYPE_EXTENDED:
128 switch (mpi_request->ExtPageType) {
129 case MPI2_CONFIG_EXTPAGETYPE_SAS_IO_UNIT:
130 desc = "sas_io_unit";
131 break;
132 case MPI2_CONFIG_EXTPAGETYPE_SAS_EXPANDER:
133 desc = "sas_expander";
134 break;
135 case MPI2_CONFIG_EXTPAGETYPE_SAS_DEVICE:
136 desc = "sas_device";
137 break;
138 case MPI2_CONFIG_EXTPAGETYPE_SAS_PHY:
139 desc = "sas_phy";
140 break;
141 case MPI2_CONFIG_EXTPAGETYPE_LOG:
142 desc = "log";
143 break;
144 case MPI2_CONFIG_EXTPAGETYPE_ENCLOSURE:
145 desc = "enclosure";
146 break;
147 case MPI2_CONFIG_EXTPAGETYPE_RAID_CONFIG:
148 desc = "raid_config";
149 break;
150 case MPI2_CONFIG_EXTPAGETYPE_DRIVER_MAPPING:
Masanari Iida07f42252013-03-20 11:00:34 +0900151 desc = "driver_mapping";
Sreekanth Reddyf92363d2012-11-30 07:44:21 +0530152 break;
153 }
154 break;
155 }
156
157 if (!desc)
158 return;
159
160 pr_info(MPT3SAS_FMT
161 "%s: %s(%d), action(%d), form(0x%08x), smid(%d)\n",
162 ioc->name, calling_function_name, desc,
163 mpi_request->Header.PageNumber, mpi_request->Action,
164 le32_to_cpu(mpi_request->PageAddress), smid);
165
166 if (!mpi_reply)
167 return;
168
169 if (mpi_reply->IOCStatus || mpi_reply->IOCLogInfo)
170 pr_info(MPT3SAS_FMT
171 "\tiocstatus(0x%04x), loginfo(0x%08x)\n",
172 ioc->name, le16_to_cpu(mpi_reply->IOCStatus),
173 le32_to_cpu(mpi_reply->IOCLogInfo));
174}
Sreekanth Reddyf92363d2012-11-30 07:44:21 +0530175
176/**
177 * _config_alloc_config_dma_memory - obtain physical memory
178 * @ioc: per adapter object
179 * @mem: struct config_request
180 *
181 * A wrapper for obtaining dma-able memory for config page request.
182 *
183 * Returns 0 for success, non-zero for failure.
184 */
185static int
186_config_alloc_config_dma_memory(struct MPT3SAS_ADAPTER *ioc,
187 struct config_request *mem)
188{
189 int r = 0;
190
191 if (mem->sz > ioc->config_page_sz) {
192 mem->page = dma_alloc_coherent(&ioc->pdev->dev, mem->sz,
193 &mem->page_dma, GFP_KERNEL);
194 if (!mem->page) {
195 pr_err(MPT3SAS_FMT
196 "%s: dma_alloc_coherent failed asking for (%d) bytes!!\n",
197 ioc->name, __func__, mem->sz);
198 r = -ENOMEM;
199 }
200 } else { /* use tmp buffer if less than 512 bytes */
201 mem->page = ioc->config_page;
202 mem->page_dma = ioc->config_page_dma;
203 }
204 return r;
205}
206
207/**
208 * _config_free_config_dma_memory - wrapper to free the memory
209 * @ioc: per adapter object
210 * @mem: struct config_request
211 *
212 * A wrapper to free dma-able memory when using _config_alloc_config_dma_memory.
213 *
214 * Returns 0 for success, non-zero for failure.
215 */
216static void
217_config_free_config_dma_memory(struct MPT3SAS_ADAPTER *ioc,
218 struct config_request *mem)
219{
220 if (mem->sz > ioc->config_page_sz)
221 dma_free_coherent(&ioc->pdev->dev, mem->sz, mem->page,
222 mem->page_dma);
223}
224
225/**
226 * mpt3sas_config_done - config page completion routine
227 * @ioc: per adapter object
228 * @smid: system request message index
229 * @msix_index: MSIX table index supplied by the OS
230 * @reply: reply message frame(lower 32bit addr)
231 * Context: none.
232 *
233 * The callback handler when using _config_request.
234 *
235 * Return 1 meaning mf should be freed from _base_interrupt
236 * 0 means the mf is freed from this function.
237 */
238u8
239mpt3sas_config_done(struct MPT3SAS_ADAPTER *ioc, u16 smid, u8 msix_index,
240 u32 reply)
241{
242 MPI2DefaultReply_t *mpi_reply;
243
244 if (ioc->config_cmds.status == MPT3_CMD_NOT_USED)
245 return 1;
246 if (ioc->config_cmds.smid != smid)
247 return 1;
248 ioc->config_cmds.status |= MPT3_CMD_COMPLETE;
249 mpi_reply = mpt3sas_base_get_reply_virt_addr(ioc, reply);
250 if (mpi_reply) {
251 ioc->config_cmds.status |= MPT3_CMD_REPLY_VALID;
252 memcpy(ioc->config_cmds.reply, mpi_reply,
253 mpi_reply->MsgLength*4);
254 }
255 ioc->config_cmds.status &= ~MPT3_CMD_PENDING;
Sreekanth Reddyf92363d2012-11-30 07:44:21 +0530256 _config_display_some_debug(ioc, smid, "config_done", mpi_reply);
Sreekanth Reddyf92363d2012-11-30 07:44:21 +0530257 ioc->config_cmds.smid = USHRT_MAX;
258 complete(&ioc->config_cmds.done);
259 return 1;
260}
261
262/**
263 * _config_request - main routine for sending config page requests
264 * @ioc: per adapter object
265 * @mpi_request: request message frame
266 * @mpi_reply: reply mf payload returned from firmware
267 * @timeout: timeout in seconds
268 * @config_page: contents of the config page
269 * @config_page_sz: size of config page
270 * Context: sleep
271 *
272 * A generic API for config page requests to firmware.
273 *
274 * The ioc->config_cmds.status flag should be MPT3_CMD_NOT_USED before calling
275 * this API.
276 *
277 * The callback index is set inside `ioc->config_cb_idx.
278 *
279 * Returns 0 for success, non-zero for failure.
280 */
281static int
282_config_request(struct MPT3SAS_ADAPTER *ioc, Mpi2ConfigRequest_t
283 *mpi_request, Mpi2ConfigReply_t *mpi_reply, int timeout,
284 void *config_page, u16 config_page_sz)
285{
286 u16 smid;
287 u32 ioc_state;
Sreekanth Reddyf92363d2012-11-30 07:44:21 +0530288 Mpi2ConfigRequest_t *config_request;
289 int r;
290 u8 retry_count, issue_host_reset = 0;
291 u16 wait_state_count;
292 struct config_request mem;
293 u32 ioc_status = UINT_MAX;
294
295 mutex_lock(&ioc->config_cmds.mutex);
296 if (ioc->config_cmds.status != MPT3_CMD_NOT_USED) {
297 pr_err(MPT3SAS_FMT "%s: config_cmd in use\n",
298 ioc->name, __func__);
299 mutex_unlock(&ioc->config_cmds.mutex);
300 return -EAGAIN;
301 }
302
303 retry_count = 0;
304 memset(&mem, 0, sizeof(struct config_request));
305
306 mpi_request->VF_ID = 0; /* TODO */
307 mpi_request->VP_ID = 0;
308
309 if (config_page) {
310 mpi_request->Header.PageVersion = mpi_reply->Header.PageVersion;
311 mpi_request->Header.PageNumber = mpi_reply->Header.PageNumber;
312 mpi_request->Header.PageType = mpi_reply->Header.PageType;
313 mpi_request->Header.PageLength = mpi_reply->Header.PageLength;
314 mpi_request->ExtPageLength = mpi_reply->ExtPageLength;
315 mpi_request->ExtPageType = mpi_reply->ExtPageType;
316 if (mpi_request->Header.PageLength)
317 mem.sz = mpi_request->Header.PageLength * 4;
318 else
319 mem.sz = le16_to_cpu(mpi_reply->ExtPageLength) * 4;
320 r = _config_alloc_config_dma_memory(ioc, &mem);
321 if (r != 0)
322 goto out;
323 if (mpi_request->Action ==
324 MPI2_CONFIG_ACTION_PAGE_WRITE_CURRENT ||
325 mpi_request->Action ==
326 MPI2_CONFIG_ACTION_PAGE_WRITE_NVRAM) {
327 ioc->base_add_sg_single(&mpi_request->PageBufferSGE,
328 MPT3_CONFIG_COMMON_WRITE_SGLFLAGS | mem.sz,
329 mem.page_dma);
330 memcpy(mem.page, config_page, min_t(u16, mem.sz,
331 config_page_sz));
332 } else {
333 memset(config_page, 0, config_page_sz);
334 ioc->base_add_sg_single(&mpi_request->PageBufferSGE,
335 MPT3_CONFIG_COMMON_SGLFLAGS | mem.sz, mem.page_dma);
336 memset(mem.page, 0, min_t(u16, mem.sz, config_page_sz));
337 }
338 }
339
340 retry_config:
341 if (retry_count) {
342 if (retry_count > 2) { /* attempt only 2 retries */
343 r = -EFAULT;
344 goto free_mem;
345 }
346 pr_info(MPT3SAS_FMT "%s: attempting retry (%d)\n",
347 ioc->name, __func__, retry_count);
348 }
349 wait_state_count = 0;
350 ioc_state = mpt3sas_base_get_iocstate(ioc, 1);
351 while (ioc_state != MPI2_IOC_STATE_OPERATIONAL) {
352 if (wait_state_count++ == MPT3_CONFIG_PAGE_DEFAULT_TIMEOUT) {
353 pr_err(MPT3SAS_FMT
354 "%s: failed due to ioc not operational\n",
355 ioc->name, __func__);
356 ioc->config_cmds.status = MPT3_CMD_NOT_USED;
357 r = -EFAULT;
358 goto free_mem;
359 }
360 ssleep(1);
361 ioc_state = mpt3sas_base_get_iocstate(ioc, 1);
362 pr_info(MPT3SAS_FMT
363 "%s: waiting for operational state(count=%d)\n",
364 ioc->name, __func__, wait_state_count);
365 }
366 if (wait_state_count)
367 pr_info(MPT3SAS_FMT "%s: ioc is operational\n",
368 ioc->name, __func__);
369
370 smid = mpt3sas_base_get_smid(ioc, ioc->config_cb_idx);
371 if (!smid) {
372 pr_err(MPT3SAS_FMT "%s: failed obtaining a smid\n",
373 ioc->name, __func__);
374 ioc->config_cmds.status = MPT3_CMD_NOT_USED;
375 r = -EAGAIN;
376 goto free_mem;
377 }
378
379 r = 0;
380 memset(mpi_reply, 0, sizeof(Mpi2ConfigReply_t));
381 ioc->config_cmds.status = MPT3_CMD_PENDING;
382 config_request = mpt3sas_base_get_msg_frame(ioc, smid);
383 ioc->config_cmds.smid = smid;
384 memcpy(config_request, mpi_request, sizeof(Mpi2ConfigRequest_t));
Sreekanth Reddyf92363d2012-11-30 07:44:21 +0530385 _config_display_some_debug(ioc, smid, "config_request", NULL);
Sreekanth Reddyf92363d2012-11-30 07:44:21 +0530386 init_completion(&ioc->config_cmds.done);
Suganath Prabu Subramani81c16f82016-10-26 13:34:40 +0530387 ioc->put_smid_default(ioc, smid);
Calvin Owens8bbb1cf2016-07-28 21:38:22 -0700388 wait_for_completion_timeout(&ioc->config_cmds.done, timeout*HZ);
Sreekanth Reddyf92363d2012-11-30 07:44:21 +0530389 if (!(ioc->config_cmds.status & MPT3_CMD_COMPLETE)) {
390 pr_err(MPT3SAS_FMT "%s: timeout\n",
391 ioc->name, __func__);
392 _debug_dump_mf(mpi_request,
393 sizeof(Mpi2ConfigRequest_t)/4);
394 retry_count++;
395 if (ioc->config_cmds.smid == smid)
396 mpt3sas_base_free_smid(ioc, smid);
397 if ((ioc->shost_recovery) || (ioc->config_cmds.status &
398 MPT3_CMD_RESET) || ioc->pci_error_recovery)
399 goto retry_config;
400 issue_host_reset = 1;
401 r = -EFAULT;
402 goto free_mem;
403 }
404
405 if (ioc->config_cmds.status & MPT3_CMD_REPLY_VALID) {
406 memcpy(mpi_reply, ioc->config_cmds.reply,
407 sizeof(Mpi2ConfigReply_t));
408
409 /* Reply Frame Sanity Checks to workaround FW issues */
410 if ((mpi_request->Header.PageType & 0xF) !=
411 (mpi_reply->Header.PageType & 0xF)) {
412 _debug_dump_mf(mpi_request, ioc->request_sz/4);
413 _debug_dump_reply(mpi_reply, ioc->request_sz/4);
414 panic(KERN_WARNING MPT3SAS_FMT "%s: Firmware BUG:" \
415 " mpi_reply mismatch: Requested PageType(0x%02x)" \
416 " Reply PageType(0x%02x)\n", \
417 ioc->name, __func__,
418 (mpi_request->Header.PageType & 0xF),
419 (mpi_reply->Header.PageType & 0xF));
420 }
421
422 if (((mpi_request->Header.PageType & 0xF) ==
423 MPI2_CONFIG_PAGETYPE_EXTENDED) &&
424 mpi_request->ExtPageType != mpi_reply->ExtPageType) {
425 _debug_dump_mf(mpi_request, ioc->request_sz/4);
426 _debug_dump_reply(mpi_reply, ioc->request_sz/4);
427 panic(KERN_WARNING MPT3SAS_FMT "%s: Firmware BUG:" \
428 " mpi_reply mismatch: Requested ExtPageType(0x%02x)"
429 " Reply ExtPageType(0x%02x)\n",
430 ioc->name, __func__, mpi_request->ExtPageType,
431 mpi_reply->ExtPageType);
432 }
433 ioc_status = le16_to_cpu(mpi_reply->IOCStatus)
434 & MPI2_IOCSTATUS_MASK;
435 }
436
437 if (retry_count)
438 pr_info(MPT3SAS_FMT "%s: retry (%d) completed!!\n", \
439 ioc->name, __func__, retry_count);
440
441 if ((ioc_status == MPI2_IOCSTATUS_SUCCESS) &&
442 config_page && mpi_request->Action ==
443 MPI2_CONFIG_ACTION_PAGE_READ_CURRENT) {
444 u8 *p = (u8 *)mem.page;
445
446 /* Config Page Sanity Checks to workaround FW issues */
447 if (p) {
448 if ((mpi_request->Header.PageType & 0xF) !=
449 (p[3] & 0xF)) {
450 _debug_dump_mf(mpi_request, ioc->request_sz/4);
451 _debug_dump_reply(mpi_reply, ioc->request_sz/4);
452 _debug_dump_config(p, min_t(u16, mem.sz,
453 config_page_sz)/4);
454 panic(KERN_WARNING MPT3SAS_FMT
455 "%s: Firmware BUG:" \
456 " config page mismatch:"
457 " Requested PageType(0x%02x)"
458 " Reply PageType(0x%02x)\n",
459 ioc->name, __func__,
460 (mpi_request->Header.PageType & 0xF),
461 (p[3] & 0xF));
462 }
463
464 if (((mpi_request->Header.PageType & 0xF) ==
465 MPI2_CONFIG_PAGETYPE_EXTENDED) &&
466 (mpi_request->ExtPageType != p[6])) {
467 _debug_dump_mf(mpi_request, ioc->request_sz/4);
468 _debug_dump_reply(mpi_reply, ioc->request_sz/4);
469 _debug_dump_config(p, min_t(u16, mem.sz,
470 config_page_sz)/4);
471 panic(KERN_WARNING MPT3SAS_FMT
472 "%s: Firmware BUG:" \
473 " config page mismatch:"
474 " Requested ExtPageType(0x%02x)"
475 " Reply ExtPageType(0x%02x)\n",
476 ioc->name, __func__,
477 mpi_request->ExtPageType, p[6]);
478 }
479 }
480 memcpy(config_page, mem.page, min_t(u16, mem.sz,
481 config_page_sz));
482 }
483
484 free_mem:
485 if (config_page)
486 _config_free_config_dma_memory(ioc, &mem);
487 out:
488 ioc->config_cmds.status = MPT3_CMD_NOT_USED;
489 mutex_unlock(&ioc->config_cmds.mutex);
490
491 if (issue_host_reset)
Calvin Owens98c56ad2016-07-28 21:38:21 -0700492 mpt3sas_base_hard_reset_handler(ioc, FORCE_BIG_HAMMER);
Sreekanth Reddyf92363d2012-11-30 07:44:21 +0530493 return r;
494}
495
496/**
497 * mpt3sas_config_get_manufacturing_pg0 - obtain manufacturing page 0
498 * @ioc: per adapter object
499 * @mpi_reply: reply mf payload returned from firmware
500 * @config_page: contents of the config page
501 * Context: sleep.
502 *
503 * Returns 0 for success, non-zero for failure.
504 */
505int
506mpt3sas_config_get_manufacturing_pg0(struct MPT3SAS_ADAPTER *ioc,
507 Mpi2ConfigReply_t *mpi_reply, Mpi2ManufacturingPage0_t *config_page)
508{
509 Mpi2ConfigRequest_t mpi_request;
510 int r;
511
512 memset(&mpi_request, 0, sizeof(Mpi2ConfigRequest_t));
513 mpi_request.Function = MPI2_FUNCTION_CONFIG;
514 mpi_request.Action = MPI2_CONFIG_ACTION_PAGE_HEADER;
515 mpi_request.Header.PageType = MPI2_CONFIG_PAGETYPE_MANUFACTURING;
516 mpi_request.Header.PageNumber = 0;
517 mpi_request.Header.PageVersion = MPI2_MANUFACTURING0_PAGEVERSION;
518 ioc->build_zero_len_sge_mpi(ioc, &mpi_request.PageBufferSGE);
519 r = _config_request(ioc, &mpi_request, mpi_reply,
520 MPT3_CONFIG_PAGE_DEFAULT_TIMEOUT, NULL, 0);
521 if (r)
522 goto out;
523
524 mpi_request.Action = MPI2_CONFIG_ACTION_PAGE_READ_CURRENT;
525 r = _config_request(ioc, &mpi_request, mpi_reply,
526 MPT3_CONFIG_PAGE_DEFAULT_TIMEOUT, config_page,
527 sizeof(*config_page));
528 out:
529 return r;
530}
531
532/**
533 * mpt3sas_config_get_manufacturing_pg7 - obtain manufacturing page 7
534 * @ioc: per adapter object
535 * @mpi_reply: reply mf payload returned from firmware
536 * @config_page: contents of the config page
537 * @sz: size of buffer passed in config_page
538 * Context: sleep.
539 *
540 * Returns 0 for success, non-zero for failure.
541 */
542int
543mpt3sas_config_get_manufacturing_pg7(struct MPT3SAS_ADAPTER *ioc,
544 Mpi2ConfigReply_t *mpi_reply, Mpi2ManufacturingPage7_t *config_page,
545 u16 sz)
546{
547 Mpi2ConfigRequest_t mpi_request;
548 int r;
549
550 memset(&mpi_request, 0, sizeof(Mpi2ConfigRequest_t));
551 mpi_request.Function = MPI2_FUNCTION_CONFIG;
552 mpi_request.Action = MPI2_CONFIG_ACTION_PAGE_HEADER;
553 mpi_request.Header.PageType = MPI2_CONFIG_PAGETYPE_MANUFACTURING;
554 mpi_request.Header.PageNumber = 7;
555 mpi_request.Header.PageVersion = MPI2_MANUFACTURING7_PAGEVERSION;
556 ioc->build_zero_len_sge_mpi(ioc, &mpi_request.PageBufferSGE);
557 r = _config_request(ioc, &mpi_request, mpi_reply,
558 MPT3_CONFIG_PAGE_DEFAULT_TIMEOUT, NULL, 0);
559 if (r)
560 goto out;
561
562 mpi_request.Action = MPI2_CONFIG_ACTION_PAGE_READ_CURRENT;
563 r = _config_request(ioc, &mpi_request, mpi_reply,
564 MPT3_CONFIG_PAGE_DEFAULT_TIMEOUT, config_page,
565 sz);
566 out:
567 return r;
568}
569
570/**
571 * mpt3sas_config_get_manufacturing_pg10 - obtain manufacturing page 10
572 * @ioc: per adapter object
573 * @mpi_reply: reply mf payload returned from firmware
574 * @config_page: contents of the config page
575 * Context: sleep.
576 *
577 * Returns 0 for success, non-zero for failure.
578 */
579int
580mpt3sas_config_get_manufacturing_pg10(struct MPT3SAS_ADAPTER *ioc,
581 Mpi2ConfigReply_t *mpi_reply,
582 struct Mpi2ManufacturingPage10_t *config_page)
583{
584 Mpi2ConfigRequest_t mpi_request;
585 int r;
586
587 memset(&mpi_request, 0, sizeof(Mpi2ConfigRequest_t));
588 mpi_request.Function = MPI2_FUNCTION_CONFIG;
589 mpi_request.Action = MPI2_CONFIG_ACTION_PAGE_HEADER;
590 mpi_request.Header.PageType = MPI2_CONFIG_PAGETYPE_MANUFACTURING;
591 mpi_request.Header.PageNumber = 10;
592 mpi_request.Header.PageVersion = MPI2_MANUFACTURING0_PAGEVERSION;
593 ioc->build_zero_len_sge_mpi(ioc, &mpi_request.PageBufferSGE);
594 r = _config_request(ioc, &mpi_request, mpi_reply,
595 MPT3_CONFIG_PAGE_DEFAULT_TIMEOUT, NULL, 0);
596 if (r)
597 goto out;
598
599 mpi_request.Action = MPI2_CONFIG_ACTION_PAGE_READ_CURRENT;
600 r = _config_request(ioc, &mpi_request, mpi_reply,
601 MPT3_CONFIG_PAGE_DEFAULT_TIMEOUT, config_page,
602 sizeof(*config_page));
603 out:
604 return r;
605}
606
607/**
608 * mpt3sas_config_get_manufacturing_pg11 - obtain manufacturing page 11
609 * @ioc: per adapter object
610 * @mpi_reply: reply mf payload returned from firmware
611 * @config_page: contents of the config page
612 * Context: sleep.
613 *
614 * Returns 0 for success, non-zero for failure.
615 */
616int
617mpt3sas_config_get_manufacturing_pg11(struct MPT3SAS_ADAPTER *ioc,
618 Mpi2ConfigReply_t *mpi_reply,
619 struct Mpi2ManufacturingPage11_t *config_page)
620{
621 Mpi2ConfigRequest_t mpi_request;
622 int r;
623
624 memset(&mpi_request, 0, sizeof(Mpi2ConfigRequest_t));
625 mpi_request.Function = MPI2_FUNCTION_CONFIG;
626 mpi_request.Action = MPI2_CONFIG_ACTION_PAGE_HEADER;
627 mpi_request.Header.PageType = MPI2_CONFIG_PAGETYPE_MANUFACTURING;
628 mpi_request.Header.PageNumber = 11;
629 mpi_request.Header.PageVersion = MPI2_MANUFACTURING0_PAGEVERSION;
630 ioc->build_zero_len_sge_mpi(ioc, &mpi_request.PageBufferSGE);
631 r = _config_request(ioc, &mpi_request, mpi_reply,
632 MPT3_CONFIG_PAGE_DEFAULT_TIMEOUT, NULL, 0);
633 if (r)
634 goto out;
635
636 mpi_request.Action = MPI2_CONFIG_ACTION_PAGE_READ_CURRENT;
637 r = _config_request(ioc, &mpi_request, mpi_reply,
638 MPT3_CONFIG_PAGE_DEFAULT_TIMEOUT, config_page,
639 sizeof(*config_page));
640 out:
641 return r;
642}
643
644/**
645 * mpt3sas_config_set_manufacturing_pg11 - set manufacturing page 11
646 * @ioc: per adapter object
647 * @mpi_reply: reply mf payload returned from firmware
648 * @config_page: contents of the config page
649 * Context: sleep.
650 *
651 * Returns 0 for success, non-zero for failure.
652 */
653int
654mpt3sas_config_set_manufacturing_pg11(struct MPT3SAS_ADAPTER *ioc,
655 Mpi2ConfigReply_t *mpi_reply,
656 struct Mpi2ManufacturingPage11_t *config_page)
657{
658 Mpi2ConfigRequest_t mpi_request;
659 int r;
660
661 memset(&mpi_request, 0, sizeof(Mpi2ConfigRequest_t));
662 mpi_request.Function = MPI2_FUNCTION_CONFIG;
663 mpi_request.Action = MPI2_CONFIG_ACTION_PAGE_HEADER;
664 mpi_request.Header.PageType = MPI2_CONFIG_PAGETYPE_MANUFACTURING;
665 mpi_request.Header.PageNumber = 11;
666 mpi_request.Header.PageVersion = MPI2_MANUFACTURING0_PAGEVERSION;
667 ioc->build_zero_len_sge_mpi(ioc, &mpi_request.PageBufferSGE);
668 r = _config_request(ioc, &mpi_request, mpi_reply,
669 MPT3_CONFIG_PAGE_DEFAULT_TIMEOUT, NULL, 0);
670 if (r)
671 goto out;
672
673 mpi_request.Action = MPI2_CONFIG_ACTION_PAGE_WRITE_CURRENT;
674 r = _config_request(ioc, &mpi_request, mpi_reply,
675 MPT3_CONFIG_PAGE_DEFAULT_TIMEOUT, config_page,
676 sizeof(*config_page));
677 mpi_request.Action = MPI2_CONFIG_ACTION_PAGE_WRITE_NVRAM;
678 r = _config_request(ioc, &mpi_request, mpi_reply,
679 MPT3_CONFIG_PAGE_DEFAULT_TIMEOUT, config_page,
680 sizeof(*config_page));
681 out:
682 return r;
683}
684
685/**
686 * mpt3sas_config_get_bios_pg2 - obtain bios page 2
687 * @ioc: per adapter object
688 * @mpi_reply: reply mf payload returned from firmware
689 * @config_page: contents of the config page
690 * Context: sleep.
691 *
692 * Returns 0 for success, non-zero for failure.
693 */
694int
695mpt3sas_config_get_bios_pg2(struct MPT3SAS_ADAPTER *ioc,
696 Mpi2ConfigReply_t *mpi_reply, Mpi2BiosPage2_t *config_page)
697{
698 Mpi2ConfigRequest_t mpi_request;
699 int r;
700
701 memset(&mpi_request, 0, sizeof(Mpi2ConfigRequest_t));
702 mpi_request.Function = MPI2_FUNCTION_CONFIG;
703 mpi_request.Action = MPI2_CONFIG_ACTION_PAGE_HEADER;
704 mpi_request.Header.PageType = MPI2_CONFIG_PAGETYPE_BIOS;
705 mpi_request.Header.PageNumber = 2;
706 mpi_request.Header.PageVersion = MPI2_BIOSPAGE2_PAGEVERSION;
707 ioc->build_zero_len_sge_mpi(ioc, &mpi_request.PageBufferSGE);
708 r = _config_request(ioc, &mpi_request, mpi_reply,
709 MPT3_CONFIG_PAGE_DEFAULT_TIMEOUT, NULL, 0);
710 if (r)
711 goto out;
712
713 mpi_request.Action = MPI2_CONFIG_ACTION_PAGE_READ_CURRENT;
714 r = _config_request(ioc, &mpi_request, mpi_reply,
715 MPT3_CONFIG_PAGE_DEFAULT_TIMEOUT, config_page,
716 sizeof(*config_page));
717 out:
718 return r;
719}
720
721/**
722 * mpt3sas_config_get_bios_pg3 - obtain bios page 3
723 * @ioc: per adapter object
724 * @mpi_reply: reply mf payload returned from firmware
725 * @config_page: contents of the config page
726 * Context: sleep.
727 *
728 * Returns 0 for success, non-zero for failure.
729 */
730int
731mpt3sas_config_get_bios_pg3(struct MPT3SAS_ADAPTER *ioc, Mpi2ConfigReply_t
732 *mpi_reply, Mpi2BiosPage3_t *config_page)
733{
734 Mpi2ConfigRequest_t mpi_request;
735 int r;
736
737 memset(&mpi_request, 0, sizeof(Mpi2ConfigRequest_t));
738 mpi_request.Function = MPI2_FUNCTION_CONFIG;
739 mpi_request.Action = MPI2_CONFIG_ACTION_PAGE_HEADER;
740 mpi_request.Header.PageType = MPI2_CONFIG_PAGETYPE_BIOS;
741 mpi_request.Header.PageNumber = 3;
742 mpi_request.Header.PageVersion = MPI2_BIOSPAGE3_PAGEVERSION;
743 ioc->build_zero_len_sge_mpi(ioc, &mpi_request.PageBufferSGE);
744 r = _config_request(ioc, &mpi_request, mpi_reply,
745 MPT3_CONFIG_PAGE_DEFAULT_TIMEOUT, NULL, 0);
746 if (r)
747 goto out;
748
749 mpi_request.Action = MPI2_CONFIG_ACTION_PAGE_READ_CURRENT;
750 r = _config_request(ioc, &mpi_request, mpi_reply,
751 MPT3_CONFIG_PAGE_DEFAULT_TIMEOUT, config_page,
752 sizeof(*config_page));
753 out:
754 return r;
755}
756
757/**
758 * mpt3sas_config_get_iounit_pg0 - obtain iounit page 0
759 * @ioc: per adapter object
760 * @mpi_reply: reply mf payload returned from firmware
761 * @config_page: contents of the config page
762 * Context: sleep.
763 *
764 * Returns 0 for success, non-zero for failure.
765 */
766int
767mpt3sas_config_get_iounit_pg0(struct MPT3SAS_ADAPTER *ioc,
768 Mpi2ConfigReply_t *mpi_reply, Mpi2IOUnitPage0_t *config_page)
769{
770 Mpi2ConfigRequest_t mpi_request;
771 int r;
772
773 memset(&mpi_request, 0, sizeof(Mpi2ConfigRequest_t));
774 mpi_request.Function = MPI2_FUNCTION_CONFIG;
775 mpi_request.Action = MPI2_CONFIG_ACTION_PAGE_HEADER;
776 mpi_request.Header.PageType = MPI2_CONFIG_PAGETYPE_IO_UNIT;
777 mpi_request.Header.PageNumber = 0;
778 mpi_request.Header.PageVersion = MPI2_IOUNITPAGE0_PAGEVERSION;
779 ioc->build_zero_len_sge_mpi(ioc, &mpi_request.PageBufferSGE);
780 r = _config_request(ioc, &mpi_request, mpi_reply,
781 MPT3_CONFIG_PAGE_DEFAULT_TIMEOUT, NULL, 0);
782 if (r)
783 goto out;
784
785 mpi_request.Action = MPI2_CONFIG_ACTION_PAGE_READ_CURRENT;
786 r = _config_request(ioc, &mpi_request, mpi_reply,
787 MPT3_CONFIG_PAGE_DEFAULT_TIMEOUT, config_page,
788 sizeof(*config_page));
789 out:
790 return r;
791}
792
793/**
794 * mpt3sas_config_get_iounit_pg1 - obtain iounit page 1
795 * @ioc: per adapter object
796 * @mpi_reply: reply mf payload returned from firmware
797 * @config_page: contents of the config page
798 * Context: sleep.
799 *
800 * Returns 0 for success, non-zero for failure.
801 */
802int
803mpt3sas_config_get_iounit_pg1(struct MPT3SAS_ADAPTER *ioc,
804 Mpi2ConfigReply_t *mpi_reply, Mpi2IOUnitPage1_t *config_page)
805{
806 Mpi2ConfigRequest_t mpi_request;
807 int r;
808
809 memset(&mpi_request, 0, sizeof(Mpi2ConfigRequest_t));
810 mpi_request.Function = MPI2_FUNCTION_CONFIG;
811 mpi_request.Action = MPI2_CONFIG_ACTION_PAGE_HEADER;
812 mpi_request.Header.PageType = MPI2_CONFIG_PAGETYPE_IO_UNIT;
813 mpi_request.Header.PageNumber = 1;
814 mpi_request.Header.PageVersion = MPI2_IOUNITPAGE1_PAGEVERSION;
815 ioc->build_zero_len_sge_mpi(ioc, &mpi_request.PageBufferSGE);
816 r = _config_request(ioc, &mpi_request, mpi_reply,
817 MPT3_CONFIG_PAGE_DEFAULT_TIMEOUT, NULL, 0);
818 if (r)
819 goto out;
820
821 mpi_request.Action = MPI2_CONFIG_ACTION_PAGE_READ_CURRENT;
822 r = _config_request(ioc, &mpi_request, mpi_reply,
823 MPT3_CONFIG_PAGE_DEFAULT_TIMEOUT, config_page,
824 sizeof(*config_page));
825 out:
826 return r;
827}
828
829/**
830 * mpt3sas_config_set_iounit_pg1 - set iounit page 1
831 * @ioc: per adapter object
832 * @mpi_reply: reply mf payload returned from firmware
833 * @config_page: contents of the config page
834 * Context: sleep.
835 *
836 * Returns 0 for success, non-zero for failure.
837 */
838int
839mpt3sas_config_set_iounit_pg1(struct MPT3SAS_ADAPTER *ioc,
840 Mpi2ConfigReply_t *mpi_reply, Mpi2IOUnitPage1_t *config_page)
841{
842 Mpi2ConfigRequest_t mpi_request;
843 int r;
844
845 memset(&mpi_request, 0, sizeof(Mpi2ConfigRequest_t));
846 mpi_request.Function = MPI2_FUNCTION_CONFIG;
847 mpi_request.Action = MPI2_CONFIG_ACTION_PAGE_HEADER;
848 mpi_request.Header.PageType = MPI2_CONFIG_PAGETYPE_IO_UNIT;
849 mpi_request.Header.PageNumber = 1;
850 mpi_request.Header.PageVersion = MPI2_IOUNITPAGE1_PAGEVERSION;
851 ioc->build_zero_len_sge_mpi(ioc, &mpi_request.PageBufferSGE);
852 r = _config_request(ioc, &mpi_request, mpi_reply,
853 MPT3_CONFIG_PAGE_DEFAULT_TIMEOUT, NULL, 0);
854 if (r)
855 goto out;
856
857 mpi_request.Action = MPI2_CONFIG_ACTION_PAGE_WRITE_CURRENT;
858 r = _config_request(ioc, &mpi_request, mpi_reply,
859 MPT3_CONFIG_PAGE_DEFAULT_TIMEOUT, config_page,
860 sizeof(*config_page));
861 out:
862 return r;
863}
864
Sreekanth Reddy42263092015-11-11 17:30:29 +0530865/**
866 * mpt3sas_config_get_iounit_pg3 - obtain iounit page 3
867 * @ioc: per adapter object
868 * @mpi_reply: reply mf payload returned from firmware
869 * @config_page: contents of the config page
870 * @sz: size of buffer passed in config_page
871 * Context: sleep.
872 *
873 * Returns 0 for success, non-zero for failure.
874 */
875int
876mpt3sas_config_get_iounit_pg3(struct MPT3SAS_ADAPTER *ioc,
877 Mpi2ConfigReply_t *mpi_reply, Mpi2IOUnitPage3_t *config_page, u16 sz)
878{
879 Mpi2ConfigRequest_t mpi_request;
880 int r;
881
882 memset(&mpi_request, 0, sizeof(Mpi2ConfigRequest_t));
883 mpi_request.Function = MPI2_FUNCTION_CONFIG;
884 mpi_request.Action = MPI2_CONFIG_ACTION_PAGE_HEADER;
885 mpi_request.Header.PageType = MPI2_CONFIG_PAGETYPE_IO_UNIT;
886 mpi_request.Header.PageNumber = 3;
887 mpi_request.Header.PageVersion = MPI2_IOUNITPAGE3_PAGEVERSION;
888 ioc->build_zero_len_sge_mpi(ioc, &mpi_request.PageBufferSGE);
889 r = _config_request(ioc, &mpi_request, mpi_reply,
890 MPT3_CONFIG_PAGE_DEFAULT_TIMEOUT, NULL, 0);
891 if (r)
892 goto out;
893
894 mpi_request.Action = MPI2_CONFIG_ACTION_PAGE_READ_CURRENT;
895 r = _config_request(ioc, &mpi_request, mpi_reply,
896 MPT3_CONFIG_PAGE_DEFAULT_TIMEOUT, config_page, sz);
897 out:
898 return r;
899}
Sreekanth Reddy42263092015-11-11 17:30:29 +0530900
Sreekanth Reddyf92363d2012-11-30 07:44:21 +0530901/**
Sreekanth Reddy2d8ce8c2015-01-12 11:38:56 +0530902 * mpt3sas_config_get_iounit_pg8 - obtain iounit page 8
903 * @ioc: per adapter object
904 * @mpi_reply: reply mf payload returned from firmware
905 * @config_page: contents of the config page
906 * Context: sleep.
907 *
908 * Returns 0 for success, non-zero for failure.
909 */
910int
911mpt3sas_config_get_iounit_pg8(struct MPT3SAS_ADAPTER *ioc,
912 Mpi2ConfigReply_t *mpi_reply, Mpi2IOUnitPage8_t *config_page)
913{
914 Mpi2ConfigRequest_t mpi_request;
915 int r;
916
917 memset(&mpi_request, 0, sizeof(Mpi2ConfigRequest_t));
918 mpi_request.Function = MPI2_FUNCTION_CONFIG;
919 mpi_request.Action = MPI2_CONFIG_ACTION_PAGE_HEADER;
920 mpi_request.Header.PageType = MPI2_CONFIG_PAGETYPE_IO_UNIT;
921 mpi_request.Header.PageNumber = 8;
922 mpi_request.Header.PageVersion = MPI2_IOUNITPAGE8_PAGEVERSION;
923 ioc->build_zero_len_sge_mpi(ioc, &mpi_request.PageBufferSGE);
924 r = _config_request(ioc, &mpi_request, mpi_reply,
925 MPT3_CONFIG_PAGE_DEFAULT_TIMEOUT, NULL, 0);
926 if (r)
927 goto out;
928
929 mpi_request.Action = MPI2_CONFIG_ACTION_PAGE_READ_CURRENT;
930 r = _config_request(ioc, &mpi_request, mpi_reply,
931 MPT3_CONFIG_PAGE_DEFAULT_TIMEOUT, config_page,
932 sizeof(*config_page));
933 out:
934 return r;
935}
936
937/**
Sreekanth Reddyf92363d2012-11-30 07:44:21 +0530938 * mpt3sas_config_get_ioc_pg8 - obtain ioc page 8
939 * @ioc: per adapter object
940 * @mpi_reply: reply mf payload returned from firmware
941 * @config_page: contents of the config page
942 * Context: sleep.
943 *
944 * Returns 0 for success, non-zero for failure.
945 */
946int
947mpt3sas_config_get_ioc_pg8(struct MPT3SAS_ADAPTER *ioc,
948 Mpi2ConfigReply_t *mpi_reply, Mpi2IOCPage8_t *config_page)
949{
950 Mpi2ConfigRequest_t mpi_request;
951 int r;
952
953 memset(&mpi_request, 0, sizeof(Mpi2ConfigRequest_t));
954 mpi_request.Function = MPI2_FUNCTION_CONFIG;
955 mpi_request.Action = MPI2_CONFIG_ACTION_PAGE_HEADER;
956 mpi_request.Header.PageType = MPI2_CONFIG_PAGETYPE_IOC;
957 mpi_request.Header.PageNumber = 8;
958 mpi_request.Header.PageVersion = MPI2_IOCPAGE8_PAGEVERSION;
959 ioc->build_zero_len_sge_mpi(ioc, &mpi_request.PageBufferSGE);
960 r = _config_request(ioc, &mpi_request, mpi_reply,
961 MPT3_CONFIG_PAGE_DEFAULT_TIMEOUT, NULL, 0);
962 if (r)
963 goto out;
964
965 mpi_request.Action = MPI2_CONFIG_ACTION_PAGE_READ_CURRENT;
966 r = _config_request(ioc, &mpi_request, mpi_reply,
967 MPT3_CONFIG_PAGE_DEFAULT_TIMEOUT, config_page,
968 sizeof(*config_page));
969 out:
970 return r;
971}
972
973/**
974 * mpt3sas_config_get_sas_device_pg0 - obtain sas device page 0
975 * @ioc: per adapter object
976 * @mpi_reply: reply mf payload returned from firmware
977 * @config_page: contents of the config page
978 * @form: GET_NEXT_HANDLE or HANDLE
979 * @handle: device handle
980 * Context: sleep.
981 *
982 * Returns 0 for success, non-zero for failure.
983 */
984int
985mpt3sas_config_get_sas_device_pg0(struct MPT3SAS_ADAPTER *ioc,
986 Mpi2ConfigReply_t *mpi_reply, Mpi2SasDevicePage0_t *config_page,
987 u32 form, u32 handle)
988{
989 Mpi2ConfigRequest_t mpi_request;
990 int r;
991
992 memset(&mpi_request, 0, sizeof(Mpi2ConfigRequest_t));
993 mpi_request.Function = MPI2_FUNCTION_CONFIG;
994 mpi_request.Action = MPI2_CONFIG_ACTION_PAGE_HEADER;
995 mpi_request.Header.PageType = MPI2_CONFIG_PAGETYPE_EXTENDED;
996 mpi_request.ExtPageType = MPI2_CONFIG_EXTPAGETYPE_SAS_DEVICE;
997 mpi_request.Header.PageVersion = MPI2_SASDEVICE0_PAGEVERSION;
998 mpi_request.Header.PageNumber = 0;
999 ioc->build_zero_len_sge_mpi(ioc, &mpi_request.PageBufferSGE);
1000 r = _config_request(ioc, &mpi_request, mpi_reply,
1001 MPT3_CONFIG_PAGE_DEFAULT_TIMEOUT, NULL, 0);
1002 if (r)
1003 goto out;
1004
1005 mpi_request.PageAddress = cpu_to_le32(form | handle);
1006 mpi_request.Action = MPI2_CONFIG_ACTION_PAGE_READ_CURRENT;
1007 r = _config_request(ioc, &mpi_request, mpi_reply,
1008 MPT3_CONFIG_PAGE_DEFAULT_TIMEOUT, config_page,
1009 sizeof(*config_page));
1010 out:
1011 return r;
1012}
1013
1014/**
1015 * mpt3sas_config_get_sas_device_pg1 - obtain sas device page 1
1016 * @ioc: per adapter object
1017 * @mpi_reply: reply mf payload returned from firmware
1018 * @config_page: contents of the config page
1019 * @form: GET_NEXT_HANDLE or HANDLE
1020 * @handle: device handle
1021 * Context: sleep.
1022 *
1023 * Returns 0 for success, non-zero for failure.
1024 */
1025int
1026mpt3sas_config_get_sas_device_pg1(struct MPT3SAS_ADAPTER *ioc,
1027 Mpi2ConfigReply_t *mpi_reply, Mpi2SasDevicePage1_t *config_page,
1028 u32 form, u32 handle)
1029{
1030 Mpi2ConfigRequest_t mpi_request;
1031 int r;
1032
1033 memset(&mpi_request, 0, sizeof(Mpi2ConfigRequest_t));
1034 mpi_request.Function = MPI2_FUNCTION_CONFIG;
1035 mpi_request.Action = MPI2_CONFIG_ACTION_PAGE_HEADER;
1036 mpi_request.Header.PageType = MPI2_CONFIG_PAGETYPE_EXTENDED;
1037 mpi_request.ExtPageType = MPI2_CONFIG_EXTPAGETYPE_SAS_DEVICE;
1038 mpi_request.Header.PageVersion = MPI2_SASDEVICE1_PAGEVERSION;
1039 mpi_request.Header.PageNumber = 1;
1040 ioc->build_zero_len_sge_mpi(ioc, &mpi_request.PageBufferSGE);
1041 r = _config_request(ioc, &mpi_request, mpi_reply,
1042 MPT3_CONFIG_PAGE_DEFAULT_TIMEOUT, NULL, 0);
1043 if (r)
1044 goto out;
1045
1046 mpi_request.PageAddress = cpu_to_le32(form | handle);
1047 mpi_request.Action = MPI2_CONFIG_ACTION_PAGE_READ_CURRENT;
1048 r = _config_request(ioc, &mpi_request, mpi_reply,
1049 MPT3_CONFIG_PAGE_DEFAULT_TIMEOUT, config_page,
1050 sizeof(*config_page));
1051 out:
1052 return r;
1053}
1054
1055/**
1056 * mpt3sas_config_get_number_hba_phys - obtain number of phys on the host
1057 * @ioc: per adapter object
1058 * @num_phys: pointer returned with the number of phys
1059 * Context: sleep.
1060 *
1061 * Returns 0 for success, non-zero for failure.
1062 */
1063int
1064mpt3sas_config_get_number_hba_phys(struct MPT3SAS_ADAPTER *ioc, u8 *num_phys)
1065{
1066 Mpi2ConfigRequest_t mpi_request;
1067 int r;
1068 u16 ioc_status;
1069 Mpi2ConfigReply_t mpi_reply;
1070 Mpi2SasIOUnitPage0_t config_page;
1071
1072 *num_phys = 0;
1073 memset(&mpi_request, 0, sizeof(Mpi2ConfigRequest_t));
1074 mpi_request.Function = MPI2_FUNCTION_CONFIG;
1075 mpi_request.Action = MPI2_CONFIG_ACTION_PAGE_HEADER;
1076 mpi_request.Header.PageType = MPI2_CONFIG_PAGETYPE_EXTENDED;
1077 mpi_request.ExtPageType = MPI2_CONFIG_EXTPAGETYPE_SAS_IO_UNIT;
1078 mpi_request.Header.PageNumber = 0;
1079 mpi_request.Header.PageVersion = MPI2_SASIOUNITPAGE0_PAGEVERSION;
1080 ioc->build_zero_len_sge_mpi(ioc, &mpi_request.PageBufferSGE);
1081 r = _config_request(ioc, &mpi_request, &mpi_reply,
1082 MPT3_CONFIG_PAGE_DEFAULT_TIMEOUT, NULL, 0);
1083 if (r)
1084 goto out;
1085
1086 mpi_request.Action = MPI2_CONFIG_ACTION_PAGE_READ_CURRENT;
1087 r = _config_request(ioc, &mpi_request, &mpi_reply,
1088 MPT3_CONFIG_PAGE_DEFAULT_TIMEOUT, &config_page,
1089 sizeof(Mpi2SasIOUnitPage0_t));
1090 if (!r) {
1091 ioc_status = le16_to_cpu(mpi_reply.IOCStatus) &
1092 MPI2_IOCSTATUS_MASK;
1093 if (ioc_status == MPI2_IOCSTATUS_SUCCESS)
1094 *num_phys = config_page.NumPhys;
1095 }
1096 out:
1097 return r;
1098}
1099
1100/**
1101 * mpt3sas_config_get_sas_iounit_pg0 - obtain sas iounit page 0
1102 * @ioc: per adapter object
1103 * @mpi_reply: reply mf payload returned from firmware
1104 * @config_page: contents of the config page
1105 * @sz: size of buffer passed in config_page
1106 * Context: sleep.
1107 *
1108 * Calling function should call config_get_number_hba_phys prior to
1109 * this function, so enough memory is allocated for config_page.
1110 *
1111 * Returns 0 for success, non-zero for failure.
1112 */
1113int
1114mpt3sas_config_get_sas_iounit_pg0(struct MPT3SAS_ADAPTER *ioc,
1115 Mpi2ConfigReply_t *mpi_reply, Mpi2SasIOUnitPage0_t *config_page,
1116 u16 sz)
1117{
1118 Mpi2ConfigRequest_t mpi_request;
1119 int r;
1120
1121 memset(&mpi_request, 0, sizeof(Mpi2ConfigRequest_t));
1122 mpi_request.Function = MPI2_FUNCTION_CONFIG;
1123 mpi_request.Action = MPI2_CONFIG_ACTION_PAGE_HEADER;
1124 mpi_request.Header.PageType = MPI2_CONFIG_PAGETYPE_EXTENDED;
1125 mpi_request.ExtPageType = MPI2_CONFIG_EXTPAGETYPE_SAS_IO_UNIT;
1126 mpi_request.Header.PageNumber = 0;
1127 mpi_request.Header.PageVersion = MPI2_SASIOUNITPAGE0_PAGEVERSION;
1128 ioc->build_zero_len_sge_mpi(ioc, &mpi_request.PageBufferSGE);
1129 r = _config_request(ioc, &mpi_request, mpi_reply,
1130 MPT3_CONFIG_PAGE_DEFAULT_TIMEOUT, NULL, 0);
1131 if (r)
1132 goto out;
1133
1134 mpi_request.Action = MPI2_CONFIG_ACTION_PAGE_READ_CURRENT;
1135 r = _config_request(ioc, &mpi_request, mpi_reply,
1136 MPT3_CONFIG_PAGE_DEFAULT_TIMEOUT, config_page, sz);
1137 out:
1138 return r;
1139}
1140
1141/**
1142 * mpt3sas_config_get_sas_iounit_pg1 - obtain sas iounit page 1
1143 * @ioc: per adapter object
1144 * @mpi_reply: reply mf payload returned from firmware
1145 * @config_page: contents of the config page
1146 * @sz: size of buffer passed in config_page
1147 * Context: sleep.
1148 *
1149 * Calling function should call config_get_number_hba_phys prior to
1150 * this function, so enough memory is allocated for config_page.
1151 *
1152 * Returns 0 for success, non-zero for failure.
1153 */
1154int
1155mpt3sas_config_get_sas_iounit_pg1(struct MPT3SAS_ADAPTER *ioc,
1156 Mpi2ConfigReply_t *mpi_reply, Mpi2SasIOUnitPage1_t *config_page,
1157 u16 sz)
1158{
1159 Mpi2ConfigRequest_t mpi_request;
1160 int r;
1161
1162 memset(&mpi_request, 0, sizeof(Mpi2ConfigRequest_t));
1163 mpi_request.Function = MPI2_FUNCTION_CONFIG;
1164 mpi_request.Action = MPI2_CONFIG_ACTION_PAGE_HEADER;
1165 mpi_request.Header.PageType = MPI2_CONFIG_PAGETYPE_EXTENDED;
1166 mpi_request.ExtPageType = MPI2_CONFIG_EXTPAGETYPE_SAS_IO_UNIT;
1167 mpi_request.Header.PageNumber = 1;
1168 mpi_request.Header.PageVersion = MPI2_SASIOUNITPAGE1_PAGEVERSION;
1169 ioc->build_zero_len_sge_mpi(ioc, &mpi_request.PageBufferSGE);
1170 r = _config_request(ioc, &mpi_request, mpi_reply,
1171 MPT3_CONFIG_PAGE_DEFAULT_TIMEOUT, NULL, 0);
1172 if (r)
1173 goto out;
1174
1175 mpi_request.Action = MPI2_CONFIG_ACTION_PAGE_READ_CURRENT;
1176 r = _config_request(ioc, &mpi_request, mpi_reply,
1177 MPT3_CONFIG_PAGE_DEFAULT_TIMEOUT, config_page, sz);
1178 out:
1179 return r;
1180}
1181
1182/**
1183 * mpt3sas_config_set_sas_iounit_pg1 - send sas iounit page 1
1184 * @ioc: per adapter object
1185 * @mpi_reply: reply mf payload returned from firmware
1186 * @config_page: contents of the config page
1187 * @sz: size of buffer passed in config_page
1188 * Context: sleep.
1189 *
1190 * Calling function should call config_get_number_hba_phys prior to
1191 * this function, so enough memory is allocated for config_page.
1192 *
1193 * Returns 0 for success, non-zero for failure.
1194 */
1195int
1196mpt3sas_config_set_sas_iounit_pg1(struct MPT3SAS_ADAPTER *ioc,
1197 Mpi2ConfigReply_t *mpi_reply, Mpi2SasIOUnitPage1_t *config_page,
1198 u16 sz)
1199{
1200 Mpi2ConfigRequest_t mpi_request;
1201 int r;
1202
1203 memset(&mpi_request, 0, sizeof(Mpi2ConfigRequest_t));
1204 mpi_request.Function = MPI2_FUNCTION_CONFIG;
1205 mpi_request.Action = MPI2_CONFIG_ACTION_PAGE_HEADER;
1206 mpi_request.Header.PageType = MPI2_CONFIG_PAGETYPE_EXTENDED;
1207 mpi_request.ExtPageType = MPI2_CONFIG_EXTPAGETYPE_SAS_IO_UNIT;
1208 mpi_request.Header.PageNumber = 1;
1209 mpi_request.Header.PageVersion = MPI2_SASIOUNITPAGE1_PAGEVERSION;
1210 ioc->build_zero_len_sge_mpi(ioc, &mpi_request.PageBufferSGE);
1211 r = _config_request(ioc, &mpi_request, mpi_reply,
1212 MPT3_CONFIG_PAGE_DEFAULT_TIMEOUT, NULL, 0);
1213 if (r)
1214 goto out;
1215
1216 mpi_request.Action = MPI2_CONFIG_ACTION_PAGE_WRITE_CURRENT;
1217 _config_request(ioc, &mpi_request, mpi_reply,
1218 MPT3_CONFIG_PAGE_DEFAULT_TIMEOUT, config_page, sz);
1219 mpi_request.Action = MPI2_CONFIG_ACTION_PAGE_WRITE_NVRAM;
1220 r = _config_request(ioc, &mpi_request, mpi_reply,
1221 MPT3_CONFIG_PAGE_DEFAULT_TIMEOUT, config_page, sz);
1222 out:
1223 return r;
1224}
1225
1226/**
1227 * mpt3sas_config_get_expander_pg0 - obtain expander page 0
1228 * @ioc: per adapter object
1229 * @mpi_reply: reply mf payload returned from firmware
1230 * @config_page: contents of the config page
1231 * @form: GET_NEXT_HANDLE or HANDLE
1232 * @handle: expander handle
1233 * Context: sleep.
1234 *
1235 * Returns 0 for success, non-zero for failure.
1236 */
1237int
1238mpt3sas_config_get_expander_pg0(struct MPT3SAS_ADAPTER *ioc, Mpi2ConfigReply_t
1239 *mpi_reply, Mpi2ExpanderPage0_t *config_page, u32 form, u32 handle)
1240{
1241 Mpi2ConfigRequest_t mpi_request;
1242 int r;
1243
1244 memset(&mpi_request, 0, sizeof(Mpi2ConfigRequest_t));
1245 mpi_request.Function = MPI2_FUNCTION_CONFIG;
1246 mpi_request.Action = MPI2_CONFIG_ACTION_PAGE_HEADER;
1247 mpi_request.Header.PageType = MPI2_CONFIG_PAGETYPE_EXTENDED;
1248 mpi_request.ExtPageType = MPI2_CONFIG_EXTPAGETYPE_SAS_EXPANDER;
1249 mpi_request.Header.PageNumber = 0;
1250 mpi_request.Header.PageVersion = MPI2_SASEXPANDER0_PAGEVERSION;
1251 ioc->build_zero_len_sge_mpi(ioc, &mpi_request.PageBufferSGE);
1252 r = _config_request(ioc, &mpi_request, mpi_reply,
1253 MPT3_CONFIG_PAGE_DEFAULT_TIMEOUT, NULL, 0);
1254 if (r)
1255 goto out;
1256
1257 mpi_request.PageAddress = cpu_to_le32(form | handle);
1258 mpi_request.Action = MPI2_CONFIG_ACTION_PAGE_READ_CURRENT;
1259 r = _config_request(ioc, &mpi_request, mpi_reply,
1260 MPT3_CONFIG_PAGE_DEFAULT_TIMEOUT, config_page,
1261 sizeof(*config_page));
1262 out:
1263 return r;
1264}
1265
1266/**
1267 * mpt3sas_config_get_expander_pg1 - obtain expander page 1
1268 * @ioc: per adapter object
1269 * @mpi_reply: reply mf payload returned from firmware
1270 * @config_page: contents of the config page
1271 * @phy_number: phy number
1272 * @handle: expander handle
1273 * Context: sleep.
1274 *
1275 * Returns 0 for success, non-zero for failure.
1276 */
1277int
1278mpt3sas_config_get_expander_pg1(struct MPT3SAS_ADAPTER *ioc, Mpi2ConfigReply_t
1279 *mpi_reply, Mpi2ExpanderPage1_t *config_page, u32 phy_number,
1280 u16 handle)
1281{
1282 Mpi2ConfigRequest_t mpi_request;
1283 int r;
1284
1285 memset(&mpi_request, 0, sizeof(Mpi2ConfigRequest_t));
1286 mpi_request.Function = MPI2_FUNCTION_CONFIG;
1287 mpi_request.Action = MPI2_CONFIG_ACTION_PAGE_HEADER;
1288 mpi_request.Header.PageType = MPI2_CONFIG_PAGETYPE_EXTENDED;
1289 mpi_request.ExtPageType = MPI2_CONFIG_EXTPAGETYPE_SAS_EXPANDER;
1290 mpi_request.Header.PageNumber = 1;
1291 mpi_request.Header.PageVersion = MPI2_SASEXPANDER1_PAGEVERSION;
1292 ioc->build_zero_len_sge_mpi(ioc, &mpi_request.PageBufferSGE);
1293 r = _config_request(ioc, &mpi_request, mpi_reply,
1294 MPT3_CONFIG_PAGE_DEFAULT_TIMEOUT, NULL, 0);
1295 if (r)
1296 goto out;
1297
1298 mpi_request.PageAddress =
1299 cpu_to_le32(MPI2_SAS_EXPAND_PGAD_FORM_HNDL_PHY_NUM |
1300 (phy_number << MPI2_SAS_EXPAND_PGAD_PHYNUM_SHIFT) | handle);
1301 mpi_request.Action = MPI2_CONFIG_ACTION_PAGE_READ_CURRENT;
1302 r = _config_request(ioc, &mpi_request, mpi_reply,
1303 MPT3_CONFIG_PAGE_DEFAULT_TIMEOUT, config_page,
1304 sizeof(*config_page));
1305 out:
1306 return r;
1307}
1308
1309/**
1310 * mpt3sas_config_get_enclosure_pg0 - obtain enclosure page 0
1311 * @ioc: per adapter object
1312 * @mpi_reply: reply mf payload returned from firmware
1313 * @config_page: contents of the config page
1314 * @form: GET_NEXT_HANDLE or HANDLE
1315 * @handle: expander handle
1316 * Context: sleep.
1317 *
1318 * Returns 0 for success, non-zero for failure.
1319 */
1320int
1321mpt3sas_config_get_enclosure_pg0(struct MPT3SAS_ADAPTER *ioc, Mpi2ConfigReply_t
1322 *mpi_reply, Mpi2SasEnclosurePage0_t *config_page, u32 form, u32 handle)
1323{
1324 Mpi2ConfigRequest_t mpi_request;
1325 int r;
1326
1327 memset(&mpi_request, 0, sizeof(Mpi2ConfigRequest_t));
1328 mpi_request.Function = MPI2_FUNCTION_CONFIG;
1329 mpi_request.Action = MPI2_CONFIG_ACTION_PAGE_HEADER;
1330 mpi_request.Header.PageType = MPI2_CONFIG_PAGETYPE_EXTENDED;
1331 mpi_request.ExtPageType = MPI2_CONFIG_EXTPAGETYPE_ENCLOSURE;
1332 mpi_request.Header.PageNumber = 0;
1333 mpi_request.Header.PageVersion = MPI2_SASENCLOSURE0_PAGEVERSION;
1334 ioc->build_zero_len_sge_mpi(ioc, &mpi_request.PageBufferSGE);
1335 r = _config_request(ioc, &mpi_request, mpi_reply,
1336 MPT3_CONFIG_PAGE_DEFAULT_TIMEOUT, NULL, 0);
1337 if (r)
1338 goto out;
1339
1340 mpi_request.PageAddress = cpu_to_le32(form | handle);
1341 mpi_request.Action = MPI2_CONFIG_ACTION_PAGE_READ_CURRENT;
1342 r = _config_request(ioc, &mpi_request, mpi_reply,
1343 MPT3_CONFIG_PAGE_DEFAULT_TIMEOUT, config_page,
1344 sizeof(*config_page));
1345 out:
1346 return r;
1347}
1348
1349/**
1350 * mpt3sas_config_get_phy_pg0 - obtain phy page 0
1351 * @ioc: per adapter object
1352 * @mpi_reply: reply mf payload returned from firmware
1353 * @config_page: contents of the config page
1354 * @phy_number: phy number
1355 * Context: sleep.
1356 *
1357 * Returns 0 for success, non-zero for failure.
1358 */
1359int
1360mpt3sas_config_get_phy_pg0(struct MPT3SAS_ADAPTER *ioc, Mpi2ConfigReply_t
1361 *mpi_reply, Mpi2SasPhyPage0_t *config_page, u32 phy_number)
1362{
1363 Mpi2ConfigRequest_t mpi_request;
1364 int r;
1365
1366 memset(&mpi_request, 0, sizeof(Mpi2ConfigRequest_t));
1367 mpi_request.Function = MPI2_FUNCTION_CONFIG;
1368 mpi_request.Action = MPI2_CONFIG_ACTION_PAGE_HEADER;
1369 mpi_request.Header.PageType = MPI2_CONFIG_PAGETYPE_EXTENDED;
1370 mpi_request.ExtPageType = MPI2_CONFIG_EXTPAGETYPE_SAS_PHY;
1371 mpi_request.Header.PageNumber = 0;
1372 mpi_request.Header.PageVersion = MPI2_SASPHY0_PAGEVERSION;
1373 ioc->build_zero_len_sge_mpi(ioc, &mpi_request.PageBufferSGE);
1374 r = _config_request(ioc, &mpi_request, mpi_reply,
1375 MPT3_CONFIG_PAGE_DEFAULT_TIMEOUT, NULL, 0);
1376 if (r)
1377 goto out;
1378
1379 mpi_request.PageAddress =
1380 cpu_to_le32(MPI2_SAS_PHY_PGAD_FORM_PHY_NUMBER | phy_number);
1381 mpi_request.Action = MPI2_CONFIG_ACTION_PAGE_READ_CURRENT;
1382 r = _config_request(ioc, &mpi_request, mpi_reply,
1383 MPT3_CONFIG_PAGE_DEFAULT_TIMEOUT, config_page,
1384 sizeof(*config_page));
1385 out:
1386 return r;
1387}
1388
1389/**
1390 * mpt3sas_config_get_phy_pg1 - obtain phy page 1
1391 * @ioc: per adapter object
1392 * @mpi_reply: reply mf payload returned from firmware
1393 * @config_page: contents of the config page
1394 * @phy_number: phy number
1395 * Context: sleep.
1396 *
1397 * Returns 0 for success, non-zero for failure.
1398 */
1399int
1400mpt3sas_config_get_phy_pg1(struct MPT3SAS_ADAPTER *ioc, Mpi2ConfigReply_t
1401 *mpi_reply, Mpi2SasPhyPage1_t *config_page, u32 phy_number)
1402{
1403 Mpi2ConfigRequest_t mpi_request;
1404 int r;
1405
1406 memset(&mpi_request, 0, sizeof(Mpi2ConfigRequest_t));
1407 mpi_request.Function = MPI2_FUNCTION_CONFIG;
1408 mpi_request.Action = MPI2_CONFIG_ACTION_PAGE_HEADER;
1409 mpi_request.Header.PageType = MPI2_CONFIG_PAGETYPE_EXTENDED;
1410 mpi_request.ExtPageType = MPI2_CONFIG_EXTPAGETYPE_SAS_PHY;
1411 mpi_request.Header.PageNumber = 1;
1412 mpi_request.Header.PageVersion = MPI2_SASPHY1_PAGEVERSION;
1413 ioc->build_zero_len_sge_mpi(ioc, &mpi_request.PageBufferSGE);
1414 r = _config_request(ioc, &mpi_request, mpi_reply,
1415 MPT3_CONFIG_PAGE_DEFAULT_TIMEOUT, NULL, 0);
1416 if (r)
1417 goto out;
1418
1419 mpi_request.PageAddress =
1420 cpu_to_le32(MPI2_SAS_PHY_PGAD_FORM_PHY_NUMBER | phy_number);
1421 mpi_request.Action = MPI2_CONFIG_ACTION_PAGE_READ_CURRENT;
1422 r = _config_request(ioc, &mpi_request, mpi_reply,
1423 MPT3_CONFIG_PAGE_DEFAULT_TIMEOUT, config_page,
1424 sizeof(*config_page));
1425 out:
1426 return r;
1427}
1428
1429/**
1430 * mpt3sas_config_get_raid_volume_pg1 - obtain raid volume page 1
1431 * @ioc: per adapter object
1432 * @mpi_reply: reply mf payload returned from firmware
1433 * @config_page: contents of the config page
1434 * @form: GET_NEXT_HANDLE or HANDLE
1435 * @handle: volume handle
1436 * Context: sleep.
1437 *
1438 * Returns 0 for success, non-zero for failure.
1439 */
1440int
1441mpt3sas_config_get_raid_volume_pg1(struct MPT3SAS_ADAPTER *ioc,
1442 Mpi2ConfigReply_t *mpi_reply, Mpi2RaidVolPage1_t *config_page, u32 form,
1443 u32 handle)
1444{
1445 Mpi2ConfigRequest_t mpi_request;
1446 int r;
1447
1448 memset(&mpi_request, 0, sizeof(Mpi2ConfigRequest_t));
1449 mpi_request.Function = MPI2_FUNCTION_CONFIG;
1450 mpi_request.Action = MPI2_CONFIG_ACTION_PAGE_HEADER;
1451 mpi_request.Header.PageType = MPI2_CONFIG_PAGETYPE_RAID_VOLUME;
1452 mpi_request.Header.PageNumber = 1;
1453 mpi_request.Header.PageVersion = MPI2_RAIDVOLPAGE1_PAGEVERSION;
1454 ioc->build_zero_len_sge_mpi(ioc, &mpi_request.PageBufferSGE);
1455 r = _config_request(ioc, &mpi_request, mpi_reply,
1456 MPT3_CONFIG_PAGE_DEFAULT_TIMEOUT, NULL, 0);
1457 if (r)
1458 goto out;
1459
1460 mpi_request.PageAddress = cpu_to_le32(form | handle);
1461 mpi_request.Action = MPI2_CONFIG_ACTION_PAGE_READ_CURRENT;
1462 r = _config_request(ioc, &mpi_request, mpi_reply,
1463 MPT3_CONFIG_PAGE_DEFAULT_TIMEOUT, config_page,
1464 sizeof(*config_page));
1465 out:
1466 return r;
1467}
1468
1469/**
1470 * mpt3sas_config_get_number_pds - obtain number of phys disk assigned to volume
1471 * @ioc: per adapter object
1472 * @handle: volume handle
1473 * @num_pds: returns pds count
1474 * Context: sleep.
1475 *
1476 * Returns 0 for success, non-zero for failure.
1477 */
1478int
1479mpt3sas_config_get_number_pds(struct MPT3SAS_ADAPTER *ioc, u16 handle,
1480 u8 *num_pds)
1481{
1482 Mpi2ConfigRequest_t mpi_request;
1483 Mpi2RaidVolPage0_t config_page;
1484 Mpi2ConfigReply_t mpi_reply;
1485 int r;
1486 u16 ioc_status;
1487
1488 memset(&mpi_request, 0, sizeof(Mpi2ConfigRequest_t));
1489 *num_pds = 0;
1490 mpi_request.Function = MPI2_FUNCTION_CONFIG;
1491 mpi_request.Action = MPI2_CONFIG_ACTION_PAGE_HEADER;
1492 mpi_request.Header.PageType = MPI2_CONFIG_PAGETYPE_RAID_VOLUME;
1493 mpi_request.Header.PageNumber = 0;
1494 mpi_request.Header.PageVersion = MPI2_RAIDVOLPAGE0_PAGEVERSION;
1495 ioc->build_zero_len_sge_mpi(ioc, &mpi_request.PageBufferSGE);
1496 r = _config_request(ioc, &mpi_request, &mpi_reply,
1497 MPT3_CONFIG_PAGE_DEFAULT_TIMEOUT, NULL, 0);
1498 if (r)
1499 goto out;
1500
1501 mpi_request.PageAddress =
1502 cpu_to_le32(MPI2_RAID_VOLUME_PGAD_FORM_HANDLE | handle);
1503 mpi_request.Action = MPI2_CONFIG_ACTION_PAGE_READ_CURRENT;
1504 r = _config_request(ioc, &mpi_request, &mpi_reply,
1505 MPT3_CONFIG_PAGE_DEFAULT_TIMEOUT, &config_page,
1506 sizeof(Mpi2RaidVolPage0_t));
1507 if (!r) {
1508 ioc_status = le16_to_cpu(mpi_reply.IOCStatus) &
1509 MPI2_IOCSTATUS_MASK;
1510 if (ioc_status == MPI2_IOCSTATUS_SUCCESS)
1511 *num_pds = config_page.NumPhysDisks;
1512 }
1513
1514 out:
1515 return r;
1516}
1517
1518/**
1519 * mpt3sas_config_get_raid_volume_pg0 - obtain raid volume page 0
1520 * @ioc: per adapter object
1521 * @mpi_reply: reply mf payload returned from firmware
1522 * @config_page: contents of the config page
1523 * @form: GET_NEXT_HANDLE or HANDLE
1524 * @handle: volume handle
1525 * @sz: size of buffer passed in config_page
1526 * Context: sleep.
1527 *
1528 * Returns 0 for success, non-zero for failure.
1529 */
1530int
1531mpt3sas_config_get_raid_volume_pg0(struct MPT3SAS_ADAPTER *ioc,
1532 Mpi2ConfigReply_t *mpi_reply, Mpi2RaidVolPage0_t *config_page, u32 form,
1533 u32 handle, u16 sz)
1534{
1535 Mpi2ConfigRequest_t mpi_request;
1536 int r;
1537
1538 memset(&mpi_request, 0, sizeof(Mpi2ConfigRequest_t));
1539 mpi_request.Function = MPI2_FUNCTION_CONFIG;
1540 mpi_request.Action = MPI2_CONFIG_ACTION_PAGE_HEADER;
1541 mpi_request.Header.PageType = MPI2_CONFIG_PAGETYPE_RAID_VOLUME;
1542 mpi_request.Header.PageNumber = 0;
1543 mpi_request.Header.PageVersion = MPI2_RAIDVOLPAGE0_PAGEVERSION;
1544 ioc->build_zero_len_sge_mpi(ioc, &mpi_request.PageBufferSGE);
1545 r = _config_request(ioc, &mpi_request, mpi_reply,
1546 MPT3_CONFIG_PAGE_DEFAULT_TIMEOUT, NULL, 0);
1547 if (r)
1548 goto out;
1549
1550 mpi_request.PageAddress = cpu_to_le32(form | handle);
1551 mpi_request.Action = MPI2_CONFIG_ACTION_PAGE_READ_CURRENT;
1552 r = _config_request(ioc, &mpi_request, mpi_reply,
1553 MPT3_CONFIG_PAGE_DEFAULT_TIMEOUT, config_page, sz);
1554 out:
1555 return r;
1556}
1557
1558/**
1559 * mpt3sas_config_get_phys_disk_pg0 - obtain phys disk page 0
1560 * @ioc: per adapter object
1561 * @mpi_reply: reply mf payload returned from firmware
1562 * @config_page: contents of the config page
1563 * @form: GET_NEXT_PHYSDISKNUM, PHYSDISKNUM, DEVHANDLE
1564 * @form_specific: specific to the form
1565 * Context: sleep.
1566 *
1567 * Returns 0 for success, non-zero for failure.
1568 */
1569int
1570mpt3sas_config_get_phys_disk_pg0(struct MPT3SAS_ADAPTER *ioc, Mpi2ConfigReply_t
1571 *mpi_reply, Mpi2RaidPhysDiskPage0_t *config_page, u32 form,
1572 u32 form_specific)
1573{
1574 Mpi2ConfigRequest_t mpi_request;
1575 int r;
1576
1577 memset(&mpi_request, 0, sizeof(Mpi2ConfigRequest_t));
1578 mpi_request.Function = MPI2_FUNCTION_CONFIG;
1579 mpi_request.Action = MPI2_CONFIG_ACTION_PAGE_HEADER;
1580 mpi_request.Header.PageType = MPI2_CONFIG_PAGETYPE_RAID_PHYSDISK;
1581 mpi_request.Header.PageNumber = 0;
1582 mpi_request.Header.PageVersion = MPI2_RAIDPHYSDISKPAGE0_PAGEVERSION;
1583 ioc->build_zero_len_sge_mpi(ioc, &mpi_request.PageBufferSGE);
1584 r = _config_request(ioc, &mpi_request, mpi_reply,
1585 MPT3_CONFIG_PAGE_DEFAULT_TIMEOUT, NULL, 0);
1586 if (r)
1587 goto out;
1588
1589 mpi_request.PageAddress = cpu_to_le32(form | form_specific);
1590 mpi_request.Action = MPI2_CONFIG_ACTION_PAGE_READ_CURRENT;
1591 r = _config_request(ioc, &mpi_request, mpi_reply,
1592 MPT3_CONFIG_PAGE_DEFAULT_TIMEOUT, config_page,
1593 sizeof(*config_page));
1594 out:
1595 return r;
1596}
1597
1598/**
1599 * mpt3sas_config_get_volume_handle - returns volume handle for give hidden
1600 * raid components
1601 * @ioc: per adapter object
1602 * @pd_handle: phys disk handle
1603 * @volume_handle: volume handle
1604 * Context: sleep.
1605 *
1606 * Returns 0 for success, non-zero for failure.
1607 */
1608int
1609mpt3sas_config_get_volume_handle(struct MPT3SAS_ADAPTER *ioc, u16 pd_handle,
1610 u16 *volume_handle)
1611{
1612 Mpi2RaidConfigurationPage0_t *config_page = NULL;
1613 Mpi2ConfigRequest_t mpi_request;
1614 Mpi2ConfigReply_t mpi_reply;
1615 int r, i, config_page_sz;
1616 u16 ioc_status;
1617 int config_num;
1618 u16 element_type;
1619 u16 phys_disk_dev_handle;
1620
1621 *volume_handle = 0;
1622 memset(&mpi_request, 0, sizeof(Mpi2ConfigRequest_t));
1623 mpi_request.Function = MPI2_FUNCTION_CONFIG;
1624 mpi_request.Action = MPI2_CONFIG_ACTION_PAGE_HEADER;
1625 mpi_request.Header.PageType = MPI2_CONFIG_PAGETYPE_EXTENDED;
1626 mpi_request.ExtPageType = MPI2_CONFIG_EXTPAGETYPE_RAID_CONFIG;
1627 mpi_request.Header.PageVersion = MPI2_RAIDCONFIG0_PAGEVERSION;
1628 mpi_request.Header.PageNumber = 0;
1629 ioc->build_zero_len_sge_mpi(ioc, &mpi_request.PageBufferSGE);
1630 r = _config_request(ioc, &mpi_request, &mpi_reply,
1631 MPT3_CONFIG_PAGE_DEFAULT_TIMEOUT, NULL, 0);
1632 if (r)
1633 goto out;
1634
1635 mpi_request.Action = MPI2_CONFIG_ACTION_PAGE_READ_CURRENT;
1636 config_page_sz = (le16_to_cpu(mpi_reply.ExtPageLength) * 4);
1637 config_page = kmalloc(config_page_sz, GFP_KERNEL);
1638 if (!config_page) {
1639 r = -1;
1640 goto out;
1641 }
1642
1643 config_num = 0xff;
1644 while (1) {
1645 mpi_request.PageAddress = cpu_to_le32(config_num +
1646 MPI2_RAID_PGAD_FORM_GET_NEXT_CONFIGNUM);
1647 r = _config_request(ioc, &mpi_request, &mpi_reply,
1648 MPT3_CONFIG_PAGE_DEFAULT_TIMEOUT, config_page,
1649 config_page_sz);
1650 if (r)
1651 goto out;
1652 r = -1;
1653 ioc_status = le16_to_cpu(mpi_reply.IOCStatus) &
1654 MPI2_IOCSTATUS_MASK;
1655 if (ioc_status != MPI2_IOCSTATUS_SUCCESS)
1656 goto out;
1657 for (i = 0; i < config_page->NumElements; i++) {
1658 element_type = le16_to_cpu(config_page->
1659 ConfigElement[i].ElementFlags) &
1660 MPI2_RAIDCONFIG0_EFLAGS_MASK_ELEMENT_TYPE;
1661 if (element_type ==
1662 MPI2_RAIDCONFIG0_EFLAGS_VOL_PHYS_DISK_ELEMENT ||
1663 element_type ==
1664 MPI2_RAIDCONFIG0_EFLAGS_OCE_ELEMENT) {
1665 phys_disk_dev_handle =
1666 le16_to_cpu(config_page->ConfigElement[i].
1667 PhysDiskDevHandle);
1668 if (phys_disk_dev_handle == pd_handle) {
1669 *volume_handle =
1670 le16_to_cpu(config_page->
1671 ConfigElement[i].VolDevHandle);
1672 r = 0;
1673 goto out;
1674 }
1675 } else if (element_type ==
1676 MPI2_RAIDCONFIG0_EFLAGS_HOT_SPARE_ELEMENT) {
1677 *volume_handle = 0;
1678 r = 0;
1679 goto out;
1680 }
1681 }
1682 config_num = config_page->ConfigNum;
1683 }
1684 out:
1685 kfree(config_page);
1686 return r;
1687}
1688
1689/**
1690 * mpt3sas_config_get_volume_wwid - returns wwid given the volume handle
1691 * @ioc: per adapter object
1692 * @volume_handle: volume handle
1693 * @wwid: volume wwid
1694 * Context: sleep.
1695 *
1696 * Returns 0 for success, non-zero for failure.
1697 */
1698int
1699mpt3sas_config_get_volume_wwid(struct MPT3SAS_ADAPTER *ioc, u16 volume_handle,
1700 u64 *wwid)
1701{
1702 Mpi2ConfigReply_t mpi_reply;
1703 Mpi2RaidVolPage1_t raid_vol_pg1;
1704
1705 *wwid = 0;
1706 if (!(mpt3sas_config_get_raid_volume_pg1(ioc, &mpi_reply,
1707 &raid_vol_pg1, MPI2_RAID_VOLUME_PGAD_FORM_HANDLE,
1708 volume_handle))) {
1709 *wwid = le64_to_cpu(raid_vol_pg1.WWID);
1710 return 0;
1711 } else
1712 return -1;
1713}