提交 1d3104b8 编写于 作者: D David Hilvert 提交者: Mauro Carvalho Chehab

V4L/DVB (7589): ibmcam: improve support for the IBM PC Camera Pro

This patch modifies Dmitri's original ibmcam driver for Linux to improve
support for the IBM PC Camera Pro.  It may also offer improved support for
other models classified by the driver as 'Model 3', such as the IBM PC Camera
Pro Max.

See http://auricle.dyndns.org/xvp610/Signed-off-by: NDavid Hilvert <dhilvert@gmail.com>
Signed-off-by: NAndrew Morton <akpm@linux-foundation.org>
Signed-off-by: NMauro Carvalho Chehab <mchehab@infradead.org>
上级 17de9a4e
...@@ -802,6 +802,21 @@ static enum ParseState ibmcam_model2_320x240_parse_lines( ...@@ -802,6 +802,21 @@ static enum ParseState ibmcam_model2_320x240_parse_lines(
return scan_Continue; return scan_Continue;
} }
/*
* ibmcam_model3_parse_lines()
*
* | Even lines | Odd Lines |
* -----------------------------------|
* |YYY........Y|UYVYUYVY.........UYVY|
* |YYY........Y|UYVYUYVY.........UYVY|
* |............|.....................|
* |YYY........Y|UYVYUYVY.........UYVY|
* |------------+---------------------|
*
* There is one (U, V) chroma pair for every four luma (Y) values. This
* function reads a pair of lines at a time and obtains missing chroma values
* from adjacent pixels.
*/
static enum ParseState ibmcam_model3_parse_lines( static enum ParseState ibmcam_model3_parse_lines(
struct uvd *uvd, struct uvd *uvd,
struct usbvideo_frame *frame, struct usbvideo_frame *frame,
...@@ -816,6 +831,7 @@ static enum ParseState ibmcam_model3_parse_lines( ...@@ -816,6 +831,7 @@ static enum ParseState ibmcam_model3_parse_lines(
const int ccm = 128; /* Color correction median - see below */ const int ccm = 128; /* Color correction median - see below */
int i, u, v, rw, data_w=0, data_h=0, color_corr; int i, u, v, rw, data_w=0, data_h=0, color_corr;
static unsigned char lineBuffer[640*3]; static unsigned char lineBuffer[640*3];
int line;
color_corr = (uvd->vpic.colour - 0x8000) >> 8; /* -128..+127 = -ccm..+(ccm-1)*/ color_corr = (uvd->vpic.colour - 0x8000) >> 8; /* -128..+127 = -ccm..+(ccm-1)*/
RESTRICT_TO_RANGE(color_corr, -ccm, ccm+1); RESTRICT_TO_RANGE(color_corr, -ccm, ccm+1);
...@@ -869,15 +885,15 @@ static enum ParseState ibmcam_model3_parse_lines( ...@@ -869,15 +885,15 @@ static enum ParseState ibmcam_model3_parse_lines(
return scan_NextFrame; return scan_NextFrame;
} }
/* Make sure there's enough data for the entire line */ /* Make sure that lineBuffer can store two lines of data */
len = 3 * data_w; /* <y-data> <uv-data> */ len = 3 * data_w; /* <y-data> <uyvy-data> */
assert(len <= sizeof(lineBuffer)); assert(len <= sizeof(lineBuffer));
/* Make sure there's enough data for the entire line */ /* Make sure there's enough data for two lines */
if (RingQueue_GetLength(&uvd->dp) < len) if (RingQueue_GetLength(&uvd->dp) < len)
return scan_Out; return scan_Out;
/* Suck one line out of the ring queue */ /* Suck two lines of data out of the ring queue */
RingQueue_Dequeue(&uvd->dp, lineBuffer, len); RingQueue_Dequeue(&uvd->dp, lineBuffer, len);
data = lineBuffer; data = lineBuffer;
...@@ -887,15 +903,23 @@ static enum ParseState ibmcam_model3_parse_lines( ...@@ -887,15 +903,23 @@ static enum ParseState ibmcam_model3_parse_lines(
rw = (int)VIDEOSIZE_Y(frame->request) - (int)(frame->curline) - 1; rw = (int)VIDEOSIZE_Y(frame->request) - (int)(frame->curline) - 1;
RESTRICT_TO_RANGE(rw, 0, VIDEOSIZE_Y(frame->request)-1); RESTRICT_TO_RANGE(rw, 0, VIDEOSIZE_Y(frame->request)-1);
/* Iterate over two lines. */
for (line = 0; line < 2; line++) {
for (i = 0; i < VIDEOSIZE_X(frame->request); i++) { for (i = 0; i < VIDEOSIZE_X(frame->request); i++) {
int y, rv, gv, bv; /* RGB components */ int y;
int rv, gv, bv; /* RGB components */
if (i < data_w) { if (i >= data_w) {
y = data[i]; /* Luminosity is the first line */ RGB24_PUTPIXEL(frame, i, rw, 0, 0, 0);
continue;
}
/* first line is YYY...Y; second is UYVY...UYVY */
y = data[(line == 0) ? i : (i*2 + 1)];
/* Apply static color correction */ /* Apply static color correction */
u = color[i*2] + hue_corr; u = color[(i/2)*4] + hue_corr;
v = color[i*2 + 1] + hue2_corr; v = color[(i/2)*4 + 2] + hue2_corr;
/* Apply color correction */ /* Apply color correction */
if (color_corr != 0) { if (color_corr != 0) {
...@@ -903,13 +927,21 @@ static enum ParseState ibmcam_model3_parse_lines( ...@@ -903,13 +927,21 @@ static enum ParseState ibmcam_model3_parse_lines(
u = 128 + ((ccm + color_corr) * (u - 128)) / ccm; u = 128 + ((ccm + color_corr) * (u - 128)) / ccm;
v = 128 + ((ccm + color_corr) * (v - 128)) / ccm; v = 128 + ((ccm + color_corr) * (v - 128)) / ccm;
} }
} else
y = 0, u = v = 128;
YUV_TO_RGB_BY_THE_BOOK(y, u, v, rv, gv, bv); YUV_TO_RGB_BY_THE_BOOK(y, u, v, rv, gv, bv);
RGB24_PUTPIXEL(frame, i, rw, rv, gv, bv); /* Done by deinterlacing now */ RGB24_PUTPIXEL(frame, i, rw, rv, gv, bv); /* No deinterlacing */
} }
frame->deinterlace = Deinterlace_FillEvenLines;
/* Check for the end of requested data */
if (rw == 0)
break;
/* Prepare for the second line */
rw--;
data = lineBuffer + data_w;
}
frame->deinterlace = Deinterlace_None;
/* /*
* Account for number of bytes that we wrote into output V4L frame. * Account for number of bytes that we wrote into output V4L frame.
......
Markdown is supported
0% .
You are about to add 0 people to the discussion. Proceed with caution.
先完成此消息的编辑!
想要评论请 注册