aboutsummaryrefslogtreecommitdiffstats
path: root/drivers/firmware
diff options
context:
space:
mode:
authorSeiji Aguchi <seiji.aguchi@hds.com>2013-01-11 13:10:05 -0500
committerTony Luck <tony.luck@intel.com>2013-01-11 13:21:56 -0500
commite59310adf5eebce108f78b6c47bb330aae2e1666 (patch)
tree755e6f7348fd37094a856595f4ab33ffc5635ce2 /drivers/firmware
parent9f244e9cfd70c7c0f82d3c92ce772ab2a92d9f64 (diff)
efi_pstore: Avoid deadlock in non-blocking paths
[Issue] There is a scenario which efi_pstore may hang up: - cpuA grabs efivars->lock - cpuB panics and calls smp_send_stop - smp_send_stop sends IRQ to cpuA - after 1 second, cpuB gives up on cpuA and sends an NMI instead - cpuA is now in an NMI handler while still holding efivars->lock - cpuB is deadlocked This case may happen if a firmware has a bug and cpuA is stuck talking with it. [Solution] This patch changes a spin_lock to a spin_trylock in non-blocking paths. and if the spin_lock has already taken by another cpu, it returns without accessing to a firmware to avoid the deadlock. Signed-off-by: Seiji Aguchi <seiji.aguchi@hds.com> Acked-by: Don Zickus <dzickus@redhat.com> Signed-off-by: Tony Luck <tony.luck@intel.com>
Diffstat (limited to 'drivers/firmware')
-rw-r--r--drivers/firmware/efivars.c11
1 files changed, 10 insertions, 1 deletions
diff --git a/drivers/firmware/efivars.c b/drivers/firmware/efivars.c
index 7b1c37497c9a..ef5070d86f88 100644
--- a/drivers/firmware/efivars.c
+++ b/drivers/firmware/efivars.c
@@ -1209,7 +1209,16 @@ static int efi_pstore_write(enum pstore_type_id type,
1209 u64 storage_space, remaining_space, max_variable_size; 1209 u64 storage_space, remaining_space, max_variable_size;
1210 efi_status_t status = EFI_NOT_FOUND; 1210 efi_status_t status = EFI_NOT_FOUND;
1211 1211
1212 spin_lock(&efivars->lock); 1212 if (pstore_cannot_block_path(reason)) {
1213 /*
1214 * If the lock is taken by another cpu in non-blocking path,
1215 * this driver returns without entering firmware to avoid
1216 * hanging up.
1217 */
1218 if (!spin_trylock(&efivars->lock))
1219 return -EBUSY;
1220 } else
1221 spin_lock(&efivars->lock);
1213 1222
1214 /* 1223 /*
1215 * Check if there is a space enough to log. 1224 * Check if there is a space enough to log.