Commit 47fa9613 authored by Russell King's avatar Russell King Committed by Ulf Hansson

mmc: sdhci: avoid walking SG list for writes

If we are writing data to the card, there is no point in walking the
scatterlist to find out if there are any unaligned entries; this is a
needless waste of CPU cycles.  Avoid this by checking for a non-read
tranfer first.
Signed-off-by: default avatarRussell King <rmk+kernel@arm.linux.org.uk>
Signed-off-by: default avatarAdrian Hunter <adrian.hunter@intel.com>
Tested-by: default avatarGregory CLEMENT <gregory.clement@free-electrons.com>
Signed-off-by: default avatarUlf Hansson <ulf.hansson@linaro.org>
parent acc3ad13
...@@ -559,37 +559,39 @@ static void sdhci_adma_table_post(struct sdhci_host *host, ...@@ -559,37 +559,39 @@ static void sdhci_adma_table_post(struct sdhci_host *host,
void *align; void *align;
char *buffer; char *buffer;
unsigned long flags; unsigned long flags;
bool has_unaligned;
if (data->flags & MMC_DATA_READ) if (data->flags & MMC_DATA_READ)
direction = DMA_FROM_DEVICE; direction = DMA_FROM_DEVICE;
else else
direction = DMA_TO_DEVICE; direction = DMA_TO_DEVICE;
/* Do a quick scan of the SG list for any unaligned mappings */ if (data->flags & MMC_DATA_READ) {
has_unaligned = false; bool has_unaligned = false;
for_each_sg(data->sg, sg, host->sg_count, i)
if (sg_dma_address(sg) & SDHCI_ADMA2_MASK) {
has_unaligned = true;
break;
}
if (has_unaligned && data->flags & MMC_DATA_READ) { /* Do a quick scan of the SG list for any unaligned mappings */
dma_sync_sg_for_cpu(mmc_dev(host->mmc), data->sg, for_each_sg(data->sg, sg, host->sg_count, i)
data->sg_len, direction); if (sg_dma_address(sg) & SDHCI_ADMA2_MASK) {
has_unaligned = true;
break;
}
align = host->align_buffer; if (has_unaligned) {
dma_sync_sg_for_cpu(mmc_dev(host->mmc), data->sg,
data->sg_len, direction);
for_each_sg(data->sg, sg, host->sg_count, i) { align = host->align_buffer;
if (sg_dma_address(sg) & SDHCI_ADMA2_MASK) {
size = SDHCI_ADMA2_ALIGN -
(sg_dma_address(sg) & SDHCI_ADMA2_MASK);
buffer = sdhci_kmap_atomic(sg, &flags); for_each_sg(data->sg, sg, host->sg_count, i) {
memcpy(buffer, align, size); if (sg_dma_address(sg) & SDHCI_ADMA2_MASK) {
sdhci_kunmap_atomic(buffer, &flags); size = SDHCI_ADMA2_ALIGN -
(sg_dma_address(sg) & SDHCI_ADMA2_MASK);
align += SDHCI_ADMA2_ALIGN; buffer = sdhci_kmap_atomic(sg, &flags);
memcpy(buffer, align, size);
sdhci_kunmap_atomic(buffer, &flags);
align += SDHCI_ADMA2_ALIGN;
}
} }
} }
} }
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment