since it is also used to implement priority handling
        (:ref:`VIDIOC_G_PRIORITY`).
 
-The users of :c:type:`v4l2_fh` (in the V4L2 framework, not the driver) know
-whether a driver uses :c:type:`v4l2_fh` as its ``file->private_data`` pointer
-by testing the ``V4L2_FL_USES_V4L2_FH`` bit in :c:type:`video_device`->flags.
-This bit is set whenever :c:func:`v4l2_fh_init` is called.
-
-struct v4l2_fh is allocated as a part of the driver's own file handle
-structure and ``file->private_data`` is set to it in the driver's ``open()``
-function by the driver. The :c:type:`v4l2_fh` file handle can be retrieved
-from the :c:type:`file` using :c:func:`file_to_v4l2_fh`. Drivers must not
-access ``file->private_data`` directly.
-
-In many cases the struct v4l2_fh will be embedded in a larger
-structure. In that case you should call:
-
-#) :c:func:`v4l2_fh_init` and :c:func:`v4l2_fh_add` in ``open()``
-#) :c:func:`v4l2_fh_del` and :c:func:`v4l2_fh_exit` in ``release()``
-
-Drivers can extract their own file handle structure by using the container_of
-macro.
+struct v4l2_fh is allocated in the driver's ``open()`` file operation handler.
+It is typically embedded in a larger driver-specific structure. The
+:c:type:`v4l2_fh` must be initialized with a call to :c:func:`v4l2_fh_init`,
+and added to the video device with :c:func:`v4l2_fh_add`. This associates the
+:c:type:`v4l2_fh` with the :c:type:`file` by setting ``file->private_data`` to
+point to the :c:type:`v4l2_fh`.
+
+Similarly, the struct v4l2_fh is freed in the driver's ``release()`` file
+operation handler. It must be removed from the video device with
+:c:func:`v4l2_fh_del` and cleaned up with :c:func:`v4l2_fh_exit` before being
+freed.
+
+Drivers must not access ``file->private_data`` directly. They can retrieve the
+:c:type:`v4l2_fh` associated with a :c:type:`file` by calling
+:c:func:`file_to_v4l2_fh`. Drivers can extract their own file handle structure
+by using the container_of macro.
 
 Example:
 
 
                ...
 
-               file->private_data = &my_fh->fh;
-               v4l2_fh_add(&my_fh->fh);
+               v4l2_fh_add(&my_fh->fh, file);
                return 0;
        }
 
   :c:type:`v4l2_file_operations`->open() handler.
 
 :c:func:`v4l2_fh_add <v4l2_fh_add>`
-(:c:type:`fh <v4l2_fh>`)
+(:c:type:`fh <v4l2_fh>`, struct file \*filp)
 
 - Add a :c:type:`v4l2_fh` to :c:type:`video_device` file handle list.
   Must be called once the file handle is completely initialized.
 
 - Same, but it calls v4l2_fh_is_singular with filp->private_data.
 
+.. note::
+        The V4L2 framework knows whether a driver uses :c:type:`v4l2_fh` as its
+        ``file->private_data`` pointer by testing the ``V4L2_FL_USES_V4L2_FH``
+        bit in :c:type:`video_device`->flags. This bit is set whenever
+        :c:func:`v4l2_fh_init` is called.
+
 
 V4L2 fh functions and data structures
 ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
 
 
        ...
 
-       file->private_data = &my_fh->fh;
-       v4l2_fh_add(&my_fh->fh);
+       v4l2_fh_add(&my_fh->fh, file);
        return 0;
 }
 
   初始化文件句柄。这*必须*在驱动的 v4l2_file_operations->open()
   函数中执行。
 
-void v4l2_fh_add(struct v4l2_fh *fh)
+void v4l2_fh_add(struct v4l2_fh *fh, struct file *filp)
 
   添加一个 v4l2_fh 到 video_device 文件句柄列表。一旦文件句柄
   初始化完成就必须调用。
 
        item->type = s->type;
 
        item->open_id = cx->open_id++;
