Commit 2b3e284a authored by Hans de Goede's avatar Hans de Goede Committed by Mauro Carvalho Chehab

[media] gspca_sonixb: Rewrite start of frame detection

Our old start of frame detection code wrongly assumes that the sof
marker always lives at the beginning of the frame. At least for the
0c45:602a camera this is not the case. This patch also improves
the framerate from 28 fps to 30 fps with the 0c45:6005 and 0c45:6007
Signed-off-by: default avatarHans de Goede <hdegoede@redhat.com>
Signed-off-by: default avatarMauro Carvalho Chehab <mchehab@redhat.com>
parent d6746d55
...@@ -56,6 +56,8 @@ struct sd { ...@@ -56,6 +56,8 @@ struct sd {
int prev_avg_lum; int prev_avg_lum;
int exp_too_low_cnt; int exp_too_low_cnt;
int exp_too_high_cnt; int exp_too_high_cnt;
int header_read;
u8 header[12]; /* Header without sof marker */
unsigned short exposure; unsigned short exposure;
unsigned char gain; unsigned char gain;
...@@ -1177,13 +1179,10 @@ static void sd_stopN(struct gspca_dev *gspca_dev) ...@@ -1177,13 +1179,10 @@ static void sd_stopN(struct gspca_dev *gspca_dev)
sd_init(gspca_dev); sd_init(gspca_dev);
} }
static void sd_pkt_scan(struct gspca_dev *gspca_dev, static u8* find_sof(struct gspca_dev *gspca_dev, u8 *data, int len)
u8 *data, /* isoc packet */
int len) /* iso packet length */
{ {
int i;
struct sd *sd = (struct sd *) gspca_dev; struct sd *sd = (struct sd *) gspca_dev;
struct cam *cam = &gspca_dev->cam; int i, header_size = (sd->bridge == BRIDGE_103) ? 18 : 12;
/* frames start with: /* frames start with:
* ff ff 00 c4 c4 96 synchro * ff ff 00 c4 c4 96 synchro
...@@ -1194,28 +1193,102 @@ static void sd_pkt_scan(struct gspca_dev *gspca_dev, ...@@ -1194,28 +1193,102 @@ static void sd_pkt_scan(struct gspca_dev *gspca_dev,
* ll mm brightness sum outside auto exposure * ll mm brightness sum outside auto exposure
* (xx xx xx xx xx) audio values for snc103 * (xx xx xx xx xx) audio values for snc103
*/ */
if (len > 6 && len < 24) { for (i = 0; i < len; i++) {
for (i = 0; i < len - 6; i++) { switch (sd->header_read) {
if (data[0 + i] == 0xff case 0:
&& data[1 + i] == 0xff if (data[i] == 0xff)
&& data[2 + i] == 0x00 sd->header_read++;
&& data[3 + i] == 0xc4 break;
&& data[4 + i] == 0xc4 case 1:
&& data[5 + i] == 0x96) { /* start of frame */ if (data[i] == 0xff)
int lum = -1; sd->header_read++;
int pkt_type = LAST_PACKET; else
int fr_h_sz = (sd->bridge == BRIDGE_103) ? sd->header_read = 0;
18 : 12; break;
case 2:
if (len - i < fr_h_sz) { if (data[i] == 0x00)
PDEBUG(D_STREAM, "packet too short to" sd->header_read++;
" get avg brightness"); else if (data[i] != 0xff)
} else if (sd->bridge == BRIDGE_103) { sd->header_read = 0;
lum = data[i + 9] + break;
(data[i + 10] << 8); case 3:
if (data[i] == 0xc4)
sd->header_read++;
else if (data[i] == 0xff)
sd->header_read = 1;
else
sd->header_read = 0;
break;
case 4:
if (data[i] == 0xc4)
sd->header_read++;
else if (data[i] == 0xff)
sd->header_read = 1;
else
sd->header_read = 0;
break;
case 5:
if (data[i] == 0x96)
sd->header_read++;
else if (data[i] == 0xff)
sd->header_read = 1;
else
sd->header_read = 0;
break;
default:
sd->header[sd->header_read - 6] = data[i];
sd->header_read++;
if (sd->header_read == header_size) {
sd->header_read = 0;
return data + i + 1;
}
}
}
return NULL;
}
static void sd_pkt_scan(struct gspca_dev *gspca_dev,
u8 *data, /* isoc packet */
int len) /* iso packet length */
{
int fr_h_sz = 0, lum_offset = 0, len_after_sof = 0;
struct sd *sd = (struct sd *) gspca_dev;
struct cam *cam = &gspca_dev->cam;
u8 *sof;
sof = find_sof(gspca_dev, data, len);
if (sof) {
if (sd->bridge == BRIDGE_103) {
fr_h_sz = 18;
lum_offset = 3;
} else { } else {
lum = data[i + 8] + (data[i + 9] << 8); fr_h_sz = 12;
lum_offset = 2;
}
len_after_sof = len - (sof - data);
len = (sof - data) - fr_h_sz;
if (len < 0)
len = 0;
}
if (cam->cam_mode[gspca_dev->curr_mode].priv & MODE_RAW) {
/* In raw mode we sometimes get some garbage after the frame
ignore this */
int used;
int size = cam->cam_mode[gspca_dev->curr_mode].sizeimage;
used = gspca_dev->image_len;
if (used + len > size)
len = size - used;
} }
gspca_frame_add(gspca_dev, INTER_PACKET, data, len);
if (sof) {
int lum = sd->header[lum_offset] +
(sd->header[lum_offset + 1] << 8);
/* When exposure changes midway a frame we /* When exposure changes midway a frame we
get a lum of 0 in this case drop 2 frames get a lum of 0 in this case drop 2 frames
as the frames directly after an exposure as the frames directly after an exposure
...@@ -1231,34 +1304,13 @@ static void sd_pkt_scan(struct gspca_dev *gspca_dev, ...@@ -1231,34 +1304,13 @@ static void sd_pkt_scan(struct gspca_dev *gspca_dev,
sd->prev_avg_lum = lum; sd->prev_avg_lum = lum;
atomic_set(&sd->avg_lum, lum); atomic_set(&sd->avg_lum, lum);
if (sd->frames_to_drop) { if (sd->frames_to_drop)
sd->frames_to_drop--; sd->frames_to_drop--;
pkt_type = DISCARD_PACKET; else
} gspca_frame_add(gspca_dev, LAST_PACKET, NULL, 0);
gspca_frame_add(gspca_dev, pkt_type,
NULL, 0);
data += i + fr_h_sz;
len -= i + fr_h_sz;
gspca_frame_add(gspca_dev, FIRST_PACKET,
data, len);
return;
}
}
}
if (cam->cam_mode[gspca_dev->curr_mode].priv & MODE_RAW) {
/* In raw mode we sometimes get some garbage after the frame
ignore this */
int used;
int size = cam->cam_mode[gspca_dev->curr_mode].sizeimage;
used = gspca_dev->image_len; gspca_frame_add(gspca_dev, FIRST_PACKET, sof, len_after_sof);
if (used + len > size)
len = size - used;
} }
gspca_frame_add(gspca_dev, INTER_PACKET, data, len);
} }
static int sd_setbrightness(struct gspca_dev *gspca_dev, __s32 val) static int sd_setbrightness(struct gspca_dev *gspca_dev, __s32 val)
......
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