| Kuninori Morimoto | e8d548d | 2011-06-06 14:18:03 +0900 | [diff] [blame] | 1 | /* | 
|  | 2 | * Renesas USB driver | 
|  | 3 | * | 
|  | 4 | * Copyright (C) 2011 Renesas Solutions Corp. | 
|  | 5 | * Kuninori Morimoto <kuninori.morimoto.gx@renesas.com> | 
|  | 6 | * | 
|  | 7 | * This program is distributed in the hope that it will be useful, | 
|  | 8 | * but WITHOUT ANY WARRANTY; without even the implied warranty of | 
|  | 9 | * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | 
|  | 10 | * GNU General Public License for more details. | 
|  | 11 | * | 
|  | 12 | * You should have received a copy of the GNU General Public License | 
|  | 13 | * along with this program; if not, write to the Free Software | 
|  | 14 | * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA  02110-1301  USA | 
|  | 15 | * | 
|  | 16 | */ | 
|  | 17 | #include <linux/delay.h> | 
|  | 18 | #include <linux/io.h> | 
| Rafael J. Wysocki | 9c646cf | 2011-07-26 20:51:01 +0200 | [diff] [blame] | 19 | #include <linux/scatterlist.h> | 
| Paul Bolle | cc502bb | 2012-06-03 18:39:13 +0200 | [diff] [blame] | 20 | #include "common.h" | 
|  | 21 | #include "pipe.h" | 
| Kuninori Morimoto | e8d548d | 2011-06-06 14:18:03 +0900 | [diff] [blame] | 22 |  | 
| Kuninori Morimoto | d3af90a | 2011-06-06 14:18:44 +0900 | [diff] [blame] | 23 | #define usbhsf_get_cfifo(p)	(&((p)->fifo_info.cfifo)) | 
| Kuninori Morimoto | e73a989 | 2011-06-06 14:19:03 +0900 | [diff] [blame] | 24 | #define usbhsf_get_d0fifo(p)	(&((p)->fifo_info.d0fifo)) | 
|  | 25 | #define usbhsf_get_d1fifo(p)	(&((p)->fifo_info.d1fifo)) | 
| Shimoda, Yoshihiro | 5ea4399 | 2012-01-05 15:37:22 +0900 | [diff] [blame] | 26 | #define usbhsf_is_cfifo(p, f)	(usbhsf_get_cfifo(p) == f) | 
| Kuninori Morimoto | d3af90a | 2011-06-06 14:18:44 +0900 | [diff] [blame] | 27 |  | 
| Kuninori Morimoto | d77e3f4 | 2011-06-06 14:18:50 +0900 | [diff] [blame] | 28 | #define usbhsf_fifo_is_busy(f)	((f)->pipe) /* see usbhs_pipe_select_fifo */ | 
|  | 29 |  | 
| Kuninori Morimoto | e8d548d | 2011-06-06 14:18:03 +0900 | [diff] [blame] | 30 | /* | 
| Kuninori Morimoto | 233f519 | 2011-07-07 00:23:24 -0700 | [diff] [blame] | 31 | *		packet initialize | 
|  | 32 | */ | 
|  | 33 | void usbhs_pkt_init(struct usbhs_pkt *pkt) | 
|  | 34 | { | 
|  | 35 | pkt->dma = DMA_ADDR_INVALID; | 
|  | 36 | INIT_LIST_HEAD(&pkt->node); | 
|  | 37 | } | 
|  | 38 |  | 
|  | 39 | /* | 
|  | 40 | *		packet control function | 
| Kuninori Morimoto | 4bd0481 | 2011-06-06 14:18:07 +0900 | [diff] [blame] | 41 | */ | 
| Kuninori Morimoto | 97664a2 | 2011-06-06 14:18:38 +0900 | [diff] [blame] | 42 | static int usbhsf_null_handle(struct usbhs_pkt *pkt, int *is_done) | 
| Kuninori Morimoto | dad6739 | 2011-06-06 14:18:28 +0900 | [diff] [blame] | 43 | { | 
|  | 44 | struct usbhs_priv *priv = usbhs_pipe_to_priv(pkt->pipe); | 
|  | 45 | struct device *dev = usbhs_priv_to_dev(priv); | 
|  | 46 |  | 
|  | 47 | dev_err(dev, "null handler\n"); | 
|  | 48 |  | 
|  | 49 | return -EINVAL; | 
|  | 50 | } | 
|  | 51 |  | 
|  | 52 | static struct usbhs_pkt_handle usbhsf_null_handler = { | 
|  | 53 | .prepare = usbhsf_null_handle, | 
|  | 54 | .try_run = usbhsf_null_handle, | 
|  | 55 | }; | 
|  | 56 |  | 
| Kuninori Morimoto | 659d495 | 2011-06-06 14:18:23 +0900 | [diff] [blame] | 57 | void usbhs_pkt_push(struct usbhs_pipe *pipe, struct usbhs_pkt *pkt, | 
| Kuninori Morimoto | b331872 | 2011-10-10 22:04:41 -0700 | [diff] [blame] | 58 | void (*done)(struct usbhs_priv *priv, | 
|  | 59 | struct usbhs_pkt *pkt), | 
| Kuninori Morimoto | 3edeee3 | 2011-12-08 18:28:54 -0800 | [diff] [blame] | 60 | void *buf, int len, int zero, int sequence) | 
| Kuninori Morimoto | 6acb95d | 2011-06-06 14:18:16 +0900 | [diff] [blame] | 61 | { | 
| Kuninori Morimoto | dad6739 | 2011-06-06 14:18:28 +0900 | [diff] [blame] | 62 | struct usbhs_priv *priv = usbhs_pipe_to_priv(pipe); | 
|  | 63 | struct device *dev = usbhs_priv_to_dev(priv); | 
| Kuninori Morimoto | 97664a2 | 2011-06-06 14:18:38 +0900 | [diff] [blame] | 64 | unsigned long flags; | 
|  | 65 |  | 
| Kuninori Morimoto | b331872 | 2011-10-10 22:04:41 -0700 | [diff] [blame] | 66 | if (!done) { | 
|  | 67 | dev_err(dev, "no done function\n"); | 
|  | 68 | return; | 
|  | 69 | } | 
|  | 70 |  | 
| Kuninori Morimoto | a2c76b8 | 2011-10-18 20:05:50 -0700 | [diff] [blame] | 71 | /********************  spin lock ********************/ | 
|  | 72 | usbhs_lock(priv, flags); | 
|  | 73 |  | 
| Kuninori Morimoto | 0c6ef98 | 2011-10-10 22:00:59 -0700 | [diff] [blame] | 74 | if (!pipe->handler) { | 
| Kuninori Morimoto | dad6739 | 2011-06-06 14:18:28 +0900 | [diff] [blame] | 75 | dev_err(dev, "no handler function\n"); | 
| Kuninori Morimoto | 0c6ef98 | 2011-10-10 22:00:59 -0700 | [diff] [blame] | 76 | pipe->handler = &usbhsf_null_handler; | 
| Kuninori Morimoto | dad6739 | 2011-06-06 14:18:28 +0900 | [diff] [blame] | 77 | } | 
|  | 78 |  | 
| Guennadi Liakhovetski | d526128 | 2012-02-14 11:37:17 +0100 | [diff] [blame] | 79 | list_move_tail(&pkt->node, &pipe->list); | 
| Kuninori Morimoto | 6acb95d | 2011-06-06 14:18:16 +0900 | [diff] [blame] | 80 |  | 
| Kuninori Morimoto | 0c6ef98 | 2011-10-10 22:00:59 -0700 | [diff] [blame] | 81 | /* | 
|  | 82 | * each pkt must hold own handler. | 
|  | 83 | * because handler might be changed by its situation. | 
|  | 84 | * dma handler -> pio handler. | 
|  | 85 | */ | 
| Kuninori Morimoto | 659d495 | 2011-06-06 14:18:23 +0900 | [diff] [blame] | 86 | pkt->pipe	= pipe; | 
|  | 87 | pkt->buf	= buf; | 
| Kuninori Morimoto | 0c6ef98 | 2011-10-10 22:00:59 -0700 | [diff] [blame] | 88 | pkt->handler	= pipe->handler; | 
| Kuninori Morimoto | 659d495 | 2011-06-06 14:18:23 +0900 | [diff] [blame] | 89 | pkt->length	= len; | 
|  | 90 | pkt->zero	= zero; | 
|  | 91 | pkt->actual	= 0; | 
| Kuninori Morimoto | b331872 | 2011-10-10 22:04:41 -0700 | [diff] [blame] | 92 | pkt->done	= done; | 
| Kuninori Morimoto | 3edeee3 | 2011-12-08 18:28:54 -0800 | [diff] [blame] | 93 | pkt->sequence	= sequence; | 
| Kuninori Morimoto | 97664a2 | 2011-06-06 14:18:38 +0900 | [diff] [blame] | 94 |  | 
|  | 95 | usbhs_unlock(priv, flags); | 
|  | 96 | /********************  spin unlock ******************/ | 
| Kuninori Morimoto | 6acb95d | 2011-06-06 14:18:16 +0900 | [diff] [blame] | 97 | } | 
|  | 98 |  | 
| Kuninori Morimoto | 97664a2 | 2011-06-06 14:18:38 +0900 | [diff] [blame] | 99 | static void __usbhsf_pkt_del(struct usbhs_pkt *pkt) | 
| Kuninori Morimoto | 6acb95d | 2011-06-06 14:18:16 +0900 | [diff] [blame] | 100 | { | 
|  | 101 | list_del_init(&pkt->node); | 
|  | 102 | } | 
|  | 103 |  | 
| Kuninori Morimoto | 97664a2 | 2011-06-06 14:18:38 +0900 | [diff] [blame] | 104 | static struct usbhs_pkt *__usbhsf_pkt_get(struct usbhs_pipe *pipe) | 
| Kuninori Morimoto | 6acb95d | 2011-06-06 14:18:16 +0900 | [diff] [blame] | 105 | { | 
|  | 106 | if (list_empty(&pipe->list)) | 
|  | 107 | return NULL; | 
|  | 108 |  | 
| Guennadi Liakhovetski | d526128 | 2012-02-14 11:37:17 +0100 | [diff] [blame] | 109 | return list_first_entry(&pipe->list, struct usbhs_pkt, node); | 
| Kuninori Morimoto | 6acb95d | 2011-06-06 14:18:16 +0900 | [diff] [blame] | 110 | } | 
|  | 111 |  | 
| Kuninori Morimoto | 97664a2 | 2011-06-06 14:18:38 +0900 | [diff] [blame] | 112 | struct usbhs_pkt *usbhs_pkt_pop(struct usbhs_pipe *pipe, struct usbhs_pkt *pkt) | 
|  | 113 | { | 
|  | 114 | struct usbhs_priv *priv = usbhs_pipe_to_priv(pipe); | 
|  | 115 | unsigned long flags; | 
|  | 116 |  | 
|  | 117 | /********************  spin lock ********************/ | 
|  | 118 | usbhs_lock(priv, flags); | 
|  | 119 |  | 
|  | 120 | if (!pkt) | 
|  | 121 | pkt = __usbhsf_pkt_get(pipe); | 
|  | 122 |  | 
|  | 123 | if (pkt) | 
|  | 124 | __usbhsf_pkt_del(pkt); | 
|  | 125 |  | 
|  | 126 | usbhs_unlock(priv, flags); | 
|  | 127 | /********************  spin unlock ******************/ | 
|  | 128 |  | 
|  | 129 | return pkt; | 
|  | 130 | } | 
|  | 131 |  | 
| Kuninori Morimoto | 51b8a02 | 2011-10-10 21:58:45 -0700 | [diff] [blame] | 132 | enum { | 
|  | 133 | USBHSF_PKT_PREPARE, | 
|  | 134 | USBHSF_PKT_TRY_RUN, | 
|  | 135 | USBHSF_PKT_DMA_DONE, | 
|  | 136 | }; | 
|  | 137 |  | 
|  | 138 | static int usbhsf_pkt_handler(struct usbhs_pipe *pipe, int type) | 
| Kuninori Morimoto | 97664a2 | 2011-06-06 14:18:38 +0900 | [diff] [blame] | 139 | { | 
|  | 140 | struct usbhs_priv *priv = usbhs_pipe_to_priv(pipe); | 
| Kuninori Morimoto | 97664a2 | 2011-06-06 14:18:38 +0900 | [diff] [blame] | 141 | struct usbhs_pkt *pkt; | 
|  | 142 | struct device *dev = usbhs_priv_to_dev(priv); | 
|  | 143 | int (*func)(struct usbhs_pkt *pkt, int *is_done); | 
|  | 144 | unsigned long flags; | 
|  | 145 | int ret = 0; | 
|  | 146 | int is_done = 0; | 
|  | 147 |  | 
|  | 148 | /********************  spin lock ********************/ | 
|  | 149 | usbhs_lock(priv, flags); | 
|  | 150 |  | 
|  | 151 | pkt = __usbhsf_pkt_get(pipe); | 
|  | 152 | if (!pkt) | 
|  | 153 | goto __usbhs_pkt_handler_end; | 
|  | 154 |  | 
|  | 155 | switch (type) { | 
|  | 156 | case USBHSF_PKT_PREPARE: | 
|  | 157 | func = pkt->handler->prepare; | 
|  | 158 | break; | 
|  | 159 | case USBHSF_PKT_TRY_RUN: | 
|  | 160 | func = pkt->handler->try_run; | 
|  | 161 | break; | 
| Kuninori Morimoto | e73a989 | 2011-06-06 14:19:03 +0900 | [diff] [blame] | 162 | case USBHSF_PKT_DMA_DONE: | 
|  | 163 | func = pkt->handler->dma_done; | 
|  | 164 | break; | 
| Kuninori Morimoto | 97664a2 | 2011-06-06 14:18:38 +0900 | [diff] [blame] | 165 | default: | 
|  | 166 | dev_err(dev, "unknown pkt hander\n"); | 
|  | 167 | goto __usbhs_pkt_handler_end; | 
|  | 168 | } | 
|  | 169 |  | 
|  | 170 | ret = func(pkt, &is_done); | 
|  | 171 |  | 
|  | 172 | if (is_done) | 
|  | 173 | __usbhsf_pkt_del(pkt); | 
|  | 174 |  | 
|  | 175 | __usbhs_pkt_handler_end: | 
|  | 176 | usbhs_unlock(priv, flags); | 
|  | 177 | /********************  spin unlock ******************/ | 
|  | 178 |  | 
| Kuninori Morimoto | 0432eed | 2011-06-06 14:18:54 +0900 | [diff] [blame] | 179 | if (is_done) { | 
| Kuninori Morimoto | b331872 | 2011-10-10 22:04:41 -0700 | [diff] [blame] | 180 | pkt->done(priv, pkt); | 
| Kuninori Morimoto | 0432eed | 2011-06-06 14:18:54 +0900 | [diff] [blame] | 181 | usbhs_pkt_start(pipe); | 
|  | 182 | } | 
| Kuninori Morimoto | 97664a2 | 2011-06-06 14:18:38 +0900 | [diff] [blame] | 183 |  | 
|  | 184 | return ret; | 
|  | 185 | } | 
|  | 186 |  | 
| Kuninori Morimoto | 51b8a02 | 2011-10-10 21:58:45 -0700 | [diff] [blame] | 187 | void usbhs_pkt_start(struct usbhs_pipe *pipe) | 
|  | 188 | { | 
|  | 189 | usbhsf_pkt_handler(pipe, USBHSF_PKT_PREPARE); | 
|  | 190 | } | 
|  | 191 |  | 
| Kuninori Morimoto | 4bd0481 | 2011-06-06 14:18:07 +0900 | [diff] [blame] | 192 | /* | 
| Kuninori Morimoto | 659d495 | 2011-06-06 14:18:23 +0900 | [diff] [blame] | 193 | *		irq enable/disable function | 
|  | 194 | */ | 
|  | 195 | #define usbhsf_irq_empty_ctrl(p, e) usbhsf_irq_callback_ctrl(p, bempsts, e) | 
|  | 196 | #define usbhsf_irq_ready_ctrl(p, e) usbhsf_irq_callback_ctrl(p, brdysts, e) | 
|  | 197 | #define usbhsf_irq_callback_ctrl(pipe, status, enable)			\ | 
|  | 198 | ({								\ | 
|  | 199 | struct usbhs_priv *priv = usbhs_pipe_to_priv(pipe);	\ | 
|  | 200 | struct usbhs_mod *mod = usbhs_mod_get_current(priv);	\ | 
|  | 201 | u16 status = (1 << usbhs_pipe_number(pipe));		\ | 
|  | 202 | if (!mod)						\ | 
|  | 203 | return;						\ | 
|  | 204 | if (enable)						\ | 
|  | 205 | mod->irq_##status |= status;			\ | 
|  | 206 | else							\ | 
|  | 207 | mod->irq_##status &= ~status;			\ | 
|  | 208 | usbhs_irq_callback_update(priv, mod);			\ | 
|  | 209 | }) | 
|  | 210 |  | 
|  | 211 | static void usbhsf_tx_irq_ctrl(struct usbhs_pipe *pipe, int enable) | 
|  | 212 | { | 
|  | 213 | /* | 
|  | 214 | * And DCP pipe can NOT use "ready interrupt" for "send" | 
|  | 215 | * it should use "empty" interrupt. | 
|  | 216 | * see | 
|  | 217 | *   "Operation" - "Interrupt Function" - "BRDY Interrupt" | 
|  | 218 | * | 
|  | 219 | * on the other hand, normal pipe can use "ready interrupt" for "send" | 
|  | 220 | * even though it is single/double buffer | 
|  | 221 | */ | 
|  | 222 | if (usbhs_pipe_is_dcp(pipe)) | 
|  | 223 | usbhsf_irq_empty_ctrl(pipe, enable); | 
|  | 224 | else | 
|  | 225 | usbhsf_irq_ready_ctrl(pipe, enable); | 
|  | 226 | } | 
|  | 227 |  | 
|  | 228 | static void usbhsf_rx_irq_ctrl(struct usbhs_pipe *pipe, int enable) | 
|  | 229 | { | 
|  | 230 | usbhsf_irq_ready_ctrl(pipe, enable); | 
|  | 231 | } | 
|  | 232 |  | 
|  | 233 | /* | 
| Kuninori Morimoto | e8d548d | 2011-06-06 14:18:03 +0900 | [diff] [blame] | 234 | *		FIFO ctrl | 
|  | 235 | */ | 
| Kuninori Morimoto | d3af90a | 2011-06-06 14:18:44 +0900 | [diff] [blame] | 236 | static void usbhsf_send_terminator(struct usbhs_pipe *pipe, | 
|  | 237 | struct usbhs_fifo *fifo) | 
| Kuninori Morimoto | e8d548d | 2011-06-06 14:18:03 +0900 | [diff] [blame] | 238 | { | 
|  | 239 | struct usbhs_priv *priv = usbhs_pipe_to_priv(pipe); | 
|  | 240 |  | 
| Kuninori Morimoto | d3af90a | 2011-06-06 14:18:44 +0900 | [diff] [blame] | 241 | usbhs_bset(priv, fifo->ctr, BVAL, BVAL); | 
| Kuninori Morimoto | e8d548d | 2011-06-06 14:18:03 +0900 | [diff] [blame] | 242 | } | 
|  | 243 |  | 
| Kuninori Morimoto | d3af90a | 2011-06-06 14:18:44 +0900 | [diff] [blame] | 244 | static int usbhsf_fifo_barrier(struct usbhs_priv *priv, | 
|  | 245 | struct usbhs_fifo *fifo) | 
| Kuninori Morimoto | e8d548d | 2011-06-06 14:18:03 +0900 | [diff] [blame] | 246 | { | 
|  | 247 | int timeout = 1024; | 
|  | 248 |  | 
|  | 249 | do { | 
|  | 250 | /* The FIFO port is accessible */ | 
| Kuninori Morimoto | d3af90a | 2011-06-06 14:18:44 +0900 | [diff] [blame] | 251 | if (usbhs_read(priv, fifo->ctr) & FRDY) | 
| Kuninori Morimoto | e8d548d | 2011-06-06 14:18:03 +0900 | [diff] [blame] | 252 | return 0; | 
|  | 253 |  | 
|  | 254 | udelay(10); | 
|  | 255 | } while (timeout--); | 
|  | 256 |  | 
|  | 257 | return -EBUSY; | 
|  | 258 | } | 
|  | 259 |  | 
| Kuninori Morimoto | d3af90a | 2011-06-06 14:18:44 +0900 | [diff] [blame] | 260 | static void usbhsf_fifo_clear(struct usbhs_pipe *pipe, | 
|  | 261 | struct usbhs_fifo *fifo) | 
| Kuninori Morimoto | e8d548d | 2011-06-06 14:18:03 +0900 | [diff] [blame] | 262 | { | 
|  | 263 | struct usbhs_priv *priv = usbhs_pipe_to_priv(pipe); | 
|  | 264 |  | 
|  | 265 | if (!usbhs_pipe_is_dcp(pipe)) | 
| Kuninori Morimoto | d3af90a | 2011-06-06 14:18:44 +0900 | [diff] [blame] | 266 | usbhsf_fifo_barrier(priv, fifo); | 
| Kuninori Morimoto | e8d548d | 2011-06-06 14:18:03 +0900 | [diff] [blame] | 267 |  | 
| Kuninori Morimoto | d3af90a | 2011-06-06 14:18:44 +0900 | [diff] [blame] | 268 | usbhs_write(priv, fifo->ctr, BCLR); | 
| Kuninori Morimoto | e8d548d | 2011-06-06 14:18:03 +0900 | [diff] [blame] | 269 | } | 
|  | 270 |  | 
| Kuninori Morimoto | d3af90a | 2011-06-06 14:18:44 +0900 | [diff] [blame] | 271 | static int usbhsf_fifo_rcv_len(struct usbhs_priv *priv, | 
|  | 272 | struct usbhs_fifo *fifo) | 
| Kuninori Morimoto | e8d548d | 2011-06-06 14:18:03 +0900 | [diff] [blame] | 273 | { | 
| Kuninori Morimoto | d3af90a | 2011-06-06 14:18:44 +0900 | [diff] [blame] | 274 | return usbhs_read(priv, fifo->ctr) & DTLN_MASK; | 
| Kuninori Morimoto | e8d548d | 2011-06-06 14:18:03 +0900 | [diff] [blame] | 275 | } | 
|  | 276 |  | 
| Kuninori Morimoto | d77e3f4 | 2011-06-06 14:18:50 +0900 | [diff] [blame] | 277 | static void usbhsf_fifo_unselect(struct usbhs_pipe *pipe, | 
|  | 278 | struct usbhs_fifo *fifo) | 
|  | 279 | { | 
|  | 280 | struct usbhs_priv *priv = usbhs_pipe_to_priv(pipe); | 
|  | 281 |  | 
|  | 282 | usbhs_pipe_select_fifo(pipe, NULL); | 
|  | 283 | usbhs_write(priv, fifo->sel, 0); | 
|  | 284 | } | 
|  | 285 |  | 
| Kuninori Morimoto | d3af90a | 2011-06-06 14:18:44 +0900 | [diff] [blame] | 286 | static int usbhsf_fifo_select(struct usbhs_pipe *pipe, | 
|  | 287 | struct usbhs_fifo *fifo, | 
|  | 288 | int write) | 
| Kuninori Morimoto | e8d548d | 2011-06-06 14:18:03 +0900 | [diff] [blame] | 289 | { | 
|  | 290 | struct usbhs_priv *priv = usbhs_pipe_to_priv(pipe); | 
|  | 291 | struct device *dev = usbhs_priv_to_dev(priv); | 
|  | 292 | int timeout = 1024; | 
|  | 293 | u16 mask = ((1 << 5) | 0xF);		/* mask of ISEL | CURPIPE */ | 
|  | 294 | u16 base = usbhs_pipe_number(pipe);	/* CURPIPE */ | 
|  | 295 |  | 
| Kuninori Morimoto | d77e3f4 | 2011-06-06 14:18:50 +0900 | [diff] [blame] | 296 | if (usbhs_pipe_is_busy(pipe) || | 
|  | 297 | usbhsf_fifo_is_busy(fifo)) | 
|  | 298 | return -EBUSY; | 
|  | 299 |  | 
| Kuninori Morimoto | 9235207 | 2011-10-10 22:02:57 -0700 | [diff] [blame] | 300 | if (usbhs_pipe_is_dcp(pipe)) { | 
| Kuninori Morimoto | e8d548d | 2011-06-06 14:18:03 +0900 | [diff] [blame] | 301 | base |= (1 == write) << 5;	/* ISEL */ | 
|  | 302 |  | 
| Kuninori Morimoto | 9235207 | 2011-10-10 22:02:57 -0700 | [diff] [blame] | 303 | if (usbhs_mod_is_host(priv)) | 
|  | 304 | usbhs_dcp_dir_for_host(pipe, write); | 
|  | 305 | } | 
|  | 306 |  | 
| Kuninori Morimoto | e8d548d | 2011-06-06 14:18:03 +0900 | [diff] [blame] | 307 | /* "base" will be used below  */ | 
| Shimoda, Yoshihiro | 5ea4399 | 2012-01-05 15:37:22 +0900 | [diff] [blame] | 308 | if (usbhs_get_dparam(priv, has_sudmac) && !usbhsf_is_cfifo(priv, fifo)) | 
|  | 309 | usbhs_write(priv, fifo->sel, base); | 
|  | 310 | else | 
|  | 311 | usbhs_write(priv, fifo->sel, base | MBW_32); | 
| Kuninori Morimoto | e8d548d | 2011-06-06 14:18:03 +0900 | [diff] [blame] | 312 |  | 
|  | 313 | /* check ISEL and CURPIPE value */ | 
|  | 314 | while (timeout--) { | 
| Kuninori Morimoto | d77e3f4 | 2011-06-06 14:18:50 +0900 | [diff] [blame] | 315 | if (base == (mask & usbhs_read(priv, fifo->sel))) { | 
|  | 316 | usbhs_pipe_select_fifo(pipe, fifo); | 
| Kuninori Morimoto | e8d548d | 2011-06-06 14:18:03 +0900 | [diff] [blame] | 317 | return 0; | 
| Kuninori Morimoto | d77e3f4 | 2011-06-06 14:18:50 +0900 | [diff] [blame] | 318 | } | 
| Kuninori Morimoto | e8d548d | 2011-06-06 14:18:03 +0900 | [diff] [blame] | 319 | udelay(10); | 
|  | 320 | } | 
|  | 321 |  | 
|  | 322 | dev_err(dev, "fifo select error\n"); | 
|  | 323 |  | 
|  | 324 | return -EIO; | 
|  | 325 | } | 
|  | 326 |  | 
|  | 327 | /* | 
| Kuninori Morimoto | 9e74d60 | 2011-10-10 22:07:08 -0700 | [diff] [blame] | 328 | *		DCP status stage | 
|  | 329 | */ | 
|  | 330 | static int usbhs_dcp_dir_switch_to_write(struct usbhs_pkt *pkt, int *is_done) | 
|  | 331 | { | 
|  | 332 | struct usbhs_pipe *pipe = pkt->pipe; | 
|  | 333 | struct usbhs_priv *priv = usbhs_pipe_to_priv(pipe); | 
|  | 334 | struct usbhs_fifo *fifo = usbhsf_get_cfifo(priv); /* CFIFO */ | 
|  | 335 | struct device *dev = usbhs_priv_to_dev(priv); | 
|  | 336 | int ret; | 
|  | 337 |  | 
|  | 338 | usbhs_pipe_disable(pipe); | 
|  | 339 |  | 
|  | 340 | ret = usbhsf_fifo_select(pipe, fifo, 1); | 
|  | 341 | if (ret < 0) { | 
|  | 342 | dev_err(dev, "%s() faile\n", __func__); | 
|  | 343 | return ret; | 
|  | 344 | } | 
|  | 345 |  | 
|  | 346 | usbhs_pipe_sequence_data1(pipe); /* DATA1 */ | 
|  | 347 |  | 
|  | 348 | usbhsf_fifo_clear(pipe, fifo); | 
|  | 349 | usbhsf_send_terminator(pipe, fifo); | 
|  | 350 |  | 
|  | 351 | usbhsf_fifo_unselect(pipe, fifo); | 
|  | 352 |  | 
|  | 353 | usbhsf_tx_irq_ctrl(pipe, 1); | 
|  | 354 | usbhs_pipe_enable(pipe); | 
|  | 355 |  | 
|  | 356 | return ret; | 
|  | 357 | } | 
|  | 358 |  | 
|  | 359 | static int usbhs_dcp_dir_switch_to_read(struct usbhs_pkt *pkt, int *is_done) | 
|  | 360 | { | 
|  | 361 | struct usbhs_pipe *pipe = pkt->pipe; | 
|  | 362 | struct usbhs_priv *priv = usbhs_pipe_to_priv(pipe); | 
|  | 363 | struct usbhs_fifo *fifo = usbhsf_get_cfifo(priv); /* CFIFO */ | 
|  | 364 | struct device *dev = usbhs_priv_to_dev(priv); | 
|  | 365 | int ret; | 
|  | 366 |  | 
|  | 367 | usbhs_pipe_disable(pipe); | 
|  | 368 |  | 
|  | 369 | ret = usbhsf_fifo_select(pipe, fifo, 0); | 
|  | 370 | if (ret < 0) { | 
|  | 371 | dev_err(dev, "%s() fail\n", __func__); | 
|  | 372 | return ret; | 
|  | 373 | } | 
|  | 374 |  | 
|  | 375 | usbhs_pipe_sequence_data1(pipe); /* DATA1 */ | 
|  | 376 | usbhsf_fifo_clear(pipe, fifo); | 
|  | 377 |  | 
|  | 378 | usbhsf_fifo_unselect(pipe, fifo); | 
|  | 379 |  | 
|  | 380 | usbhsf_rx_irq_ctrl(pipe, 1); | 
|  | 381 | usbhs_pipe_enable(pipe); | 
|  | 382 |  | 
|  | 383 | return ret; | 
|  | 384 |  | 
|  | 385 | } | 
|  | 386 |  | 
|  | 387 | static int usbhs_dcp_dir_switch_done(struct usbhs_pkt *pkt, int *is_done) | 
|  | 388 | { | 
|  | 389 | struct usbhs_pipe *pipe = pkt->pipe; | 
|  | 390 |  | 
|  | 391 | if (pkt->handler == &usbhs_dcp_status_stage_in_handler) | 
|  | 392 | usbhsf_tx_irq_ctrl(pipe, 0); | 
|  | 393 | else | 
|  | 394 | usbhsf_rx_irq_ctrl(pipe, 0); | 
|  | 395 |  | 
|  | 396 | pkt->actual = pkt->length; | 
|  | 397 | *is_done = 1; | 
|  | 398 |  | 
|  | 399 | return 0; | 
|  | 400 | } | 
|  | 401 |  | 
|  | 402 | struct usbhs_pkt_handle usbhs_dcp_status_stage_in_handler = { | 
|  | 403 | .prepare = usbhs_dcp_dir_switch_to_write, | 
|  | 404 | .try_run = usbhs_dcp_dir_switch_done, | 
|  | 405 | }; | 
|  | 406 |  | 
|  | 407 | struct usbhs_pkt_handle usbhs_dcp_status_stage_out_handler = { | 
|  | 408 | .prepare = usbhs_dcp_dir_switch_to_read, | 
|  | 409 | .try_run = usbhs_dcp_dir_switch_done, | 
|  | 410 | }; | 
|  | 411 |  | 
|  | 412 | /* | 
|  | 413 | *		DCP data stage (push) | 
|  | 414 | */ | 
|  | 415 | static int usbhsf_dcp_data_stage_try_push(struct usbhs_pkt *pkt, int *is_done) | 
|  | 416 | { | 
|  | 417 | struct usbhs_pipe *pipe = pkt->pipe; | 
|  | 418 |  | 
|  | 419 | usbhs_pipe_sequence_data1(pipe); /* DATA1 */ | 
|  | 420 |  | 
|  | 421 | /* | 
|  | 422 | * change handler to PIO push | 
|  | 423 | */ | 
|  | 424 | pkt->handler = &usbhs_fifo_pio_push_handler; | 
|  | 425 |  | 
|  | 426 | return pkt->handler->prepare(pkt, is_done); | 
|  | 427 | } | 
|  | 428 |  | 
|  | 429 | struct usbhs_pkt_handle usbhs_dcp_data_stage_out_handler = { | 
|  | 430 | .prepare = usbhsf_dcp_data_stage_try_push, | 
|  | 431 | }; | 
|  | 432 |  | 
|  | 433 | /* | 
|  | 434 | *		DCP data stage (pop) | 
|  | 435 | */ | 
|  | 436 | static int usbhsf_dcp_data_stage_prepare_pop(struct usbhs_pkt *pkt, | 
|  | 437 | int *is_done) | 
|  | 438 | { | 
|  | 439 | struct usbhs_pipe *pipe = pkt->pipe; | 
|  | 440 | struct usbhs_priv *priv = usbhs_pipe_to_priv(pipe); | 
|  | 441 | struct usbhs_fifo *fifo = usbhsf_get_cfifo(priv); | 
|  | 442 |  | 
|  | 443 | if (usbhs_pipe_is_busy(pipe)) | 
|  | 444 | return 0; | 
|  | 445 |  | 
|  | 446 | /* | 
|  | 447 | * prepare pop for DCP should | 
|  | 448 | *  - change DCP direction, | 
|  | 449 | *  - clear fifo | 
|  | 450 | *  - DATA1 | 
|  | 451 | */ | 
|  | 452 | usbhs_pipe_disable(pipe); | 
|  | 453 |  | 
|  | 454 | usbhs_pipe_sequence_data1(pipe); /* DATA1 */ | 
|  | 455 |  | 
|  | 456 | usbhsf_fifo_select(pipe, fifo, 0); | 
|  | 457 | usbhsf_fifo_clear(pipe, fifo); | 
|  | 458 | usbhsf_fifo_unselect(pipe, fifo); | 
|  | 459 |  | 
|  | 460 | /* | 
|  | 461 | * change handler to PIO pop | 
|  | 462 | */ | 
|  | 463 | pkt->handler = &usbhs_fifo_pio_pop_handler; | 
|  | 464 |  | 
|  | 465 | return pkt->handler->prepare(pkt, is_done); | 
|  | 466 | } | 
|  | 467 |  | 
|  | 468 | struct usbhs_pkt_handle usbhs_dcp_data_stage_in_handler = { | 
|  | 469 | .prepare = usbhsf_dcp_data_stage_prepare_pop, | 
|  | 470 | }; | 
|  | 471 |  | 
|  | 472 | /* | 
| Kuninori Morimoto | 233f519 | 2011-07-07 00:23:24 -0700 | [diff] [blame] | 473 | *		PIO push handler | 
| Kuninori Morimoto | e8d548d | 2011-06-06 14:18:03 +0900 | [diff] [blame] | 474 | */ | 
| Kuninori Morimoto | 0cb7e61 | 2011-06-06 14:18:58 +0900 | [diff] [blame] | 475 | static int usbhsf_pio_try_push(struct usbhs_pkt *pkt, int *is_done) | 
| Kuninori Morimoto | e8d548d | 2011-06-06 14:18:03 +0900 | [diff] [blame] | 476 | { | 
| Kuninori Morimoto | 4bd0481 | 2011-06-06 14:18:07 +0900 | [diff] [blame] | 477 | struct usbhs_pipe *pipe = pkt->pipe; | 
| Kuninori Morimoto | e8d548d | 2011-06-06 14:18:03 +0900 | [diff] [blame] | 478 | struct usbhs_priv *priv = usbhs_pipe_to_priv(pipe); | 
| Kuninori Morimoto | 659d495 | 2011-06-06 14:18:23 +0900 | [diff] [blame] | 479 | struct device *dev = usbhs_priv_to_dev(priv); | 
| Kuninori Morimoto | d3af90a | 2011-06-06 14:18:44 +0900 | [diff] [blame] | 480 | struct usbhs_fifo *fifo = usbhsf_get_cfifo(priv); /* CFIFO */ | 
|  | 481 | void __iomem *addr = priv->base + fifo->port; | 
| Kuninori Morimoto | 659d495 | 2011-06-06 14:18:23 +0900 | [diff] [blame] | 482 | u8 *buf; | 
| Kuninori Morimoto | e8d548d | 2011-06-06 14:18:03 +0900 | [diff] [blame] | 483 | int maxp = usbhs_pipe_get_maxpacket(pipe); | 
|  | 484 | int total_len; | 
| Kuninori Morimoto | 4bd0481 | 2011-06-06 14:18:07 +0900 | [diff] [blame] | 485 | int i, ret, len; | 
| Kuninori Morimoto | 97664a2 | 2011-06-06 14:18:38 +0900 | [diff] [blame] | 486 | int is_short; | 
| Kuninori Morimoto | e8d548d | 2011-06-06 14:18:03 +0900 | [diff] [blame] | 487 |  | 
| Kuninori Morimoto | 3edeee3 | 2011-12-08 18:28:54 -0800 | [diff] [blame] | 488 | usbhs_pipe_data_sequence(pipe, pkt->sequence); | 
|  | 489 | pkt->sequence = -1; /* -1 sequence will be ignored */ | 
|  | 490 |  | 
| Kuninori Morimoto | d3af90a | 2011-06-06 14:18:44 +0900 | [diff] [blame] | 491 | ret = usbhsf_fifo_select(pipe, fifo, 1); | 
| Kuninori Morimoto | e8d548d | 2011-06-06 14:18:03 +0900 | [diff] [blame] | 492 | if (ret < 0) | 
| Kuninori Morimoto | d77e3f4 | 2011-06-06 14:18:50 +0900 | [diff] [blame] | 493 | return 0; | 
| Kuninori Morimoto | e8d548d | 2011-06-06 14:18:03 +0900 | [diff] [blame] | 494 |  | 
| Kuninori Morimoto | dad6739 | 2011-06-06 14:18:28 +0900 | [diff] [blame] | 495 | ret = usbhs_pipe_is_accessible(pipe); | 
| Kuninori Morimoto | 4ef85e0 | 2011-07-03 17:42:47 -0700 | [diff] [blame] | 496 | if (ret < 0) { | 
|  | 497 | /* inaccessible pipe is not an error */ | 
|  | 498 | ret = 0; | 
| Kuninori Morimoto | 659d495 | 2011-06-06 14:18:23 +0900 | [diff] [blame] | 499 | goto usbhs_fifo_write_busy; | 
| Kuninori Morimoto | 4ef85e0 | 2011-07-03 17:42:47 -0700 | [diff] [blame] | 500 | } | 
| Kuninori Morimoto | e8d548d | 2011-06-06 14:18:03 +0900 | [diff] [blame] | 501 |  | 
| Kuninori Morimoto | d3af90a | 2011-06-06 14:18:44 +0900 | [diff] [blame] | 502 | ret = usbhsf_fifo_barrier(priv, fifo); | 
| Kuninori Morimoto | e8d548d | 2011-06-06 14:18:03 +0900 | [diff] [blame] | 503 | if (ret < 0) | 
| Kuninori Morimoto | 659d495 | 2011-06-06 14:18:23 +0900 | [diff] [blame] | 504 | goto usbhs_fifo_write_busy; | 
| Kuninori Morimoto | e8d548d | 2011-06-06 14:18:03 +0900 | [diff] [blame] | 505 |  | 
| Kuninori Morimoto | 659d495 | 2011-06-06 14:18:23 +0900 | [diff] [blame] | 506 | buf		= pkt->buf    + pkt->actual; | 
|  | 507 | len		= pkt->length - pkt->actual; | 
|  | 508 | len		= min(len, maxp); | 
|  | 509 | total_len	= len; | 
|  | 510 | is_short	= total_len < maxp; | 
| Kuninori Morimoto | e8d548d | 2011-06-06 14:18:03 +0900 | [diff] [blame] | 511 |  | 
|  | 512 | /* | 
|  | 513 | * FIXME | 
|  | 514 | * | 
|  | 515 | * 32-bit access only | 
|  | 516 | */ | 
| Kuninori Morimoto | 659d495 | 2011-06-06 14:18:23 +0900 | [diff] [blame] | 517 | if (len >= 4 && !((unsigned long)buf & 0x03)) { | 
| Kuninori Morimoto | e8d548d | 2011-06-06 14:18:03 +0900 | [diff] [blame] | 518 | iowrite32_rep(addr, buf, len / 4); | 
|  | 519 | len %= 4; | 
|  | 520 | buf += total_len - len; | 
|  | 521 | } | 
|  | 522 |  | 
|  | 523 | /* the rest operation */ | 
|  | 524 | for (i = 0; i < len; i++) | 
|  | 525 | iowrite8(buf[i], addr + (0x03 - (i & 0x03))); | 
|  | 526 |  | 
| Kuninori Morimoto | 659d495 | 2011-06-06 14:18:23 +0900 | [diff] [blame] | 527 | /* | 
|  | 528 | * variable update | 
|  | 529 | */ | 
|  | 530 | pkt->actual += total_len; | 
|  | 531 |  | 
|  | 532 | if (pkt->actual < pkt->length) | 
| Kuninori Morimoto | 97664a2 | 2011-06-06 14:18:38 +0900 | [diff] [blame] | 533 | *is_done = 0;		/* there are remainder data */ | 
| Kuninori Morimoto | 659d495 | 2011-06-06 14:18:23 +0900 | [diff] [blame] | 534 | else if (is_short) | 
| Kuninori Morimoto | 97664a2 | 2011-06-06 14:18:38 +0900 | [diff] [blame] | 535 | *is_done = 1;		/* short packet */ | 
| Kuninori Morimoto | 659d495 | 2011-06-06 14:18:23 +0900 | [diff] [blame] | 536 | else | 
| Kuninori Morimoto | 97664a2 | 2011-06-06 14:18:38 +0900 | [diff] [blame] | 537 | *is_done = !pkt->zero;	/* send zero packet ? */ | 
| Kuninori Morimoto | 659d495 | 2011-06-06 14:18:23 +0900 | [diff] [blame] | 538 |  | 
|  | 539 | /* | 
|  | 540 | * pipe/irq handling | 
|  | 541 | */ | 
|  | 542 | if (is_short) | 
| Kuninori Morimoto | d3af90a | 2011-06-06 14:18:44 +0900 | [diff] [blame] | 543 | usbhsf_send_terminator(pipe, fifo); | 
| Kuninori Morimoto | e8d548d | 2011-06-06 14:18:03 +0900 | [diff] [blame] | 544 |  | 
| Kuninori Morimoto | 97664a2 | 2011-06-06 14:18:38 +0900 | [diff] [blame] | 545 | usbhsf_tx_irq_ctrl(pipe, !*is_done); | 
| Kuninori Morimoto | 4bd0481 | 2011-06-06 14:18:07 +0900 | [diff] [blame] | 546 | usbhs_pipe_enable(pipe); | 
|  | 547 |  | 
| Kuninori Morimoto | 659d495 | 2011-06-06 14:18:23 +0900 | [diff] [blame] | 548 | dev_dbg(dev, "  send %d (%d/ %d/ %d/ %d)\n", | 
|  | 549 | usbhs_pipe_number(pipe), | 
| Kuninori Morimoto | 97664a2 | 2011-06-06 14:18:38 +0900 | [diff] [blame] | 550 | pkt->length, pkt->actual, *is_done, pkt->zero); | 
| Kuninori Morimoto | 659d495 | 2011-06-06 14:18:23 +0900 | [diff] [blame] | 551 |  | 
|  | 552 | /* | 
|  | 553 | * Transmission end | 
|  | 554 | */ | 
| Kuninori Morimoto | 97664a2 | 2011-06-06 14:18:38 +0900 | [diff] [blame] | 555 | if (*is_done) { | 
| Kuninori Morimoto | 659d495 | 2011-06-06 14:18:23 +0900 | [diff] [blame] | 556 | if (usbhs_pipe_is_dcp(pipe)) | 
|  | 557 | usbhs_dcp_control_transfer_done(pipe); | 
| Kuninori Morimoto | 4bd0481 | 2011-06-06 14:18:07 +0900 | [diff] [blame] | 558 | } | 
|  | 559 |  | 
| Kuninori Morimoto | d77e3f4 | 2011-06-06 14:18:50 +0900 | [diff] [blame] | 560 | usbhsf_fifo_unselect(pipe, fifo); | 
|  | 561 |  | 
| Kuninori Morimoto | 4bd0481 | 2011-06-06 14:18:07 +0900 | [diff] [blame] | 562 | return 0; | 
| Kuninori Morimoto | 659d495 | 2011-06-06 14:18:23 +0900 | [diff] [blame] | 563 |  | 
|  | 564 | usbhs_fifo_write_busy: | 
| Kuninori Morimoto | d77e3f4 | 2011-06-06 14:18:50 +0900 | [diff] [blame] | 565 | usbhsf_fifo_unselect(pipe, fifo); | 
|  | 566 |  | 
| Kuninori Morimoto | 659d495 | 2011-06-06 14:18:23 +0900 | [diff] [blame] | 567 | /* | 
|  | 568 | * pipe is busy. | 
|  | 569 | * retry in interrupt | 
|  | 570 | */ | 
|  | 571 | usbhsf_tx_irq_ctrl(pipe, 1); | 
|  | 572 |  | 
|  | 573 | return ret; | 
| Kuninori Morimoto | e8d548d | 2011-06-06 14:18:03 +0900 | [diff] [blame] | 574 | } | 
|  | 575 |  | 
| Kuninori Morimoto | 0cb7e61 | 2011-06-06 14:18:58 +0900 | [diff] [blame] | 576 | struct usbhs_pkt_handle usbhs_fifo_pio_push_handler = { | 
|  | 577 | .prepare = usbhsf_pio_try_push, | 
|  | 578 | .try_run = usbhsf_pio_try_push, | 
| Kuninori Morimoto | dad6739 | 2011-06-06 14:18:28 +0900 | [diff] [blame] | 579 | }; | 
|  | 580 |  | 
| Kuninori Morimoto | 233f519 | 2011-07-07 00:23:24 -0700 | [diff] [blame] | 581 | /* | 
|  | 582 | *		PIO pop handler | 
|  | 583 | */ | 
| Kuninori Morimoto | 97664a2 | 2011-06-06 14:18:38 +0900 | [diff] [blame] | 584 | static int usbhsf_prepare_pop(struct usbhs_pkt *pkt, int *is_done) | 
| Kuninori Morimoto | e8d548d | 2011-06-06 14:18:03 +0900 | [diff] [blame] | 585 | { | 
| Kuninori Morimoto | dad6739 | 2011-06-06 14:18:28 +0900 | [diff] [blame] | 586 | struct usbhs_pipe *pipe = pkt->pipe; | 
| Kuninori Morimoto | d77e3f4 | 2011-06-06 14:18:50 +0900 | [diff] [blame] | 587 |  | 
|  | 588 | if (usbhs_pipe_is_busy(pipe)) | 
|  | 589 | return 0; | 
| Kuninori Morimoto | e8d548d | 2011-06-06 14:18:03 +0900 | [diff] [blame] | 590 |  | 
|  | 591 | /* | 
| Kuninori Morimoto | d77e3f4 | 2011-06-06 14:18:50 +0900 | [diff] [blame] | 592 | * pipe enable to prepare packet receive | 
| Kuninori Morimoto | e8d548d | 2011-06-06 14:18:03 +0900 | [diff] [blame] | 593 | */ | 
| Kuninori Morimoto | 3edeee3 | 2011-12-08 18:28:54 -0800 | [diff] [blame] | 594 | usbhs_pipe_data_sequence(pipe, pkt->sequence); | 
|  | 595 | pkt->sequence = -1; /* -1 sequence will be ignored */ | 
| Kuninori Morimoto | e8d548d | 2011-06-06 14:18:03 +0900 | [diff] [blame] | 596 |  | 
|  | 597 | usbhs_pipe_enable(pipe); | 
| Kuninori Morimoto | 659d495 | 2011-06-06 14:18:23 +0900 | [diff] [blame] | 598 | usbhsf_rx_irq_ctrl(pipe, 1); | 
| Kuninori Morimoto | e8d548d | 2011-06-06 14:18:03 +0900 | [diff] [blame] | 599 |  | 
| Kuninori Morimoto | d3af90a | 2011-06-06 14:18:44 +0900 | [diff] [blame] | 600 | return 0; | 
| Kuninori Morimoto | e8d548d | 2011-06-06 14:18:03 +0900 | [diff] [blame] | 601 | } | 
|  | 602 |  | 
| Kuninori Morimoto | 0cb7e61 | 2011-06-06 14:18:58 +0900 | [diff] [blame] | 603 | static int usbhsf_pio_try_pop(struct usbhs_pkt *pkt, int *is_done) | 
| Kuninori Morimoto | e8d548d | 2011-06-06 14:18:03 +0900 | [diff] [blame] | 604 | { | 
| Kuninori Morimoto | 4bd0481 | 2011-06-06 14:18:07 +0900 | [diff] [blame] | 605 | struct usbhs_pipe *pipe = pkt->pipe; | 
| Kuninori Morimoto | e8d548d | 2011-06-06 14:18:03 +0900 | [diff] [blame] | 606 | struct usbhs_priv *priv = usbhs_pipe_to_priv(pipe); | 
| Kuninori Morimoto | 659d495 | 2011-06-06 14:18:23 +0900 | [diff] [blame] | 607 | struct device *dev = usbhs_priv_to_dev(priv); | 
| Kuninori Morimoto | d3af90a | 2011-06-06 14:18:44 +0900 | [diff] [blame] | 608 | struct usbhs_fifo *fifo = usbhsf_get_cfifo(priv); /* CFIFO */ | 
|  | 609 | void __iomem *addr = priv->base + fifo->port; | 
| Kuninori Morimoto | 659d495 | 2011-06-06 14:18:23 +0900 | [diff] [blame] | 610 | u8 *buf; | 
|  | 611 | u32 data = 0; | 
|  | 612 | int maxp = usbhs_pipe_get_maxpacket(pipe); | 
| Kuninori Morimoto | 4bd0481 | 2011-06-06 14:18:07 +0900 | [diff] [blame] | 613 | int rcv_len, len; | 
| Kuninori Morimoto | e8d548d | 2011-06-06 14:18:03 +0900 | [diff] [blame] | 614 | int i, ret; | 
| Kuninori Morimoto | 4bd0481 | 2011-06-06 14:18:07 +0900 | [diff] [blame] | 615 | int total_len = 0; | 
| Kuninori Morimoto | e8d548d | 2011-06-06 14:18:03 +0900 | [diff] [blame] | 616 |  | 
| Kuninori Morimoto | d3af90a | 2011-06-06 14:18:44 +0900 | [diff] [blame] | 617 | ret = usbhsf_fifo_select(pipe, fifo, 0); | 
| Kuninori Morimoto | e8d548d | 2011-06-06 14:18:03 +0900 | [diff] [blame] | 618 | if (ret < 0) | 
| Kuninori Morimoto | d77e3f4 | 2011-06-06 14:18:50 +0900 | [diff] [blame] | 619 | return 0; | 
| Kuninori Morimoto | e8d548d | 2011-06-06 14:18:03 +0900 | [diff] [blame] | 620 |  | 
| Kuninori Morimoto | d3af90a | 2011-06-06 14:18:44 +0900 | [diff] [blame] | 621 | ret = usbhsf_fifo_barrier(priv, fifo); | 
| Kuninori Morimoto | e8d548d | 2011-06-06 14:18:03 +0900 | [diff] [blame] | 622 | if (ret < 0) | 
| Kuninori Morimoto | d77e3f4 | 2011-06-06 14:18:50 +0900 | [diff] [blame] | 623 | goto usbhs_fifo_read_busy; | 
| Kuninori Morimoto | e8d548d | 2011-06-06 14:18:03 +0900 | [diff] [blame] | 624 |  | 
| Kuninori Morimoto | d3af90a | 2011-06-06 14:18:44 +0900 | [diff] [blame] | 625 | rcv_len = usbhsf_fifo_rcv_len(priv, fifo); | 
| Kuninori Morimoto | e8d548d | 2011-06-06 14:18:03 +0900 | [diff] [blame] | 626 |  | 
| Kuninori Morimoto | 659d495 | 2011-06-06 14:18:23 +0900 | [diff] [blame] | 627 | buf		= pkt->buf    + pkt->actual; | 
|  | 628 | len		= pkt->length - pkt->actual; | 
|  | 629 | len		= min(len, rcv_len); | 
|  | 630 | total_len	= len; | 
|  | 631 |  | 
| Kuninori Morimoto | e8d548d | 2011-06-06 14:18:03 +0900 | [diff] [blame] | 632 | /* | 
| Kuninori Morimoto | 6ff5d09 | 2011-10-10 22:05:51 -0700 | [diff] [blame] | 633 | * update actual length first here to decide disable pipe. | 
|  | 634 | * if this pipe keeps BUF status and all data were popped, | 
|  | 635 | * then, next interrupt/token will be issued again | 
|  | 636 | */ | 
|  | 637 | pkt->actual += total_len; | 
|  | 638 |  | 
|  | 639 | if ((pkt->actual == pkt->length) ||	/* receive all data */ | 
|  | 640 | (total_len < maxp)) {		/* short packet */ | 
|  | 641 | *is_done = 1; | 
|  | 642 | usbhsf_rx_irq_ctrl(pipe, 0); | 
|  | 643 | usbhs_pipe_disable(pipe);	/* disable pipe first */ | 
|  | 644 | } | 
|  | 645 |  | 
|  | 646 | /* | 
| Kuninori Morimoto | e8d548d | 2011-06-06 14:18:03 +0900 | [diff] [blame] | 647 | * Buffer clear if Zero-Length packet | 
|  | 648 | * | 
|  | 649 | * see | 
|  | 650 | * "Operation" - "FIFO Buffer Memory" - "FIFO Port Function" | 
|  | 651 | */ | 
|  | 652 | if (0 == rcv_len) { | 
| Kuninori Morimoto | 3edeee3 | 2011-12-08 18:28:54 -0800 | [diff] [blame] | 653 | pkt->zero = 1; | 
| Kuninori Morimoto | d3af90a | 2011-06-06 14:18:44 +0900 | [diff] [blame] | 654 | usbhsf_fifo_clear(pipe, fifo); | 
| Kuninori Morimoto | 4bd0481 | 2011-06-06 14:18:07 +0900 | [diff] [blame] | 655 | goto usbhs_fifo_read_end; | 
| Kuninori Morimoto | e8d548d | 2011-06-06 14:18:03 +0900 | [diff] [blame] | 656 | } | 
|  | 657 |  | 
| Kuninori Morimoto | e8d548d | 2011-06-06 14:18:03 +0900 | [diff] [blame] | 658 | /* | 
|  | 659 | * FIXME | 
|  | 660 | * | 
|  | 661 | * 32-bit access only | 
|  | 662 | */ | 
| Kuninori Morimoto | 659d495 | 2011-06-06 14:18:23 +0900 | [diff] [blame] | 663 | if (len >= 4 && !((unsigned long)buf & 0x03)) { | 
| Kuninori Morimoto | e8d548d | 2011-06-06 14:18:03 +0900 | [diff] [blame] | 664 | ioread32_rep(addr, buf, len / 4); | 
|  | 665 | len %= 4; | 
| Kuninori Morimoto | 659d495 | 2011-06-06 14:18:23 +0900 | [diff] [blame] | 666 | buf += total_len - len; | 
| Kuninori Morimoto | e8d548d | 2011-06-06 14:18:03 +0900 | [diff] [blame] | 667 | } | 
|  | 668 |  | 
|  | 669 | /* the rest operation */ | 
|  | 670 | for (i = 0; i < len; i++) { | 
|  | 671 | if (!(i & 0x03)) | 
|  | 672 | data = ioread32(addr); | 
|  | 673 |  | 
|  | 674 | buf[i] = (data >> ((i & 0x03) * 8)) & 0xff; | 
|  | 675 | } | 
|  | 676 |  | 
| Kuninori Morimoto | 4bd0481 | 2011-06-06 14:18:07 +0900 | [diff] [blame] | 677 | usbhs_fifo_read_end: | 
| Kuninori Morimoto | 659d495 | 2011-06-06 14:18:23 +0900 | [diff] [blame] | 678 | dev_dbg(dev, "  recv %d (%d/ %d/ %d/ %d)\n", | 
|  | 679 | usbhs_pipe_number(pipe), | 
| Kuninori Morimoto | 97664a2 | 2011-06-06 14:18:38 +0900 | [diff] [blame] | 680 | pkt->length, pkt->actual, *is_done, pkt->zero); | 
| Kuninori Morimoto | 4bd0481 | 2011-06-06 14:18:07 +0900 | [diff] [blame] | 681 |  | 
| Kuninori Morimoto | d77e3f4 | 2011-06-06 14:18:50 +0900 | [diff] [blame] | 682 | usbhs_fifo_read_busy: | 
|  | 683 | usbhsf_fifo_unselect(pipe, fifo); | 
|  | 684 |  | 
|  | 685 | return ret; | 
| Kuninori Morimoto | e8d548d | 2011-06-06 14:18:03 +0900 | [diff] [blame] | 686 | } | 
| Kuninori Morimoto | dad6739 | 2011-06-06 14:18:28 +0900 | [diff] [blame] | 687 |  | 
| Kuninori Morimoto | 0cb7e61 | 2011-06-06 14:18:58 +0900 | [diff] [blame] | 688 | struct usbhs_pkt_handle usbhs_fifo_pio_pop_handler = { | 
| Kuninori Morimoto | dad6739 | 2011-06-06 14:18:28 +0900 | [diff] [blame] | 689 | .prepare = usbhsf_prepare_pop, | 
| Kuninori Morimoto | 0cb7e61 | 2011-06-06 14:18:58 +0900 | [diff] [blame] | 690 | .try_run = usbhsf_pio_try_pop, | 
| Kuninori Morimoto | dad6739 | 2011-06-06 14:18:28 +0900 | [diff] [blame] | 691 | }; | 
|  | 692 |  | 
|  | 693 | /* | 
| Kuninori Morimoto | 233f519 | 2011-07-07 00:23:24 -0700 | [diff] [blame] | 694 | *		DCP ctrol statge handler | 
| Kuninori Morimoto | dad6739 | 2011-06-06 14:18:28 +0900 | [diff] [blame] | 695 | */ | 
| Kuninori Morimoto | 97664a2 | 2011-06-06 14:18:38 +0900 | [diff] [blame] | 696 | static int usbhsf_ctrl_stage_end(struct usbhs_pkt *pkt, int *is_done) | 
| Kuninori Morimoto | dad6739 | 2011-06-06 14:18:28 +0900 | [diff] [blame] | 697 | { | 
| Kuninori Morimoto | 97664a2 | 2011-06-06 14:18:38 +0900 | [diff] [blame] | 698 | usbhs_dcp_control_transfer_done(pkt->pipe); | 
| Kuninori Morimoto | dad6739 | 2011-06-06 14:18:28 +0900 | [diff] [blame] | 699 |  | 
| Kuninori Morimoto | 97664a2 | 2011-06-06 14:18:38 +0900 | [diff] [blame] | 700 | *is_done = 1; | 
| Kuninori Morimoto | dad6739 | 2011-06-06 14:18:28 +0900 | [diff] [blame] | 701 |  | 
|  | 702 | return 0; | 
|  | 703 | } | 
|  | 704 |  | 
|  | 705 | struct usbhs_pkt_handle usbhs_ctrl_stage_end_handler = { | 
|  | 706 | .prepare = usbhsf_ctrl_stage_end, | 
|  | 707 | .try_run = usbhsf_ctrl_stage_end, | 
|  | 708 | }; | 
|  | 709 |  | 
|  | 710 | /* | 
| Kuninori Morimoto | e73a989 | 2011-06-06 14:19:03 +0900 | [diff] [blame] | 711 | *		DMA fifo functions | 
|  | 712 | */ | 
|  | 713 | static struct dma_chan *usbhsf_dma_chan_get(struct usbhs_fifo *fifo, | 
|  | 714 | struct usbhs_pkt *pkt) | 
|  | 715 | { | 
|  | 716 | if (&usbhs_fifo_dma_push_handler == pkt->handler) | 
|  | 717 | return fifo->tx_chan; | 
|  | 718 |  | 
|  | 719 | if (&usbhs_fifo_dma_pop_handler == pkt->handler) | 
|  | 720 | return fifo->rx_chan; | 
|  | 721 |  | 
|  | 722 | return NULL; | 
|  | 723 | } | 
|  | 724 |  | 
|  | 725 | static struct usbhs_fifo *usbhsf_get_dma_fifo(struct usbhs_priv *priv, | 
|  | 726 | struct usbhs_pkt *pkt) | 
|  | 727 | { | 
|  | 728 | struct usbhs_fifo *fifo; | 
|  | 729 |  | 
|  | 730 | /* DMA :: D0FIFO */ | 
|  | 731 | fifo = usbhsf_get_d0fifo(priv); | 
|  | 732 | if (usbhsf_dma_chan_get(fifo, pkt) && | 
|  | 733 | !usbhsf_fifo_is_busy(fifo)) | 
|  | 734 | return fifo; | 
|  | 735 |  | 
|  | 736 | /* DMA :: D1FIFO */ | 
|  | 737 | fifo = usbhsf_get_d1fifo(priv); | 
|  | 738 | if (usbhsf_dma_chan_get(fifo, pkt) && | 
|  | 739 | !usbhsf_fifo_is_busy(fifo)) | 
|  | 740 | return fifo; | 
|  | 741 |  | 
|  | 742 | return NULL; | 
|  | 743 | } | 
|  | 744 |  | 
|  | 745 | #define usbhsf_dma_start(p, f)	__usbhsf_dma_ctrl(p, f, DREQE) | 
|  | 746 | #define usbhsf_dma_stop(p, f)	__usbhsf_dma_ctrl(p, f, 0) | 
|  | 747 | static void __usbhsf_dma_ctrl(struct usbhs_pipe *pipe, | 
|  | 748 | struct usbhs_fifo *fifo, | 
|  | 749 | u16 dreqe) | 
|  | 750 | { | 
|  | 751 | struct usbhs_priv *priv = usbhs_pipe_to_priv(pipe); | 
|  | 752 |  | 
|  | 753 | usbhs_bset(priv, fifo->sel, DREQE, dreqe); | 
|  | 754 | } | 
|  | 755 |  | 
|  | 756 | #define usbhsf_dma_map(p)	__usbhsf_dma_map_ctrl(p, 1) | 
|  | 757 | #define usbhsf_dma_unmap(p)	__usbhsf_dma_map_ctrl(p, 0) | 
|  | 758 | static int __usbhsf_dma_map_ctrl(struct usbhs_pkt *pkt, int map) | 
|  | 759 | { | 
|  | 760 | struct usbhs_pipe *pipe = pkt->pipe; | 
|  | 761 | struct usbhs_priv *priv = usbhs_pipe_to_priv(pipe); | 
|  | 762 | struct usbhs_pipe_info *info = usbhs_priv_to_pipeinfo(priv); | 
|  | 763 |  | 
|  | 764 | return info->dma_map_ctrl(pkt, map); | 
|  | 765 | } | 
|  | 766 |  | 
|  | 767 | static void usbhsf_dma_complete(void *arg); | 
| Guennadi Liakhovetski | 6e4b74e | 2012-02-14 11:37:21 +0100 | [diff] [blame] | 768 | static void xfer_work(struct work_struct *work) | 
| Kuninori Morimoto | e73a989 | 2011-06-06 14:19:03 +0900 | [diff] [blame] | 769 | { | 
| Guennadi Liakhovetski | 6e4b74e | 2012-02-14 11:37:21 +0100 | [diff] [blame] | 770 | struct usbhs_pkt *pkt = container_of(work, struct usbhs_pkt, work); | 
| Kuninori Morimoto | e73a989 | 2011-06-06 14:19:03 +0900 | [diff] [blame] | 771 | struct usbhs_pipe *pipe = pkt->pipe; | 
|  | 772 | struct usbhs_fifo *fifo = usbhs_pipe_to_fifo(pipe); | 
|  | 773 | struct usbhs_priv *priv = usbhs_pipe_to_priv(pipe); | 
| Kuninori Morimoto | e73a989 | 2011-06-06 14:19:03 +0900 | [diff] [blame] | 774 | struct dma_async_tx_descriptor *desc; | 
|  | 775 | struct dma_chan *chan = usbhsf_dma_chan_get(fifo, pkt); | 
|  | 776 | struct device *dev = usbhs_priv_to_dev(priv); | 
| Vinod Koul | 55ba4e5 | 2011-10-14 21:57:58 +0530 | [diff] [blame] | 777 | enum dma_transfer_direction dir; | 
| Kuninori Morimoto | e73a989 | 2011-06-06 14:19:03 +0900 | [diff] [blame] | 778 |  | 
| Vinod Koul | 55ba4e5 | 2011-10-14 21:57:58 +0530 | [diff] [blame] | 779 | dir = usbhs_pipe_is_dir_in(pipe) ? DMA_DEV_TO_MEM : DMA_MEM_TO_DEV; | 
| Kuninori Morimoto | e73a989 | 2011-06-06 14:19:03 +0900 | [diff] [blame] | 780 |  | 
| Kuninori Morimoto | 2f0de9d | 2012-07-08 23:10:28 -0700 | [diff] [blame] | 781 | desc = dmaengine_prep_slave_single(chan, pkt->dma + pkt->actual, | 
|  | 782 | pkt->trans, dir, | 
| Alexandre Bounine | 1605282 | 2012-03-08 16:11:18 -0500 | [diff] [blame] | 783 | DMA_PREP_INTERRUPT | DMA_CTRL_ACK); | 
| Kuninori Morimoto | e73a989 | 2011-06-06 14:19:03 +0900 | [diff] [blame] | 784 | if (!desc) | 
|  | 785 | return; | 
|  | 786 |  | 
|  | 787 | desc->callback		= usbhsf_dma_complete; | 
|  | 788 | desc->callback_param	= pipe; | 
|  | 789 |  | 
| Kuninori Morimoto | 2f0de9d | 2012-07-08 23:10:28 -0700 | [diff] [blame] | 790 | if (dmaengine_submit(desc) < 0) { | 
| Kuninori Morimoto | e73a989 | 2011-06-06 14:19:03 +0900 | [diff] [blame] | 791 | dev_err(dev, "Failed to submit dma descriptor\n"); | 
|  | 792 | return; | 
|  | 793 | } | 
|  | 794 |  | 
|  | 795 | dev_dbg(dev, "  %s %d (%d/ %d)\n", | 
|  | 796 | fifo->name, usbhs_pipe_number(pipe), pkt->length, pkt->zero); | 
|  | 797 |  | 
|  | 798 | usbhsf_dma_start(pipe, fifo); | 
|  | 799 | dma_async_issue_pending(chan); | 
|  | 800 | } | 
|  | 801 |  | 
| Kuninori Morimoto | 233f519 | 2011-07-07 00:23:24 -0700 | [diff] [blame] | 802 | /* | 
|  | 803 | *		DMA push handler | 
|  | 804 | */ | 
| Kuninori Morimoto | e73a989 | 2011-06-06 14:19:03 +0900 | [diff] [blame] | 805 | static int usbhsf_dma_prepare_push(struct usbhs_pkt *pkt, int *is_done) | 
|  | 806 | { | 
|  | 807 | struct usbhs_pipe *pipe = pkt->pipe; | 
|  | 808 | struct usbhs_priv *priv = usbhs_pipe_to_priv(pipe); | 
|  | 809 | struct usbhs_fifo *fifo; | 
|  | 810 | int len = pkt->length - pkt->actual; | 
|  | 811 | int ret; | 
|  | 812 |  | 
|  | 813 | if (usbhs_pipe_is_busy(pipe)) | 
|  | 814 | return 0; | 
|  | 815 |  | 
|  | 816 | /* use PIO if packet is less than pio_dma_border or pipe is DCP */ | 
|  | 817 | if ((len < usbhs_get_dparam(priv, pio_dma_border)) || | 
|  | 818 | usbhs_pipe_is_dcp(pipe)) | 
|  | 819 | goto usbhsf_pio_prepare_push; | 
|  | 820 |  | 
| Kuninori Morimoto | 77975ee | 2012-08-22 18:01:13 -0700 | [diff] [blame] | 821 | if (len & 0x7) /* 8byte alignment */ | 
| Kuninori Morimoto | e73a989 | 2011-06-06 14:19:03 +0900 | [diff] [blame] | 822 | goto usbhsf_pio_prepare_push; | 
|  | 823 |  | 
| Kuninori Morimoto | c9ae0c9 | 2011-10-26 01:21:07 -0700 | [diff] [blame] | 824 | if ((uintptr_t)(pkt->buf + pkt->actual) & 0x7) /* 8byte alignment */ | 
| Kuninori Morimoto | 9a12d09 | 2011-07-03 17:42:19 -0700 | [diff] [blame] | 825 | goto usbhsf_pio_prepare_push; | 
|  | 826 |  | 
| Kuninori Morimoto | e73a989 | 2011-06-06 14:19:03 +0900 | [diff] [blame] | 827 | /* get enable DMA fifo */ | 
|  | 828 | fifo = usbhsf_get_dma_fifo(priv, pkt); | 
|  | 829 | if (!fifo) | 
|  | 830 | goto usbhsf_pio_prepare_push; | 
|  | 831 |  | 
|  | 832 | if (usbhsf_dma_map(pkt) < 0) | 
|  | 833 | goto usbhsf_pio_prepare_push; | 
|  | 834 |  | 
|  | 835 | ret = usbhsf_fifo_select(pipe, fifo, 0); | 
|  | 836 | if (ret < 0) | 
|  | 837 | goto usbhsf_pio_prepare_push_unmap; | 
|  | 838 |  | 
|  | 839 | pkt->trans = len; | 
|  | 840 |  | 
| Guennadi Liakhovetski | 6e4b74e | 2012-02-14 11:37:21 +0100 | [diff] [blame] | 841 | INIT_WORK(&pkt->work, xfer_work); | 
|  | 842 | schedule_work(&pkt->work); | 
| Kuninori Morimoto | e73a989 | 2011-06-06 14:19:03 +0900 | [diff] [blame] | 843 |  | 
|  | 844 | return 0; | 
|  | 845 |  | 
|  | 846 | usbhsf_pio_prepare_push_unmap: | 
|  | 847 | usbhsf_dma_unmap(pkt); | 
|  | 848 | usbhsf_pio_prepare_push: | 
|  | 849 | /* | 
|  | 850 | * change handler to PIO | 
|  | 851 | */ | 
|  | 852 | pkt->handler = &usbhs_fifo_pio_push_handler; | 
|  | 853 |  | 
|  | 854 | return pkt->handler->prepare(pkt, is_done); | 
|  | 855 | } | 
|  | 856 |  | 
|  | 857 | static int usbhsf_dma_push_done(struct usbhs_pkt *pkt, int *is_done) | 
|  | 858 | { | 
|  | 859 | struct usbhs_pipe *pipe = pkt->pipe; | 
|  | 860 |  | 
|  | 861 | pkt->actual = pkt->trans; | 
|  | 862 |  | 
|  | 863 | *is_done = !pkt->zero;	/* send zero packet ? */ | 
|  | 864 |  | 
|  | 865 | usbhsf_dma_stop(pipe, pipe->fifo); | 
|  | 866 | usbhsf_dma_unmap(pkt); | 
|  | 867 | usbhsf_fifo_unselect(pipe, pipe->fifo); | 
|  | 868 |  | 
|  | 869 | return 0; | 
|  | 870 | } | 
|  | 871 |  | 
|  | 872 | struct usbhs_pkt_handle usbhs_fifo_dma_push_handler = { | 
|  | 873 | .prepare	= usbhsf_dma_prepare_push, | 
|  | 874 | .dma_done	= usbhsf_dma_push_done, | 
|  | 875 | }; | 
|  | 876 |  | 
| Kuninori Morimoto | 233f519 | 2011-07-07 00:23:24 -0700 | [diff] [blame] | 877 | /* | 
|  | 878 | *		DMA pop handler | 
|  | 879 | */ | 
| Kuninori Morimoto | e73a989 | 2011-06-06 14:19:03 +0900 | [diff] [blame] | 880 | static int usbhsf_dma_try_pop(struct usbhs_pkt *pkt, int *is_done) | 
|  | 881 | { | 
|  | 882 | struct usbhs_pipe *pipe = pkt->pipe; | 
|  | 883 | struct usbhs_priv *priv = usbhs_pipe_to_priv(pipe); | 
|  | 884 | struct usbhs_fifo *fifo; | 
|  | 885 | int len, ret; | 
|  | 886 |  | 
|  | 887 | if (usbhs_pipe_is_busy(pipe)) | 
|  | 888 | return 0; | 
|  | 889 |  | 
|  | 890 | if (usbhs_pipe_is_dcp(pipe)) | 
|  | 891 | goto usbhsf_pio_prepare_pop; | 
|  | 892 |  | 
|  | 893 | /* get enable DMA fifo */ | 
|  | 894 | fifo = usbhsf_get_dma_fifo(priv, pkt); | 
|  | 895 | if (!fifo) | 
|  | 896 | goto usbhsf_pio_prepare_pop; | 
|  | 897 |  | 
| Kuninori Morimoto | c9ae0c9 | 2011-10-26 01:21:07 -0700 | [diff] [blame] | 898 | if ((uintptr_t)(pkt->buf + pkt->actual) & 0x7) /* 8byte alignment */ | 
| Kuninori Morimoto | 9a12d09 | 2011-07-03 17:42:19 -0700 | [diff] [blame] | 899 | goto usbhsf_pio_prepare_pop; | 
|  | 900 |  | 
| Kuninori Morimoto | e73a989 | 2011-06-06 14:19:03 +0900 | [diff] [blame] | 901 | ret = usbhsf_fifo_select(pipe, fifo, 0); | 
|  | 902 | if (ret < 0) | 
|  | 903 | goto usbhsf_pio_prepare_pop; | 
|  | 904 |  | 
|  | 905 | /* use PIO if packet is less than pio_dma_border */ | 
|  | 906 | len = usbhsf_fifo_rcv_len(priv, fifo); | 
|  | 907 | len = min(pkt->length - pkt->actual, len); | 
| Kuninori Morimoto | 77975ee | 2012-08-22 18:01:13 -0700 | [diff] [blame] | 908 | if (len & 0x7) /* 8byte alignment */ | 
| Kuninori Morimoto | e73a989 | 2011-06-06 14:19:03 +0900 | [diff] [blame] | 909 | goto usbhsf_pio_prepare_pop_unselect; | 
|  | 910 |  | 
|  | 911 | if (len < usbhs_get_dparam(priv, pio_dma_border)) | 
|  | 912 | goto usbhsf_pio_prepare_pop_unselect; | 
|  | 913 |  | 
|  | 914 | ret = usbhsf_fifo_barrier(priv, fifo); | 
|  | 915 | if (ret < 0) | 
|  | 916 | goto usbhsf_pio_prepare_pop_unselect; | 
|  | 917 |  | 
|  | 918 | if (usbhsf_dma_map(pkt) < 0) | 
|  | 919 | goto usbhsf_pio_prepare_pop_unselect; | 
|  | 920 |  | 
|  | 921 | /* DMA */ | 
|  | 922 |  | 
|  | 923 | /* | 
|  | 924 | * usbhs_fifo_dma_pop_handler :: prepare | 
|  | 925 | * enabled irq to come here. | 
|  | 926 | * but it is no longer needed for DMA. disable it. | 
|  | 927 | */ | 
|  | 928 | usbhsf_rx_irq_ctrl(pipe, 0); | 
|  | 929 |  | 
|  | 930 | pkt->trans = len; | 
|  | 931 |  | 
| Guennadi Liakhovetski | 6e4b74e | 2012-02-14 11:37:21 +0100 | [diff] [blame] | 932 | INIT_WORK(&pkt->work, xfer_work); | 
|  | 933 | schedule_work(&pkt->work); | 
| Kuninori Morimoto | e73a989 | 2011-06-06 14:19:03 +0900 | [diff] [blame] | 934 |  | 
|  | 935 | return 0; | 
|  | 936 |  | 
|  | 937 | usbhsf_pio_prepare_pop_unselect: | 
|  | 938 | usbhsf_fifo_unselect(pipe, fifo); | 
|  | 939 | usbhsf_pio_prepare_pop: | 
|  | 940 |  | 
|  | 941 | /* | 
|  | 942 | * change handler to PIO | 
|  | 943 | */ | 
|  | 944 | pkt->handler = &usbhs_fifo_pio_pop_handler; | 
|  | 945 |  | 
|  | 946 | return pkt->handler->try_run(pkt, is_done); | 
|  | 947 | } | 
|  | 948 |  | 
|  | 949 | static int usbhsf_dma_pop_done(struct usbhs_pkt *pkt, int *is_done) | 
|  | 950 | { | 
|  | 951 | struct usbhs_pipe *pipe = pkt->pipe; | 
|  | 952 | int maxp = usbhs_pipe_get_maxpacket(pipe); | 
|  | 953 |  | 
|  | 954 | usbhsf_dma_stop(pipe, pipe->fifo); | 
|  | 955 | usbhsf_dma_unmap(pkt); | 
|  | 956 | usbhsf_fifo_unselect(pipe, pipe->fifo); | 
|  | 957 |  | 
|  | 958 | pkt->actual += pkt->trans; | 
|  | 959 |  | 
|  | 960 | if ((pkt->actual == pkt->length) ||	/* receive all data */ | 
|  | 961 | (pkt->trans < maxp)) {		/* short packet */ | 
|  | 962 | *is_done = 1; | 
|  | 963 | } else { | 
|  | 964 | /* re-enable */ | 
|  | 965 | usbhsf_prepare_pop(pkt, is_done); | 
|  | 966 | } | 
|  | 967 |  | 
|  | 968 | return 0; | 
|  | 969 | } | 
|  | 970 |  | 
|  | 971 | struct usbhs_pkt_handle usbhs_fifo_dma_pop_handler = { | 
|  | 972 | .prepare	= usbhsf_prepare_pop, | 
|  | 973 | .try_run	= usbhsf_dma_try_pop, | 
|  | 974 | .dma_done	= usbhsf_dma_pop_done | 
|  | 975 | }; | 
|  | 976 |  | 
|  | 977 | /* | 
|  | 978 | *		DMA setting | 
|  | 979 | */ | 
|  | 980 | static bool usbhsf_dma_filter(struct dma_chan *chan, void *param) | 
|  | 981 | { | 
|  | 982 | struct sh_dmae_slave *slave = param; | 
|  | 983 |  | 
|  | 984 | /* | 
|  | 985 | * FIXME | 
|  | 986 | * | 
|  | 987 | * usbhs doesn't recognize id = 0 as valid DMA | 
|  | 988 | */ | 
| Guennadi Liakhovetski | f19b7e0 | 2012-05-09 17:09:19 +0200 | [diff] [blame] | 989 | if (0 == slave->shdma_slave.slave_id) | 
| Kuninori Morimoto | e73a989 | 2011-06-06 14:19:03 +0900 | [diff] [blame] | 990 | return false; | 
|  | 991 |  | 
|  | 992 | chan->private = slave; | 
|  | 993 |  | 
|  | 994 | return true; | 
|  | 995 | } | 
|  | 996 |  | 
|  | 997 | static void usbhsf_dma_quit(struct usbhs_priv *priv, struct usbhs_fifo *fifo) | 
|  | 998 | { | 
|  | 999 | if (fifo->tx_chan) | 
|  | 1000 | dma_release_channel(fifo->tx_chan); | 
|  | 1001 | if (fifo->rx_chan) | 
|  | 1002 | dma_release_channel(fifo->rx_chan); | 
|  | 1003 |  | 
|  | 1004 | fifo->tx_chan = NULL; | 
|  | 1005 | fifo->rx_chan = NULL; | 
|  | 1006 | } | 
|  | 1007 |  | 
|  | 1008 | static void usbhsf_dma_init(struct usbhs_priv *priv, | 
|  | 1009 | struct usbhs_fifo *fifo) | 
|  | 1010 | { | 
|  | 1011 | struct device *dev = usbhs_priv_to_dev(priv); | 
|  | 1012 | dma_cap_mask_t mask; | 
|  | 1013 |  | 
|  | 1014 | dma_cap_zero(mask); | 
|  | 1015 | dma_cap_set(DMA_SLAVE, mask); | 
|  | 1016 | fifo->tx_chan = dma_request_channel(mask, usbhsf_dma_filter, | 
|  | 1017 | &fifo->tx_slave); | 
|  | 1018 |  | 
|  | 1019 | dma_cap_zero(mask); | 
|  | 1020 | dma_cap_set(DMA_SLAVE, mask); | 
|  | 1021 | fifo->rx_chan = dma_request_channel(mask, usbhsf_dma_filter, | 
|  | 1022 | &fifo->rx_slave); | 
|  | 1023 |  | 
|  | 1024 | if (fifo->tx_chan || fifo->rx_chan) | 
| Kuninori Morimoto | 4ce6880 | 2011-06-21 09:33:43 +0900 | [diff] [blame] | 1025 | dev_dbg(dev, "enable DMAEngine (%s%s%s)\n", | 
| Kuninori Morimoto | e73a989 | 2011-06-06 14:19:03 +0900 | [diff] [blame] | 1026 | fifo->name, | 
|  | 1027 | fifo->tx_chan ? "[TX]" : "    ", | 
|  | 1028 | fifo->rx_chan ? "[RX]" : "    "); | 
|  | 1029 | } | 
|  | 1030 |  | 
|  | 1031 | /* | 
| Kuninori Morimoto | dad6739 | 2011-06-06 14:18:28 +0900 | [diff] [blame] | 1032 | *		irq functions | 
|  | 1033 | */ | 
|  | 1034 | static int usbhsf_irq_empty(struct usbhs_priv *priv, | 
|  | 1035 | struct usbhs_irq_state *irq_state) | 
|  | 1036 | { | 
|  | 1037 | struct usbhs_pipe *pipe; | 
| Kuninori Morimoto | dad6739 | 2011-06-06 14:18:28 +0900 | [diff] [blame] | 1038 | struct device *dev = usbhs_priv_to_dev(priv); | 
|  | 1039 | int i, ret; | 
|  | 1040 |  | 
|  | 1041 | if (!irq_state->bempsts) { | 
|  | 1042 | dev_err(dev, "debug %s !!\n", __func__); | 
|  | 1043 | return -EIO; | 
|  | 1044 | } | 
|  | 1045 |  | 
|  | 1046 | dev_dbg(dev, "irq empty [0x%04x]\n", irq_state->bempsts); | 
|  | 1047 |  | 
|  | 1048 | /* | 
|  | 1049 | * search interrupted "pipe" | 
|  | 1050 | * not "uep". | 
|  | 1051 | */ | 
|  | 1052 | usbhs_for_each_pipe_with_dcp(pipe, priv, i) { | 
|  | 1053 | if (!(irq_state->bempsts & (1 << i))) | 
|  | 1054 | continue; | 
|  | 1055 |  | 
| Kuninori Morimoto | 51b8a02 | 2011-10-10 21:58:45 -0700 | [diff] [blame] | 1056 | ret = usbhsf_pkt_handler(pipe, USBHSF_PKT_TRY_RUN); | 
| Kuninori Morimoto | dad6739 | 2011-06-06 14:18:28 +0900 | [diff] [blame] | 1057 | if (ret < 0) | 
|  | 1058 | dev_err(dev, "irq_empty run_error %d : %d\n", i, ret); | 
|  | 1059 | } | 
|  | 1060 |  | 
|  | 1061 | return 0; | 
|  | 1062 | } | 
|  | 1063 |  | 
|  | 1064 | static int usbhsf_irq_ready(struct usbhs_priv *priv, | 
|  | 1065 | struct usbhs_irq_state *irq_state) | 
|  | 1066 | { | 
|  | 1067 | struct usbhs_pipe *pipe; | 
| Kuninori Morimoto | dad6739 | 2011-06-06 14:18:28 +0900 | [diff] [blame] | 1068 | struct device *dev = usbhs_priv_to_dev(priv); | 
|  | 1069 | int i, ret; | 
|  | 1070 |  | 
|  | 1071 | if (!irq_state->brdysts) { | 
|  | 1072 | dev_err(dev, "debug %s !!\n", __func__); | 
|  | 1073 | return -EIO; | 
|  | 1074 | } | 
|  | 1075 |  | 
|  | 1076 | dev_dbg(dev, "irq ready [0x%04x]\n", irq_state->brdysts); | 
|  | 1077 |  | 
|  | 1078 | /* | 
|  | 1079 | * search interrupted "pipe" | 
|  | 1080 | * not "uep". | 
|  | 1081 | */ | 
|  | 1082 | usbhs_for_each_pipe_with_dcp(pipe, priv, i) { | 
|  | 1083 | if (!(irq_state->brdysts & (1 << i))) | 
|  | 1084 | continue; | 
|  | 1085 |  | 
| Kuninori Morimoto | 51b8a02 | 2011-10-10 21:58:45 -0700 | [diff] [blame] | 1086 | ret = usbhsf_pkt_handler(pipe, USBHSF_PKT_TRY_RUN); | 
| Kuninori Morimoto | dad6739 | 2011-06-06 14:18:28 +0900 | [diff] [blame] | 1087 | if (ret < 0) | 
|  | 1088 | dev_err(dev, "irq_ready run_error %d : %d\n", i, ret); | 
|  | 1089 | } | 
|  | 1090 |  | 
|  | 1091 | return 0; | 
|  | 1092 | } | 
|  | 1093 |  | 
| Kuninori Morimoto | e73a989 | 2011-06-06 14:19:03 +0900 | [diff] [blame] | 1094 | static void usbhsf_dma_complete(void *arg) | 
|  | 1095 | { | 
|  | 1096 | struct usbhs_pipe *pipe = arg; | 
|  | 1097 | struct usbhs_priv *priv = usbhs_pipe_to_priv(pipe); | 
|  | 1098 | struct device *dev = usbhs_priv_to_dev(priv); | 
|  | 1099 | int ret; | 
|  | 1100 |  | 
| Kuninori Morimoto | 51b8a02 | 2011-10-10 21:58:45 -0700 | [diff] [blame] | 1101 | ret = usbhsf_pkt_handler(pipe, USBHSF_PKT_DMA_DONE); | 
| Kuninori Morimoto | e73a989 | 2011-06-06 14:19:03 +0900 | [diff] [blame] | 1102 | if (ret < 0) | 
|  | 1103 | dev_err(dev, "dma_complete run_error %d : %d\n", | 
|  | 1104 | usbhs_pipe_number(pipe), ret); | 
|  | 1105 | } | 
|  | 1106 |  | 
| Kuninori Morimoto | dad6739 | 2011-06-06 14:18:28 +0900 | [diff] [blame] | 1107 | /* | 
|  | 1108 | *		fifo init | 
|  | 1109 | */ | 
|  | 1110 | void usbhs_fifo_init(struct usbhs_priv *priv) | 
|  | 1111 | { | 
|  | 1112 | struct usbhs_mod *mod = usbhs_mod_get_current(priv); | 
| Kuninori Morimoto | d77e3f4 | 2011-06-06 14:18:50 +0900 | [diff] [blame] | 1113 | struct usbhs_fifo *cfifo = usbhsf_get_cfifo(priv); | 
| Kuninori Morimoto | e73a989 | 2011-06-06 14:19:03 +0900 | [diff] [blame] | 1114 | struct usbhs_fifo *d0fifo = usbhsf_get_d0fifo(priv); | 
|  | 1115 | struct usbhs_fifo *d1fifo = usbhsf_get_d1fifo(priv); | 
| Kuninori Morimoto | dad6739 | 2011-06-06 14:18:28 +0900 | [diff] [blame] | 1116 |  | 
|  | 1117 | mod->irq_empty		= usbhsf_irq_empty; | 
|  | 1118 | mod->irq_ready		= usbhsf_irq_ready; | 
|  | 1119 | mod->irq_bempsts	= 0; | 
|  | 1120 | mod->irq_brdysts	= 0; | 
| Kuninori Morimoto | d77e3f4 | 2011-06-06 14:18:50 +0900 | [diff] [blame] | 1121 |  | 
|  | 1122 | cfifo->pipe	= NULL; | 
| Kuninori Morimoto | e73a989 | 2011-06-06 14:19:03 +0900 | [diff] [blame] | 1123 | cfifo->tx_chan	= NULL; | 
|  | 1124 | cfifo->rx_chan	= NULL; | 
|  | 1125 |  | 
|  | 1126 | d0fifo->pipe	= NULL; | 
|  | 1127 | d0fifo->tx_chan	= NULL; | 
|  | 1128 | d0fifo->rx_chan	= NULL; | 
|  | 1129 |  | 
|  | 1130 | d1fifo->pipe	= NULL; | 
|  | 1131 | d1fifo->tx_chan	= NULL; | 
|  | 1132 | d1fifo->rx_chan	= NULL; | 
|  | 1133 |  | 
|  | 1134 | usbhsf_dma_init(priv, usbhsf_get_d0fifo(priv)); | 
|  | 1135 | usbhsf_dma_init(priv, usbhsf_get_d1fifo(priv)); | 
| Kuninori Morimoto | dad6739 | 2011-06-06 14:18:28 +0900 | [diff] [blame] | 1136 | } | 
|  | 1137 |  | 
|  | 1138 | void usbhs_fifo_quit(struct usbhs_priv *priv) | 
|  | 1139 | { | 
|  | 1140 | struct usbhs_mod *mod = usbhs_mod_get_current(priv); | 
|  | 1141 |  | 
|  | 1142 | mod->irq_empty		= NULL; | 
|  | 1143 | mod->irq_ready		= NULL; | 
|  | 1144 | mod->irq_bempsts	= 0; | 
|  | 1145 | mod->irq_brdysts	= 0; | 
| Kuninori Morimoto | e73a989 | 2011-06-06 14:19:03 +0900 | [diff] [blame] | 1146 |  | 
|  | 1147 | usbhsf_dma_quit(priv, usbhsf_get_d0fifo(priv)); | 
|  | 1148 | usbhsf_dma_quit(priv, usbhsf_get_d1fifo(priv)); | 
| Kuninori Morimoto | dad6739 | 2011-06-06 14:18:28 +0900 | [diff] [blame] | 1149 | } | 
| Kuninori Morimoto | d3af90a | 2011-06-06 14:18:44 +0900 | [diff] [blame] | 1150 |  | 
|  | 1151 | int usbhs_fifo_probe(struct usbhs_priv *priv) | 
|  | 1152 | { | 
|  | 1153 | struct usbhs_fifo *fifo; | 
|  | 1154 |  | 
|  | 1155 | /* CFIFO */ | 
|  | 1156 | fifo = usbhsf_get_cfifo(priv); | 
| Kuninori Morimoto | e73a989 | 2011-06-06 14:19:03 +0900 | [diff] [blame] | 1157 | fifo->name	= "CFIFO"; | 
| Kuninori Morimoto | d3af90a | 2011-06-06 14:18:44 +0900 | [diff] [blame] | 1158 | fifo->port	= CFIFO; | 
|  | 1159 | fifo->sel	= CFIFOSEL; | 
|  | 1160 | fifo->ctr	= CFIFOCTR; | 
|  | 1161 |  | 
| Kuninori Morimoto | e73a989 | 2011-06-06 14:19:03 +0900 | [diff] [blame] | 1162 | /* D0FIFO */ | 
|  | 1163 | fifo = usbhsf_get_d0fifo(priv); | 
|  | 1164 | fifo->name	= "D0FIFO"; | 
|  | 1165 | fifo->port	= D0FIFO; | 
|  | 1166 | fifo->sel	= D0FIFOSEL; | 
|  | 1167 | fifo->ctr	= D0FIFOCTR; | 
| Guennadi Liakhovetski | f19b7e0 | 2012-05-09 17:09:19 +0200 | [diff] [blame] | 1168 | fifo->tx_slave.shdma_slave.slave_id	= usbhs_get_dparam(priv, d0_tx_id); | 
|  | 1169 | fifo->rx_slave.shdma_slave.slave_id	= usbhs_get_dparam(priv, d0_rx_id); | 
| Kuninori Morimoto | e73a989 | 2011-06-06 14:19:03 +0900 | [diff] [blame] | 1170 |  | 
|  | 1171 | /* D1FIFO */ | 
|  | 1172 | fifo = usbhsf_get_d1fifo(priv); | 
|  | 1173 | fifo->name	= "D1FIFO"; | 
|  | 1174 | fifo->port	= D1FIFO; | 
|  | 1175 | fifo->sel	= D1FIFOSEL; | 
|  | 1176 | fifo->ctr	= D1FIFOCTR; | 
| Guennadi Liakhovetski | f19b7e0 | 2012-05-09 17:09:19 +0200 | [diff] [blame] | 1177 | fifo->tx_slave.shdma_slave.slave_id	= usbhs_get_dparam(priv, d1_tx_id); | 
|  | 1178 | fifo->rx_slave.shdma_slave.slave_id	= usbhs_get_dparam(priv, d1_rx_id); | 
| Kuninori Morimoto | e73a989 | 2011-06-06 14:19:03 +0900 | [diff] [blame] | 1179 |  | 
| Kuninori Morimoto | d3af90a | 2011-06-06 14:18:44 +0900 | [diff] [blame] | 1180 | return 0; | 
|  | 1181 | } | 
|  | 1182 |  | 
|  | 1183 | void usbhs_fifo_remove(struct usbhs_priv *priv) | 
|  | 1184 | { | 
|  | 1185 | } |