-       filp->private_data = &item->fh;
-       v4l2_fh_add(&item->fh);
+       v4l2_fh_add(&item->fh, filp);
 
        if (item->type == CX18_ENC_STREAM_TYPE_RAD &&
                        v4l2_fh_is_singular_file(filp)) {
 
        v4l2_fh_init(&item->fh, &s->vdev);
        item->itv = itv;
        item->type = s->type;
-
-       filp->private_data = &item->fh;
-       v4l2_fh_add(&item->fh);
+       v4l2_fh_add(&item->fh, filp);
 
        if (item->type == IVTV_ENC_STREAM_TYPE_RAD &&
                        v4l2_fh_is_singular_file(filp)) {
 
 
        fh->port = port;
        v4l2_fh_init(&fh->fh, video_devdata(file));
-       v4l2_fh_add(&fh->fh);
-       file->private_data = &fh->fh;
+       v4l2_fh_add(&fh->fh, file);
 
        return 0;
 }
 
 
        fh->port = port;
        v4l2_fh_init(&fh->fh, video_devdata(file));
-       v4l2_fh_add(&fh->fh);
-       file->private_data = &fh->fh;
+       v4l2_fh_add(&fh->fh, file);
 
        return 0;
 }
 
        }
 
        list_add(&channel->list, &dev->channels);
-       file->private_data = &channel->fh;
-       v4l2_fh_add(&channel->fh);
+       v4l2_fh_add(&channel->fh, file);
 
        allegro_channel_adjust(channel);
 
 
                return ret;
        }
        v4l2_fh_init(&ctx->fh, video_devdata(file));
-       file->private_data = &ctx->fh;
-       v4l2_fh_add(&ctx->fh);
+       v4l2_fh_add(&ctx->fh, file);
 
        ge2d_setup_ctrls(ctx);
 
 
        inst->min_buffer_cap = 2;
        inst->min_buffer_out = 2;
        v4l2_fh_init(&inst->fh, func->vfd);
-       v4l2_fh_add(&inst->fh);
+       v4l2_fh_add(&inst->fh, file);
 
        ret = call_vop(inst, ctrl_init);
        if (ret)
        }
 
        inst->fh.ctrl_handler = &inst->ctrl_handler;
-       file->private_data = &inst->fh;
        inst->state = VPU_CODEC_STATE_DEINIT;
        inst->workqueue = alloc_ordered_workqueue("vpu_inst", WQ_MEM_RECLAIM);
        if (inst->workqueue) {
 
        if (ctx->ops->seq_end_work)
                INIT_WORK(&ctx->seq_end_work, ctx->ops->seq_end_work);
        v4l2_fh_init(&ctx->fh, video_devdata(file));
-       file->private_data = &ctx->fh;
-       v4l2_fh_add(&ctx->fh);
+       v4l2_fh_add(&ctx->fh, file);
        ctx->dev = dev;
        ctx->idx = idx;
 
 
                return -ENOMEM;
 
        v4l2_fh_init(&inst->v4l2_fh, vdev);
-       filp->private_data = &inst->v4l2_fh;
-       v4l2_fh_add(&inst->v4l2_fh);
+       v4l2_fh_add(&inst->v4l2_fh, filp);
 
        INIT_LIST_HEAD(&inst->list);
 
 
                return -ENOMEM;
 
        v4l2_fh_init(&inst->v4l2_fh, vdev);
-       filp->private_data = &inst->v4l2_fh;
-       v4l2_fh_add(&inst->v4l2_fh);
+       v4l2_fh_add(&inst->v4l2_fh, filp);
 
        INIT_LIST_HEAD(&inst->list);
 
 
        }
 
        v4l2_fh_init(&ctx->fh, vdev);
-       file->private_data = &ctx->fh;
-       v4l2_fh_add(&ctx->fh);
+       v4l2_fh_add(&ctx->fh, file);
 
        ctx->e5010 = e5010;
        ctx->fh.m2m_ctx = v4l2_m2m_ctx_init(e5010->m2m_dev, ctx, queue_init);
 
                return -ENOMEM;
 
        v4l2_fh_init(&ctx->fh, video_devdata(file));
-       file->private_data = &ctx->fh;
        ctx->dev = pcdev;
 
        ctx->fh.m2m_ctx = v4l2_m2m_ctx_init(pcdev->m2m_dev, ctx, &queue_init);
        }
 
        ctx->colorspace = V4L2_COLORSPACE_REC709;
