}
        mutex_unlock(&channel->io_mutex);
 
-       mbo = most_get_mbo(channel->iface, channel->channel_id);
+       mbo = most_get_mbo(channel->iface, channel->channel_id, &cdev_aim);
 
        if (!mbo) {
                if ((filp->f_flags & O_NONBLOCK))
                if (wait_event_interruptible(
                            channel->wq,
                            (mbo = most_get_mbo(channel->iface,
-                                               channel->channel_id)) ||
+                                               channel->channel_id,
+                                               &cdev_aim)) ||
                            (!channel->dev)))
                        return -ERESTARTSYS;
        }
 
 static struct device *class_glue_dir;
 static struct ida mdev_id;
 static int modref;
+static int dummy_num_buffers;
 
 struct most_c_obj {
        struct kobject kobj;
        struct most_aim *second_aim;
        int first_aim_refs;
        int second_aim_refs;
+       int first_num_buffers;
+       int second_num_buffers;
        struct list_head trash_fifo;
        struct task_struct *hdm_enqueue_task;
        struct mutex stop_task_mutex;
        }
 
        spin_lock_irqsave(&c->fifo_lock, flags);
+       ++*mbo->num_buffers_ptr;
        list_add_tail(&mbo->list, &c->fifo);
        spin_unlock_irqrestore(&c->fifo_lock, flags);
 
                        goto _error1;
                }
                mbo->complete = compl;
+               mbo->num_buffers_ptr = &dummy_num_buffers;
                if (dir == MOST_CH_RX) {
                        nq_hdm_mbo(mbo);
                        atomic_inc(&c->mbo_nq_level);
  * This attempts to get a free buffer out of the channel fifo.
  * Returns a pointer to MBO on success or NULL otherwise.
  */
-struct mbo *most_get_mbo(struct most_interface *iface, int id)
+struct mbo *most_get_mbo(struct most_interface *iface, int id,
+                        struct most_aim *aim)
 {
        struct mbo *mbo;
        struct most_c_obj *c;
        unsigned long flags;
+       int *num_buffers_ptr;
 
        c = get_channel_by_iface(iface, id);
        if (unlikely(!c))
                return NULL;
+
+       if (c->first_aim_refs && c->second_aim_refs &&
+           ((aim == c->first_aim && c->first_num_buffers <= 0) ||
+            (aim == c->second_aim && c->second_num_buffers <= 0)))
+               return NULL;
+
+       if (aim == c->first_aim)
+               num_buffers_ptr = &c->first_num_buffers;
+       else if (aim == c->second_aim)
+               num_buffers_ptr = &c->second_num_buffers;
+       else
+               num_buffers_ptr = &dummy_num_buffers;
+
        spin_lock_irqsave(&c->fifo_lock, flags);
        if (list_empty(&c->fifo)) {
                spin_unlock_irqrestore(&c->fifo_lock, flags);
                return NULL;
        }
        mbo = list_pop_mbo(&c->fifo);
+       --*num_buffers_ptr;
        spin_unlock_irqrestore(&c->fifo_lock, flags);
+
+       mbo->num_buffers_ptr = num_buffers_ptr;
        mbo->buffer_length = c->cfg.buffer_size;
        return mbo;
 }
                goto error;
 
        c->is_starving = 0;
+       c->first_num_buffers = c->cfg.num_buffers / 2;
+       c->second_num_buffers = c->cfg.num_buffers - c->first_num_buffers;
        atomic_set(&c->mbo_ref, num_buffer);
 
 out:
 
        void *priv;
        struct list_head list;
        struct most_interface *ifp;
+       int *num_buffers_ptr;
        u16 hdm_channel_id;
        void *virt_address;
        dma_addr_t bus_address;
 void most_resume_enqueue(struct most_interface *iface, int channel_idx);
 int most_register_aim(struct most_aim *aim);
 int most_deregister_aim(struct most_aim *aim);
-struct mbo *most_get_mbo(struct most_interface *iface, int channel_idx);
+struct mbo *most_get_mbo(struct most_interface *iface, int channel_idx,
+                        struct most_aim *);
 void most_put_mbo(struct mbo *mbo);
 int most_start_channel(struct most_interface *iface, int channel_idx,
                       struct most_aim *);