struct dvb_device *dvbdev = file->private_data;
        struct dmxdev *dmxdev = dvbdev->priv;
        struct dmx_frontend *front;
+#ifndef DVB_MMAP
+       bool need_ringbuffer = false;
+#else
+       const bool need_ringbuffer = true;
+#endif
 
        dprintk("%s\n", __func__);
 
                return -ENODEV;
        }
 
-       if (((file->f_flags & O_ACCMODE) == O_RDONLY) ||
-           ((file->f_flags & O_ACCMODE) == O_RDWR)) {
+#ifndef DVB_MMAP
+       if ((file->f_flags & O_ACCMODE) == O_RDONLY)
+               need_ringbuffer = true;
+#else
+       if ((file->f_flags & O_ACCMODE) == O_RDWR) {
+               if (!(dmxdev->capabilities & DMXDEV_CAP_DUPLEX)) {
+                       mutex_unlock(&dmxdev->mutex);
+                       return -EOPNOTSUPP;
+               }
+       }
+#endif
+
+       if (need_ringbuffer) {
                void *mem;
 
                if (!dvbdev->readers) {
 {
        struct dvb_device *dvbdev = file->private_data;
        struct dmxdev *dmxdev = dvbdev->priv;
+#ifndef DVB_MMAP
+       bool need_ringbuffer = false;
+#else
+       const bool need_ringbuffer = true;
+#endif
 
        mutex_lock(&dmxdev->mutex);
 
                dmxdev->demux->connect_frontend(dmxdev->demux,
                                                dmxdev->dvr_orig_fe);
        }
-       if (((file->f_flags & O_ACCMODE) == O_RDONLY) ||
-           ((file->f_flags & O_ACCMODE) == O_RDWR)) {
+#ifndef DVB_MMAP
+       if ((file->f_flags & O_ACCMODE) == O_RDONLY)
+               need_ringbuffer = true;
+#endif
+
+       if (need_ringbuffer) {
                if (dvb_vb2_is_streaming(&dmxdev->dvr_vb2_ctx))
                        dvb_vb2_stream_off(&dmxdev->dvr_vb2_ctx);
                dvb_vb2_release(&dmxdev->dvr_vb2_ctx);
 {
        struct dmxdev_filter *dmxdevfilter = filter->priv;
        int ret;
+
        if (!dvb_vb2_is_streaming(&dmxdevfilter->vb2_ctx) &&
            dmxdevfilter->buffer.error) {
                wake_up(&dmxdevfilter->buffer.queue);
 {
        struct dmxdev_filter *dmxdevfilter = feed->priv;
        struct dvb_ringbuffer *buffer;
+#ifdef DVB_MMAP
        struct dvb_vb2_ctx *ctx;
+#endif
        int ret;
 
        spin_lock(&dmxdevfilter->dev->lock);
        if (dmxdevfilter->params.pes.output == DMX_OUT_TAP ||
            dmxdevfilter->params.pes.output == DMX_OUT_TSDEMUX_TAP) {
                buffer = &dmxdevfilter->buffer;
+#ifdef DVB_MMAP
                ctx = &dmxdevfilter->vb2_ctx;
+#endif
        } else {
                buffer = &dmxdevfilter->dev->dvr_buffer;
+#ifdef DVB_MMAP
                ctx = &dmxdevfilter->dev->dvr_vb2_ctx;
+#endif
        }
 
        if (dvb_vb2_is_streaming(ctx)) {
                mutex_unlock(&dmxdevfilter->mutex);
                break;
 
+#ifdef DVB_MMAP
        case DMX_REQBUFS:
                if (mutex_lock_interruptible(&dmxdevfilter->mutex)) {
                        mutex_unlock(&dmxdev->mutex);
                ret = dvb_vb2_dqbuf(&dmxdevfilter->vb2_ctx, parg);
                mutex_unlock(&dmxdevfilter->mutex);
                break;
-
+#endif
        default:
                ret = -EINVAL;
                break;
        return mask;
 }
 
+#ifdef DVB_MMAP
 static int dvb_demux_mmap(struct file *file, struct vm_area_struct *vma)
 {
        struct dmxdev_filter *dmxdevfilter = file->private_data;
 
        return ret;
 }
+#endif
 
 static int dvb_demux_release(struct inode *inode, struct file *file)
 {
        .release = dvb_demux_release,
        .poll = dvb_demux_poll,
        .llseek = default_llseek,
+#ifdef DVB_MMAP
        .mmap = dvb_demux_mmap,
+#endif
 };
 
 static const struct dvb_device dvbdev_demux = {
                ret = dvb_dvr_set_buffer_size(dmxdev, arg);
                break;
 
+#ifdef DVB_MMAP
        case DMX_REQBUFS:
                ret = dvb_vb2_reqbufs(&dmxdev->dvr_vb2_ctx, parg);
                break;
        case DMX_DQBUF:
                ret = dvb_vb2_dqbuf(&dmxdev->dvr_vb2_ctx, parg);
                break;
-
+#endif
        default:
                ret = -EINVAL;
                break;
        struct dvb_device *dvbdev = file->private_data;
        struct dmxdev *dmxdev = dvbdev->priv;
        unsigned int mask = 0;
+#ifndef DVB_MMAP
+       bool need_ringbuffer = false;
+#else
+       const bool need_ringbuffer = true;
+#endif
 
        dprintk("%s\n", __func__);
 
 
        poll_wait(file, &dmxdev->dvr_buffer.queue, wait);
 
-       if (((file->f_flags & O_ACCMODE) == O_RDONLY) ||
-           ((file->f_flags & O_ACCMODE) == O_RDWR)) {
+#ifndef DVB_MMAP
+       if ((file->f_flags & O_ACCMODE) == O_RDONLY)
+               need_ringbuffer = true;
+#endif
+       if (need_ringbuffer) {
                if (dmxdev->dvr_buffer.error)
                        mask |= (POLLIN | POLLRDNORM | POLLPRI | POLLERR);
 
        return mask;
 }
 
+#ifdef DVB_MMAP
 static int dvb_dvr_mmap(struct file *file, struct vm_area_struct *vma)
 {
        struct dvb_device *dvbdev = file->private_data;
        mutex_unlock(&dmxdev->mutex);
        return ret;
 }
+#endif
 
 static const struct file_operations dvb_dvr_fops = {
        .owner = THIS_MODULE,
        .release = dvb_dvr_release,
        .poll = dvb_dvr_poll,
        .llseek = default_llseek,
+#ifdef DVB_MMAP
        .mmap = dvb_dvr_mmap,
+#endif
 };
 
 static const struct dvb_device dvbdev_dvr = {
 
        char    name[DVB_VB2_NAME_MAX + 1];
 };
 
-int dvb_vb2_init(struct dvb_vb2_ctx *ctx, const char *name, int non_blocking);
-int dvb_vb2_release(struct dvb_vb2_ctx *ctx);
 int dvb_vb2_stream_on(struct dvb_vb2_ctx *ctx);
 int dvb_vb2_stream_off(struct dvb_vb2_ctx *ctx);
+#ifndef DVB_MMAP
+static inline int dvb_vb2_init(struct dvb_vb2_ctx *ctx,
+                              const char *name, int non_blocking)
+{
+       return 0;
+};
+static inline int dvb_vb2_release(struct dvb_vb2_ctx *ctx)
+{
+       return 0;
+};
+#define dvb_vb2_is_streaming(ctx) (0)
+#define dvb_vb2_fill_buffer(ctx, file, wait) (0)
+
+static inline unsigned int dvb_vb2_poll(struct dvb_vb2_ctx *ctx,
+                                       struct file *file,
+                                       poll_table *wait)
+{
+       return 0;
+}
+#else
+int dvb_vb2_init(struct dvb_vb2_ctx *ctx, const char *name, int non_blocking);
+int dvb_vb2_release(struct dvb_vb2_ctx *ctx);
 int dvb_vb2_is_streaming(struct dvb_vb2_ctx *ctx);
 int dvb_vb2_fill_buffer(struct dvb_vb2_ctx *ctx,
                        const unsigned char *src, int len);
+unsigned int dvb_vb2_poll(struct dvb_vb2_ctx *ctx, struct file *file,
+                         poll_table *wait);
+#endif
+
 
 int dvb_vb2_reqbufs(struct dvb_vb2_ctx *ctx, struct dmx_requestbuffers *req);
 int dvb_vb2_querybuf(struct dvb_vb2_ctx *ctx, struct dmx_buffer *b);
 int dvb_vb2_qbuf(struct dvb_vb2_ctx *ctx, struct dmx_buffer *b);
 int dvb_vb2_dqbuf(struct dvb_vb2_ctx *ctx, struct dmx_buffer *b);
 int dvb_vb2_mmap(struct dvb_vb2_ctx *ctx, struct vm_area_struct *vma);
-unsigned int dvb_vb2_poll(struct dvb_vb2_ctx *ctx, struct file *file,
-                         poll_table *wait);
 
 #endif /* _DVB_VB2_H */