// SPDX-License-Identifier: GPL-2.0-or-later
/*
* Syntek STK1135 subdriver
*
* Copyright (c) 2013 Ondrej Zary
*
* Based on Syntekdriver (stk11xx) by Nicolas VIVIEN:
* http://syntekdriver.sourceforge.net
*/
#define pr_fmt(fmt) KBUILD_MODNAME
": " fmt
#define MODULE_NAME
"stk1135"
#include "gspca.h"
#include "stk1135.h"
MODULE_AUTHOR(
"Ondrej Zary" );
MODULE_DESCRIPTION(
"Syntek STK1135 USB Camera Driver" );
MODULE_LICENSE(
"GPL" );
/* specific webcam descriptor */
struct sd {
struct gspca_dev gspca_dev;
/* !! must be the first item */
u8 pkt_seq;
u8 sensor_page;
bool flip_status;
u8 flip_debounce;
struct v4l2_ctrl *hflip;
struct v4l2_ctrl *vflip;
};
static const struct v4l2_pix_format stk1135_modes[] = {
/* default mode (this driver supports variable resolution) */
{
640 ,
480 , V4L2_PIX_FMT_SBGGR8, V4L2_FIELD_NONE,
.bytesperline =
640 ,
.sizeimage =
640 *
480 ,
.colorspace = V4L2_COLORSPACE_SRGB},
};
/* -- read a register -- */
static u8 reg_r(
struct gspca_dev *gspca_dev, u16 index)
{
struct usb_device *dev = gspca_dev->dev;
int ret;
if (gspca_dev->usb_err <
0 )
return 0 ;
ret = usb_control_msg(dev, usb_rcvctrlpipe(dev,
0 ),
0 x00,
USB_DIR_IN | USB_TYPE_VENDOR | USB_RECIP_DEVICE,
0 x00,
index,
gspca_dev->usb_buf,
1 ,
500 );
gspca_dbg(gspca_dev, D_USBI,
"reg_r 0x%x=0x%02x\n" ,
index, gspca_dev->usb_buf[
0 ]);
if (ret <
0 ) {
pr_err(
"reg_r 0x%x err %d\n" , index, ret);
gspca_dev->usb_err = ret;
return 0 ;
}
return gspca_dev->usb_buf[
0 ];
}
/* -- write a register -- */
static void reg_w(
struct gspca_dev *gspca_dev, u16 index, u8 val)
{
int ret;
struct usb_device *dev = gspca_dev->dev;
if (gspca_dev->usb_err <
0 )
return ;
ret = usb_control_msg(dev, usb_sndctrlpipe(dev,
0 ),
0 x01,
USB_DIR_OUT | USB_TYPE_VENDOR | USB_RECIP_DEVICE,
val,
index,
NULL,
0 ,
500 );
gspca_dbg(gspca_dev, D_USBO,
"reg_w 0x%x:=0x%02x\n" , index, val);
if (ret <
0 ) {
pr_err(
"reg_w 0x%x err %d\n" , index, ret);
gspca_dev->usb_err = ret;
}
}
static void reg_w_mask(
struct gspca_dev *gspca_dev, u16 index, u8 val, u8 mask)
{
val = (reg_r(gspca_dev, index) & ~mask) | (val & mask);
reg_w(gspca_dev, index, val);
}
/* this function is called at probe time */
static int sd_config(
struct gspca_dev *gspca_dev,
const struct usb_device_id *id)
{
gspca_dev->cam.cam_mode = stk1135_modes;
gspca_dev->cam.nmodes = ARRAY_SIZE(stk1135_modes);
return 0 ;
}
static int stk1135_serial_wait_ready(
struct gspca_dev *gspca_dev)
{
int i =
0 ;
u8 val;
do {
val = reg_r(gspca_dev, STK1135_REG_SICTL +
1 );
if (i++ >
500 ) {
/* maximum retry count */
pr_err(
"serial bus timeout: status=0x%02x\n" , val);
return -
1 ;
}
/* repeat if BUSY or WRITE/READ not finished */
}
while ((val &
0 x10) || !(val &
0 x05));
return 0 ;
}
static u8 sensor_read_8(
struct gspca_dev *gspca_dev, u8 addr)
{
reg_w(gspca_dev, STK1135_REG_SBUSR, addr);
/* begin read */
reg_w(gspca_dev, STK1135_REG_SICTL,
0 x20);
/* wait until finished */
if (stk1135_serial_wait_ready(gspca_dev)) {
pr_err(
"Sensor read failed\n" );
return 0 ;
}
return reg_r(gspca_dev, STK1135_REG_SBUSR +
1 );
}
static u16 sensor_read_16(
struct gspca_dev *gspca_dev, u8 addr)
{
return (sensor_read_8(gspca_dev, addr) <<
8 ) |
sensor_read_8(gspca_dev,
0 xf1);
}
static void sensor_write_8(
struct gspca_dev *gspca_dev, u8 addr, u8 data)
{
/* load address and data registers */
reg_w(gspca_dev, STK1135_REG_SBUSW, addr);
reg_w(gspca_dev, STK1135_REG_SBUSW +
1 , data);
/* begin write */
reg_w(gspca_dev, STK1135_REG_SICTL,
0 x01);
/* wait until finished */
if (stk1135_serial_wait_ready(gspca_dev)) {
pr_err(
"Sensor write failed\n" );
return ;
}
}
static void sensor_write_16(
struct gspca_dev *gspca_dev, u8 addr, u16 data)
{
sensor_write_8(gspca_dev, addr, data >>
8 );
sensor_write_8(gspca_dev,
0 xf1, data &
0 xff);
}
static void sensor_set_page(
struct gspca_dev *gspca_dev, u8 page)
{
struct sd *sd = (
struct sd *) gspca_dev;
if (page != sd->sensor_page) {
sensor_write_16(gspca_dev,
0 xf0, page);
sd->sensor_page = page;
}
}
static u16 sensor_read(
struct gspca_dev *gspca_dev, u16 reg)
{
sensor_set_page(gspca_dev, reg >>
8 );
return sensor_read_16(gspca_dev, reg &
0 xff);
}
static void sensor_write(
struct gspca_dev *gspca_dev, u16 reg, u16 val)
{
sensor_set_page(gspca_dev, reg >>
8 );
sensor_write_16(gspca_dev, reg &
0 xff, val);
}
static void sensor_write_mask(
struct gspca_dev *gspca_dev,
u16 reg, u16 val, u16 mask)
{
val = (sensor_read(gspca_dev, reg) & ~mask) | (val & mask);
sensor_write(gspca_dev, reg, val);
}
struct sensor_val {
u16 reg;
u16 val;
};
/* configure MT9M112 sensor */
static void stk1135_configure_mt9m112(
struct gspca_dev *gspca_dev)
{
static const struct sensor_val cfg[] = {
/* restart&reset, chip enable, reserved */
{
0 x00d,
0 x000b }, {
0 x00d,
0 x0008 }, {
0 x035,
0 x0022 },
/* mode ctl: AWB on, AE both, clip aper corr, defect corr, AE */
{
0 x106,
0 x700e },
{
0 x2dd,
0 x18e0 },
/* B-R thresholds, */
/* AWB */
{
0 x21f,
0 x0180 },
/* Cb and Cr limits */
{
0 x220,
0 xc814 }, {
0 x221,
0 x8080 },
/* lum limits, RGB gain */
{
0 x222,
0 xa078 }, {
0 x223,
0 xa078 },
/* R, B limit */
{
0 x224,
0 x5f20 }, {
0 x228,
0 xea02 },
/* mtx adj lim, adv ctl */
{
0 x229,
0 x867a },
/* wide gates */
/* Color correction */
/* imager gains base, delta, delta signs */
{
0 x25e,
0 x594c }, {
0 x25f,
0 x4d51 }, {
0 x260,
0 x0002 },
/* AWB adv ctl 2, gain offs */
{
0 x2ef,
0 x0008 }, {
0 x2f2,
0 x0000 },
/* base matrix signs, scale K1-5, K6-9 */
{
0 x202,
0 x00ee }, {
0 x203,
0 x3923 }, {
0 x204,
0 x0724 },
/* base matrix coef */
{
0 x209,
0 x00cd }, {
0 x20a,
0 x0093 }, {
0 x20b,
0 x0004 },
/*K1-3*/
{
0 x20c,
0 x005c }, {
0 x20d,
0 x00d9 }, {
0 x20e,
0 x0053 },
/*K4-6*/
{
0 x20f,
0 x0008 }, {
0 x210,
0 x0091 }, {
0 x211,
0 x00cf },
/*K7-9*/
{
0 x215,
0 x0000 },
/* delta mtx signs */
/* delta matrix coef */
{
0 x216,
0 x0000 }, {
0 x217,
0 x0000 }, {
0 x218,
0 x0000 },
/*D1-3*/
{
0 x219,
0 x0000 }, {
0 x21a,
0 x0000 }, {
0 x21b,
0 x0000 },
/*D4-6*/
{
0 x21c,
0 x0000 }, {
0 x21d,
0 x0000 }, {
0 x21e,
0 x0000 },
/*D7-9*/
/* enable & disable manual WB to apply color corr. settings */
{
0 x106,
0 xf00e }, {
0 x106,
0 x700e },
/* Lens shading correction */
{
0 x180,
0 x0007 },
/* control */
/* vertical knee 0, 2+1, 4+3 */
{
0 x181,
0 xde13 }, {
0 x182,
0 xebe2 }, {
0 x183,
0 x00f6 },
/* R */
{
0 x184,
0 xe114 }, {
0 x185,
0 xeadd }, {
0 x186,
0 xfdf6 },
/* G */
{
0 x187,
0 xe511 }, {
0 x188,
0 xede6 }, {
0 x189,
0 xfbf7 },
/* B */
/* horizontal knee 0, 2+1, 4+3, 5 */
{
0 x18a,
0 xd613 }, {
0 x18b,
0 xedec },
/* R .. */
{
0 x18c,
0 xf9f2 }, {
0 x18d,
0 x0000 },
/* .. R */
{
0 x18e,
0 xd815 }, {
0 x18f,
0 xe9ea },
/* G .. */
{
0 x190,
0 xf9f1 }, {
0 x191,
0 x0002 },
/* .. G */
{
0 x192,
0 xde10 }, {
0 x193,
0 xefef },
/* B .. */
{
0 x194,
0 xfbf4 }, {
0 x195,
0 x0002 },
/* .. B */
/* vertical knee 6+5, 8+7 */
{
0 x1b6,
0 x0e06 }, {
0 x1b7,
0 x2713 },
/* R */
{
0 x1b8,
0 x1106 }, {
0 x1b9,
0 x2713 },
/* G */
{
0 x1ba,
0 x0c03 }, {
0 x1bb,
0 x2a0f },
/* B */
/* horizontal knee 7+6, 9+8, 10 */
{
0 x1bc,
0 x1208 }, {
0 x1bd,
0 x1a16 }, {
0 x1be,
0 x0022 },
/* R */
{
0 x1bf,
0 x150a }, {
0 x1c0,
0 x1c1a }, {
0 x1c1,
0 x002d },
/* G */
{
0 x1c2,
0 x1109 }, {
0 x1c3,
0 x1414 }, {
0 x1c4,
0 x002a },
/* B */
{
0 x106,
0 x740e },
/* enable lens shading correction */
/* Gamma correction - context A */
{
0 x153,
0 x0b03 }, {
0 x154,
0 x4722 }, {
0 x155,
0 xac82 },
{
0 x156,
0 xdac7 }, {
0 x157,
0 xf5e9 }, {
0 x158,
0 xff00 },
/* Gamma correction - context B */
{
0 x1dc,
0 x0b03 }, {
0 x1dd,
0 x4722 }, {
0 x1de,
0 xac82 },
{
0 x1df,
0 xdac7 }, {
0 x1e0,
0 xf5e9 }, {
0 x1e1,
0 xff00 },
/* output format: RGB, invert output pixclock, output bayer */
{
0 x13a,
0 x4300 }, {
0 x19b,
0 x4300 },
/* for context A, B */
{
0 x108,
0 x0180 },
/* format control - enable bayer row flip */
{
0 x22f,
0 xd100 }, {
0 x29c,
0 xd100 },
/* AE A, B */
/* default prg conf, prg ctl - by 0x2d2, prg advance - PA1 */
{
0 x2d2,
0 x0000 }, {
0 x2cc,
0 x0004 }, {
0 x2cb,
0 x0001 },
{
0 x22e,
0 x0c3c }, {
0 x267,
0 x1010 },
/* AE tgt ctl, gain lim */
/* PLL */
{
0 x065,
0 xa000 },
/* clk ctl - enable PLL (clear bit 14) */
{
0 x066,
0 x2003 }, {
0 x067,
0 x0501 },
/* PLL M=128, N=3, P=1 */
{
0 x065,
0 x2000 },
/* disable PLL bypass (clear bit 15) */
{
0 x005,
0 x01b8 }, {
0 x007,
0 x00d8 },
/* horiz blanking B, A */
/* AE line size, shutter delay limit */
{
0 x239,
0 x06c0 }, {
0 x23b,
0 x040e },
/* for context A */
{
0 x23a,
0 x06c0 }, {
0 x23c,
0 x0564 },
/* for context B */
/* shutter width basis 60Hz, 50Hz */
{
0 x257,
0 x0208 }, {
0 x258,
0 x0271 },
/* for context A */
{
0 x259,
0 x0209 }, {
0 x25a,
0 x0271 },
/* for context B */
{
0 x25c,
0 x120d }, {
0 x25d,
0 x1712 },
/* flicker 60Hz, 50Hz */
{
0 x264,
0 x5e1c },
/* reserved */
/* flicker, AE gain limits, gain zone limits */
{
0 x25b,
0 x0003 }, {
0 x236,
0 x7810 }, {
0 x237,
0 x8304 },
{
0 x008,
0 x0021 },
/* vert blanking A */
};
int i;
u16 width, height;
for (i =
0 ; i < ARRAY_SIZE(cfg); i++)
sensor_write(gspca_dev, cfg[i].reg, cfg[i].val);
/* set output size */
width = gspca_dev->pixfmt.width;
height = gspca_dev->pixfmt.height;
if (width <=
640 && height <=
512 ) {
/* context A (half readout speed)*/
sensor_write(gspca_dev,
0 x1a7, width);
sensor_write(gspca_dev,
0 x1aa, height);
/* set read mode context A */
sensor_write(gspca_dev,
0 x0c8,
0 x0000);
/* set resize, read mode, vblank, hblank context A */
sensor_write(gspca_dev,
0 x2c8,
0 x0000);
}
else {
/* context B (full readout speed) */
sensor_write(gspca_dev,
0 x1a1, width);
sensor_write(gspca_dev,
0 x1a4, height);
/* set read mode context B */
sensor_write(gspca_dev,
0 x0c8,
0 x0008);
/* set resize, read mode, vblank, hblank context B */
sensor_write(gspca_dev,
0 x2c8,
0 x040b);
}
}
static void stk1135_configure_clock(
struct gspca_dev *gspca_dev)
{
/* configure SCLKOUT */
reg_w(gspca_dev, STK1135_REG_TMGEN,
0 x12);
/* set 1 clock per pixel */
/* and positive edge clocked pulse high when pixel counter = 0 */
reg_w(gspca_dev, STK1135_REG_TCP1 +
0 ,
0 x41);
reg_w(gspca_dev, STK1135_REG_TCP1 +
1 ,
0 x00);
reg_w(gspca_dev, STK1135_REG_TCP1 +
2 ,
0 x00);
reg_w(gspca_dev, STK1135_REG_TCP1 +
3 ,
0 x00);
/* enable CLKOUT for sensor */
reg_w(gspca_dev, STK1135_REG_SENSO +
0 ,
0 x10);
/* disable STOP clock */
reg_w(gspca_dev, STK1135_REG_SENSO +
1 ,
0 x00);
/* set lower 8 bits of PLL feedback divider */
reg_w(gspca_dev, STK1135_REG_SENSO +
3 ,
0 x07);
/* set other PLL parameters */
reg_w(gspca_dev, STK1135_REG_PLLFD,
0 x06);
/* enable timing generator */
reg_w(gspca_dev, STK1135_REG_TMGEN,
0 x80);
/* enable PLL */
reg_w(gspca_dev, STK1135_REG_SENSO +
2 ,
0 x04);
/* set serial interface clock divider (30MHz/0x1f*16+2) = 60240 kHz) */
reg_w(gspca_dev, STK1135_REG_SICTL +
2 ,
0 x1f);
/* wait a while for sensor to catch up */
udelay(
1000 );
}
static void stk1135_camera_disable(
struct gspca_dev *gspca_dev)
{
/* set capture end Y position to 0 */
reg_w(gspca_dev, STK1135_REG_CIEPO +
2 ,
0 x00);
reg_w(gspca_dev, STK1135_REG_CIEPO +
3 ,
0 x00);
/* disable capture */
reg_w_mask(gspca_dev, STK1135_REG_SCTRL,
0 x00,
0 x80);
/* enable sensor standby and diasble chip enable */
sensor_write_mask(gspca_dev,
0 x00d,
0 x0004,
0 x000c);
/* disable PLL */
reg_w_mask(gspca_dev, STK1135_REG_SENSO +
2 ,
0 x00,
0 x01);
/* disable timing generator */
reg_w(gspca_dev, STK1135_REG_TMGEN,
0 x00);
/* enable STOP clock */
reg_w(gspca_dev, STK1135_REG_SENSO +
1 ,
0 x20);
/* disable CLKOUT for sensor */
reg_w(gspca_dev, STK1135_REG_SENSO,
0 x00);
/* disable sensor (GPIO5) and enable GPIO0,3,6 (?) - sensor standby? */
reg_w(gspca_dev, STK1135_REG_GCTRL,
0 x49);
}
/* this function is called at probe and resume time */
static int sd_init(
struct gspca_dev *gspca_dev)
{
u16 sensor_id;
char *sensor_name;
struct sd *sd = (
struct sd *) gspca_dev;
/* set GPIO3,4,5,6 direction to output */
reg_w(gspca_dev, STK1135_REG_GCTRL +
2 ,
0 x78);
/* enable sensor (GPIO5) */
reg_w(gspca_dev, STK1135_REG_GCTRL, (
1 <<
5 ));
/* disable ROM interface */
reg_w(gspca_dev, STK1135_REG_GCTRL +
3 ,
0 x80);
/* enable interrupts from GPIO8 (flip sensor) and GPIO9 (???) */
reg_w(gspca_dev, STK1135_REG_ICTRL +
1 ,
0 x00);
reg_w(gspca_dev, STK1135_REG_ICTRL +
3 ,
0 x03);
/* enable remote wakeup from GPIO9 (???) */
reg_w(gspca_dev, STK1135_REG_RMCTL +
1 ,
0 x00);
reg_w(gspca_dev, STK1135_REG_RMCTL +
3 ,
0 x02);
/* reset serial interface */
reg_w(gspca_dev, STK1135_REG_SICTL,
0 x80);
reg_w(gspca_dev, STK1135_REG_SICTL,
0 x00);
/* set sensor address */
reg_w(gspca_dev, STK1135_REG_SICTL +
3 ,
0 xba);
/* disable alt 2-wire serial interface */
reg_w(gspca_dev, STK1135_REG_ASIC +
3 ,
0 x00);
stk1135_configure_clock(gspca_dev);
/* read sensor ID */
sd->sensor_page =
0 xff;
sensor_id = sensor_read(gspca_dev,
0 x000);
switch (sensor_id) {
case 0 x148c:
sensor_name =
"MT9M112" ;
break ;
default :
sensor_name =
"unknown" ;
}
pr_info(
"Detected sensor type %s (0x%x)\n" , sensor_name, sensor_id);
stk1135_camera_disable(gspca_dev);
return gspca_dev->usb_err;
}
/* -- start the camera -- */
static int sd_start(
struct gspca_dev *gspca_dev)
{
struct sd *sd = (
struct sd *) gspca_dev;
u16 width, height;
/* enable sensor (GPIO5) */
reg_w(gspca_dev, STK1135_REG_GCTRL, (
1 <<
5 ));
stk1135_configure_clock(gspca_dev);
/* set capture start position X = 0, Y = 0 */
reg_w(gspca_dev, STK1135_REG_CISPO +
0 ,
0 x00);
reg_w(gspca_dev, STK1135_REG_CISPO +
1 ,
0 x00);
reg_w(gspca_dev, STK1135_REG_CISPO +
2 ,
0 x00);
reg_w(gspca_dev, STK1135_REG_CISPO +
3 ,
0 x00);
/* set capture end position */
width = gspca_dev->pixfmt.width;
height = gspca_dev->pixfmt.height;
reg_w(gspca_dev, STK1135_REG_CIEPO +
0 , width &
0 xff);
reg_w(gspca_dev, STK1135_REG_CIEPO +
1 , width >>
8 );
reg_w(gspca_dev, STK1135_REG_CIEPO +
2 , height &
0 xff);
reg_w(gspca_dev, STK1135_REG_CIEPO +
3 , height >>
8 );
/* set 8-bit mode */
reg_w(gspca_dev, STK1135_REG_SCTRL,
0 x20);
stk1135_configure_mt9m112(gspca_dev);
/* enable capture */
reg_w_mask(gspca_dev, STK1135_REG_SCTRL,
0 x80,
0 x80);
if (gspca_dev->usb_err >=
0 )
gspca_dbg(gspca_dev, D_STREAM,
"camera started alt: 0x%02x\n" ,
gspca_dev->alt);
sd->pkt_seq =
0 ;
return gspca_dev->usb_err;
}
static void sd_stopN(
struct gspca_dev *gspca_dev)
{
struct usb_device *dev = gspca_dev->dev;
usb_set_interface(dev, gspca_dev->iface,
0 );
stk1135_camera_disable(gspca_dev);
gspca_dbg(gspca_dev, D_STREAM,
"camera stopped\n" );
}
static void sd_pkt_scan(
struct gspca_dev *gspca_dev,
u8 *data,
/* isoc packet */
int len)
/* iso packet length */
{
struct sd *sd = (
struct sd *) gspca_dev;
int skip =
sizeof (
struct stk1135_pkt_header);
bool flip;
enum gspca_packet_type pkt_type = INTER_PACKET;
struct stk1135_pkt_header *hdr = (
void *)data;
u8 seq;
if (len <
4 ) {
gspca_dbg(gspca_dev, D_PACK,
"received short packet (less than 4 bytes)\n" );
return ;
}
/* GPIO 8 is flip sensor (1 = normal position, 0 = flipped to back) */
flip = !(le16_to_cpu(hdr->gpio) & (
1 <<
8 ));
/* it's a switch, needs software debounce */
if (sd->flip_status != flip)
sd->flip_debounce++;
else
sd->flip_debounce =
0 ;
/* check sequence number (not present in new frame packets) */
if (!(hdr->flags & STK1135_HDR_FRAME_START)) {
seq = hdr->seq & STK1135_HDR_SEQ_MASK;
if (seq != sd->pkt_seq) {
gspca_dbg(gspca_dev, D_PACK,
"received out-of-sequence packet\n" );
/* resync sequence and discard packet */
sd->pkt_seq = seq;
gspca_dev->last_packet_type = DISCARD_PACKET;
return ;
}
}
sd->pkt_seq++;
if (sd->pkt_seq > STK1135_HDR_SEQ_MASK)
sd->pkt_seq =
0 ;
if (len ==
sizeof (
struct stk1135_pkt_header))
return ;
if (hdr->flags & STK1135_HDR_FRAME_START) {
/* new frame */
skip =
8 ;
/* the header is longer */
gspca_frame_add(gspca_dev, LAST_PACKET, data,
0 );
pkt_type = FIRST_PACKET;
}
gspca_frame_add(gspca_dev, pkt_type, data + skip, len - skip);
}
static void sethflip(
struct gspca_dev *gspca_dev, s32 val)
{
struct sd *sd = (
struct sd *) gspca_dev;
if (sd->flip_status)
val = !val;
sensor_write_mask(gspca_dev,
0 x020, val ?
0 x0002 :
0 x0000 ,
0 x0002);
}
static void setvflip(
struct gspca_dev *gspca_dev, s32 val)
{
struct sd *sd = (
struct sd *) gspca_dev;
if (sd->flip_status)
val = !val;
sensor_write_mask(gspca_dev,
0 x020, val ?
0 x0001 :
0 x0000 ,
0 x0001);
}
static void stk1135_dq_callback(
struct gspca_dev *gspca_dev)
{
struct sd *sd = (
struct sd *) gspca_dev;
if (sd->flip_debounce >
100 ) {
sd->flip_status = !sd->flip_status;
sethflip(gspca_dev, v4l2_ctrl_g_ctrl(sd->hflip));
setvflip(gspca_dev, v4l2_ctrl_g_ctrl(sd->vflip));
}
}
static int sd_s_ctrl(
struct v4l2_ctrl *ctrl)
{
struct gspca_dev *gspca_dev =
container_of(ctrl->handler,
struct gspca_dev, ctrl_handler);
gspca_dev->usb_err =
0 ;
if (!gspca_dev->streaming)
return 0 ;
switch (ctrl->id) {
case V4L2_CID_HFLIP:
sethflip(gspca_dev, ctrl->val);
break ;
case V4L2_CID_VFLIP:
setvflip(gspca_dev, ctrl->val);
break ;
}
return gspca_dev->usb_err;
}
static const struct v4l2_ctrl_ops sd_ctrl_ops = {
.s_ctrl = sd_s_ctrl,
};
static int sd_init_controls(
struct gspca_dev *gspca_dev)
{
struct sd *sd = (
struct sd *) gspca_dev;
struct v4l2_ctrl_handler *hdl = &gspca_dev->ctrl_handler;
gspca_dev->vdev.ctrl_handler = hdl;
v4l2_ctrl_handler_init(hdl,
2 );
sd->hflip = v4l2_ctrl_new_std(hdl, &sd_ctrl_ops,
V4L2_CID_HFLIP,
0 ,
1 ,
1 ,
0 );
sd->vflip = v4l2_ctrl_new_std(hdl, &sd_ctrl_ops,
V4L2_CID_VFLIP,
0 ,
1 ,
1 ,
0 );
if (hdl->error) {
pr_err(
"Could not initialize controls\n" );
return hdl->error;
}
return 0 ;
}
static void stk1135_try_fmt(
struct gspca_dev *gspca_dev,
struct v4l2_format *fmt)
{
fmt->fmt.pix.width = clamp(fmt->fmt.pix.width,
32 U,
1280 U);
fmt->fmt.pix.height = clamp(fmt->fmt.pix.height,
32 U,
1024 U);
/* round up to even numbers */
fmt->fmt.pix.width += (fmt->fmt.pix.width &
1 );
fmt->fmt.pix.height += (fmt->fmt.pix.height &
1 );
fmt->fmt.pix.bytesperline = fmt->fmt.pix.width;
fmt->fmt.pix.sizeimage = fmt->fmt.pix.width * fmt->fmt.pix.height;
}
static int stk1135_enum_framesizes(
struct gspca_dev *gspca_dev,
struct v4l2_frmsizeenum *fsize)
{
if (fsize->index !=
0 || fsize->pixel_format != V4L2_PIX_FMT_SBGGR8)
return -EINVAL;
fsize->type = V4L2_FRMSIZE_TYPE_STEPWISE;
fsize->stepwise.min_width =
32 ;
fsize->stepwise.min_height =
32 ;
fsize->stepwise.max_width =
1280 ;
fsize->stepwise.max_height =
1024 ;
fsize->stepwise.step_width =
2 ;
fsize->stepwise.step_height =
2 ;
return 0 ;
}
/* sub-driver description */
static const struct sd_desc sd_desc = {
.name = MODULE_NAME,
.config = sd_config,
.init = sd_init,
.init_controls = sd_init_controls,
.start = sd_start,
.stopN = sd_stopN,
.pkt_scan = sd_pkt_scan,
.dq_callback = stk1135_dq_callback,
.try_fmt = stk1135_try_fmt,
.enum_framesizes = stk1135_enum_framesizes,
};
/* -- module initialisation -- */
static const struct usb_device_id device_table[] = {
{USB_DEVICE(
0 x174f,
0 x6a31)},
/* ASUS laptop, MT9M112 sensor */
{}
};
MODULE_DEVICE_TABLE(usb, device_table);
/* -- device connect -- */
static int sd_probe(
struct usb_interface *intf,
const struct usb_device_id *id)
{
return gspca_dev_probe(intf, id, &sd_desc,
sizeof (
struct sd),
THIS_MODULE);
}
static struct usb_driver sd_driver = {
.name = MODULE_NAME,
.id_table = device_table,
.probe = sd_probe,
.disconnect = gspca_disconnect,
#ifdef CONFIG_PM
.suspend = gspca_suspend,
.resume = gspca_resume,
.reset_resume = gspca_resume,
#endif
};
module_usb_driver(sd_driver);
Messung V0.5 in Prozent C=94 H=94 G=93
¤ Dauer der Verarbeitung: 0.11 Sekunden
(vorverarbeitet am 2026-06-07)
¤
*© Formatika GbR, Deutschland