diff options
author | David Flater <dave@flaterco.com> | 2012-08-27 22:25:21 -0400 |
---|---|---|
committer | Takashi Iwai <tiwai@suse.de> | 2012-08-28 13:58:12 -0400 |
commit | 1338fc97d07a04e74a7b75ff28b7751061f4cf38 (patch) | |
tree | a9432867f53befa3ce1e63a11144b8ff4b5b6887 /sound | |
parent | a184d4e4591bb53fd8b91d6e8b85cffa261fb83e (diff) |
ALSA: emu8000: fix emu8000 DRAM sized 512 KiB too small
v2: Fixed result still wrong in the case of 512 KiB DRAM. Oops.
Applicable to 3.5.3 mainline.
In emu8000.c, size_dram determines the amount of memory on the sound card by
doing write/readback tests starting at 512 KiB and incrementing by 512 KiB.
On success, detected_size is updated to the successful address and testing
continues. On failure, the loop is immediately exited. The resulting
detected_size is 512 KiB too small except in two special cases:
1. If there is no memory, the initial 0 value of detected_size is used, which
is correct.
2. If the address space wraps around, detected_size is updated before the
bailout, so the result is correct.
The patch corrects all cases and was tested with an AWE64 Gold. Before:
EMU8000 [0x620]: 3584 Kb on-board memory detected
asfxload 4GMGSMT.SF2 (4174814 B) fails.
After:
EMU8000 [0x620]: 4096 Kb on-board memory detected
asfxload 4GMGSMT.SF2 succeeds.
I do not have a card with 512 KiB to test with, but by forcibly enabling the
added conditional I verified on the AWE64 Gold that it detects 512 KiB
(successfully reading from the first memory location) and does not hang the
card.
C.f. Bug 46451 https://bugzilla.kernel.org/show_bug.cgi?id=46451
Signed-off-by: David Flater <dave@flaterco.com>
Signed-off-by: Takashi Iwai <tiwai@suse.de>
Diffstat (limited to 'sound')
-rw-r--r-- | sound/isa/sb/emu8000.c | 15 |
1 files changed, 12 insertions, 3 deletions
diff --git a/sound/isa/sb/emu8000.c b/sound/isa/sb/emu8000.c index 71887874679c..2aae6a0efbcd 100644 --- a/sound/isa/sb/emu8000.c +++ b/sound/isa/sb/emu8000.c | |||
@@ -417,9 +417,6 @@ size_dram(struct snd_emu8000 *emu) | |||
417 | EMU8000_SMLD_READ(emu); /* discard stale data */ | 417 | EMU8000_SMLD_READ(emu); /* discard stale data */ |
418 | if (EMU8000_SMLD_READ(emu) != UNIQUE_ID2) | 418 | if (EMU8000_SMLD_READ(emu) != UNIQUE_ID2) |
419 | break; /* no memory at this address */ | 419 | break; /* no memory at this address */ |
420 | |||
421 | detected_size = size; | ||
422 | |||
423 | snd_emu8000_read_wait(emu); | 420 | snd_emu8000_read_wait(emu); |
424 | 421 | ||
425 | /* | 422 | /* |
@@ -432,6 +429,18 @@ size_dram(struct snd_emu8000 *emu) | |||
432 | if (EMU8000_SMLD_READ(emu) != UNIQUE_ID1) | 429 | if (EMU8000_SMLD_READ(emu) != UNIQUE_ID1) |
433 | break; /* we must have wrapped around */ | 430 | break; /* we must have wrapped around */ |
434 | snd_emu8000_read_wait(emu); | 431 | snd_emu8000_read_wait(emu); |
432 | |||
433 | /* Otherwise, it's valid memory. */ | ||
434 | detected_size = size + 512 * 1024; | ||
435 | } | ||
436 | |||
437 | /* Distinguish 512 KiB from 0. */ | ||
438 | if (detected_size == 0) { | ||
439 | snd_emu8000_read_wait(emu); | ||
440 | EMU8000_SMALR_WRITE(emu, EMU8000_DRAM_OFFSET); | ||
441 | EMU8000_SMLD_READ(emu); /* discard stale data */ | ||
442 | if (EMU8000_SMLD_READ(emu) == UNIQUE_ID1) | ||
443 | detected_size = 512 * 1024; | ||
435 | } | 444 | } |
436 | 445 | ||
437 | /* wait until FULL bit in SMAxW register is false */ | 446 | /* wait until FULL bit in SMAxW register is false */ |