2008-02-05 20:24:09 +03: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 0213 9 , 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 02:34:47 +03:00
* buffer_icap_get_status - Get the contents of the status register .
2008-03-17 20:36:30 +03:00
* @ drvdata : a pointer to the drvdata .
2008-02-05 20:24:09 +03: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-17 20:36:30 +03:00
u32 buffer_icap_get_status ( struct hwicap_drvdata * drvdata )
2008-02-05 20:24:09 +03:00
{
2008-03-17 20:36:30 +03:00
return in_be32 ( drvdata - > base_address + XHI_STATUS_REG_OFFSET ) ;
2008-02-05 20:24:09 +03:00
}
/**
2008-02-25 02:34:47 +03: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-05 20:24:09 +03: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 02:34:47 +03:00
* buffer_icap_busy - Return true if the icap device is busy
* @ base_address : is the base address of the device
2008-02-05 20:24:09 +03: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-17 20:36:30 +03:00
u32 status = in_be32 ( base_address + XHI_STATUS_REG_OFFSET ) ;
return ( status & 1 ) = = XHI_NOT_FINISHED ;
2008-02-05 20:24:09 +03:00
}
/**
2008-02-25 02:34:47 +03: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-05 20:24:09 +03: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 02:34:47 +03: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-05 20:24:09 +03: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 02:34:47 +03: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-05 20:24:09 +03: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 02:34:47 +03: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-05 20:24:09 +03: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 02:34:47 +03: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-05 20:24:09 +03: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 02:34:47 +03: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-05 20:24:09 +03: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 02:34:47 +03:00
* buffer_icap_reset - Reset the logic of the icap device .
* @ drvdata : a pointer to the drvdata .
2008-02-05 20:24:09 +03: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 02:34:47 +03: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-05 20:24:09 +03: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 02:34:47 +03: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-05 20:24:09 +03: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 ;
} ;