diff options
author | Adrian Hunter <adrian.hunter@nokia.com> | 2009-09-22 19:44:37 -0400 |
---|---|---|
committer | Linus Torvalds <torvalds@linux-foundation.org> | 2009-09-23 10:39:34 -0400 |
commit | ef0b27d4ccacac32afc3d1c0e8a95e4091dfbc8c (patch) | |
tree | ed507b7f505620915f96df8757f36c78fd24c6b6 /drivers/mmc/core/mmc_ops.c | |
parent | 53509f0fe28e049e772897aa8fa1f5183b6823a2 (diff) |
mmc: check status after MMC SWITCH command
According to the standard, the SWITCH command should be followed by a
SEND_STATUS command to check for errors.
Signed-off-by: Adrian Hunter <adrian.hunter@nokia.com>
Acked-by: Matt Fleming <matt@console-pimps.org>
Cc: Ian Molton <ian@mnementh.co.uk>
Cc: "Roberto A. Foglietta" <roberto.foglietta@gmail.com>
Cc: Jarkko Lavinen <jarkko.lavinen@nokia.com>
Cc: Denis Karpov <ext-denis.2.karpov@nokia.com>
Cc: Pierre Ossman <pierre@ossman.eu>
Cc: Philip Langdale <philipl@overt.org>
Cc: "Madhusudhan" <madhu.cr@ti.com>
Cc: <linux-mmc@vger.kernel.org>
Signed-off-by: Andrew Morton <akpm@linux-foundation.org>
Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
Diffstat (limited to 'drivers/mmc/core/mmc_ops.c')
-rw-r--r-- | drivers/mmc/core/mmc_ops.c | 23 |
1 files changed, 23 insertions, 0 deletions
diff --git a/drivers/mmc/core/mmc_ops.c b/drivers/mmc/core/mmc_ops.c index 355c6042cf65..d2cb5c634392 100644 --- a/drivers/mmc/core/mmc_ops.c +++ b/drivers/mmc/core/mmc_ops.c | |||
@@ -390,6 +390,7 @@ int mmc_switch(struct mmc_card *card, u8 set, u8 index, u8 value) | |||
390 | { | 390 | { |
391 | int err; | 391 | int err; |
392 | struct mmc_command cmd; | 392 | struct mmc_command cmd; |
393 | u32 status; | ||
393 | 394 | ||
394 | BUG_ON(!card); | 395 | BUG_ON(!card); |
395 | BUG_ON(!card->host); | 396 | BUG_ON(!card->host); |
@@ -407,6 +408,28 @@ int mmc_switch(struct mmc_card *card, u8 set, u8 index, u8 value) | |||
407 | if (err) | 408 | if (err) |
408 | return err; | 409 | return err; |
409 | 410 | ||
411 | /* Must check status to be sure of no errors */ | ||
412 | do { | ||
413 | err = mmc_send_status(card, &status); | ||
414 | if (err) | ||
415 | return err; | ||
416 | if (card->host->caps & MMC_CAP_WAIT_WHILE_BUSY) | ||
417 | break; | ||
418 | if (mmc_host_is_spi(card->host)) | ||
419 | break; | ||
420 | } while (R1_CURRENT_STATE(status) == 7); | ||
421 | |||
422 | if (mmc_host_is_spi(card->host)) { | ||
423 | if (status & R1_SPI_ILLEGAL_COMMAND) | ||
424 | return -EBADMSG; | ||
425 | } else { | ||
426 | if (status & 0xFDFFA000) | ||
427 | printk(KERN_WARNING "%s: unexpected status %#x after " | ||
428 | "switch", mmc_hostname(card->host), status); | ||
429 | if (status & R1_SWITCH_ERROR) | ||
430 | return -EBADMSG; | ||
431 | } | ||
432 | |||
410 | return 0; | 433 | return 0; |
411 | } | 434 | } |
412 | 435 | ||