-       v4l2_fh_add(&ctx->fh);
+       v4l2_fh_add(&ctx->fh, file);
 
        dprintk(pcdev, "Created instance %p, m2m_ctx: %p\n",
                ctx, ctx->fh.m2m_ctx);
 
        INIT_LIST_HEAD(&ctx->dst_done_queue);
        spin_lock_init(&ctx->done_queue_lock);
        v4l2_fh_init(&ctx->fh, vfd);
-       file->private_data = &ctx->fh;
-       v4l2_fh_add(&ctx->fh);
+       v4l2_fh_add(&ctx->fh, file);
 
        ctx->jpeg = jpeg;
        ctx->fh.m2m_ctx = v4l2_m2m_ctx_init(jpeg->m2m_dev, ctx,
 
        mutex_init(&ctx->slock);
        ctx->id = mdp->id_counter++;
        v4l2_fh_init(&ctx->fh, vfd);
-       file->private_data = &ctx->fh;
        ret = mtk_mdp_ctrls_create(ctx);
        if (ret)
                goto error_ctrls;
 
        /* Use separate control handler per file handle */
        ctx->fh.ctrl_handler = &ctx->ctrl_handler;
-       v4l2_fh_add(&ctx->fh);
+       v4l2_fh_add(&ctx->fh, file);
        INIT_LIST_HEAD(&ctx->list);
 
        ctx->mdp_dev = mdp;
 
        ctx->mdp_dev = mdp;
 
        v4l2_fh_init(&ctx->fh, vdev);
-       file->private_data = &ctx->fh;
        ret = mdp_m2m_ctrls_create(ctx);
        if (ret)
                goto err_exit_fh;
 
        /* Use separate control handler per file handle */
        ctx->fh.ctrl_handler = &ctx->ctrl_handler;
-       v4l2_fh_add(&ctx->fh);
+       v4l2_fh_add(&ctx->fh, file);
 
        mutex_init(&ctx->ctx_lock);
        ctx->m2m_ctx = v4l2_m2m_ctx_init(mdp->m2m_dev, ctx, mdp_m2m_queue_init);
 
        mutex_lock(&dev->dev_mutex);
        ctx->id = dev->id_counter++;
        v4l2_fh_init(&ctx->fh, video_devdata(file));
-       file->private_data = &ctx->fh;
-       v4l2_fh_add(&ctx->fh);
+       v4l2_fh_add(&ctx->fh, file);
        INIT_LIST_HEAD(&ctx->list);
        ctx->dev = dev;
        if (ctx->dev->vdec_pdata->is_subdev_supported) {
 
         */
        ctx->id = dev->id_counter++;
        v4l2_fh_init(&ctx->fh, video_devdata(file));
-       file->private_data = &ctx->fh;
-       v4l2_fh_add(&ctx->fh);
+       v4l2_fh_add(&ctx->fh, file);
        INIT_LIST_HEAD(&ctx->list);
        ctx->dev = dev;
        init_waitqueue_head(&ctx->queue[0]);
 
                goto free_ctrls;
        }
 
-       file->private_data = &ctx->fh;
-       v4l2_fh_add(&ctx->fh);
+       v4l2_fh_add(&ctx->fh, file);
 
        tegra_reset_coded_fmt(ctx);
        tegra_try_coded_fmt(file, &ctx->fh, &ctx->coded_fmt);
 
 
        mutex_init(&ctx->vq_mutex);
        v4l2_fh_init(&ctx->fh, video_devdata(file));
-       file->private_data = &ctx->fh;
        ctx->dw_dev = dw_dev;
 
        ctx->q_data[DW100_QUEUE_SRC].fmt = &formats[0];
                goto err;
        }
 
-       v4l2_fh_add(&ctx->fh);
+       v4l2_fh_add(&ctx->fh, file);
 
        return 0;
 
 
        }
 
        v4l2_fh_init(&ctx->fh, mxc_vfd);
-       file->private_data = &ctx->fh;
-       v4l2_fh_add(&ctx->fh);
+       v4l2_fh_add(&ctx->fh, file);
 
        ctx->mxc_jpeg = mxc_jpeg;
 
 
        }
 
        v4l2_fh_init(&ctx->fh, video_devdata(file));
