return 0;
 }
 
+static int zoran_init_video_device(struct zoran *zr, struct video_device *video_dev, int dir)
+{
+       int err;
+
+       /* Now add the template and register the device unit. */
+       *video_dev = zoran_template;
+       video_dev->v4l2_dev = &zr->v4l2_dev;
+       video_dev->lock = &zr->lock;
+       video_dev->device_caps = V4L2_CAP_STREAMING | dir;
+
+       strscpy(video_dev->name, ZR_DEVNAME(zr), sizeof(video_dev->name));
+       /*
+        * It's not a mem2mem device, but you can both capture and output from one and the same
+        * device. This should really be split up into two device nodes, but that's a job for
+        * another day.
+        */
+       video_dev->vfl_dir = VFL_DIR_M2M;
+       zoran_queue_init(zr, &zr->vq, V4L2_BUF_TYPE_VIDEO_CAPTURE);
+
+       err = video_register_device(video_dev, VFL_TYPE_VIDEO, video_nr[zr->id]);
+       if (err < 0)
+               return err;
+       video_set_drvdata(video_dev, zr);
+       return 0;
+}
+
+static void zoran_exit_video_devices(struct zoran *zr)
+{
+       video_unregister_device(zr->video_dev);
+       kfree(zr->video_dev);
+}
+
+static int zoran_init_video_devices(struct zoran *zr)
+{
+       int err;
+
+       zr->video_dev = video_device_alloc();
+       if (!zr->video_dev)
+               return -ENOMEM;
+
+       err = zoran_init_video_device(zr, zr->video_dev, V4L2_CAP_VIDEO_CAPTURE);
+       if (err)
+               kfree(zr->video_dev);
+       return err;
+}
+
 void zoran_open_init_params(struct zoran *zr)
 {
        int i;
        zoran_open_init_params(zr);
 
        /* allocate memory *before* doing anything to the hardware in case allocation fails */
-       zr->video_dev = video_device_alloc();
-       if (!zr->video_dev) {
-               err = -ENOMEM;
-               goto exit;
-       }
        zr->stat_com = dma_alloc_coherent(&zr->pci_dev->dev,
                                          BUZ_NUM_STAT_COM * sizeof(u32),
                                          &zr->p_sc, GFP_KERNEL);
        if (!zr->stat_com) {
-               err = -ENOMEM;
-               goto exit_video;
+               return -ENOMEM;
        }
        for (j = 0; j < BUZ_NUM_STAT_COM; j++)
                zr->stat_com[j] = cpu_to_le32(1); /* mark as unavailable to zr36057 */
                goto exit_statcom;
        }
 
-       /* Now add the template and register the device unit. */
-       *zr->video_dev = zoran_template;
-       zr->video_dev->v4l2_dev = &zr->v4l2_dev;
-       zr->video_dev->lock = &zr->lock;
-       zr->video_dev->device_caps = V4L2_CAP_STREAMING | V4L2_CAP_VIDEO_CAPTURE;
-
-       strscpy(zr->video_dev->name, ZR_DEVNAME(zr), sizeof(zr->video_dev->name));
-       /*
-        * It's not a mem2mem device, but you can both capture and output from one and the same
-        * device. This should really be split up into two device nodes, but that's a job for
-        * another day.
-        */
-       zr->video_dev->vfl_dir = VFL_DIR_M2M;
-
-       zoran_queue_init(zr, &zr->vq);
-
-       err = video_register_device(zr->video_dev, VFL_TYPE_VIDEO, video_nr[zr->id]);
-       if (err < 0)
+       err = zoran_init_video_devices(zr);
+       if (err)
                goto exit_statcomb;
-       video_set_drvdata(zr->video_dev, zr);
 
        zoran_init_hardware(zr);
        if (!pass_through) {
        dma_free_coherent(&zr->pci_dev->dev, BUZ_NUM_STAT_COM * sizeof(u32) * 2, zr->stat_comb, zr->p_scb);
 exit_statcom:
        dma_free_coherent(&zr->pci_dev->dev, BUZ_NUM_STAT_COM * sizeof(u32), zr->stat_com, zr->p_sc);
-exit_video:
-       kfree(zr->video_dev);
-exit:
        return err;
 }
 
        dma_free_coherent(&zr->pci_dev->dev, BUZ_NUM_STAT_COM * sizeof(u32) * 2, zr->stat_comb, zr->p_scb);
        pci_release_regions(pdev);
        pci_disable_device(zr->pci_dev);
-       video_unregister_device(zr->video_dev);
+       zoran_exit_video_devices(zr);
 exit_free:
        v4l2_ctrl_handler_free(&zr->hdl);
        v4l2_device_unregister(&zr->v4l2_dev);