1b285192aSMauro Carvalho Chehab /* 2b285192aSMauro Carvalho Chehab * cx18 buffer queues 3b285192aSMauro Carvalho Chehab * 4b285192aSMauro Carvalho Chehab * Derived from ivtv-queue.c 5b285192aSMauro Carvalho Chehab * 6b285192aSMauro Carvalho Chehab * Copyright (C) 2007 Hans Verkuil <hverkuil@xs4all.nl> 7b285192aSMauro Carvalho Chehab * Copyright (C) 2008 Andy Walls <awalls@md.metrocast.net> 8b285192aSMauro Carvalho Chehab * 9b285192aSMauro Carvalho Chehab * This program is free software; you can redistribute it and/or modify 10b285192aSMauro Carvalho Chehab * it under the terms of the GNU General Public License as published by 11b285192aSMauro Carvalho Chehab * the Free Software Foundation; either version 2 of the License, or 12b285192aSMauro Carvalho Chehab * (at your option) any later version. 13b285192aSMauro Carvalho Chehab * 14b285192aSMauro Carvalho Chehab * This program is distributed in the hope that it will be useful, 15b285192aSMauro Carvalho Chehab * but WITHOUT ANY WARRANTY; without even the implied warranty of 16b285192aSMauro Carvalho Chehab * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the 17b285192aSMauro Carvalho Chehab * GNU General Public License for more details. 18b285192aSMauro Carvalho Chehab * 19b285192aSMauro Carvalho Chehab * You should have received a copy of the GNU General Public License 20b285192aSMauro Carvalho Chehab * along with this program; if not, write to the Free Software 21b285192aSMauro Carvalho Chehab * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 22b285192aSMauro Carvalho Chehab * 02111-1307 USA 23b285192aSMauro Carvalho Chehab */ 24b285192aSMauro Carvalho Chehab 25b285192aSMauro Carvalho Chehab #include "cx18-driver.h" 26b285192aSMauro Carvalho Chehab #include "cx18-queue.h" 27b285192aSMauro Carvalho Chehab #include "cx18-streams.h" 28b285192aSMauro Carvalho Chehab #include "cx18-scb.h" 29b285192aSMauro Carvalho Chehab #include "cx18-io.h" 30b285192aSMauro Carvalho Chehab 31b285192aSMauro Carvalho Chehab void cx18_buf_swap(struct cx18_buffer *buf) 32b285192aSMauro Carvalho Chehab { 33b285192aSMauro Carvalho Chehab int i; 34b285192aSMauro Carvalho Chehab 35b285192aSMauro Carvalho Chehab for (i = 0; i < buf->bytesused; i += 4) 36b285192aSMauro Carvalho Chehab swab32s((u32 *)(buf->buf + i)); 37b285192aSMauro Carvalho Chehab } 38b285192aSMauro Carvalho Chehab 39b285192aSMauro Carvalho Chehab void _cx18_mdl_swap(struct cx18_mdl *mdl) 40b285192aSMauro Carvalho Chehab { 41b285192aSMauro Carvalho Chehab struct cx18_buffer *buf; 42b285192aSMauro Carvalho Chehab 43b285192aSMauro Carvalho Chehab list_for_each_entry(buf, &mdl->buf_list, list) { 44b285192aSMauro Carvalho Chehab if (buf->bytesused == 0) 45b285192aSMauro Carvalho Chehab break; 46b285192aSMauro Carvalho Chehab cx18_buf_swap(buf); 47b285192aSMauro Carvalho Chehab } 48b285192aSMauro Carvalho Chehab } 49b285192aSMauro Carvalho Chehab 50b285192aSMauro Carvalho Chehab void cx18_queue_init(struct cx18_queue *q) 51b285192aSMauro Carvalho Chehab { 52b285192aSMauro Carvalho Chehab INIT_LIST_HEAD(&q->list); 53b285192aSMauro Carvalho Chehab atomic_set(&q->depth, 0); 54b285192aSMauro Carvalho Chehab q->bytesused = 0; 55b285192aSMauro Carvalho Chehab } 56b285192aSMauro Carvalho Chehab 57b285192aSMauro Carvalho Chehab struct cx18_queue *_cx18_enqueue(struct cx18_stream *s, struct cx18_mdl *mdl, 58b285192aSMauro Carvalho Chehab struct cx18_queue *q, int to_front) 59b285192aSMauro Carvalho Chehab { 60b285192aSMauro Carvalho Chehab /* clear the mdl if it is not to be enqueued to the full queue */ 61b285192aSMauro Carvalho Chehab if (q != &s->q_full) { 62b285192aSMauro Carvalho Chehab mdl->bytesused = 0; 63b285192aSMauro Carvalho Chehab mdl->readpos = 0; 64b285192aSMauro Carvalho Chehab mdl->m_flags = 0; 65b285192aSMauro Carvalho Chehab mdl->skipped = 0; 66b285192aSMauro Carvalho Chehab mdl->curr_buf = NULL; 67b285192aSMauro Carvalho Chehab } 68b285192aSMauro Carvalho Chehab 69b285192aSMauro Carvalho Chehab /* q_busy is restricted to a max buffer count imposed by firmware */ 70b285192aSMauro Carvalho Chehab if (q == &s->q_busy && 71b285192aSMauro Carvalho Chehab atomic_read(&q->depth) >= CX18_MAX_FW_MDLS_PER_STREAM) 72b285192aSMauro Carvalho Chehab q = &s->q_free; 73b285192aSMauro Carvalho Chehab 74b285192aSMauro Carvalho Chehab spin_lock(&q->lock); 75b285192aSMauro Carvalho Chehab 76b285192aSMauro Carvalho Chehab if (to_front) 77b285192aSMauro Carvalho Chehab list_add(&mdl->list, &q->list); /* LIFO */ 78b285192aSMauro Carvalho Chehab else 79b285192aSMauro Carvalho Chehab list_add_tail(&mdl->list, &q->list); /* FIFO */ 80b285192aSMauro Carvalho Chehab q->bytesused += mdl->bytesused - mdl->readpos; 81b285192aSMauro Carvalho Chehab atomic_inc(&q->depth); 82b285192aSMauro Carvalho Chehab 83b285192aSMauro Carvalho Chehab spin_unlock(&q->lock); 84b285192aSMauro Carvalho Chehab return q; 85b285192aSMauro Carvalho Chehab } 86b285192aSMauro Carvalho Chehab 87b285192aSMauro Carvalho Chehab struct cx18_mdl *cx18_dequeue(struct cx18_stream *s, struct cx18_queue *q) 88b285192aSMauro Carvalho Chehab { 89b285192aSMauro Carvalho Chehab struct cx18_mdl *mdl = NULL; 90b285192aSMauro Carvalho Chehab 91b285192aSMauro Carvalho Chehab spin_lock(&q->lock); 92b285192aSMauro Carvalho Chehab if (!list_empty(&q->list)) { 93b285192aSMauro Carvalho Chehab mdl = list_first_entry(&q->list, struct cx18_mdl, list); 94b285192aSMauro Carvalho Chehab list_del_init(&mdl->list); 95b285192aSMauro Carvalho Chehab q->bytesused -= mdl->bytesused - mdl->readpos; 96b285192aSMauro Carvalho Chehab mdl->skipped = 0; 97b285192aSMauro Carvalho Chehab atomic_dec(&q->depth); 98b285192aSMauro Carvalho Chehab } 99b285192aSMauro Carvalho Chehab spin_unlock(&q->lock); 100b285192aSMauro Carvalho Chehab return mdl; 101b285192aSMauro Carvalho Chehab } 102b285192aSMauro Carvalho Chehab 103b285192aSMauro Carvalho Chehab static void _cx18_mdl_update_bufs_for_cpu(struct cx18_stream *s, 104b285192aSMauro Carvalho Chehab struct cx18_mdl *mdl) 105b285192aSMauro Carvalho Chehab { 106b285192aSMauro Carvalho Chehab struct cx18_buffer *buf; 107b285192aSMauro Carvalho Chehab u32 buf_size = s->buf_size; 108b285192aSMauro Carvalho Chehab u32 bytesused = mdl->bytesused; 109b285192aSMauro Carvalho Chehab 110b285192aSMauro Carvalho Chehab list_for_each_entry(buf, &mdl->buf_list, list) { 111b285192aSMauro Carvalho Chehab buf->readpos = 0; 112b285192aSMauro Carvalho Chehab if (bytesused >= buf_size) { 113b285192aSMauro Carvalho Chehab buf->bytesused = buf_size; 114b285192aSMauro Carvalho Chehab bytesused -= buf_size; 115b285192aSMauro Carvalho Chehab } else { 116b285192aSMauro Carvalho Chehab buf->bytesused = bytesused; 117b285192aSMauro Carvalho Chehab bytesused = 0; 118b285192aSMauro Carvalho Chehab } 119b285192aSMauro Carvalho Chehab cx18_buf_sync_for_cpu(s, buf); 120b285192aSMauro Carvalho Chehab } 121b285192aSMauro Carvalho Chehab } 122b285192aSMauro Carvalho Chehab 123b285192aSMauro Carvalho Chehab static inline void cx18_mdl_update_bufs_for_cpu(struct cx18_stream *s, 124b285192aSMauro Carvalho Chehab struct cx18_mdl *mdl) 125b285192aSMauro Carvalho Chehab { 126b285192aSMauro Carvalho Chehab struct cx18_buffer *buf; 127b285192aSMauro Carvalho Chehab 128b285192aSMauro Carvalho Chehab if (list_is_singular(&mdl->buf_list)) { 129b285192aSMauro Carvalho Chehab buf = list_first_entry(&mdl->buf_list, struct cx18_buffer, 130b285192aSMauro Carvalho Chehab list); 131b285192aSMauro Carvalho Chehab buf->bytesused = mdl->bytesused; 132b285192aSMauro Carvalho Chehab buf->readpos = 0; 133b285192aSMauro Carvalho Chehab cx18_buf_sync_for_cpu(s, buf); 134b285192aSMauro Carvalho Chehab } else { 135b285192aSMauro Carvalho Chehab _cx18_mdl_update_bufs_for_cpu(s, mdl); 136b285192aSMauro Carvalho Chehab } 137b285192aSMauro Carvalho Chehab } 138b285192aSMauro Carvalho Chehab 139b285192aSMauro Carvalho Chehab struct cx18_mdl *cx18_queue_get_mdl(struct cx18_stream *s, u32 id, 140b285192aSMauro Carvalho Chehab u32 bytesused) 141b285192aSMauro Carvalho Chehab { 142b285192aSMauro Carvalho Chehab struct cx18 *cx = s->cx; 143b285192aSMauro Carvalho Chehab struct cx18_mdl *mdl; 144b285192aSMauro Carvalho Chehab struct cx18_mdl *tmp; 145b285192aSMauro Carvalho Chehab struct cx18_mdl *ret = NULL; 146b285192aSMauro Carvalho Chehab LIST_HEAD(sweep_up); 147b285192aSMauro Carvalho Chehab 148b285192aSMauro Carvalho Chehab /* 149b285192aSMauro Carvalho Chehab * We don't have to acquire multiple q locks here, because we are 150b285192aSMauro Carvalho Chehab * serialized by the single threaded work handler. 151b285192aSMauro Carvalho Chehab * MDLs from the firmware will thus remain in order as 152b285192aSMauro Carvalho Chehab * they are moved from q_busy to q_full or to the dvb ring buffer. 153b285192aSMauro Carvalho Chehab */ 154b285192aSMauro Carvalho Chehab spin_lock(&s->q_busy.lock); 155b285192aSMauro Carvalho Chehab list_for_each_entry_safe(mdl, tmp, &s->q_busy.list, list) { 156b285192aSMauro Carvalho Chehab /* 157b285192aSMauro Carvalho Chehab * We should find what the firmware told us is done, 158b285192aSMauro Carvalho Chehab * right at the front of the queue. If we don't, we likely have 159b285192aSMauro Carvalho Chehab * missed an mdl done message from the firmware. 160b285192aSMauro Carvalho Chehab * Once we skip an mdl repeatedly, relative to the size of 161b285192aSMauro Carvalho Chehab * q_busy, we have high confidence we've missed it. 162b285192aSMauro Carvalho Chehab */ 163b285192aSMauro Carvalho Chehab if (mdl->id != id) { 164b285192aSMauro Carvalho Chehab mdl->skipped++; 165b285192aSMauro Carvalho Chehab if (mdl->skipped >= atomic_read(&s->q_busy.depth)-1) { 166b285192aSMauro Carvalho Chehab /* mdl must have fallen out of rotation */ 167*6beb1388SMauro Carvalho Chehab CX18_WARN("Skipped %s, MDL %d, %d times - it must have dropped out of rotation\n", 168*6beb1388SMauro Carvalho Chehab s->name, mdl->id, 169b285192aSMauro Carvalho Chehab mdl->skipped); 170b285192aSMauro Carvalho Chehab /* Sweep it up to put it back into rotation */ 171b285192aSMauro Carvalho Chehab list_move_tail(&mdl->list, &sweep_up); 172b285192aSMauro Carvalho Chehab atomic_dec(&s->q_busy.depth); 173b285192aSMauro Carvalho Chehab } 174b285192aSMauro Carvalho Chehab continue; 175b285192aSMauro Carvalho Chehab } 176b285192aSMauro Carvalho Chehab /* 177b285192aSMauro Carvalho Chehab * We pull the desired mdl off of the queue here. Something 178b285192aSMauro Carvalho Chehab * will have to put it back on a queue later. 179b285192aSMauro Carvalho Chehab */ 180b285192aSMauro Carvalho Chehab list_del_init(&mdl->list); 181b285192aSMauro Carvalho Chehab atomic_dec(&s->q_busy.depth); 182b285192aSMauro Carvalho Chehab ret = mdl; 183b285192aSMauro Carvalho Chehab break; 184b285192aSMauro Carvalho Chehab } 185b285192aSMauro Carvalho Chehab spin_unlock(&s->q_busy.lock); 186b285192aSMauro Carvalho Chehab 187b285192aSMauro Carvalho Chehab /* 188b285192aSMauro Carvalho Chehab * We found the mdl for which we were looking. Get it ready for 189b285192aSMauro Carvalho Chehab * the caller to put on q_full or in the dvb ring buffer. 190b285192aSMauro Carvalho Chehab */ 191b285192aSMauro Carvalho Chehab if (ret != NULL) { 192b285192aSMauro Carvalho Chehab ret->bytesused = bytesused; 193b285192aSMauro Carvalho Chehab ret->skipped = 0; 194b285192aSMauro Carvalho Chehab /* 0'ed readpos, m_flags & curr_buf when mdl went on q_busy */ 195b285192aSMauro Carvalho Chehab cx18_mdl_update_bufs_for_cpu(s, ret); 196b285192aSMauro Carvalho Chehab if (s->type != CX18_ENC_STREAM_TYPE_TS) 197b285192aSMauro Carvalho Chehab set_bit(CX18_F_M_NEED_SWAP, &ret->m_flags); 198b285192aSMauro Carvalho Chehab } 199b285192aSMauro Carvalho Chehab 200b285192aSMauro Carvalho Chehab /* Put any mdls the firmware is ignoring back into normal rotation */ 201b285192aSMauro Carvalho Chehab list_for_each_entry_safe(mdl, tmp, &sweep_up, list) { 202b285192aSMauro Carvalho Chehab list_del_init(&mdl->list); 203b285192aSMauro Carvalho Chehab cx18_enqueue(s, mdl, &s->q_free); 204b285192aSMauro Carvalho Chehab } 205b285192aSMauro Carvalho Chehab return ret; 206b285192aSMauro Carvalho Chehab } 207b285192aSMauro Carvalho Chehab 208b285192aSMauro Carvalho Chehab /* Move all mdls of a queue, while flushing the mdl */ 209b285192aSMauro Carvalho Chehab static void cx18_queue_flush(struct cx18_stream *s, 210b285192aSMauro Carvalho Chehab struct cx18_queue *q_src, struct cx18_queue *q_dst) 211b285192aSMauro Carvalho Chehab { 212b285192aSMauro Carvalho Chehab struct cx18_mdl *mdl; 213b285192aSMauro Carvalho Chehab 214b285192aSMauro Carvalho Chehab /* It only makes sense to flush to q_free or q_idle */ 215b285192aSMauro Carvalho Chehab if (q_src == q_dst || q_dst == &s->q_full || q_dst == &s->q_busy) 216b285192aSMauro Carvalho Chehab return; 217b285192aSMauro Carvalho Chehab 218b285192aSMauro Carvalho Chehab spin_lock(&q_src->lock); 219b285192aSMauro Carvalho Chehab spin_lock(&q_dst->lock); 220b285192aSMauro Carvalho Chehab while (!list_empty(&q_src->list)) { 221b285192aSMauro Carvalho Chehab mdl = list_first_entry(&q_src->list, struct cx18_mdl, list); 222b285192aSMauro Carvalho Chehab list_move_tail(&mdl->list, &q_dst->list); 223b285192aSMauro Carvalho Chehab mdl->bytesused = 0; 224b285192aSMauro Carvalho Chehab mdl->readpos = 0; 225b285192aSMauro Carvalho Chehab mdl->m_flags = 0; 226b285192aSMauro Carvalho Chehab mdl->skipped = 0; 227b285192aSMauro Carvalho Chehab mdl->curr_buf = NULL; 228b285192aSMauro Carvalho Chehab atomic_inc(&q_dst->depth); 229b285192aSMauro Carvalho Chehab } 230b285192aSMauro Carvalho Chehab cx18_queue_init(q_src); 231b285192aSMauro Carvalho Chehab spin_unlock(&q_src->lock); 232b285192aSMauro Carvalho Chehab spin_unlock(&q_dst->lock); 233b285192aSMauro Carvalho Chehab } 234b285192aSMauro Carvalho Chehab 235b285192aSMauro Carvalho Chehab void cx18_flush_queues(struct cx18_stream *s) 236b285192aSMauro Carvalho Chehab { 237b285192aSMauro Carvalho Chehab cx18_queue_flush(s, &s->q_busy, &s->q_free); 238b285192aSMauro Carvalho Chehab cx18_queue_flush(s, &s->q_full, &s->q_free); 239b285192aSMauro Carvalho Chehab } 240b285192aSMauro Carvalho Chehab 241b285192aSMauro Carvalho Chehab /* 242b285192aSMauro Carvalho Chehab * Note, s->buf_pool is not protected by a lock, 243b285192aSMauro Carvalho Chehab * the stream better not have *anything* going on when calling this 244b285192aSMauro Carvalho Chehab */ 245b285192aSMauro Carvalho Chehab void cx18_unload_queues(struct cx18_stream *s) 246b285192aSMauro Carvalho Chehab { 247b285192aSMauro Carvalho Chehab struct cx18_queue *q_idle = &s->q_idle; 248b285192aSMauro Carvalho Chehab struct cx18_mdl *mdl; 249b285192aSMauro Carvalho Chehab struct cx18_buffer *buf; 250b285192aSMauro Carvalho Chehab 251b285192aSMauro Carvalho Chehab /* Move all MDLS to q_idle */ 252b285192aSMauro Carvalho Chehab cx18_queue_flush(s, &s->q_busy, q_idle); 253b285192aSMauro Carvalho Chehab cx18_queue_flush(s, &s->q_full, q_idle); 254b285192aSMauro Carvalho Chehab cx18_queue_flush(s, &s->q_free, q_idle); 255b285192aSMauro Carvalho Chehab 256b285192aSMauro Carvalho Chehab /* Reset MDL id's and move all buffers back to the stream's buf_pool */ 257b285192aSMauro Carvalho Chehab spin_lock(&q_idle->lock); 258b285192aSMauro Carvalho Chehab list_for_each_entry(mdl, &q_idle->list, list) { 259b285192aSMauro Carvalho Chehab while (!list_empty(&mdl->buf_list)) { 260b285192aSMauro Carvalho Chehab buf = list_first_entry(&mdl->buf_list, 261b285192aSMauro Carvalho Chehab struct cx18_buffer, list); 262b285192aSMauro Carvalho Chehab list_move_tail(&buf->list, &s->buf_pool); 263b285192aSMauro Carvalho Chehab buf->bytesused = 0; 264b285192aSMauro Carvalho Chehab buf->readpos = 0; 265b285192aSMauro Carvalho Chehab } 266b285192aSMauro Carvalho Chehab mdl->id = s->mdl_base_idx; /* reset id to a "safe" value */ 267b285192aSMauro Carvalho Chehab /* all other mdl fields were cleared by cx18_queue_flush() */ 268b285192aSMauro Carvalho Chehab } 269b285192aSMauro Carvalho Chehab spin_unlock(&q_idle->lock); 270b285192aSMauro Carvalho Chehab } 271b285192aSMauro Carvalho Chehab 272b285192aSMauro Carvalho Chehab /* 273b285192aSMauro Carvalho Chehab * Note, s->buf_pool is not protected by a lock, 274b285192aSMauro Carvalho Chehab * the stream better not have *anything* going on when calling this 275b285192aSMauro Carvalho Chehab */ 276b285192aSMauro Carvalho Chehab void cx18_load_queues(struct cx18_stream *s) 277b285192aSMauro Carvalho Chehab { 278b285192aSMauro Carvalho Chehab struct cx18 *cx = s->cx; 279b285192aSMauro Carvalho Chehab struct cx18_mdl *mdl; 280b285192aSMauro Carvalho Chehab struct cx18_buffer *buf; 281b285192aSMauro Carvalho Chehab int mdl_id; 282b285192aSMauro Carvalho Chehab int i; 283b285192aSMauro Carvalho Chehab u32 partial_buf_size; 284b285192aSMauro Carvalho Chehab 285b285192aSMauro Carvalho Chehab /* 286b285192aSMauro Carvalho Chehab * Attach buffers to MDLs, give the MDLs ids, and add MDLs to q_free 287b285192aSMauro Carvalho Chehab * Excess MDLs are left on q_idle 288b285192aSMauro Carvalho Chehab * Excess buffers are left in buf_pool and/or on an MDL in q_idle 289b285192aSMauro Carvalho Chehab */ 290b285192aSMauro Carvalho Chehab mdl_id = s->mdl_base_idx; 291b285192aSMauro Carvalho Chehab for (mdl = cx18_dequeue(s, &s->q_idle), i = s->bufs_per_mdl; 292b285192aSMauro Carvalho Chehab mdl != NULL && i == s->bufs_per_mdl; 293b285192aSMauro Carvalho Chehab mdl = cx18_dequeue(s, &s->q_idle)) { 294b285192aSMauro Carvalho Chehab 295b285192aSMauro Carvalho Chehab mdl->id = mdl_id; 296b285192aSMauro Carvalho Chehab 297b285192aSMauro Carvalho Chehab for (i = 0; i < s->bufs_per_mdl; i++) { 298b285192aSMauro Carvalho Chehab if (list_empty(&s->buf_pool)) 299b285192aSMauro Carvalho Chehab break; 300b285192aSMauro Carvalho Chehab 301b285192aSMauro Carvalho Chehab buf = list_first_entry(&s->buf_pool, struct cx18_buffer, 302b285192aSMauro Carvalho Chehab list); 303b285192aSMauro Carvalho Chehab list_move_tail(&buf->list, &mdl->buf_list); 304b285192aSMauro Carvalho Chehab 305b285192aSMauro Carvalho Chehab /* update the firmware's MDL array with this buffer */ 306b285192aSMauro Carvalho Chehab cx18_writel(cx, buf->dma_handle, 307b285192aSMauro Carvalho Chehab &cx->scb->cpu_mdl[mdl_id + i].paddr); 308b285192aSMauro Carvalho Chehab cx18_writel(cx, s->buf_size, 309b285192aSMauro Carvalho Chehab &cx->scb->cpu_mdl[mdl_id + i].length); 310b285192aSMauro Carvalho Chehab } 311b285192aSMauro Carvalho Chehab 312b285192aSMauro Carvalho Chehab if (i == s->bufs_per_mdl) { 313b285192aSMauro Carvalho Chehab /* 314b285192aSMauro Carvalho Chehab * The encoder doesn't honor s->mdl_size. So in the 315b285192aSMauro Carvalho Chehab * case of a non-integral number of buffers to meet 316b285192aSMauro Carvalho Chehab * mdl_size, we lie about the size of the last buffer 317b285192aSMauro Carvalho Chehab * in the MDL to get the encoder to really only send 318b285192aSMauro Carvalho Chehab * us mdl_size bytes per MDL transfer. 319b285192aSMauro Carvalho Chehab */ 320b285192aSMauro Carvalho Chehab partial_buf_size = s->mdl_size % s->buf_size; 321b285192aSMauro Carvalho Chehab if (partial_buf_size) { 322b285192aSMauro Carvalho Chehab cx18_writel(cx, partial_buf_size, 323b285192aSMauro Carvalho Chehab &cx->scb->cpu_mdl[mdl_id + i - 1].length); 324b285192aSMauro Carvalho Chehab } 325b285192aSMauro Carvalho Chehab cx18_enqueue(s, mdl, &s->q_free); 326b285192aSMauro Carvalho Chehab } else { 327b285192aSMauro Carvalho Chehab /* Not enough buffers for this MDL; we won't use it */ 328b285192aSMauro Carvalho Chehab cx18_push(s, mdl, &s->q_idle); 329b285192aSMauro Carvalho Chehab } 330b285192aSMauro Carvalho Chehab mdl_id += i; 331b285192aSMauro Carvalho Chehab } 332b285192aSMauro Carvalho Chehab } 333b285192aSMauro Carvalho Chehab 334b285192aSMauro Carvalho Chehab void _cx18_mdl_sync_for_device(struct cx18_stream *s, struct cx18_mdl *mdl) 335b285192aSMauro Carvalho Chehab { 336b285192aSMauro Carvalho Chehab int dma = s->dma; 337b285192aSMauro Carvalho Chehab u32 buf_size = s->buf_size; 338b285192aSMauro Carvalho Chehab struct pci_dev *pci_dev = s->cx->pci_dev; 339b285192aSMauro Carvalho Chehab struct cx18_buffer *buf; 340b285192aSMauro Carvalho Chehab 341b285192aSMauro Carvalho Chehab list_for_each_entry(buf, &mdl->buf_list, list) 342b285192aSMauro Carvalho Chehab pci_dma_sync_single_for_device(pci_dev, buf->dma_handle, 343b285192aSMauro Carvalho Chehab buf_size, dma); 344b285192aSMauro Carvalho Chehab } 345b285192aSMauro Carvalho Chehab 346b285192aSMauro Carvalho Chehab int cx18_stream_alloc(struct cx18_stream *s) 347b285192aSMauro Carvalho Chehab { 348b285192aSMauro Carvalho Chehab struct cx18 *cx = s->cx; 349b285192aSMauro Carvalho Chehab int i; 350b285192aSMauro Carvalho Chehab 351b285192aSMauro Carvalho Chehab if (s->buffers == 0) 352b285192aSMauro Carvalho Chehab return 0; 353b285192aSMauro Carvalho Chehab 354*6beb1388SMauro Carvalho Chehab CX18_DEBUG_INFO("Allocate %s stream: %d x %d buffers (%d.%02d kB total)\n", 355b285192aSMauro Carvalho Chehab s->name, s->buffers, s->buf_size, 356b285192aSMauro Carvalho Chehab s->buffers * s->buf_size / 1024, 357b285192aSMauro Carvalho Chehab (s->buffers * s->buf_size * 100 / 1024) % 100); 358b285192aSMauro Carvalho Chehab 359b285192aSMauro Carvalho Chehab if (((char __iomem *)&cx->scb->cpu_mdl[cx->free_mdl_idx + s->buffers] - 360b285192aSMauro Carvalho Chehab (char __iomem *)cx->scb) > SCB_RESERVED_SIZE) { 361b285192aSMauro Carvalho Chehab unsigned bufsz = (((char __iomem *)cx->scb) + SCB_RESERVED_SIZE - 362b285192aSMauro Carvalho Chehab ((char __iomem *)cx->scb->cpu_mdl)); 363b285192aSMauro Carvalho Chehab 364b285192aSMauro Carvalho Chehab CX18_ERR("Too many buffers, cannot fit in SCB area\n"); 365339f06c5SMauro Carvalho Chehab CX18_ERR("Max buffers = %zu\n", 366b285192aSMauro Carvalho Chehab bufsz / sizeof(struct cx18_mdl_ent)); 367b285192aSMauro Carvalho Chehab return -ENOMEM; 368b285192aSMauro Carvalho Chehab } 369b285192aSMauro Carvalho Chehab 370b285192aSMauro Carvalho Chehab s->mdl_base_idx = cx->free_mdl_idx; 371b285192aSMauro Carvalho Chehab 372b285192aSMauro Carvalho Chehab /* allocate stream buffers and MDLs */ 373b285192aSMauro Carvalho Chehab for (i = 0; i < s->buffers; i++) { 374b285192aSMauro Carvalho Chehab struct cx18_mdl *mdl; 375b285192aSMauro Carvalho Chehab struct cx18_buffer *buf; 376b285192aSMauro Carvalho Chehab 377b285192aSMauro Carvalho Chehab /* 1 MDL per buffer to handle the worst & also default case */ 378b285192aSMauro Carvalho Chehab mdl = kzalloc(sizeof(struct cx18_mdl), GFP_KERNEL|__GFP_NOWARN); 379b285192aSMauro Carvalho Chehab if (mdl == NULL) 380b285192aSMauro Carvalho Chehab break; 381b285192aSMauro Carvalho Chehab 382b285192aSMauro Carvalho Chehab buf = kzalloc(sizeof(struct cx18_buffer), 383b285192aSMauro Carvalho Chehab GFP_KERNEL|__GFP_NOWARN); 384b285192aSMauro Carvalho Chehab if (buf == NULL) { 385b285192aSMauro Carvalho Chehab kfree(mdl); 386b285192aSMauro Carvalho Chehab break; 387b285192aSMauro Carvalho Chehab } 388b285192aSMauro Carvalho Chehab 389b285192aSMauro Carvalho Chehab buf->buf = kmalloc(s->buf_size, GFP_KERNEL|__GFP_NOWARN); 390b285192aSMauro Carvalho Chehab if (buf->buf == NULL) { 391b285192aSMauro Carvalho Chehab kfree(mdl); 392b285192aSMauro Carvalho Chehab kfree(buf); 393b285192aSMauro Carvalho Chehab break; 394b285192aSMauro Carvalho Chehab } 395b285192aSMauro Carvalho Chehab 396b285192aSMauro Carvalho Chehab INIT_LIST_HEAD(&mdl->list); 397b285192aSMauro Carvalho Chehab INIT_LIST_HEAD(&mdl->buf_list); 398b285192aSMauro Carvalho Chehab mdl->id = s->mdl_base_idx; /* a somewhat safe value */ 399b285192aSMauro Carvalho Chehab cx18_enqueue(s, mdl, &s->q_idle); 400b285192aSMauro Carvalho Chehab 401b285192aSMauro Carvalho Chehab INIT_LIST_HEAD(&buf->list); 402b285192aSMauro Carvalho Chehab buf->dma_handle = pci_map_single(s->cx->pci_dev, 403b285192aSMauro Carvalho Chehab buf->buf, s->buf_size, s->dma); 404b285192aSMauro Carvalho Chehab cx18_buf_sync_for_cpu(s, buf); 405b285192aSMauro Carvalho Chehab list_add_tail(&buf->list, &s->buf_pool); 406b285192aSMauro Carvalho Chehab } 407b285192aSMauro Carvalho Chehab if (i == s->buffers) { 408b285192aSMauro Carvalho Chehab cx->free_mdl_idx += s->buffers; 409b285192aSMauro Carvalho Chehab return 0; 410b285192aSMauro Carvalho Chehab } 411b285192aSMauro Carvalho Chehab CX18_ERR("Couldn't allocate buffers for %s stream\n", s->name); 412b285192aSMauro Carvalho Chehab cx18_stream_free(s); 413b285192aSMauro Carvalho Chehab return -ENOMEM; 414b285192aSMauro Carvalho Chehab } 415b285192aSMauro Carvalho Chehab 416b285192aSMauro Carvalho Chehab void cx18_stream_free(struct cx18_stream *s) 417b285192aSMauro Carvalho Chehab { 418b285192aSMauro Carvalho Chehab struct cx18_mdl *mdl; 419b285192aSMauro Carvalho Chehab struct cx18_buffer *buf; 420b285192aSMauro Carvalho Chehab struct cx18 *cx = s->cx; 421b285192aSMauro Carvalho Chehab 422b285192aSMauro Carvalho Chehab CX18_DEBUG_INFO("Deallocating buffers for %s stream\n", s->name); 423b285192aSMauro Carvalho Chehab 424b285192aSMauro Carvalho Chehab /* move all buffers to buf_pool and all MDLs to q_idle */ 425b285192aSMauro Carvalho Chehab cx18_unload_queues(s); 426b285192aSMauro Carvalho Chehab 427b285192aSMauro Carvalho Chehab /* empty q_idle */ 428b285192aSMauro Carvalho Chehab while ((mdl = cx18_dequeue(s, &s->q_idle))) 429b285192aSMauro Carvalho Chehab kfree(mdl); 430b285192aSMauro Carvalho Chehab 431b285192aSMauro Carvalho Chehab /* empty buf_pool */ 432b285192aSMauro Carvalho Chehab while (!list_empty(&s->buf_pool)) { 433b285192aSMauro Carvalho Chehab buf = list_first_entry(&s->buf_pool, struct cx18_buffer, list); 434b285192aSMauro Carvalho Chehab list_del_init(&buf->list); 435b285192aSMauro Carvalho Chehab 436b285192aSMauro Carvalho Chehab pci_unmap_single(s->cx->pci_dev, buf->dma_handle, 437b285192aSMauro Carvalho Chehab s->buf_size, s->dma); 438b285192aSMauro Carvalho Chehab kfree(buf->buf); 439b285192aSMauro Carvalho Chehab kfree(buf); 440b285192aSMauro Carvalho Chehab } 441b285192aSMauro Carvalho Chehab } 442