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 */ 167b285192aSMauro Carvalho Chehab CX18_WARN("Skipped %s, MDL %d, %d " 168b285192aSMauro Carvalho Chehab "times - it must have dropped out of " 169b285192aSMauro Carvalho Chehab "rotation\n", s->name, mdl->id, 170b285192aSMauro Carvalho Chehab mdl->skipped); 171b285192aSMauro Carvalho Chehab /* Sweep it up to put it back into rotation */ 172b285192aSMauro Carvalho Chehab list_move_tail(&mdl->list, &sweep_up); 173b285192aSMauro Carvalho Chehab atomic_dec(&s->q_busy.depth); 174b285192aSMauro Carvalho Chehab } 175b285192aSMauro Carvalho Chehab continue; 176b285192aSMauro Carvalho Chehab } 177b285192aSMauro Carvalho Chehab /* 178b285192aSMauro Carvalho Chehab * We pull the desired mdl off of the queue here. Something 179b285192aSMauro Carvalho Chehab * will have to put it back on a queue later. 180b285192aSMauro Carvalho Chehab */ 181b285192aSMauro Carvalho Chehab list_del_init(&mdl->list); 182b285192aSMauro Carvalho Chehab atomic_dec(&s->q_busy.depth); 183b285192aSMauro Carvalho Chehab ret = mdl; 184b285192aSMauro Carvalho Chehab break; 185b285192aSMauro Carvalho Chehab } 186b285192aSMauro Carvalho Chehab spin_unlock(&s->q_busy.lock); 187b285192aSMauro Carvalho Chehab 188b285192aSMauro Carvalho Chehab /* 189b285192aSMauro Carvalho Chehab * We found the mdl for which we were looking. Get it ready for 190b285192aSMauro Carvalho Chehab * the caller to put on q_full or in the dvb ring buffer. 191b285192aSMauro Carvalho Chehab */ 192b285192aSMauro Carvalho Chehab if (ret != NULL) { 193b285192aSMauro Carvalho Chehab ret->bytesused = bytesused; 194b285192aSMauro Carvalho Chehab ret->skipped = 0; 195b285192aSMauro Carvalho Chehab /* 0'ed readpos, m_flags & curr_buf when mdl went on q_busy */ 196b285192aSMauro Carvalho Chehab cx18_mdl_update_bufs_for_cpu(s, ret); 197b285192aSMauro Carvalho Chehab if (s->type != CX18_ENC_STREAM_TYPE_TS) 198b285192aSMauro Carvalho Chehab set_bit(CX18_F_M_NEED_SWAP, &ret->m_flags); 199b285192aSMauro Carvalho Chehab } 200b285192aSMauro Carvalho Chehab 201b285192aSMauro Carvalho Chehab /* Put any mdls the firmware is ignoring back into normal rotation */ 202b285192aSMauro Carvalho Chehab list_for_each_entry_safe(mdl, tmp, &sweep_up, list) { 203b285192aSMauro Carvalho Chehab list_del_init(&mdl->list); 204b285192aSMauro Carvalho Chehab cx18_enqueue(s, mdl, &s->q_free); 205b285192aSMauro Carvalho Chehab } 206b285192aSMauro Carvalho Chehab return ret; 207b285192aSMauro Carvalho Chehab } 208b285192aSMauro Carvalho Chehab 209b285192aSMauro Carvalho Chehab /* Move all mdls of a queue, while flushing the mdl */ 210b285192aSMauro Carvalho Chehab static void cx18_queue_flush(struct cx18_stream *s, 211b285192aSMauro Carvalho Chehab struct cx18_queue *q_src, struct cx18_queue *q_dst) 212b285192aSMauro Carvalho Chehab { 213b285192aSMauro Carvalho Chehab struct cx18_mdl *mdl; 214b285192aSMauro Carvalho Chehab 215b285192aSMauro Carvalho Chehab /* It only makes sense to flush to q_free or q_idle */ 216b285192aSMauro Carvalho Chehab if (q_src == q_dst || q_dst == &s->q_full || q_dst == &s->q_busy) 217b285192aSMauro Carvalho Chehab return; 218b285192aSMauro Carvalho Chehab 219b285192aSMauro Carvalho Chehab spin_lock(&q_src->lock); 220b285192aSMauro Carvalho Chehab spin_lock(&q_dst->lock); 221b285192aSMauro Carvalho Chehab while (!list_empty(&q_src->list)) { 222b285192aSMauro Carvalho Chehab mdl = list_first_entry(&q_src->list, struct cx18_mdl, list); 223b285192aSMauro Carvalho Chehab list_move_tail(&mdl->list, &q_dst->list); 224b285192aSMauro Carvalho Chehab mdl->bytesused = 0; 225b285192aSMauro Carvalho Chehab mdl->readpos = 0; 226b285192aSMauro Carvalho Chehab mdl->m_flags = 0; 227b285192aSMauro Carvalho Chehab mdl->skipped = 0; 228b285192aSMauro Carvalho Chehab mdl->curr_buf = NULL; 229b285192aSMauro Carvalho Chehab atomic_inc(&q_dst->depth); 230b285192aSMauro Carvalho Chehab } 231b285192aSMauro Carvalho Chehab cx18_queue_init(q_src); 232b285192aSMauro Carvalho Chehab spin_unlock(&q_src->lock); 233b285192aSMauro Carvalho Chehab spin_unlock(&q_dst->lock); 234b285192aSMauro Carvalho Chehab } 235b285192aSMauro Carvalho Chehab 236b285192aSMauro Carvalho Chehab void cx18_flush_queues(struct cx18_stream *s) 237b285192aSMauro Carvalho Chehab { 238b285192aSMauro Carvalho Chehab cx18_queue_flush(s, &s->q_busy, &s->q_free); 239b285192aSMauro Carvalho Chehab cx18_queue_flush(s, &s->q_full, &s->q_free); 240b285192aSMauro Carvalho Chehab } 241b285192aSMauro Carvalho Chehab 242b285192aSMauro Carvalho Chehab /* 243b285192aSMauro Carvalho Chehab * Note, s->buf_pool is not protected by a lock, 244b285192aSMauro Carvalho Chehab * the stream better not have *anything* going on when calling this 245b285192aSMauro Carvalho Chehab */ 246b285192aSMauro Carvalho Chehab void cx18_unload_queues(struct cx18_stream *s) 247b285192aSMauro Carvalho Chehab { 248b285192aSMauro Carvalho Chehab struct cx18_queue *q_idle = &s->q_idle; 249b285192aSMauro Carvalho Chehab struct cx18_mdl *mdl; 250b285192aSMauro Carvalho Chehab struct cx18_buffer *buf; 251b285192aSMauro Carvalho Chehab 252b285192aSMauro Carvalho Chehab /* Move all MDLS to q_idle */ 253b285192aSMauro Carvalho Chehab cx18_queue_flush(s, &s->q_busy, q_idle); 254b285192aSMauro Carvalho Chehab cx18_queue_flush(s, &s->q_full, q_idle); 255b285192aSMauro Carvalho Chehab cx18_queue_flush(s, &s->q_free, q_idle); 256b285192aSMauro Carvalho Chehab 257b285192aSMauro Carvalho Chehab /* Reset MDL id's and move all buffers back to the stream's buf_pool */ 258b285192aSMauro Carvalho Chehab spin_lock(&q_idle->lock); 259b285192aSMauro Carvalho Chehab list_for_each_entry(mdl, &q_idle->list, list) { 260b285192aSMauro Carvalho Chehab while (!list_empty(&mdl->buf_list)) { 261b285192aSMauro Carvalho Chehab buf = list_first_entry(&mdl->buf_list, 262b285192aSMauro Carvalho Chehab struct cx18_buffer, list); 263b285192aSMauro Carvalho Chehab list_move_tail(&buf->list, &s->buf_pool); 264b285192aSMauro Carvalho Chehab buf->bytesused = 0; 265b285192aSMauro Carvalho Chehab buf->readpos = 0; 266b285192aSMauro Carvalho Chehab } 267b285192aSMauro Carvalho Chehab mdl->id = s->mdl_base_idx; /* reset id to a "safe" value */ 268b285192aSMauro Carvalho Chehab /* all other mdl fields were cleared by cx18_queue_flush() */ 269b285192aSMauro Carvalho Chehab } 270b285192aSMauro Carvalho Chehab spin_unlock(&q_idle->lock); 271b285192aSMauro Carvalho Chehab } 272b285192aSMauro Carvalho Chehab 273b285192aSMauro Carvalho Chehab /* 274b285192aSMauro Carvalho Chehab * Note, s->buf_pool is not protected by a lock, 275b285192aSMauro Carvalho Chehab * the stream better not have *anything* going on when calling this 276b285192aSMauro Carvalho Chehab */ 277b285192aSMauro Carvalho Chehab void cx18_load_queues(struct cx18_stream *s) 278b285192aSMauro Carvalho Chehab { 279b285192aSMauro Carvalho Chehab struct cx18 *cx = s->cx; 280b285192aSMauro Carvalho Chehab struct cx18_mdl *mdl; 281b285192aSMauro Carvalho Chehab struct cx18_buffer *buf; 282b285192aSMauro Carvalho Chehab int mdl_id; 283b285192aSMauro Carvalho Chehab int i; 284b285192aSMauro Carvalho Chehab u32 partial_buf_size; 285b285192aSMauro Carvalho Chehab 286b285192aSMauro Carvalho Chehab /* 287b285192aSMauro Carvalho Chehab * Attach buffers to MDLs, give the MDLs ids, and add MDLs to q_free 288b285192aSMauro Carvalho Chehab * Excess MDLs are left on q_idle 289b285192aSMauro Carvalho Chehab * Excess buffers are left in buf_pool and/or on an MDL in q_idle 290b285192aSMauro Carvalho Chehab */ 291b285192aSMauro Carvalho Chehab mdl_id = s->mdl_base_idx; 292b285192aSMauro Carvalho Chehab for (mdl = cx18_dequeue(s, &s->q_idle), i = s->bufs_per_mdl; 293b285192aSMauro Carvalho Chehab mdl != NULL && i == s->bufs_per_mdl; 294b285192aSMauro Carvalho Chehab mdl = cx18_dequeue(s, &s->q_idle)) { 295b285192aSMauro Carvalho Chehab 296b285192aSMauro Carvalho Chehab mdl->id = mdl_id; 297b285192aSMauro Carvalho Chehab 298b285192aSMauro Carvalho Chehab for (i = 0; i < s->bufs_per_mdl; i++) { 299b285192aSMauro Carvalho Chehab if (list_empty(&s->buf_pool)) 300b285192aSMauro Carvalho Chehab break; 301b285192aSMauro Carvalho Chehab 302b285192aSMauro Carvalho Chehab buf = list_first_entry(&s->buf_pool, struct cx18_buffer, 303b285192aSMauro Carvalho Chehab list); 304b285192aSMauro Carvalho Chehab list_move_tail(&buf->list, &mdl->buf_list); 305b285192aSMauro Carvalho Chehab 306b285192aSMauro Carvalho Chehab /* update the firmware's MDL array with this buffer */ 307b285192aSMauro Carvalho Chehab cx18_writel(cx, buf->dma_handle, 308b285192aSMauro Carvalho Chehab &cx->scb->cpu_mdl[mdl_id + i].paddr); 309b285192aSMauro Carvalho Chehab cx18_writel(cx, s->buf_size, 310b285192aSMauro Carvalho Chehab &cx->scb->cpu_mdl[mdl_id + i].length); 311b285192aSMauro Carvalho Chehab } 312b285192aSMauro Carvalho Chehab 313b285192aSMauro Carvalho Chehab if (i == s->bufs_per_mdl) { 314b285192aSMauro Carvalho Chehab /* 315b285192aSMauro Carvalho Chehab * The encoder doesn't honor s->mdl_size. So in the 316b285192aSMauro Carvalho Chehab * case of a non-integral number of buffers to meet 317b285192aSMauro Carvalho Chehab * mdl_size, we lie about the size of the last buffer 318b285192aSMauro Carvalho Chehab * in the MDL to get the encoder to really only send 319b285192aSMauro Carvalho Chehab * us mdl_size bytes per MDL transfer. 320b285192aSMauro Carvalho Chehab */ 321b285192aSMauro Carvalho Chehab partial_buf_size = s->mdl_size % s->buf_size; 322b285192aSMauro Carvalho Chehab if (partial_buf_size) { 323b285192aSMauro Carvalho Chehab cx18_writel(cx, partial_buf_size, 324b285192aSMauro Carvalho Chehab &cx->scb->cpu_mdl[mdl_id + i - 1].length); 325b285192aSMauro Carvalho Chehab } 326b285192aSMauro Carvalho Chehab cx18_enqueue(s, mdl, &s->q_free); 327b285192aSMauro Carvalho Chehab } else { 328b285192aSMauro Carvalho Chehab /* Not enough buffers for this MDL; we won't use it */ 329b285192aSMauro Carvalho Chehab cx18_push(s, mdl, &s->q_idle); 330b285192aSMauro Carvalho Chehab } 331b285192aSMauro Carvalho Chehab mdl_id += i; 332b285192aSMauro Carvalho Chehab } 333b285192aSMauro Carvalho Chehab } 334b285192aSMauro Carvalho Chehab 335b285192aSMauro Carvalho Chehab void _cx18_mdl_sync_for_device(struct cx18_stream *s, struct cx18_mdl *mdl) 336b285192aSMauro Carvalho Chehab { 337b285192aSMauro Carvalho Chehab int dma = s->dma; 338b285192aSMauro Carvalho Chehab u32 buf_size = s->buf_size; 339b285192aSMauro Carvalho Chehab struct pci_dev *pci_dev = s->cx->pci_dev; 340b285192aSMauro Carvalho Chehab struct cx18_buffer *buf; 341b285192aSMauro Carvalho Chehab 342b285192aSMauro Carvalho Chehab list_for_each_entry(buf, &mdl->buf_list, list) 343b285192aSMauro Carvalho Chehab pci_dma_sync_single_for_device(pci_dev, buf->dma_handle, 344b285192aSMauro Carvalho Chehab buf_size, dma); 345b285192aSMauro Carvalho Chehab } 346b285192aSMauro Carvalho Chehab 347b285192aSMauro Carvalho Chehab int cx18_stream_alloc(struct cx18_stream *s) 348b285192aSMauro Carvalho Chehab { 349b285192aSMauro Carvalho Chehab struct cx18 *cx = s->cx; 350b285192aSMauro Carvalho Chehab int i; 351b285192aSMauro Carvalho Chehab 352b285192aSMauro Carvalho Chehab if (s->buffers == 0) 353b285192aSMauro Carvalho Chehab return 0; 354b285192aSMauro Carvalho Chehab 355b285192aSMauro Carvalho Chehab CX18_DEBUG_INFO("Allocate %s stream: %d x %d buffers " 356b285192aSMauro Carvalho Chehab "(%d.%02d kB total)\n", 357b285192aSMauro Carvalho Chehab s->name, s->buffers, s->buf_size, 358b285192aSMauro Carvalho Chehab s->buffers * s->buf_size / 1024, 359b285192aSMauro Carvalho Chehab (s->buffers * s->buf_size * 100 / 1024) % 100); 360b285192aSMauro Carvalho Chehab 361b285192aSMauro Carvalho Chehab if (((char __iomem *)&cx->scb->cpu_mdl[cx->free_mdl_idx + s->buffers] - 362b285192aSMauro Carvalho Chehab (char __iomem *)cx->scb) > SCB_RESERVED_SIZE) { 363b285192aSMauro Carvalho Chehab unsigned bufsz = (((char __iomem *)cx->scb) + SCB_RESERVED_SIZE - 364b285192aSMauro Carvalho Chehab ((char __iomem *)cx->scb->cpu_mdl)); 365b285192aSMauro Carvalho Chehab 366b285192aSMauro Carvalho Chehab CX18_ERR("Too many buffers, cannot fit in SCB area\n"); 367*339f06c5SMauro Carvalho Chehab CX18_ERR("Max buffers = %zu\n", 368b285192aSMauro Carvalho Chehab bufsz / sizeof(struct cx18_mdl_ent)); 369b285192aSMauro Carvalho Chehab return -ENOMEM; 370b285192aSMauro Carvalho Chehab } 371b285192aSMauro Carvalho Chehab 372b285192aSMauro Carvalho Chehab s->mdl_base_idx = cx->free_mdl_idx; 373b285192aSMauro Carvalho Chehab 374b285192aSMauro Carvalho Chehab /* allocate stream buffers and MDLs */ 375b285192aSMauro Carvalho Chehab for (i = 0; i < s->buffers; i++) { 376b285192aSMauro Carvalho Chehab struct cx18_mdl *mdl; 377b285192aSMauro Carvalho Chehab struct cx18_buffer *buf; 378b285192aSMauro Carvalho Chehab 379b285192aSMauro Carvalho Chehab /* 1 MDL per buffer to handle the worst & also default case */ 380b285192aSMauro Carvalho Chehab mdl = kzalloc(sizeof(struct cx18_mdl), GFP_KERNEL|__GFP_NOWARN); 381b285192aSMauro Carvalho Chehab if (mdl == NULL) 382b285192aSMauro Carvalho Chehab break; 383b285192aSMauro Carvalho Chehab 384b285192aSMauro Carvalho Chehab buf = kzalloc(sizeof(struct cx18_buffer), 385b285192aSMauro Carvalho Chehab GFP_KERNEL|__GFP_NOWARN); 386b285192aSMauro Carvalho Chehab if (buf == NULL) { 387b285192aSMauro Carvalho Chehab kfree(mdl); 388b285192aSMauro Carvalho Chehab break; 389b285192aSMauro Carvalho Chehab } 390b285192aSMauro Carvalho Chehab 391b285192aSMauro Carvalho Chehab buf->buf = kmalloc(s->buf_size, GFP_KERNEL|__GFP_NOWARN); 392b285192aSMauro Carvalho Chehab if (buf->buf == NULL) { 393b285192aSMauro Carvalho Chehab kfree(mdl); 394b285192aSMauro Carvalho Chehab kfree(buf); 395b285192aSMauro Carvalho Chehab break; 396b285192aSMauro Carvalho Chehab } 397b285192aSMauro Carvalho Chehab 398b285192aSMauro Carvalho Chehab INIT_LIST_HEAD(&mdl->list); 399b285192aSMauro Carvalho Chehab INIT_LIST_HEAD(&mdl->buf_list); 400b285192aSMauro Carvalho Chehab mdl->id = s->mdl_base_idx; /* a somewhat safe value */ 401b285192aSMauro Carvalho Chehab cx18_enqueue(s, mdl, &s->q_idle); 402b285192aSMauro Carvalho Chehab 403b285192aSMauro Carvalho Chehab INIT_LIST_HEAD(&buf->list); 404b285192aSMauro Carvalho Chehab buf->dma_handle = pci_map_single(s->cx->pci_dev, 405b285192aSMauro Carvalho Chehab buf->buf, s->buf_size, s->dma); 406b285192aSMauro Carvalho Chehab cx18_buf_sync_for_cpu(s, buf); 407b285192aSMauro Carvalho Chehab list_add_tail(&buf->list, &s->buf_pool); 408b285192aSMauro Carvalho Chehab } 409b285192aSMauro Carvalho Chehab if (i == s->buffers) { 410b285192aSMauro Carvalho Chehab cx->free_mdl_idx += s->buffers; 411b285192aSMauro Carvalho Chehab return 0; 412b285192aSMauro Carvalho Chehab } 413b285192aSMauro Carvalho Chehab CX18_ERR("Couldn't allocate buffers for %s stream\n", s->name); 414b285192aSMauro Carvalho Chehab cx18_stream_free(s); 415b285192aSMauro Carvalho Chehab return -ENOMEM; 416b285192aSMauro Carvalho Chehab } 417b285192aSMauro Carvalho Chehab 418b285192aSMauro Carvalho Chehab void cx18_stream_free(struct cx18_stream *s) 419b285192aSMauro Carvalho Chehab { 420b285192aSMauro Carvalho Chehab struct cx18_mdl *mdl; 421b285192aSMauro Carvalho Chehab struct cx18_buffer *buf; 422b285192aSMauro Carvalho Chehab struct cx18 *cx = s->cx; 423b285192aSMauro Carvalho Chehab 424b285192aSMauro Carvalho Chehab CX18_DEBUG_INFO("Deallocating buffers for %s stream\n", s->name); 425b285192aSMauro Carvalho Chehab 426b285192aSMauro Carvalho Chehab /* move all buffers to buf_pool and all MDLs to q_idle */ 427b285192aSMauro Carvalho Chehab cx18_unload_queues(s); 428b285192aSMauro Carvalho Chehab 429b285192aSMauro Carvalho Chehab /* empty q_idle */ 430b285192aSMauro Carvalho Chehab while ((mdl = cx18_dequeue(s, &s->q_idle))) 431b285192aSMauro Carvalho Chehab kfree(mdl); 432b285192aSMauro Carvalho Chehab 433b285192aSMauro Carvalho Chehab /* empty buf_pool */ 434b285192aSMauro Carvalho Chehab while (!list_empty(&s->buf_pool)) { 435b285192aSMauro Carvalho Chehab buf = list_first_entry(&s->buf_pool, struct cx18_buffer, list); 436b285192aSMauro Carvalho Chehab list_del_init(&buf->list); 437b285192aSMauro Carvalho Chehab 438b285192aSMauro Carvalho Chehab pci_unmap_single(s->cx->pci_dev, buf->dma_handle, 439b285192aSMauro Carvalho Chehab s->buf_size, s->dma); 440b285192aSMauro Carvalho Chehab kfree(buf->buf); 441b285192aSMauro Carvalho Chehab kfree(buf); 442b285192aSMauro Carvalho Chehab } 443b285192aSMauro Carvalho Chehab } 444