mirror of
				https://github.com/torvalds/linux.git
				synced 2025-11-04 10:40:15 +02:00 
			
		
		
		
	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: Per Forlin <per.forlin@linaro.org> Acked-by: Ulf Hansson <ulf.hansson@stericsson.com> Reviewed-by: Nicolas Pitre <nicolas.pitre@linaro.org> Signed-off-by: Chris Ball <cjb@laptop.org>
This commit is contained in:
		
							parent
							
								
									253d6a280f
								
							
						
					
					
						commit
						06e8935feb
					
				
					 2 changed files with 33 additions and 1 deletions
				
			
		| 
						 | 
					@ -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)
 | 
				
			||||||
	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)
 | 
				
			||||||
	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)
 | 
				
			||||||
	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)
 | 
				
			||||||
	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, ®);
 | 
						ret = mmc_io_rw_direct(func->card, 0, 0, SDIO_CCCR_IENx, 0, ®);
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -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 */
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
		Loading…
	
		Reference in a new issue