2020-06-18 16:55:50 +00:00
|
|
|
/* SPDX-License-Identifier: BSD-3-Clause
|
|
|
|
* Copyright(c) 2010-2019 Intel Corporation
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "opae_spi.h"
|
|
|
|
#include "ifpga_compat.h"
|
|
|
|
|
|
|
|
/*transaction opcodes*/
|
|
|
|
#define SPI_TRAN_SEQ_WRITE 0x04 /* SPI transaction sequential write */
|
|
|
|
#define SPI_TRAN_SEQ_READ 0x14 /* SPI transaction sequential read */
|
|
|
|
#define SPI_TRAN_NON_SEQ_WRITE 0x00 /* SPI transaction non-sequential write */
|
|
|
|
#define SPI_TRAN_NON_SEQ_READ 0x10 /* SPI transaction non-sequential read*/
|
|
|
|
|
|
|
|
/*specail packet characters*/
|
|
|
|
#define SPI_PACKET_SOP 0x7a
|
|
|
|
#define SPI_PACKET_EOP 0x7b
|
|
|
|
#define SPI_PACKET_CHANNEL 0x7c
|
|
|
|
#define SPI_PACKET_ESC 0x7d
|
|
|
|
|
|
|
|
/*special byte characters*/
|
|
|
|
#define SPI_BYTE_IDLE 0x4a
|
|
|
|
#define SPI_BYTE_ESC 0x4d
|
|
|
|
|
|
|
|
#define SPI_REG_BYTES 4
|
|
|
|
|
|
|
|
#define INIT_SPI_TRAN_HEADER(trans_type, size, address) \
|
|
|
|
({ \
|
|
|
|
header.trans_type = trans_type; \
|
|
|
|
header.reserve = 0; \
|
|
|
|
header.size = cpu_to_be16(size); \
|
|
|
|
header.addr = cpu_to_be32(addr); \
|
|
|
|
})
|
|
|
|
|
|
|
|
#ifdef OPAE_SPI_DEBUG
|
|
|
|
static void print_buffer(const char *string, void *buffer, int len)
|
|
|
|
{
|
|
|
|
int i;
|
|
|
|
unsigned char *p = buffer;
|
|
|
|
|
|
|
|
printf("%s print buffer, len=%d\n", string, len);
|
|
|
|
|
|
|
|
for (i = 0; i < len; i++)
|
|
|
|
printf("%x ", *(p+i));
|
|
|
|
printf("\n");
|
|
|
|
}
|
|
|
|
#else
|
|
|
|
static void print_buffer(const char *string, void *buffer, int len)
|
|
|
|
{
|
|
|
|
UNUSED(string);
|
|
|
|
UNUSED(buffer);
|
|
|
|
UNUSED(len);
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
static unsigned char xor_20(unsigned char val)
|
|
|
|
{
|
|
|
|
return val^0x20;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void reorder_phy_data(u8 bits_per_word,
|
|
|
|
void *buf, unsigned int len)
|
|
|
|
{
|
|
|
|
unsigned int count = len / (bits_per_word/8);
|
|
|
|
u32 *p;
|
|
|
|
|
|
|
|
if (bits_per_word == 32) {
|
|
|
|
p = (u32 *)buf;
|
|
|
|
while (count--) {
|
|
|
|
*p = cpu_to_be32(*p);
|
|
|
|
p++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
enum {
|
|
|
|
SPI_FOUND_SOP,
|
|
|
|
SPI_FOUND_EOP,
|
|
|
|
SPI_NOT_FOUND,
|
|
|
|
};
|
|
|
|
|
|
|
|
static int resp_find_sop_eop(unsigned char *resp, unsigned int len,
|
|
|
|
int flags)
|
|
|
|
{
|
|
|
|
int ret = SPI_NOT_FOUND;
|
|
|
|
|
|
|
|
unsigned char *b = resp;
|
|
|
|
|
|
|
|
/* find SOP */
|
|
|
|
if (flags != SPI_FOUND_SOP) {
|
|
|
|
while (b < resp + len && *b != SPI_PACKET_SOP)
|
|
|
|
b++;
|
|
|
|
|
|
|
|
if (*b != SPI_PACKET_SOP)
|
|
|
|
goto done;
|
|
|
|
|
|
|
|
ret = SPI_FOUND_SOP;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* find EOP */
|
|
|
|
while (b < resp + len && *b != SPI_PACKET_EOP)
|
|
|
|
b++;
|
|
|
|
|
|
|
|
if (*b != SPI_PACKET_EOP)
|
|
|
|
goto done;
|
|
|
|
|
|
|
|
ret = SPI_FOUND_EOP;
|
|
|
|
|
|
|
|
done:
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void phy_tx_pad(unsigned char *phy_buf, unsigned int phy_buf_len,
|
|
|
|
unsigned int *aligned_len)
|
|
|
|
{
|
|
|
|
unsigned char *p = &phy_buf[phy_buf_len - 1], *dst_p;
|
|
|
|
|
|
|
|
*aligned_len = IFPGA_ALIGN(phy_buf_len, 4);
|
|
|
|
|
|
|
|
if (*aligned_len == phy_buf_len)
|
|
|
|
return;
|
|
|
|
|
|
|
|
dst_p = &phy_buf[*aligned_len - 1];
|
|
|
|
|
|
|
|
/* move EOP and bytes after EOP to the end of aligned size */
|
|
|
|
while (p > phy_buf) {
|
|
|
|
*dst_p = *p;
|
|
|
|
|
|
|
|
if (*p == SPI_PACKET_EOP)
|
|
|
|
break;
|
|
|
|
|
|
|
|
p--;
|
|
|
|
dst_p--;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* fill the hole with PHY_IDLE */
|
|
|
|
while (p < dst_p)
|
|
|
|
*p++ = SPI_BYTE_IDLE;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int byte_to_core_convert(struct spi_transaction_dev *dev,
|
|
|
|
unsigned int send_len, unsigned char *send_data,
|
|
|
|
unsigned int resp_len, unsigned char *resp_data,
|
|
|
|
unsigned int *valid_resp_len)
|
|
|
|
{
|
|
|
|
unsigned int i;
|
|
|
|
int ret = 0;
|
|
|
|
unsigned char *send_packet = dev->buffer->bytes_send;
|
|
|
|
unsigned char *resp_packet = dev->buffer->bytes_resp;
|
|
|
|
unsigned char *p;
|
|
|
|
unsigned char current_byte;
|
|
|
|
unsigned char *tx_buffer;
|
|
|
|
unsigned int tx_len = 0;
|
|
|
|
unsigned char *rx_buffer;
|
|
|
|
unsigned int rx_len = 0;
|
|
|
|
int retry = 0;
|
|
|
|
int spi_flags;
|
|
|
|
unsigned long timeout = msecs_to_timer_cycles(1000);
|
|
|
|
unsigned long ticks;
|
|
|
|
unsigned int resp_max_len = 2 * resp_len;
|
|
|
|
|
|
|
|
print_buffer("before bytes:", send_data, send_len);
|
|
|
|
|
|
|
|
p = send_packet;
|
|
|
|
|
|
|
|
for (i = 0; i < send_len; i++) {
|
|
|
|
current_byte = send_data[i];
|
|
|
|
switch (current_byte) {
|
|
|
|
case SPI_BYTE_IDLE:
|
2021-01-28 17:08:59 +00:00
|
|
|
*p++ = SPI_BYTE_ESC;
|
2020-06-18 16:55:50 +00:00
|
|
|
*p++ = xor_20(current_byte);
|
|
|
|
break;
|
|
|
|
case SPI_BYTE_ESC:
|
|
|
|
*p++ = SPI_BYTE_ESC;
|
|
|
|
*p++ = xor_20(current_byte);
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
*p++ = current_byte;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
tx_len = p - send_packet;
|
|
|
|
|
|
|
|
print_buffer("before spi:", send_packet, tx_len);
|
|
|
|
|
|
|
|
phy_tx_pad(send_packet, tx_len, &tx_len);
|
|
|
|
print_buffer("after pad:", send_packet, tx_len);
|
|
|
|
|
|
|
|
reorder_phy_data(32, send_packet, tx_len);
|
|
|
|
|
|
|
|
print_buffer("after order to spi:", send_packet, tx_len);
|
|
|
|
|
|
|
|
/* call spi */
|
|
|
|
tx_buffer = send_packet;
|
|
|
|
rx_buffer = resp_packet;
|
|
|
|
rx_len = resp_max_len;
|
|
|
|
spi_flags = SPI_NOT_FOUND;
|
|
|
|
|
|
|
|
read_again:
|
|
|
|
ret = spi_command(dev->dev, dev->chipselect, tx_len, tx_buffer,
|
|
|
|
rx_len, rx_buffer);
|
|
|
|
if (ret)
|
|
|
|
return -EBUSY;
|
|
|
|
|
|
|
|
print_buffer("read from spi:", rx_buffer, rx_len);
|
|
|
|
|
|
|
|
/* look for SOP firstly*/
|
|
|
|
ret = resp_find_sop_eop(rx_buffer, rx_len - 1, spi_flags);
|
|
|
|
if (ret != SPI_FOUND_EOP) {
|
|
|
|
tx_buffer = NULL;
|
|
|
|
tx_len = 0;
|
|
|
|
ticks = rte_get_timer_cycles();
|
|
|
|
if (time_after(ticks, timeout) &&
|
|
|
|
retry++ > SPI_MAX_RETRY) {
|
|
|
|
dev_err(NULL, "Have retry %d, found invalid packet data\n",
|
|
|
|
retry);
|
|
|
|
return -EBUSY;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (ret == SPI_FOUND_SOP) {
|
|
|
|
rx_buffer += rx_len;
|
|
|
|
resp_max_len += rx_len;
|
|
|
|
}
|
|
|
|
|
|
|
|
spi_flags = ret;
|
|
|
|
goto read_again;
|
|
|
|
}
|
|
|
|
|
|
|
|
print_buffer("found valid data:", resp_packet, resp_max_len);
|
|
|
|
|
|
|
|
/* analyze response packet */
|
|
|
|
i = 0;
|
|
|
|
p = resp_data;
|
|
|
|
while (i < resp_max_len) {
|
|
|
|
current_byte = resp_packet[i];
|
|
|
|
switch (current_byte) {
|
|
|
|
case SPI_BYTE_IDLE:
|
|
|
|
i++;
|
|
|
|
break;
|
|
|
|
case SPI_BYTE_ESC:
|
|
|
|
i++;
|
|
|
|
current_byte = resp_packet[i];
|
|
|
|
*p++ = xor_20(current_byte);
|
|
|
|
i++;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
*p++ = current_byte;
|
|
|
|
i++;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* receive "4a" means the SPI is idle, not valid data */
|
|
|
|
*valid_resp_len = p - resp_data;
|
|
|
|
if (*valid_resp_len == 0) {
|
|
|
|
dev_err(NULL, "error: repond package without valid data\n");
|
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int packet_to_byte_conver(struct spi_transaction_dev *dev,
|
|
|
|
unsigned int send_len, unsigned char *send_buf,
|
|
|
|
unsigned int resp_len, unsigned char *resp_buf,
|
|
|
|
unsigned int *valid)
|
|
|
|
{
|
|
|
|
int ret = 0;
|
|
|
|
unsigned int i;
|
|
|
|
unsigned char current_byte;
|
|
|
|
unsigned int resp_max_len;
|
|
|
|
unsigned char *send_packet = dev->buffer->packet_send;
|
|
|
|
unsigned char *resp_packet = dev->buffer->packet_resp;
|
|
|
|
unsigned char *p;
|
|
|
|
unsigned int valid_resp_len = 0;
|
|
|
|
|
|
|
|
print_buffer("before packet:", send_buf, send_len);
|
|
|
|
|
|
|
|
resp_max_len = 2 * resp_len + 4;
|
|
|
|
|
|
|
|
p = send_packet;
|
|
|
|
|
|
|
|
/* SOP header */
|
|
|
|
*p++ = SPI_PACKET_SOP;
|
|
|
|
|
|
|
|
*p++ = SPI_PACKET_CHANNEL;
|
|
|
|
*p++ = 0;
|
|
|
|
|
|
|
|
/* append the data into a packet */
|
|
|
|
for (i = 0; i < send_len; i++) {
|
|
|
|
current_byte = send_buf[i];
|
|
|
|
|
|
|
|
/* EOP for last byte */
|
|
|
|
if (i == send_len - 1)
|
|
|
|
*p++ = SPI_PACKET_EOP;
|
|
|
|
|
|
|
|
switch (current_byte) {
|
|
|
|
case SPI_PACKET_SOP:
|
|
|
|
case SPI_PACKET_EOP:
|
|
|
|
case SPI_PACKET_CHANNEL:
|
|
|
|
case SPI_PACKET_ESC:
|
|
|
|
*p++ = SPI_PACKET_ESC;
|
|
|
|
*p++ = xor_20(current_byte);
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
*p++ = current_byte;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
ret = byte_to_core_convert(dev, p - send_packet,
|
|
|
|
send_packet, resp_max_len, resp_packet,
|
|
|
|
&valid_resp_len);
|
|
|
|
if (ret)
|
|
|
|
return -EBUSY;
|
|
|
|
|
|
|
|
print_buffer("after byte conver:", resp_packet, valid_resp_len);
|
|
|
|
|
|
|
|
/* analyze the response packet */
|
|
|
|
p = resp_buf;
|
|
|
|
|
|
|
|
/* look for SOP */
|
|
|
|
for (i = 0; i < valid_resp_len; i++) {
|
|
|
|
if (resp_packet[i] == SPI_PACKET_SOP)
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (i == valid_resp_len) {
|
|
|
|
dev_err(NULL, "error on analyze response packet 0x%x\n",
|
|
|
|
resp_packet[i]);
|
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
|
|
|
|
i++;
|
|
|
|
|
|
|
|
/* continue parsing data after SOP */
|
|
|
|
while (i < valid_resp_len) {
|
|
|
|
current_byte = resp_packet[i];
|
|
|
|
|
|
|
|
switch (current_byte) {
|
|
|
|
case SPI_PACKET_ESC:
|
|
|
|
case SPI_PACKET_CHANNEL:
|
|
|
|
case SPI_PACKET_SOP:
|
|
|
|
i++;
|
|
|
|
current_byte = resp_packet[i];
|
|
|
|
*p++ = xor_20(current_byte);
|
|
|
|
i++;
|
|
|
|
break;
|
|
|
|
case SPI_PACKET_EOP:
|
|
|
|
i++;
|
|
|
|
current_byte = resp_packet[i];
|
|
|
|
if (current_byte == SPI_PACKET_ESC ||
|
|
|
|
current_byte == SPI_PACKET_CHANNEL ||
|
|
|
|
current_byte == SPI_PACKET_SOP) {
|
|
|
|
i++;
|
|
|
|
current_byte = resp_packet[i];
|
|
|
|
*p++ = xor_20(current_byte);
|
|
|
|
} else
|
|
|
|
*p++ = current_byte;
|
|
|
|
i = valid_resp_len;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
*p++ = current_byte;
|
|
|
|
i++;
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
*valid = p - resp_buf;
|
|
|
|
|
|
|
|
print_buffer("after packet:", resp_buf, *valid);
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int do_transaction(struct spi_transaction_dev *dev, unsigned int addr,
|
|
|
|
unsigned int size, unsigned char *data,
|
|
|
|
unsigned int trans_type)
|
|
|
|
{
|
|
|
|
|
|
|
|
struct spi_tran_header header;
|
|
|
|
unsigned char *transaction = dev->buffer->tran_send;
|
|
|
|
unsigned char *response = dev->buffer->tran_resp;
|
|
|
|
unsigned char *p;
|
|
|
|
int ret = 0;
|
|
|
|
unsigned int i;
|
|
|
|
unsigned int valid_len = 0;
|
|
|
|
|
|
|
|
/* make transacation header */
|
|
|
|
INIT_SPI_TRAN_HEADER(trans_type, size, addr);
|
|
|
|
|
|
|
|
/* fill the header */
|
|
|
|
p = transaction;
|
|
|
|
opae_memcpy(p, &header, sizeof(struct spi_tran_header));
|
|
|
|
p = p + sizeof(struct spi_tran_header);
|
|
|
|
|
|
|
|
switch (trans_type) {
|
|
|
|
case SPI_TRAN_SEQ_WRITE:
|
|
|
|
case SPI_TRAN_NON_SEQ_WRITE:
|
|
|
|
for (i = 0; i < size; i++)
|
|
|
|
*p++ = *data++;
|
|
|
|
|
|
|
|
ret = packet_to_byte_conver(dev, size + HEADER_LEN,
|
|
|
|
transaction, RESPONSE_LEN, response,
|
|
|
|
&valid_len);
|
|
|
|
if (ret)
|
|
|
|
return -EBUSY;
|
|
|
|
|
|
|
|
/* check the result */
|
|
|
|
if (size != ((unsigned int)(response[2] & 0xff) << 8 |
|
|
|
|
(unsigned int)(response[3] & 0xff)))
|
|
|
|
ret = -EBUSY;
|
|
|
|
|
|
|
|
break;
|
|
|
|
case SPI_TRAN_SEQ_READ:
|
|
|
|
case SPI_TRAN_NON_SEQ_READ:
|
|
|
|
ret = packet_to_byte_conver(dev, HEADER_LEN,
|
|
|
|
transaction, size, response,
|
|
|
|
&valid_len);
|
|
|
|
if (ret || valid_len != size)
|
|
|
|
return -EBUSY;
|
|
|
|
|
|
|
|
for (i = 0; i < size; i++)
|
|
|
|
*data++ = *response++;
|
|
|
|
|
|
|
|
ret = 0;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
int spi_transaction_read(struct spi_transaction_dev *dev, unsigned int addr,
|
|
|
|
unsigned int size, unsigned char *data)
|
|
|
|
{
|
|
|
|
int ret;
|
|
|
|
|
|
|
|
pthread_mutex_lock(&dev->lock);
|
|
|
|
ret = do_transaction(dev, addr, size, data,
|
|
|
|
(size > SPI_REG_BYTES) ?
|
|
|
|
SPI_TRAN_SEQ_READ : SPI_TRAN_NON_SEQ_READ);
|
|
|
|
pthread_mutex_unlock(&dev->lock);
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
int spi_transaction_write(struct spi_transaction_dev *dev, unsigned int addr,
|
|
|
|
unsigned int size, unsigned char *data)
|
|
|
|
{
|
|
|
|
int ret;
|
|
|
|
|
|
|
|
pthread_mutex_lock(&dev->lock);
|
|
|
|
ret = do_transaction(dev, addr, size, data,
|
|
|
|
(size > SPI_REG_BYTES) ?
|
|
|
|
SPI_TRAN_SEQ_WRITE : SPI_TRAN_NON_SEQ_WRITE);
|
|
|
|
pthread_mutex_unlock(&dev->lock);
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
struct spi_transaction_dev *spi_transaction_init(struct altera_spi_device *dev,
|
|
|
|
int chipselect)
|
|
|
|
{
|
|
|
|
struct spi_transaction_dev *spi_tran_dev;
|
|
|
|
int ret;
|
|
|
|
|
|
|
|
spi_tran_dev = opae_malloc(sizeof(struct spi_transaction_dev));
|
|
|
|
if (!spi_tran_dev)
|
|
|
|
return NULL;
|
|
|
|
|
|
|
|
spi_tran_dev->dev = dev;
|
|
|
|
spi_tran_dev->chipselect = chipselect;
|
|
|
|
|
|
|
|
spi_tran_dev->buffer = opae_malloc(sizeof(struct spi_tran_buffer));
|
|
|
|
if (!spi_tran_dev->buffer)
|
|
|
|
goto err;
|
|
|
|
|
|
|
|
ret = pthread_mutex_init(&spi_tran_dev->lock, NULL);
|
|
|
|
if (ret) {
|
|
|
|
dev_err(spi_tran_dev, "fail to init mutex lock\n");
|
|
|
|
goto err;
|
|
|
|
}
|
|
|
|
|
|
|
|
return spi_tran_dev;
|
|
|
|
|
|
|
|
err:
|
|
|
|
opae_free(spi_tran_dev);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
void spi_transaction_remove(struct spi_transaction_dev *dev)
|
|
|
|
{
|
|
|
|
if (dev && dev->buffer)
|
|
|
|
opae_free(dev->buffer);
|
|
|
|
if (dev) {
|
|
|
|
pthread_mutex_destroy(&dev->lock);
|
|
|
|
opae_free(dev);
|
|
|
|
}
|
|
|
|
}
|