mirror of
				git://git.openwrt.org/openwrt/openwrt.git
				synced 2025-10-26 11:34:27 -04:00 
			
		
		
		
	
		
			
				
	
	
		
			427 lines
		
	
	
		
			12 KiB
		
	
	
	
		
			Diff
		
	
	
	
	
	
			
		
		
	
	
			427 lines
		
	
	
		
			12 KiB
		
	
	
	
		
			Diff
		
	
	
	
	
	
| Index: linux-2.6.21.7/drivers/mmc/at91_mci.c
 | |
| ===================================================================
 | |
| --- linux-2.6.21.7.orig/drivers/mmc/at91_mci.c
 | |
| +++ linux-2.6.21.7/drivers/mmc/at91_mci.c
 | |
| @@ -79,7 +79,8 @@
 | |
|  
 | |
|  #define DRIVER_NAME "at91_mci"
 | |
|  
 | |
| -#undef	SUPPORT_4WIRE
 | |
| +//#undef	SUPPORT_4WIRE
 | |
| +#define 	SUPPORT_4WIRE
 | |
|  
 | |
|  #define FL_SENT_COMMAND	(1 << 0)
 | |
|  #define FL_SENT_STOP	(1 << 1)
 | |
| @@ -132,7 +133,7 @@ struct at91mci_host
 | |
|  /*
 | |
|   * Copy from sg to a dma block - used for transfers
 | |
|   */
 | |
| -static inline void at91mci_sg_to_dma(struct at91mci_host *host, struct mmc_data *data)
 | |
| +static inline void at91_mci_sg_to_dma(struct at91mci_host *host, struct mmc_data *data)
 | |
