aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorDan Williams <dan.j.williams@intel.com>2019-02-13 12:04:07 -0500
committerDan Williams <dan.j.williams@intel.com>2019-02-20 17:18:59 -0500
commit5479b2757f26fe9908fc341d105b2097fe820b6f (patch)
tree8bf25b81a20115dcd9d417428af9ae414ab505c7
parente34b8252a3d2893ca55c82dbfcdaa302fa03d400 (diff)
nfit/ars: Allow root to busy-poll the ARS state machine
The ARS implementation implements exponential back-off on the poll interval to prevent high-frequency access to the DIMM / platform interface. Depending on when the ARS completes the poll interval may exceed the completion event by minutes. Allow root to reset the timeout each time it probes the status. A one-second timeout is still enforced, but root can otherwise can control the poll interval. Fixes: bc6ba8085842 ("nfit, address-range-scrub: rework and simplify ARS...") Cc: <stable@vger.kernel.org> Reported-by: Erwin Tsaur <erwin.tsaur@oracle.com> Reviewed-by: Toshi Kani <toshi.kani@hpe.com> Signed-off-by: Dan Williams <dan.j.williams@intel.com>
-rw-r--r--drivers/acpi/nfit/core.c8
-rw-r--r--drivers/acpi/nfit/nfit.h1
2 files changed, 9 insertions, 0 deletions
diff --git a/drivers/acpi/nfit/core.c b/drivers/acpi/nfit/core.c
index 90312892093e..629cf91649d2 100644
--- a/drivers/acpi/nfit/core.c
+++ b/drivers/acpi/nfit/core.c
@@ -1333,6 +1333,13 @@ static ssize_t scrub_show(struct device *dev,
1333 busy = test_bit(ARS_BUSY, &acpi_desc->scrub_flags) 1333 busy = test_bit(ARS_BUSY, &acpi_desc->scrub_flags)
1334 && !test_bit(ARS_CANCEL, &acpi_desc->scrub_flags); 1334 && !test_bit(ARS_CANCEL, &acpi_desc->scrub_flags);
1335 rc = sprintf(buf, "%d%s", acpi_desc->scrub_count, busy ? "+\n" : "\n"); 1335 rc = sprintf(buf, "%d%s", acpi_desc->scrub_count, busy ? "+\n" : "\n");
1336 /* Allow an admin to poll the busy state at a higher rate */
1337 if (busy && capable(CAP_SYS_RAWIO) && !test_and_set_bit(ARS_POLL,
1338 &acpi_desc->scrub_flags)) {
1339 acpi_desc->scrub_tmo = 1;
1340 mod_delayed_work(nfit_wq, &acpi_desc->dwork, HZ);
1341 }
1342
1336 mutex_unlock(&acpi_desc->init_mutex); 1343 mutex_unlock(&acpi_desc->init_mutex);
1337 device_unlock(dev); 1344 device_unlock(dev);
1338 return rc; 1345 return rc;
@@ -3187,6 +3194,7 @@ static void acpi_nfit_scrub(struct work_struct *work)
3187 else 3194 else
3188 notify_ars_done(acpi_desc); 3195 notify_ars_done(acpi_desc);
3189 memset(acpi_desc->ars_status, 0, acpi_desc->max_ars); 3196 memset(acpi_desc->ars_status, 0, acpi_desc->max_ars);
3197 clear_bit(ARS_POLL, &acpi_desc->scrub_flags);
3190 mutex_unlock(&acpi_desc->init_mutex); 3198 mutex_unlock(&acpi_desc->init_mutex);
3191} 3199}
3192 3200
diff --git a/drivers/acpi/nfit/nfit.h b/drivers/acpi/nfit/nfit.h
index 897ce10192a0..d14bad687fb8 100644
--- a/drivers/acpi/nfit/nfit.h
+++ b/drivers/acpi/nfit/nfit.h
@@ -213,6 +213,7 @@ struct nfit_mem {
213enum scrub_flags { 213enum scrub_flags {
214 ARS_BUSY, 214 ARS_BUSY,
215 ARS_CANCEL, 215 ARS_CANCEL,
216 ARS_POLL,
216}; 217};
217 218
218struct acpi_nfit_desc { 219struct acpi_nfit_desc {