2  * Copyright(c) 2007 - 2008 Intel Corporation. All rights reserved.
 
   4  * This program is free software; you can redistribute it and/or modify it
 
   5  * under the terms and conditions of the GNU General Public License,
 
   6  * version 2, as published by the Free Software Foundation.
 
   8  * This program is distributed in the hope it will be useful, but WITHOUT
 
   9  * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
 
  10  * FITNESS FOR A PARTICULAR PURPOSE.  See the GNU General Public License for
 
  13  * You should have received a copy of the GNU General Public License along with
 
  14  * this program; if not, write to the Free Software Foundation, Inc.,
 
  15  * 51 Franklin St - Fifth Floor, Boston, MA 02110-1301 USA.
 
  17  * Maintained at www.Open-FCoE.org
 
  23  * This block discovers all FC-4 remote ports, including FCP initiators. It
 
  24  * also handles RSCN events and re-discovery if necessary.
 
  30  * The disc mutex is can be locked when acquiring rport locks, but may not
 
  31  * be held when acquiring the lport lock. Refer to fc_lport.c for more
 
  35 #include <linux/timer.h>
 
  36 #include <linux/err.h>
 
  37 #include <asm/unaligned.h>
 
  39 #include <scsi/fc/fc_gs.h>
 
  41 #include <scsi/libfc.h>
 
  43 #define FC_DISC_RETRY_LIMIT     3       /* max retries */
 
  44 #define FC_DISC_RETRY_DELAY     500UL   /* (msecs) delay */
 
  46 #define FC_DISC_DELAY           3
 
  48 static int fc_disc_debug;
 
  50 #define FC_DEBUG_DISC(fmt...)                   \
 
  56 static void fc_disc_gpn_ft_req(struct fc_disc *);
 
  57 static void fc_disc_gpn_ft_resp(struct fc_seq *, struct fc_frame *, void *);
 
  58 static int fc_disc_new_target(struct fc_disc *, struct fc_rport *,
 
  59                               struct fc_rport_identifiers *);
 
  60 static void fc_disc_del_target(struct fc_disc *, struct fc_rport *);
 
  61 static void fc_disc_done(struct fc_disc *);
 
  62 static void fc_disc_timeout(struct work_struct *);
 
  63 static void fc_disc_single(struct fc_disc *, struct fc_disc_port *);
 
  64 static void fc_disc_restart(struct fc_disc *);
 
  67  * fc_disc_lookup_rport - lookup a remote port by port_id
 
  68  * @lport: Fibre Channel host port instance
 
  69  * @port_id: remote port port_id to match
 
  71 struct fc_rport *fc_disc_lookup_rport(const struct fc_lport *lport,
 
  74         const struct fc_disc *disc = &lport->disc;
 
  75         struct fc_rport *rport, *found = NULL;
 
  76         struct fc_rport_libfc_priv *rdata;
 
  79         list_for_each_entry(rdata, &disc->rports, peers) {
 
  80                 rport = PRIV_TO_RPORT(rdata);
 
  81                 if (rport->port_id == port_id) {
 
  95  * fc_disc_stop_rports - delete all the remote ports associated with the lport
 
  96  * @disc: The discovery job to stop rports on
 
  98  * Locking Note: This function expects that the lport mutex is locked before
 
 101 void fc_disc_stop_rports(struct fc_disc *disc)
 
 103         struct fc_lport *lport;
 
 104         struct fc_rport *rport;
 
 105         struct fc_rport_libfc_priv *rdata, *next;
 
 109         mutex_lock(&disc->disc_mutex);
 
 110         list_for_each_entry_safe(rdata, next, &disc->rports, peers) {
 
 111                 rport = PRIV_TO_RPORT(rdata);
 
 112                 list_del(&rdata->peers);
 
 113                 lport->tt.rport_logoff(rport);
 
 116         mutex_unlock(&disc->disc_mutex);
 
 120  * fc_disc_rport_callback - Event handler for rport events
 
 121  * @lport: The lport which is receiving the event
 
 122  * @rport: The rport which the event has occured on
 
 123  * @event: The event that occured
 
 125  * Locking Note: The rport lock should not be held when calling
 
 128 static void fc_disc_rport_callback(struct fc_lport *lport,
 
 129                                    struct fc_rport *rport,
 
 130                                    enum fc_rport_event event)
 
 132         struct fc_rport_libfc_priv *rdata = rport->dd_data;
 
 133         struct fc_disc *disc = &lport->disc;
 
 136         FC_DEBUG_DISC("Received a %d event for port (%6x)\n", event,
 
 139         if (event == RPORT_EV_CREATED) {
 
 142                         mutex_lock(&disc->disc_mutex);
 
 143                         list_add_tail(&rdata->peers, &disc->rports);
 
 144                         mutex_unlock(&disc->disc_mutex);
 
 149                 FC_DEBUG_DISC("The rport (%6x) is not maintained "
 
 150                               "by the discovery layer\n", rport->port_id);
 
 154  * fc_disc_recv_rscn_req - Handle Registered State Change Notification (RSCN)
 
 155  * @sp: Current sequence of the RSCN exchange
 
 157  * @lport: Fibre Channel host port instance
 
 159  * Locking Note: This function expects that the disc_mutex is locked
 
 160  *               before it is called.
 
 162 static void fc_disc_recv_rscn_req(struct fc_seq *sp, struct fc_frame *fp,
 
 163                                   struct fc_disc *disc)
 
 165         struct fc_lport *lport;
 
 166         struct fc_rport *rport;
 
 167         struct fc_rport_libfc_priv *rdata;
 
 168         struct fc_els_rscn *rp;
 
 169         struct fc_els_rscn_page *pp;
 
 170         struct fc_seq_els_data rjt_data;
 
 173         enum fc_els_rscn_ev_qual ev_qual;
 
 174         enum fc_els_rscn_addr_fmt fmt;
 
 175         LIST_HEAD(disc_ports);
 
 176         struct fc_disc_port *dp, *next;
 
 180         FC_DEBUG_DISC("Received an RSCN event on port (%6x)\n",
 
 181                       fc_host_port_id(lport->host));
 
 183         /* make sure the frame contains an RSCN message */
 
 184         rp = fc_frame_payload_get(fp, sizeof(*rp));
 
 187         /* make sure the page length is as expected (4 bytes) */
 
 188         if (rp->rscn_page_len != sizeof(*pp))
 
 190         /* get the RSCN payload length */
 
 191         len = ntohs(rp->rscn_plen);
 
 192         if (len < sizeof(*rp))
 
 194         /* make sure the frame contains the expected payload */
 
 195         rp = fc_frame_payload_get(fp, len);
 
 198         /* payload must be a multiple of the RSCN page size */
 
 200         if (len % sizeof(*pp))
 
 203         for (pp = (void *)(rp + 1); len > 0; len -= sizeof(*pp), pp++) {
 
 204                 ev_qual = pp->rscn_page_flags >> ELS_RSCN_EV_QUAL_BIT;
 
 205                 ev_qual &= ELS_RSCN_EV_QUAL_MASK;
 
 206                 fmt = pp->rscn_page_flags >> ELS_RSCN_ADDR_FMT_BIT;
 
 207                 fmt &= ELS_RSCN_ADDR_FMT_MASK;
 
 209                  * if we get an address format other than port
 
 210                  * (area, domain, fabric), then do a full discovery
 
 213                 case ELS_ADDR_FMT_PORT:
 
 214                         FC_DEBUG_DISC("Port address format for port (%6x)\n",
 
 215                                       ntoh24(pp->rscn_fid));
 
 216                         dp = kzalloc(sizeof(*dp), GFP_KERNEL);
 
 222                         dp->ids.port_id = ntoh24(pp->rscn_fid);
 
 223                         dp->ids.port_name = -1;
 
 224                         dp->ids.node_name = -1;
 
 225                         dp->ids.roles = FC_RPORT_ROLE_UNKNOWN;
 
 226                         list_add_tail(&dp->peers, &disc_ports);
 
 228                 case ELS_ADDR_FMT_AREA:
 
 229                 case ELS_ADDR_FMT_DOM:
 
 230                 case ELS_ADDR_FMT_FAB:
 
 232                         FC_DEBUG_DISC("Address format is (%d)\n", fmt);
 
 237         lport->tt.seq_els_rsp_send(sp, ELS_LS_ACC, NULL);
 
 239                 FC_DEBUG_DISC("RSCN received: rediscovering\n");
 
 240                 fc_disc_restart(disc);
 
 242                 FC_DEBUG_DISC("RSCN received: not rediscovering. "
 
 243                               "redisc %d state %d in_prog %d\n",
 
 244                               redisc, lport->state, disc->pending);
 
 245                 list_for_each_entry_safe(dp, next, &disc_ports, peers) {
 
 246                         list_del(&dp->peers);
 
 247                         rport = lport->tt.rport_lookup(lport, dp->ids.port_id);
 
 249                                 rdata = RPORT_TO_PRIV(rport);
 
 250                                 list_del(&rdata->peers);
 
 251                                 lport->tt.rport_logoff(rport);
 
 253                         fc_disc_single(disc, dp);
 
 259         FC_DEBUG_DISC("Received a bad RSCN frame\n");
 
 261         rjt_data.reason = ELS_RJT_LOGIC;
 
 262         rjt_data.explan = ELS_EXPL_NONE;
 
 263         lport->tt.seq_els_rsp_send(sp, ELS_LS_RJT, &rjt_data);
 
 268  * fc_disc_recv_req - Handle incoming requests
 
 269  * @sp: Current sequence of the request exchange
 
 271  * @lport: The FC local port
 
 273  * Locking Note: This function is called from the EM and will lock
 
 274  *               the disc_mutex before calling the handler for the
 
 277 static void fc_disc_recv_req(struct fc_seq *sp, struct fc_frame *fp,
 
 278                              struct fc_lport *lport)
 
 281         struct fc_disc *disc = &lport->disc;
 
 283         op = fc_frame_payload_op(fp);
 
 286                 mutex_lock(&disc->disc_mutex);
 
 287                 fc_disc_recv_rscn_req(sp, fp, disc);
 
 288                 mutex_unlock(&disc->disc_mutex);
 
 291                 FC_DBG("Received an unsupported request. opcode (%x)\n", op);
 
 297  * fc_disc_restart - Restart discovery
 
 298  * @lport: FC discovery context
 
 300  * Locking Note: This function expects that the disc mutex
 
 303 static void fc_disc_restart(struct fc_disc *disc)
 
 305         struct fc_rport *rport;
 
 306         struct fc_rport_libfc_priv *rdata, *next;
 
 307         struct fc_lport *lport = disc->lport;
 
 309         FC_DEBUG_DISC("Restarting discovery for port (%6x)\n",
 
 310                       fc_host_port_id(lport->host));
 
 312         list_for_each_entry_safe(rdata, next, &disc->rports, peers) {
 
 313                 rport = PRIV_TO_RPORT(rdata);
 
 314                 FC_DEBUG_DISC("list_del(%6x)\n", rport->port_id);
 
 315                 list_del(&rdata->peers);
 
 316                 lport->tt.rport_logoff(rport);
 
 321                 fc_disc_gpn_ft_req(disc);
 
 325  * fc_disc_start - Fibre Channel Target discovery
 
 326  * @lport: FC local port
 
 328  * Returns non-zero if discovery cannot be started.
 
 330 static void fc_disc_start(void (*disc_callback)(struct fc_lport *,
 
 332                           struct fc_lport *lport)
 
 334         struct fc_rport *rport;
 
 335         struct fc_rport_identifiers ids;
 
 336         struct fc_disc *disc = &lport->disc;
 
 339          * At this point we may have a new disc job or an existing
 
 340          * one. Either way, let's lock when we make changes to it
 
 341          * and send the GPN_FT request.
 
 343         mutex_lock(&disc->disc_mutex);
 
 345         disc->disc_callback = disc_callback;
 
 348          * If not ready, or already running discovery, just set request flag.
 
 353                 mutex_unlock(&disc->disc_mutex);
 
 358          * Handle point-to-point mode as a simple discovery
 
 359          * of the remote port. Yucky, yucky, yuck, yuck!
 
 361         rport = disc->lport->ptp_rp;
 
 363                 ids.port_id = rport->port_id;
 
 364                 ids.port_name = rport->port_name;
 
 365                 ids.node_name = rport->node_name;
 
 366                 ids.roles = FC_RPORT_ROLE_UNKNOWN;
 
 367                 get_device(&rport->dev);
 
 369                 if (!fc_disc_new_target(disc, rport, &ids)) {
 
 370                         disc->event = DISC_EV_SUCCESS;
 
 373                 put_device(&rport->dev);
 
 375                 fc_disc_gpn_ft_req(disc);       /* get ports by FC-4 type */
 
 378         mutex_unlock(&disc->disc_mutex);
 
 381 static struct fc_rport_operations fc_disc_rport_ops = {
 
 382         .event_callback = fc_disc_rport_callback,
 
 386  * fc_disc_new_target - Handle new target found by discovery
 
 387  * @lport: FC local port
 
 388  * @rport: The previous FC remote port (NULL if new remote port)
 
 389  * @ids: Identifiers for the new FC remote port
 
 391  * Locking Note: This function expects that the disc_mutex is locked
 
 392  *               before it is called.
 
 394 static int fc_disc_new_target(struct fc_disc *disc,
 
 395                               struct fc_rport *rport,
 
 396                               struct fc_rport_identifiers *ids)
 
 398         struct fc_lport *lport = disc->lport;
 
 399         struct fc_rport_libfc_priv *rp;
 
 402         if (rport && ids->port_name) {
 
 403                 if (rport->port_name == -1) {
 
 405                          * Set WWN and fall through to notify of create.
 
 407                         fc_rport_set_name(rport, ids->port_name,
 
 409                 } else if (rport->port_name != ids->port_name) {
 
 411                          * This is a new port with the same FCID as
 
 412                          * a previously-discovered port.  Presumably the old
 
 413                          * port logged out and a new port logged in and was
 
 414                          * assigned the same FCID.  This should be rare.
 
 415                          * Delete the old one and fall thru to re-create.
 
 417                         fc_disc_del_target(disc, rport);
 
 421         if (((ids->port_name != -1) || (ids->port_id != -1)) &&
 
 422             ids->port_id != fc_host_port_id(lport->host) &&
 
 423             ids->port_name != lport->wwpn) {
 
 425                         rport = lport->tt.rport_lookup(lport, ids->port_id);
 
 427                                 struct fc_disc_port dp;
 
 429                                 dp.ids.port_id = ids->port_id;
 
 430                                 dp.ids.port_name = ids->port_name;
 
 431                                 dp.ids.node_name = ids->node_name;
 
 432                                 dp.ids.roles = ids->roles;
 
 433                                 rport = fc_rport_rogue_create(&dp);
 
 440                         rp->ops = &fc_disc_rport_ops;
 
 441                         rp->rp_state = RPORT_ST_INIT;
 
 442                         lport->tt.rport_login(rport);
 
 449  * fc_disc_del_target - Delete a target
 
 450  * @disc: FC discovery context
 
 451  * @rport: The remote port to be removed
 
 453 static void fc_disc_del_target(struct fc_disc *disc, struct fc_rport *rport)
 
 455         struct fc_lport *lport = disc->lport;
 
 456         struct fc_rport_libfc_priv *rdata = RPORT_TO_PRIV(rport);
 
 457         list_del(&rdata->peers);
 
 458         lport->tt.rport_logoff(rport);
 
 462  * fc_disc_done - Discovery has been completed
 
 463  * @disc: FC discovery context
 
 465 static void fc_disc_done(struct fc_disc *disc)
 
 467         struct fc_lport *lport = disc->lport;
 
 469         FC_DEBUG_DISC("Discovery complete for port (%6x)\n",
 
 470                       fc_host_port_id(lport->host));
 
 472         disc->disc_callback(lport, disc->event);
 
 473         disc->event = DISC_EV_NONE;
 
 476                 fc_disc_gpn_ft_req(disc);
 
 482  * fc_disc_error - Handle error on dNS request
 
 483  * @disc: FC discovery context
 
 484  * @fp: The frame pointer
 
 486 static void fc_disc_error(struct fc_disc *disc, struct fc_frame *fp)
 
 488         struct fc_lport *lport = disc->lport;
 
 489         unsigned long delay = 0;
 
 491                 FC_DBG("Error %ld, retries %d/%d\n",
 
 492                        PTR_ERR(fp), disc->retry_count,
 
 493                        FC_DISC_RETRY_LIMIT);
 
 495         if (!fp || PTR_ERR(fp) == -FC_EX_TIMEOUT) {
 
 497                  * Memory allocation failure, or the exchange timed out,
 
 500                 if (disc->retry_count < FC_DISC_RETRY_LIMIT) {
 
 501                         /* go ahead and retry */
 
 503                                 delay = msecs_to_jiffies(FC_DISC_RETRY_DELAY);
 
 505                                 delay = msecs_to_jiffies(lport->e_d_tov);
 
 507                                 /* timeout faster first time */
 
 508                                 if (!disc->retry_count)
 
 512                         schedule_delayed_work(&disc->disc_work, delay);
 
 514                         /* exceeded retries */
 
 515                         disc->event = DISC_EV_FAILED;
 
 522  * fc_disc_gpn_ft_req - Send Get Port Names by FC-4 type (GPN_FT) request
 
 523  * @lport: FC discovery context
 
 525  * Locking Note: This function expects that the disc_mutex is locked
 
 526  *               before it is called.
 
 528 static void fc_disc_gpn_ft_req(struct fc_disc *disc)
 
 531         struct fc_lport *lport = disc->lport;
 
 533         WARN_ON(!fc_lport_test_ready(lport));
 
 540         fp = fc_frame_alloc(lport,
 
 541                             sizeof(struct fc_ct_hdr) +
 
 542                             sizeof(struct fc_ns_gid_ft));
 
 546         if (lport->tt.elsct_send(lport, NULL, fp,
 
 549                                  disc, lport->e_d_tov))
 
 552         fc_disc_error(disc, fp);
 
 556  * fc_disc_gpn_ft_parse - Parse the list of IDs and names resulting from a request
 
 557  * @lport: Fibre Channel host port instance
 
 558  * @buf: GPN_FT response buffer
 
 559  * @len: size of response buffer
 
 561 static int fc_disc_gpn_ft_parse(struct fc_disc *disc, void *buf, size_t len)
 
 563         struct fc_lport *lport;
 
 564         struct fc_gpn_ft_resp *np;
 
 569         struct fc_disc_port dp;
 
 570         struct fc_rport *rport;
 
 571         struct fc_rport_libfc_priv *rdata;
 
 576          * Handle partial name record left over from previous call.
 
 580         np = (struct fc_gpn_ft_resp *)bp;
 
 581         tlen = disc->buf_len;
 
 583                 WARN_ON(tlen >= sizeof(*np));
 
 584                 plen = sizeof(*np) - tlen;
 
 586                 WARN_ON(plen >= sizeof(*np));
 
 589                 np = &disc->partial_buf;
 
 590                 memcpy((char *)np + tlen, bp, plen);
 
 593                  * Set bp so that the loop below will advance it to the
 
 594                  * first valid full name element.
 
 599                 disc->buf_len = (unsigned char) plen;
 
 600                 if (plen == sizeof(*np))
 
 605          * Handle full name records, including the one filled from above.
 
 606          * Normally, np == bp and plen == len, but from the partial case above,
 
 607          * bp, len describe the overall buffer, and np, plen describe the
 
 608          * partial buffer, which if would usually be full now.
 
 609          * After the first time through the loop, things return to "normal".
 
 611         while (plen >= sizeof(*np)) {
 
 613                 dp.ids.port_id = ntoh24(np->fp_fid);
 
 614                 dp.ids.port_name = ntohll(np->fp_wwpn);
 
 615                 dp.ids.node_name = -1;
 
 616                 dp.ids.roles = FC_RPORT_ROLE_UNKNOWN;
 
 618                 if ((dp.ids.port_id != fc_host_port_id(lport->host)) &&
 
 619                     (dp.ids.port_name != lport->wwpn)) {
 
 620                         rport = fc_rport_rogue_create(&dp);
 
 622                                 rdata = rport->dd_data;
 
 623                                 rdata->ops = &fc_disc_rport_ops;
 
 624                                 rdata->local_port = lport;
 
 625                                 lport->tt.rport_login(rport);
 
 627                                 FC_DBG("Failed to allocate memory for "
 
 628                                        "the newly discovered port (%6x)\n",
 
 632                 if (np->fp_flags & FC_NS_FID_LAST) {
 
 633                         disc->event = DISC_EV_SUCCESS;
 
 640                 np = (struct fc_gpn_ft_resp *)bp;
 
 645          * Save any partial record at the end of the buffer for next time.
 
 647         if (error == 0 && len > 0 && len < sizeof(*np)) {
 
 648                 if (np != &disc->partial_buf) {
 
 649                         FC_DEBUG_DISC("Partial buffer remains "
 
 650                                       "for discovery by (%6x)\n",
 
 651                                       fc_host_port_id(lport->host));
 
 652                         memcpy(&disc->partial_buf, np, len);
 
 654                 disc->buf_len = (unsigned char) len;
 
 662  * Handle retry of memory allocation for remote ports.
 
 664 static void fc_disc_timeout(struct work_struct *work)
 
 666         struct fc_disc *disc = container_of(work,
 
 669         mutex_lock(&disc->disc_mutex);
 
 670         if (disc->requested && !disc->pending)
 
 671                 fc_disc_gpn_ft_req(disc);
 
 672         mutex_unlock(&disc->disc_mutex);
 
 676  * fc_disc_gpn_ft_resp - Handle a response frame from Get Port Names (GPN_FT)
 
 677  * @sp: Current sequence of GPN_FT exchange
 
 678  * @fp: response frame
 
 679  * @lp_arg: Fibre Channel host port instance
 
 681  * Locking Note: This function expects that the disc_mutex is locked
 
 682  *               before it is called.
 
 684 static void fc_disc_gpn_ft_resp(struct fc_seq *sp, struct fc_frame *fp,
 
 687         struct fc_disc *disc = disc_arg;
 
 688         struct fc_ct_hdr *cp;
 
 689         struct fc_frame_header *fh;
 
 690         unsigned int seq_cnt;
 
 695         FC_DEBUG_DISC("Received a GPN_FT response on port (%6x)\n",
 
 696                       fc_host_port_id(disc->lport->host));
 
 699                 fc_disc_error(disc, fp);
 
 703         WARN_ON(!fc_frame_is_linear(fp));       /* buffer must be contiguous */
 
 704         fh = fc_frame_header_get(fp);
 
 705         len = fr_len(fp) - sizeof(*fh);
 
 706         seq_cnt = ntohs(fh->fh_seq_cnt);
 
 707         if (fr_sof(fp) == FC_SOF_I3 && seq_cnt == 0 &&
 
 708             disc->seq_count == 0) {
 
 709                 cp = fc_frame_payload_get(fp, sizeof(*cp));
 
 711                         FC_DBG("GPN_FT response too short, len %d\n",
 
 713                 } else if (ntohs(cp->ct_cmd) == FC_FS_ACC) {
 
 716                          * Accepted.  Parse response.
 
 720                 } else if (ntohs(cp->ct_cmd) == FC_FS_RJT) {
 
 721                         FC_DBG("GPN_FT rejected reason %x exp %x "
 
 722                                "(check zoning)\n", cp->ct_reason,
 
 724                         disc->event = DISC_EV_FAILED;
 
 727                         FC_DBG("GPN_FT unexpected response code %x\n",
 
 730         } else if (fr_sof(fp) == FC_SOF_N3 &&
 
 731                    seq_cnt == disc->seq_count) {
 
 734                 FC_DBG("GPN_FT unexpected frame - out of sequence? "
 
 735                        "seq_cnt %x expected %x sof %x eof %x\n",
 
 736                        seq_cnt, disc->seq_count, fr_sof(fp), fr_eof(fp));
 
 739                 error = fc_disc_gpn_ft_parse(disc, buf, len);
 
 741                         fc_disc_error(disc, fp);
 
 749  * fc_disc_single - Discover the directory information for a single target
 
 750  * @lport: FC local port
 
 751  * @dp: The port to rediscover
 
 753  * Locking Note: This function expects that the disc_mutex is locked
 
 754  *               before it is called.
 
 756 static void fc_disc_single(struct fc_disc *disc, struct fc_disc_port *dp)
 
 758         struct fc_lport *lport;
 
 759         struct fc_rport *rport;
 
 760         struct fc_rport *new_rport;
 
 761         struct fc_rport_libfc_priv *rdata;
 
 765         if (dp->ids.port_id == fc_host_port_id(lport->host))
 
 768         rport = lport->tt.rport_lookup(lport, dp->ids.port_id);
 
 770                 fc_disc_del_target(disc, rport);
 
 772         new_rport = fc_rport_rogue_create(dp);
 
 774                 rdata = new_rport->dd_data;
 
 775                 rdata->ops = &fc_disc_rport_ops;
 
 777                 lport->tt.rport_login(new_rport);
 
 785  * fc_disc_stop - Stop discovery for a given lport
 
 786  * @lport: The lport that discovery should stop for
 
 788 void fc_disc_stop(struct fc_lport *lport)
 
 790         struct fc_disc *disc = &lport->disc;
 
 793                 cancel_delayed_work_sync(&disc->disc_work);
 
 794                 fc_disc_stop_rports(disc);
 
 799  * fc_disc_stop_final - Stop discovery for a given lport
 
 800  * @lport: The lport that discovery should stop for
 
 802  * This function will block until discovery has been
 
 803  * completely stopped and all rports have been deleted.
 
 805 void fc_disc_stop_final(struct fc_lport *lport)
 
 808         lport->tt.rport_flush_queue();
 
 812  * fc_disc_init - Initialize the discovery block
 
 813  * @lport: FC local port
 
 815 int fc_disc_init(struct fc_lport *lport)
 
 817         struct fc_disc *disc;
 
 819         if (!lport->tt.disc_start)
 
 820                 lport->tt.disc_start = fc_disc_start;
 
 822         if (!lport->tt.disc_stop)
 
 823                 lport->tt.disc_stop = fc_disc_stop;
 
 825         if (!lport->tt.disc_stop_final)
 
 826                 lport->tt.disc_stop_final = fc_disc_stop_final;
 
 828         if (!lport->tt.disc_recv_req)
 
 829                 lport->tt.disc_recv_req = fc_disc_recv_req;
 
 831         if (!lport->tt.rport_lookup)
 
 832                 lport->tt.rport_lookup = fc_disc_lookup_rport;
 
 835         INIT_DELAYED_WORK(&disc->disc_work, fc_disc_timeout);
 
 836         mutex_init(&disc->disc_mutex);
 
 837         INIT_LIST_HEAD(&disc->rports);
 
 840         disc->delay = FC_DISC_DELAY;
 
 841         disc->event = DISC_EV_NONE;
 
 845 EXPORT_SYMBOL(fc_disc_init);