s390/dasd: Process FCES path event notification
authorJan Höppner <hoeppner@linux.ibm.com>
Thu, 8 Oct 2020 13:13:36 +0000 (15:13 +0200)
committerJens Axboe <axboe@kernel.dk>
Mon, 16 Nov 2020 15:14:38 +0000 (08:14 -0700)
If the Fibre Channel Endpoint-Security status of a path changes, a
corresponding path event is received from the CIO layer.

Process this event by re-reading the FCES information.

As the information is retrieved for all paths on a single CU in one
call, the internal status can also be updated for all paths and no
processing per path is necessary.

Signed-off-by: Jan Höppner <hoeppner@linux.ibm.com>
Signed-off-by: Stefan Haberland <sth@linux.ibm.com>
Reviewed-by: Stefan Haberland <sth@linux.ibm.com>
Reviewed-by: Cornelia Huck <cohuck@redhat.com>
Signed-off-by: Jens Axboe <axboe@kernel.dk>
drivers/s390/block/dasd.c
drivers/s390/block/dasd_eckd.c
drivers/s390/block/dasd_int.h

index 8d7a53e..874345e 100644 (file)
@@ -2107,20 +2107,25 @@ static void __dasd_device_start_head(struct dasd_device *device)
 
 static void __dasd_device_check_path_events(struct dasd_device *device)
 {
+       __u8 tbvpm, fcsecpm;
        int rc;
 
-       if (!dasd_path_get_tbvpm(device))
+       tbvpm = dasd_path_get_tbvpm(device);
+       fcsecpm = dasd_path_get_fcsecpm(device);
+
+       if (!tbvpm && !fcsecpm)
                return;
 
        if (device->stopped &
            ~(DASD_STOPPED_DC_WAIT | DASD_UNRESUMED_PM))
                return;
-       rc = device->discipline->pe_handler(device,
-                                           dasd_path_get_tbvpm(device));
-       if (rc)
+       rc = device->discipline->pe_handler(device, tbvpm, fcsecpm);
+       if (rc) {
                dasd_device_set_timer(device, 50);
-       else
+       } else {
                dasd_path_clear_all_verify(device);
+               dasd_path_clear_all_fcsec(device);
+       }
 };
 
 /*
@@ -3869,6 +3874,10 @@ void dasd_generic_path_event(struct ccw_device *cdev, int *path_event)
                        if (device->discipline->kick_validate)
                                device->discipline->kick_validate(device);
                }
+               if (path_event[chp] & PE_PATH_FCES_EVENT) {
+                       dasd_path_fcsec_update(device, chp);
+                       dasd_schedule_device_bh(device);
+               }
        }
        hpfpm = dasd_path_get_hpfpm(device);
        ifccpm = dasd_path_get_ifccpm(device);
index 2e1cfac..0d319c2 100644 (file)
@@ -111,6 +111,7 @@ struct pe_handler_work_data {
        __u8 rcd_buffer[DASD_ECKD_RCD_DATA_SIZE];
        int isglobal;
        __u8 tbvpm;
+       __u8 fcsecpm;
 };
 static struct pe_handler_work_data *pe_handler_worker;
 static DEFINE_MUTEX(dasd_pe_handler_mutex);
@@ -1466,7 +1467,10 @@ static void do_pe_handler_work(struct work_struct *work)
                return;
        }
 
-       dasd_eckd_path_available_action(device, data);
+       if (data->tbvpm)
+               dasd_eckd_path_available_action(device, data);
+       if (data->fcsecpm)
+               dasd_eckd_read_fc_security(device);
 
        clear_bit(DASD_FLAG_PATH_VERIFY, &device->flags);
        dasd_put_device(device);
@@ -1476,7 +1480,8 @@ static void do_pe_handler_work(struct work_struct *work)
                kfree(data);
 }
 
-static int dasd_eckd_pe_handler(struct dasd_device *device, __u8 lpm)
+static int dasd_eckd_pe_handler(struct dasd_device *device,
+                               __u8 tbvpm, __u8 fcsecpm)
 {
        struct pe_handler_work_data *data;
 
@@ -1495,7 +1500,8 @@ static int dasd_eckd_pe_handler(struct dasd_device *device, __u8 lpm)
        INIT_WORK(&data->worker, do_pe_handler_work);
        dasd_get_device(device);
        data->device = device;
-       data->tbvpm = lpm;
+       data->tbvpm = tbvpm;
+       data->fcsecpm = fcsecpm;
        schedule_work(&data->worker);
        return 0;
 }
index 4cfed3b..10f411c 100644 (file)
@@ -298,7 +298,7 @@ struct dasd_discipline {
         * configuration.
         */
        int (*verify_path)(struct dasd_device *, __u8);
-       int (*pe_handler)(struct dasd_device *, __u8);
+       int (*pe_handler)(struct dasd_device *, __u8, __u8);
 
        /*
         * Last things to do when a device is set online, and first things
@@ -423,6 +423,7 @@ extern struct dasd_discipline *dasd_diag_discipline_pointer;
 #define DASD_PATH_NOHPF        6
 #define DASD_PATH_CUIR        7
 #define DASD_PATH_IFCC        8
+#define DASD_PATH_FCSEC               9
 
 #define DASD_THRHLD_MAX                4294967295U
 #define DASD_INTERVAL_MAX      4294967295U
@@ -966,6 +967,29 @@ static inline void dasd_path_clear_all_verify(struct dasd_device *device)
                dasd_path_clear_verify(device, chp);
 }
 
+static inline void dasd_path_fcsec(struct dasd_device *device, int chp)
+{
+       __set_bit(DASD_PATH_FCSEC, &device->path[chp].flags);
+}
+
+static inline void dasd_path_clear_fcsec(struct dasd_device *device, int chp)
+{
+       __clear_bit(DASD_PATH_FCSEC, &device->path[chp].flags);
+}
+
+static inline int dasd_path_need_fcsec(struct dasd_device *device, int chp)
+{
+       return test_bit(DASD_PATH_FCSEC, &device->path[chp].flags);
+}
+
+static inline void dasd_path_clear_all_fcsec(struct dasd_device *device)
+{
+       int chp;
+
+       for (chp = 0; chp < 8; chp++)
+               dasd_path_clear_fcsec(device, chp);
+}
+
 static inline void dasd_path_operational(struct dasd_device *device, int chp)
 {
        __set_bit(DASD_PATH_OPERATIONAL, &device->path[chp].flags);
@@ -1091,6 +1115,17 @@ static inline __u8 dasd_path_get_tbvpm(struct dasd_device *device)
        return tbvpm;
 }
 
+static inline int dasd_path_get_fcsecpm(struct dasd_device *device)
+{
+       int chp;
+
+       for (chp = 0; chp < 8; chp++)
+               if (dasd_path_need_fcsec(device, chp))
+                       return 1;
+
+       return 0;
+}
+
 static inline __u8 dasd_path_get_nppm(struct dasd_device *device)
 {
        int chp;
@@ -1348,6 +1383,11 @@ static inline void dasd_path_notoper(struct dasd_device *device, int chp)
        dasd_path_clear_nonpreferred(device, chp);
 }
 
+static inline void dasd_path_fcsec_update(struct dasd_device *device, int chp)
+{
+       dasd_path_fcsec(device, chp);
+}
+
 /*
  * remove all paths from normal operation
  */