|  {
 | |
|  	unsigned int len, i, size;
 | |
|  	unsigned *dmabuf = host->buffer;
 | |
| @@ -181,7 +182,7 @@ static inline void at91mci_sg_to_dma(str
 | |
|  /*
 | |
|   * Prepare a dma read
 | |
|   */
 | |
| -static void at91mci_pre_dma_read(struct at91mci_host *host)
 | |
| +static void at91_mci_pre_dma_read(struct at91mci_host *host)
 | |
|  {
 | |
|  	int i;
 | |
|  	struct scatterlist *sg;
 | |
| @@ -249,23 +250,24 @@ static void at91mci_pre_dma_read(struct 
 | |
|  /*
 | |
|   * Handle after a dma read
 | |
|   */
 | |
| -static void at91mci_post_dma_read(struct at91mci_host *host)
 | |
| +static int at91_mci_post_dma_read(struct at91mci_host *host)
 | |
|  {
 | |
|  	struct mmc_command *cmd;
 | |
|  	struct mmc_data *data;
 | |
| +	int completed = 0;
 | |
|  
 | |
|  	pr_debug("post dma read\n");
 | |
|  
 | |
|  	cmd = host->cmd;
 | |
|  	if (!cmd) {
 | |
|  		pr_debug("no command\n");
 | |
| -		return;
 | |
| +		return 1;
 | |
|  	}
 | |
|  
 | |
|  	data = cmd->data;
 | |
|  	if (!data) {
 | |
|  		pr_debug("no data\n");
 | |
| -		return;
 | |
| +		return 1;
 | |
|  	}
 | |
|  
 | |
|  	while (host->in_use_index < host->transfer_index) {
 | |
| @@ -300,39 +302,14 @@ static void at91mci_post_dma_read(struct
 | |
|  
 | |
|  	/* Is there another transfer to trigger? */
 | |
|  	if (host->transfer_index < data->sg_len)
 | |
| -		at91mci_pre_dma_read(host);
 | |
| +		at91_mci_pre_dma_read(host);
 | |
|  	else {
 | |
| +		at91_mci_write(host, AT91_MCI_IDR, AT91_MCI_ENDRX);
 | |
|  		at91_mci_write(host, AT91_MCI_IER, AT91_MCI_RXBUFF);
 | |
| -		at91_mci_write(host, ATMEL_PDC_PTCR, ATMEL_PDC_RXTDIS | ATMEL_PDC_TXTDIS);
 | |
|  	}
 | |
|  
 | |
|  	pr_debug("post dma read done\n");
 | |
| -}
 | |
| -
 | |
| -/*
 | |
| - * Handle transmitted data
 | |
| - */
 | |
| -static void at91_mci_handle_transmitted(struct at91mci_host *host)
 | |
| -{
 | |
| -	struct mmc_command *cmd;
 | |
| -	struct mmc_data *data;
 | |
| -
 | |
| -	pr_debug("Handling the transmit\n");
 | |
| -
 | |
| -	/* Disable the transfer */
 | |
| -	at91_mci_write(host, ATMEL_PDC_PTCR, ATMEL_PDC_RXTDIS | ATMEL_PDC_TXTDIS);
 | |
| -
 | |
| -	/* Now wait for cmd ready */
 | |
| -	at91_mci_write(host, AT91_MCI_IDR, AT91_MCI_TXBUFE);
 | |
| -	at91_mci_write(host, AT91_MCI_IER, AT91_MCI_NOTBUSY);
 | |
| -
 | |
| -	cmd = host->cmd;
 | |
| -	if (!cmd) return;
 | |
| -
 | |
| -	data = cmd->data;
 | |
| -	if (!data) return;
 | |
| -
 | |
| -	data->bytes_xfered = host->total_length;
 | |
| +	return completed;
 | |
|  }
 | |
|  
 | |
|  /*
 | |
| @@ -340,10 +317,17 @@ static void at91_mci_handle_transmitted(
 | |
|   */
 | |
|  static void at91_mci_enable(struct at91mci_host *host)
 | |
|  {
 | |
| +	unsigned int mr;
 | |
| +
 | |
|  	at91_mci_write(host, AT91_MCI_CR, AT91_MCI_MCIEN);
 | |
|  	at91_mci_write(host, AT91_MCI_IDR, 0xffffffff);
 | |
|  	at91_mci_write(host, AT91_MCI_DTOR, AT91_MCI_DTOMUL_1M | AT91_MCI_DTOCYC);
 | |
| -	at91_mci_write(host, AT91_MCI_MR, AT91_MCI_PDCMODE | 0x34a);
 | |
| +	mr = AT91_MCI_PDCMODE | 0x34a;
 | |
| +
 | |
| +	if (cpu_is_at91sam9260() || cpu_is_at91sam9263())
 | |
| +		mr |= AT91_MCI_RDPROOF | AT91_MCI_WRPROOF;
 | |
| +
 | |
| +	at91_mci_write(host, AT91_MCI_MR, mr);
 | |
|  
 | |
|  	/* use Slot A or B (only one at same time) */
 | |
|  	at91_mci_write(host, AT91_MCI_SDCR, host->board->slot_b);
 | |
| @@ -359,9 +343,8 @@ static void at91_mci_disable(struct at91
 | |
|  
 | |
|  /*
 | |
|   * Send a command
 | |
| - * return the interrupts to enable
 | |
|   */
 | |
| -static unsigned int at91_mci_send_command(struct at91mci_host *host, struct mmc_command *cmd)
 | |
| +static void at91_mci_send_command(struct at91mci_host *host, struct mmc_command *cmd)
 | |
|  {
 | |
|  	unsigned int cmdr, mr;
 | |
|  	unsigned int block_length;
 | |
| @@ -372,8 +355,7 @@ static unsigned int at91_mci_send_comman
 | |
|  
 | |
|  	host->cmd = cmd;
 | |
|  
 | |
| -	/* Not sure if this is needed */
 | |
| -#if 0
 | |
| +	/* Needed for leaving busy state before CMD1 */
 | |
|  	if ((at91_mci_read(host, AT91_MCI_SR) & AT91_MCI_RTOE) && (cmd->opcode == 1)) {
 | |
|  		pr_debug("Clearing timeout\n");
 | |
|  		at91_mci_write(host, AT91_MCI_ARGR, 0);
 | |
| @@ -383,7 +365,7 @@ static unsigned int at91_mci_send_comman
 | |
|  			pr_debug("Clearing: SR = %08X\n", at91_mci_read(host, AT91_MCI_SR));
 | |
|  		}
 | |
|  	}
 | |
| -#endif
 | |
| +
 | |
|  	cmdr = cmd->opcode;
 | |
|  
 | |
|  	if (mmc_resp_type(cmd) == MMC_RSP_NONE)
 | |
| @@ -440,50 +422,48 @@ static unsigned int at91_mci_send_comman
 | |
|  		at91_mci_write(host, ATMEL_PDC_TCR, 0);
 | |
|  		at91_mci_write(host, ATMEL_PDC_TNPR, 0);
 | |
|  		at91_mci_write(host, ATMEL_PDC_TNCR, 0);
 | |
| +		ier = AT91_MCI_CMDRDY;
 | |
| +	} else {
 | |
| +		/* zero block length in PDC mode */
 | |
| +		mr = at91_mci_read(host, AT91_MCI_MR) & 0x7fff;
 | |
| +		at91_mci_write(host, AT91_MCI_MR, mr | (block_length << 16) | AT91_MCI_PDCMODE);
 | |
| +
 | |
| +		/*
 | |
| +		 * Disable the PDC controller
 | |
| +		 */
 | |
| +		at91_mci_write(host, ATMEL_PDC_PTCR, ATMEL_PDC_RXTDIS | ATMEL_PDC_TXTDIS);
 | |
|  
 | |
| -		at91_mci_write(host, AT91_MCI_ARGR, cmd->arg);
 | |
| -		at91_mci_write(host, AT91_MCI_CMDR, cmdr);
 | |
| -		return AT91_MCI_CMDRDY;
 | |
| -	}
 | |
| -
 | |
| -	mr = at91_mci_read(host, AT91_MCI_MR) & 0x7fff;	/* zero block length and PDC mode */
 | |
| -	at91_mci_write(host, AT91_MCI_MR, mr | (block_length << 16) | AT91_MCI_PDCMODE);
 | |
| -
 | |
| -	/*
 | |
| -	 * Disable the PDC controller
 | |
| -	 */
 | |
| -	at91_mci_write(host, ATMEL_PDC_PTCR, ATMEL_PDC_RXTDIS | ATMEL_PDC_TXTDIS);
 | |
| -
 | |
| -	if (cmdr & AT91_MCI_TRCMD_START) {
 | |
| -		data->bytes_xfered = 0;
 | |
| -		host->transfer_index = 0;
 | |
| -		host->in_use_index = 0;
 | |
| -		if (cmdr & AT91_MCI_TRDIR) {
 | |
| -			/*
 | |
| -			 * Handle a read
 | |
| -			 */
 | |
| -			host->buffer = NULL;
 | |
| -			host->total_length = 0;
 | |
| +		if (cmdr & AT91_MCI_TRCMD_START) {
 | |
| +			data->bytes_xfered = 0;
 | |
| +			host->transfer_index = 0;
 | |
| +			host->in_use_index = 0;
 | |
| +			if (cmdr & AT91_MCI_TRDIR) {
 | |
| +				/*
 | |
| +				 * Handle a read
 | |
| +				 */
 | |
| +				host->buffer = NULL;
 | |
| +				host->total_length = 0;
 | |
|  
 | |
| -			at91mci_pre_dma_read(host);
 | |
| -			ier = AT91_MCI_ENDRX /* | AT91_MCI_RXBUFF */;
 | |
| -		}
 | |
| -		else {
 | |
| -			/*
 | |
| -			 * Handle a write
 | |
| -			 */
 | |
| -			host->total_length = block_length * blocks;
 | |
| -			host->buffer = dma_alloc_coherent(NULL,
 | |
| -						  host->total_length,
 | |
| -						  &host->physical_address, GFP_KERNEL);
 | |
| -
 | |
| -			at91mci_sg_to_dma(host, data);
 | |
| -
 | |
| -			pr_debug("Transmitting %d bytes\n", host->total_length);
 | |
| -
 | |
| -			at91_mci_write(host, ATMEL_PDC_TPR, host->physical_address);
 | |
| -			at91_mci_write(host, ATMEL_PDC_TCR, host->total_length / 4);
 | |
| -			ier = AT91_MCI_TXBUFE;
 | |
| +				at91_mci_pre_dma_read(host);
 | |
| +				ier = AT91_MCI_ENDRX /* | AT91_MCI_RXBUFF */;
 | |
| +			}
 | |
| +			else {
 | |
| +				/*
 | |
| +			 	* Handle a write
 | |
| +			 	*/
 | |
| +				host->total_length = block_length * blocks;
 | |
| +				host->buffer = dma_alloc_coherent(NULL,
 | |
| +							host->total_length,
 | |
| +							&host->physical_address, GFP_KERNEL);
 | |
| +
 | |
| +				at91_mci_sg_to_dma(host, data);
 | |
| +
 | |
| +				pr_debug("Transmitting %d bytes\n", host->total_length);
 | |
| +
 | |
| +				at91_mci_write(host, ATMEL_PDC_TPR, host->physical_address);
 | |
| +				at91_mci_write(host, ATMEL_PDC_TCR, host->total_length / 4);
 | |
| +				ier = AT91_MCI_CMDRDY;
 | |
| +			}
 | |
|  		}
 | |
|  	}
 | |
|  
 | |
| @@ -498,39 +478,24 @@ static unsigned int at91_mci_send_comman
 | |
|  	if (cmdr & AT91_MCI_TRCMD_START) {
 | |
|  		if (cmdr & AT91_MCI_TRDIR)
 | |
|  			at91_mci_write(host, ATMEL_PDC_PTCR, ATMEL_PDC_RXTEN);
 | |
| -		else
 | |
| -			at91_mci_write(host, ATMEL_PDC_PTCR, ATMEL_PDC_TXTEN);
 | |
|  	}
 | |
| -	return ier;
 | |
| -}
 | |
|  
 | |
| -/*
 | |
| - * Wait for a command to complete
 | |
| - */
 | |
| -static void at91mci_process_command(struct at91mci_host *host, struct mmc_command *cmd)
 | |
| -{
 | |
| -	unsigned int ier;
 | |
| -
 | |
| -	ier = at91_mci_send_command(host, cmd);
 | |
| -
 | |
| -	pr_debug("setting ier to %08X\n", ier);
 | |
| -
 | |
| -	/* Stop on errors or the required value */
 | |
| +	/* Enable selected interrupts */
 | |
|  	at91_mci_write(host, AT91_MCI_IER, AT91_MCI_ERRORS | ier);
 | |
|  }
 | |
|  
 | |
|  /*
 | |
|   * Process the next step in the request
 | |
|   */
 | |
| -static void at91mci_process_next(struct at91mci_host *host)
 | |
| +static void at91_mci_process_next(struct at91mci_host *host)
 | |
|  {
 | |
|  	if (!(host->flags & FL_SENT_COMMAND)) {
 | |
|  		host->flags |= FL_SENT_COMMAND;
 | |
| -		at91mci_process_command(host, host->request->cmd);
 | |
| +		at91_mci_send_command(host, host->request->cmd);
 | |
|  	}
 | |
|  	else if ((!(host->flags & FL_SENT_STOP)) && host->request->stop) {
 | |
|  		host->flags |= FL_SENT_STOP;
 | |
| -		at91mci_process_command(host, host->request->stop);
 | |
| +		at91_mci_send_command(host, host->request->stop);
 | |
|  	}
 | |
|  	else
 | |
|  		mmc_request_done(host->mmc, host->request);
 | |
| @@ -539,7 +504,7 @@ static void at91mci_process_next(struct 
 | |
|  /*
 | |
|   * Handle a command that has been completed
 | |
|   */
 | |
| -static void at91mci_completed_command(struct at91mci_host *host)
 | |
| +static void at91_mci_completed_command(struct at91mci_host *host)
 | |
|  {
 | |
|  	struct mmc_command *cmd = host->cmd;
 | |
|  	unsigned int status;
 | |
| @@ -583,7 +548,7 @@ static void at91mci_completed_command(st
 | |
|  	else
 | |
|  		cmd->error = MMC_ERR_NONE;
 | |
|  
 | |
| -	at91mci_process_next(host);
 | |
| +	at91_mci_process_next(host);
 | |
|  }
 | |
|  
 | |
|  /*
 | |
| @@ -595,7 +560,60 @@ static void at91_mci_request(struct mmc_
 | |
|  	host->request = mrq;
 | |
|  	host->flags = 0;
 | |
|  
 | |
| -	at91mci_process_next(host);
 | |
| +	at91_mci_process_next(host);
 | |
| +}
 | |
| +
 | |
| +/*
 | |
| + * Handle transmitted data
 | |
| + */
 | |
| +static void at91_mci_handle_transmitted(struct at91mci_host *host)
 | |
| +{
 | |
| +	struct mmc_command *cmd;
 | |
| +	struct mmc_data *data;
 | |
| +
 | |
| +	pr_debug("Handling the transmit\n");
 | |
| +
 | |
| +	/* Disable the transfer */
 | |
| +	at91_mci_write(host, ATMEL_PDC_PTCR, ATMEL_PDC_RXTDIS | ATMEL_PDC_TXTDIS);
 | |
| +
 | |
| +	/* Now wait for cmd ready */
 | |
| +	at91_mci_write(host, AT91_MCI_IDR, AT91_MCI_TXBUFE);
 | |
| +
 | |
| +	cmd = host->cmd;
 | |
| +	if (!cmd) return;
 | |
| +
 | |
| +	data = cmd->data;
 | |
| +	if (!data) return;
 | |
| +
 | |
| +	if (cmd->opcode == MMC_WRITE_MULTIPLE_BLOCK) {
 | |
| +		pr_debug("multiple write : wait for BLKE...\n");
 | |
| +		at91_mci_write(host, AT91_MCI_IER, AT91_MCI_BLKE);
 | |
| +	} else
 | |
| +		at91_mci_write(host, AT91_MCI_IER, AT91_MCI_NOTBUSY);
 | |
| +
 | |
| +	data->bytes_xfered = host->total_length;
 | |
| +}
 | |
| +
 | |
| +
 | |
| +/*Handle after command sent ready*/
 | |
| +static int at91_mci_handle_cmdrdy(struct at91mci_host *host)
 | |
| +{
 | |
| +	if (!host->cmd)
 | |
| +		return 1;
 | |
| +	else if (!host->cmd->data) {
 | |
| +		if (host->flags & FL_SENT_STOP) {
 | |
| +			/*After multi block write, we mus wait for NOTBUSY*/
 | |
| +			at91_mci_write(host, AT91_MCI_IER, AT91_MCI_NOTBUSY);
 | |
| +		} else return 1;
 | |
| +	} else if (host->cmd->data->flags & MMC_DATA_WRITE) {
 | |
| +		/*After sending multi-block-write command, start DMA transfer*/
 | |
| +		at91_mci_write(host, AT91_MCI_IER, AT91_MCI_TXBUFE);
 | |
| +		at91_mci_write(host, AT91_MCI_IER, AT91_MCI_BLKE);
 | |
| +		at91_mci_write(host, ATMEL_PDC_PTCR, ATMEL_PDC_TXTEN);
 | |
| +	}
 | |
| +
 | |
| +	/* command not completed, have to wait */
 | |
| +	return 0;
 | |
|  }
 | |
|  
 | |
|  /*
 | |
| @@ -698,29 +716,33 @@ static irqreturn_t at91_mci_irq(int irq,
 | |
|  			at91_mci_handle_transmitted(host);
 | |
|  		}
 | |
|  
 | |
| +		if (int_status & AT91_MCI_ENDRX) {
 | |
| +			pr_debug("ENDRX\n");
 | |
| +			at91_mci_post_dma_read(host);
 | |
| +		}
 | |
| +
 | |
|  		if (int_status & AT91_MCI_RXBUFF) {
 | |
|  			pr_debug("RX buffer full\n");
 | |
| -			at91_mci_write(host, AT91_MCI_IER, AT91_MCI_CMDRDY);
 | |
| +			at91_mci_write(host, ATMEL_PDC_PTCR, ATMEL_PDC_RXTDIS | ATMEL_PDC_TXTDIS);
 | |
| +			at91_mci_write(host, AT91_MCI_IDR, AT91_MCI_RXBUFF | AT91_MCI_ENDRX);
 | |
| +			completed = 1;
 | |
|  		}
 | |
|  
 | |
|  		if (int_status & AT91_MCI_ENDTX)
 | |
|  			pr_debug("Transmit has ended\n");
 | |
|  
 | |
| -		if (int_status & AT91_MCI_ENDRX) {
 | |
| -			pr_debug("Receive has ended\n");
 | |
| -			at91mci_post_dma_read(host);
 | |
| -		}
 | |
| -
 | |
|  		if (int_status & AT91_MCI_NOTBUSY) {
 | |
|  			pr_debug("Card is ready\n");
 | |
| -			at91_mci_write(host, AT91_MCI_IER, AT91_MCI_CMDRDY);
 | |
| +			completed = 1;
 | |
|  		}
 | |
|  
 | |
|  		if (int_status & AT91_MCI_DTIP)
 | |
|  			pr_debug("Data transfer in progress\n");
 | |
|  
 | |
| -		if (int_status & AT91_MCI_BLKE)
 | |
| +		if (int_status & AT91_MCI_BLKE) {
 | |
|  			pr_debug("Block transfer has ended\n");
 | |
| +			completed = 1;
 | |
| +		}
 | |
|  
 | |
|  		if (int_status & AT91_MCI_TXRDY)
 | |
|  			pr_debug("Ready to transmit\n");
 | |
| @@ -730,14 +752,14 @@ static irqreturn_t at91_mci_irq(int irq,
 | |
|  
 | |
|  		if (int_status & AT91_MCI_CMDRDY) {
 | |
|  			pr_debug("Command ready\n");
 | |
| -			completed = 1;
 | |
| +			completed = at91_mci_handle_cmdrdy(host);
 | |
|  		}
 | |
|  	}
 | |
|  
 | |
|  	if (completed) {
 | |
|  		pr_debug("Completed command\n");
 | |
|  		at91_mci_write(host, AT91_MCI_IDR, 0xffffffff);
 | |
| -		at91mci_completed_command(host);
 | |
| +		at91_mci_completed_command(host);
 | |
|  	} else
 | |
|  		at91_mci_write(host, AT91_MCI_IDR, int_status);
 | |
|  
 |