2 * Marvell UMI head file
4 * Copyright 2011 Marvell. <jyli@marvell.com>
6 * This file is licensed under GPLv2.
8 * This program is free software; you can redistribute it and/or
9 * modify it under the terms of the GNU General Public License as
10 * published by the Free Software Foundation; version 2 of the
13 * This program is distributed in the hope that it will be useful,
14 * but WITHOUT ANY WARRANTY; without even the implied warranty of
15 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
16 * General Public License for more details.
18 * You should have received a copy of the GNU General Public License
19 * along with this program; if not, write to the Free Software
20 * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307
27 #define MAX_BASE_ADDRESS 6
32 #define VER_BUILD 1500
34 #define MV_DRIVER_NAME "mvumi"
35 #define PCI_DEVICE_ID_MARVELL_MV9143 0x9143
36 #define PCI_DEVICE_ID_MARVELL_MV9580 0x9580
38 #define MVUMI_INTERNAL_CMD_WAIT_TIME 45
39 #define MVUMI_INQUIRY_LENGTH 44
40 #define MVUMI_INQUIRY_UUID_OFF 36
41 #define MVUMI_INQUIRY_UUID_LEN 8
43 #define IS_DMA64 (sizeof(dma_addr_t) == 8)
45 enum mvumi_qc_result
{
46 MV_QUEUE_COMMAND_RESULT_SENT
= 0,
47 MV_QUEUE_COMMAND_RESULT_NO_RESOURCE
,
50 struct mvumi_hw_regs
{
52 void *main_int_cause_reg
;
53 void *enpointa_mask_reg
;
54 void *enpointb_mask_reg
;
57 void *rstoutn_mask_reg
;
58 void *sys_soft_rst_reg
;
61 void *pciea_to_arm_drbl_reg
;
62 void *arm_to_pciea_drbl_reg
;
63 void *arm_to_pciea_mask_reg
;
64 void *pciea_to_arm_msg0
;
65 void *pciea_to_arm_msg1
;
66 void *arm_to_pciea_msg0
;
67 void *arm_to_pciea_msg1
;
73 /* For Message Unit */
76 void *inb_aval_count_basel
;
77 void *inb_aval_count_baseh
;
78 void *inb_write_pointer
;
79 void *inb_read_pointer
;
80 void *outb_list_basel
;
81 void *outb_list_baseh
;
82 void *outb_copy_basel
;
83 void *outb_copy_baseh
;
84 void *outb_copy_pointer
;
85 void *outb_read_pointer
;
89 void *outb_coal_timeout
;
91 /* Bit setting for HW */
96 u32 int_drbl_int_mask
;
97 u32 int_main_int_mask
;
98 u32 cl_pointer_toggle
;
105 struct mvumi_dyn_list_entry
{
112 #define SCSI_CMD_MARVELL_SPECIFIC 0xE1
113 #define CDB_CORE_MODULE 0x1
114 #define CDB_CORE_SHUTDOWN 0xB
117 DRBL_HANDSHAKE
= 1 << 0,
118 DRBL_SOFT_RESET
= 1 << 1,
119 DRBL_BUS_CHANGE
= 1 << 2,
120 DRBL_EVENT_NOTIFY
= 1 << 3,
121 DRBL_MU_RESET
= 1 << 4,
122 DRBL_HANDSHAKE_ISR
= DRBL_HANDSHAKE
,
125 * Command flag is the flag for the CDB command itself
127 /* 1-non data; 0-data command */
128 CMD_FLAG_NON_DATA
= 1 << 0,
129 CMD_FLAG_DMA
= 1 << 1,
130 CMD_FLAG_PIO
= 1 << 2,
131 /* 1-host read data */
132 CMD_FLAG_DATA_IN
= 1 << 3,
133 /* 1-host write data */
134 CMD_FLAG_DATA_OUT
= 1 << 4,
135 CMD_FLAG_PRDT_IN_HOST
= 1 << 5,
138 #define APICDB0_EVENT 0xF4
139 #define APICDB1_EVENT_GETEVENT 0
140 #define APICDB1_HOST_GETEVENT 1
141 #define MAX_EVENTS_RETURNED 6
143 #define DEVICE_OFFLINE 0
144 #define DEVICE_ONLINE 1
146 struct mvumi_hotplug_event
{
152 struct mvumi_driver_event
{
160 u8 sense_data_length
;
165 struct mvumi_event_req
{
167 unsigned char reserved
[3];
168 struct mvumi_driver_event events
[MAX_EVENTS_RETURNED
];
171 struct mvumi_events_wq
{
172 struct work_struct work_q
;
173 struct mvumi_hba
*mhba
;
178 #define HS_CAPABILITY_SUPPORT_COMPACT_SG (1U << 4)
179 #define HS_CAPABILITY_SUPPORT_PRD_HOST (1U << 5)
180 #define HS_CAPABILITY_SUPPORT_DYN_SRC (1U << 6)
181 #define HS_CAPABILITY_NEW_PAGE_IO_DEPTH_DEF (1U << 14)
183 #define MVUMI_MAX_SG_ENTRY 32
184 #define SGD_EOT (1L << 27)
185 #define SGD_EOT_CP (1L << 22)
193 struct mvumi_compact_sgl
{
199 #define GET_COMPACT_SGD_SIZE(sgd) \
200 ((((struct mvumi_compact_sgl *)(sgd))->flags) & 0x3FFFFFL)
202 #define SET_COMPACT_SGD_SIZE(sgd, sz) do { \
203 (((struct mvumi_compact_sgl *)(sgd))->flags) &= ~0x3FFFFFL; \
204 (((struct mvumi_compact_sgl *)(sgd))->flags) |= (sz); \
206 #define sgd_getsz(_mhba, sgd, sz) do { \
207 if (_mhba->hba_capability & HS_CAPABILITY_SUPPORT_COMPACT_SG) \
208 (sz) = GET_COMPACT_SGD_SIZE(sgd); \
210 (sz) = (sgd)->size; \
213 #define sgd_setsz(_mhba, sgd, sz) do { \
214 if (_mhba->hba_capability & HS_CAPABILITY_SUPPORT_COMPACT_SG) \
215 SET_COMPACT_SGD_SIZE(sgd, sz); \
217 (sgd)->size = (sz); \
220 #define sgd_inc(_mhba, sgd) do { \
221 if (_mhba->hba_capability & HS_CAPABILITY_SUPPORT_COMPACT_SG) \
222 sgd = (struct mvumi_sgl *)(((unsigned char *) (sgd)) + 12); \
224 sgd = (struct mvumi_sgl *)(((unsigned char *) (sgd)) + 16); \
228 struct list_head entry
;
232 unsigned short type
; /* enum Resource_Type */
237 RESOURCE_CACHED_MEMORY
= 0,
238 RESOURCE_UNCACHED_MEMORY
241 struct mvumi_sense_data
{
247 u8 incorrect_length
:1;
251 u8 additional_sense_length
;
252 u8 command_specific_information
[4];
253 u8 additional_sense_code
;
254 u8 additional_sense_code_qualifier
;
255 u8 field_replaceable_unit_code
;
256 u8 sense_key_specific
[3];
259 /* Request initiator must set the status to REQ_STATUS_PENDING. */
260 #define REQ_STATUS_PENDING 0x80
263 struct list_head queue_pointer
;
264 struct mvumi_msg_frame
*frame
;
265 dma_addr_t frame_phys
;
266 struct scsi_cmnd
*scmd
;
269 unsigned short request_id
;
270 unsigned char cmd_status
;
274 * the function type of the in bound frame
276 #define CL_FUN_SCSI_CMD 0x1
278 struct mvumi_msg_frame
{
285 u32 data_transfer_length
;
288 u8 cdb
[MAX_COMMAND_SIZE
];
293 * the respond flag for data_payload of the out bound frame
295 #define CL_RSP_FLAG_NODATA 0x0
296 #define CL_RSP_FLAG_SENSEDATA 0x1
298 struct mvumi_rsp_frame
{
302 u8 rsp_flag
; /* Indicates the type of Data_Payload.*/
307 struct mvumi_ob_data
{
308 struct list_head list
;
309 unsigned char data
[0];
312 struct version_info
{
319 #define FW_MAX_DELAY 30
320 #define MVUMI_FW_BUSY (1U << 0)
321 #define MVUMI_FW_ATTACH (1U << 1)
322 #define MVUMI_FW_ALLOC (1U << 2)
325 * State is the state of the MU
327 #define FW_STATE_IDLE 0
328 #define FW_STATE_STARTING 1
329 #define FW_STATE_HANDSHAKING 2
330 #define FW_STATE_STARTED 3
331 #define FW_STATE_ABORT 4
333 #define HANDSHAKE_SIGNATURE 0x5A5A5A5AL
334 #define HANDSHAKE_READYSTATE 0x55AA5AA5L
335 #define HANDSHAKE_DONESTATE 0x55AAA55AL
337 /* HandShake Status definition */
338 #define HS_STATUS_OK 1
339 #define HS_STATUS_ERR 2
340 #define HS_STATUS_INVALID 3
342 /* HandShake State/Cmd definition */
345 #define HS_S_PAGE_ADDR 3
346 #define HS_S_QUERY_PAGE 4
347 #define HS_S_SEND_PAGE 5
350 #define HS_PAGE_VERIFY_SIZE 128
352 #define HS_GET_STATE(a) (a & 0xFFFF)
353 #define HS_GET_STATUS(a) ((a & 0xFFFF0000) >> 16)
354 #define HS_SET_STATE(a, b) (a |= (b & 0xFFFF))
355 #define HS_SET_STATUS(a, b) (a |= ((b & 0xFFFF) << 16))
357 /* handshake frame */
358 struct mvumi_hs_frame
{
360 /* host information */
363 struct version_info host_ver
; /* bios or driver version */
365 /* controller information */
371 /* communication list configuration */
383 u64 seconds_since1970
;
386 struct mvumi_hs_header
{
390 u32 frame_content
[1];
394 * the page code type of the handshake header
396 #define HS_PAGE_FIRM_CAP 0x1
397 #define HS_PAGE_HOST_INFO 0x2
398 #define HS_PAGE_FIRM_CTL 0x3
399 #define HS_PAGE_CL_INFO 0x4
400 #define HS_PAGE_TOTAL 0x5
402 #define HSP_SIZE(i) sizeof(struct mvumi_hs_page##i)
404 #define HSP_MAX_SIZE ({ \
406 m1 = max(HSP_SIZE(1), HSP_SIZE(3)); \
407 m2 = max(HSP_SIZE(2), HSP_SIZE(4)); \
408 size = max(m1, m2); \
412 /* The format of the page code for Firmware capability */
413 struct mvumi_hs_page1
{
419 u16 max_devices_support
;
422 u32 max_transfer_size
;
423 struct version_info fw_ver
;
424 u8 cl_in_max_entry_size
;
425 u8 cl_out_max_entry_size
;
426 u8 cl_inout_list_depth
;
432 /* The format of the page code for Host information */
433 struct mvumi_hs_page2
{
441 struct version_info host_ver
;
446 u64 seconds_since1970
;
449 /* The format of the page code for firmware control */
450 struct mvumi_hs_page3
{
456 u32 host_bufferaddr_l
;
457 u32 host_bufferaddr_h
;
458 u32 host_eventaddr_l
;
459 u32 host_eventaddr_h
;
462 struct mvumi_hs_page4
{
477 unsigned short *stack
;
482 struct mvumi_device
{
483 struct list_head list
;
484 struct scsi_device
*sdev
;
491 void *base_addr
[MAX_BASE_ADDRESS
];
492 u32 pci_base
[MAX_BASE_ADDRESS
];
494 struct list_head cmd_pool
;
495 struct Scsi_Host
*shost
;
496 wait_queue_head_t int_cmd_wait_q
;
497 struct pci_dev
*pdev
;
498 unsigned int unique_id
;
499 atomic_t fw_outstanding
;
500 struct mvumi_instance_template
*instancet
;
503 dma_addr_t ib_list_phys
;
506 dma_addr_t ib_frame_phys
;
509 dma_addr_t ob_list_phys
;
512 dma_addr_t ib_shadow_phys
;
515 dma_addr_t ob_shadow_phys
;
517 void *handshake_page
;
518 dma_addr_t handshake_page_phys
;
520 unsigned int global_isr
;
521 unsigned int isr_status
;
523 unsigned short max_sge
;
524 unsigned short max_target_id
;
525 unsigned char *target_map
;
527 unsigned int list_num_io
;
528 unsigned int ib_max_size
;
529 unsigned int ob_max_size
;
530 unsigned int ib_max_size_setting
;
531 unsigned int ob_max_size_setting
;
532 unsigned int max_transfer_size
;
533 unsigned char hba_total_pages
;
534 unsigned char fw_flag
;
535 unsigned char request_id_enabled
;
536 unsigned char eot_flag
;
537 unsigned short hba_capability
;
538 unsigned short io_seq
;
540 unsigned int ib_cur_slot
;
541 unsigned int ob_cur_slot
;
542 unsigned int fw_state
;
543 struct mutex sas_discovery_mutex
;
545 struct list_head ob_data_list
;
546 struct list_head free_ob_list
;
547 struct list_head res_list
;
548 struct list_head waiting_req_list
;
550 struct mvumi_tag tag_pool
;
551 struct mvumi_cmd
**tag_cmd
;
552 struct mvumi_hw_regs
*regs
;
553 struct mutex device_lock
;
554 struct list_head mhba_dev_list
;
555 struct list_head shost_dev_list
;
556 struct task_struct
*dm_thread
;
560 struct mvumi_instance_template
{
561 void (*fire_cmd
) (struct mvumi_hba
*, struct mvumi_cmd
*);
562 void (*enable_intr
) (struct mvumi_hba
*);
563 void (*disable_intr
) (struct mvumi_hba
*);
564 int (*clear_intr
) (void *);
565 unsigned int (*read_fw_status_reg
) (struct mvumi_hba
*);
566 unsigned int (*check_ib_list
) (struct mvumi_hba
*);
567 int (*check_ob_list
) (struct mvumi_hba
*, unsigned int *,
569 int (*reset_host
) (struct mvumi_hba
*);
572 extern struct timezone sys_tz
;