Message ID | 20190605204427.28333-1-brad@nextdimension.cc (mailing list archive) |
---|---|
State | New, archived |
Headers | show |
Series | None | expand |
On Wed, Jun 05, 2019 at 03:44:27PM -0500, Brad Love wrote: > Includes support to identify and use two Hauppauge device. > HVR-1955: > - LGDT3306a ATSC/QAM demod > - si2177 tuner > - cx25840 decoder for analog tv/composite/s-video/audio > > HVR-1975 dual-frontend: > - LGDT3306a ATSC/QAM demod > - si2168 DVB-C/T/T2 demod > - si2177 tuner > - cx25840 decoder for analog tv/composite/s-video/audio > > Signed-off-by: Brad Love <brad@nextdimension.cc> Reviewed-by: Sean Young <sean@mess.org> > --- > Since v6: > - back off one checkpatch strict fix, causes discarded const compiler warning > Since v5: > - le16_to_cpu fix > Since v4: > - checkpatch strict fixes > - change !0 initializations > Since v2: > - Fix build with VIDEO_PVRUSB2_DVB enabled > Changes since v1: > - Fix build with VIDEO_PVRUSB2_DVB disabled > - Insert 160xxx code lower, so 75xxx profile is not split > - Reorganize 160xxx board profile > - Share config where possible > > drivers/media/usb/pvrusb2/Kconfig | 2 + > .../media/usb/pvrusb2/pvrusb2-cx2584x-v4l.c | 25 +++ > drivers/media/usb/pvrusb2/pvrusb2-devattr.c | 166 ++++++++++++++++++ > drivers/media/usb/pvrusb2/pvrusb2-devattr.h | 1 + > drivers/media/usb/pvrusb2/pvrusb2-fx2-cmd.h | 4 + > drivers/media/usb/pvrusb2/pvrusb2-hdw.c | 36 +++- > 6 files changed, 233 insertions(+), 1 deletion(-) > > diff --git a/drivers/media/usb/pvrusb2/Kconfig b/drivers/media/usb/pvrusb2/Kconfig > index 64f9df067269..e6a4f730591b 100644 > --- a/drivers/media/usb/pvrusb2/Kconfig > +++ b/drivers/media/usb/pvrusb2/Kconfig > @@ -41,6 +41,8 @@ config VIDEO_PVRUSB2_DVB > select DVB_S5H1409 if MEDIA_SUBDRV_AUTOSELECT > select DVB_S5H1411 if MEDIA_SUBDRV_AUTOSELECT > select DVB_TDA10048 if MEDIA_SUBDRV_AUTOSELECT > + select DVB_LGDT3306A if MEDIA_SUBDRV_AUTOSELECT > + select DVB_SI2168 if MEDIA_SUBDRV_AUTOSELECT > select MEDIA_TUNER_TDA18271 if MEDIA_SUBDRV_AUTOSELECT > select MEDIA_TUNER_SIMPLE if MEDIA_SUBDRV_AUTOSELECT > select MEDIA_TUNER_TDA8290 if MEDIA_SUBDRV_AUTOSELECT > diff --git a/drivers/media/usb/pvrusb2/pvrusb2-cx2584x-v4l.c b/drivers/media/usb/pvrusb2/pvrusb2-cx2584x-v4l.c > index d5bec0f69bec..36016ab3aef0 100644 > --- a/drivers/media/usb/pvrusb2/pvrusb2-cx2584x-v4l.c > +++ b/drivers/media/usb/pvrusb2/pvrusb2-cx2584x-v4l.c > @@ -111,10 +111,35 @@ static const struct routing_scheme routing_defav400 = { > .cnt = ARRAY_SIZE(routing_schemeav400), > }; > > +static const struct routing_scheme_item routing_scheme160xxx[] = { > + [PVR2_CVAL_INPUT_TV] = { > + .vid = CX25840_COMPOSITE7, > + .aud = CX25840_AUDIO8, > + }, > + [PVR2_CVAL_INPUT_RADIO] = { > + .vid = CX25840_COMPOSITE4, > + .aud = CX25840_AUDIO6, > + }, > + [PVR2_CVAL_INPUT_COMPOSITE] = { > + .vid = CX25840_COMPOSITE3, > + .aud = CX25840_AUDIO_SERIAL, > + }, > + [PVR2_CVAL_INPUT_SVIDEO] = { > + .vid = CX25840_SVIDEO1, > + .aud = CX25840_AUDIO_SERIAL, > + }, > +}; > + > +static const struct routing_scheme routing_def160xxx = { > + .def = routing_scheme160xxx, > + .cnt = ARRAY_SIZE(routing_scheme160xxx), > +}; > + > static const struct routing_scheme *routing_schemes[] = { > [PVR2_ROUTING_SCHEME_HAUPPAUGE] = &routing_def0, > [PVR2_ROUTING_SCHEME_GOTVIEW] = &routing_defgv, > [PVR2_ROUTING_SCHEME_AV400] = &routing_defav400, > + [PVR2_ROUTING_SCHEME_HAUP160XXX] = &routing_def160xxx, > }; > > void pvr2_cx25840_subdev_update(struct pvr2_hdw *hdw, struct v4l2_subdev *sd) > diff --git a/drivers/media/usb/pvrusb2/pvrusb2-devattr.c b/drivers/media/usb/pvrusb2/pvrusb2-devattr.c > index a4a27e071c6d..ef57a6329863 100644 > --- a/drivers/media/usb/pvrusb2/pvrusb2-devattr.c > +++ b/drivers/media/usb/pvrusb2/pvrusb2-devattr.c > @@ -37,6 +37,9 @@ pvr2_device_desc structures. > #include "tda18271.h" > #include "tda8290.h" > #include "tuner-simple.h" > +#include "si2157.h" > +#include "lgdt3306a.h" > +#include "si2168.h" > #endif > > > @@ -525,7 +528,166 @@ static const struct pvr2_device_desc pvr2_device_751xx = { > #endif > }; > > +/*------------------------------------------------------------------------*/ > +/* Hauppauge PVR-USB2 Model 160000 / 160111 -- HVR-1955 / HVR-1975 */ > + > +#ifdef CONFIG_VIDEO_PVRUSB2_DVB > +static int pvr2_si2157_attach(struct pvr2_dvb_adapter *adap); > +static int pvr2_si2168_attach(struct pvr2_dvb_adapter *adap); > +static int pvr2_dual_fe_attach(struct pvr2_dvb_adapter *adap); > +static int pvr2_lgdt3306a_attach(struct pvr2_dvb_adapter *adap); > + > +static const struct pvr2_dvb_props pvr2_160000_dvb_props = { > + .frontend_attach = pvr2_dual_fe_attach, > + .tuner_attach = pvr2_si2157_attach, > +}; > + > +static const struct pvr2_dvb_props pvr2_160111_dvb_props = { > + .frontend_attach = pvr2_lgdt3306a_attach, > + .tuner_attach = pvr2_si2157_attach, > +}; > + > +static int pvr2_si2157_attach(struct pvr2_dvb_adapter *adap) > +{ > + struct si2157_config si2157_config = {}; > + > + si2157_config.inversion = 1; > + si2157_config.fe = adap->fe[0]; > + > + adap->i2c_client_tuner = dvb_module_probe("si2157", "si2177", > + &adap->channel.hdw->i2c_adap, > + 0x60, &si2157_config); > + > + if (!adap->i2c_client_tuner) > + return -ENODEV; > + > + return 0; > +} > + > +static int pvr2_si2168_attach(struct pvr2_dvb_adapter *adap) > +{ > + struct si2168_config si2168_config = {}; > + struct i2c_adapter *adapter; > + > + pr_debug("%s()\n", __func__); > + > + si2168_config.fe = &adap->fe[1]; > + si2168_config.i2c_adapter = &adapter; > + si2168_config.ts_mode = SI2168_TS_PARALLEL; /*2, 1-serial, 2-parallel.*/ > + si2168_config.ts_clock_gapped = 1; /*0-disabled, 1-enabled.*/ > + si2168_config.ts_clock_inv = 0; /*0-not-invert, 1-invert*/ > + si2168_config.spectral_inversion = 1; /*0-not-invert, 1-invert*/ > + > + adap->i2c_client_demod[1] = dvb_module_probe("si2168", NULL, > + &adap->channel.hdw->i2c_adap, > + 0x64, &si2168_config); > + > + if (!adap->i2c_client_demod[1]) > + return -ENODEV; > + > + return 0; > +} > > +static int pvr2_lgdt3306a_attach(struct pvr2_dvb_adapter *adap) > +{ > + struct lgdt3306a_config lgdt3306a_config; > + struct i2c_adapter *adapter; > + > + pr_debug("%s()\n", __func__); > + > + lgdt3306a_config.fe = &adap->fe[0]; > + lgdt3306a_config.i2c_adapter = &adapter; > + lgdt3306a_config.deny_i2c_rptr = 1; > + lgdt3306a_config.spectral_inversion = 1; > + lgdt3306a_config.qam_if_khz = 4000; > + lgdt3306a_config.vsb_if_khz = 3250; > + lgdt3306a_config.mpeg_mode = LGDT3306A_MPEG_PARALLEL; > + lgdt3306a_config.tpclk_edge = LGDT3306A_TPCLK_FALLING_EDGE; > + lgdt3306a_config.tpvalid_polarity = LGDT3306A_TP_VALID_LOW; > + lgdt3306a_config.xtalMHz = 25, /* demod clock MHz; 24/25 supported */ > + > + adap->i2c_client_demod[0] = dvb_module_probe("lgdt3306a", NULL, > + &adap->channel.hdw->i2c_adap, > + 0x59, &lgdt3306a_config); > + > + if (!adap->i2c_client_demod[0]) > + return -ENODEV; > + > + return 0; > +} > + > +static int pvr2_dual_fe_attach(struct pvr2_dvb_adapter *adap) > +{ > + pr_debug("%s()\n", __func__); > + > + if (pvr2_lgdt3306a_attach(adap) != 0) > + return -ENODEV; > + > + if (pvr2_si2168_attach(adap) != 0) { > + dvb_module_release(adap->i2c_client_demod[0]); > + return -ENODEV; > + } > + > + return 0; > +} > +#endif > + > +#define PVR2_FIRMWARE_160xxx "v4l-pvrusb2-160xxx-01.fw" > +static const char *pvr2_fw1_names_160xxx[] = { > + PVR2_FIRMWARE_160xxx, > +}; > + > +static const struct pvr2_device_client_desc pvr2_cli_160xxx[] = { > + { .module_id = PVR2_CLIENT_ID_CX25840 }, > +}; > + > +static const struct pvr2_device_desc pvr2_device_160000 = { > + .description = "WinTV HVR-1975 Model 160000", > + .shortname = "160000", > + .client_table.lst = pvr2_cli_160xxx, > + .client_table.cnt = ARRAY_SIZE(pvr2_cli_160xxx), > + .fx2_firmware.lst = pvr2_fw1_names_160xxx, > + .fx2_firmware.cnt = ARRAY_SIZE(pvr2_fw1_names_160xxx), > + .default_tuner_type = TUNER_ABSENT, > + .flag_has_cx25840 = 1, > + .flag_has_hauppauge_rom = 1, > + .flag_has_analogtuner = 1, > + .flag_has_composite = 1, > + .flag_has_svideo = 1, > + .flag_fx2_16kb = 1, > + .signal_routing_scheme = PVR2_ROUTING_SCHEME_HAUPPAUGE, > + .digital_control_scheme = PVR2_DIGITAL_SCHEME_HAUPPAUGE, > + .default_std_mask = V4L2_STD_NTSC_M, > + .led_scheme = PVR2_LED_SCHEME_HAUPPAUGE, > + .ir_scheme = PVR2_IR_SCHEME_ZILOG, > +#ifdef CONFIG_VIDEO_PVRUSB2_DVB > + .dvb_props = &pvr2_160000_dvb_props, > +#endif > +}; > + > +static const struct pvr2_device_desc pvr2_device_160111 = { > + .description = "WinTV HVR-1955 Model 160111", > + .shortname = "160111", > + .client_table.lst = pvr2_cli_160xxx, > + .client_table.cnt = ARRAY_SIZE(pvr2_cli_160xxx), > + .fx2_firmware.lst = pvr2_fw1_names_160xxx, > + .fx2_firmware.cnt = ARRAY_SIZE(pvr2_fw1_names_160xxx), > + .default_tuner_type = TUNER_ABSENT, > + .flag_has_cx25840 = 1, > + .flag_has_hauppauge_rom = 1, > + .flag_has_analogtuner = 1, > + .flag_has_composite = 1, > + .flag_has_svideo = 1, > + .flag_fx2_16kb = 1, > + .signal_routing_scheme = PVR2_ROUTING_SCHEME_HAUPPAUGE, > + .digital_control_scheme = PVR2_DIGITAL_SCHEME_HAUPPAUGE, > + .default_std_mask = V4L2_STD_NTSC_M, > + .led_scheme = PVR2_LED_SCHEME_HAUPPAUGE, > + .ir_scheme = PVR2_IR_SCHEME_ZILOG, > +#ifdef CONFIG_VIDEO_PVRUSB2_DVB > + .dvb_props = &pvr2_160111_dvb_props, > +#endif > +}; > > /*------------------------------------------------------------------------*/ > > @@ -552,6 +714,10 @@ struct usb_device_id pvr2_device_table[] = { > .driver_info = (kernel_ulong_t)&pvr2_device_751xx}, > { USB_DEVICE(0x0ccd, 0x0039), > .driver_info = (kernel_ulong_t)&pvr2_device_av400}, > + { USB_DEVICE(0x2040, 0x7502), > + .driver_info = (kernel_ulong_t)&pvr2_device_160111}, > + { USB_DEVICE(0x2040, 0x7510), > + .driver_info = (kernel_ulong_t)&pvr2_device_160000}, > { } > }; > > diff --git a/drivers/media/usb/pvrusb2/pvrusb2-devattr.h b/drivers/media/usb/pvrusb2/pvrusb2-devattr.h > index c1e7d4822cd1..ea0b2bf429e9 100644 > --- a/drivers/media/usb/pvrusb2/pvrusb2-devattr.h > +++ b/drivers/media/usb/pvrusb2/pvrusb2-devattr.h > @@ -66,6 +66,7 @@ struct pvr2_string_table { > #define PVR2_ROUTING_SCHEME_GOTVIEW 1 > #define PVR2_ROUTING_SCHEME_ONAIR 2 > #define PVR2_ROUTING_SCHEME_AV400 3 > +#define PVR2_ROUTING_SCHEME_HAUP160XXX 4 > > #define PVR2_DIGITAL_SCHEME_NONE 0 > #define PVR2_DIGITAL_SCHEME_HAUPPAUGE 1 > diff --git a/drivers/media/usb/pvrusb2/pvrusb2-fx2-cmd.h b/drivers/media/usb/pvrusb2/pvrusb2-fx2-cmd.h > index 0a01de4e54db..640b033815ec 100644 > --- a/drivers/media/usb/pvrusb2/pvrusb2-fx2-cmd.h > +++ b/drivers/media/usb/pvrusb2/pvrusb2-fx2-cmd.h > @@ -38,6 +38,10 @@ > > #define FX2CMD_FWPOST1 0x52u > > +/* These 2 only exist on Model 160xxx */ > +#define FX2CMD_HCW_DEMOD_RESET_PIN 0xd4u > +#define FX2CMD_HCW_MAKO_SLEEP_PIN 0xd5u > + > #define FX2CMD_POWER_OFF 0xdcu > #define FX2CMD_POWER_ON 0xdeu > > diff --git a/drivers/media/usb/pvrusb2/pvrusb2-hdw.c b/drivers/media/usb/pvrusb2/pvrusb2-hdw.c > index 191439109788..957913146e88 100644 > --- a/drivers/media/usb/pvrusb2/pvrusb2-hdw.c > +++ b/drivers/media/usb/pvrusb2/pvrusb2-hdw.c > @@ -316,6 +316,8 @@ static const struct pvr2_fx2cmd_descdef pvr2_fx2cmd_desc[] = { > {FX2CMD_ONAIR_DTV_STREAMING_OFF, "onair dtv stream off"}, > {FX2CMD_ONAIR_DTV_POWER_ON, "onair dtv power on"}, > {FX2CMD_ONAIR_DTV_POWER_OFF, "onair dtv power off"}, > + {FX2CMD_HCW_DEMOD_RESET_PIN, "hcw demod reset pin"}, > + {FX2CMD_HCW_MAKO_SLEEP_PIN, "hcw mako sleep pin"}, > }; > > > @@ -2139,10 +2141,28 @@ static void pvr2_hdw_setup_low(struct pvr2_hdw *hdw) > ((0) << 16)); > } > > - // This step MUST happen after the earlier powerup step. > + /* This step MUST happen after the earlier powerup step */ > pvr2_i2c_core_init(hdw); > if (!pvr2_hdw_dev_ok(hdw)) return; > > + /* Reset demod only on Hauppauge 160xxx platform */ > + if (le16_to_cpu(hdw->usb_dev->descriptor.idVendor) == 0x2040 && > + (le16_to_cpu(hdw->usb_dev->descriptor.idProduct) == 0x7502 || > + le16_to_cpu(hdw->usb_dev->descriptor.idProduct) == 0x7510)) { > + pr_info("%s(): resetting 160xxx demod\n", __func__); > + /* TODO: not sure this is proper place to reset once only */ > + pvr2_issue_simple_cmd(hdw, > + FX2CMD_HCW_DEMOD_RESET_PIN | > + (1 << 8) | > + ((0) << 16)); > + usleep_range(10000, 10500); > + pvr2_issue_simple_cmd(hdw, > + FX2CMD_HCW_DEMOD_RESET_PIN | > + (1 << 8) | > + ((1) << 16)); > + usleep_range(10000, 10500); > + } > + > pvr2_hdw_load_modules(hdw); > if (!pvr2_hdw_dev_ok(hdw)) return; > > @@ -4012,6 +4032,20 @@ int pvr2_hdw_cmd_decoder_reset(struct pvr2_hdw *hdw) > static int pvr2_hdw_cmd_hcw_demod_reset(struct pvr2_hdw *hdw, int onoff) > { > hdw->flag_ok = !0; > + > + /* Use this for Hauppauge 160xxx only */ > + if (le16_to_cpu(hdw->usb_dev->descriptor.idVendor) == 0x2040 && > + (le16_to_cpu(hdw->usb_dev->descriptor.idProduct) == 0x7502 || > + le16_to_cpu(hdw->usb_dev->descriptor.idProduct) == 0x7510)) { > + pr_debug("%s(): resetting demod on Hauppauge 160xxx platform skipped\n", > + __func__); > + /* Can't reset 160xxx or it will trash Demod tristate */ > + return pvr2_issue_simple_cmd(hdw, > + FX2CMD_HCW_MAKO_SLEEP_PIN | > + (1 << 8) | > + ((onoff ? 1 : 0) << 16)); > + } > + > return pvr2_issue_simple_cmd(hdw, > FX2CMD_HCW_DEMOD_RESETIN | > (1 << 8) | > -- > 2.21.0
diff --git a/drivers/media/usb/pvrusb2/Kconfig b/drivers/media/usb/pvrusb2/Kconfig index 64f9df067269..e6a4f730591b 100644 --- a/drivers/media/usb/pvrusb2/Kconfig +++ b/drivers/media/usb/pvrusb2/Kconfig @@ -41,6 +41,8 @@ config VIDEO_PVRUSB2_DVB select DVB_S5H1409 if MEDIA_SUBDRV_AUTOSELECT select DVB_S5H1411 if MEDIA_SUBDRV_AUTOSELECT select DVB_TDA10048 if MEDIA_SUBDRV_AUTOSELECT + select DVB_LGDT3306A if MEDIA_SUBDRV_AUTOSELECT + select DVB_SI2168 if MEDIA_SUBDRV_AUTOSELECT select MEDIA_TUNER_TDA18271 if MEDIA_SUBDRV_AUTOSELECT select MEDIA_TUNER_SIMPLE if MEDIA_SUBDRV_AUTOSELECT select MEDIA_TUNER_TDA8290 if MEDIA_SUBDRV_AUTOSELECT diff --git a/drivers/media/usb/pvrusb2/pvrusb2-cx2584x-v4l.c b/drivers/media/usb/pvrusb2/pvrusb2-cx2584x-v4l.c index d5bec0f69bec..36016ab3aef0 100644 --- a/drivers/media/usb/pvrusb2/pvrusb2-cx2584x-v4l.c +++ b/drivers/media/usb/pvrusb2/pvrusb2-cx2584x-v4l.c @@ -111,10 +111,35 @@ static const struct routing_scheme routing_defav400 = { .cnt = ARRAY_SIZE(routing_schemeav400), }; +static const struct routing_scheme_item routing_scheme160xxx[] = { + [PVR2_CVAL_INPUT_TV] = { + .vid = CX25840_COMPOSITE7, + .aud = CX25840_AUDIO8, + }, + [PVR2_CVAL_INPUT_RADIO] = { + .vid = CX25840_COMPOSITE4, + .aud = CX25840_AUDIO6, + }, + [PVR2_CVAL_INPUT_COMPOSITE] = { + .vid = CX25840_COMPOSITE3, + .aud = CX25840_AUDIO_SERIAL, + }, + [PVR2_CVAL_INPUT_SVIDEO] = { + .vid = CX25840_SVIDEO1, + .aud = CX25840_AUDIO_SERIAL, + }, +}; + +static const struct routing_scheme routing_def160xxx = { + .def = routing_scheme160xxx, + .cnt = ARRAY_SIZE(routing_scheme160xxx), +}; + static const struct routing_scheme *routing_schemes[] = { [PVR2_ROUTING_SCHEME_HAUPPAUGE] = &routing_def0, [PVR2_ROUTING_SCHEME_GOTVIEW] = &routing_defgv, [PVR2_ROUTING_SCHEME_AV400] = &routing_defav400, + [PVR2_ROUTING_SCHEME_HAUP160XXX] = &routing_def160xxx, }; void pvr2_cx25840_subdev_update(struct pvr2_hdw *hdw, struct v4l2_subdev *sd) diff --git a/drivers/media/usb/pvrusb2/pvrusb2-devattr.c b/drivers/media/usb/pvrusb2/pvrusb2-devattr.c index a4a27e071c6d..ef57a6329863 100644 --- a/drivers/media/usb/pvrusb2/pvrusb2-devattr.c +++ b/drivers/media/usb/pvrusb2/pvrusb2-devattr.c @@ -37,6 +37,9 @@ pvr2_device_desc structures. #include "tda18271.h" #include "tda8290.h" #include "tuner-simple.h" +#include "si2157.h" +#include "lgdt3306a.h" +#include "si2168.h" #endif @@ -525,7 +528,166 @@ static const struct pvr2_device_desc pvr2_device_751xx = { #endif }; +/*------------------------------------------------------------------------*/ +/* Hauppauge PVR-USB2 Model 160000 / 160111 -- HVR-1955 / HVR-1975 */ + +#ifdef CONFIG_VIDEO_PVRUSB2_DVB +static int pvr2_si2157_attach(struct pvr2_dvb_adapter *adap); +static int pvr2_si2168_attach(struct pvr2_dvb_adapter *adap); +static int pvr2_dual_fe_attach(struct pvr2_dvb_adapter *adap); +static int pvr2_lgdt3306a_attach(struct pvr2_dvb_adapter *adap); + +static const struct pvr2_dvb_props pvr2_160000_dvb_props = { + .frontend_attach = pvr2_dual_fe_attach, + .tuner_attach = pvr2_si2157_attach, +}; + +static const struct pvr2_dvb_props pvr2_160111_dvb_props = { + .frontend_attach = pvr2_lgdt3306a_attach, + .tuner_attach = pvr2_si2157_attach, +}; + +static int pvr2_si2157_attach(struct pvr2_dvb_adapter *adap) +{ + struct si2157_config si2157_config = {}; + + si2157_config.inversion = 1; + si2157_config.fe = adap->fe[0]; + + adap->i2c_client_tuner = dvb_module_probe("si2157", "si2177", + &adap->channel.hdw->i2c_adap, + 0x60, &si2157_config); + + if (!adap->i2c_client_tuner) + return -ENODEV; + + return 0; +} + +static int pvr2_si2168_attach(struct pvr2_dvb_adapter *adap) +{ + struct si2168_config si2168_config = {}; + struct i2c_adapter *adapter; + + pr_debug("%s()\n", __func__); + + si2168_config.fe = &adap->fe[1]; + si2168_config.i2c_adapter = &adapter; + si2168_config.ts_mode = SI2168_TS_PARALLEL; /*2, 1-serial, 2-parallel.*/ + si2168_config.ts_clock_gapped = 1; /*0-disabled, 1-enabled.*/ + si2168_config.ts_clock_inv = 0; /*0-not-invert, 1-invert*/ + si2168_config.spectral_inversion = 1; /*0-not-invert, 1-invert*/ + + adap->i2c_client_demod[1] = dvb_module_probe("si2168", NULL, + &adap->channel.hdw->i2c_adap, + 0x64, &si2168_config); + + if (!adap->i2c_client_demod[1]) + return -ENODEV; + + return 0; +} +static int pvr2_lgdt3306a_attach(struct pvr2_dvb_adapter *adap) +{ + struct lgdt3306a_config lgdt3306a_config; + struct i2c_adapter *adapter; + + pr_debug("%s()\n", __func__); + + lgdt3306a_config.fe = &adap->fe[0]; + lgdt3306a_config.i2c_adapter = &adapter; + lgdt3306a_config.deny_i2c_rptr = 1; + lgdt3306a_config.spectral_inversion = 1; + lgdt3306a_config.qam_if_khz = 4000; + lgdt3306a_config.vsb_if_khz = 3250; + lgdt3306a_config.mpeg_mode = LGDT3306A_MPEG_PARALLEL; + lgdt3306a_config.tpclk_edge = LGDT3306A_TPCLK_FALLING_EDGE; + lgdt3306a_config.tpvalid_polarity = LGDT3306A_TP_VALID_LOW; + lgdt3306a_config.xtalMHz = 25, /* demod clock MHz; 24/25 supported */ + + adap->i2c_client_demod[0] = dvb_module_probe("lgdt3306a", NULL, + &adap->channel.hdw->i2c_adap, + 0x59, &lgdt3306a_config); + + if (!adap->i2c_client_demod[0]) + return -ENODEV; + + return 0; +} + +static int pvr2_dual_fe_attach(struct pvr2_dvb_adapter *adap) +{ + pr_debug("%s()\n", __func__); + + if (pvr2_lgdt3306a_attach(adap) != 0) + return -ENODEV; + + if (pvr2_si2168_attach(adap) != 0) { + dvb_module_release(adap->i2c_client_demod[0]); + return -ENODEV; + } + + return 0; +} +#endif + +#define PVR2_FIRMWARE_160xxx "v4l-pvrusb2-160xxx-01.fw" +static const char *pvr2_fw1_names_160xxx[] = { + PVR2_FIRMWARE_160xxx, +}; + +static const struct pvr2_device_client_desc pvr2_cli_160xxx[] = { + { .module_id = PVR2_CLIENT_ID_CX25840 }, +}; + +static const struct pvr2_device_desc pvr2_device_160000 = { + .description = "WinTV HVR-1975 Model 160000", + .shortname = "160000", + .client_table.lst = pvr2_cli_160xxx, + .client_table.cnt = ARRAY_SIZE(pvr2_cli_160xxx), + .fx2_firmware.lst = pvr2_fw1_names_160xxx, + .fx2_firmware.cnt = ARRAY_SIZE(pvr2_fw1_names_160xxx), + .default_tuner_type = TUNER_ABSENT, + .flag_has_cx25840 = 1, + .flag_has_hauppauge_rom = 1, + .flag_has_analogtuner = 1, + .flag_has_composite = 1, + .flag_has_svideo = 1, + .flag_fx2_16kb = 1, + .signal_routing_scheme = PVR2_ROUTING_SCHEME_HAUPPAUGE, + .digital_control_scheme = PVR2_DIGITAL_SCHEME_HAUPPAUGE, + .default_std_mask = V4L2_STD_NTSC_M, + .led_scheme = PVR2_LED_SCHEME_HAUPPAUGE, + .ir_scheme = PVR2_IR_SCHEME_ZILOG, +#ifdef CONFIG_VIDEO_PVRUSB2_DVB + .dvb_props = &pvr2_160000_dvb_props, +#endif +}; + +static const struct pvr2_device_desc pvr2_device_160111 = { + .description = "WinTV HVR-1955 Model 160111", + .shortname = "160111", + .client_table.lst = pvr2_cli_160xxx, + .client_table.cnt = ARRAY_SIZE(pvr2_cli_160xxx), + .fx2_firmware.lst = pvr2_fw1_names_160xxx, + .fx2_firmware.cnt = ARRAY_SIZE(pvr2_fw1_names_160xxx), + .default_tuner_type = TUNER_ABSENT, + .flag_has_cx25840 = 1, + .flag_has_hauppauge_rom = 1, + .flag_has_analogtuner = 1, + .flag_has_composite = 1, + .flag_has_svideo = 1, + .flag_fx2_16kb = 1, + .signal_routing_scheme = PVR2_ROUTING_SCHEME_HAUPPAUGE, + .digital_control_scheme = PVR2_DIGITAL_SCHEME_HAUPPAUGE, + .default_std_mask = V4L2_STD_NTSC_M, + .led_scheme = PVR2_LED_SCHEME_HAUPPAUGE, + .ir_scheme = PVR2_IR_SCHEME_ZILOG, +#ifdef CONFIG_VIDEO_PVRUSB2_DVB + .dvb_props = &pvr2_160111_dvb_props, +#endif +}; /*------------------------------------------------------------------------*/ @@ -552,6 +714,10 @@ struct usb_device_id pvr2_device_table[] = { .driver_info = (kernel_ulong_t)&pvr2_device_751xx}, { USB_DEVICE(0x0ccd, 0x0039), .driver_info = (kernel_ulong_t)&pvr2_device_av400}, + { USB_DEVICE(0x2040, 0x7502), + .driver_info = (kernel_ulong_t)&pvr2_device_160111}, + { USB_DEVICE(0x2040, 0x7510), + .driver_info = (kernel_ulong_t)&pvr2_device_160000}, { } }; diff --git a/drivers/media/usb/pvrusb2/pvrusb2-devattr.h b/drivers/media/usb/pvrusb2/pvrusb2-devattr.h index c1e7d4822cd1..ea0b2bf429e9 100644 --- a/drivers/media/usb/pvrusb2/pvrusb2-devattr.h +++ b/drivers/media/usb/pvrusb2/pvrusb2-devattr.h @@ -66,6 +66,7 @@ struct pvr2_string_table { #define PVR2_ROUTING_SCHEME_GOTVIEW 1 #define PVR2_ROUTING_SCHEME_ONAIR 2 #define PVR2_ROUTING_SCHEME_AV400 3 +#define PVR2_ROUTING_SCHEME_HAUP160XXX 4 #define PVR2_DIGITAL_SCHEME_NONE 0 #define PVR2_DIGITAL_SCHEME_HAUPPAUGE 1 diff --git a/drivers/media/usb/pvrusb2/pvrusb2-fx2-cmd.h b/drivers/media/usb/pvrusb2/pvrusb2-fx2-cmd.h index 0a01de4e54db..640b033815ec 100644 --- a/drivers/media/usb/pvrusb2/pvrusb2-fx2-cmd.h +++ b/drivers/media/usb/pvrusb2/pvrusb2-fx2-cmd.h @@ -38,6 +38,10 @@ #define FX2CMD_FWPOST1 0x52u +/* These 2 only exist on Model 160xxx */ +#define FX2CMD_HCW_DEMOD_RESET_PIN 0xd4u +#define FX2CMD_HCW_MAKO_SLEEP_PIN 0xd5u + #define FX2CMD_POWER_OFF 0xdcu #define FX2CMD_POWER_ON 0xdeu diff --git a/drivers/media/usb/pvrusb2/pvrusb2-hdw.c b/drivers/media/usb/pvrusb2/pvrusb2-hdw.c index 191439109788..957913146e88 100644 --- a/drivers/media/usb/pvrusb2/pvrusb2-hdw.c +++ b/drivers/media/usb/pvrusb2/pvrusb2-hdw.c @@ -316,6 +316,8 @@ static const struct pvr2_fx2cmd_descdef pvr2_fx2cmd_desc[] = { {FX2CMD_ONAIR_DTV_STREAMING_OFF, "onair dtv stream off"}, {FX2CMD_ONAIR_DTV_POWER_ON, "onair dtv power on"}, {FX2CMD_ONAIR_DTV_POWER_OFF, "onair dtv power off"}, + {FX2CMD_HCW_DEMOD_RESET_PIN, "hcw demod reset pin"}, + {FX2CMD_HCW_MAKO_SLEEP_PIN, "hcw mako sleep pin"}, }; @@ -2139,10 +2141,28 @@ static void pvr2_hdw_setup_low(struct pvr2_hdw *hdw) ((0) << 16)); } - // This step MUST happen after the earlier powerup step. + /* This step MUST happen after the earlier powerup step */ pvr2_i2c_core_init(hdw); if (!pvr2_hdw_dev_ok(hdw)) return; + /* Reset demod only on Hauppauge 160xxx platform */ + if (le16_to_cpu(hdw->usb_dev->descriptor.idVendor) == 0x2040 && + (le16_to_cpu(hdw->usb_dev->descriptor.idProduct) == 0x7502 || + le16_to_cpu(hdw->usb_dev->descriptor.idProduct) == 0x7510)) { + pr_info("%s(): resetting 160xxx demod\n", __func__); + /* TODO: not sure this is proper place to reset once only */ + pvr2_issue_simple_cmd(hdw, + FX2CMD_HCW_DEMOD_RESET_PIN | + (1 << 8) | + ((0) << 16)); + usleep_range(10000, 10500); + pvr2_issue_simple_cmd(hdw, + FX2CMD_HCW_DEMOD_RESET_PIN | + (1 << 8) | + ((1) << 16)); + usleep_range(10000, 10500); + } + pvr2_hdw_load_modules(hdw); if (!pvr2_hdw_dev_ok(hdw)) return; @@ -4012,6 +4032,20 @@ int pvr2_hdw_cmd_decoder_reset(struct pvr2_hdw *hdw) static int pvr2_hdw_cmd_hcw_demod_reset(struct pvr2_hdw *hdw, int onoff) { hdw->flag_ok = !0; + + /* Use this for Hauppauge 160xxx only */ + if (le16_to_cpu(hdw->usb_dev->descriptor.idVendor) == 0x2040 && + (le16_to_cpu(hdw->usb_dev->descriptor.idProduct) == 0x7502 || + le16_to_cpu(hdw->usb_dev->descriptor.idProduct) == 0x7510)) { + pr_debug("%s(): resetting demod on Hauppauge 160xxx platform skipped\n", + __func__); + /* Can't reset 160xxx or it will trash Demod tristate */ + return pvr2_issue_simple_cmd(hdw, + FX2CMD_HCW_MAKO_SLEEP_PIN | + (1 << 8) | + ((onoff ? 1 : 0) << 16)); + } + return pvr2_issue_simple_cmd(hdw, FX2CMD_HCW_DEMOD_RESETIN | (1 << 8) |
Includes support to identify and use two Hauppauge device. HVR-1955: - LGDT3306a ATSC/QAM demod - si2177 tuner - cx25840 decoder for analog tv/composite/s-video/audio HVR-1975 dual-frontend: - LGDT3306a ATSC/QAM demod - si2168 DVB-C/T/T2 demod - si2177 tuner - cx25840 decoder for analog tv/composite/s-video/audio Signed-off-by: Brad Love <brad@nextdimension.cc> --- Since v6: - back off one checkpatch strict fix, causes discarded const compiler warning Since v5: - le16_to_cpu fix Since v4: - checkpatch strict fixes - change !0 initializations Since v2: - Fix build with VIDEO_PVRUSB2_DVB enabled Changes since v1: - Fix build with VIDEO_PVRUSB2_DVB disabled - Insert 160xxx code lower, so 75xxx profile is not split - Reorganize 160xxx board profile - Share config where possible drivers/media/usb/pvrusb2/Kconfig | 2 + .../media/usb/pvrusb2/pvrusb2-cx2584x-v4l.c | 25 +++ drivers/media/usb/pvrusb2/pvrusb2-devattr.c | 166 ++++++++++++++++++ drivers/media/usb/pvrusb2/pvrusb2-devattr.h | 1 + drivers/media/usb/pvrusb2/pvrusb2-fx2-cmd.h | 4 + drivers/media/usb/pvrusb2/pvrusb2-hdw.c | 36 +++- 6 files changed, 233 insertions(+), 1 deletion(-)