-       file->private_data = &ctx->fh;
        ctx->dev = dev;
        hdl = &ctx->hdl;
        v4l2_ctrl_handler_init(hdl, 4);
                goto open_unlock;
        }
 
-       v4l2_fh_add(&ctx->fh);
+       v4l2_fh_add(&ctx->fh, file);
        atomic_inc(&dev->num_inst);
 
        dprintk(dev, "Created instance: %p, m2m_ctx: %p\n",
 
        mutex_init(&ctx->vb2_lock);
 
        v4l2_fh_init(&ctx->fh, vdev);
-       file->private_data = &ctx->fh;
 
        ctx->fh.m2m_ctx = v4l2_m2m_ctx_init(m2m->m2m_dev, ctx,
                                            &mxc_isi_m2m_queue_init);
        if (ret)
                goto err_ctrls;
 
-       v4l2_fh_add(&ctx->fh);
+       v4l2_fh_add(&ctx->fh, file);
 
        return 0;
 
 
                return -ENOMEM;
 
        v4l2_fh_init(&ctx->fh, video_devdata(file));
-       file->private_data = &ctx->fh;
        ctx->dev = pcdev;
 
        if (mutex_lock_interruptible(&pcdev->dev_mutex)) {
        clk_prepare_enable(pcdev->clk_emma_ahb);
        ctx->q_data[V4L2_M2M_SRC].fmt = &formats[1];
        ctx->q_data[V4L2_M2M_DST].fmt = &formats[0];
-       v4l2_fh_add(&ctx->fh);
+       v4l2_fh_add(&ctx->fh, file);
        mutex_unlock(&pcdev->dev_mutex);
 
        dprintk(pcdev, "Created instance %p, m2m_ctx: %p\n", ctx, ctx->fh.m2m_ctx);
 
 {
        v4l2_fh_init(&inst->fh, inst->core->vdev_dec);
        inst->fh.ctrl_handler = &inst->ctrl_handler;
-       v4l2_fh_add(&inst->fh);
-       filp->private_data = &inst->fh;
+       v4l2_fh_add(&inst->fh, filp);
 }
 
 static void iris_v4l2_fh_deinit(struct iris_inst *inst, struct file *filp)
 
        v4l2_fh_init(&inst->fh, core->vdev_dec);
 
        inst->fh.ctrl_handler = &inst->ctrl_handler;
-       v4l2_fh_add(&inst->fh);
+       v4l2_fh_add(&inst->fh, file);
        inst->fh.m2m_ctx = inst->m2m_ctx;
-       file->private_data = &inst->fh;
 
        return 0;
 
 
        v4l2_fh_init(&inst->fh, core->vdev_enc);
 
        inst->fh.ctrl_handler = &inst->ctrl_handler;
-       v4l2_fh_add(&inst->fh);
+       v4l2_fh_add(&inst->fh, file);
        inst->fh.m2m_ctx = inst->m2m_ctx;
-       file->private_data = &inst->fh;
 
        return 0;
 
 
        }
 
        v4l2_fh_init(&ctx->fh, video_devdata(file));
-       file->private_data = &ctx->fh;
        ctx->fdp1 = fdp1;
 
        /* Initialise Queues */
        if (ret < 0)
                goto error_pm;
 
-       v4l2_fh_add(&ctx->fh);
+       v4l2_fh_add(&ctx->fh, file);
 
        dprintk(fdp1, "Created instance: %p, m2m_ctx: %p\n",
                ctx, ctx->fh.m2m_ctx);
 
 
        v4l2_fh_init(&ctx->fh, vfd);
        ctx->fh.ctrl_handler = &ctx->ctrl_handler;
-       file->private_data = &ctx->fh;
-       v4l2_fh_add(&ctx->fh);
+       v4l2_fh_add(&ctx->fh, file);
 
        ctx->jpu = jpu;
        ctx->encoder = vfd == &jpu->vfd_encoder;
 
                return -ENOMEM;
 
        v4l2_fh_init(vfh, &video->video);
-       v4l2_fh_add(vfh);
-
-       file->private_data = vfh;
+       v4l2_fh_add(vfh, file);
 
        ret = vsp1_device_get(video->vsp1);
        if (ret < 0) {
 
                return ret;
        }
        v4l2_fh_init(&ctx->fh, video_devdata(file));
-       file->private_data = &ctx->fh;
-       v4l2_fh_add(&ctx->fh);
+       v4l2_fh_add(&ctx->fh, file);
 
        rga_setup_ctrls(ctx);
 
 
        if (ret)
                goto err_cleanup_m2m_ctx;
 
-       filp->private_data = &ctx->fh;
-       v4l2_fh_add(&ctx->fh);
+       v4l2_fh_add(&ctx->fh, filp);
 
        return 0;
 
 
 
        /* Use separate control handler per file handle */
        ctx->fh.ctrl_handler = &ctx->ctrl_handler;
-       file->private_data = &ctx->fh;
-       v4l2_fh_add(&ctx->fh);
+       v4l2_fh_add(&ctx->fh, file);
 
        ctx->gsc_dev = gsc;
        /* Default color format */
 
 
        /* Use separate control handler per file handle */
        ctx->fh.ctrl_handler = &ctx->ctrls.handler;
-       file->private_data = &ctx->fh;
-       v4l2_fh_add(&ctx->fh);
+       v4l2_fh_add(&ctx->fh, file);
 
        /* Setup the device context for memory-to-memory mode */
        ctx->state = FIMC_CTX_M2M;
 
                return ret;
        }
        v4l2_fh_init(&ctx->fh, video_devdata(file));
