📄 tvp5150.c
字号:
tvp5150_read(c, TVP5150_LINE_MODE_REG_41)); printk("tvp5150: Line mode registers = 0x%02x\n", tvp5150_read(c, TVP5150_LINE_MODE_REG_42)); printk("tvp5150: Line mode registers = 0x%02x\n", tvp5150_read(c, TVP5150_LINE_MODE_REG_43)); printk("tvp5150: Line mode registers = 0x%02x\n", tvp5150_read(c, TVP5150_LINE_MODE_REG_44)); printk("tvp5150: Full field mode register = 0x%02x\n", tvp5150_read(c, TVP5150_FULL_FIELD_MODE_REG));}/**************************************************************************** Basic functions ****************************************************************************/enum tvp5150_input { TVP5150_ANALOG_CH0 = 0, TVP5150_SVIDEO = 1, TVP5150_ANALOG_CH1 = 2, TVP5150_BLACK_SCREEN = 8};static inline void tvp5150_selmux(struct i2c_client *c, enum tvp5150_input input){ struct tvp5150 *decoder = i2c_get_clientdata(c); if (!decoder->enable) input |= TVP5150_BLACK_SCREEN; tvp5150_write(c, TVP5150_VD_IN_SRC_SEL_1, input);};static inline void tvp5150_reset(struct i2c_client *c){ struct tvp5150 *decoder = i2c_get_clientdata(c); tvp5150_write(c, TVP5150_CONF_SHARED_PIN, 2); /* Automatic offset and AGC enabled */ tvp5150_write(c, TVP5150_ANAL_CHL_CTL, 0x15); /* Normal Operation */// tvp5150_write(c, TVP5150_OP_MODE_CTL, 0x00); /* Activate YCrCb output 0x9 or 0xd ? */ tvp5150_write(c, TVP5150_MISC_CTL, 0x6f); /* Activates video std autodetection for all standards */ tvp5150_write(c, TVP5150_AUTOSW_MSK, 0x0); /* Default format: 0x47, 4:2:2: 0x40 */ tvp5150_write(c, TVP5150_DATA_RATE_SEL, 0x47); tvp5150_selmux(c, decoder->input); tvp5150_write(c, TVP5150_CHROMA_PROC_CTL_1, 0x0c); tvp5150_write(c, TVP5150_CHROMA_PROC_CTL_2, 0x54); tvp5150_write(c, 0x27, 0x20); /* ?????????? */ tvp5150_write(c, TVP5150_VIDEO_STD, 0x0); /* Auto switch */ tvp5150_write(c, TVP5150_BRIGHT_CTL, decoder->bright >> 8); tvp5150_write(c, TVP5150_CONTRAST_CTL, decoder->contrast >> 8); tvp5150_write(c, TVP5150_SATURATION_CTL, decoder->contrast >> 8); tvp5150_write(c, TVP5150_HUE_CTL, (decoder->hue - 32768) >> 8);};static int tvp5150_get_ctrl(struct i2c_client *c, struct v4l2_control *ctrl){/* struct tvp5150 *decoder = i2c_get_clientdata(c); */ switch (ctrl->id) { case V4L2_CID_BRIGHTNESS: ctrl->value = tvp5150_read(c, TVP5150_BRIGHT_CTL); return 0; case V4L2_CID_CONTRAST: ctrl->value = tvp5150_read(c, TVP5150_CONTRAST_CTL); return 0; case V4L2_CID_SATURATION: ctrl->value = tvp5150_read(c, TVP5150_SATURATION_CTL); return 0; case V4L2_CID_HUE: ctrl->value = tvp5150_read(c, TVP5150_HUE_CTL); return 0; default: return -EINVAL; }}static int tvp5150_set_ctrl(struct i2c_client *c, struct v4l2_control *ctrl){/* struct tvp5150 *decoder = i2c_get_clientdata(c); */ switch (ctrl->id) { case V4L2_CID_BRIGHTNESS: tvp5150_write(c, TVP5150_BRIGHT_CTL, ctrl->value); return 0; case V4L2_CID_CONTRAST: tvp5150_write(c, TVP5150_CONTRAST_CTL, ctrl->value); return 0; case V4L2_CID_SATURATION: tvp5150_write(c, TVP5150_SATURATION_CTL, ctrl->value); return 0; case V4L2_CID_HUE: tvp5150_write(c, TVP5150_HUE_CTL, ctrl->value); return 0; default: return -EINVAL; }}/**************************************************************************** I2C Command ****************************************************************************/static int tvp5150_command(struct i2c_client *client, unsigned int cmd, void *arg){ struct tvp5150 *decoder = i2c_get_clientdata(client); switch (cmd) { case 0: case DECODER_INIT: tvp5150_reset(client); break; case DECODER_DUMP: dump_reg(client); break; case DECODER_GET_CAPABILITIES: { struct video_decoder_capability *cap = arg; cap->flags = VIDEO_DECODER_PAL | VIDEO_DECODER_NTSC | VIDEO_DECODER_SECAM | VIDEO_DECODER_AUTO | VIDEO_DECODER_CCIR; cap->inputs = 3; cap->outputs = 1; break; } case DECODER_GET_STATUS: { break; } case DECODER_SET_GPIO: break; case DECODER_SET_VBI_BYPASS: break; case DECODER_SET_NORM: { int *iarg = arg; switch (*iarg) { case VIDEO_MODE_NTSC: break; case VIDEO_MODE_PAL: break; case VIDEO_MODE_SECAM: break; case VIDEO_MODE_AUTO: break; default: return -EINVAL; } decoder->norm = *iarg; break; } case DECODER_SET_INPUT: { int *iarg = arg; if (*iarg < 0 || *iarg > 3) { return -EINVAL; } decoder->input = *iarg; tvp5150_selmux(client, decoder->input); break; } case DECODER_SET_OUTPUT: { int *iarg = arg; /* not much choice of outputs */ if (*iarg != 0) { return -EINVAL; } break; } case DECODER_ENABLE_OUTPUT: { int *iarg = arg; decoder->enable = (*iarg != 0); tvp5150_selmux(client, decoder->input); break; } case VIDIOC_QUERYCTRL: { struct v4l2_queryctrl *qc = arg; u8 i, n; dprintk(1, KERN_DEBUG "VIDIOC_QUERYCTRL"); n = sizeof(tvp5150_qctrl) / sizeof(tvp5150_qctrl[0]); for (i = 0; i < n; i++) if (qc->id && qc->id == tvp5150_qctrl[i].id) { memcpy(qc, &(tvp5150_qctrl[i]), sizeof(*qc)); return 0; } return -EINVAL; } case VIDIOC_G_CTRL: { struct v4l2_control *ctrl = arg; dprintk(1, KERN_DEBUG "VIDIOC_G_CTRL"); return tvp5150_get_ctrl(client, ctrl); } case VIDIOC_S_CTRL_OLD: /* ??? */ case VIDIOC_S_CTRL: { struct v4l2_control *ctrl = arg; u8 i, n; dprintk(1, KERN_DEBUG "VIDIOC_S_CTRL"); n = sizeof(tvp5150_qctrl) / sizeof(tvp5150_qctrl[0]); for (i = 0; i < n; i++) if (ctrl->id == tvp5150_qctrl[i].id) { if (ctrl->value < tvp5150_qctrl[i].minimum || ctrl->value > tvp5150_qctrl[i].maximum) return -ERANGE; dprintk(1, KERN_DEBUG "VIDIOC_S_CTRL: id=%d, value=%d", ctrl->id, ctrl->value); return tvp5150_set_ctrl(client, ctrl); } return -EINVAL; } case DECODER_SET_PICTURE: { struct video_picture *pic = arg; if (decoder->bright != pic->brightness) { /* We want 0 to 255 we get 0-65535 */ decoder->bright = pic->brightness; tvp5150_write(client, TVP5150_BRIGHT_CTL, decoder->bright >> 8); } if (decoder->contrast != pic->contrast) { /* We want 0 to 255 we get 0-65535 */ decoder->contrast = pic->contrast; tvp5150_write(client, TVP5150_CONTRAST_CTL, decoder->contrast >> 8); } if (decoder->sat != pic->colour) { /* We want 0 to 255 we get 0-65535 */ decoder->sat = pic->colour; tvp5150_write(client, TVP5150_SATURATION_CTL, decoder->contrast >> 8); } if (decoder->hue != pic->hue) { /* We want -128 to 127 we get 0-65535 */ decoder->hue = pic->hue; tvp5150_write(client, TVP5150_HUE_CTL, (decoder->hue - 32768) >> 8); } break; } default: return -EINVAL; } return 0;}/**************************************************************************** I2C Client & Driver ****************************************************************************/static struct i2c_driver driver;static struct i2c_client client_template = { .name = "(unset)", .flags = I2C_CLIENT_ALLOW_USE, .driver = &driver,};static int tvp5150_detect_client(struct i2c_adapter *adapter, int address, int kind){ struct i2c_client *client; struct tvp5150 *core; int rv; dprintk(1, KERN_INFO "tvp5150.c: detecting tvp5150 client on address 0x%x\n", address << 1); client_template.adapter = adapter; client_template.addr = address; /* Check if the adapter supports the needed features */ if (!i2c_check_functionality (adapter, I2C_FUNC_SMBUS_READ_BYTE | I2C_FUNC_SMBUS_WRITE_BYTE_DATA)) return 0; client = kmalloc(sizeof(struct i2c_client), GFP_KERNEL); if (client == 0) return -ENOMEM; memcpy(client, &client_template, sizeof(struct i2c_client)); core = kmalloc(sizeof(struct tvp5150), GFP_KERNEL); if (core == 0) { kfree(client); return -ENOMEM; } memset(core, 0, sizeof(struct tvp5150)); i2c_set_clientdata(client, core); rv = i2c_attach_client(client); core->norm = VIDEO_MODE_AUTO; core->input = 2; core->enable = 1; core->bright = 32768; core->contrast = 32768; core->hue = 32768; core->sat = 32768; if (rv) { kfree(client); kfree(core); return rv; } if (debug > 1) dump_reg(client); return 0;}static int tvp5150_attach_adapter(struct i2c_adapter *adapter){ dprintk(1, KERN_INFO "tvp5150.c: starting probe for adapter %s (0x%x)\n", adapter->name, adapter->id); return i2c_probe(adapter, &addr_data, &tvp5150_detect_client);}static int tvp5150_detach_client(struct i2c_client *client){ struct tvp5150 *decoder = i2c_get_clientdata(client); int err; err = i2c_detach_client(client); if (err) { return err; } kfree(decoder); kfree(client); return 0;}/* ----------------------------------------------------------------------- */static struct i2c_driver driver = { .owner = THIS_MODULE, .name = "tvp5150", /* FIXME */ .id = I2C_DRIVERID_SAA7110, .flags = I2C_DF_NOTIFY, .attach_adapter = tvp5150_attach_adapter, .detach_client = tvp5150_detach_client, .command = tvp5150_command,};static int __init tvp5150_init(void){ return i2c_add_driver(&driver);}static void __exit tvp5150_exit(void){ i2c_del_driver(&driver);}module_init(tvp5150_init);module_exit(tvp5150_exit);
⌨️ 快捷键说明
复制代码
Ctrl + C
搜索代码
Ctrl + F
全屏模式
F11
切换主题
Ctrl + Shift + D
显示快捷键
?
增大字号
Ctrl + =
减小字号
Ctrl + -