aboutsummaryrefslogtreecommitdiffstats
path: root/drivers/media/video/em28xx/em28xx-video.c
diff options
context:
space:
mode:
Diffstat (limited to 'drivers/media/video/em28xx/em28xx-video.c')
-rw-r--r--drivers/media/video/em28xx/em28xx-video.c869
1 files changed, 427 insertions, 442 deletions
diff --git a/drivers/media/video/em28xx/em28xx-video.c b/drivers/media/video/em28xx/em28xx-video.c
index d8d02c44c9d2..1b0e10d5edd4 100644
--- a/drivers/media/video/em28xx/em28xx-video.c
+++ b/drivers/media/video/em28xx/em28xx-video.c
@@ -1012,19 +1012,15 @@ static int em28xx_set_fmt(struct em28xx *dev, unsigned int cmd, struct v4l2_form
1012 width /= 2; 1012 width /= 2;
1013 } 1013 }
1014 1014
1015 if ((hscale = 1015 if ((hscale = (((unsigned long)maxw) << 12) / width - 4096L) >= 0x4000)
1016 (((unsigned long)maxw) << 12) / width - 4096L) >=
1017 0x4000)
1018 hscale = 0x3fff; 1016 hscale = 0x3fff;
1019 width =
1020 (((unsigned long)maxw) << 12) / (hscale + 4096L);
1021 1017
1022 if ((vscale = 1018 width = (((unsigned long)maxw) << 12) / (hscale + 4096L);
1023 (((unsigned long)maxh) << 12) / height - 4096L) >= 1019
1024 0x4000) 1020 if ((vscale = (((unsigned long)maxh) << 12) / height - 4096L) >= 0x4000)
1025 vscale = 0x3fff; 1021 vscale = 0x3fff;
1026 height = 1022
1027 (((unsigned long)maxh) << 12) / (vscale + 4096L); 1023 height = (((unsigned long)maxh) << 12) / (vscale + 4096L);
1028 1024
1029 format->fmt.pix.width = width; 1025 format->fmt.pix.width = width;
1030 format->fmt.pix.height = height; 1026 format->fmt.pix.height = height;
@@ -1035,10 +1031,9 @@ static int em28xx_set_fmt(struct em28xx *dev, unsigned int cmd, struct v4l2_form
1035 format->fmt.pix.field = V4L2_FIELD_INTERLACED; 1031 format->fmt.pix.field = V4L2_FIELD_INTERLACED;
1036 1032
1037 em28xx_videodbg("%s: returned %dx%d (%d, %d)\n", 1033 em28xx_videodbg("%s: returned %dx%d (%d, %d)\n",
1038 cmd == 1034 cmd == VIDIOC_TRY_FMT ?
1039 VIDIOC_TRY_FMT ? "VIDIOC_TRY_FMT" : 1035 "VIDIOC_TRY_FMT" :"VIDIOC_S_FMT",
1040 "VIDIOC_S_FMT", format->fmt.pix.width, 1036 format->fmt.pix.width, format->fmt.pix.height, hscale, vscale);
1041 format->fmt.pix.height, hscale, vscale);
1042 1037
1043 if (cmd == VIDIOC_TRY_FMT) 1038 if (cmd == VIDIOC_TRY_FMT)
1044 return 0; 1039 return 0;
@@ -1064,7 +1059,7 @@ static int em28xx_set_fmt(struct em28xx *dev, unsigned int cmd, struct v4l2_form
1064 dev->width = width; 1059 dev->width = width;
1065 dev->height = height; 1060 dev->height = height;
1066 dev->frame_size = dev->width * dev->height * 2; 1061 dev->frame_size = dev->width * dev->height * 2;
1067 dev->field_size = dev->frame_size >> 1; /*both_fileds ? dev->frame_size>>1 : dev->frame_size; */ 1062 dev->field_size = dev->frame_size >> 1;
1068 dev->bytesperline = dev->width * 2; 1063 dev->bytesperline = dev->width * 2;
1069 dev->hscale = hscale; 1064 dev->hscale = hscale;
1070 dev->vscale = vscale; 1065 dev->vscale = vscale;
@@ -1092,374 +1087,367 @@ static int em28xx_do_ioctl(struct inode *inode, struct file *filp,
1092 switch (cmd) { 1087 switch (cmd) {
1093 /* ---------- tv norms ---------- */ 1088 /* ---------- tv norms ---------- */
1094 case VIDIOC_ENUMSTD: 1089 case VIDIOC_ENUMSTD:
1095 { 1090 {
1096 struct v4l2_standard *e = arg; 1091 struct v4l2_standard *e = arg;
1097 unsigned int i; 1092 unsigned int i;
1098 1093
1099 i = e->index; 1094 i = e->index;
1100 if (i >= TVNORMS) 1095 if (i >= TVNORMS)
1101 return -EINVAL; 1096 return -EINVAL;
1102 ret = v4l2_video_std_construct(e, tvnorms[e->index].id, 1097 ret = v4l2_video_std_construct(e, tvnorms[e->index].id,
1103 tvnorms[e->index].name); 1098 tvnorms[e->index].name);
1104 e->index = i; 1099 e->index = i;
1105 if (ret < 0) 1100 if (ret < 0)
1106 return ret; 1101 return ret;
1107 return 0; 1102 return 0;
1108 } 1103 }
1109 case VIDIOC_G_STD: 1104 case VIDIOC_G_STD:
1110 { 1105 {
1111 v4l2_std_id *id = arg; 1106 v4l2_std_id *id = arg;
1112 1107
1113 *id = dev->tvnorm->id; 1108 *id = dev->tvnorm->id;
1114 return 0; 1109 return 0;
1115 } 1110 }
1116 case VIDIOC_S_STD: 1111 case VIDIOC_S_STD:
1117 { 1112 {
1118 v4l2_std_id *id = arg; 1113 v4l2_std_id *id = arg;
1119 unsigned int i; 1114 unsigned int i;
1120 1115
1116 for (i = 0; i < TVNORMS; i++)
1117 if (*id == tvnorms[i].id)
1118 break;
1119 if (i == TVNORMS)
1121 for (i = 0; i < TVNORMS; i++) 1120 for (i = 0; i < TVNORMS; i++)
1122 if (*id == tvnorms[i].id) 1121 if (*id & tvnorms[i].id)
1123 break; 1122 break;
1124 if (i == TVNORMS) 1123 if (i == TVNORMS)
1125 for (i = 0; i < TVNORMS; i++) 1124 return -EINVAL;
1126 if (*id & tvnorms[i].id)
1127 break;
1128 if (i == TVNORMS)
1129 return -EINVAL;
1130 1125
1131 down(&dev->lock); 1126 down(&dev->lock);
1132 dev->tvnorm = &tvnorms[i]; 1127 dev->tvnorm = &tvnorms[i];
1133 1128
1134 em28xx_set_norm(dev, dev->width, dev->height); 1129 em28xx_set_norm(dev, dev->width, dev->height);
1135 1130
1136 em28xx_i2c_call_clients(dev, DECODER_SET_NORM, 1131 em28xx_i2c_call_clients(dev, DECODER_SET_NORM,
1137 &tvnorms[i].mode); 1132 &tvnorms[i].mode);
1138 em28xx_i2c_call_clients(dev, VIDIOC_S_STD, 1133 em28xx_i2c_call_clients(dev, VIDIOC_S_STD,
1139 &dev->tvnorm->id); 1134 &dev->tvnorm->id);
1140 1135
1141 up(&dev->lock); 1136 up(&dev->lock);
1142 1137
1143 return 0; 1138 return 0;
1144 } 1139 }
1145 1140
1146 /* ------ input switching ---------- */ 1141 /* ------ input switching ---------- */
1147 case VIDIOC_ENUMINPUT: 1142 case VIDIOC_ENUMINPUT:
1148 { 1143 {
1149 struct v4l2_input *i = arg; 1144 struct v4l2_input *i = arg;
1150 unsigned int n; 1145 unsigned int n;
1151 static const char *iname[] = { 1146 static const char *iname[] = {
1152 [EM28XX_VMUX_COMPOSITE1] = "Composite1", 1147 [EM28XX_VMUX_COMPOSITE1] = "Composite1",
1153 [EM28XX_VMUX_COMPOSITE2] = "Composite2", 1148 [EM28XX_VMUX_COMPOSITE2] = "Composite2",
1154 [EM28XX_VMUX_COMPOSITE3] = "Composite3", 1149 [EM28XX_VMUX_COMPOSITE3] = "Composite3",
1155 [EM28XX_VMUX_COMPOSITE4] = "Composite4", 1150 [EM28XX_VMUX_COMPOSITE4] = "Composite4",
1156 [EM28XX_VMUX_SVIDEO] = "S-Video", 1151 [EM28XX_VMUX_SVIDEO] = "S-Video",
1157 [EM28XX_VMUX_TELEVISION] = "Television", 1152 [EM28XX_VMUX_TELEVISION] = "Television",
1158 [EM28XX_VMUX_CABLE] = "Cable TV", 1153 [EM28XX_VMUX_CABLE] = "Cable TV",
1159 [EM28XX_VMUX_DVB] = "DVB", 1154 [EM28XX_VMUX_DVB] = "DVB",
1160 [EM28XX_VMUX_DEBUG] = "for debug only", 1155 [EM28XX_VMUX_DEBUG] = "for debug only",
1161 }; 1156 };
1162 1157
1163 n = i->index; 1158 n = i->index;
1164 if (n >= MAX_EM28XX_INPUT) 1159 if (n >= MAX_EM28XX_INPUT)
1165 return -EINVAL; 1160 return -EINVAL;
1166 if (0 == INPUT(n)->type) 1161 if (0 == INPUT(n)->type)
1167 return -EINVAL; 1162 return -EINVAL;
1168 memset(i, 0, sizeof(*i)); 1163 memset(i, 0, sizeof(*i));
1169 i->index = n; 1164 i->index = n;
1170 i->type = V4L2_INPUT_TYPE_CAMERA; 1165 i->type = V4L2_INPUT_TYPE_CAMERA;
1171 strcpy(i->name, iname[INPUT(n)->type]); 1166 strcpy(i->name, iname[INPUT(n)->type]);
1172 if ((EM28XX_VMUX_TELEVISION == INPUT(n)->type) || 1167 if ((EM28XX_VMUX_TELEVISION == INPUT(n)->type) ||
1173 (EM28XX_VMUX_CABLE == INPUT(n)->type)) 1168 (EM28XX_VMUX_CABLE == INPUT(n)->type))
1174 i->type = V4L2_INPUT_TYPE_TUNER; 1169 i->type = V4L2_INPUT_TYPE_TUNER;
1175 for (n = 0; n < ARRAY_SIZE(tvnorms); n++) 1170 for (n = 0; n < ARRAY_SIZE(tvnorms); n++)
1176 i->std |= tvnorms[n].id; 1171 i->std |= tvnorms[n].id;
1177 return 0; 1172 return 0;
1178 } 1173 }
1179
1180 case VIDIOC_G_INPUT: 1174 case VIDIOC_G_INPUT:
1181 { 1175 {
1182 int *i = arg; 1176 int *i = arg;
1183 *i = dev->ctl_input; 1177 *i = dev->ctl_input;
1184
1185 return 0;
1186 }
1187 1178
1179 return 0;
1180 }
1188 case VIDIOC_S_INPUT: 1181 case VIDIOC_S_INPUT:
1189 { 1182 {
1190 int *index = arg; 1183 int *index = arg;
1191
1192 if (*index >= MAX_EM28XX_INPUT)
1193 return -EINVAL;
1194 if (0 == INPUT(*index)->type)
1195 return -EINVAL;
1196 1184
1197 down(&dev->lock); 1185 if (*index >= MAX_EM28XX_INPUT)
1198 video_mux(dev, *index); 1186 return -EINVAL;
1199 up(&dev->lock); 1187 if (0 == INPUT(*index)->type)
1188 return -EINVAL;
1200 1189
1201 return 0; 1190 down(&dev->lock);
1202 } 1191 video_mux(dev, *index);
1192 up(&dev->lock);
1203 1193
1194 return 0;
1195 }
1204 case VIDIOC_G_AUDIO: 1196 case VIDIOC_G_AUDIO:
1205 { 1197 {
1206 struct v4l2_audio *a = arg; 1198 struct v4l2_audio *a = arg;
1207 unsigned int index = a->index; 1199 unsigned int index = a->index;
1208 1200
1209 if (a->index > 1) 1201 if (a->index > 1)
1210 return -EINVAL; 1202 return -EINVAL;
1211 memset(a, 0, sizeof(*a)); 1203 memset(a, 0, sizeof(*a));
1212 index = dev->ctl_ainput; 1204 index = dev->ctl_ainput;
1213 1205
1214 if (index == 0) { 1206 if (index == 0) {
1215 strcpy(a->name, "Television"); 1207 strcpy(a->name, "Television");
1216 } else { 1208 } else {
1217 strcpy(a->name, "Line In"); 1209 strcpy(a->name, "Line In");
1218 }
1219 a->capability = V4L2_AUDCAP_STEREO;
1220 a->index = index;
1221 return 0;
1222 } 1210 }
1223 1211 a->capability = V4L2_AUDCAP_STEREO;
1212 a->index = index;
1213 return 0;
1214 }
1224 case VIDIOC_S_AUDIO: 1215 case VIDIOC_S_AUDIO:
1225 { 1216 {
1226 struct v4l2_audio *a = arg; 1217 struct v4l2_audio *a = arg;
1227 if (a->index != dev->ctl_ainput)
1228 return -EINVAL;
1229 1218
1230 return 0; 1219 if (a->index != dev->ctl_ainput)
1231 } 1220 return -EINVAL;
1221
1222 return 0;
1223 }
1232 1224
1233 /* --- controls ---------------------------------------------- */ 1225 /* --- controls ---------------------------------------------- */
1234 case VIDIOC_QUERYCTRL: 1226 case VIDIOC_QUERYCTRL:
1235 { 1227 {
1236 struct v4l2_queryctrl *qc = arg; 1228 struct v4l2_queryctrl *qc = arg;
1237 int i, id=qc->id; 1229 int i, id=qc->id;
1238 1230
1239 memset(qc,0,sizeof(*qc)); 1231 memset(qc,0,sizeof(*qc));
1240 qc->id=id; 1232 qc->id=id;
1241 1233
1242 if (!dev->has_msp34xx) { 1234 if (!dev->has_msp34xx) {
1243 for (i = 0; i < ARRAY_SIZE(em28xx_qctrl); i++) { 1235 for (i = 0; i < ARRAY_SIZE(em28xx_qctrl); i++) {
1244 if (qc->id && qc->id == em28xx_qctrl[i].id) { 1236 if (qc->id && qc->id == em28xx_qctrl[i].id) {
1245 memcpy(qc, &(em28xx_qctrl[i]), 1237 memcpy(qc, &(em28xx_qctrl[i]),
1246 sizeof(*qc)); 1238 sizeof(*qc));
1247 return 0;
1248 }
1249 }
1250 }
1251 if (dev->decoder == EM28XX_TVP5150) {
1252 em28xx_i2c_call_clients(dev,cmd,qc);
1253 if (qc->type)
1254 return 0;
1255 else
1256 return -EINVAL;
1257 }
1258 for (i = 0; i < ARRAY_SIZE(saa711x_qctrl); i++) {
1259 if (qc->id && qc->id == saa711x_qctrl[i].id) {
1260 memcpy(qc, &(saa711x_qctrl[i]),
1261 sizeof(*qc));
1262 return 0; 1239 return 0;
1263 } 1240 }
1264 } 1241 }
1265 1242 }
1266 return -EINVAL; 1243 if (dev->decoder == EM28XX_TVP5150) {
1244 em28xx_i2c_call_clients(dev,cmd,qc);
1245 if (qc->type)
1246 return 0;
1247 else
1248 return -EINVAL;
1249 }
1250 for (i = 0; i < ARRAY_SIZE(saa711x_qctrl); i++) {
1251 if (qc->id && qc->id == saa711x_qctrl[i].id) {
1252 memcpy(qc, &(saa711x_qctrl[i]),
1253 sizeof(*qc));
1254 return 0;
1255 }
1267 } 1256 }
1268 1257
1258 return -EINVAL;
1259 }
1269 case VIDIOC_G_CTRL: 1260 case VIDIOC_G_CTRL:
1270 { 1261 {
1271 struct v4l2_control *ctrl = arg; 1262 struct v4l2_control *ctrl = arg;
1272 int retval=-EINVAL; 1263 int retval=-EINVAL;
1273
1274 if (!dev->has_msp34xx)
1275 retval=em28xx_get_ctrl(dev, ctrl);
1276 if (retval==-EINVAL) {
1277 if (dev->decoder == EM28XX_TVP5150) {
1278 em28xx_i2c_call_clients(dev,cmd,arg);
1279 return 0;
1280 }
1281 1264
1282 return saa711x_get_ctrl(dev, ctrl); 1265 if (!dev->has_msp34xx)
1283 } else return retval; 1266 retval=em28xx_get_ctrl(dev, ctrl);
1284 } 1267 if (retval==-EINVAL) {
1268 if (dev->decoder == EM28XX_TVP5150) {
1269 em28xx_i2c_call_clients(dev,cmd,arg);
1270 return 0;
1271 }
1285 1272
1273 return saa711x_get_ctrl(dev, ctrl);
1274 } else return retval;
1275 }
1286 case VIDIOC_S_CTRL: 1276 case VIDIOC_S_CTRL:
1287 { 1277 {
1288 struct v4l2_control *ctrl = arg; 1278 struct v4l2_control *ctrl = arg;
1289 u8 i; 1279 u8 i;
1290 1280
1291 if (!dev->has_msp34xx){ 1281 if (!dev->has_msp34xx){
1292 for (i = 0; i < ARRAY_SIZE(em28xx_qctrl); i++) { 1282 for (i = 0; i < ARRAY_SIZE(em28xx_qctrl); i++) {
1293 if (ctrl->id == em28xx_qctrl[i].id) { 1283 if (ctrl->id == em28xx_qctrl[i].id) {
1294 if (ctrl->value < 1284 if (ctrl->value <
1295 em28xx_qctrl[i].minimum 1285 em28xx_qctrl[i].minimum
1296 || ctrl->value > 1286 || ctrl->value >
1297 em28xx_qctrl[i].maximum) 1287 em28xx_qctrl[i].maximum)
1298 return -ERANGE; 1288 return -ERANGE;
1299 return em28xx_set_ctrl(dev, ctrl); 1289 return em28xx_set_ctrl(dev, ctrl);
1300 }
1301 } 1290 }
1302 } 1291 }
1292 }
1303 1293
1304 if (dev->decoder == EM28XX_TVP5150) { 1294 if (dev->decoder == EM28XX_TVP5150) {
1305 em28xx_i2c_call_clients(dev,cmd,arg); 1295 em28xx_i2c_call_clients(dev,cmd,arg);
1306 return 0; 1296 return 0;
1307 } else if (!dev->has_msp34xx) { 1297 } else if (!dev->has_msp34xx) {
1308 for (i = 0; i < ARRAY_SIZE(em28xx_qctrl); i++) { 1298 for (i = 0; i < ARRAY_SIZE(em28xx_qctrl); i++) {
1309 if (ctrl->id == em28xx_qctrl[i].id) { 1299 if (ctrl->id == em28xx_qctrl[i].id) {
1310 if (ctrl->value < 1300 if (ctrl->value <
1311 em28xx_qctrl[i].minimum 1301 em28xx_qctrl[i].minimum
1312 || ctrl->value > 1302 || ctrl->value >
1313 em28xx_qctrl[i].maximum) 1303 em28xx_qctrl[i].maximum)
1314 return -ERANGE; 1304 return -ERANGE;
1315 return em28xx_set_ctrl(dev, ctrl); 1305 return em28xx_set_ctrl(dev, ctrl);
1316 }
1317 } 1306 }
1318 for (i = 0; i < ARRAY_SIZE(saa711x_qctrl); i++) { 1307 }
1319 if (ctrl->id == saa711x_qctrl[i].id) { 1308 for (i = 0; i < ARRAY_SIZE(saa711x_qctrl); i++) {
1320 if (ctrl->value < 1309 if (ctrl->id == saa711x_qctrl[i].id) {
1321 saa711x_qctrl[i].minimum 1310 if (ctrl->value <
1322 || ctrl->value > 1311 saa711x_qctrl[i].minimum
1323 saa711x_qctrl[i].maximum) 1312 || ctrl->value >
1324 return -ERANGE; 1313 saa711x_qctrl[i].maximum)
1325 return saa711x_set_ctrl(dev, ctrl); 1314 return -ERANGE;
1326 } 1315 return saa711x_set_ctrl(dev, ctrl);
1327 } 1316 }
1328 } 1317 }
1329
1330 return -EINVAL;
1331 } 1318 }
1332 1319
1333 /* --- tuner ioctls ------------------------------------------ */ 1320 return -EINVAL;
1321 }
1322 /* --- tuner ioctls ------------------------------------------ */
1334 case VIDIOC_G_TUNER: 1323 case VIDIOC_G_TUNER:
1335 { 1324 {
1336 struct v4l2_tuner *t = arg; 1325 struct v4l2_tuner *t = arg;
1337 int status = 0; 1326 int status = 0;
1338 1327
1339 if (0 != t->index) 1328 if (0 != t->index)
1340 return -EINVAL; 1329 return -EINVAL;
1341 1330
1342 memset(t, 0, sizeof(*t)); 1331 memset(t, 0, sizeof(*t));
1343 strcpy(t->name, "Tuner"); 1332 strcpy(t->name, "Tuner");
1344 t->type = V4L2_TUNER_ANALOG_TV; 1333 t->type = V4L2_TUNER_ANALOG_TV;
1345 t->capability = V4L2_TUNER_CAP_NORM; 1334 t->capability = V4L2_TUNER_CAP_NORM;
1346 t->rangehigh = 0xffffffffUL; /* FIXME: set correct range */ 1335 t->rangehigh = 0xffffffffUL; /* FIXME: set correct range */
1347/* t->signal = 0xffff;*/ 1336/* t->signal = 0xffff;*/
1348/* em28xx_i2c_call_clients(dev,VIDIOC_G_TUNER,t);*/ 1337/* em28xx_i2c_call_clients(dev,VIDIOC_G_TUNER,t);*/
1349 /* No way to get signal strength? */ 1338 /* No way to get signal strength? */
1350 down(&dev->lock); 1339 down(&dev->lock);
1351 em28xx_i2c_call_clients(dev, DECODER_GET_STATUS, 1340 em28xx_i2c_call_clients(dev, DECODER_GET_STATUS,
1352 &status); 1341 &status);
1353 up(&dev->lock); 1342 up(&dev->lock);
1354 t->signal = 1343 t->signal =
1355 (status & DECODER_STATUS_GOOD) != 0 ? 0xffff : 0; 1344 (status & DECODER_STATUS_GOOD) != 0 ? 0xffff : 0;
1356 1345
1357 em28xx_videodbg("VIDIO_G_TUNER: signal=%x, afc=%x\n", t->signal, 1346 em28xx_videodbg("VIDIO_G_TUNER: signal=%x, afc=%x\n", t->signal,
1358 t->afc); 1347 t->afc);
1359 return 0; 1348 return 0;
1360 } 1349 }
1361 case VIDIOC_S_TUNER: 1350 case VIDIOC_S_TUNER:
1362 { 1351 {
1363 struct v4l2_tuner *t = arg; 1352 struct v4l2_tuner *t = arg;
1364 int status = 0; 1353 int status = 0;
1365 1354
1366 if (0 != t->index) 1355 if (0 != t->index)
1367 return -EINVAL; 1356 return -EINVAL;
1368 memset(t, 0, sizeof(*t)); 1357 memset(t, 0, sizeof(*t));
1369 strcpy(t->name, "Tuner"); 1358 strcpy(t->name, "Tuner");
1370 t->type = V4L2_TUNER_ANALOG_TV; 1359 t->type = V4L2_TUNER_ANALOG_TV;
1371 t->capability = V4L2_TUNER_CAP_NORM; 1360 t->capability = V4L2_TUNER_CAP_NORM;
1372 t->rangehigh = 0xffffffffUL; /* FIXME: set correct range */ 1361 t->rangehigh = 0xffffffffUL; /* FIXME: set correct range */
1373/* t->signal = 0xffff; */ 1362/* t->signal = 0xffff; */
1374 /* No way to get signal strength? */ 1363 /* No way to get signal strength? */
1375 down(&dev->lock); 1364 down(&dev->lock);
1376 em28xx_i2c_call_clients(dev, DECODER_GET_STATUS, 1365 em28xx_i2c_call_clients(dev, DECODER_GET_STATUS,
1377 &status); 1366 &status);
1378 up(&dev->lock); 1367 up(&dev->lock);
1379 t->signal = 1368 t->signal =
1380 (status & DECODER_STATUS_GOOD) != 0 ? 0xffff : 0; 1369 (status & DECODER_STATUS_GOOD) != 0 ? 0xffff : 0;
1381 1370
1382 em28xx_videodbg("VIDIO_S_TUNER: signal=%x, afc=%x\n", 1371 em28xx_videodbg("VIDIO_S_TUNER: signal=%x, afc=%x\n",
1383 t->signal, t->afc); 1372 t->signal, t->afc);
1384 return 0; 1373 return 0;
1385 } 1374 }
1386 case VIDIOC_G_FREQUENCY: 1375 case VIDIOC_G_FREQUENCY:
1387 { 1376 {
1388 struct v4l2_frequency *f = arg; 1377 struct v4l2_frequency *f = arg;
1389 1378
1390 memset(f, 0, sizeof(*f)); 1379 memset(f, 0, sizeof(*f));
1391 f->type = V4L2_TUNER_ANALOG_TV; 1380 f->type = V4L2_TUNER_ANALOG_TV;
1392 f->frequency = dev->ctl_freq; 1381 f->frequency = dev->ctl_freq;
1393 1382
1394 return 0; 1383 return 0;
1395 } 1384 }
1396 case VIDIOC_S_FREQUENCY: 1385 case VIDIOC_S_FREQUENCY:
1397 { 1386 {
1398 struct v4l2_frequency *f = arg; 1387 struct v4l2_frequency *f = arg;
1399
1400 if (0 != f->tuner)
1401 return -EINVAL;
1402 1388
1403 if (V4L2_TUNER_ANALOG_TV != f->type) 1389 if (0 != f->tuner)
1404 return -EINVAL; 1390 return -EINVAL;
1405 1391
1406 down(&dev->lock); 1392 if (V4L2_TUNER_ANALOG_TV != f->type)
1407 dev->ctl_freq = f->frequency; 1393 return -EINVAL;
1408 em28xx_i2c_call_clients(dev, VIDIOC_S_FREQUENCY, f);
1409 up(&dev->lock);
1410 return 0;
1411 }
1412 1394
1395 down(&dev->lock);
1396 dev->ctl_freq = f->frequency;
1397 em28xx_i2c_call_clients(dev, VIDIOC_S_FREQUENCY, f);
1398 up(&dev->lock);
1399 return 0;
1400 }
1413 case VIDIOC_CROPCAP: 1401 case VIDIOC_CROPCAP:
1414 { 1402 {
1415 struct v4l2_cropcap *cc = arg; 1403 struct v4l2_cropcap *cc = arg;
1416 1404
1417 if (cc->type != V4L2_BUF_TYPE_VIDEO_CAPTURE) 1405 if (cc->type != V4L2_BUF_TYPE_VIDEO_CAPTURE)
1418 return -EINVAL; 1406 return -EINVAL;
1419 cc->bounds.left = 0; 1407 cc->bounds.left = 0;
1420 cc->bounds.top = 0; 1408 cc->bounds.top = 0;
1421 cc->bounds.width = dev->width; 1409 cc->bounds.width = dev->width;
1422 cc->bounds.height = dev->height; 1410 cc->bounds.height = dev->height;
1423 cc->defrect = cc->bounds; 1411 cc->defrect = cc->bounds;
1424 cc->pixelaspect.numerator = 54; /* 4:3 FIXME: remove magic numbers */ 1412 cc->pixelaspect.numerator = 54; /* 4:3 FIXME: remove magic numbers */
1425 cc->pixelaspect.denominator = 59; 1413 cc->pixelaspect.denominator = 59;
1426 return 0; 1414 return 0;
1427 } 1415 }
1428 case VIDIOC_STREAMON: 1416 case VIDIOC_STREAMON:
1429 { 1417 {
1430 int *type = arg; 1418 int *type = arg;
1431 1419
1432 if (*type != V4L2_BUF_TYPE_VIDEO_CAPTURE 1420 if (*type != V4L2_BUF_TYPE_VIDEO_CAPTURE
1433 || dev->io != IO_MMAP) 1421 || dev->io != IO_MMAP)
1434 return -EINVAL; 1422 return -EINVAL;
1435 1423
1436 if (list_empty(&dev->inqueue)) 1424 if (list_empty(&dev->inqueue))
1437 return -EINVAL; 1425 return -EINVAL;
1438 1426
1439 dev->stream = STREAM_ON; /* FIXME: Start video capture here? */ 1427 dev->stream = STREAM_ON; /* FIXME: Start video capture here? */
1440 1428
1441 em28xx_videodbg("VIDIOC_STREAMON: starting stream\n"); 1429 em28xx_videodbg("VIDIOC_STREAMON: starting stream\n");
1442 1430
1443 return 0; 1431 return 0;
1444 } 1432 }
1445 case VIDIOC_STREAMOFF: 1433 case VIDIOC_STREAMOFF:
1446 { 1434 {
1447 int *type = arg; 1435 int *type = arg;
1448 int ret; 1436 int ret;
1449
1450 if (*type != V4L2_BUF_TYPE_VIDEO_CAPTURE
1451 || dev->io != IO_MMAP)
1452 return -EINVAL;
1453 1437
1454 if (dev->stream == STREAM_ON) { 1438 if (*type != V4L2_BUF_TYPE_VIDEO_CAPTURE
1455 em28xx_videodbg ("VIDIOC_STREAMOFF: interrupting stream\n"); 1439 || dev->io != IO_MMAP)
1456 if ((ret = em28xx_stream_interrupt(dev))) 1440 return -EINVAL;
1457 return ret;
1458 }
1459 em28xx_empty_framequeues(dev);
1460 1441
1461 return 0; 1442 if (dev->stream == STREAM_ON) {
1443 em28xx_videodbg ("VIDIOC_STREAMOFF: interrupting stream\n");
1444 if ((ret = em28xx_stream_interrupt(dev)))
1445 return ret;
1462 } 1446 }
1447 em28xx_empty_framequeues(dev);
1448
1449 return 0;
1450 }
1463 default: 1451 default:
1464 return v4l_compat_translate_ioctl(inode, filp, cmd, arg, 1452 return v4l_compat_translate_ioctl(inode, filp, cmd, arg,
1465 driver_ioctl); 1453 driver_ioctl);
@@ -1489,40 +1477,38 @@ static int em28xx_video_do_ioctl(struct inode *inode, struct file *filp,
1489 /* --- capabilities ------------------------------------------ */ 1477 /* --- capabilities ------------------------------------------ */
1490 case VIDIOC_QUERYCAP: 1478 case VIDIOC_QUERYCAP:
1491 { 1479 {
1492 struct v4l2_capability *cap = arg; 1480 struct v4l2_capability *cap = arg;
1493 1481
1494 memset(cap, 0, sizeof(*cap)); 1482 memset(cap, 0, sizeof(*cap));
1495 strlcpy(cap->driver, "em28xx", sizeof(cap->driver)); 1483 strlcpy(cap->driver, "em28xx", sizeof(cap->driver));
1496 strlcpy(cap->card, em28xx_boards[dev->model].name, 1484 strlcpy(cap->card, em28xx_boards[dev->model].name,
1497 sizeof(cap->card)); 1485 sizeof(cap->card));
1498 strlcpy(cap->bus_info, dev->udev->dev.bus_id, 1486 strlcpy(cap->bus_info, dev->udev->dev.bus_id,
1499 sizeof(cap->bus_info)); 1487 sizeof(cap->bus_info));
1500 cap->version = EM28XX_VERSION_CODE; 1488 cap->version = EM28XX_VERSION_CODE;
1501 cap->capabilities = 1489 cap->capabilities =
1502 V4L2_CAP_SLICED_VBI_CAPTURE | 1490 V4L2_CAP_SLICED_VBI_CAPTURE |
1503 V4L2_CAP_VIDEO_CAPTURE | 1491 V4L2_CAP_VIDEO_CAPTURE |
1504 V4L2_CAP_AUDIO | 1492 V4L2_CAP_AUDIO |
1505 V4L2_CAP_READWRITE | V4L2_CAP_STREAMING; 1493 V4L2_CAP_READWRITE | V4L2_CAP_STREAMING;
1506 if (dev->has_tuner) 1494 if (dev->has_tuner)
1507 cap->capabilities |= V4L2_CAP_TUNER; 1495 cap->capabilities |= V4L2_CAP_TUNER;
1508 return 0; 1496 return 0;
1509 } 1497 }
1510 1498 /* --- capture ioctls ---------------------------------------- */
1511 /* --- capture ioctls ---------------------------------------- */
1512 case VIDIOC_ENUM_FMT: 1499 case VIDIOC_ENUM_FMT:
1513 { 1500 {
1514 struct v4l2_fmtdesc *fmtd = arg; 1501 struct v4l2_fmtdesc *fmtd = arg;
1515
1516 if (fmtd->index != 0)
1517 return -EINVAL;
1518 memset(fmtd, 0, sizeof(*fmtd));
1519 fmtd->type = V4L2_BUF_TYPE_VIDEO_CAPTURE;
1520 strcpy(fmtd->description, "Packed YUY2");
1521 fmtd->pixelformat = V4L2_PIX_FMT_YUYV;
1522 memset(fmtd->reserved, 0, sizeof(fmtd->reserved));
1523 return 0;
1524 }
1525 1502
1503 if (fmtd->index != 0)
1504 return -EINVAL;
1505 memset(fmtd, 0, sizeof(*fmtd));
1506 fmtd->type = V4L2_BUF_TYPE_VIDEO_CAPTURE;
1507 strcpy(fmtd->description, "Packed YUY2");
1508 fmtd->pixelformat = V4L2_PIX_FMT_YUYV;
1509 memset(fmtd->reserved, 0, sizeof(fmtd->reserved));
1510 return 0;
1511 }
1526 case VIDIOC_G_FMT: 1512 case VIDIOC_G_FMT:
1527 return em28xx_get_fmt(dev, (struct v4l2_format *) arg); 1513 return em28xx_get_fmt(dev, (struct v4l2_format *) arg);
1528 1514
@@ -1531,131 +1517,130 @@ static int em28xx_video_do_ioctl(struct inode *inode, struct file *filp,
1531 return em28xx_set_fmt(dev, cmd, (struct v4l2_format *)arg); 1517 return em28xx_set_fmt(dev, cmd, (struct v4l2_format *)arg);
1532 1518
1533 case VIDIOC_REQBUFS: 1519 case VIDIOC_REQBUFS:
1534 { 1520 {
1535 struct v4l2_requestbuffers *rb = arg; 1521 struct v4l2_requestbuffers *rb = arg;
1536 u32 i; 1522 u32 i;
1537 int ret; 1523 int ret;
1538
1539 if (rb->type != V4L2_BUF_TYPE_VIDEO_CAPTURE ||
1540 rb->memory != V4L2_MEMORY_MMAP)
1541 return -EINVAL;
1542 1524
1543 if (dev->io == IO_READ) { 1525 if (rb->type != V4L2_BUF_TYPE_VIDEO_CAPTURE ||
1544 em28xx_videodbg ("method is set to read;" 1526 rb->memory != V4L2_MEMORY_MMAP)
1545 " close and open the device again to" 1527 return -EINVAL;
1546 " choose the mmap I/O method\n");
1547 return -EINVAL;
1548 }
1549 1528
1550 for (i = 0; i < dev->num_frames; i++) 1529 if (dev->io == IO_READ) {
1551 if (dev->frame[i].vma_use_count) { 1530 em28xx_videodbg ("method is set to read;"
1552 em28xx_videodbg ("VIDIOC_REQBUFS failed; previous buffers are still mapped\n"); 1531 " close and open the device again to"
1553 return -EINVAL; 1532 " choose the mmap I/O method\n");
1554 } 1533 return -EINVAL;
1534 }
1555 1535
1556 if (dev->stream == STREAM_ON) { 1536 for (i = 0; i < dev->num_frames; i++)
1557 em28xx_videodbg("VIDIOC_REQBUFS: interrupting stream\n"); 1537 if (dev->frame[i].vma_use_count) {
1558 if ((ret = em28xx_stream_interrupt(dev))) 1538 em28xx_videodbg ("VIDIOC_REQBUFS failed; previous buffers are still mapped\n");
1559 return ret; 1539 return -EINVAL;
1560 } 1540 }
1561 1541
1562 em28xx_empty_framequeues(dev); 1542 if (dev->stream == STREAM_ON) {
1543 em28xx_videodbg("VIDIOC_REQBUFS: interrupting stream\n");
1544 if ((ret = em28xx_stream_interrupt(dev)))
1545 return ret;
1546 }
1563 1547
1564 em28xx_release_buffers(dev); 1548 em28xx_empty_framequeues(dev);
1565 if (rb->count)
1566 rb->count =
1567 em28xx_request_buffers(dev, rb->count);
1568 1549
1569 dev->frame_current = NULL; 1550 em28xx_release_buffers(dev);
1551 if (rb->count)
1552 rb->count =
1553 em28xx_request_buffers(dev, rb->count);
1570 1554
1571 em28xx_videodbg ("VIDIOC_REQBUFS: setting io method to mmap: num bufs %i\n", 1555 dev->frame_current = NULL;
1572 rb->count);
1573 dev->io = rb->count ? IO_MMAP : IO_NONE;
1574 return 0;
1575 }
1576 1556
1557 em28xx_videodbg ("VIDIOC_REQBUFS: setting io method to mmap: num bufs %i\n",
1558 rb->count);
1559 dev->io = rb->count ? IO_MMAP : IO_NONE;
1560 return 0;
1561 }
1577 case VIDIOC_QUERYBUF: 1562 case VIDIOC_QUERYBUF:
1578 { 1563 {
1579 struct v4l2_buffer *b = arg; 1564 struct v4l2_buffer *b = arg;
1580 1565
1581 if (b->type != V4L2_BUF_TYPE_VIDEO_CAPTURE || 1566 if (b->type != V4L2_BUF_TYPE_VIDEO_CAPTURE ||
1582 b->index >= dev->num_frames || dev->io != IO_MMAP) 1567 b->index >= dev->num_frames || dev->io != IO_MMAP)
1583 return -EINVAL; 1568 return -EINVAL;
1584 1569
1585 memcpy(b, &dev->frame[b->index].buf, sizeof(*b)); 1570 memcpy(b, &dev->frame[b->index].buf, sizeof(*b));
1586 1571
1587 if (dev->frame[b->index].vma_use_count) { 1572 if (dev->frame[b->index].vma_use_count) {
1588 b->flags |= V4L2_BUF_FLAG_MAPPED; 1573 b->flags |= V4L2_BUF_FLAG_MAPPED;
1589 }
1590 if (dev->frame[b->index].state == F_DONE)
1591 b->flags |= V4L2_BUF_FLAG_DONE;
1592 else if (dev->frame[b->index].state != F_UNUSED)
1593 b->flags |= V4L2_BUF_FLAG_QUEUED;
1594 return 0;
1595 } 1574 }
1575 if (dev->frame[b->index].state == F_DONE)
1576 b->flags |= V4L2_BUF_FLAG_DONE;
1577 else if (dev->frame[b->index].state != F_UNUSED)
1578 b->flags |= V4L2_BUF_FLAG_QUEUED;
1579 return 0;
1580 }
1596 case VIDIOC_QBUF: 1581 case VIDIOC_QBUF:
1597 { 1582 {
1598 struct v4l2_buffer *b = arg; 1583 struct v4l2_buffer *b = arg;
1599 unsigned long lock_flags; 1584 unsigned long lock_flags;
1600 1585
1601 if (b->type != V4L2_BUF_TYPE_VIDEO_CAPTURE || 1586 if (b->type != V4L2_BUF_TYPE_VIDEO_CAPTURE ||
1602 b->index >= dev->num_frames || dev->io != IO_MMAP) { 1587 b->index >= dev->num_frames || dev->io != IO_MMAP) {
1603 return -EINVAL; 1588 return -EINVAL;
1604 } 1589 }
1605 1590
1606 if (dev->frame[b->index].state != F_UNUSED) { 1591 if (dev->frame[b->index].state != F_UNUSED) {
1607 return -EAGAIN; 1592 return -EAGAIN;
1608 } 1593 }
1609 dev->frame[b->index].state = F_QUEUED; 1594 dev->frame[b->index].state = F_QUEUED;
1610 1595
1611 /* add frame to fifo */ 1596 /* add frame to fifo */
1612 spin_lock_irqsave(&dev->queue_lock, lock_flags); 1597 spin_lock_irqsave(&dev->queue_lock, lock_flags);
1613 list_add_tail(&dev->frame[b->index].frame, 1598 list_add_tail(&dev->frame[b->index].frame,
1614 &dev->inqueue); 1599 &dev->inqueue);
1615 spin_unlock_irqrestore(&dev->queue_lock, lock_flags); 1600 spin_unlock_irqrestore(&dev->queue_lock, lock_flags);
1616 1601
1617 return 0; 1602 return 0;
1618 } 1603 }
1619 case VIDIOC_DQBUF: 1604 case VIDIOC_DQBUF:
1620 { 1605 {
1621 struct v4l2_buffer *b = arg; 1606 struct v4l2_buffer *b = arg;
1622 struct em28xx_frame_t *f; 1607 struct em28xx_frame_t *f;
1623 unsigned long lock_flags; 1608 unsigned long lock_flags;
1624 int ret = 0; 1609 int ret = 0;
1625 1610
1626 if (b->type != V4L2_BUF_TYPE_VIDEO_CAPTURE 1611 if (b->type != V4L2_BUF_TYPE_VIDEO_CAPTURE
1627 || dev->io != IO_MMAP) 1612 || dev->io != IO_MMAP)
1628 return -EINVAL; 1613 return -EINVAL;
1629 1614
1630 if (list_empty(&dev->outqueue)) { 1615 if (list_empty(&dev->outqueue)) {
1631 if (dev->stream == STREAM_OFF) 1616 if (dev->stream == STREAM_OFF)
1632 return -EINVAL; 1617 return -EINVAL;
1633 if (filp->f_flags & O_NONBLOCK) 1618 if (filp->f_flags & O_NONBLOCK)
1634 return -EAGAIN; 1619 return -EAGAIN;
1635 ret = wait_event_interruptible 1620 ret = wait_event_interruptible
1636 (dev->wait_frame, 1621 (dev->wait_frame,
1637 (!list_empty(&dev->outqueue)) || 1622 (!list_empty(&dev->outqueue)) ||
1638 (dev->state & DEV_DISCONNECTED)); 1623 (dev->state & DEV_DISCONNECTED));
1639 if (ret) 1624 if (ret)
1640 return ret; 1625 return ret;
1641 if (dev->state & DEV_DISCONNECTED) 1626 if (dev->state & DEV_DISCONNECTED)
1642 return -ENODEV; 1627 return -ENODEV;
1643 } 1628 }
1644 1629
1645 spin_lock_irqsave(&dev->queue_lock, lock_flags); 1630 spin_lock_irqsave(&dev->queue_lock, lock_flags);
1646 f = list_entry(dev->outqueue.next, 1631 f = list_entry(dev->outqueue.next,
1647 struct em28xx_frame_t, frame); 1632 struct em28xx_frame_t, frame);
1648 list_del(dev->outqueue.next); 1633 list_del(dev->outqueue.next);
1649 spin_unlock_irqrestore(&dev->queue_lock, lock_flags); 1634 spin_unlock_irqrestore(&dev->queue_lock, lock_flags);
1650 1635
1651 f->state = F_UNUSED; 1636 f->state = F_UNUSED;
1652 memcpy(b, &f->buf, sizeof(*b)); 1637 memcpy(b, &f->buf, sizeof(*b));
1653 1638
1654 if (f->vma_use_count) 1639 if (f->vma_use_count)
1655 b->flags |= V4L2_BUF_FLAG_MAPPED; 1640 b->flags |= V4L2_BUF_FLAG_MAPPED;
1656 1641
1657 return 0; 1642 return 0;
1658 } 1643 }
1659 default: 1644 default:
1660 return em28xx_do_ioctl(inode, filp, dev, cmd, arg, 1645 return em28xx_do_ioctl(inode, filp, dev, cmd, arg,
1661 em28xx_video_do_ioctl); 1646 em28xx_video_do_ioctl);