2011-06-06 13:18:03 +08:00
|
|
|
/*
|
|
|
|
* Renesas USB driver
|
|
|
|
*
|
|
|
|
* Copyright (C) 2011 Renesas Solutions Corp.
|
|
|
|
* Kuninori Morimoto <kuninori.morimoto.gx@renesas.com>
|
|
|
|
*
|
|
|
|
* This program is distributed in the hope that it will be useful,
|
|
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
* GNU General Public License for more details.
|
|
|
|
*
|
|
|
|
* 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., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
#include <linux/delay.h>
|
|
|
|
#include <linux/io.h>
|
|
|
|
#include "./common.h"
|
|
|
|
#include "./pipe.h"
|
|
|
|
|
2011-06-06 13:18:07 +08:00
|
|
|
/*
|
|
|
|
* packet info function
|
|
|
|
*/
|
|
|
|
void usbhs_pkt_update(struct usbhs_pkt *pkt,
|
|
|
|
struct usbhs_pipe *pipe,
|
|
|
|
void *buf, int len)
|
|
|
|
{
|
|
|
|
pkt->pipe = pipe;
|
|
|
|
pkt->buf = buf;
|
|
|
|
pkt->length = len;
|
|
|
|
pkt->actual = 0;
|
|
|
|
pkt->maxp = 0;
|
|
|
|
}
|
|
|
|
|
2011-06-06 13:18:03 +08:00
|
|
|
/*
|
|
|
|
* FIFO ctrl
|
|
|
|
*/
|
|
|
|
static void usbhsf_send_terminator(struct usbhs_pipe *pipe)
|
|
|
|
{
|
|
|
|
struct usbhs_priv *priv = usbhs_pipe_to_priv(pipe);
|
|
|
|
|
|
|
|
usbhs_bset(priv, CFIFOCTR, BVAL, BVAL);
|
|
|
|
}
|
|
|
|
|
|
|
|
static int usbhsf_fifo_barrier(struct usbhs_priv *priv)
|
|
|
|
{
|
|
|
|
int timeout = 1024;
|
|
|
|
|
|
|
|
do {
|
|
|
|
/* The FIFO port is accessible */
|
|
|
|
if (usbhs_read(priv, CFIFOCTR) & FRDY)
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
udelay(10);
|
|
|
|
} while (timeout--);
|
|
|
|
|
|
|
|
return -EBUSY;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void usbhsf_fifo_clear(struct usbhs_pipe *pipe)
|
|
|
|
{
|
|
|
|
struct usbhs_priv *priv = usbhs_pipe_to_priv(pipe);
|
|
|
|
|
|
|
|
if (!usbhs_pipe_is_dcp(pipe))
|
|
|
|
usbhsf_fifo_barrier(priv);
|
|
|
|
|
|
|
|
usbhs_write(priv, CFIFOCTR, BCLR);
|
|
|
|
}
|
|
|
|
|
|
|
|
static int usbhsf_fifo_rcv_len(struct usbhs_priv *priv)
|
|
|
|
{
|
|
|
|
return usbhs_read(priv, CFIFOCTR) & DTLN_MASK;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int usbhsf_fifo_select(struct usbhs_pipe *pipe, int write)
|
|
|
|
{
|
|
|
|
struct usbhs_priv *priv = usbhs_pipe_to_priv(pipe);
|
|
|
|
struct device *dev = usbhs_priv_to_dev(priv);
|
|
|
|
int timeout = 1024;
|
|
|
|
u16 mask = ((1 << 5) | 0xF); /* mask of ISEL | CURPIPE */
|
|
|
|
u16 base = usbhs_pipe_number(pipe); /* CURPIPE */
|
|
|
|
|
|
|
|
if (usbhs_pipe_is_dcp(pipe))
|
|
|
|
base |= (1 == write) << 5; /* ISEL */
|
|
|
|
|
|
|
|
/* "base" will be used below */
|
|
|
|
usbhs_write(priv, CFIFOSEL, base | MBW_32);
|
|
|
|
|
|
|
|
/* check ISEL and CURPIPE value */
|
|
|
|
while (timeout--) {
|
|
|
|
if (base == (mask & usbhs_read(priv, CFIFOSEL)))
|
|
|
|
return 0;
|
|
|
|
udelay(10);
|
|
|
|
}
|
|
|
|
|
|
|
|
dev_err(dev, "fifo select error\n");
|
|
|
|
|
|
|
|
return -EIO;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* PIO fifo functions
|
|
|
|
*/
|
|
|
|
int usbhs_fifo_prepare_write(struct usbhs_pipe *pipe)
|
|
|
|
{
|
|
|
|
return usbhsf_fifo_select(pipe, 1);
|
|
|
|
}
|
|
|
|
|
2011-06-06 13:18:07 +08:00
|
|
|
int usbhs_fifo_write(struct usbhs_pkt *pkt)
|
2011-06-06 13:18:03 +08:00
|
|
|
{
|
2011-06-06 13:18:07 +08:00
|
|
|
struct usbhs_pipe *pipe = pkt->pipe;
|
2011-06-06 13:18:03 +08:00
|
|
|
struct usbhs_priv *priv = usbhs_pipe_to_priv(pipe);
|
2011-06-06 13:18:07 +08:00
|
|
|
struct usbhs_pipe_info *info = usbhs_priv_to_pipeinfo(priv);
|
2011-06-06 13:18:03 +08:00
|
|
|
void __iomem *addr = priv->base + CFIFO;
|
|
|
|
int maxp = usbhs_pipe_get_maxpacket(pipe);
|
|
|
|
int total_len;
|
2011-06-06 13:18:07 +08:00
|
|
|
u8 *buf = pkt->buf;
|
|
|
|
int i, ret, len;
|
2011-06-06 13:18:03 +08:00
|
|
|
|
|
|
|
ret = usbhs_pipe_is_accessible(pipe);
|
|
|
|
if (ret < 0)
|
|
|
|
return ret;
|
|
|
|
|
|
|
|
ret = usbhsf_fifo_select(pipe, 1);
|
|
|
|
if (ret < 0)
|
|
|
|
return ret;
|
|
|
|
|
|
|
|
ret = usbhsf_fifo_barrier(priv);
|
|
|
|
if (ret < 0)
|
|
|
|
return ret;
|
|
|
|
|
2011-06-06 13:18:07 +08:00
|
|
|
len = min(pkt->length, maxp);
|
2011-06-06 13:18:03 +08:00
|
|
|
total_len = len;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* FIXME
|
|
|
|
*
|
|
|
|
* 32-bit access only
|
|
|
|
*/
|
|
|
|
if (len >= 4 &&
|
|
|
|
!((unsigned long)buf & 0x03)) {
|
|
|
|
iowrite32_rep(addr, buf, len / 4);
|
|
|
|
len %= 4;
|
|
|
|
buf += total_len - len;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* the rest operation */
|
|
|
|
for (i = 0; i < len; i++)
|
|
|
|
iowrite8(buf[i], addr + (0x03 - (i & 0x03)));
|
|
|
|
|
|
|
|
if (total_len < maxp)
|
|
|
|
usbhsf_send_terminator(pipe);
|
|
|
|
|
2011-06-06 13:18:07 +08:00
|
|
|
usbhs_pipe_enable(pipe);
|
|
|
|
|
|
|
|
/* update pkt */
|
|
|
|
if (info->tx_done) {
|
|
|
|
pkt->actual = total_len;
|
|
|
|
pkt->maxp = maxp;
|
|
|
|
info->tx_done(pkt);
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
2011-06-06 13:18:03 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
int usbhs_fifo_prepare_read(struct usbhs_pipe *pipe)
|
|
|
|
{
|
|
|
|
int ret;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* select pipe and enable it to prepare packet receive
|
|
|
|
*/
|
|
|
|
ret = usbhsf_fifo_select(pipe, 0);
|
|
|
|
if (ret < 0)
|
|
|
|
return ret;
|
|
|
|
|
|
|
|
usbhs_pipe_enable(pipe);
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2011-06-06 13:18:07 +08:00
|
|
|
int usbhs_fifo_read(struct usbhs_pkt *pkt)
|
2011-06-06 13:18:03 +08:00
|
|
|
{
|
2011-06-06 13:18:07 +08:00
|
|
|
struct usbhs_pipe *pipe = pkt->pipe;
|
2011-06-06 13:18:03 +08:00
|
|
|
struct usbhs_priv *priv = usbhs_pipe_to_priv(pipe);
|
2011-06-06 13:18:07 +08:00
|
|
|
struct usbhs_pipe_info *info = usbhs_priv_to_pipeinfo(priv);
|
2011-06-06 13:18:03 +08:00
|
|
|
void __iomem *addr = priv->base + CFIFO;
|
2011-06-06 13:18:07 +08:00
|
|
|
u8 *buf = pkt->buf;
|
|
|
|
int rcv_len, len;
|
2011-06-06 13:18:03 +08:00
|
|
|
int i, ret;
|
2011-06-06 13:18:07 +08:00
|
|
|
int total_len = 0;
|
2011-06-06 13:18:03 +08:00
|
|
|
u32 data = 0;
|
|
|
|
|
|
|
|
ret = usbhsf_fifo_select(pipe, 0);
|
|
|
|
if (ret < 0)
|
|
|
|
return ret;
|
|
|
|
|
|
|
|
ret = usbhsf_fifo_barrier(priv);
|
|
|
|
if (ret < 0)
|
|
|
|
return ret;
|
|
|
|
|
|
|
|
rcv_len = usbhsf_fifo_rcv_len(priv);
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Buffer clear if Zero-Length packet
|
|
|
|
*
|
|
|
|
* see
|
|
|
|
* "Operation" - "FIFO Buffer Memory" - "FIFO Port Function"
|
|
|
|
*/
|
|
|
|
if (0 == rcv_len) {
|
|
|
|
usbhsf_fifo_clear(pipe);
|
2011-06-06 13:18:07 +08:00
|
|
|
goto usbhs_fifo_read_end;
|
2011-06-06 13:18:03 +08:00
|
|
|
}
|
|
|
|
|
2011-06-06 13:18:07 +08:00
|
|
|
len = min(rcv_len, pkt->length);
|
2011-06-06 13:18:03 +08:00
|
|
|
total_len = len;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* FIXME
|
|
|
|
*
|
|
|
|
* 32-bit access only
|
|
|
|
*/
|
|
|
|
if (len >= 4 &&
|
|
|
|
!((unsigned long)buf & 0x03)) {
|
|
|
|
ioread32_rep(addr, buf, len / 4);
|
|
|
|
len %= 4;
|
|
|
|
buf += rcv_len - len;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* the rest operation */
|
|
|
|
for (i = 0; i < len; i++) {
|
|
|
|
if (!(i & 0x03))
|
|
|
|
data = ioread32(addr);
|
|
|
|
|
|
|
|
buf[i] = (data >> ((i & 0x03) * 8)) & 0xff;
|
|
|
|
}
|
|
|
|
|
2011-06-06 13:18:07 +08:00
|
|
|
usbhs_fifo_read_end:
|
|
|
|
if (info->rx_done) {
|
|
|
|
/* update pkt */
|
|
|
|
pkt->actual = total_len;
|
|
|
|
pkt->maxp = usbhs_pipe_get_maxpacket(pipe);
|
|
|
|
info->rx_done(pkt);
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
2011-06-06 13:18:03 +08:00
|
|
|
}
|