-       file->private_data = &ctx->fh;
-       v4l2_fh_add(&ctx->fh);
+       v4l2_fh_add(&ctx->fh, file);
 
        g2d_setup_ctrls(ctx);
 
 
        v4l2_fh_init(&ctx->fh, vfd);
        /* Use separate control handler per file handle */
        ctx->fh.ctrl_handler = &ctx->ctrl_handler;
-       file->private_data = &ctx->fh;
-       v4l2_fh_add(&ctx->fh);
+       v4l2_fh_add(&ctx->fh, file);
 
        ctx->jpeg = jpeg;
        if (vfd == jpeg->vfd_encoder) {
 
        }
        init_waitqueue_head(&ctx->queue);
        v4l2_fh_init(&ctx->fh, vdev);
-       file->private_data = &ctx->fh;
-       v4l2_fh_add(&ctx->fh);
+       v4l2_fh_add(&ctx->fh, file);
        ctx->dev = dev;
        INIT_LIST_HEAD(&ctx->src_queue);
        INIT_LIST_HEAD(&ctx->dst_queue);
 
 
        /* Use separate control handler per file handle */
        ctx->fh.ctrl_handler = &ctx->ctrl_handler;
-       file->private_data = &ctx->fh;
-       v4l2_fh_add(&ctx->fh);
+       v4l2_fh_add(&ctx->fh, file);
 
        /* Default format */
        ctx->src = bdisp_dflt_fmt;
 
        ctx->dev = delta;
 
        v4l2_fh_init(&ctx->fh, video_devdata(file));
-       file->private_data = &ctx->fh;
-       v4l2_fh_add(&ctx->fh);
+       v4l2_fh_add(&ctx->fh, file);
 
        INIT_WORK(&ctx->run_work, delta_run_work);
        mutex_init(&ctx->lock);
 
 
        INIT_WORK(&ctx->run_work, hva_run_work);
        v4l2_fh_init(&ctx->fh, video_devdata(file));
-       file->private_data = &ctx->fh;
-       v4l2_fh_add(&ctx->fh);
+       v4l2_fh_add(&ctx->fh, file);
 
        ret = hva_ctrls_setup(ctx);
        if (ret) {
 
        }
 
        v4l2_fh_init(&ctx->fh, video_devdata(file));
-       file->private_data = &ctx->fh;
-       v4l2_fh_add(&ctx->fh);
+       v4l2_fh_add(&ctx->fh, file);
 
        dma2d_setup_ctrls(ctx);
 
 
        deinterlace_prepare_format(&ctx->dst_fmt);
 
        v4l2_fh_init(&ctx->fh, video_devdata(file));
-       file->private_data = &ctx->fh;
        ctx->dev = dev;
 
        ctx->fh.m2m_ctx = v4l2_m2m_ctx_init(dev->m2m_dev, ctx,
                goto err_free;
        }
 
