EXPORT_SYMBOL(v4l2_subdev_call_wrappers);
 
 #if defined(CONFIG_VIDEO_V4L2_SUBDEV_API)
+
+static struct v4l2_subdev_state *
+subdev_ioctl_get_state(struct v4l2_subdev *sd, struct v4l2_subdev_fh *subdev_fh,
+                      unsigned int cmd, void *arg)
+{
+       u32 which;
+
+       switch (cmd) {
+       default:
+               return NULL;
+       case VIDIOC_SUBDEV_G_FMT:
+       case VIDIOC_SUBDEV_S_FMT:
+               which = ((struct v4l2_subdev_format *)arg)->which;
+               break;
+       case VIDIOC_SUBDEV_G_CROP:
+       case VIDIOC_SUBDEV_S_CROP:
+               which = ((struct v4l2_subdev_crop *)arg)->which;
+               break;
+       case VIDIOC_SUBDEV_ENUM_MBUS_CODE:
+               which = ((struct v4l2_subdev_mbus_code_enum *)arg)->which;
+               break;
+       case VIDIOC_SUBDEV_ENUM_FRAME_SIZE:
+               which = ((struct v4l2_subdev_frame_size_enum *)arg)->which;
+               break;
+       case VIDIOC_SUBDEV_ENUM_FRAME_INTERVAL:
+               which = ((struct v4l2_subdev_frame_interval_enum *)arg)->which;
+               break;
+       case VIDIOC_SUBDEV_G_SELECTION:
+       case VIDIOC_SUBDEV_S_SELECTION:
+               which = ((struct v4l2_subdev_selection *)arg)->which;
+               break;
+       }
+
+       return which == V4L2_SUBDEV_FORMAT_TRY ?
+                            subdev_fh->state :
+                            v4l2_subdev_get_active_state(sd);
+}
+
 static long subdev_do_ioctl(struct file *file, unsigned int cmd, void *arg)
 {
        struct video_device *vdev = video_devdata(file);
        struct v4l2_fh *vfh = file->private_data;
        struct v4l2_subdev_fh *subdev_fh = to_v4l2_subdev_fh(vfh);
        bool ro_subdev = test_bit(V4L2_FL_SUBDEV_RO_DEVNODE, &vdev->flags);
+       struct v4l2_subdev_state *state;
        int rval;
 
+       state = subdev_ioctl_get_state(sd, subdev_fh, cmd, arg);
+
        switch (cmd) {
        case VIDIOC_SUBDEV_QUERYCAP: {
                struct v4l2_subdev_capability *cap = arg;
 
                memset(format->reserved, 0, sizeof(format->reserved));
                memset(format->format.reserved, 0, sizeof(format->format.reserved));
-               return v4l2_subdev_call(sd, pad, get_fmt, subdev_fh->state, format);
+               return v4l2_subdev_call(sd, pad, get_fmt, state, format);
        }
 
        case VIDIOC_SUBDEV_S_FMT: {
 
                memset(format->reserved, 0, sizeof(format->reserved));
                memset(format->format.reserved, 0, sizeof(format->format.reserved));
-               return v4l2_subdev_call(sd, pad, set_fmt, subdev_fh->state, format);
+               return v4l2_subdev_call(sd, pad, set_fmt, state, format);
        }
 
        case VIDIOC_SUBDEV_G_CROP: {
                sel.target = V4L2_SEL_TGT_CROP;
 
                rval = v4l2_subdev_call(
-                       sd, pad, get_selection, subdev_fh->state, &sel);
+                       sd, pad, get_selection, state, &sel);
 
                crop->rect = sel.r;
 
                sel.r = crop->rect;
 
                rval = v4l2_subdev_call(
-                       sd, pad, set_selection, subdev_fh->state, &sel);
+                       sd, pad, set_selection, state, &sel);
 
                crop->rect = sel.r;
 
                struct v4l2_subdev_mbus_code_enum *code = arg;
 
                memset(code->reserved, 0, sizeof(code->reserved));
-               return v4l2_subdev_call(sd, pad, enum_mbus_code, subdev_fh->state,
+               return v4l2_subdev_call(sd, pad, enum_mbus_code, state,
                                        code);
        }
 
                struct v4l2_subdev_frame_size_enum *fse = arg;
 
                memset(fse->reserved, 0, sizeof(fse->reserved));
-               return v4l2_subdev_call(sd, pad, enum_frame_size, subdev_fh->state,
+               return v4l2_subdev_call(sd, pad, enum_frame_size, state,
                                        fse);
        }
 
                struct v4l2_subdev_frame_interval_enum *fie = arg;
 
                memset(fie->reserved, 0, sizeof(fie->reserved));
-               return v4l2_subdev_call(sd, pad, enum_frame_interval, subdev_fh->state,
+               return v4l2_subdev_call(sd, pad, enum_frame_interval, state,
                                        fie);
        }
 
 
                memset(sel->reserved, 0, sizeof(sel->reserved));
                return v4l2_subdev_call(
-                       sd, pad, get_selection, subdev_fh->state, sel);
+                       sd, pad, get_selection, state, sel);
        }
 
        case VIDIOC_SUBDEV_S_SELECTION: {
 
                memset(sel->reserved, 0, sizeof(sel->reserved));
                return v4l2_subdev_call(
-                       sd, pad, set_selection, subdev_fh->state, sel);
+                       sd, pad, set_selection, state, sel);
        }
 
        case VIDIOC_G_EDID: {
        if (is_media_entity_v4l2_subdev(pad->entity)) {
                struct v4l2_subdev *sd =
                        media_entity_to_v4l2_subdev(pad->entity);
+               struct v4l2_subdev_state *state;
+
+               state = v4l2_subdev_get_active_state(sd);
 
                fmt->which = V4L2_SUBDEV_FORMAT_ACTIVE;
                fmt->pad = pad->index;
-               return v4l2_subdev_call(sd, pad, get_fmt, NULL, fmt);
+               return v4l2_subdev_call(sd, pad, get_fmt, state, fmt);
        }
 
        WARN(pad->entity->function != MEDIA_ENT_F_IO_V4L,