| 
									
										
										
										
											2008-02-06 04:24:09 +11:00
										 |  |  | /*****************************************************************************
 | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  *     Author: Xilinx, Inc. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  *     This program is free software; you can redistribute it and/or modify it | 
					
						
							|  |  |  |  *     under the terms of the GNU General Public License as published by the | 
					
						
							|  |  |  |  *     Free Software Foundation; either version 2 of the License, or (at your | 
					
						
							|  |  |  |  *     option) any later version. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  *     XILINX IS PROVIDING THIS DESIGN, CODE, OR INFORMATION "AS IS" | 
					
						
							|  |  |  |  *     AS A COURTESY TO YOU, SOLELY FOR USE IN DEVELOPING PROGRAMS AND | 
					
						
							|  |  |  |  *     SOLUTIONS FOR XILINX DEVICES.  BY PROVIDING THIS DESIGN, CODE, | 
					
						
							|  |  |  |  *     OR INFORMATION AS ONE POSSIBLE IMPLEMENTATION OF THIS FEATURE, | 
					
						
							|  |  |  |  *     APPLICATION OR STANDARD, XILINX IS MAKING NO REPRESENTATION | 
					
						
							|  |  |  |  *     THAT THIS IMPLEMENTATION IS FREE FROM ANY CLAIMS OF INFRINGEMENT, | 
					
						
							|  |  |  |  *     AND YOU ARE RESPONSIBLE FOR OBTAINING ANY RIGHTS YOU MAY REQUIRE | 
					
						
							|  |  |  |  *     FOR YOUR IMPLEMENTATION.  XILINX EXPRESSLY DISCLAIMS ANY | 
					
						
							|  |  |  |  *     WARRANTY WHATSOEVER WITH RESPECT TO THE ADEQUACY OF THE | 
					
						
							|  |  |  |  *     IMPLEMENTATION, INCLUDING BUT NOT LIMITED TO ANY WARRANTIES OR | 
					
						
							|  |  |  |  *     REPRESENTATIONS THAT THIS IMPLEMENTATION IS FREE FROM CLAIMS OF | 
					
						
							|  |  |  |  *     INFRINGEMENT, IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS | 
					
						
							|  |  |  |  *     FOR A PARTICULAR PURPOSE. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  *     (c) Copyright 2003-2008 Xilinx Inc. | 
					
						
							|  |  |  |  *     All rights reserved. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  *     You should have received a copy of the GNU General Public License along | 
					
						
							|  |  |  |  *     with this program; if not, write to the Free Software Foundation, Inc., | 
					
						
							|  |  |  |  *     675 Mass Ave, Cambridge, MA 02139, USA. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  *****************************************************************************/ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "buffer_icap.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* Indicates how many bytes will fit in a buffer. (1 BRAM) */ | 
					
						
							|  |  |  | #define XHI_MAX_BUFFER_BYTES        2048
 | 
					
						
							|  |  |  | #define XHI_MAX_BUFFER_INTS         (XHI_MAX_BUFFER_BYTES >> 2)
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* File access and error constants */ | 
					
						
							|  |  |  | #define XHI_DEVICE_READ_ERROR       -1
 | 
					
						
							|  |  |  | #define XHI_DEVICE_WRITE_ERROR      -2
 | 
					
						
							|  |  |  | #define XHI_BUFFER_OVERFLOW_ERROR   -3
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #define XHI_DEVICE_READ             0x1
 | 
					
						
							|  |  |  | #define XHI_DEVICE_WRITE            0x0
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* Constants for checking transfer status */ | 
					
						
							|  |  |  | #define XHI_CYCLE_DONE              0
 | 
					
						
							|  |  |  | #define XHI_CYCLE_EXECUTING         1
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* buffer_icap register offsets */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* Size of transfer, read & write */ | 
					
						
							|  |  |  | #define XHI_SIZE_REG_OFFSET        0x800L
 | 
					
						
							|  |  |  | /* offset into bram, read & write */ | 
					
						
							|  |  |  | #define XHI_BRAM_OFFSET_REG_OFFSET 0x804L
 | 
					
						
							|  |  |  | /* Read not Configure, direction of transfer.  Write only */ | 
					
						
							|  |  |  | #define XHI_RNC_REG_OFFSET         0x808L
 | 
					
						
							|  |  |  | /* Indicates transfer complete. Read only */ | 
					
						
							|  |  |  | #define XHI_STATUS_REG_OFFSET      0x80CL
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* Constants for setting the RNC register */ | 
					
						
							|  |  |  | #define XHI_CONFIGURE              0x0UL
 | 
					
						
							|  |  |  | #define XHI_READBACK               0x1UL
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* Constants for the Done register */ | 
					
						
							|  |  |  | #define XHI_NOT_FINISHED           0x0UL
 | 
					
						
							|  |  |  | #define XHI_FINISHED               0x1UL
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #define XHI_BUFFER_START 0
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /**
 | 
					
						
							| 
									
										
										
										
											2008-02-25 10:34:47 +11:00
										 |  |  |  * buffer_icap_get_status - Get the contents of the status register. | 
					
						
							| 
									
										
										
										
											2008-03-18 04:36:30 +11:00
										 |  |  |  * @drvdata: a pointer to the drvdata. | 
					
						
							| 
									
										
										
										
											2008-02-06 04:24:09 +11:00
										 |  |  |  * | 
					
						
							|  |  |  |  * The status register contains the ICAP status and the done bit. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * D8 - cfgerr | 
					
						
							|  |  |  |  * D7 - dalign | 
					
						
							|  |  |  |  * D6 - rip | 
					
						
							|  |  |  |  * D5 - in_abort_l | 
					
						
							|  |  |  |  * D4 - Always 1 | 
					
						
							|  |  |  |  * D3 - Always 1 | 
					
						
							|  |  |  |  * D2 - Always 1 | 
					
						
							|  |  |  |  * D1 - Always 1 | 
					
						
							|  |  |  |  * D0 - Done bit | 
					
						
							|  |  |  |  **/ | 
					
						
							| 
									
										
										
										
											2008-03-18 04:36:30 +11:00
										 |  |  | u32 buffer_icap_get_status(struct hwicap_drvdata *drvdata) | 
					
						
							| 
									
										
										
										
											2008-02-06 04:24:09 +11:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2008-03-18 04:36:30 +11:00
										 |  |  | 	return in_be32(drvdata->base_address + XHI_STATUS_REG_OFFSET); | 
					
						
							| 
									
										
										
										
											2008-02-06 04:24:09 +11:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /**
 | 
					
						
							| 
									
										
										
										
											2008-02-25 10:34:47 +11:00
										 |  |  |  * buffer_icap_get_bram - Reads data from the storage buffer bram. | 
					
						
							|  |  |  |  * @base_address: contains the base address of the component. | 
					
						
							|  |  |  |  * @offset: The word offset from which the data should be read. | 
					
						
							| 
									
										
										
										
											2008-02-06 04:24:09 +11:00
										 |  |  |  * | 
					
						
							|  |  |  |  * A bram is used as a configuration memory cache.  One frame of data can | 
					
						
							|  |  |  |  * be stored in this "storage buffer". | 
					
						
							|  |  |  |  **/ | 
					
						
							|  |  |  | static inline u32 buffer_icap_get_bram(void __iomem *base_address, | 
					
						
							|  |  |  | 		u32 offset) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	return in_be32(base_address + (offset << 2)); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /**
 | 
					
						
							| 
									
										
										
										
											2008-02-25 10:34:47 +11:00
										 |  |  |  * buffer_icap_busy - Return true if the icap device is busy | 
					
						
							|  |  |  |  * @base_address: is the base address of the device | 
					
						
							| 
									
										
										
										
											2008-02-06 04:24:09 +11:00
										 |  |  |  * | 
					
						
							|  |  |  |  * The queries the low order bit of the status register, which | 
					
						
							|  |  |  |  * indicates whether the current configuration or readback operation | 
					
						
							|  |  |  |  * has completed. | 
					
						
							|  |  |  |  **/ | 
					
						
							|  |  |  | static inline bool buffer_icap_busy(void __iomem *base_address) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2008-03-18 04:36:30 +11:00
										 |  |  | 	u32 status = in_be32(base_address + XHI_STATUS_REG_OFFSET); | 
					
						
							|  |  |  | 	return (status & 1) == XHI_NOT_FINISHED; | 
					
						
							| 
									
										
										
										
											2008-02-06 04:24:09 +11:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /**
 | 
					
						
							| 
									
										
										
										
											2008-02-25 10:34:47 +11:00
										 |  |  |  * buffer_icap_set_size - Set the size register. | 
					
						
							|  |  |  |  * @base_address: is the base address of the device | 
					
						
							|  |  |  |  * @data: The size in bytes. | 
					
						
							| 
									
										
										
										
											2008-02-06 04:24:09 +11:00
										 |  |  |  * | 
					
						
							|  |  |  |  * The size register holds the number of 8 bit bytes to transfer between | 
					
						
							|  |  |  |  * bram and the icap (or icap to bram). | 
					
						
							|  |  |  |  **/ | 
					
						
							|  |  |  | static inline void buffer_icap_set_size(void __iomem *base_address, | 
					
						
							|  |  |  | 		u32 data) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	out_be32(base_address + XHI_SIZE_REG_OFFSET, data); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /**
 | 
					
						
							| 
									
										
										
										
											2008-02-25 10:34:47 +11:00
										 |  |  |  * buffer_icap_set_offset - Set the bram offset register. | 
					
						
							|  |  |  |  * @base_address: contains the base address of the device. | 
					
						
							|  |  |  |  * @data: is the value to be written to the data register. | 
					
						
							| 
									
										
										
										
											2008-02-06 04:24:09 +11:00
										 |  |  |  * | 
					
						
							|  |  |  |  * The bram offset register holds the starting bram address to transfer | 
					
						
							|  |  |  |  * data from during configuration or write data to during readback. | 
					
						
							|  |  |  |  **/ | 
					
						
							|  |  |  | static inline void buffer_icap_set_offset(void __iomem *base_address, | 
					
						
							|  |  |  | 		u32 data) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	out_be32(base_address + XHI_BRAM_OFFSET_REG_OFFSET, data); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /**
 | 
					
						
							| 
									
										
										
										
											2008-02-25 10:34:47 +11:00
										 |  |  |  * buffer_icap_set_rnc - Set the RNC (Readback not Configure) register. | 
					
						
							|  |  |  |  * @base_address: contains the base address of the device. | 
					
						
							|  |  |  |  * @data: is the value to be written to the data register. | 
					
						
							| 
									
										
										
										
											2008-02-06 04:24:09 +11:00
										 |  |  |  * | 
					
						
							|  |  |  |  * The RNC register determines the direction of the data transfer.  It | 
					
						
							|  |  |  |  * controls whether a configuration or readback take place.  Writing to | 
					
						
							|  |  |  |  * this register initiates the transfer.  A value of 1 initiates a | 
					
						
							|  |  |  |  * readback while writing a value of 0 initiates a configuration. | 
					
						
							|  |  |  |  **/ | 
					
						
							|  |  |  | static inline void buffer_icap_set_rnc(void __iomem *base_address, | 
					
						
							|  |  |  | 		u32 data) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	out_be32(base_address + XHI_RNC_REG_OFFSET, data); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /**
 | 
					
						
							| 
									
										
										
										
											2008-02-25 10:34:47 +11:00
										 |  |  |  * buffer_icap_set_bram - Write data to the storage buffer bram. | 
					
						
							|  |  |  |  * @base_address: contains the base address of the component. | 
					
						
							|  |  |  |  * @offset: The word offset at which the data should be written. | 
					
						
							|  |  |  |  * @data: The value to be written to the bram offset. | 
					
						
							| 
									
										
										
										
											2008-02-06 04:24:09 +11:00
										 |  |  |  * | 
					
						
							|  |  |  |  * A bram is used as a configuration memory cache.  One frame of data can | 
					
						
							|  |  |  |  * be stored in this "storage buffer". | 
					
						
							|  |  |  |  **/ | 
					
						
							|  |  |  | static inline void buffer_icap_set_bram(void __iomem *base_address, | 
					
						
							|  |  |  | 		u32 offset, u32 data) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	out_be32(base_address + (offset << 2), data); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /**
 | 
					
						
							| 
									
										
										
										
											2008-02-25 10:34:47 +11:00
										 |  |  |  * buffer_icap_device_read - Transfer bytes from ICAP to the storage buffer. | 
					
						
							|  |  |  |  * @drvdata: a pointer to the drvdata. | 
					
						
							|  |  |  |  * @offset: The storage buffer start address. | 
					
						
							|  |  |  |  * @count: The number of words (32 bit) to read from the | 
					
						
							| 
									
										
										
										
											2008-02-06 04:24:09 +11:00
										 |  |  |  *           device (ICAP). | 
					
						
							|  |  |  |  **/ | 
					
						
							|  |  |  | static int buffer_icap_device_read(struct hwicap_drvdata *drvdata, | 
					
						
							|  |  |  | 		u32 offset, u32 count) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	s32 retries = 0; | 
					
						
							|  |  |  | 	void __iomem *base_address = drvdata->base_address; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (buffer_icap_busy(base_address)) | 
					
						
							|  |  |  | 		return -EBUSY; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if ((offset + count) > XHI_MAX_BUFFER_INTS) | 
					
						
							|  |  |  | 		return -EINVAL; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* setSize count*4 to get bytes. */ | 
					
						
							|  |  |  | 	buffer_icap_set_size(base_address, (count << 2)); | 
					
						
							|  |  |  | 	buffer_icap_set_offset(base_address, offset); | 
					
						
							|  |  |  | 	buffer_icap_set_rnc(base_address, XHI_READBACK); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	while (buffer_icap_busy(base_address)) { | 
					
						
							|  |  |  | 		retries++; | 
					
						
							|  |  |  | 		if (retries > XHI_MAX_RETRIES) | 
					
						
							|  |  |  | 			return -EBUSY; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /**
 | 
					
						
							| 
									
										
										
										
											2008-02-25 10:34:47 +11:00
										 |  |  |  * buffer_icap_device_write - Transfer bytes from ICAP to the storage buffer. | 
					
						
							|  |  |  |  * @drvdata: a pointer to the drvdata. | 
					
						
							|  |  |  |  * @offset: The storage buffer start address. | 
					
						
							|  |  |  |  * @count: The number of words (32 bit) to read from the | 
					
						
							| 
									
										
										
										
											2008-02-06 04:24:09 +11:00
										 |  |  |  *           device (ICAP). | 
					
						
							|  |  |  |  **/ | 
					
						
							|  |  |  | static int buffer_icap_device_write(struct hwicap_drvdata *drvdata, | 
					
						
							|  |  |  | 		u32 offset, u32 count) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	s32 retries = 0; | 
					
						
							|  |  |  | 	void __iomem *base_address = drvdata->base_address; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (buffer_icap_busy(base_address)) | 
					
						
							|  |  |  | 		return -EBUSY; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if ((offset + count) > XHI_MAX_BUFFER_INTS) | 
					
						
							|  |  |  | 		return -EINVAL; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* setSize count*4 to get bytes. */ | 
					
						
							|  |  |  | 	buffer_icap_set_size(base_address, count << 2); | 
					
						
							|  |  |  | 	buffer_icap_set_offset(base_address, offset); | 
					
						
							|  |  |  | 	buffer_icap_set_rnc(base_address, XHI_CONFIGURE); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	while (buffer_icap_busy(base_address)) { | 
					
						
							|  |  |  | 		retries++; | 
					
						
							|  |  |  | 		if (retries > XHI_MAX_RETRIES) | 
					
						
							|  |  |  | 			return -EBUSY; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /**
 | 
					
						
							| 
									
										
										
										
											2008-02-25 10:34:47 +11:00
										 |  |  |  * buffer_icap_reset - Reset the logic of the icap device. | 
					
						
							|  |  |  |  * @drvdata: a pointer to the drvdata. | 
					
						
							| 
									
										
										
										
											2008-02-06 04:24:09 +11:00
										 |  |  |  * | 
					
						
							|  |  |  |  * Writing to the status register resets the ICAP logic in an internal | 
					
						
							|  |  |  |  * version of the core.  For the version of the core published in EDK, | 
					
						
							|  |  |  |  * this is a noop. | 
					
						
							|  |  |  |  **/ | 
					
						
							|  |  |  | void buffer_icap_reset(struct hwicap_drvdata *drvdata) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     out_be32(drvdata->base_address + XHI_STATUS_REG_OFFSET, 0xFEFE); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /**
 | 
					
						
							| 
									
										
										
										
											2008-02-25 10:34:47 +11:00
										 |  |  |  * buffer_icap_set_configuration - Load a partial bitstream from system memory. | 
					
						
							|  |  |  |  * @drvdata: a pointer to the drvdata. | 
					
						
							|  |  |  |  * @data: Kernel address of the partial bitstream. | 
					
						
							|  |  |  |  * @size: the size of the partial bitstream in 32 bit words. | 
					
						
							| 
									
										
										
										
											2008-02-06 04:24:09 +11:00
										 |  |  |  **/ | 
					
						
							|  |  |  | int buffer_icap_set_configuration(struct hwicap_drvdata *drvdata, u32 *data, | 
					
						
							|  |  |  | 			     u32 size) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	int status; | 
					
						
							|  |  |  | 	s32 buffer_count = 0; | 
					
						
							|  |  |  | 	s32 num_writes = 0; | 
					
						
							|  |  |  | 	bool dirty = 0; | 
					
						
							|  |  |  | 	u32 i; | 
					
						
							|  |  |  | 	void __iomem *base_address = drvdata->base_address; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* Loop through all the data */ | 
					
						
							|  |  |  | 	for (i = 0, buffer_count = 0; i < size; i++) { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		/* Copy data to bram */ | 
					
						
							|  |  |  | 		buffer_icap_set_bram(base_address, buffer_count, data[i]); | 
					
						
							|  |  |  | 		dirty = 1; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		if (buffer_count < XHI_MAX_BUFFER_INTS - 1) { | 
					
						
							|  |  |  | 			buffer_count++; | 
					
						
							|  |  |  | 			continue; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		/* Write data to ICAP */ | 
					
						
							|  |  |  | 		status = buffer_icap_device_write( | 
					
						
							|  |  |  | 				drvdata, | 
					
						
							|  |  |  | 				XHI_BUFFER_START, | 
					
						
							|  |  |  | 				XHI_MAX_BUFFER_INTS); | 
					
						
							|  |  |  | 		if (status != 0) { | 
					
						
							|  |  |  | 			/* abort. */ | 
					
						
							|  |  |  | 			buffer_icap_reset(drvdata); | 
					
						
							|  |  |  | 			return status; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		buffer_count = 0; | 
					
						
							|  |  |  | 		num_writes++; | 
					
						
							|  |  |  | 		dirty = 0; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* Write unwritten data to ICAP */ | 
					
						
							|  |  |  | 	if (dirty) { | 
					
						
							|  |  |  | 		/* Write data to ICAP */ | 
					
						
							|  |  |  | 		status = buffer_icap_device_write(drvdata, XHI_BUFFER_START, | 
					
						
							|  |  |  | 					     buffer_count); | 
					
						
							|  |  |  | 		if (status != 0) { | 
					
						
							|  |  |  | 			/* abort. */ | 
					
						
							|  |  |  | 			buffer_icap_reset(drvdata); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		return status; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return 0; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /**
 | 
					
						
							| 
									
										
										
										
											2008-02-25 10:34:47 +11:00
										 |  |  |  * buffer_icap_get_configuration - Read configuration data from the device. | 
					
						
							|  |  |  |  * @drvdata: a pointer to the drvdata. | 
					
						
							|  |  |  |  * @data: Address of the data representing the partial bitstream | 
					
						
							|  |  |  |  * @size: the size of the partial bitstream in 32 bit words. | 
					
						
							| 
									
										
										
										
											2008-02-06 04:24:09 +11:00
										 |  |  |  **/ | 
					
						
							|  |  |  | int buffer_icap_get_configuration(struct hwicap_drvdata *drvdata, u32 *data, | 
					
						
							|  |  |  | 			     u32 size) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	int status; | 
					
						
							|  |  |  | 	s32 buffer_count = 0; | 
					
						
							|  |  |  | 	s32 read_count = 0; | 
					
						
							|  |  |  | 	u32 i; | 
					
						
							|  |  |  | 	void __iomem *base_address = drvdata->base_address; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* Loop through all the data */ | 
					
						
							|  |  |  | 	for (i = 0, buffer_count = XHI_MAX_BUFFER_INTS; i < size; i++) { | 
					
						
							|  |  |  | 		if (buffer_count == XHI_MAX_BUFFER_INTS) { | 
					
						
							|  |  |  | 			u32 words_remaining = size - i; | 
					
						
							|  |  |  | 			u32 words_to_read = | 
					
						
							|  |  |  | 				words_remaining < | 
					
						
							|  |  |  | 				XHI_MAX_BUFFER_INTS ? words_remaining : | 
					
						
							|  |  |  | 				XHI_MAX_BUFFER_INTS; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			/* Read data from ICAP */ | 
					
						
							|  |  |  | 			status = buffer_icap_device_read( | 
					
						
							|  |  |  | 					drvdata, | 
					
						
							|  |  |  | 					XHI_BUFFER_START, | 
					
						
							|  |  |  | 					words_to_read); | 
					
						
							|  |  |  | 			if (status != 0) { | 
					
						
							|  |  |  | 				/* abort. */ | 
					
						
							|  |  |  | 				buffer_icap_reset(drvdata); | 
					
						
							|  |  |  | 				return status; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			buffer_count = 0; | 
					
						
							|  |  |  | 			read_count++; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		/* Copy data from bram */ | 
					
						
							|  |  |  | 		data[i] = buffer_icap_get_bram(base_address, buffer_count); | 
					
						
							|  |  |  | 		buffer_count++; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return 0; | 
					
						
							|  |  |  | }; |