-       v4l2_fh_add(&ctx->fh);
+       v4l2_fh_add(&ctx->fh, file);
 
        mutex_unlock(&dev->dev_mutex);
 
 
        rotate_set_cap_format(ctx, &ctx->dst_fmt, ctx->rotate);
 
        v4l2_fh_init(&ctx->fh, video_devdata(file));
-       file->private_data = &ctx->fh;
        ctx->dev = dev;
 
        ctx->fh.m2m_ctx = v4l2_m2m_ctx_init(dev->m2m_dev, ctx,
                goto err_free;
        }
 
-       v4l2_fh_add(&ctx->fh);
+       v4l2_fh_add(&ctx->fh, file);
 
        ret = rotate_setup_ctrls(ctx);
        if (ret)
 
                return -ENOMEM;
 
        v4l2_fh_init(&handle->vfh, &video->video);
-       v4l2_fh_add(&handle->vfh);
+       v4l2_fh_add(&handle->vfh, file);
 
        /* If this is the first user, initialise the pipeline. */
        if (omap3isp_get(video->isp) == NULL) {
        handle->timeperframe.denominator = 1;
 
        handle->video = video;
-       file->private_data = &handle->vfh;
 
 done:
        if (ret < 0) {
 
        init_adb_hdrs(ctx);
 
        v4l2_fh_init(&ctx->fh, video_devdata(file));
-       file->private_data = &ctx->fh;
 
        hdl = &ctx->hdl;
        v4l2_ctrl_handler_init(hdl, 1);
                goto exit_fh;
        }
 
-       v4l2_fh_add(&ctx->fh);
+       v4l2_fh_add(&ctx->fh, file);
 
        /*
         * for now, just report the creation of the first instance, we can later
 
        }
 
        v4l2_fh_init(&ctx->fh, vdev);
-       filp->private_data = &ctx->fh;
-       v4l2_fh_add(&ctx->fh);
+       v4l2_fh_add(&ctx->fh, filp);
 
        hantro_reset_fmts(ctx);
 
 
                ctx->is_stateless = true;
 
        v4l2_fh_init(&ctx->fh, video_devdata(file));
-       file->private_data = &ctx->fh;
        ctx->dev = dev;
        hdl = &ctx->hdl;
        v4l2_ctrl_handler_init(hdl, 5);
                goto open_unlock;
        }
 
-       v4l2_fh_add(&ctx->fh);
+       v4l2_fh_add(&ctx->fh, file);
 
 open_unlock:
        mutex_unlock(vfd->lock);
 
        }
 
        v4l2_fh_init(&ctx->fh, video_devdata(file));
-       file->private_data = &ctx->fh;
        ctx->dev = dev;
        hdl = &ctx->hdl;
        v4l2_ctrl_handler_init(hdl, 4);
                goto open_unlock;
        }
 
-       v4l2_fh_add(&ctx->fh);
+       v4l2_fh_add(&ctx->fh, file);
        atomic_inc(&dev->num_inst);
 
        dprintk(dev, 1, "Created instance: %p, m2m_ctx: %p\n",
 
        ctx->tpg_str_buf = kzalloc(TPG_STR_BUF_SZ, GFP_KERNEL);
 
        v4l2_fh_init(&ctx->fh, video_devdata(file));
-       file->private_data = &ctx->fh;
        ctx->dev = dev;
 
        rc = visl_init_ctrls(ctx);
        if (rc)
                goto free_m2m_ctx;
 
-       v4l2_fh_add(&ctx->fh);
+       v4l2_fh_add(&ctx->fh, file);
 
        dprintk(dev, "Created instance: %p, m2m_ctx: %p\n",
                ctx, ctx->fh.m2m_ctx);
 
                return -ENOMEM;
        fh->legacy_mode = true;
        v4l2_fh_init(&fh->fh, video_devdata(file));
-       v4l2_fh_add(&fh->fh);
-       file->private_data = &fh->fh;
+       v4l2_fh_add(&fh->fh, file);
        return 0;
 }
 
 
        }
 
        fhp->file = file;
-       file->private_data = &fhp->fh;
 
        fhp->fw_mode_flag = pvr2_hdw_cpufw_get_enabled(hdw);
-       v4l2_fh_add(&fhp->fh);
+       v4l2_fh_add(&fhp->fh, file);
 
        return 0;
 }
 
                return -ENOMEM;
 
        v4l2_fh_init(&handle->vfh, &stream->vdev);
-       v4l2_fh_add(&handle->vfh);
+       v4l2_fh_add(&handle->vfh, file);
        handle->chain = stream->chain;
        handle->stream = stream;
-       file->private_data = &handle->vfh;
 
        return 0;
 }
 
 }
 EXPORT_SYMBOL_GPL(v4l2_fh_init);
 
-void v4l2_fh_add(struct v4l2_fh *fh)
+void v4l2_fh_add(struct v4l2_fh *fh, struct file *filp)
 {
        unsigned long flags;
 
+       filp->private_data = fh;
+
        v4l2_prio_open(fh->vdev->prio, &fh->prio);
        spin_lock_irqsave(&fh->vdev->fh_lock, flags);
        list_add(&fh->list, &fh->vdev->fh_list);
        struct video_device *vdev = video_devdata(filp);
        struct v4l2_fh *fh = kzalloc(sizeof(*fh), GFP_KERNEL);
 
-       filp->private_data = fh;
        if (fh == NULL)
                return -ENOMEM;
        v4l2_fh_init(fh, vdev);
-       v4l2_fh_add(fh);
+       v4l2_fh_add(fh, filp);
        return 0;
 }
 EXPORT_SYMBOL_GPL(v4l2_fh_open);
 
        }
 
        v4l2_fh_init(&subdev_fh->vfh, vdev);
-       v4l2_fh_add(&subdev_fh->vfh);
-       file->private_data = &subdev_fh->vfh;
+       v4l2_fh_add(&subdev_fh->vfh, file);
 
        if (sd->v4l2_dev->mdev && sd->entity.graph_obj.mdev->dev) {
                struct module *owner;
 
        ctx->rot_mode = IPU_ROTATE_NONE;
 
        v4l2_fh_init(&ctx->fh, video_devdata(file));
-       file->private_data = &ctx->fh;
-       v4l2_fh_add(&ctx->fh);
+       v4l2_fh_add(&ctx->fh, file);
        ctx->priv = priv;
 
        ctx->fh.m2m_ctx = v4l2_m2m_ctx_init(priv->m2m_dev, ctx,
 
 
        v4l2_fh_init(&sess->fh, core->vdev_dec);
        sess->fh.ctrl_handler = &sess->ctrl_handler;
-       v4l2_fh_add(&sess->fh);
+       v4l2_fh_add(&sess->fh, file);
        sess->fh.m2m_ctx = sess->m2m_ctx;
-       file->private_data = &sess->fh;
 
        return 0;
 
 
        }
 
        v4l2_fh_init(&ctx->fh, video_devdata(file));
-       file->private_data = &ctx->fh;
        ctx->dev = dev;
        ctx->bit_depth = 8;
 
        if (ret)
                goto err_m2m_release;
 
-       v4l2_fh_add(&ctx->fh);
+       v4l2_fh_add(&ctx->fh, file);
 
        mutex_unlock(&dev->dev_mutex);
 
 
 
        fh->mdev = mdev;
        v4l2_fh_init(&fh->fh, vdev);
-       filp->private_data = &fh->fh;
-
-       v4l2_fh_add(&fh->fh);
+       v4l2_fh_add(&fh->fh, filp);
 
        ret = most_start_channel(mdev->iface, mdev->ch_idx, &comp);
        if (ret) {
 
                return -ENOMEM;
 
        v4l2_fh_init(&handle->vfh, vdev);
-       v4l2_fh_add(&handle->vfh);
+       v4l2_fh_add(&handle->vfh, file);
 
        handle->device = &uvc->video;
-       file->private_data = &handle->vfh;
 
        return 0;
 }
 
  * v4l2_fh_add - Add the fh to the list of file handles on a video_device.
  *
  * @fh: pointer to &struct v4l2_fh
+ * @filp: pointer to &struct file associated with @fh
+ *
+ * The function sets filp->private_data to point to @fh.
  *
  * .. note::
  *    The @fh file handle must be initialised first.
  */
-void v4l2_fh_add(struct v4l2_fh *fh);
+void v4l2_fh_add(struct v4l2_fh *fh, struct file *filp);
 
 /**
  * v4l2_fh_open - Ancillary routine that can be used as the open\(\) op