| /* |
| * QLOGIC LINUX SOFTWARE |
| * |
| * QLogic ISP2x00 device driver for Linux 2.6.x |
| * Copyright (C) 2003-2005 QLogic Corporation |
| * (www.qlogic.com) |
| * |
| * This program is free software; you can redistribute it and/or modify it |
| * under the terms of the GNU General Public License as published by the |
| * Free Software Foundation; either version 2, or (at your option) any |
| * later version. |
| * |
| * This program is distributed in the hope that it will be useful, but |
| * WITHOUT ANY WARRANTY; without even the implied warranty of |
| * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU |
| * General Public License for more details. |
| * |
| */ |
| #include "qla_def.h" |
| |
| /** |
| * IO descriptor handle definitions. |
| * |
| * Signature form: |
| * |
| * |31------28|27-------------------12|11-------0| |
| * | Type | Rolling Signature | Index | |
| * |----------|-----------------------|----------| |
| * |
| **/ |
| |
| #define HDL_TYPE_SCSI 0 |
| #define HDL_TYPE_ASYNC_IOCB 0x0A |
| |
| #define HDL_INDEX_BITS 12 |
| #define HDL_ITER_BITS 16 |
| #define HDL_TYPE_BITS 4 |
| |
| #define HDL_INDEX_MASK ((1UL << HDL_INDEX_BITS) - 1) |
| #define HDL_ITER_MASK ((1UL << HDL_ITER_BITS) - 1) |
| #define HDL_TYPE_MASK ((1UL << HDL_TYPE_BITS) - 1) |
| |
| #define HDL_INDEX_SHIFT 0 |
| #define HDL_ITER_SHIFT (HDL_INDEX_SHIFT + HDL_INDEX_BITS) |
| #define HDL_TYPE_SHIFT (HDL_ITER_SHIFT + HDL_ITER_BITS) |
| |
| /* Local Prototypes. */ |
| static inline uint32_t qla2x00_to_handle(uint16_t, uint16_t, uint16_t); |
| static inline uint16_t qla2x00_handle_to_idx(uint32_t); |
| static inline uint32_t qla2x00_iodesc_to_handle(struct io_descriptor *); |
| static inline struct io_descriptor *qla2x00_handle_to_iodesc(scsi_qla_host_t *, |
| uint32_t); |
| |
| static inline struct io_descriptor *qla2x00_alloc_iodesc(scsi_qla_host_t *); |
| static inline void qla2x00_free_iodesc(struct io_descriptor *); |
| static inline void qla2x00_init_io_descriptors(scsi_qla_host_t *); |
| |
| static void qla2x00_iodesc_timeout(unsigned long); |
| static inline void qla2x00_add_iodesc_timer(struct io_descriptor *); |
| static inline void qla2x00_remove_iodesc_timer(struct io_descriptor *); |
| |
| static inline void qla2x00_update_login_fcport(scsi_qla_host_t *, |
| struct mbx_entry *, fc_port_t *); |
| |
| static int qla2x00_send_abort_iocb(scsi_qla_host_t *, struct io_descriptor *, |
| uint32_t, int); |
| static int qla2x00_send_abort_iocb_cb(scsi_qla_host_t *, struct io_descriptor *, |
| struct mbx_entry *); |
| |
| static int qla2x00_send_adisc_iocb(scsi_qla_host_t *, struct io_descriptor *, |
| int); |
| static int qla2x00_send_adisc_iocb_cb(scsi_qla_host_t *, struct io_descriptor *, |
| struct mbx_entry *); |
| |
| static int qla2x00_send_logout_iocb(scsi_qla_host_t *, struct io_descriptor *, |
| int); |
| static int qla2x00_send_logout_iocb_cb(scsi_qla_host_t *, |
| struct io_descriptor *, struct mbx_entry *); |
| |
| static int qla2x00_send_login_iocb(scsi_qla_host_t *, struct io_descriptor *, |
| port_id_t *, int); |
| static int qla2x00_send_login_iocb_cb(scsi_qla_host_t *, struct io_descriptor *, |
| struct mbx_entry *); |
| |
| /** |
| * Mailbox IOCB callback array. |
| **/ |
| static int (*iocb_function_cb_list[LAST_IOCB_CB]) |
| (scsi_qla_host_t *, struct io_descriptor *, struct mbx_entry *) = { |
| |
| qla2x00_send_abort_iocb_cb, |
| qla2x00_send_adisc_iocb_cb, |
| qla2x00_send_logout_iocb_cb, |
| qla2x00_send_login_iocb_cb, |
| }; |
| |
| |
| /** |
| * Generic IO descriptor handle routines. |
| **/ |
| |
| /** |
| * qla2x00_to_handle() - Create a descriptor handle. |
| * @type: descriptor type |
| * @iter: descriptor rolling signature |
| * @idx: index to the descriptor array |
| * |
| * Returns a composite handle based in the @type, @iter, and @idx. |
| */ |
| static inline uint32_t |
| qla2x00_to_handle(uint16_t type, uint16_t iter, uint16_t idx) |
| { |
| return ((uint32_t)(((uint32_t)type << HDL_TYPE_SHIFT) | |
| ((uint32_t)iter << HDL_ITER_SHIFT) | |
| ((uint32_t)idx << HDL_INDEX_SHIFT))); |
| } |
| |
| /** |
| * qla2x00_handle_to_idx() - Retrive the index for a given handle. |
| * @handle: descriptor handle |
| * |
| * Returns the index specified by the @handle. |
| */ |
| static inline uint16_t |
| qla2x00_handle_to_idx(uint32_t handle) |
| { |
| return ((uint16_t)(((handle) >> HDL_INDEX_SHIFT) & HDL_INDEX_MASK)); |
| } |
| |
| /** |
| * qla2x00_iodesc_to_handle() - Convert an IO descriptor to a unique handle. |
| * @iodesc: io descriptor |
| * |
| * Returns a unique handle for @iodesc. |
| */ |
| static inline uint32_t |
| qla2x00_iodesc_to_handle(struct io_descriptor *iodesc) |
| { |
| uint32_t handle; |
| |
| handle = qla2x00_to_handle(HDL_TYPE_ASYNC_IOCB, |
| ++iodesc->ha->iodesc_signature, iodesc->idx); |
| iodesc->signature = handle; |
| |
| return (handle); |
| } |
| |
| /** |
| * qla2x00_handle_to_iodesc() - Retrieve an IO descriptor given a unique handle. |
| * @ha: HA context |
| * @handle: handle to io descriptor |
| * |
| * Returns a pointer to the io descriptor, or NULL, if the io descriptor does |
| * not exist or the io descriptors signature does not @handle. |
| */ |
| static inline struct io_descriptor * |
| qla2x00_handle_to_iodesc(scsi_qla_host_t *ha, uint32_t handle) |
| { |
| uint16_t idx; |
| struct io_descriptor *iodesc; |
| |
| idx = qla2x00_handle_to_idx(handle); |
| iodesc = &ha->io_descriptors[idx]; |
| if (iodesc) |
| if (iodesc->signature != handle) |
| iodesc = NULL; |
| |
| return (iodesc); |
| } |
| |
| |
| /** |
| * IO descriptor allocation routines. |
| **/ |
| |
| /** |
| * qla2x00_alloc_iodesc() - Allocate an IO descriptor from the pool. |
| * @ha: HA context |
| * |
| * Returns a pointer to the allocated io descriptor, or NULL, if none available. |
| */ |
| static inline struct io_descriptor * |
| qla2x00_alloc_iodesc(scsi_qla_host_t *ha) |
| { |
| uint16_t iter; |
| struct io_descriptor *iodesc; |
| |
| iodesc = NULL; |
| for (iter = 0; iter < MAX_IO_DESCRIPTORS; iter++) { |
| if (ha->io_descriptors[iter].used) |
| continue; |
| |
| iodesc = &ha->io_descriptors[iter]; |
| iodesc->used = 1; |
| iodesc->idx = iter; |
| init_timer(&iodesc->timer); |
| iodesc->ha = ha; |
| iodesc->signature = qla2x00_iodesc_to_handle(iodesc); |
| break; |
| } |
| |
| return (iodesc); |
| } |
| |
| /** |
| * qla2x00_free_iodesc() - Free an IO descriptor. |
| * @iodesc: io descriptor |
| * |
| * NOTE: The io descriptors timer *must* be stopped before it can be free'd. |
| */ |
| static inline void |
| qla2x00_free_iodesc(struct io_descriptor *iodesc) |
| { |
| iodesc->used = 0; |
| iodesc->signature = 0; |
| } |
| |
| /** |
| * qla2x00_remove_iodesc_timer() - Remove an active timer from an IO descriptor. |
| * @iodesc: io descriptor |
| */ |
| static inline void |
| qla2x00_remove_iodesc_timer(struct io_descriptor *iodesc) |
| { |
| if (iodesc->timer.function != NULL) { |
| del_timer_sync(&iodesc->timer); |
| iodesc->timer.data = (unsigned long) NULL; |
| iodesc->timer.function = NULL; |
| } |
| } |
| |
| /** |
| * qla2x00_init_io_descriptors() - Initialize the pool of IO descriptors. |
| * @ha: HA context |
| */ |
| static inline void |
| qla2x00_init_io_descriptors(scsi_qla_host_t *ha) |
| { |
| uint16_t iter; |
| |
| for (iter = 0; iter < MAX_IO_DESCRIPTORS; iter++) { |
| if (!ha->io_descriptors[iter].used) |
| continue; |
| |
| qla2x00_remove_iodesc_timer(&ha->io_descriptors[iter]); |
| qla2x00_free_iodesc(&ha->io_descriptors[iter]); |
| } |
| } |
| |
| |
| /** |
| * IO descriptor timer routines. |
| **/ |
| |
| /** |
| * qla2x00_iodesc_timeout() - Timeout IO descriptor handler. |
| * @data: io descriptor |
| */ |
| static void |
| qla2x00_iodesc_timeout(unsigned long data) |
| { |
| struct io_descriptor *iodesc; |
| |
| iodesc = (struct io_descriptor *) data; |
| |
| DEBUG14(printk("scsi(%ld): IO descriptor timeout, index=%x " |
| "signature=%08x, scheduling ISP abort.\n", iodesc->ha->host_no, |
| iodesc->idx, iodesc->signature)); |
| |
| qla2x00_free_iodesc(iodesc); |
| |
| qla_printk(KERN_WARNING, iodesc->ha, |
| "IO descriptor timeout. Scheduling ISP abort.\n"); |
| set_bit(ISP_ABORT_NEEDED, &iodesc->ha->dpc_flags); |
| } |
| |
| /** |
| * qla2x00_add_iodesc_timer() - Add and start a timer for an IO descriptor. |
| * @iodesc: io descriptor |
| * |
| * NOTE: |
| * The firmware shall timeout an outstanding mailbox IOCB in 2 * R_A_TOV (in |
| * tenths of a second) after it hits the wire. But, if there are any request |
| * resource contraints (i.e. during heavy I/O), exchanges can be held off for |
| * at most R_A_TOV. Therefore, the driver will wait 4 * R_A_TOV before |
| * scheduling a recovery (big hammer). |
| */ |
| static inline void |
| qla2x00_add_iodesc_timer(struct io_descriptor *iodesc) |
| { |
| unsigned long timeout; |
| |
| timeout = (iodesc->ha->r_a_tov * 4) / 10; |
| init_timer(&iodesc->timer); |
| iodesc->timer.data = (unsigned long) iodesc; |
| iodesc->timer.expires = jiffies + (timeout * HZ); |
| iodesc->timer.function = |
| (void (*) (unsigned long)) qla2x00_iodesc_timeout; |
| add_timer(&iodesc->timer); |
| } |
| |
| /** |
| * IO descriptor support routines. |
| **/ |
| |
| /** |
| * qla2x00_update_login_fcport() - Update fcport data after login processing. |
| * @ha: HA context |
| * @mbxstat: Mailbox command status IOCB |
| * @fcport: port to update |
| */ |
| static inline void |
| qla2x00_update_login_fcport(scsi_qla_host_t *ha, struct mbx_entry *mbxstat, |
| fc_port_t *fcport) |
| { |
| if (le16_to_cpu(mbxstat->mb1) & BIT_0) { |
| fcport->port_type = FCT_INITIATOR; |
| } else { |
| fcport->port_type = FCT_TARGET; |
| if (le16_to_cpu(mbxstat->mb1) & BIT_1) { |
| fcport->flags |= FCF_TAPE_PRESENT; |
| } |
| } |
| fcport->login_retry = 0; |
| fcport->port_login_retry_count = ha->port_down_retry_count * |
| PORT_RETRY_TIME; |
| atomic_set(&fcport->port_down_timer, ha->port_down_retry_count * |
| PORT_RETRY_TIME); |
| fcport->flags |= FCF_FABRIC_DEVICE; |
| fcport->flags &= ~FCF_FAILOVER_NEEDED; |
| fcport->iodesc_idx_sent = IODESC_INVALID_INDEX; |
| atomic_set(&fcport->state, FCS_ONLINE); |
| if (fcport->rport) |
| fc_remote_port_unblock(fcport->rport); |
| } |
| |
| |
| /** |
| * Mailbox IOCB commands. |
| **/ |
| |
| /** |
| * qla2x00_get_mbx_iocb_entry() - Retrieve an IOCB from the request queue. |
| * @ha: HA context |
| * @handle: handle to io descriptor |
| * |
| * Returns a pointer to the reqest entry, or NULL, if none were available. |
| */ |
| static inline struct mbx_entry * |
| qla2x00_get_mbx_iocb_entry(scsi_qla_host_t *ha, uint32_t handle) |
| { |
| uint16_t cnt; |
| struct device_reg_2xxx __iomem *reg = &ha->iobase->isp; |
| struct mbx_entry *mbxentry; |
| |
| mbxentry = NULL; |
| |
| if (ha->req_q_cnt < 3) { |
| cnt = qla2x00_debounce_register(ISP_REQ_Q_OUT(ha, reg)); |
| if (ha->req_ring_index < cnt) |
| ha->req_q_cnt = cnt - ha->req_ring_index; |
| else |
| ha->req_q_cnt = ha->request_q_length - |
| (ha->req_ring_index - cnt); |
| } |
| if (ha->req_q_cnt >= 3) { |
| mbxentry = (struct mbx_entry *)ha->request_ring_ptr; |
| |
| memset(mbxentry, 0, sizeof(struct mbx_entry)); |
| mbxentry->entry_type = MBX_IOCB_TYPE; |
| mbxentry->entry_count = 1; |
| mbxentry->sys_define1 = SOURCE_ASYNC_IOCB; |
| mbxentry->handle = handle; |
| } |
| return (mbxentry); |
| } |
| |
| /** |
| * qla2x00_send_abort_iocb() - Issue an abort IOCB to the firmware. |
| * @ha: HA context |
| * @iodesc: io descriptor |
| * @handle_to_abort: firmware handle to abort |
| * @ha_locked: is function called with the hardware lock |
| * |
| * Returns QLA_SUCCESS if the IOCB was issued. |
| */ |
| static int |
| qla2x00_send_abort_iocb(scsi_qla_host_t *ha, struct io_descriptor *iodesc, |
| uint32_t handle_to_abort, int ha_locked) |
| { |
| unsigned long flags = 0; |
| struct mbx_entry *mbxentry; |
| |
| /* Send marker if required. */ |
| if (qla2x00_issue_marker(ha, ha_locked) != QLA_SUCCESS) |
| return (QLA_FUNCTION_FAILED); |
| |
| if (!ha_locked) |
| spin_lock_irqsave(&ha->hardware_lock, flags); |
| |
| /* Build abort mailbox IOCB. */ |
| mbxentry = qla2x00_get_mbx_iocb_entry(ha, iodesc->signature); |
| if (mbxentry == NULL) { |
| if (!ha_locked) |
| spin_unlock_irqrestore(&ha->hardware_lock, flags); |
| |
| return (QLA_FUNCTION_FAILED); |
| } |
| mbxentry->mb0 = __constant_cpu_to_le16(MBC_ABORT_COMMAND); |
| mbxentry->mb1 = mbxentry->loop_id.extended = |
| cpu_to_le16(iodesc->remote_fcport->loop_id); |
| mbxentry->mb2 = LSW(handle_to_abort); |
| mbxentry->mb3 = MSW(handle_to_abort); |
| wmb(); |
| |
| qla2x00_add_iodesc_timer(iodesc); |
| |
| /* Issue command to ISP. */ |
| qla2x00_isp_cmd(ha); |
| |
| if (!ha_locked) |
| spin_unlock_irqrestore(&ha->hardware_lock, flags); |
| |
| DEBUG14(printk("scsi(%ld): Sending Abort IOCB (%08x) to [%x], aborting " |
| "%08x.\n", ha->host_no, iodesc->signature, |
| iodesc->remote_fcport->loop_id, handle_to_abort)); |
| |
| return (QLA_SUCCESS); |
| } |
| |
| /** |
| * qla2x00_send_abort_iocb_cb() - Abort IOCB callback. |
| * @ha: HA context |
| * @iodesc: io descriptor |
| * @mbxstat: mailbox status IOCB |
| * |
| * Returns QLA_SUCCESS if @iodesc can be freed by the caller, else, @iodesc |
| * will be used for a retry. |
| */ |
| static int |
| qla2x00_send_abort_iocb_cb(scsi_qla_host_t *ha, struct io_descriptor *iodesc, |
| struct mbx_entry *mbxstat) |
| { |
| DEBUG14(printk("scsi(%ld): Abort IOCB -- sent to [%x/%02x%02x%02x], " |
| "status=%x mb0=%x.\n", ha->host_no, iodesc->remote_fcport->loop_id, |
| iodesc->d_id.b.domain, iodesc->d_id.b.area, iodesc->d_id.b.al_pa, |
| le16_to_cpu(mbxstat->status), le16_to_cpu(mbxstat->mb0))); |
| |
| return (QLA_SUCCESS); |
| } |
| |
| |
| /** |
| * qla2x00_send_adisc_iocb() - Issue a Get Port Database IOCB to the firmware. |
| * @ha: HA context |
| * @iodesc: io descriptor |
| * @ha_locked: is function called with the hardware lock |
| * |
| * Returns QLA_SUCCESS if the IOCB was issued. |
| */ |
| static int |
| qla2x00_send_adisc_iocb(scsi_qla_host_t *ha, struct io_descriptor *iodesc, |
| int ha_locked) |
| { |
| unsigned long flags = 0; |
| struct mbx_entry *mbxentry; |
| |
| /* Send marker if required. */ |
| if (qla2x00_issue_marker(ha, ha_locked) != QLA_SUCCESS) |
| return (QLA_FUNCTION_FAILED); |
| |
| if (!ha_locked) |
| spin_lock_irqsave(&ha->hardware_lock, flags); |
| |
| /* Build Get Port Database IOCB. */ |
| mbxentry = qla2x00_get_mbx_iocb_entry(ha, iodesc->signature); |
| if (mbxentry == NULL) { |
| if (!ha_locked) |
| spin_unlock_irqrestore(&ha->hardware_lock, flags); |
| |
| return (QLA_FUNCTION_FAILED); |
| } |
| mbxentry->mb0 = __constant_cpu_to_le16(MBC_GET_PORT_DATABASE); |
| mbxentry->mb1 = mbxentry->loop_id.extended = |
| cpu_to_le16(iodesc->remote_fcport->loop_id); |
| mbxentry->mb2 = cpu_to_le16(MSW(LSD(ha->iodesc_pd_dma))); |
| mbxentry->mb3 = cpu_to_le16(LSW(LSD(ha->iodesc_pd_dma))); |
| mbxentry->mb6 = cpu_to_le16(MSW(MSD(ha->iodesc_pd_dma))); |
| mbxentry->mb7 = cpu_to_le16(LSW(MSD(ha->iodesc_pd_dma))); |
| mbxentry->mb10 = __constant_cpu_to_le16(BIT_0); |
| wmb(); |
| |
| qla2x00_add_iodesc_timer(iodesc); |
| |
| /* Issue command to ISP. */ |
| qla2x00_isp_cmd(ha); |
| |
| if (!ha_locked) |
| spin_unlock_irqrestore(&ha->hardware_lock, flags); |
| |
| DEBUG14(printk("scsi(%ld): Sending Adisc IOCB (%08x) to [%x].\n", |
| ha->host_no, iodesc->signature, iodesc->remote_fcport->loop_id)); |
| |
| return (QLA_SUCCESS); |
| } |
| |
| /** |
| * qla2x00_send_adisc_iocb_cb() - Get Port Database IOCB callback. |
| * @ha: HA context |
| * @iodesc: io descriptor |
| * @mbxstat: mailbox status IOCB |
| * |
| * Returns QLA_SUCCESS if @iodesc can be freed by the caller, else, @iodesc |
| * will be used for a retry. |
| */ |
| static int |
| qla2x00_send_adisc_iocb_cb(scsi_qla_host_t *ha, struct io_descriptor *iodesc, |
| struct mbx_entry *mbxstat) |
| { |
| fc_port_t *remote_fcport; |
| |
| remote_fcport = iodesc->remote_fcport; |
| |
| /* Ensure the port IDs are consistent. */ |
| if (remote_fcport->d_id.b24 != iodesc->d_id.b24) { |
| DEBUG14(printk("scsi(%ld): Adisc IOCB -- ignoring, remote port " |
| "id changed from [%02x%02x%02x] to [%02x%02x%02x].\n", |
| ha->host_no, remote_fcport->d_id.b.domain, |
| remote_fcport->d_id.b.area, remote_fcport->d_id.b.al_pa, |
| iodesc->d_id.b.domain, iodesc->d_id.b.area, |
| iodesc->d_id.b.al_pa)); |
| |
| return (QLA_SUCCESS); |
| } |
| |
| /* Only process the last command. */ |
| if (remote_fcport->iodesc_idx_sent != iodesc->idx) { |
| DEBUG14(printk("scsi(%ld): Adisc IOCB -- ignoring, sent to " |
| "[%02x%02x%02x], expected %x, received %x.\n", ha->host_no, |
| iodesc->d_id.b.domain, iodesc->d_id.b.area, |
| iodesc->d_id.b.al_pa, remote_fcport->iodesc_idx_sent, |
| iodesc->idx)); |
| |
| return (QLA_SUCCESS); |
| } |
| |
| if (le16_to_cpu(mbxstat->status) == CS_COMPLETE) { |
| DEBUG14(printk("scsi(%ld): Adisc IOCB -- marking " |
| "[%x/%02x%02x%02x] online.\n", ha->host_no, |
| remote_fcport->loop_id, remote_fcport->d_id.b.domain, |
| remote_fcport->d_id.b.area, remote_fcport->d_id.b.al_pa)); |
| |
| atomic_set(&remote_fcport->state, FCS_ONLINE); |
| } else { |
| DEBUG14(printk("scsi(%ld): Adisc IOCB -- marking " |
| "[%x/%02x%02x%02x] lost, status=%x mb0=%x.\n", ha->host_no, |
| remote_fcport->loop_id, remote_fcport->d_id.b.domain, |
| remote_fcport->d_id.b.area, remote_fcport->d_id.b.al_pa, |
| le16_to_cpu(mbxstat->status), le16_to_cpu(mbxstat->mb0))); |
| |
| if (atomic_read(&remote_fcport->state) != FCS_DEVICE_DEAD) |
| atomic_set(&remote_fcport->state, FCS_DEVICE_LOST); |
| } |
| remote_fcport->iodesc_idx_sent = IODESC_INVALID_INDEX; |
| |
| return (QLA_SUCCESS); |
| } |
| |
| |
| /** |
| * qla2x00_send_logout_iocb() - Issue a fabric port logout IOCB to the firmware. |
| * @ha: HA context |
| * @iodesc: io descriptor |
| * @ha_locked: is function called with the hardware lock |
| * |
| * Returns QLA_SUCCESS if the IOCB was issued. |
| */ |
| static int |
| qla2x00_send_logout_iocb(scsi_qla_host_t *ha, struct io_descriptor *iodesc, |
| int ha_locked) |
| { |
| unsigned long flags = 0; |
| struct mbx_entry *mbxentry; |
| |
| /* Send marker if required. */ |
| if (qla2x00_issue_marker(ha, ha_locked) != QLA_SUCCESS) |
| return (QLA_FUNCTION_FAILED); |
| |
| if (!ha_locked) |
| spin_lock_irqsave(&ha->hardware_lock, flags); |
| |
| /* Build fabric port logout mailbox IOCB. */ |
| mbxentry = qla2x00_get_mbx_iocb_entry(ha, iodesc->signature); |
| if (mbxentry == NULL) { |
| if (!ha_locked) |
| spin_unlock_irqrestore(&ha->hardware_lock, flags); |
| |
| return (QLA_FUNCTION_FAILED); |
| } |
| mbxentry->mb0 = __constant_cpu_to_le16(MBC_LOGOUT_FABRIC_PORT); |
| mbxentry->mb1 = mbxentry->loop_id.extended = |
| cpu_to_le16(iodesc->remote_fcport->loop_id); |
| wmb(); |
| |
| qla2x00_add_iodesc_timer(iodesc); |
| |
| /* Issue command to ISP. */ |
| qla2x00_isp_cmd(ha); |
| |
| if (!ha_locked) |
| spin_unlock_irqrestore(&ha->hardware_lock, flags); |
| |
| DEBUG14(printk("scsi(%ld): Sending Logout IOCB (%08x) to [%x].\n", |
| ha->host_no, iodesc->signature, iodesc->remote_fcport->loop_id)); |
| |
| return (QLA_SUCCESS); |
| } |
| |
| /** |
| * qla2x00_send_logout_iocb_cb() - Fabric port logout IOCB callback. |
| * @ha: HA context |
| * @iodesc: io descriptor |
| * @mbxstat: mailbox status IOCB |
| * |
| * Returns QLA_SUCCESS if @iodesc can be freed by the caller, else, @iodesc |
| * will be used for a retry. |
| */ |
| static int |
| qla2x00_send_logout_iocb_cb(scsi_qla_host_t *ha, struct io_descriptor *iodesc, |
| struct mbx_entry *mbxstat) |
| { |
| DEBUG14(printk("scsi(%ld): Logout IOCB -- sent to [%x/%02x%02x%02x], " |
| "status=%x mb0=%x mb1=%x.\n", ha->host_no, |
| iodesc->remote_fcport->loop_id, |
| iodesc->remote_fcport->d_id.b.domain, |
| iodesc->remote_fcport->d_id.b.area, |
| iodesc->remote_fcport->d_id.b.al_pa, le16_to_cpu(mbxstat->status), |
| le16_to_cpu(mbxstat->mb0), le16_to_cpu(mbxstat->mb1))); |
| |
| return (QLA_SUCCESS); |
| } |
| |
| |
| /** |
| * qla2x00_send_login_iocb() - Issue a fabric port login IOCB to the firmware. |
| * @ha: HA context |
| * @iodesc: io descriptor |
| * @d_id: port id for device |
| * @ha_locked: is function called with the hardware lock |
| * |
| * Returns QLA_SUCCESS if the IOCB was issued. |
| */ |
| static int |
| qla2x00_send_login_iocb(scsi_qla_host_t *ha, struct io_descriptor *iodesc, |
| port_id_t *d_id, int ha_locked) |
| { |
| unsigned long flags = 0; |
| struct mbx_entry *mbxentry; |
| |
| /* Send marker if required. */ |
| if (qla2x00_issue_marker(ha, ha_locked) != QLA_SUCCESS) |
| return (QLA_FUNCTION_FAILED); |
| |
| if (!ha_locked) |
| spin_lock_irqsave(&ha->hardware_lock, flags); |
| |
| /* Build fabric port login mailbox IOCB. */ |
| mbxentry = qla2x00_get_mbx_iocb_entry(ha, iodesc->signature); |
| if (mbxentry == NULL) { |
| if (!ha_locked) |
| spin_unlock_irqrestore(&ha->hardware_lock, flags); |
| |
| return (QLA_FUNCTION_FAILED); |
| } |
| mbxentry->mb0 = __constant_cpu_to_le16(MBC_LOGIN_FABRIC_PORT); |
| mbxentry->mb1 = mbxentry->loop_id.extended = |
| cpu_to_le16(iodesc->remote_fcport->loop_id); |
| mbxentry->mb2 = cpu_to_le16(d_id->b.domain); |
| mbxentry->mb3 = cpu_to_le16(d_id->b.area << 8 | d_id->b.al_pa); |
| mbxentry->mb10 = __constant_cpu_to_le16(BIT_0); |
| wmb(); |
| |
| qla2x00_add_iodesc_timer(iodesc); |
| |
| /* Issue command to ISP. */ |
| qla2x00_isp_cmd(ha); |
| |
| if (!ha_locked) |
| spin_unlock_irqrestore(&ha->hardware_lock, flags); |
| |
| DEBUG14(printk("scsi(%ld): Sending Login IOCB (%08x) to " |
| "[%x/%02x%02x%02x].\n", ha->host_no, iodesc->signature, |
| iodesc->remote_fcport->loop_id, d_id->b.domain, d_id->b.area, |
| d_id->b.al_pa)); |
| |
| return (QLA_SUCCESS); |
| } |
| |
| /** |
| * qla2x00_send_login_iocb_cb() - Fabric port logout IOCB callback. |
| * @ha: HA context |
| * @iodesc: io descriptor |
| * @mbxstat: mailbox status IOCB |
| * |
| * Returns QLA_SUCCESS if @iodesc can be freed by the caller, else, @iodesc |
| * will be used for a retry. |
| */ |
| static int |
| qla2x00_send_login_iocb_cb(scsi_qla_host_t *ha, struct io_descriptor *iodesc, |
| struct mbx_entry *mbxstat) |
| { |
| int rval; |
| fc_port_t *fcport, *remote_fcport, *exist_fcport; |
| struct io_descriptor *abort_iodesc, *login_iodesc; |
| uint16_t status, mb[8]; |
| uint16_t reuse; |
| uint16_t remote_loopid; |
| port_id_t remote_did, inuse_did; |
| |
| remote_fcport = iodesc->remote_fcport; |
| |
| /* Only process the last command. */ |
| if (remote_fcport->iodesc_idx_sent != iodesc->idx) { |
| DEBUG14(printk("scsi(%ld): Login IOCB -- ignoring, sent to " |
| "[%02x%02x%02x], expected %x, received %x.\n", |
| ha->host_no, iodesc->d_id.b.domain, iodesc->d_id.b.area, |
| iodesc->d_id.b.al_pa, remote_fcport->iodesc_idx_sent, |
| iodesc->idx)); |
| |
| /* Free RSCN fcport resources. */ |
| if (remote_fcport->port_type == FCT_RSCN) { |
| DEBUG14(printk("scsi(%ld): Login IOCB -- Freeing RSCN " |
| "fcport %p [%x/%02x%02x%02x] given ignored Login " |
| "IOCB.\n", ha->host_no, remote_fcport, |
| remote_fcport->loop_id, |
| remote_fcport->d_id.b.domain, |
| remote_fcport->d_id.b.area, |
| remote_fcport->d_id.b.al_pa)); |
| |
| list_del(&remote_fcport->list); |
| kfree(remote_fcport); |
| } |
| return (QLA_SUCCESS); |
| } |
| |
| status = le16_to_cpu(mbxstat->status); |
| mb[0] = le16_to_cpu(mbxstat->mb0); |
| mb[1] = le16_to_cpu(mbxstat->mb1); |
| mb[2] = le16_to_cpu(mbxstat->mb2); |
| mb[6] = le16_to_cpu(mbxstat->mb6); |
| mb[7] = le16_to_cpu(mbxstat->mb7); |
| |
| /* Good status? */ |
| if ((status == CS_COMPLETE || status == CS_COMPLETE_CHKCOND) && |
| mb[0] == MBS_COMMAND_COMPLETE) { |
| |
| DEBUG14(printk("scsi(%ld): Login IOCB -- status=%x mb1=%x pn=" |
| "%02x%02x%02x%02x%02x%02x%02x%02x.\n", ha->host_no, status, |
| mb[1], mbxstat->port_name[0], mbxstat->port_name[1], |
| mbxstat->port_name[2], mbxstat->port_name[3], |
| mbxstat->port_name[4], mbxstat->port_name[5], |
| mbxstat->port_name[6], mbxstat->port_name[7])); |
| |
| memcpy(remote_fcport->node_name, mbxstat->node_name, WWN_SIZE); |
| memcpy(remote_fcport->port_name, mbxstat->port_name, WWN_SIZE); |
| |
| /* Is the device already in our fcports list? */ |
| if (remote_fcport->port_type != FCT_RSCN) { |
| DEBUG14(printk("scsi(%ld): Login IOCB -- marking " |
| "[%x/%02x%02x%02x] online.\n", ha->host_no, |
| remote_fcport->loop_id, |
| remote_fcport->d_id.b.domain, |
| remote_fcport->d_id.b.area, |
| remote_fcport->d_id.b.al_pa)); |
| |
| qla2x00_update_login_fcport(ha, mbxstat, remote_fcport); |
| |
| return (QLA_SUCCESS); |
| } |
| |
| /* Does the RSCN portname already exist in our fcports list? */ |
| exist_fcport = NULL; |
| list_for_each_entry(fcport, &ha->fcports, list) { |
| if (memcmp(remote_fcport->port_name, fcport->port_name, |
| WWN_SIZE) == 0) { |
| exist_fcport = fcport; |
| break; |
| } |
| } |
| if (exist_fcport != NULL) { |
| DEBUG14(printk("scsi(%ld): Login IOCB -- found RSCN " |
| "fcport in fcports list [%p].\n", ha->host_no, |
| exist_fcport)); |
| |
| /* Abort any ADISC that could have been sent. */ |
| if (exist_fcport->iodesc_idx_sent != iodesc->idx && |
| exist_fcport->iodesc_idx_sent < |
| MAX_IO_DESCRIPTORS && |
| ha->io_descriptors[exist_fcport->iodesc_idx_sent]. |
| cb_idx == ADISC_PORT_IOCB_CB) { |
| |
| abort_iodesc = qla2x00_alloc_iodesc(ha); |
| if (abort_iodesc) { |
| DEBUG14(printk("scsi(%ld): Login IOCB " |
| "-- issuing abort to outstanding " |
| "Adisc [%x/%02x%02x%02x].\n", |
| ha->host_no, remote_fcport->loop_id, |
| exist_fcport->d_id.b.domain, |
| exist_fcport->d_id.b.area, |
| exist_fcport->d_id.b.al_pa)); |
| |
| abort_iodesc->cb_idx = ABORT_IOCB_CB; |
| abort_iodesc->d_id.b24 = |
| exist_fcport->d_id.b24; |
| abort_iodesc->remote_fcport = |
| exist_fcport; |
| exist_fcport->iodesc_idx_sent = |
| abort_iodesc->idx; |
| qla2x00_send_abort_iocb(ha, |
| abort_iodesc, ha->io_descriptors[ |
| exist_fcport->iodesc_idx_sent]. |
| signature, 1); |
| } else { |
| DEBUG14(printk("scsi(%ld): Login IOCB " |
| "-- unable to abort outstanding " |
| "Adisc [%x/%02x%02x%02x].\n", |
| ha->host_no, remote_fcport->loop_id, |
| exist_fcport->d_id.b.domain, |
| exist_fcport->d_id.b.area, |
| exist_fcport->d_id.b.al_pa)); |
| } |
| } |
| |
| /* |
| * If the existing fcport is waiting to send an ADISC |
| * or LOGIN, then reuse remote fcport (RSCN) to |
| * continue waiting. |
| */ |
| reuse = 0; |
| remote_loopid = remote_fcport->loop_id; |
| remote_did.b24 = remote_fcport->d_id.b24; |
| if (exist_fcport->iodesc_idx_sent == |
| IODESC_ADISC_NEEDED || |
| exist_fcport->iodesc_idx_sent == |
| IODESC_LOGIN_NEEDED) { |
| DEBUG14(printk("scsi(%ld): Login IOCB -- " |
| "existing fcport [%x/%02x%02x%02x] " |
| "waiting for IO descriptor, reuse RSCN " |
| "fcport.\n", ha->host_no, |
| exist_fcport->loop_id, |
| exist_fcport->d_id.b.domain, |
| exist_fcport->d_id.b.area, |
| exist_fcport->d_id.b.al_pa)); |
| |
| reuse++; |
| remote_fcport->iodesc_idx_sent = |
| exist_fcport->iodesc_idx_sent; |
| exist_fcport->iodesc_idx_sent = |
| IODESC_INVALID_INDEX; |
| remote_fcport->loop_id = exist_fcport->loop_id; |
| remote_fcport->d_id.b24 = |
| exist_fcport->d_id.b24; |
| } |
| |
| /* Logout the old loopid. */ |
| if (!reuse && |
| exist_fcport->loop_id != remote_fcport->loop_id && |
| exist_fcport->loop_id != FC_NO_LOOP_ID) { |
| login_iodesc = qla2x00_alloc_iodesc(ha); |
| if (login_iodesc) { |
| DEBUG14(printk("scsi(%ld): Login IOCB " |
| "-- issuing logout to free old " |
| "loop id [%x/%02x%02x%02x].\n", |
| ha->host_no, exist_fcport->loop_id, |
| exist_fcport->d_id.b.domain, |
| exist_fcport->d_id.b.area, |
| exist_fcport->d_id.b.al_pa)); |
| |
| login_iodesc->cb_idx = |
| LOGOUT_PORT_IOCB_CB; |
| login_iodesc->d_id.b24 = |
| exist_fcport->d_id.b24; |
| login_iodesc->remote_fcport = |
| exist_fcport; |
| exist_fcport->iodesc_idx_sent = |
| login_iodesc->idx; |
| qla2x00_send_logout_iocb(ha, |
| login_iodesc, 1); |
| } else { |
| /* Ran out of IO descriptiors. */ |
| DEBUG14(printk("scsi(%ld): Login IOCB " |
| "-- unable to logout to free old " |
| "loop id [%x/%02x%02x%02x].\n", |
| ha->host_no, exist_fcport->loop_id, |
| exist_fcport->d_id.b.domain, |
| exist_fcport->d_id.b.area, |
| exist_fcport->d_id.b.al_pa)); |
| |
| exist_fcport->iodesc_idx_sent = |
| IODESC_INVALID_INDEX; |
| } |
| |
| } |
| |
| /* Update existing fcport with remote fcport info. */ |
| DEBUG14(printk("scsi(%ld): Login IOCB -- marking " |
| "existing fcport [%x/%02x%02x%02x] online.\n", |
| ha->host_no, remote_loopid, remote_did.b.domain, |
| remote_did.b.area, remote_did.b.al_pa)); |
| |
| memcpy(exist_fcport->node_name, |
| remote_fcport->node_name, WWN_SIZE); |
| exist_fcport->loop_id = remote_loopid; |
| exist_fcport->d_id.b24 = remote_did.b24; |
| qla2x00_update_login_fcport(ha, mbxstat, exist_fcport); |
| |
| /* Finally, free the remote (RSCN) fcport. */ |
| if (!reuse) { |
| DEBUG14(printk("scsi(%ld): Login IOCB -- " |
| "Freeing RSCN fcport %p " |
| "[%x/%02x%02x%02x].\n", ha->host_no, |
| remote_fcport, remote_fcport->loop_id, |
| remote_fcport->d_id.b.domain, |
| remote_fcport->d_id.b.area, |
| remote_fcport->d_id.b.al_pa)); |
| |
| list_del(&remote_fcport->list); |
| kfree(remote_fcport); |
| } |
| |
| return (QLA_SUCCESS); |
| } |
| |
| /* |
| * A new device has been added, move the RSCN fcport to our |
| * fcports list. |
| */ |
| DEBUG14(printk("scsi(%ld): Login IOCB -- adding RSCN fcport " |
| "[%x/%02x%02x%02x] to fcports list.\n", ha->host_no, |
| remote_fcport->loop_id, remote_fcport->d_id.b.domain, |
| remote_fcport->d_id.b.area, remote_fcport->d_id.b.al_pa)); |
| |
| list_del(&remote_fcport->list); |
| remote_fcport->flags = (FCF_RLC_SUPPORT | FCF_RESCAN_NEEDED); |
| qla2x00_update_login_fcport(ha, mbxstat, remote_fcport); |
| list_add_tail(&remote_fcport->list, &ha->fcports); |
| set_bit(FCPORT_RESCAN_NEEDED, &ha->dpc_flags); |
| } else { |
| /* Handle login failure. */ |
| if (remote_fcport->login_retry != 0) { |
| if (mb[0] == MBS_LOOP_ID_USED) { |
| inuse_did.b.domain = LSB(mb[1]); |
| inuse_did.b.area = MSB(mb[2]); |
| inuse_did.b.al_pa = LSB(mb[2]); |
| |
| DEBUG14(printk("scsi(%ld): Login IOCB -- loop " |
| "id [%x] used by port id [%02x%02x%02x].\n", |
| ha->host_no, remote_fcport->loop_id, |
| inuse_did.b.domain, inuse_did.b.area, |
| inuse_did.b.al_pa)); |
| |
| if (remote_fcport->d_id.b24 == |
| INVALID_PORT_ID) { |
| /* |
| * Invalid port id means we are trying |
| * to login to a remote port with just |
| * a loop id without knowing about the |
| * port id. Copy the port id and try |
| * again. |
| */ |
| remote_fcport->d_id.b24 = inuse_did.b24; |
| iodesc->d_id.b24 = inuse_did.b24; |
| } else { |
| remote_fcport->loop_id++; |
| rval = qla2x00_find_new_loop_id(ha, |
| remote_fcport); |
| if (rval == QLA_FUNCTION_FAILED) { |
| /* No more loop ids. */ |
| return (QLA_SUCCESS); |
| } |
| } |
| } else if (mb[0] == MBS_PORT_ID_USED) { |
| /* |
| * Device has another loop ID. The firmware |
| * group recommends the driver perform an |
| * implicit login with the specified ID. |
| */ |
| DEBUG14(printk("scsi(%ld): Login IOCB -- port " |
| "id [%02x%02x%02x] already assigned to " |
| "loop id [%x].\n", ha->host_no, |
| iodesc->d_id.b.domain, iodesc->d_id.b.area, |
| iodesc->d_id.b.al_pa, mb[1])); |
| |
| remote_fcport->loop_id = mb[1]; |
| |
| } else { |
| /* Unable to perform login, try again. */ |
| DEBUG14(printk("scsi(%ld): Login IOCB -- " |
| "failed login [%x/%02x%02x%02x], status=%x " |
| "mb0=%x mb1=%x mb2=%x mb6=%x mb7=%x.\n", |
| ha->host_no, remote_fcport->loop_id, |
| iodesc->d_id.b.domain, iodesc->d_id.b.area, |
| iodesc->d_id.b.al_pa, status, mb[0], mb[1], |
| mb[2], mb[6], mb[7])); |
| } |
| |
| /* Reissue Login with the same IO descriptor. */ |
| iodesc->signature = |
| qla2x00_iodesc_to_handle(iodesc); |
| iodesc->cb_idx = LOGIN_PORT_IOCB_CB; |
| iodesc->d_id.b24 = remote_fcport->d_id.b24; |
| remote_fcport->iodesc_idx_sent = iodesc->idx; |
| remote_fcport->login_retry--; |
| |
| DEBUG14(printk("scsi(%ld): Login IOCB -- retrying " |
| "login to [%x/%02x%02x%02x] (%d).\n", ha->host_no, |
| remote_fcport->loop_id, |
| remote_fcport->d_id.b.domain, |
| remote_fcport->d_id.b.area, |
| remote_fcport->d_id.b.al_pa, |
| remote_fcport->login_retry)); |
| |
| qla2x00_send_login_iocb(ha, iodesc, |
| &remote_fcport->d_id, 1); |
| |
| return (QLA_FUNCTION_FAILED); |
| } else { |
| /* No more logins, mark device dead. */ |
| DEBUG14(printk("scsi(%ld): Login IOCB -- failed " |
| "login [%x/%02x%02x%02x] after retries, status=%x " |
| "mb0=%x mb1=%x mb2=%x mb6=%x mb7=%x.\n", |
| ha->host_no, remote_fcport->loop_id, |
| iodesc->d_id.b.domain, iodesc->d_id.b.area, |
| iodesc->d_id.b.al_pa, status, mb[0], mb[1], |
| mb[2], mb[6], mb[7])); |
| |
| atomic_set(&remote_fcport->state, FCS_DEVICE_DEAD); |
| if (remote_fcport->port_type == FCT_RSCN) { |
| DEBUG14(printk("scsi(%ld): Login IOCB -- " |
| "Freeing dead RSCN fcport %p " |
| "[%x/%02x%02x%02x].\n", ha->host_no, |
| remote_fcport, remote_fcport->loop_id, |
| remote_fcport->d_id.b.domain, |
| remote_fcport->d_id.b.area, |
| remote_fcport->d_id.b.al_pa)); |
| |
| list_del(&remote_fcport->list); |
| kfree(remote_fcport); |
| } |
| } |
| } |
| |
| return (QLA_SUCCESS); |
| } |
| |
| |
| /** |
| * IO descriptor processing routines. |
| **/ |
| |
| /** |
| * qla2x00_alloc_rscn_fcport() - Allocate an RSCN type fcport. |
| * @ha: HA context |
| * @flags: allocation flags |
| * |
| * Returns a pointer to the allocated RSCN fcport, or NULL, if none available. |
| */ |
| fc_port_t * |
| qla2x00_alloc_rscn_fcport(scsi_qla_host_t *ha, gfp_t flags) |
| { |
| fc_port_t *fcport; |
| |
| fcport = qla2x00_alloc_fcport(ha, flags); |
| if (fcport == NULL) |
| return (fcport); |
| |
| /* Setup RSCN fcport structure. */ |
| fcport->port_type = FCT_RSCN; |
| |
| return (fcport); |
| } |
| |
| /** |
| * qla2x00_handle_port_rscn() - Handle port RSCN. |
| * @ha: HA context |
| * @rscn_entry: RSCN entry |
| * @fcport: fcport entry to updated |
| * |
| * Returns QLA_SUCCESS if the port RSCN was handled. |
| */ |
| int |
| qla2x00_handle_port_rscn(scsi_qla_host_t *ha, uint32_t rscn_entry, |
| fc_port_t *known_fcport, int ha_locked) |
| { |
| int rval; |
| port_id_t rscn_pid; |
| fc_port_t *fcport, *remote_fcport, *rscn_fcport; |
| struct io_descriptor *iodesc; |
| |
| remote_fcport = NULL; |
| rscn_fcport = NULL; |
| |
| /* Prepare port id based on incoming entries. */ |
| if (known_fcport) { |
| rscn_pid.b24 = known_fcport->d_id.b24; |
| remote_fcport = known_fcport; |
| |
| DEBUG14(printk("scsi(%ld): Handle RSCN -- process RSCN for " |
| "fcport [%02x%02x%02x].\n", ha->host_no, |
| remote_fcport->d_id.b.domain, remote_fcport->d_id.b.area, |
| remote_fcport->d_id.b.al_pa)); |
| } else { |
| rscn_pid.b.domain = LSB(MSW(rscn_entry)); |
| rscn_pid.b.area = MSB(LSW(rscn_entry)); |
| rscn_pid.b.al_pa = LSB(LSW(rscn_entry)); |
| |
| DEBUG14(printk("scsi(%ld): Handle RSCN -- process RSCN for " |
| "port id [%02x%02x%02x].\n", ha->host_no, |
| rscn_pid.b.domain, rscn_pid.b.area, rscn_pid.b.al_pa)); |
| |
| /* |
| * Search fcport lists for a known entry at the specified port |
| * ID. |
| */ |
| list_for_each_entry(fcport, &ha->fcports, list) { |
| if (rscn_pid.b24 == fcport->d_id.b24) { |
| remote_fcport = fcport; |
| break; |
| } |
| } |
| list_for_each_entry(fcport, &ha->rscn_fcports, list) { |
| if (rscn_pid.b24 == fcport->d_id.b24) { |
| rscn_fcport = fcport; |
| break; |
| } |
| } |
| if (remote_fcport == NULL) |
| remote_fcport = rscn_fcport; |
| } |
| |
| /* |
| * If the port is already in our fcport list and online, send an ADISC |
| * to see if it's still alive. Issue login if a new fcport or the known |
| * fcport is currently offline. |
| */ |
| if (remote_fcport) { |
| /* |
| * No need to send request if the remote fcport is currently |
| * waiting for an available io descriptor. |
| */ |
| if (known_fcport == NULL && |
| (remote_fcport->iodesc_idx_sent == IODESC_ADISC_NEEDED || |
| remote_fcport->iodesc_idx_sent == IODESC_LOGIN_NEEDED)) { |
| /* |
| * If previous waiting io descriptor is an ADISC, then |
| * the new RSCN may come from a new remote fcport being |
| * plugged into the same location. |
| */ |
| if (remote_fcport->port_type == FCT_RSCN) { |
| remote_fcport->iodesc_idx_sent = |
| IODESC_LOGIN_NEEDED; |
| } else if (remote_fcport->iodesc_idx_sent == |
| IODESC_ADISC_NEEDED) { |
| fc_port_t *new_fcport; |
| |
| remote_fcport->iodesc_idx_sent = |
| IODESC_INVALID_INDEX; |
| |
| /* Create new fcport for later login. */ |
| new_fcport = qla2x00_alloc_rscn_fcport(ha, |
| ha_locked ? GFP_ATOMIC: GFP_KERNEL); |
| if (new_fcport) { |
| DEBUG14(printk("scsi(%ld): Handle RSCN " |
| "-- creating RSCN fcport %p for " |
| "future login.\n", ha->host_no, |
| new_fcport)); |
| |
| new_fcport->d_id.b24 = |
| remote_fcport->d_id.b24; |
| new_fcport->iodesc_idx_sent = |
| IODESC_LOGIN_NEEDED; |
| |
| list_add_tail(&new_fcport->list, |
| &ha->rscn_fcports); |
| set_bit(IODESC_PROCESS_NEEDED, |
| &ha->dpc_flags); |
| } else { |
| DEBUG14(printk("scsi(%ld): Handle RSCN " |
| "-- unable to allocate RSCN fcport " |
| "for future login.\n", |
| ha->host_no)); |
| } |
| } |
| return (QLA_SUCCESS); |
| } |
| |
| /* Send ADISC if the fcport is online */ |
| if (atomic_read(&remote_fcport->state) == FCS_ONLINE || |
| remote_fcport->iodesc_idx_sent == IODESC_ADISC_NEEDED) { |
| |
| atomic_set(&remote_fcport->state, FCS_DEVICE_LOST); |
| |
| iodesc = qla2x00_alloc_iodesc(ha); |
| if (iodesc == NULL) { |
| /* Mark fcport for later adisc processing */ |
| DEBUG14(printk("scsi(%ld): Handle RSCN -- not " |
| "enough IO descriptors for Adisc, flag " |
| "for later processing.\n", ha->host_no)); |
| |
| remote_fcport->iodesc_idx_sent = |
| IODESC_ADISC_NEEDED; |
| set_bit(IODESC_PROCESS_NEEDED, &ha->dpc_flags); |
| |
| return (QLA_SUCCESS); |
| } |
| |
| iodesc->cb_idx = ADISC_PORT_IOCB_CB; |
| iodesc->d_id.b24 = rscn_pid.b24; |
| iodesc->remote_fcport = remote_fcport; |
| remote_fcport->iodesc_idx_sent = iodesc->idx; |
| qla2x00_send_adisc_iocb(ha, iodesc, ha_locked); |
| |
| return (QLA_SUCCESS); |
| } else if (remote_fcport->iodesc_idx_sent < |
| MAX_IO_DESCRIPTORS && |
| ha->io_descriptors[remote_fcport->iodesc_idx_sent].cb_idx == |
| ADISC_PORT_IOCB_CB) { |
| /* |
| * Receiving another RSCN while an ADISC is pending, |
| * abort the IOCB. Use the same descriptor for the |
| * abort. |
| */ |
| uint32_t handle_to_abort; |
| |
| iodesc = &ha->io_descriptors[ |
| remote_fcport->iodesc_idx_sent]; |
| qla2x00_remove_iodesc_timer(iodesc); |
| handle_to_abort = iodesc->signature; |
| iodesc->signature = qla2x00_iodesc_to_handle(iodesc); |
| iodesc->cb_idx = ABORT_IOCB_CB; |
| iodesc->d_id.b24 = remote_fcport->d_id.b24; |
| iodesc->remote_fcport = remote_fcport; |
| remote_fcport->iodesc_idx_sent = iodesc->idx; |
| |
| DEBUG14(printk("scsi(%ld): Handle RSCN -- issuing " |
| "abort to outstanding Adisc [%x/%02x%02x%02x].\n", |
| ha->host_no, remote_fcport->loop_id, |
| iodesc->d_id.b.domain, iodesc->d_id.b.area, |
| iodesc->d_id.b.al_pa)); |
| |
| qla2x00_send_abort_iocb(ha, iodesc, handle_to_abort, |
| ha_locked); |
| } |
| } |
| |
| /* We need to login to the remote port, find it. */ |
| if (known_fcport) { |
| remote_fcport = known_fcport; |
| } else if (rscn_fcport && rscn_fcport->d_id.b24 != INVALID_PORT_ID && |
| rscn_fcport->iodesc_idx_sent < MAX_IO_DESCRIPTORS && |
| ha->io_descriptors[rscn_fcport->iodesc_idx_sent].cb_idx == |
| LOGIN_PORT_IOCB_CB) { |
| /* |
| * Ignore duplicate RSCN on fcport which has already |
| * initiated a login IOCB. |
| */ |
| DEBUG14(printk("scsi(%ld): Handle RSCN -- ignoring, login " |
| "already sent to [%02x%02x%02x].\n", ha->host_no, |
| rscn_fcport->d_id.b.domain, rscn_fcport->d_id.b.area, |
| rscn_fcport->d_id.b.al_pa)); |
| |
| return (QLA_SUCCESS); |
| } else if (rscn_fcport && rscn_fcport->d_id.b24 != INVALID_PORT_ID && |
| rscn_fcport != remote_fcport) { |
| /* Reuse same rscn fcport. */ |
| DEBUG14(printk("scsi(%ld): Handle RSCN -- reusing RSCN fcport " |
| "[%02x%02x%02x].\n", ha->host_no, |
| rscn_fcport->d_id.b.domain, rscn_fcport->d_id.b.area, |
| rscn_fcport->d_id.b.al_pa)); |
| |
| remote_fcport = rscn_fcport; |
| } else { |
| /* Create new fcport for later login. */ |
| remote_fcport = qla2x00_alloc_rscn_fcport(ha, |
| ha_locked ? GFP_ATOMIC: GFP_KERNEL); |
| list_add_tail(&remote_fcport->list, &ha->rscn_fcports); |
| } |
| if (remote_fcport == NULL) |
| return (QLA_SUCCESS); |
| |
| /* Prepare fcport for login. */ |
| atomic_set(&remote_fcport->state, FCS_DEVICE_LOST); |
| remote_fcport->login_retry = 3; /* ha->login_retry_count; */ |
| remote_fcport->d_id.b24 = rscn_pid.b24; |
| |
| iodesc = qla2x00_alloc_iodesc(ha); |
| if (iodesc == NULL) { |
| /* Mark fcport for later adisc processing. */ |
| DEBUG14(printk("scsi(%ld): Handle RSCN -- not enough IO " |
| "descriptors for Login, flag for later processing.\n", |
| ha->host_no)); |
| |
| remote_fcport->iodesc_idx_sent = IODESC_LOGIN_NEEDED; |
| set_bit(IODESC_PROCESS_NEEDED, &ha->dpc_flags); |
| |
| return (QLA_SUCCESS); |
| } |
| |
| if (known_fcport == NULL || rscn_pid.b24 != INVALID_PORT_ID) { |
| remote_fcport->loop_id = ha->min_external_loopid; |
| |
| rval = qla2x00_find_new_loop_id(ha, remote_fcport); |
| if (rval == QLA_FUNCTION_FAILED) { |
| /* No more loop ids, failed. */ |
| DEBUG14(printk("scsi(%ld): Handle RSCN -- no available " |
| "loop id to perform Login, failed.\n", |
| ha->host_no)); |
| |
| return (rval); |
| } |
| } |
| |
| iodesc->cb_idx = LOGIN_PORT_IOCB_CB; |
| iodesc->d_id.b24 = rscn_pid.b24; |
| iodesc->remote_fcport = remote_fcport; |
| remote_fcport->iodesc_idx_sent = iodesc->idx; |
| |
| DEBUG14(printk("scsi(%ld): Handle RSCN -- attempting login to " |
| "[%x/%02x%02x%02x].\n", ha->host_no, remote_fcport->loop_id, |
| iodesc->d_id.b.domain, iodesc->d_id.b.area, iodesc->d_id.b.al_pa)); |
| |
| qla2x00_send_login_iocb(ha, iodesc, &rscn_pid, ha_locked); |
| |
| return (QLA_SUCCESS); |
| } |
| |
| /** |
| * qla2x00_process_iodesc() - Complete IO descriptor processing. |
| * @ha: HA context |
| * @mbxstat: Mailbox IOCB status |
| */ |
| void |
| qla2x00_process_iodesc(scsi_qla_host_t *ha, struct mbx_entry *mbxstat) |
| { |
| int rval; |
| uint32_t signature; |
| fc_port_t *fcport; |
| struct io_descriptor *iodesc; |
| |
| signature = mbxstat->handle; |
| |
| DEBUG14(printk("scsi(%ld): Process IODesc -- processing %08x.\n", |
| ha->host_no, signature)); |
| |
| /* Retrieve proper IO descriptor. */ |
| iodesc = qla2x00_handle_to_iodesc(ha, signature); |
| if (iodesc == NULL) { |
| DEBUG14(printk("scsi(%ld): Process IODesc -- ignoring, " |
| "incorrect signature %08x.\n", ha->host_no, signature)); |
| |
| return; |
| } |
| |
| /* Stop IO descriptor timer. */ |
| qla2x00_remove_iodesc_timer(iodesc); |
| |
| /* Verify signature match. */ |
| if (iodesc->signature != signature) { |
| DEBUG14(printk("scsi(%ld): Process IODesc -- ignoring, " |
| "signature mismatch, sent %08x, received %08x.\n", |
| ha->host_no, iodesc->signature, signature)); |
| |
| return; |
| } |
| |
| /* Go with IOCB callback. */ |
| rval = iocb_function_cb_list[iodesc->cb_idx](ha, iodesc, mbxstat); |
| if (rval != QLA_SUCCESS) { |
| /* IO descriptor reused by callback. */ |
| return; |
| } |
| |
| qla2x00_free_iodesc(iodesc); |
| |
| if (test_bit(IODESC_PROCESS_NEEDED, &ha->dpc_flags)) { |
| /* Scan our fcports list for any RSCN requests. */ |
| list_for_each_entry(fcport, &ha->fcports, list) { |
| if (fcport->iodesc_idx_sent == IODESC_ADISC_NEEDED || |
| fcport->iodesc_idx_sent == IODESC_LOGIN_NEEDED) { |
| qla2x00_handle_port_rscn(ha, 0, fcport, 1); |
| return; |
| } |
| } |
| |
| /* Scan our RSCN fcports list for any RSCN requests. */ |
| list_for_each_entry(fcport, &ha->rscn_fcports, list) { |
| if (fcport->iodesc_idx_sent == IODESC_ADISC_NEEDED || |
| fcport->iodesc_idx_sent == IODESC_LOGIN_NEEDED) { |
| qla2x00_handle_port_rscn(ha, 0, fcport, 1); |
| return; |
| } |
| } |
| } |
| clear_bit(IODESC_PROCESS_NEEDED, &ha->dpc_flags); |
| } |
| |
| /** |
| * qla2x00_cancel_io_descriptors() - Cancel all outstanding io descriptors. |
| * @ha: HA context |
| * |
| * This routine will also delete any RSCN entries related to the outstanding |
| * IO descriptors. |
| */ |
| void |
| qla2x00_cancel_io_descriptors(scsi_qla_host_t *ha) |
| { |
| fc_port_t *fcport, *fcptemp; |
| |
| clear_bit(IODESC_PROCESS_NEEDED, &ha->dpc_flags); |
| |
| /* Abort all IO descriptors. */ |
| qla2x00_init_io_descriptors(ha); |
| |
| /* Reset all pending IO descriptors in fcports list. */ |
| list_for_each_entry(fcport, &ha->fcports, list) { |
| fcport->iodesc_idx_sent = IODESC_INVALID_INDEX; |
| } |
| |
| /* Reset all pending IO descriptors in rscn fcports list. */ |
| list_for_each_entry_safe(fcport, fcptemp, &ha->rscn_fcports, list) { |
| DEBUG14(printk("scsi(%ld): Cancel IOs -- Freeing RSCN fcport " |
| "%p [%x/%02x%02x%02x].\n", ha->host_no, fcport, |
| fcport->loop_id, fcport->d_id.b.domain, fcport->d_id.b.area, |
| fcport->d_id.b.al_pa)); |
| |
| list_del(&fcport->list); |
| kfree(fcport); |
| } |
| } |