Commit 06e8935f authored by Stefan Nilsson XK's avatar Stefan Nilsson XK Committed by Chris Ball

mmc: sdio: optimized SDIO IRQ handling for single irq

If there is only 1 function interrupt registered it is possible to
improve performance by directly calling the irq handler and avoiding
the overhead of reading the CCCR registers.
Signed-off-by: default avatarPer Forlin <per.forlin@linaro.org>
Acked-by: default avatarUlf Hansson <ulf.hansson@stericsson.com>
Reviewed-by: default avatarNicolas Pitre <nicolas.pitre@linaro.org>
Signed-off-by: default avatarChris Ball <cjb@laptop.org>
parent 253d6a28
...@@ -31,6 +31,17 @@ static int process_sdio_pending_irqs(struct mmc_card *card) ...@@ -31,6 +31,17 @@ static int process_sdio_pending_irqs(struct mmc_card *card)
{ {
int i, ret, count; int i, ret, count;
unsigned char pending; unsigned char pending;
struct sdio_func *func;
/*
* Optimization, if there is only 1 function interrupt registered
* call irq handler directly
*/
func = card->sdio_single_irq;
if (func) {
func->irq_handler(func);
return 1;
}
ret = mmc_io_rw_direct(card, 0, 0, SDIO_CCCR_INTx, 0, &pending); ret = mmc_io_rw_direct(card, 0, 0, SDIO_CCCR_INTx, 0, &pending);
if (ret) { if (ret) {
...@@ -42,7 +53,7 @@ static int process_sdio_pending_irqs(struct mmc_card *card) ...@@ -42,7 +53,7 @@ static int process_sdio_pending_irqs(struct mmc_card *card)
count = 0; count = 0;
for (i = 1; i <= 7; i++) { for (i = 1; i <= 7; i++) {
if (pending & (1 << i)) { if (pending & (1 << i)) {
struct sdio_func *func = card->sdio_func[i - 1]; func = card->sdio_func[i - 1];
if (!func) { if (!func) {
printk(KERN_WARNING "%s: pending IRQ for " printk(KERN_WARNING "%s: pending IRQ for "
"non-existent function\n", "non-existent function\n",
...@@ -186,6 +197,24 @@ static int sdio_card_irq_put(struct mmc_card *card) ...@@ -186,6 +197,24 @@ static int sdio_card_irq_put(struct mmc_card *card)
return 0; return 0;
} }
/* If there is only 1 function registered set sdio_single_irq */
static void sdio_single_irq_set(struct mmc_card *card)
{
struct sdio_func *func;
int i;
card->sdio_single_irq = NULL;
if ((card->host->caps & MMC_CAP_SDIO_IRQ) &&
card->host->sdio_irqs == 1)
for (i = 0; i < card->sdio_funcs; i++) {
func = card->sdio_func[i];
if (func && func->irq_handler) {
card->sdio_single_irq = func;
break;
}
}
}
/** /**
* sdio_claim_irq - claim the IRQ for a SDIO function * sdio_claim_irq - claim the IRQ for a SDIO function
* @func: SDIO function * @func: SDIO function
...@@ -227,6 +256,7 @@ int sdio_claim_irq(struct sdio_func *func, sdio_irq_handler_t *handler) ...@@ -227,6 +256,7 @@ int sdio_claim_irq(struct sdio_func *func, sdio_irq_handler_t *handler)
ret = sdio_card_irq_get(func->card); ret = sdio_card_irq_get(func->card);
if (ret) if (ret)
func->irq_handler = NULL; func->irq_handler = NULL;
sdio_single_irq_set(func->card);
return ret; return ret;
} }
...@@ -251,6 +281,7 @@ int sdio_release_irq(struct sdio_func *func) ...@@ -251,6 +281,7 @@ int sdio_release_irq(struct sdio_func *func)
if (func->irq_handler) { if (func->irq_handler) {
func->irq_handler = NULL; func->irq_handler = NULL;
sdio_card_irq_put(func->card); sdio_card_irq_put(func->card);
sdio_single_irq_set(func->card);
} }
ret = mmc_io_rw_direct(func->card, 0, 0, SDIO_CCCR_IENx, 0, &reg); ret = mmc_io_rw_direct(func->card, 0, 0, SDIO_CCCR_IENx, 0, &reg);
......
...@@ -191,6 +191,7 @@ struct mmc_card { ...@@ -191,6 +191,7 @@ struct mmc_card {
struct sdio_cccr cccr; /* common card info */ struct sdio_cccr cccr; /* common card info */
struct sdio_cis cis; /* common tuple info */ struct sdio_cis cis; /* common tuple info */
struct sdio_func *sdio_func[SDIO_MAX_FUNCS]; /* SDIO functions (devices) */ struct sdio_func *sdio_func[SDIO_MAX_FUNCS]; /* SDIO functions (devices) */
struct sdio_func *sdio_single_irq; /* SDIO function when only one IRQ active */
unsigned num_info; /* number of info strings */ unsigned num_info; /* number of info strings */
const char **info; /* info strings */ const char **info; /* info strings */
struct sdio_func_tuple *tuples; /* unknown common tuples */ struct sdio_func_tuple *tuples; /* unknown common tuples */
......
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