Commit 04793dd0 authored by Laurent Pinchart's avatar Laurent Pinchart Committed by Mauro Carvalho Chehab

V4L/DVB (8617): uvcvideo: don't use stack-based buffers for USB transfers.

Data buffers on the stack are not allowed for USB I/O. Use dynamically
allocated buffers instead.
Signed-off-by: default avatarBruce Schmid <duck@freescale.com>
Signed-off-by: default avatarLaurent Pinchart <laurent.pinchart@skynet.be>
Signed-off-by: default avatarMauro Carvalho Chehab <mchehab@infradead.org>
parent 835f09c6
...@@ -585,13 +585,17 @@ int uvc_query_v4l2_ctrl(struct uvc_video_device *video, ...@@ -585,13 +585,17 @@ int uvc_query_v4l2_ctrl(struct uvc_video_device *video,
struct uvc_control_mapping *mapping; struct uvc_control_mapping *mapping;
struct uvc_menu_info *menu; struct uvc_menu_info *menu;
unsigned int i; unsigned int i;
__u8 data[8]; __u8 *data;
int ret; int ret;
ctrl = uvc_find_control(video, v4l2_ctrl->id, &mapping); ctrl = uvc_find_control(video, v4l2_ctrl->id, &mapping);
if (ctrl == NULL) if (ctrl == NULL)
return -EINVAL; return -EINVAL;
data = kmalloc(8, GFP_KERNEL);
if (data == NULL)
return -ENOMEM;
memset(v4l2_ctrl, 0, sizeof *v4l2_ctrl); memset(v4l2_ctrl, 0, sizeof *v4l2_ctrl);
v4l2_ctrl->id = mapping->id; v4l2_ctrl->id = mapping->id;
v4l2_ctrl->type = mapping->v4l2_type; v4l2_ctrl->type = mapping->v4l2_type;
...@@ -604,8 +608,8 @@ int uvc_query_v4l2_ctrl(struct uvc_video_device *video, ...@@ -604,8 +608,8 @@ int uvc_query_v4l2_ctrl(struct uvc_video_device *video,
if (ctrl->info->flags & UVC_CONTROL_GET_DEF) { if (ctrl->info->flags & UVC_CONTROL_GET_DEF) {
if ((ret = uvc_query_ctrl(video->dev, GET_DEF, ctrl->entity->id, if ((ret = uvc_query_ctrl(video->dev, GET_DEF, ctrl->entity->id,
video->dev->intfnum, ctrl->info->selector, video->dev->intfnum, ctrl->info->selector,
&data, ctrl->info->size)) < 0) data, ctrl->info->size)) < 0)
return ret; goto out;
v4l2_ctrl->default_value = uvc_get_le_value(data, mapping); v4l2_ctrl->default_value = uvc_get_le_value(data, mapping);
} }
...@@ -623,13 +627,15 @@ int uvc_query_v4l2_ctrl(struct uvc_video_device *video, ...@@ -623,13 +627,15 @@ int uvc_query_v4l2_ctrl(struct uvc_video_device *video,
} }
} }
return 0; ret = 0;
goto out;
case V4L2_CTRL_TYPE_BOOLEAN: case V4L2_CTRL_TYPE_BOOLEAN:
v4l2_ctrl->minimum = 0; v4l2_ctrl->minimum = 0;
v4l2_ctrl->maximum = 1; v4l2_ctrl->maximum = 1;
v4l2_ctrl->step = 1; v4l2_ctrl->step = 1;
return 0; ret = 0;
goto out;
default: default:
break; break;
...@@ -638,26 +644,29 @@ int uvc_query_v4l2_ctrl(struct uvc_video_device *video, ...@@ -638,26 +644,29 @@ int uvc_query_v4l2_ctrl(struct uvc_video_device *video,
if (ctrl->info->flags & UVC_CONTROL_GET_MIN) { if (ctrl->info->flags & UVC_CONTROL_GET_MIN) {
if ((ret = uvc_query_ctrl(video->dev, GET_MIN, ctrl->entity->id, if ((ret = uvc_query_ctrl(video->dev, GET_MIN, ctrl->entity->id,
video->dev->intfnum, ctrl->info->selector, video->dev->intfnum, ctrl->info->selector,
&data, ctrl->info->size)) < 0) data, ctrl->info->size)) < 0)
return ret; goto out;
v4l2_ctrl->minimum = uvc_get_le_value(data, mapping); v4l2_ctrl->minimum = uvc_get_le_value(data, mapping);
} }
if (ctrl->info->flags & UVC_CONTROL_GET_MAX) { if (ctrl->info->flags & UVC_CONTROL_GET_MAX) {
if ((ret = uvc_query_ctrl(video->dev, GET_MAX, ctrl->entity->id, if ((ret = uvc_query_ctrl(video->dev, GET_MAX, ctrl->entity->id,
video->dev->intfnum, ctrl->info->selector, video->dev->intfnum, ctrl->info->selector,
&data, ctrl->info->size)) < 0) data, ctrl->info->size)) < 0)
return ret; goto out;
v4l2_ctrl->maximum = uvc_get_le_value(data, mapping); v4l2_ctrl->maximum = uvc_get_le_value(data, mapping);
} }
if (ctrl->info->flags & UVC_CONTROL_GET_RES) { if (ctrl->info->flags & UVC_CONTROL_GET_RES) {
if ((ret = uvc_query_ctrl(video->dev, GET_RES, ctrl->entity->id, if ((ret = uvc_query_ctrl(video->dev, GET_RES, ctrl->entity->id,
video->dev->intfnum, ctrl->info->selector, video->dev->intfnum, ctrl->info->selector,
&data, ctrl->info->size)) < 0) data, ctrl->info->size)) < 0)
return ret; goto out;
v4l2_ctrl->step = uvc_get_le_value(data, mapping); v4l2_ctrl->step = uvc_get_le_value(data, mapping);
} }
return 0; ret = 0;
out:
kfree(data);
return ret;
} }
......
...@@ -90,17 +90,20 @@ static void uvc_fixup_buffer_size(struct uvc_video_device *video, ...@@ -90,17 +90,20 @@ static void uvc_fixup_buffer_size(struct uvc_video_device *video,
static int uvc_get_video_ctrl(struct uvc_video_device *video, static int uvc_get_video_ctrl(struct uvc_video_device *video,
struct uvc_streaming_control *ctrl, int probe, __u8 query) struct uvc_streaming_control *ctrl, int probe, __u8 query)
{ {
__u8 data[34]; __u8 *data;
__u8 size; __u16 size;
int ret; int ret;
size = video->dev->uvc_version >= 0x0110 ? 34 : 26; size = video->dev->uvc_version >= 0x0110 ? 34 : 26;
data = kmalloc(size, GFP_KERNEL);
if (data == NULL)
return -ENOMEM;
ret = __uvc_query_ctrl(video->dev, query, 0, video->streaming->intfnum, ret = __uvc_query_ctrl(video->dev, query, 0, video->streaming->intfnum,
probe ? VS_PROBE_CONTROL : VS_COMMIT_CONTROL, &data, size, probe ? VS_PROBE_CONTROL : VS_COMMIT_CONTROL, data, size,
UVC_CTRL_STREAMING_TIMEOUT); UVC_CTRL_STREAMING_TIMEOUT);
if (ret < 0) if (ret < 0)
return ret; goto out;
ctrl->bmHint = le16_to_cpup((__le16 *)&data[0]); ctrl->bmHint = le16_to_cpup((__le16 *)&data[0]);
ctrl->bFormatIndex = data[2]; ctrl->bFormatIndex = data[2];
...@@ -136,17 +139,22 @@ static int uvc_get_video_ctrl(struct uvc_video_device *video, ...@@ -136,17 +139,22 @@ static int uvc_get_video_ctrl(struct uvc_video_device *video,
*/ */
uvc_fixup_buffer_size(video, ctrl); uvc_fixup_buffer_size(video, ctrl);
return 0; out:
kfree(data);
return ret;
} }
int uvc_set_video_ctrl(struct uvc_video_device *video, int uvc_set_video_ctrl(struct uvc_video_device *video,
struct uvc_streaming_control *ctrl, int probe) struct uvc_streaming_control *ctrl, int probe)
{ {
__u8 data[34]; __u8 *data;
__u8 size; __u16 size;
int ret;
size = video->dev->uvc_version >= 0x0110 ? 34 : 26; size = video->dev->uvc_version >= 0x0110 ? 34 : 26;
memset(data, 0, sizeof data); data = kzalloc(size, GFP_KERNEL);
if (data == NULL)
return -ENOMEM;
*(__le16 *)&data[0] = cpu_to_le16(ctrl->bmHint); *(__le16 *)&data[0] = cpu_to_le16(ctrl->bmHint);
data[2] = ctrl->bFormatIndex; data[2] = ctrl->bFormatIndex;
...@@ -174,10 +182,13 @@ int uvc_set_video_ctrl(struct uvc_video_device *video, ...@@ -174,10 +182,13 @@ int uvc_set_video_ctrl(struct uvc_video_device *video,
data[33] = ctrl->bMaxVersion; data[33] = ctrl->bMaxVersion;
} }
return __uvc_query_ctrl(video->dev, SET_CUR, 0, ret = __uvc_query_ctrl(video->dev, SET_CUR, 0,
video->streaming->intfnum, video->streaming->intfnum,
probe ? VS_PROBE_CONTROL : VS_COMMIT_CONTROL, &data, size, probe ? VS_PROBE_CONTROL : VS_COMMIT_CONTROL, data, size,
UVC_CTRL_STREAMING_TIMEOUT); UVC_CTRL_STREAMING_TIMEOUT);
kfree(data);
return ret;
} }
int uvc_probe_video(struct uvc_video_device *video, int uvc_probe_video(struct uvc_video_device *video,
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment