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; | ||||
| 	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); | ||||
| 	if (ret) { | ||||
|  | @ -42,7 +53,7 @@ static int process_sdio_pending_irqs(struct mmc_card *card) | |||
| 	count = 0; | ||||
| 	for (i = 1; i <= 7; i++) { | ||||
| 		if (pending & (1 << i)) { | ||||
| 			struct sdio_func *func = card->sdio_func[i - 1]; | ||||
| 			func = card->sdio_func[i - 1]; | ||||
| 			if (!func) { | ||||
| 				printk(KERN_WARNING "%s: pending IRQ for " | ||||
| 					"non-existent function\n", | ||||
|  | @ -186,6 +197,24 @@ static int sdio_card_irq_put(struct mmc_card *card) | |||
| 	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 | ||||
|  *	@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); | ||||
| 	if (ret) | ||||
| 		func->irq_handler = NULL; | ||||
| 	sdio_single_irq_set(func->card); | ||||
| 
 | ||||
| 	return ret; | ||||
| } | ||||
|  | @ -251,6 +281,7 @@ int sdio_release_irq(struct sdio_func *func) | |||
| 	if (func->irq_handler) { | ||||
| 		func->irq_handler = NULL; | ||||
| 		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, ®); | ||||
|  |  | |||
|  | @ -191,6 +191,7 @@ struct mmc_card { | |||
| 	struct sdio_cccr	cccr;		/* common card info */ | ||||
| 	struct sdio_cis		cis;		/* common tuple info */ | ||||
| 	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 */ | ||||
| 	const char		**info;		/* info strings */ | ||||
| 	struct sdio_func_tuple	*tuples;	/* unknown common tuples */ | ||||
|  |  | |||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue
	
	 Stefan Nilsson XK
				Stefan Nilsson XK