83d290c56f
When U-Boot started using SPDX tags we were among the early adopters and there weren't a lot of other examples to borrow from. So we picked the area of the file that usually had a full license text and replaced it with an appropriate SPDX-License-Identifier: entry. Since then, the Linux Kernel has adopted SPDX tags and they place it as the very first line in a file (except where shebangs are used, then it's second line) and with slightly different comment styles than us. In part due to community overlap, in part due to better tag visibility and in part for other minor reasons, switch over to that style. This commit changes all instances where we have a single declared license in the tag as both the before and after are identical in tag contents. There's also a few places where I found we did not have a tag and have introduced one. Signed-off-by: Tom Rini <trini@konsulko.com>
380 lines
8.2 KiB
C
380 lines
8.2 KiB
C
// SPDX-License-Identifier: GPL-2.0+
|
|
/*
|
|
* (C) Copyright 2010
|
|
* Stefano Babic, DENX Software Engineering, sbabic@denx.de.
|
|
*
|
|
* (C) Copyright 2002
|
|
* Rich Ireland, Enterasys Networks, rireland@enterasys.com.
|
|
*
|
|
* ispVM functions adapted from Lattice's ispmVMEmbedded code:
|
|
* Copyright 2009 Lattice Semiconductor Corp.
|
|
*/
|
|
|
|
#include <common.h>
|
|
#include <malloc.h>
|
|
#include <fpga.h>
|
|
#include <lattice.h>
|
|
|
|
static lattice_board_specific_func *pfns;
|
|
static const char *fpga_image;
|
|
static unsigned long read_bytes;
|
|
static unsigned long bufsize;
|
|
static unsigned short expectedCRC;
|
|
|
|
/*
|
|
* External variables and functions declared in ivm_core.c module.
|
|
*/
|
|
extern unsigned short g_usCalculatedCRC;
|
|
extern unsigned short g_usDataType;
|
|
extern unsigned char *g_pucIntelBuffer;
|
|
extern unsigned char *g_pucHeapMemory;
|
|
extern unsigned short g_iHeapCounter;
|
|
extern unsigned short g_iHEAPSize;
|
|
extern unsigned short g_usIntelDataIndex;
|
|
extern unsigned short g_usIntelBufferSize;
|
|
extern char *const g_szSupportedVersions[];
|
|
|
|
|
|
/*
|
|
* ispVMDelay
|
|
*
|
|
* Users must implement a delay to observe a_usTimeDelay, where
|
|
* bit 15 of the a_usTimeDelay defines the unit.
|
|
* 1 = milliseconds
|
|
* 0 = microseconds
|
|
* Example:
|
|
* a_usTimeDelay = 0x0001 = 1 microsecond delay.
|
|
* a_usTimeDelay = 0x8001 = 1 millisecond delay.
|
|
*
|
|
* This subroutine is called upon to provide a delay from 1 millisecond to a few
|
|
* hundreds milliseconds each time.
|
|
* It is understood that due to a_usTimeDelay is defined as unsigned short, a 16
|
|
* bits integer, this function is restricted to produce a delay to 64000
|
|
* micro-seconds or 32000 milli-second maximum. The VME file will never pass on
|
|
* to this function a delay time > those maximum number. If it needs more than
|
|
* those maximum, the VME file will launch the delay function several times to
|
|
* realize a larger delay time cummulatively.
|
|
* It is perfectly alright to provide a longer delay than required. It is not
|
|
* acceptable if the delay is shorter.
|
|
*/
|
|
void ispVMDelay(unsigned short delay)
|
|
{
|
|
if (delay & 0x8000)
|
|
delay = (delay & ~0x8000) * 1000;
|
|
udelay(delay);
|
|
}
|
|
|
|
void writePort(unsigned char a_ucPins, unsigned char a_ucValue)
|
|
{
|
|
a_ucValue = a_ucValue ? 1 : 0;
|
|
|
|
switch (a_ucPins) {
|
|
case g_ucPinTDI:
|
|
pfns->jtag_set_tdi(a_ucValue);
|
|
break;
|
|
case g_ucPinTCK:
|
|
pfns->jtag_set_tck(a_ucValue);
|
|
break;
|
|
case g_ucPinTMS:
|
|
pfns->jtag_set_tms(a_ucValue);
|
|
break;
|
|
default:
|
|
printf("%s: requested unknown pin\n", __func__);
|
|
}
|
|
}
|
|
|
|
unsigned char readPort(void)
|
|
{
|
|
return pfns->jtag_get_tdo();
|
|
}
|
|
|
|
void sclock(void)
|
|
{
|
|
writePort(g_ucPinTCK, 0x01);
|
|
writePort(g_ucPinTCK, 0x00);
|
|
}
|
|
|
|
void calibration(void)
|
|
{
|
|
/* Apply 2 pulses to TCK. */
|
|
writePort(g_ucPinTCK, 0x00);
|
|
writePort(g_ucPinTCK, 0x01);
|
|
writePort(g_ucPinTCK, 0x00);
|
|
writePort(g_ucPinTCK, 0x01);
|
|
writePort(g_ucPinTCK, 0x00);
|
|
|
|
ispVMDelay(0x8001);
|
|
|
|
/* Apply 2 pulses to TCK. */
|
|
writePort(g_ucPinTCK, 0x01);
|
|
writePort(g_ucPinTCK, 0x00);
|
|
writePort(g_ucPinTCK, 0x01);
|
|
writePort(g_ucPinTCK, 0x00);
|
|
}
|
|
|
|
/*
|
|
* GetByte
|
|
*
|
|
* Returns a byte to the caller. The returned byte depends on the
|
|
* g_usDataType register. If the HEAP_IN bit is set, then the byte
|
|
* is returned from the HEAP. If the LHEAP_IN bit is set, then
|
|
* the byte is returned from the intelligent buffer. Otherwise,
|
|
* the byte is returned directly from the VME file.
|
|
*/
|
|
unsigned char GetByte(void)
|
|
{
|
|
unsigned char ucData;
|
|
unsigned int block_size = 4 * 1024;
|
|
|
|
if (g_usDataType & HEAP_IN) {
|
|
|
|
/*
|
|
* Get data from repeat buffer.
|
|
*/
|
|
|
|
if (g_iHeapCounter > g_iHEAPSize) {
|
|
|
|
/*
|
|
* Data over-run.
|
|
*/
|
|
|
|
return 0xFF;
|
|
}
|
|
|
|
ucData = g_pucHeapMemory[g_iHeapCounter++];
|
|
} else if (g_usDataType & LHEAP_IN) {
|
|
|
|
/*
|
|
* Get data from intel buffer.
|
|
*/
|
|
|
|
if (g_usIntelDataIndex >= g_usIntelBufferSize) {
|
|
return 0xFF;
|
|
}
|
|
|
|
ucData = g_pucIntelBuffer[g_usIntelDataIndex++];
|
|
} else {
|
|
if (read_bytes == bufsize) {
|
|
return 0xFF;
|
|
}
|
|
ucData = *fpga_image++;
|
|
read_bytes++;
|
|
|
|
if (!(read_bytes % block_size)) {
|
|
printf("Downloading FPGA %ld/%ld completed\r",
|
|
read_bytes,
|
|
bufsize);
|
|
}
|
|
|
|
if (expectedCRC != 0) {
|
|
ispVMCalculateCRC32(ucData);
|
|
}
|
|
}
|
|
|
|
return ucData;
|
|
}
|
|
|
|
signed char ispVM(void)
|
|
{
|
|
char szFileVersion[9] = { 0 };
|
|
signed char cRetCode = 0;
|
|
signed char cIndex = 0;
|
|
signed char cVersionIndex = 0;
|
|
unsigned char ucReadByte = 0;
|
|
unsigned short crc;
|
|
|
|
g_pucHeapMemory = NULL;
|
|
g_iHeapCounter = 0;
|
|
g_iHEAPSize = 0;
|
|
g_usIntelDataIndex = 0;
|
|
g_usIntelBufferSize = 0;
|
|
g_usCalculatedCRC = 0;
|
|
expectedCRC = 0;
|
|
ucReadByte = GetByte();
|
|
switch (ucReadByte) {
|
|
case FILE_CRC:
|
|
crc = (unsigned char)GetByte();
|
|
crc <<= 8;
|
|
crc |= GetByte();
|
|
expectedCRC = crc;
|
|
|
|
for (cIndex = 0; cIndex < 8; cIndex++)
|
|
szFileVersion[cIndex] = GetByte();
|
|
|
|
break;
|
|
default:
|
|
szFileVersion[0] = (signed char) ucReadByte;
|
|
for (cIndex = 1; cIndex < 8; cIndex++)
|
|
szFileVersion[cIndex] = GetByte();
|
|
|
|
break;
|
|
}
|
|
|
|
/*
|
|
*
|
|
* Compare the VME file version against the supported version.
|
|
*
|
|
*/
|
|
|
|
for (cVersionIndex = 0; g_szSupportedVersions[cVersionIndex] != 0;
|
|
cVersionIndex++) {
|
|
for (cIndex = 0; cIndex < 8; cIndex++) {
|
|
if (szFileVersion[cIndex] !=
|
|
g_szSupportedVersions[cVersionIndex][cIndex]) {
|
|
cRetCode = VME_VERSION_FAILURE;
|
|
break;
|
|
}
|
|
cRetCode = 0;
|
|
}
|
|
|
|
if (cRetCode == 0) {
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (cRetCode < 0) {
|
|
return VME_VERSION_FAILURE;
|
|
}
|
|
|
|
printf("VME file checked: starting downloading to FPGA\n");
|
|
|
|
ispVMStart();
|
|
|
|
cRetCode = ispVMCode();
|
|
|
|
ispVMEnd();
|
|
ispVMFreeMem();
|
|
puts("\n");
|
|
|
|
if (cRetCode == 0 && expectedCRC != 0 &&
|
|
(expectedCRC != g_usCalculatedCRC)) {
|
|
printf("Expected CRC: 0x%.4X\n", expectedCRC);
|
|
printf("Calculated CRC: 0x%.4X\n", g_usCalculatedCRC);
|
|
return VME_CRC_FAILURE;
|
|
}
|
|
return cRetCode;
|
|
}
|
|
|
|
static int lattice_validate(Lattice_desc *desc, const char *fn)
|
|
{
|
|
int ret_val = false;
|
|
|
|
if (desc) {
|
|
if ((desc->family > min_lattice_type) &&
|
|
(desc->family < max_lattice_type)) {
|
|
if ((desc->iface > min_lattice_iface_type) &&
|
|
(desc->iface < max_lattice_iface_type)) {
|
|
if (desc->size) {
|
|
ret_val = true;
|
|
} else {
|
|
printf("%s: NULL part size\n", fn);
|
|
}
|
|
} else {
|
|
printf("%s: Invalid Interface type, %d\n",
|
|
fn, desc->iface);
|
|
}
|
|
} else {
|
|
printf("%s: Invalid family type, %d\n",
|
|
fn, desc->family);
|
|
}
|
|
} else {
|
|
printf("%s: NULL descriptor!\n", fn);
|
|
}
|
|
|
|
return ret_val;
|
|
}
|
|
|
|
int lattice_load(Lattice_desc *desc, const void *buf, size_t bsize)
|
|
{
|
|
int ret_val = FPGA_FAIL;
|
|
|
|
if (!lattice_validate(desc, (char *)__func__)) {
|
|
printf("%s: Invalid device descriptor\n", __func__);
|
|
} else {
|
|
pfns = desc->iface_fns;
|
|
|
|
switch (desc->family) {
|
|
case Lattice_XP2:
|
|
fpga_image = buf;
|
|
read_bytes = 0;
|
|
bufsize = bsize;
|
|
debug("%s: Launching the Lattice ISPVME Loader:"
|
|
" addr %p size 0x%lx...\n",
|
|
__func__, fpga_image, bufsize);
|
|
ret_val = ispVM();
|
|
if (ret_val)
|
|
printf("%s: error %d downloading FPGA image\n",
|
|
__func__, ret_val);
|
|
else
|
|
puts("FPGA downloaded successfully\n");
|
|
break;
|
|
default:
|
|
printf("%s: Unsupported family type, %d\n",
|
|
__func__, desc->family);
|
|
}
|
|
}
|
|
|
|
return ret_val;
|
|
}
|
|
|
|
int lattice_dump(Lattice_desc *desc, const void *buf, size_t bsize)
|
|
{
|
|
puts("Dump not supported for Lattice FPGA\n");
|
|
|
|
return FPGA_FAIL;
|
|
|
|
}
|
|
|
|
int lattice_info(Lattice_desc *desc)
|
|
{
|
|
int ret_val = FPGA_FAIL;
|
|
|
|
if (lattice_validate(desc, (char *)__func__)) {
|
|
printf("Family: \t");
|
|
switch (desc->family) {
|
|
case Lattice_XP2:
|
|
puts("XP2\n");
|
|
break;
|
|
/* Add new family types here */
|
|
default:
|
|
printf("Unknown family type, %d\n", desc->family);
|
|
}
|
|
|
|
puts("Interface type:\t");
|
|
switch (desc->iface) {
|
|
case lattice_jtag_mode:
|
|
puts("JTAG Mode\n");
|
|
break;
|
|
/* Add new interface types here */
|
|
default:
|
|
printf("Unsupported interface type, %d\n", desc->iface);
|
|
}
|
|
|
|
printf("Device Size: \t%d bytes\n",
|
|
desc->size);
|
|
|
|
if (desc->iface_fns) {
|
|
printf("Device Function Table @ 0x%p\n",
|
|
desc->iface_fns);
|
|
switch (desc->family) {
|
|
case Lattice_XP2:
|
|
break;
|
|
/* Add new family types here */
|
|
default:
|
|
break;
|
|
}
|
|
} else {
|
|
puts("No Device Function Table.\n");
|
|
}
|
|
|
|
if (desc->desc)
|
|
printf("Model: \t%s\n", desc->desc);
|
|
|
|
ret_val = FPGA_SUCCESS;
|
|
} else {
|
|
printf("%s: Invalid device descriptor\n", __func__);
|
|
}
|
|
|
|
return ret_val;
|
|
}
|