mirror of
https://github.com/AuxXxilium/linux_dsm_epyc7002.git
synced 2025-01-18 13:36:07 +07:00
scsi: rescan VPD attributes
The VPD page information might change, so we need to be able to update it. This patch implements a VPD page rescan whenever the 'rescan' sysfs attribute is triggered. Signed-off-by: Hannes Reinecke <hare@suse.de> Reviewed-by: Johannes Thumshirn <jthumshirn@suse.de> Reviewed-by: Shane Seymour <shane.seymour@hpe.com> Signed-off-by: Martin K. Petersen <martin.petersen@oracle.com>
This commit is contained in:
parent
fe0798c5e1
commit
09e2b0b146
@ -782,7 +782,7 @@ void scsi_attach_vpd(struct scsi_device *sdev)
|
|||||||
int vpd_len = SCSI_VPD_PG_LEN;
|
int vpd_len = SCSI_VPD_PG_LEN;
|
||||||
int pg80_supported = 0;
|
int pg80_supported = 0;
|
||||||
int pg83_supported = 0;
|
int pg83_supported = 0;
|
||||||
unsigned char *vpd_buf;
|
unsigned char __rcu *vpd_buf, *orig_vpd_buf = NULL;
|
||||||
|
|
||||||
if (sdev->skip_vpd_pages)
|
if (sdev->skip_vpd_pages)
|
||||||
return;
|
return;
|
||||||
@ -828,8 +828,16 @@ void scsi_attach_vpd(struct scsi_device *sdev)
|
|||||||
kfree(vpd_buf);
|
kfree(vpd_buf);
|
||||||
goto retry_pg80;
|
goto retry_pg80;
|
||||||
}
|
}
|
||||||
|
mutex_lock(&sdev->inquiry_mutex);
|
||||||
|
orig_vpd_buf = sdev->vpd_pg80;
|
||||||
sdev->vpd_pg80_len = result;
|
sdev->vpd_pg80_len = result;
|
||||||
sdev->vpd_pg80 = vpd_buf;
|
rcu_assign_pointer(sdev->vpd_pg80, vpd_buf);
|
||||||
|
mutex_unlock(&sdev->inquiry_mutex);
|
||||||
|
synchronize_rcu();
|
||||||
|
if (orig_vpd_buf) {
|
||||||
|
kfree(orig_vpd_buf);
|
||||||
|
orig_vpd_buf = NULL;
|
||||||
|
}
|
||||||
vpd_len = SCSI_VPD_PG_LEN;
|
vpd_len = SCSI_VPD_PG_LEN;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -849,8 +857,14 @@ void scsi_attach_vpd(struct scsi_device *sdev)
|
|||||||
kfree(vpd_buf);
|
kfree(vpd_buf);
|
||||||
goto retry_pg83;
|
goto retry_pg83;
|
||||||
}
|
}
|
||||||
|
mutex_lock(&sdev->inquiry_mutex);
|
||||||
|
orig_vpd_buf = sdev->vpd_pg83;
|
||||||
sdev->vpd_pg83_len = result;
|
sdev->vpd_pg83_len = result;
|
||||||
sdev->vpd_pg83 = vpd_buf;
|
rcu_assign_pointer(sdev->vpd_pg83, vpd_buf);
|
||||||
|
mutex_unlock(&sdev->inquiry_mutex);
|
||||||
|
synchronize_rcu();
|
||||||
|
if (orig_vpd_buf)
|
||||||
|
kfree(orig_vpd_buf);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -236,6 +236,7 @@ static struct scsi_device *scsi_alloc_sdev(struct scsi_target *starget,
|
|||||||
INIT_LIST_HEAD(&sdev->starved_entry);
|
INIT_LIST_HEAD(&sdev->starved_entry);
|
||||||
INIT_LIST_HEAD(&sdev->event_list);
|
INIT_LIST_HEAD(&sdev->event_list);
|
||||||
spin_lock_init(&sdev->list_lock);
|
spin_lock_init(&sdev->list_lock);
|
||||||
|
mutex_init(&sdev->inquiry_mutex);
|
||||||
INIT_WORK(&sdev->event_work, scsi_evt_thread);
|
INIT_WORK(&sdev->event_work, scsi_evt_thread);
|
||||||
INIT_WORK(&sdev->requeue_work, scsi_requeue_run_queue);
|
INIT_WORK(&sdev->requeue_work, scsi_requeue_run_queue);
|
||||||
|
|
||||||
@ -1516,6 +1517,9 @@ EXPORT_SYMBOL(scsi_add_device);
|
|||||||
void scsi_rescan_device(struct device *dev)
|
void scsi_rescan_device(struct device *dev)
|
||||||
{
|
{
|
||||||
device_lock(dev);
|
device_lock(dev);
|
||||||
|
|
||||||
|
scsi_attach_vpd(to_scsi_device(dev));
|
||||||
|
|
||||||
if (dev->driver && try_module_get(dev->driver->owner)) {
|
if (dev->driver && try_module_get(dev->driver->owner)) {
|
||||||
struct scsi_driver *drv = to_scsi_driver(dev->driver);
|
struct scsi_driver *drv = to_scsi_driver(dev->driver);
|
||||||
|
|
||||||
|
@ -760,11 +760,15 @@ show_vpd_##_page(struct file *filp, struct kobject *kobj, \
|
|||||||
{ \
|
{ \
|
||||||
struct device *dev = container_of(kobj, struct device, kobj); \
|
struct device *dev = container_of(kobj, struct device, kobj); \
|
||||||
struct scsi_device *sdev = to_scsi_device(dev); \
|
struct scsi_device *sdev = to_scsi_device(dev); \
|
||||||
|
int ret; \
|
||||||
if (!sdev->vpd_##_page) \
|
if (!sdev->vpd_##_page) \
|
||||||
return -EINVAL; \
|
return -EINVAL; \
|
||||||
return memory_read_from_buffer(buf, count, &off, \
|
rcu_read_lock(); \
|
||||||
sdev->vpd_##_page, \
|
ret = memory_read_from_buffer(buf, count, &off, \
|
||||||
|
rcu_dereference(sdev->vpd_##_page), \
|
||||||
sdev->vpd_##_page##_len); \
|
sdev->vpd_##_page##_len); \
|
||||||
|
rcu_read_unlock(); \
|
||||||
|
return ret; \
|
||||||
} \
|
} \
|
||||||
static struct bin_attribute dev_attr_vpd_##_page = { \
|
static struct bin_attribute dev_attr_vpd_##_page = { \
|
||||||
.attr = {.name = __stringify(vpd_##_page), .mode = S_IRUGO }, \
|
.attr = {.name = __stringify(vpd_##_page), .mode = S_IRUGO }, \
|
||||||
|
@ -554,17 +554,22 @@ static void ses_match_to_enclosure(struct enclosure_device *edev,
|
|||||||
struct scsi_device *sdev)
|
struct scsi_device *sdev)
|
||||||
{
|
{
|
||||||
unsigned char *desc;
|
unsigned char *desc;
|
||||||
|
unsigned char __rcu *vpd_pg83;
|
||||||
struct efd efd = {
|
struct efd efd = {
|
||||||
.addr = 0,
|
.addr = 0,
|
||||||
};
|
};
|
||||||
|
|
||||||
ses_enclosure_data_process(edev, to_scsi_device(edev->edev.parent), 0);
|
ses_enclosure_data_process(edev, to_scsi_device(edev->edev.parent), 0);
|
||||||
|
|
||||||
if (!sdev->vpd_pg83_len)
|
rcu_read_lock();
|
||||||
|
vpd_pg83 = rcu_dereference(sdev->vpd_pg83);
|
||||||
|
if (!vpd_pg83) {
|
||||||
|
rcu_read_unlock();
|
||||||
return;
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
desc = sdev->vpd_pg83 + 4;
|
desc = vpd_pg83 + 4;
|
||||||
while (desc < sdev->vpd_pg83 + sdev->vpd_pg83_len) {
|
while (desc < vpd_pg83 + sdev->vpd_pg83_len) {
|
||||||
enum scsi_protocol proto = desc[0] >> 4;
|
enum scsi_protocol proto = desc[0] >> 4;
|
||||||
u8 code_set = desc[0] & 0x0f;
|
u8 code_set = desc[0] & 0x0f;
|
||||||
u8 piv = desc[1] & 0x80;
|
u8 piv = desc[1] & 0x80;
|
||||||
@ -578,6 +583,7 @@ static void ses_match_to_enclosure(struct enclosure_device *edev,
|
|||||||
|
|
||||||
desc += len + 4;
|
desc += len + 4;
|
||||||
}
|
}
|
||||||
|
rcu_read_unlock();
|
||||||
if (efd.addr) {
|
if (efd.addr) {
|
||||||
efd.dev = &sdev->sdev_gendev;
|
efd.dev = &sdev->sdev_gendev;
|
||||||
|
|
||||||
|
@ -109,6 +109,7 @@ struct scsi_device {
|
|||||||
char type;
|
char type;
|
||||||
char scsi_level;
|
char scsi_level;
|
||||||
char inq_periph_qual; /* PQ from INQUIRY data */
|
char inq_periph_qual; /* PQ from INQUIRY data */
|
||||||
|
struct mutex inquiry_mutex;
|
||||||
unsigned char inquiry_len; /* valid bytes in 'inquiry' */
|
unsigned char inquiry_len; /* valid bytes in 'inquiry' */
|
||||||
unsigned char * inquiry; /* INQUIRY response data */
|
unsigned char * inquiry; /* INQUIRY response data */
|
||||||
const char * vendor; /* [back_compat] point into 'inquiry' ... */
|
const char * vendor; /* [back_compat] point into 'inquiry' ... */
|
||||||
@ -117,9 +118,9 @@ struct scsi_device {
|
|||||||
|
|
||||||
#define SCSI_VPD_PG_LEN 255
|
#define SCSI_VPD_PG_LEN 255
|
||||||
int vpd_pg83_len;
|
int vpd_pg83_len;
|
||||||
unsigned char *vpd_pg83;
|
unsigned char __rcu *vpd_pg83;
|
||||||
int vpd_pg80_len;
|
int vpd_pg80_len;
|
||||||
unsigned char *vpd_pg80;
|
unsigned char __rcu *vpd_pg80;
|
||||||
unsigned char current_tag; /* current tag */
|
unsigned char current_tag; /* current tag */
|
||||||
struct scsi_target *sdev_target; /* used only for single_lun */
|
struct scsi_target *sdev_target; /* used only for single_lun */
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user