1*1da177e4SLinus Torvalds /* 2*1da177e4SLinus Torvalds * 3*1da177e4SLinus Torvalds * Linux MegaRAID device driver 4*1da177e4SLinus Torvalds * 5*1da177e4SLinus Torvalds * Copyright (c) 2003-2004 LSI Logic Corporation. 6*1da177e4SLinus Torvalds * 7*1da177e4SLinus Torvalds * This program is free software; you can redistribute it and/or 8*1da177e4SLinus Torvalds * modify it under the terms of the GNU General Public License 9*1da177e4SLinus Torvalds * as published by the Free Software Foundation; either version 10*1da177e4SLinus Torvalds * 2 of the License, or (at your option) any later version. 11*1da177e4SLinus Torvalds * 12*1da177e4SLinus Torvalds * FILE : mega_common.h 13*1da177e4SLinus Torvalds * 14*1da177e4SLinus Torvalds * Libaray of common routine used by all low-level megaraid drivers 15*1da177e4SLinus Torvalds */ 16*1da177e4SLinus Torvalds 17*1da177e4SLinus Torvalds #ifndef _MEGA_COMMON_H_ 18*1da177e4SLinus Torvalds #define _MEGA_COMMON_H_ 19*1da177e4SLinus Torvalds 20*1da177e4SLinus Torvalds #include <linux/kernel.h> 21*1da177e4SLinus Torvalds #include <linux/types.h> 22*1da177e4SLinus Torvalds #include <linux/pci.h> 23*1da177e4SLinus Torvalds #include <linux/spinlock.h> 24*1da177e4SLinus Torvalds #include <linux/interrupt.h> 25*1da177e4SLinus Torvalds #include <linux/delay.h> 26*1da177e4SLinus Torvalds #include <linux/blkdev.h> 27*1da177e4SLinus Torvalds #include <linux/list.h> 28*1da177e4SLinus Torvalds #include <linux/version.h> 29*1da177e4SLinus Torvalds #include <linux/moduleparam.h> 30*1da177e4SLinus Torvalds #include <asm/semaphore.h> 31*1da177e4SLinus Torvalds #include <scsi/scsi.h> 32*1da177e4SLinus Torvalds #include <scsi/scsi_cmnd.h> 33*1da177e4SLinus Torvalds #include <scsi/scsi_device.h> 34*1da177e4SLinus Torvalds #include <scsi/scsi_host.h> 35*1da177e4SLinus Torvalds 36*1da177e4SLinus Torvalds 37*1da177e4SLinus Torvalds #define LSI_MAX_CHANNELS 16 38*1da177e4SLinus Torvalds #define LSI_MAX_LOGICAL_DRIVES_64LD (64+1) 39*1da177e4SLinus Torvalds 40*1da177e4SLinus Torvalds 41*1da177e4SLinus Torvalds /** 42*1da177e4SLinus Torvalds * scb_t - scsi command control block 43*1da177e4SLinus Torvalds * @param ccb : command control block for individual driver 44*1da177e4SLinus Torvalds * @param list : list of control blocks 45*1da177e4SLinus Torvalds * @param gp : general purpose field for LLDs 46*1da177e4SLinus Torvalds * @param sno : all SCBs have a serial number 47*1da177e4SLinus Torvalds * @param scp : associated scsi command 48*1da177e4SLinus Torvalds * @param state : current state of scb 49*1da177e4SLinus Torvalds * @param dma_dir : direction of data transfer 50*1da177e4SLinus Torvalds * @param dma_type : transfer with sg list, buffer, or no data transfer 51*1da177e4SLinus Torvalds * @param dev_channel : actual channel on the device 52*1da177e4SLinus Torvalds * @param dev_target : actual target on the device 53*1da177e4SLinus Torvalds * @param status : completion status 54*1da177e4SLinus Torvalds * 55*1da177e4SLinus Torvalds * This is our central data structure to issue commands the each driver. 56*1da177e4SLinus Torvalds * Driver specific data structures are maintained in the ccb field. 57*1da177e4SLinus Torvalds * scb provides a field 'gp', which can be used by LLD for its own purposes 58*1da177e4SLinus Torvalds * 59*1da177e4SLinus Torvalds * dev_channel and dev_target must be initialized with the actual channel and 60*1da177e4SLinus Torvalds * target on the controller. 61*1da177e4SLinus Torvalds */ 62*1da177e4SLinus Torvalds typedef struct { 63*1da177e4SLinus Torvalds caddr_t ccb; 64*1da177e4SLinus Torvalds struct list_head list; 65*1da177e4SLinus Torvalds unsigned long gp; 66*1da177e4SLinus Torvalds unsigned int sno; 67*1da177e4SLinus Torvalds struct scsi_cmnd *scp; 68*1da177e4SLinus Torvalds uint32_t state; 69*1da177e4SLinus Torvalds uint32_t dma_direction; 70*1da177e4SLinus Torvalds uint32_t dma_type; 71*1da177e4SLinus Torvalds uint16_t dev_channel; 72*1da177e4SLinus Torvalds uint16_t dev_target; 73*1da177e4SLinus Torvalds uint32_t status; 74*1da177e4SLinus Torvalds } scb_t; 75*1da177e4SLinus Torvalds 76*1da177e4SLinus Torvalds /* 77*1da177e4SLinus Torvalds * SCB states as it transitions from one state to another 78*1da177e4SLinus Torvalds */ 79*1da177e4SLinus Torvalds #define SCB_FREE 0x0000 /* on the free list */ 80*1da177e4SLinus Torvalds #define SCB_ACTIVE 0x0001 /* off the free list */ 81*1da177e4SLinus Torvalds #define SCB_PENDQ 0x0002 /* on the pending queue */ 82*1da177e4SLinus Torvalds #define SCB_ISSUED 0x0004 /* issued - owner f/w */ 83*1da177e4SLinus Torvalds #define SCB_ABORT 0x0008 /* Got an abort for this one */ 84*1da177e4SLinus Torvalds #define SCB_RESET 0x0010 /* Got a reset for this one */ 85*1da177e4SLinus Torvalds 86*1da177e4SLinus Torvalds /* 87*1da177e4SLinus Torvalds * DMA types for scb 88*1da177e4SLinus Torvalds */ 89*1da177e4SLinus Torvalds #define MRAID_DMA_NONE 0x0000 /* no data transfer for this command */ 90*1da177e4SLinus Torvalds #define MRAID_DMA_WSG 0x0001 /* data transfer using a sg list */ 91*1da177e4SLinus Torvalds #define MRAID_DMA_WBUF 0x0002 /* data transfer using a contiguous buffer */ 92*1da177e4SLinus Torvalds 93*1da177e4SLinus Torvalds 94*1da177e4SLinus Torvalds /** 95*1da177e4SLinus Torvalds * struct adapter_t - driver's initialization structure 96*1da177e4SLinus Torvalds * @param dpc_h : tasklet handle 97*1da177e4SLinus Torvalds * @param pdev : pci configuration pointer for kernel 98*1da177e4SLinus Torvalds * @param host : pointer to host structure of mid-layer 99*1da177e4SLinus Torvalds * @param host_lock : pointer to appropriate lock 100*1da177e4SLinus Torvalds * @param lock : synchronization lock for mid-layer and driver 101*1da177e4SLinus Torvalds * @param quiescent : driver is quiescent for now. 102*1da177e4SLinus Torvalds * @param outstanding_cmds : number of commands pending in the driver 103*1da177e4SLinus Torvalds * @param kscb_list : pointer to the bulk of SCBs pointers for IO 104*1da177e4SLinus Torvalds * @param kscb_pool : pool of free scbs for IO 105*1da177e4SLinus Torvalds * @param kscb_pool_lock : lock for pool of free scbs 106*1da177e4SLinus Torvalds * @param pend_list : pending commands list 107*1da177e4SLinus Torvalds * @param pend_list_lock : exlusion lock for pending commands list 108*1da177e4SLinus Torvalds * @param completed_list : list of completed commands 109*1da177e4SLinus Torvalds * @param completed_list_lock : exclusion lock for list of completed commands 110*1da177e4SLinus Torvalds * @param sglen : max sg elements supported 111*1da177e4SLinus Torvalds * @param device_ids : to convert kernel device addr to our devices. 112*1da177e4SLinus Torvalds * @param raid_device : raid adapter specific pointer 113*1da177e4SLinus Torvalds * @param max_channel : maximum channel number supported - inclusive 114*1da177e4SLinus Torvalds * @param max_target : max target supported - inclusive 115*1da177e4SLinus Torvalds * @param max_lun : max lun supported - inclusive 116*1da177e4SLinus Torvalds * @param unique_id : unique identifier for each adapter 117*1da177e4SLinus Torvalds * @param irq : IRQ for this adapter 118*1da177e4SLinus Torvalds * @param ito : internal timeout value, (-1) means no timeout 119*1da177e4SLinus Torvalds * @param ibuf : buffer to issue internal commands 120*1da177e4SLinus Torvalds * @param ibuf_dma_h : dma handle for the above buffer 121*1da177e4SLinus Torvalds * @param uscb_list : SCB pointers for user cmds, common mgmt module 122*1da177e4SLinus Torvalds * @param uscb_pool : pool of SCBs for user commands 123*1da177e4SLinus Torvalds * @param uscb_pool_lock : exclusion lock for these SCBs 124*1da177e4SLinus Torvalds * @param max_cmds : max outstanding commands 125*1da177e4SLinus Torvalds * @param fw_version : firmware version 126*1da177e4SLinus Torvalds * @param bios_version : bios version 127*1da177e4SLinus Torvalds * @param max_cdb_sz : biggest CDB size supported. 128*1da177e4SLinus Torvalds * @param ha : is high availability present - clustering 129*1da177e4SLinus Torvalds * @param init_id : initiator ID, the default value should be 7 130*1da177e4SLinus Torvalds * @param max_sectors : max sectors per request 131*1da177e4SLinus Torvalds * @param cmd_per_lun : max outstanding commands per LUN 132*1da177e4SLinus Torvalds * @param being_detached : set when unloading, no more mgmt calls 133*1da177e4SLinus Torvalds * 134*1da177e4SLinus Torvalds * 135*1da177e4SLinus Torvalds * mraid_setup_device_map() can be called anytime after the device map is 136*1da177e4SLinus Torvalds * available and MRAID_GET_DEVICE_MAP() can be called whenever the mapping is 137*1da177e4SLinus Torvalds * required, usually from LLD's queue entry point. The formar API sets up the 138*1da177e4SLinus Torvalds * MRAID_IS_LOGICAL(adapter_t *, struct scsi_cmnd *) to find out if the 139*1da177e4SLinus Torvalds * device in question is a logical drive. 140*1da177e4SLinus Torvalds * 141*1da177e4SLinus Torvalds * quiescent flag should be set by the driver if it is not accepting more 142*1da177e4SLinus Torvalds * commands 143*1da177e4SLinus Torvalds * 144*1da177e4SLinus Torvalds * NOTE: The fields of this structures are placed to minimize cache misses 145*1da177e4SLinus Torvalds */ 146*1da177e4SLinus Torvalds 147*1da177e4SLinus Torvalds // amount of space required to store the bios and firmware version strings 148*1da177e4SLinus Torvalds #define VERSION_SIZE 16 149*1da177e4SLinus Torvalds 150*1da177e4SLinus Torvalds typedef struct { 151*1da177e4SLinus Torvalds struct tasklet_struct dpc_h; 152*1da177e4SLinus Torvalds struct pci_dev *pdev; 153*1da177e4SLinus Torvalds struct Scsi_Host *host; 154*1da177e4SLinus Torvalds spinlock_t *host_lock; 155*1da177e4SLinus Torvalds spinlock_t lock; 156*1da177e4SLinus Torvalds uint8_t quiescent; 157*1da177e4SLinus Torvalds int outstanding_cmds; 158*1da177e4SLinus Torvalds scb_t *kscb_list; 159*1da177e4SLinus Torvalds struct list_head kscb_pool; 160*1da177e4SLinus Torvalds spinlock_t kscb_pool_lock; 161*1da177e4SLinus Torvalds struct list_head pend_list; 162*1da177e4SLinus Torvalds spinlock_t pend_list_lock; 163*1da177e4SLinus Torvalds struct list_head completed_list; 164*1da177e4SLinus Torvalds spinlock_t completed_list_lock; 165*1da177e4SLinus Torvalds uint16_t sglen; 166*1da177e4SLinus Torvalds int device_ids[LSI_MAX_CHANNELS] 167*1da177e4SLinus Torvalds [LSI_MAX_LOGICAL_DRIVES_64LD]; 168*1da177e4SLinus Torvalds caddr_t raid_device; 169*1da177e4SLinus Torvalds uint8_t max_channel; 170*1da177e4SLinus Torvalds uint16_t max_target; 171*1da177e4SLinus Torvalds uint8_t max_lun; 172*1da177e4SLinus Torvalds 173*1da177e4SLinus Torvalds uint32_t unique_id; 174*1da177e4SLinus Torvalds uint8_t irq; 175*1da177e4SLinus Torvalds uint8_t ito; 176*1da177e4SLinus Torvalds caddr_t ibuf; 177*1da177e4SLinus Torvalds dma_addr_t ibuf_dma_h; 178*1da177e4SLinus Torvalds scb_t *uscb_list; 179*1da177e4SLinus Torvalds struct list_head uscb_pool; 180*1da177e4SLinus Torvalds spinlock_t uscb_pool_lock; 181*1da177e4SLinus Torvalds int max_cmds; 182*1da177e4SLinus Torvalds uint8_t fw_version[VERSION_SIZE]; 183*1da177e4SLinus Torvalds uint8_t bios_version[VERSION_SIZE]; 184*1da177e4SLinus Torvalds uint8_t max_cdb_sz; 185*1da177e4SLinus Torvalds uint8_t ha; 186*1da177e4SLinus Torvalds uint16_t init_id; 187*1da177e4SLinus Torvalds uint16_t max_sectors; 188*1da177e4SLinus Torvalds uint16_t cmd_per_lun; 189*1da177e4SLinus Torvalds atomic_t being_detached; 190*1da177e4SLinus Torvalds } adapter_t; 191*1da177e4SLinus Torvalds 192*1da177e4SLinus Torvalds #define SCSI_FREE_LIST_LOCK(adapter) (&adapter->kscb_pool_lock) 193*1da177e4SLinus Torvalds #define USER_FREE_LIST_LOCK(adapter) (&adapter->uscb_pool_lock) 194*1da177e4SLinus Torvalds #define PENDING_LIST_LOCK(adapter) (&adapter->pend_list_lock) 195*1da177e4SLinus Torvalds #define COMPLETED_LIST_LOCK(adapter) (&adapter->completed_list_lock) 196*1da177e4SLinus Torvalds 197*1da177e4SLinus Torvalds 198*1da177e4SLinus Torvalds // conversion from scsi command 199*1da177e4SLinus Torvalds #define SCP2HOST(scp) (scp)->device->host // to host 200*1da177e4SLinus Torvalds #define SCP2HOSTDATA(scp) SCP2HOST(scp)->hostdata // to soft state 201*1da177e4SLinus Torvalds #define SCP2CHANNEL(scp) (scp)->device->channel // to channel 202*1da177e4SLinus Torvalds #define SCP2TARGET(scp) (scp)->device->id // to target 203*1da177e4SLinus Torvalds #define SCP2LUN(scp) (scp)->device->lun // to LUN 204*1da177e4SLinus Torvalds 205*1da177e4SLinus Torvalds // generic macro to convert scsi command and host to controller's soft state 206*1da177e4SLinus Torvalds #define SCSIHOST2ADAP(host) (((caddr_t *)(host->hostdata))[0]) 207*1da177e4SLinus Torvalds #define SCP2ADAPTER(scp) (adapter_t *)SCSIHOST2ADAP(SCP2HOST(scp)) 208*1da177e4SLinus Torvalds 209*1da177e4SLinus Torvalds 210*1da177e4SLinus Torvalds /** 211*1da177e4SLinus Torvalds * MRAID_GET_DEVICE_MAP - device ids 212*1da177e4SLinus Torvalds * @param adp - Adapter's soft state 213*1da177e4SLinus Torvalds * @param scp - mid-layer scsi command pointer 214*1da177e4SLinus Torvalds * @param p_chan - physical channel on the controller 215*1da177e4SLinus Torvalds * @param target - target id of the device or logical drive number 216*1da177e4SLinus Torvalds * @param islogical - set if the command is for the logical drive 217*1da177e4SLinus Torvalds * 218*1da177e4SLinus Torvalds * Macro to retrieve information about device class, logical or physical and 219*1da177e4SLinus Torvalds * the corresponding physical channel and target or logical drive number 220*1da177e4SLinus Torvalds **/ 221*1da177e4SLinus Torvalds #define MRAID_IS_LOGICAL(adp, scp) \ 222*1da177e4SLinus Torvalds (SCP2CHANNEL(scp) == (adp)->max_channel) ? 1 : 0 223*1da177e4SLinus Torvalds 224*1da177e4SLinus Torvalds #define MRAID_IS_LOGICAL_SDEV(adp, sdev) \ 225*1da177e4SLinus Torvalds (sdev->channel == (adp)->max_channel) ? 1 : 0 226*1da177e4SLinus Torvalds 227*1da177e4SLinus Torvalds #define MRAID_GET_DEVICE_MAP(adp, scp, p_chan, target, islogical) \ 228*1da177e4SLinus Torvalds /* \ 229*1da177e4SLinus Torvalds * Is the request coming for the virtual channel \ 230*1da177e4SLinus Torvalds */ \ 231*1da177e4SLinus Torvalds islogical = MRAID_IS_LOGICAL(adp, scp); \ 232*1da177e4SLinus Torvalds \ 233*1da177e4SLinus Torvalds /* \ 234*1da177e4SLinus Torvalds * Get an index into our table of drive ids mapping \ 235*1da177e4SLinus Torvalds */ \ 236*1da177e4SLinus Torvalds if (islogical) { \ 237*1da177e4SLinus Torvalds p_chan = 0xFF; \ 238*1da177e4SLinus Torvalds target = \ 239*1da177e4SLinus Torvalds (adp)->device_ids[(adp)->max_channel][SCP2TARGET(scp)]; \ 240*1da177e4SLinus Torvalds } \ 241*1da177e4SLinus Torvalds else { \ 242*1da177e4SLinus Torvalds p_chan = ((adp)->device_ids[SCP2CHANNEL(scp)] \ 243*1da177e4SLinus Torvalds [SCP2TARGET(scp)] >> 8) & 0xFF; \ 244*1da177e4SLinus Torvalds target = ((adp)->device_ids[SCP2CHANNEL(scp)] \ 245*1da177e4SLinus Torvalds [SCP2TARGET(scp)] & 0xFF); \ 246*1da177e4SLinus Torvalds } 247*1da177e4SLinus Torvalds 248*1da177e4SLinus Torvalds /* 249*1da177e4SLinus Torvalds * ### Helper routines ### 250*1da177e4SLinus Torvalds */ 251*1da177e4SLinus Torvalds #define LSI_DBGLVL mraid_debug_level // each LLD must define a global 252*1da177e4SLinus Torvalds // mraid_debug_level 253*1da177e4SLinus Torvalds 254*1da177e4SLinus Torvalds #ifdef DEBUG 255*1da177e4SLinus Torvalds #if defined (_ASSERT_PANIC) 256*1da177e4SLinus Torvalds #define ASSERT_ACTION panic 257*1da177e4SLinus Torvalds #else 258*1da177e4SLinus Torvalds #define ASSERT_ACTION printk 259*1da177e4SLinus Torvalds #endif 260*1da177e4SLinus Torvalds 261*1da177e4SLinus Torvalds #define ASSERT(expression) \ 262*1da177e4SLinus Torvalds if (!(expression)) { \ 263*1da177e4SLinus Torvalds ASSERT_ACTION("assertion failed:(%s), file: %s, line: %d:%s\n", \ 264*1da177e4SLinus Torvalds #expression, __FILE__, __LINE__, __FUNCTION__); \ 265*1da177e4SLinus Torvalds } 266*1da177e4SLinus Torvalds #else 267*1da177e4SLinus Torvalds #define ASSERT(expression) 268*1da177e4SLinus Torvalds #endif 269*1da177e4SLinus Torvalds 270*1da177e4SLinus Torvalds /* 271*1da177e4SLinus Torvalds * struct mraid_pci_blk - structure holds DMA memory block info 272*1da177e4SLinus Torvalds * @param vaddr : virtual address to a memory block 273*1da177e4SLinus Torvalds * @param dma_addr : DMA handle to a memory block 274*1da177e4SLinus Torvalds * 275*1da177e4SLinus Torvalds * This structure is filled up for the caller. It is the responsibilty of the 276*1da177e4SLinus Torvalds * caller to allocate this array big enough to store addresses for all 277*1da177e4SLinus Torvalds * requested elements 278*1da177e4SLinus Torvalds */ 279*1da177e4SLinus Torvalds struct mraid_pci_blk { 280*1da177e4SLinus Torvalds caddr_t vaddr; 281*1da177e4SLinus Torvalds dma_addr_t dma_addr; 282*1da177e4SLinus Torvalds }; 283*1da177e4SLinus Torvalds 284*1da177e4SLinus Torvalds #endif // _MEGA_COMMON_H_ 285*1da177e4SLinus Torvalds 286*1da177e4SLinus Torvalds // vim: set ts=8 sw=8 tw=78: 287