diff options
author | Bhanu Prakash Gollapudi <bprakash@broadcom.com> | 2010-06-11 19:44:04 -0400 |
---|---|---|
committer | James Bottomley <James.Bottomley@suse.de> | 2010-07-27 13:01:46 -0400 |
commit | f8fc6c2c99b8085368119d6cf39b997255052826 (patch) | |
tree | d1ba9ea7fc49fd8c96143cdcf81a8a9892716ee5 /drivers/scsi/libfc | |
parent | 5d4a2e29fba5b2bef95b96a46b338ec4d76fa4fd (diff) |
[SCSI] libfc: Handle unsolicited PRLO request
Resubmitting after incorporating Joe's review comment.
Unsolicited PRLO request is now handled by sending LS_ACC,
and then relogin to the remote port if an N-port login
session exists for that remote port.
Note that this patch should be applied on top of Joe Eykholt's
"Fix remote port restart problem" patch.
Signed-off-by: Bhanu Prakash Gollapudi <bprakash@broadcom.com>
Signed-off-by: Robert Love <robert.w.love@intel.com>
Signed-off-by: James Bottomley <James.Bottomley@suse.de>
Diffstat (limited to 'drivers/scsi/libfc')
-rw-r--r-- | drivers/scsi/libfc/fc_rport.c | 71 |
1 files changed, 63 insertions, 8 deletions
diff --git a/drivers/scsi/libfc/fc_rport.c b/drivers/scsi/libfc/fc_rport.c index e33c5c7961a2..df85e19079fb 100644 --- a/drivers/scsi/libfc/fc_rport.c +++ b/drivers/scsi/libfc/fc_rport.c | |||
@@ -1573,30 +1573,85 @@ drop: | |||
1573 | * fc_rport_recv_prlo_req() - Handler for process logout (PRLO) requests | 1573 | * fc_rport_recv_prlo_req() - Handler for process logout (PRLO) requests |
1574 | * @rdata: The remote port that sent the PRLO request | 1574 | * @rdata: The remote port that sent the PRLO request |
1575 | * @sp: The sequence that the PRLO was on | 1575 | * @sp: The sequence that the PRLO was on |
1576 | * @fp: The PRLO request frame | 1576 | * @rx_fp: The PRLO request frame |
1577 | * | 1577 | * |
1578 | * Locking Note: The rport lock is exected to be held before calling | 1578 | * Locking Note: The rport lock is exected to be held before calling |
1579 | * this function. | 1579 | * this function. |
1580 | */ | 1580 | */ |
1581 | static void fc_rport_recv_prlo_req(struct fc_rport_priv *rdata, | 1581 | static void fc_rport_recv_prlo_req(struct fc_rport_priv *rdata, |
1582 | struct fc_seq *sp, | 1582 | struct fc_seq *sp, |
1583 | struct fc_frame *fp) | 1583 | struct fc_frame *rx_fp) |
1584 | { | 1584 | { |
1585 | struct fc_lport *lport = rdata->local_port; | 1585 | struct fc_lport *lport = rdata->local_port; |
1586 | |||
1587 | struct fc_frame_header *fh; | 1586 | struct fc_frame_header *fh; |
1587 | struct fc_exch *ep; | ||
1588 | struct fc_frame *fp; | ||
1589 | struct { | ||
1590 | struct fc_els_prlo prlo; | ||
1591 | struct fc_els_spp spp; | ||
1592 | } *pp; | ||
1593 | struct fc_els_spp *rspp; /* request service param page */ | ||
1594 | struct fc_els_spp *spp; /* response spp */ | ||
1595 | unsigned int len; | ||
1596 | unsigned int plen; | ||
1597 | u32 f_ctl; | ||
1588 | struct fc_seq_els_data rjt_data; | 1598 | struct fc_seq_els_data rjt_data; |
1589 | 1599 | ||
1590 | fh = fc_frame_header_get(fp); | 1600 | rjt_data.fp = NULL; |
1601 | fh = fc_frame_header_get(rx_fp); | ||
1591 | 1602 | ||
1592 | FC_RPORT_DBG(rdata, "Received PRLO request while in state %s\n", | 1603 | FC_RPORT_DBG(rdata, "Received PRLO request while in state %s\n", |
1593 | fc_rport_state(rdata)); | 1604 | fc_rport_state(rdata)); |
1594 | 1605 | ||
1595 | rjt_data.fp = NULL; | 1606 | len = fr_len(rx_fp) - sizeof(*fh); |
1596 | rjt_data.reason = ELS_RJT_UNAB; | 1607 | pp = fc_frame_payload_get(rx_fp, sizeof(*pp)); |
1597 | rjt_data.explan = ELS_EXPL_NONE; | 1608 | if (!pp) |
1609 | goto reject_len; | ||
1610 | plen = ntohs(pp->prlo.prlo_len); | ||
1611 | if (plen != 20) | ||
1612 | goto reject_len; | ||
1613 | if (plen < len) | ||
1614 | len = plen; | ||
1615 | |||
1616 | rspp = &pp->spp; | ||
1617 | |||
1618 | fp = fc_frame_alloc(lport, len); | ||
1619 | if (!fp) { | ||
1620 | rjt_data.reason = ELS_RJT_UNAB; | ||
1621 | rjt_data.explan = ELS_EXPL_INSUF_RES; | ||
1622 | goto reject; | ||
1623 | } | ||
1624 | |||
1625 | sp = lport->tt.seq_start_next(sp); | ||
1626 | WARN_ON(!sp); | ||
1627 | pp = fc_frame_payload_get(fp, len); | ||
1628 | WARN_ON(!pp); | ||
1629 | memset(pp, 0, len); | ||
1630 | pp->prlo.prlo_cmd = ELS_LS_ACC; | ||
1631 | pp->prlo.prlo_obs = 0x10; | ||
1632 | pp->prlo.prlo_len = htons(len); | ||
1633 | spp = &pp->spp; | ||
1634 | spp->spp_type = rspp->spp_type; | ||
1635 | spp->spp_type_ext = rspp->spp_type_ext; | ||
1636 | spp->spp_flags = FC_SPP_RESP_ACK; | ||
1637 | |||
1638 | fc_rport_enter_delete(rdata, RPORT_EV_LOGO); | ||
1639 | |||
1640 | f_ctl = FC_FC_EX_CTX | FC_FC_LAST_SEQ; | ||
1641 | f_ctl |= FC_FC_END_SEQ | FC_FC_SEQ_INIT; | ||
1642 | ep = fc_seq_exch(sp); | ||
1643 | fc_fill_fc_hdr(fp, FC_RCTL_ELS_REP, ep->did, ep->sid, | ||
1644 | FC_TYPE_ELS, f_ctl, 0); | ||
1645 | lport->tt.seq_send(lport, sp, fp); | ||
1646 | goto drop; | ||
1647 | |||
1648 | reject_len: | ||
1649 | rjt_data.reason = ELS_RJT_PROT; | ||
1650 | rjt_data.explan = ELS_EXPL_INV_LEN; | ||
1651 | reject: | ||
1598 | lport->tt.seq_els_rsp_send(sp, ELS_LS_RJT, &rjt_data); | 1652 | lport->tt.seq_els_rsp_send(sp, ELS_LS_RJT, &rjt_data); |
1599 | fc_frame_free(fp); | 1653 | drop: |
1654 | fc_frame_free(rx_fp); | ||
1600 | } | 1655 | } |
1601 | 1656 | ||
1602 | /** | 1657 | /** |