dib0700_devices.c 49 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213121412151216121712181219122012211222122312241225122612271228122912301231123212331234123512361237123812391240124112421243124412451246124712481249125012511252125312541255125612571258125912601261126212631264126512661267126812691270127112721273127412751276127712781279128012811282128312841285128612871288128912901291129212931294129512961297129812991300130113021303130413051306130713081309131013111312131313141315131613171318131913201321132213231324132513261327132813291330133113321333133413351336133713381339134013411342134313441345134613471348134913501351135213531354135513561357135813591360136113621363136413651366136713681369137013711372137313741375137613771378137913801381138213831384138513861387138813891390139113921393139413951396139713981399140014011402140314041405140614071408140914101411141214131414141514161417141814191420142114221423142414251426142714281429143014311432143314341435143614371438143914401441144214431444144514461447144814491450145114521453145414551456145714581459146014611462146314641465146614671468146914701471147214731474147514761477147814791480148114821483148414851486148714881489149014911492149314941495149614971498149915001501150215031504150515061507150815091510151115121513151415151516151715181519152015211522152315241525152615271528152915301531153215331534153515361537153815391540154115421543154415451546154715481549155015511552155315541555155615571558155915601561156215631564156515661567156815691570157115721573157415751576157715781579158015811582158315841585158615871588158915901591159215931594159515961597159815991600160116021603160416051606160716081609161016111612161316141615161616171618161916201621162216231624162516261627162816291630163116321633163416351636163716381639164016411642164316441645164616471648164916501651165216531654165516561657165816591660166116621663166416651666166716681669167016711672167316741675167616771678167916801681168216831684168516861687168816891690169116921693169416951696169716981699170017011702170317041705170617071708170917101711171217131714171517161717171817191720172117221723172417251726172717281729173017311732173317341735173617371738173917401741174217431744174517461747174817491750175117521753
  1. /* Linux driver for devices based on the DiBcom DiB0700 USB bridge
  2. *
  3. * This program is free software; you can redistribute it and/or modify it
  4. * under the terms of the GNU General Public License as published by the Free
  5. * Software Foundation, version 2.
  6. *
  7. * Copyright (C) 2005-7 DiBcom, SA
  8. */
  9. #include "dib0700.h"
  10. #include "dib3000mc.h"
  11. #include "dib7000m.h"
  12. #include "dib7000p.h"
  13. #include "mt2060.h"
  14. #include "mt2266.h"
  15. #include "tuner-xc2028.h"
  16. #include "xc5000.h"
  17. #include "s5h1411.h"
  18. #include "dib0070.h"
  19. static int force_lna_activation;
  20. module_param(force_lna_activation, int, 0644);
  21. MODULE_PARM_DESC(force_lna_activation, "force the activation of Low-Noise-Amplifyer(s) (LNA), "
  22. "if applicable for the device (default: 0=automatic/off).");
  23. struct dib0700_adapter_state {
  24. int (*set_param_save) (struct dvb_frontend *, struct dvb_frontend_parameters *);
  25. };
  26. /* Hauppauge Nova-T 500 (aka Bristol)
  27. * has a LNA on GPIO0 which is enabled by setting 1 */
  28. static struct mt2060_config bristol_mt2060_config[2] = {
  29. {
  30. .i2c_address = 0x60,
  31. .clock_out = 3,
  32. }, {
  33. .i2c_address = 0x61,
  34. }
  35. };
  36. static struct dibx000_agc_config bristol_dib3000p_mt2060_agc_config = {
  37. .band_caps = BAND_VHF | BAND_UHF,
  38. .setup = (1 << 8) | (5 << 5) | (0 << 4) | (0 << 3) | (0 << 2) | (2 << 0),
  39. .agc1_max = 42598,
  40. .agc1_min = 17694,
  41. .agc2_max = 45875,
  42. .agc2_min = 0,
  43. .agc1_pt1 = 0,
  44. .agc1_pt2 = 59,
  45. .agc1_slope1 = 0,
  46. .agc1_slope2 = 69,
  47. .agc2_pt1 = 0,
  48. .agc2_pt2 = 59,
  49. .agc2_slope1 = 111,
  50. .agc2_slope2 = 28,
  51. };
  52. static struct dib3000mc_config bristol_dib3000mc_config[2] = {
  53. { .agc = &bristol_dib3000p_mt2060_agc_config,
  54. .max_time = 0x196,
  55. .ln_adc_level = 0x1cc7,
  56. .output_mpeg2_in_188_bytes = 1,
  57. },
  58. { .agc = &bristol_dib3000p_mt2060_agc_config,
  59. .max_time = 0x196,
  60. .ln_adc_level = 0x1cc7,
  61. .output_mpeg2_in_188_bytes = 1,
  62. }
  63. };
  64. static int bristol_frontend_attach(struct dvb_usb_adapter *adap)
  65. {
  66. struct dib0700_state *st = adap->dev->priv;
  67. if (adap->id == 0) {
  68. dib0700_set_gpio(adap->dev, GPIO6, GPIO_OUT, 0); msleep(10);
  69. dib0700_set_gpio(adap->dev, GPIO6, GPIO_OUT, 1); msleep(10);
  70. dib0700_set_gpio(adap->dev, GPIO10, GPIO_OUT, 0); msleep(10);
  71. dib0700_set_gpio(adap->dev, GPIO10, GPIO_OUT, 1); msleep(10);
  72. if (force_lna_activation)
  73. dib0700_set_gpio(adap->dev, GPIO0, GPIO_OUT, 1);
  74. else
  75. dib0700_set_gpio(adap->dev, GPIO0, GPIO_OUT, 0);
  76. if (dib3000mc_i2c_enumeration(&adap->dev->i2c_adap, 2, DEFAULT_DIB3000P_I2C_ADDRESS, bristol_dib3000mc_config) != 0) {
  77. dib0700_set_gpio(adap->dev, GPIO6, GPIO_OUT, 0); msleep(10);
  78. return -ENODEV;
  79. }
  80. }
  81. st->mt2060_if1[adap->id] = 1220;
  82. return (adap->fe = dvb_attach(dib3000mc_attach, &adap->dev->i2c_adap,
  83. (10 + adap->id) << 1, &bristol_dib3000mc_config[adap->id])) == NULL ? -ENODEV : 0;
  84. }
  85. static int eeprom_read(struct i2c_adapter *adap,u8 adrs,u8 *pval)
  86. {
  87. struct i2c_msg msg[2] = {
  88. { .addr = 0x50, .flags = 0, .buf = &adrs, .len = 1 },
  89. { .addr = 0x50, .flags = I2C_M_RD, .buf = pval, .len = 1 },
  90. };
  91. if (i2c_transfer(adap, msg, 2) != 2) return -EREMOTEIO;
  92. return 0;
  93. }
  94. static int bristol_tuner_attach(struct dvb_usb_adapter *adap)
  95. {
  96. struct i2c_adapter *prim_i2c = &adap->dev->i2c_adap;
  97. struct i2c_adapter *tun_i2c = dib3000mc_get_tuner_i2c_master(adap->fe, 1);
  98. s8 a;
  99. int if1=1220;
  100. if (adap->dev->udev->descriptor.idVendor == cpu_to_le16(USB_VID_HAUPPAUGE) &&
  101. adap->dev->udev->descriptor.idProduct == cpu_to_le16(USB_PID_HAUPPAUGE_NOVA_T_500_2)) {
  102. if (!eeprom_read(prim_i2c,0x59 + adap->id,&a)) if1=1220+a;
  103. }
  104. return dvb_attach(mt2060_attach,adap->fe, tun_i2c,&bristol_mt2060_config[adap->id],
  105. if1) == NULL ? -ENODEV : 0;
  106. }
  107. /* STK7700D: Pinnacle/Terratec/Hauppauge Dual DVB-T Diversity */
  108. /* MT226x */
  109. static struct dibx000_agc_config stk7700d_7000p_mt2266_agc_config[2] = {
  110. {
  111. BAND_UHF, // band_caps
  112. /* P_agc_use_sd_mod1=0, P_agc_use_sd_mod2=0, P_agc_freq_pwm_div=1, P_agc_inv_pwm1=1, P_agc_inv_pwm2=1,
  113. * P_agc_inh_dc_rv_est=0, P_agc_time_est=3, P_agc_freeze=0, P_agc_nb_est=2, P_agc_write=0 */
  114. (0 << 15) | (0 << 14) | (1 << 11) | (1 << 10) | (1 << 9) | (0 << 8) | (3 << 5) | (0 << 4) | (5 << 1) | (0 << 0), // setup
  115. 1130, // inv_gain
  116. 21, // time_stabiliz
  117. 0, // alpha_level
  118. 118, // thlock
  119. 0, // wbd_inv
  120. 3530, // wbd_ref
  121. 1, // wbd_sel
  122. 0, // wbd_alpha
  123. 65535, // agc1_max
  124. 33770, // agc1_min
  125. 65535, // agc2_max
  126. 23592, // agc2_min
  127. 0, // agc1_pt1
  128. 62, // agc1_pt2
  129. 255, // agc1_pt3
  130. 64, // agc1_slope1
  131. 64, // agc1_slope2
  132. 132, // agc2_pt1
  133. 192, // agc2_pt2
  134. 80, // agc2_slope1
  135. 80, // agc2_slope2
  136. 17, // alpha_mant
  137. 27, // alpha_exp
  138. 23, // beta_mant
  139. 51, // beta_exp
  140. 1, // perform_agc_softsplit
  141. }, {
  142. BAND_VHF | BAND_LBAND, // band_caps
  143. /* P_agc_use_sd_mod1=0, P_agc_use_sd_mod2=0, P_agc_freq_pwm_div=1, P_agc_inv_pwm1=1, P_agc_inv_pwm2=1,
  144. * P_agc_inh_dc_rv_est=0, P_agc_time_est=3, P_agc_freeze=0, P_agc_nb_est=2, P_agc_write=0 */
  145. (0 << 15) | (0 << 14) | (1 << 11) | (1 << 10) | (1 << 9) | (0 << 8) | (3 << 5) | (0 << 4) | (2 << 1) | (0 << 0), // setup
  146. 2372, // inv_gain
  147. 21, // time_stabiliz
  148. 0, // alpha_level
  149. 118, // thlock
  150. 0, // wbd_inv
  151. 3530, // wbd_ref
  152. 1, // wbd_sel
  153. 0, // wbd_alpha
  154. 65535, // agc1_max
  155. 0, // agc1_min
  156. 65535, // agc2_max
  157. 23592, // agc2_min
  158. 0, // agc1_pt1
  159. 128, // agc1_pt2
  160. 128, // agc1_pt3
  161. 128, // agc1_slope1
  162. 0, // agc1_slope2
  163. 128, // agc2_pt1
  164. 253, // agc2_pt2
  165. 81, // agc2_slope1
  166. 0, // agc2_slope2
  167. 17, // alpha_mant
  168. 27, // alpha_exp
  169. 23, // beta_mant
  170. 51, // beta_exp
  171. 1, // perform_agc_softsplit
  172. }
  173. };
  174. static struct dibx000_bandwidth_config stk7700d_mt2266_pll_config = {
  175. 60000, 30000, // internal, sampling
  176. 1, 8, 3, 1, 0, // pll_cfg: prediv, ratio, range, reset, bypass
  177. 0, 0, 1, 1, 2, // misc: refdiv, bypclk_div, IO_CLK_en_core, ADClkSrc, modulo
  178. (3 << 14) | (1 << 12) | (524 << 0), // sad_cfg: refsel, sel, freq_15k
  179. 0, // ifreq
  180. 20452225, // timf
  181. };
  182. static struct dib7000p_config stk7700d_dib7000p_mt2266_config[] = {
  183. { .output_mpeg2_in_188_bytes = 1,
  184. .hostbus_diversity = 1,
  185. .tuner_is_baseband = 1,
  186. .agc_config_count = 2,
  187. .agc = stk7700d_7000p_mt2266_agc_config,
  188. .bw = &stk7700d_mt2266_pll_config,
  189. .gpio_dir = DIB7000P_GPIO_DEFAULT_DIRECTIONS,
  190. .gpio_val = DIB7000P_GPIO_DEFAULT_VALUES,
  191. .gpio_pwm_pos = DIB7000P_GPIO_DEFAULT_PWM_POS,
  192. },
  193. { .output_mpeg2_in_188_bytes = 1,
  194. .hostbus_diversity = 1,
  195. .tuner_is_baseband = 1,
  196. .agc_config_count = 2,
  197. .agc = stk7700d_7000p_mt2266_agc_config,
  198. .bw = &stk7700d_mt2266_pll_config,
  199. .gpio_dir = DIB7000P_GPIO_DEFAULT_DIRECTIONS,
  200. .gpio_val = DIB7000P_GPIO_DEFAULT_VALUES,
  201. .gpio_pwm_pos = DIB7000P_GPIO_DEFAULT_PWM_POS,
  202. }
  203. };
  204. static struct mt2266_config stk7700d_mt2266_config[2] = {
  205. { .i2c_address = 0x60
  206. },
  207. { .i2c_address = 0x60
  208. }
  209. };
  210. static int stk7700P2_frontend_attach(struct dvb_usb_adapter *adap)
  211. {
  212. if (adap->id == 0) {
  213. dib0700_set_gpio(adap->dev, GPIO6, GPIO_OUT, 1);
  214. msleep(10);
  215. dib0700_set_gpio(adap->dev, GPIO9, GPIO_OUT, 1);
  216. dib0700_set_gpio(adap->dev, GPIO4, GPIO_OUT, 1);
  217. dib0700_set_gpio(adap->dev, GPIO7, GPIO_OUT, 1);
  218. dib0700_set_gpio(adap->dev, GPIO10, GPIO_OUT, 0);
  219. msleep(10);
  220. dib0700_set_gpio(adap->dev, GPIO10, GPIO_OUT, 1);
  221. msleep(10);
  222. dib7000p_i2c_enumeration(&adap->dev->i2c_adap,1,18,stk7700d_dib7000p_mt2266_config);
  223. }
  224. adap->fe = dvb_attach(dib7000p_attach, &adap->dev->i2c_adap,0x80+(adap->id << 1),
  225. &stk7700d_dib7000p_mt2266_config[adap->id]);
  226. return adap->fe == NULL ? -ENODEV : 0;
  227. }
  228. static int stk7700d_frontend_attach(struct dvb_usb_adapter *adap)
  229. {
  230. if (adap->id == 0) {
  231. dib0700_set_gpio(adap->dev, GPIO6, GPIO_OUT, 1);
  232. msleep(10);
  233. dib0700_set_gpio(adap->dev, GPIO9, GPIO_OUT, 1);
  234. dib0700_set_gpio(adap->dev, GPIO4, GPIO_OUT, 1);
  235. dib0700_set_gpio(adap->dev, GPIO7, GPIO_OUT, 1);
  236. dib0700_set_gpio(adap->dev, GPIO10, GPIO_OUT, 0);
  237. msleep(10);
  238. dib0700_set_gpio(adap->dev, GPIO10, GPIO_OUT, 1);
  239. msleep(10);
  240. dib0700_set_gpio(adap->dev, GPIO0, GPIO_OUT, 1);
  241. dib7000p_i2c_enumeration(&adap->dev->i2c_adap,2,18,stk7700d_dib7000p_mt2266_config);
  242. }
  243. adap->fe = dvb_attach(dib7000p_attach, &adap->dev->i2c_adap,0x80+(adap->id << 1),
  244. &stk7700d_dib7000p_mt2266_config[adap->id]);
  245. return adap->fe == NULL ? -ENODEV : 0;
  246. }
  247. static int stk7700d_tuner_attach(struct dvb_usb_adapter *adap)
  248. {
  249. struct i2c_adapter *tun_i2c;
  250. tun_i2c = dib7000p_get_i2c_master(adap->fe, DIBX000_I2C_INTERFACE_TUNER, 1);
  251. return dvb_attach(mt2266_attach, adap->fe, tun_i2c,
  252. &stk7700d_mt2266_config[adap->id]) == NULL ? -ENODEV : 0;;
  253. }
  254. /* STK7700-PH: Digital/Analog Hybrid Tuner, e.h. Cinergy HT USB HE */
  255. static struct dibx000_agc_config xc3028_agc_config = {
  256. BAND_VHF | BAND_UHF, /* band_caps */
  257. /* P_agc_use_sd_mod1=0, P_agc_use_sd_mod2=0, P_agc_freq_pwm_div=0,
  258. * P_agc_inv_pwm1=0, P_agc_inv_pwm2=0, P_agc_inh_dc_rv_est=0,
  259. * P_agc_time_est=3, P_agc_freeze=0, P_agc_nb_est=2, P_agc_write=0 */
  260. (0 << 15) | (0 << 14) | (0 << 11) | (0 << 10) | (0 << 9) | (0 << 8) |
  261. (3 << 5) | (0 << 4) | (2 << 1) | (0 << 0), /* setup */
  262. 712, /* inv_gain */
  263. 21, /* time_stabiliz */
  264. 0, /* alpha_level */
  265. 118, /* thlock */
  266. 0, /* wbd_inv */
  267. 2867, /* wbd_ref */
  268. 0, /* wbd_sel */
  269. 2, /* wbd_alpha */
  270. 0, /* agc1_max */
  271. 0, /* agc1_min */
  272. 39718, /* agc2_max */
  273. 9930, /* agc2_min */
  274. 0, /* agc1_pt1 */
  275. 0, /* agc1_pt2 */
  276. 0, /* agc1_pt3 */
  277. 0, /* agc1_slope1 */
  278. 0, /* agc1_slope2 */
  279. 0, /* agc2_pt1 */
  280. 128, /* agc2_pt2 */
  281. 29, /* agc2_slope1 */
  282. 29, /* agc2_slope2 */
  283. 17, /* alpha_mant */
  284. 27, /* alpha_exp */
  285. 23, /* beta_mant */
  286. 51, /* beta_exp */
  287. 1, /* perform_agc_softsplit */
  288. };
  289. /* PLL Configuration for COFDM BW_MHz = 8.00 with external clock = 30.00 */
  290. static struct dibx000_bandwidth_config xc3028_bw_config = {
  291. 60000, 30000, /* internal, sampling */
  292. 1, 8, 3, 1, 0, /* pll_cfg: prediv, ratio, range, reset, bypass */
  293. 0, 0, 1, 1, 0, /* misc: refdiv, bypclk_div, IO_CLK_en_core, ADClkSrc,
  294. modulo */
  295. (3 << 14) | (1 << 12) | (524 << 0), /* sad_cfg: refsel, sel, freq_15k */
  296. (1 << 25) | 5816102, /* ifreq = 5.200000 MHz */
  297. 20452225, /* timf */
  298. 30000000, /* xtal_hz */
  299. };
  300. static struct dib7000p_config stk7700ph_dib7700_xc3028_config = {
  301. .output_mpeg2_in_188_bytes = 1,
  302. .tuner_is_baseband = 1,
  303. .agc_config_count = 1,
  304. .agc = &xc3028_agc_config,
  305. .bw = &xc3028_bw_config,
  306. .gpio_dir = DIB7000P_GPIO_DEFAULT_DIRECTIONS,
  307. .gpio_val = DIB7000P_GPIO_DEFAULT_VALUES,
  308. .gpio_pwm_pos = DIB7000P_GPIO_DEFAULT_PWM_POS,
  309. };
  310. static int stk7700ph_xc3028_callback(void *ptr, int component,
  311. int command, int arg)
  312. {
  313. struct dvb_usb_adapter *adap = ptr;
  314. switch (command) {
  315. case XC2028_TUNER_RESET:
  316. /* Send the tuner in then out of reset */
  317. dib7000p_set_gpio(adap->fe, 8, 0, 0); msleep(10);
  318. dib7000p_set_gpio(adap->fe, 8, 0, 1);
  319. break;
  320. case XC2028_RESET_CLK:
  321. break;
  322. default:
  323. err("%s: unknown command %d, arg %d\n", __func__,
  324. command, arg);
  325. return -EINVAL;
  326. }
  327. return 0;
  328. }
  329. static struct xc2028_ctrl stk7700ph_xc3028_ctrl = {
  330. .fname = XC2028_DEFAULT_FIRMWARE,
  331. .max_len = 64,
  332. .demod = XC3028_FE_DIBCOM52,
  333. };
  334. static struct xc2028_config stk7700ph_xc3028_config = {
  335. .i2c_addr = 0x61,
  336. .ctrl = &stk7700ph_xc3028_ctrl,
  337. };
  338. static int stk7700ph_frontend_attach(struct dvb_usb_adapter *adap)
  339. {
  340. struct usb_device_descriptor *desc = &adap->dev->udev->descriptor;
  341. if (desc->idVendor == cpu_to_le16(USB_VID_PINNACLE) &&
  342. desc->idProduct == cpu_to_le16(USB_PID_PINNACLE_EXPRESSCARD_320CX))
  343. dib0700_set_gpio(adap->dev, GPIO6, GPIO_OUT, 0);
  344. else
  345. dib0700_set_gpio(adap->dev, GPIO6, GPIO_OUT, 1);
  346. msleep(20);
  347. dib0700_set_gpio(adap->dev, GPIO9, GPIO_OUT, 1);
  348. dib0700_set_gpio(adap->dev, GPIO4, GPIO_OUT, 1);
  349. dib0700_set_gpio(adap->dev, GPIO7, GPIO_OUT, 1);
  350. dib0700_set_gpio(adap->dev, GPIO10, GPIO_OUT, 0);
  351. msleep(10);
  352. dib0700_set_gpio(adap->dev, GPIO10, GPIO_OUT, 1);
  353. msleep(20);
  354. dib0700_set_gpio(adap->dev, GPIO0, GPIO_OUT, 1);
  355. msleep(10);
  356. dib7000p_i2c_enumeration(&adap->dev->i2c_adap, 1, 18,
  357. &stk7700ph_dib7700_xc3028_config);
  358. adap->fe = dvb_attach(dib7000p_attach, &adap->dev->i2c_adap, 0x80,
  359. &stk7700ph_dib7700_xc3028_config);
  360. return adap->fe == NULL ? -ENODEV : 0;
  361. }
  362. static int stk7700ph_tuner_attach(struct dvb_usb_adapter *adap)
  363. {
  364. struct i2c_adapter *tun_i2c;
  365. tun_i2c = dib7000p_get_i2c_master(adap->fe,
  366. DIBX000_I2C_INTERFACE_TUNER, 1);
  367. stk7700ph_xc3028_config.i2c_adap = tun_i2c;
  368. /* FIXME: generalize & move to common area */
  369. adap->fe->callback = stk7700ph_xc3028_callback;
  370. return dvb_attach(xc2028_attach, adap->fe, &stk7700ph_xc3028_config)
  371. == NULL ? -ENODEV : 0;
  372. }
  373. #define DEFAULT_RC_INTERVAL 50
  374. static u8 rc_request[] = { REQUEST_POLL_RC, 0 };
  375. /* Number of keypresses to ignore before start repeating */
  376. #define RC_REPEAT_DELAY 6
  377. #define RC_REPEAT_DELAY_V1_20 10
  378. /* Used by firmware versions < 1.20 (deprecated) */
  379. static int dib0700_rc_query_legacy(struct dvb_usb_device *d, u32 *event,
  380. int *state)
  381. {
  382. u8 key[4];
  383. int i;
  384. struct dvb_usb_rc_key *keymap = d->props.rc_key_map;
  385. struct dib0700_state *st = d->priv;
  386. *event = 0;
  387. *state = REMOTE_NO_KEY_PRESSED;
  388. i=dib0700_ctrl_rd(d,rc_request,2,key,4);
  389. if (i<=0) {
  390. err("RC Query Failed");
  391. return -1;
  392. }
  393. /* losing half of KEY_0 events from Philipps rc5 remotes.. */
  394. if (key[0]==0 && key[1]==0 && key[2]==0 && key[3]==0) return 0;
  395. /* info("%d: %2X %2X %2X %2X",dvb_usb_dib0700_ir_proto,(int)key[3-2],(int)key[3-3],(int)key[3-1],(int)key[3]); */
  396. dib0700_rc_setup(d); /* reset ir sensor data to prevent false events */
  397. switch (dvb_usb_dib0700_ir_proto) {
  398. case 0: {
  399. /* NEC protocol sends repeat code as 0 0 0 FF */
  400. if ((key[3-2] == 0x00) && (key[3-3] == 0x00) &&
  401. (key[3] == 0xFF)) {
  402. st->rc_counter++;
  403. if (st->rc_counter > RC_REPEAT_DELAY) {
  404. *event = d->last_event;
  405. *state = REMOTE_KEY_PRESSED;
  406. st->rc_counter = RC_REPEAT_DELAY;
  407. }
  408. return 0;
  409. }
  410. for (i=0;i<d->props.rc_key_map_size; i++) {
  411. if (keymap[i].custom == key[3-2] && keymap[i].data == key[3-3]) {
  412. st->rc_counter = 0;
  413. *event = keymap[i].event;
  414. *state = REMOTE_KEY_PRESSED;
  415. d->last_event = keymap[i].event;
  416. return 0;
  417. }
  418. }
  419. break;
  420. }
  421. default: {
  422. /* RC-5 protocol changes toggle bit on new keypress */
  423. for (i = 0; i < d->props.rc_key_map_size; i++) {
  424. if (keymap[i].custom == key[3-2] && keymap[i].data == key[3-3]) {
  425. if (d->last_event == keymap[i].event &&
  426. key[3-1] == st->rc_toggle) {
  427. st->rc_counter++;
  428. /* prevents unwanted double hits */
  429. if (st->rc_counter > RC_REPEAT_DELAY) {
  430. *event = d->last_event;
  431. *state = REMOTE_KEY_PRESSED;
  432. st->rc_counter = RC_REPEAT_DELAY;
  433. }
  434. return 0;
  435. }
  436. st->rc_counter = 0;
  437. *event = keymap[i].event;
  438. *state = REMOTE_KEY_PRESSED;
  439. st->rc_toggle = key[3-1];
  440. d->last_event = keymap[i].event;
  441. return 0;
  442. }
  443. }
  444. break;
  445. }
  446. }
  447. err("Unknown remote controller key: %2X %2X %2X %2X", (int) key[3-2], (int) key[3-3], (int) key[3-1], (int) key[3]);
  448. d->last_event = 0;
  449. return 0;
  450. }
  451. /* This is the structure of the RC response packet starting in firmware 1.20 */
  452. struct dib0700_rc_response {
  453. u8 report_id;
  454. u8 data_state;
  455. u8 system_msb;
  456. u8 system_lsb;
  457. u8 data;
  458. u8 not_data;
  459. };
  460. /* This supports the new IR response format for firmware v1.20 */
  461. static int dib0700_rc_query_v1_20(struct dvb_usb_device *d, u32 *event,
  462. int *state)
  463. {
  464. struct dvb_usb_rc_key *keymap = d->props.rc_key_map;
  465. struct dib0700_state *st = d->priv;
  466. struct dib0700_rc_response poll_reply;
  467. u8 buf[6];
  468. int i;
  469. int status;
  470. int actlen;
  471. int found = 0;
  472. /* Set initial results in case we exit the function early */
  473. *event = 0;
  474. *state = REMOTE_NO_KEY_PRESSED;
  475. /* Firmware v1.20 provides RC data via bulk endpoint 1 */
  476. status = usb_bulk_msg(d->udev, usb_rcvbulkpipe(d->udev, 1), buf,
  477. sizeof(buf), &actlen, 50);
  478. if (status < 0) {
  479. /* No data available (meaning no key press) */
  480. return 0;
  481. }
  482. if (actlen != sizeof(buf)) {
  483. /* We didn't get back the 6 byte message we expected */
  484. err("Unexpected RC response size [%d]", actlen);
  485. return -1;
  486. }
  487. poll_reply.report_id = buf[0];
  488. poll_reply.data_state = buf[1];
  489. poll_reply.system_msb = buf[2];
  490. poll_reply.system_lsb = buf[3];
  491. poll_reply.data = buf[4];
  492. poll_reply.not_data = buf[5];
  493. /*
  494. info("rid=%02x ds=%02x sm=%02x sl=%02x d=%02x nd=%02x\n",
  495. poll_reply.report_id, poll_reply.data_state,
  496. poll_reply.system_msb, poll_reply.system_lsb,
  497. poll_reply.data, poll_reply.not_data);
  498. */
  499. if ((poll_reply.data + poll_reply.not_data) != 0xff) {
  500. /* Key failed integrity check */
  501. err("key failed integrity check: %02x %02x %02x %02x",
  502. poll_reply.system_msb, poll_reply.system_lsb,
  503. poll_reply.data, poll_reply.not_data);
  504. return -1;
  505. }
  506. /* Find the key in the map */
  507. for (i = 0; i < d->props.rc_key_map_size; i++) {
  508. if (keymap[i].custom == poll_reply.system_lsb &&
  509. keymap[i].data == poll_reply.data) {
  510. *event = keymap[i].event;
  511. found = 1;
  512. break;
  513. }
  514. }
  515. if (found == 0) {
  516. err("Unknown remote controller key: %02x %02x %02x %02x",
  517. poll_reply.system_msb, poll_reply.system_lsb,
  518. poll_reply.data, poll_reply.not_data);
  519. d->last_event = 0;
  520. return 0;
  521. }
  522. if (poll_reply.data_state == 1) {
  523. /* New key hit */
  524. st->rc_counter = 0;
  525. *event = keymap[i].event;
  526. *state = REMOTE_KEY_PRESSED;
  527. d->last_event = keymap[i].event;
  528. } else if (poll_reply.data_state == 2) {
  529. /* Key repeated */
  530. st->rc_counter++;
  531. /* prevents unwanted double hits */
  532. if (st->rc_counter > RC_REPEAT_DELAY_V1_20) {
  533. *event = d->last_event;
  534. *state = REMOTE_KEY_PRESSED;
  535. st->rc_counter = RC_REPEAT_DELAY_V1_20;
  536. }
  537. } else {
  538. err("Unknown data state [%d]", poll_reply.data_state);
  539. }
  540. return 0;
  541. }
  542. static int dib0700_rc_query(struct dvb_usb_device *d, u32 *event, int *state)
  543. {
  544. struct dib0700_state *st = d->priv;
  545. /* Because some people may have improperly named firmware files,
  546. let's figure out whether to use the new firmware call or the legacy
  547. call based on the firmware version embedded in the file */
  548. if (st->rc_func_version == 0) {
  549. u32 hwver, romver, ramver, fwtype;
  550. int ret = dib0700_get_version(d, &hwver, &romver, &ramver,
  551. &fwtype);
  552. if (ret < 0) {
  553. err("Could not determine version info");
  554. return -1;
  555. }
  556. if (ramver < 0x10200)
  557. st->rc_func_version = 1;
  558. else
  559. st->rc_func_version = 2;
  560. }
  561. if (st->rc_func_version == 2)
  562. return dib0700_rc_query_v1_20(d, event, state);
  563. else
  564. return dib0700_rc_query_legacy(d, event, state);
  565. }
  566. static struct dvb_usb_rc_key dib0700_rc_keys[] = {
  567. /* Key codes for the tiny Pinnacle remote*/
  568. { 0x07, 0x00, KEY_MUTE },
  569. { 0x07, 0x01, KEY_MENU }, // Pinnacle logo
  570. { 0x07, 0x39, KEY_POWER },
  571. { 0x07, 0x03, KEY_VOLUMEUP },
  572. { 0x07, 0x09, KEY_VOLUMEDOWN },
  573. { 0x07, 0x06, KEY_CHANNELUP },
  574. { 0x07, 0x0c, KEY_CHANNELDOWN },
  575. { 0x07, 0x0f, KEY_1 },
  576. { 0x07, 0x15, KEY_2 },
  577. { 0x07, 0x10, KEY_3 },
  578. { 0x07, 0x18, KEY_4 },
  579. { 0x07, 0x1b, KEY_5 },
  580. { 0x07, 0x1e, KEY_6 },
  581. { 0x07, 0x11, KEY_7 },
  582. { 0x07, 0x21, KEY_8 },
  583. { 0x07, 0x12, KEY_9 },
  584. { 0x07, 0x27, KEY_0 },
  585. { 0x07, 0x24, KEY_SCREEN }, // 'Square' key
  586. { 0x07, 0x2a, KEY_TEXT }, // 'T' key
  587. { 0x07, 0x2d, KEY_REWIND },
  588. { 0x07, 0x30, KEY_PLAY },
  589. { 0x07, 0x33, KEY_FASTFORWARD },
  590. { 0x07, 0x36, KEY_RECORD },
  591. { 0x07, 0x3c, KEY_STOP },
  592. { 0x07, 0x3f, KEY_CANCEL }, // '?' key
  593. /* Key codes for the Terratec Cinergy DT XS Diversity, similar to cinergyT2.c */
  594. { 0xeb, 0x01, KEY_POWER },
  595. { 0xeb, 0x02, KEY_1 },
  596. { 0xeb, 0x03, KEY_2 },
  597. { 0xeb, 0x04, KEY_3 },
  598. { 0xeb, 0x05, KEY_4 },
  599. { 0xeb, 0x06, KEY_5 },
  600. { 0xeb, 0x07, KEY_6 },
  601. { 0xeb, 0x08, KEY_7 },
  602. { 0xeb, 0x09, KEY_8 },
  603. { 0xeb, 0x0a, KEY_9 },
  604. { 0xeb, 0x0b, KEY_VIDEO },
  605. { 0xeb, 0x0c, KEY_0 },
  606. { 0xeb, 0x0d, KEY_REFRESH },
  607. { 0xeb, 0x0f, KEY_EPG },
  608. { 0xeb, 0x10, KEY_UP },
  609. { 0xeb, 0x11, KEY_LEFT },
  610. { 0xeb, 0x12, KEY_OK },
  611. { 0xeb, 0x13, KEY_RIGHT },
  612. { 0xeb, 0x14, KEY_DOWN },
  613. { 0xeb, 0x16, KEY_INFO },
  614. { 0xeb, 0x17, KEY_RED },
  615. { 0xeb, 0x18, KEY_GREEN },
  616. { 0xeb, 0x19, KEY_YELLOW },
  617. { 0xeb, 0x1a, KEY_BLUE },
  618. { 0xeb, 0x1b, KEY_CHANNELUP },
  619. { 0xeb, 0x1c, KEY_VOLUMEUP },
  620. { 0xeb, 0x1d, KEY_MUTE },
  621. { 0xeb, 0x1e, KEY_VOLUMEDOWN },
  622. { 0xeb, 0x1f, KEY_CHANNELDOWN },
  623. { 0xeb, 0x40, KEY_PAUSE },
  624. { 0xeb, 0x41, KEY_HOME },
  625. { 0xeb, 0x42, KEY_MENU }, /* DVD Menu */
  626. { 0xeb, 0x43, KEY_SUBTITLE },
  627. { 0xeb, 0x44, KEY_TEXT }, /* Teletext */
  628. { 0xeb, 0x45, KEY_DELETE },
  629. { 0xeb, 0x46, KEY_TV },
  630. { 0xeb, 0x47, KEY_DVD },
  631. { 0xeb, 0x48, KEY_STOP },
  632. { 0xeb, 0x49, KEY_VIDEO },
  633. { 0xeb, 0x4a, KEY_AUDIO }, /* Music */
  634. { 0xeb, 0x4b, KEY_SCREEN }, /* Pic */
  635. { 0xeb, 0x4c, KEY_PLAY },
  636. { 0xeb, 0x4d, KEY_BACK },
  637. { 0xeb, 0x4e, KEY_REWIND },
  638. { 0xeb, 0x4f, KEY_FASTFORWARD },
  639. { 0xeb, 0x54, KEY_PREVIOUS },
  640. { 0xeb, 0x58, KEY_RECORD },
  641. { 0xeb, 0x5c, KEY_NEXT },
  642. /* Key codes for the Haupauge WinTV Nova-TD, copied from nova-t-usb2.c (Nova-T USB2) */
  643. { 0x1e, 0x00, KEY_0 },
  644. { 0x1e, 0x01, KEY_1 },
  645. { 0x1e, 0x02, KEY_2 },
  646. { 0x1e, 0x03, KEY_3 },
  647. { 0x1e, 0x04, KEY_4 },
  648. { 0x1e, 0x05, KEY_5 },
  649. { 0x1e, 0x06, KEY_6 },
  650. { 0x1e, 0x07, KEY_7 },
  651. { 0x1e, 0x08, KEY_8 },
  652. { 0x1e, 0x09, KEY_9 },
  653. { 0x1e, 0x0a, KEY_KPASTERISK },
  654. { 0x1e, 0x0b, KEY_RED },
  655. { 0x1e, 0x0c, KEY_RADIO },
  656. { 0x1e, 0x0d, KEY_MENU },
  657. { 0x1e, 0x0e, KEY_GRAVE }, /* # */
  658. { 0x1e, 0x0f, KEY_MUTE },
  659. { 0x1e, 0x10, KEY_VOLUMEUP },
  660. { 0x1e, 0x11, KEY_VOLUMEDOWN },
  661. { 0x1e, 0x12, KEY_CHANNEL },
  662. { 0x1e, 0x14, KEY_UP },
  663. { 0x1e, 0x15, KEY_DOWN },
  664. { 0x1e, 0x16, KEY_LEFT },
  665. { 0x1e, 0x17, KEY_RIGHT },
  666. { 0x1e, 0x18, KEY_VIDEO },
  667. { 0x1e, 0x19, KEY_AUDIO },
  668. { 0x1e, 0x1a, KEY_MEDIA },
  669. { 0x1e, 0x1b, KEY_EPG },
  670. { 0x1e, 0x1c, KEY_TV },
  671. { 0x1e, 0x1e, KEY_NEXT },
  672. { 0x1e, 0x1f, KEY_BACK },
  673. { 0x1e, 0x20, KEY_CHANNELUP },
  674. { 0x1e, 0x21, KEY_CHANNELDOWN },
  675. { 0x1e, 0x24, KEY_LAST }, /* Skip backwards */
  676. { 0x1e, 0x25, KEY_OK },
  677. { 0x1e, 0x29, KEY_BLUE},
  678. { 0x1e, 0x2e, KEY_GREEN },
  679. { 0x1e, 0x30, KEY_PAUSE },
  680. { 0x1e, 0x32, KEY_REWIND },
  681. { 0x1e, 0x34, KEY_FASTFORWARD },
  682. { 0x1e, 0x35, KEY_PLAY },
  683. { 0x1e, 0x36, KEY_STOP },
  684. { 0x1e, 0x37, KEY_RECORD },
  685. { 0x1e, 0x38, KEY_YELLOW },
  686. { 0x1e, 0x3b, KEY_GOTO },
  687. { 0x1e, 0x3d, KEY_POWER },
  688. /* Key codes for the Leadtek Winfast DTV Dongle */
  689. { 0x00, 0x42, KEY_POWER },
  690. { 0x07, 0x7c, KEY_TUNER },
  691. { 0x0f, 0x4e, KEY_PRINT }, /* PREVIEW */
  692. { 0x08, 0x40, KEY_SCREEN }, /* full screen toggle*/
  693. { 0x0f, 0x71, KEY_DOT }, /* frequency */
  694. { 0x07, 0x43, KEY_0 },
  695. { 0x0c, 0x41, KEY_1 },
  696. { 0x04, 0x43, KEY_2 },
  697. { 0x0b, 0x7f, KEY_3 },
  698. { 0x0e, 0x41, KEY_4 },
  699. { 0x06, 0x43, KEY_5 },
  700. { 0x09, 0x7f, KEY_6 },
  701. { 0x0d, 0x7e, KEY_7 },
  702. { 0x05, 0x7c, KEY_8 },
  703. { 0x0a, 0x40, KEY_9 },
  704. { 0x0e, 0x4e, KEY_CLEAR },
  705. { 0x04, 0x7c, KEY_CHANNEL }, /* show channel number */
  706. { 0x0f, 0x41, KEY_LAST }, /* recall */
  707. { 0x03, 0x42, KEY_MUTE },
  708. { 0x06, 0x4c, KEY_RESERVED }, /* PIP button*/
  709. { 0x01, 0x72, KEY_SHUFFLE }, /* SNAPSHOT */
  710. { 0x0c, 0x4e, KEY_PLAYPAUSE }, /* TIMESHIFT */
  711. { 0x0b, 0x70, KEY_RECORD },
  712. { 0x03, 0x7d, KEY_VOLUMEUP },
  713. { 0x01, 0x7d, KEY_VOLUMEDOWN },
  714. { 0x02, 0x42, KEY_CHANNELUP },
  715. { 0x00, 0x7d, KEY_CHANNELDOWN },
  716. /* Key codes for Nova-TD "credit card" remote control. */
  717. { 0x1d, 0x00, KEY_0 },
  718. { 0x1d, 0x01, KEY_1 },
  719. { 0x1d, 0x02, KEY_2 },
  720. { 0x1d, 0x03, KEY_3 },
  721. { 0x1d, 0x04, KEY_4 },
  722. { 0x1d, 0x05, KEY_5 },
  723. { 0x1d, 0x06, KEY_6 },
  724. { 0x1d, 0x07, KEY_7 },
  725. { 0x1d, 0x08, KEY_8 },
  726. { 0x1d, 0x09, KEY_9 },
  727. { 0x1d, 0x0a, KEY_TEXT },
  728. { 0x1d, 0x0d, KEY_MENU },
  729. { 0x1d, 0x0f, KEY_MUTE },
  730. { 0x1d, 0x10, KEY_VOLUMEUP },
  731. { 0x1d, 0x11, KEY_VOLUMEDOWN },
  732. { 0x1d, 0x12, KEY_CHANNEL },
  733. { 0x1d, 0x14, KEY_UP },
  734. { 0x1d, 0x15, KEY_DOWN },
  735. { 0x1d, 0x16, KEY_LEFT },
  736. { 0x1d, 0x17, KEY_RIGHT },
  737. { 0x1d, 0x1c, KEY_TV },
  738. { 0x1d, 0x1e, KEY_NEXT },
  739. { 0x1d, 0x1f, KEY_BACK },
  740. { 0x1d, 0x20, KEY_CHANNELUP },
  741. { 0x1d, 0x21, KEY_CHANNELDOWN },
  742. { 0x1d, 0x24, KEY_LAST },
  743. { 0x1d, 0x25, KEY_OK },
  744. { 0x1d, 0x30, KEY_PAUSE },
  745. { 0x1d, 0x32, KEY_REWIND },
  746. { 0x1d, 0x34, KEY_FASTFORWARD },
  747. { 0x1d, 0x35, KEY_PLAY },
  748. { 0x1d, 0x36, KEY_STOP },
  749. { 0x1d, 0x37, KEY_RECORD },
  750. { 0x1d, 0x3b, KEY_GOTO },
  751. { 0x1d, 0x3d, KEY_POWER },
  752. };
  753. /* STK7700P: Hauppauge Nova-T Stick, AVerMedia Volar */
  754. static struct dibx000_agc_config stk7700p_7000m_mt2060_agc_config = {
  755. BAND_UHF | BAND_VHF, // band_caps
  756. /* P_agc_use_sd_mod1=0, P_agc_use_sd_mod2=0, P_agc_freq_pwm_div=5, P_agc_inv_pwm1=0, P_agc_inv_pwm2=0,
  757. * P_agc_inh_dc_rv_est=0, P_agc_time_est=3, P_agc_freeze=0, P_agc_nb_est=2, P_agc_write=0 */
  758. (0 << 15) | (0 << 14) | (5 << 11) | (0 << 10) | (0 << 9) | (0 << 8) | (3 << 5) | (0 << 4) | (2 << 1) | (0 << 0), // setup
  759. 712, // inv_gain
  760. 41, // time_stabiliz
  761. 0, // alpha_level
  762. 118, // thlock
  763. 0, // wbd_inv
  764. 4095, // wbd_ref
  765. 0, // wbd_sel
  766. 0, // wbd_alpha
  767. 42598, // agc1_max
  768. 17694, // agc1_min
  769. 45875, // agc2_max
  770. 2621, // agc2_min
  771. 0, // agc1_pt1
  772. 76, // agc1_pt2
  773. 139, // agc1_pt3
  774. 52, // agc1_slope1
  775. 59, // agc1_slope2
  776. 107, // agc2_pt1
  777. 172, // agc2_pt2
  778. 57, // agc2_slope1
  779. 70, // agc2_slope2
  780. 21, // alpha_mant
  781. 25, // alpha_exp
  782. 28, // beta_mant
  783. 48, // beta_exp
  784. 1, // perform_agc_softsplit
  785. { 0, // split_min
  786. 107, // split_max
  787. 51800, // global_split_min
  788. 24700 // global_split_max
  789. },
  790. };
  791. static struct dibx000_agc_config stk7700p_7000p_mt2060_agc_config = {
  792. BAND_UHF | BAND_VHF,
  793. /* P_agc_use_sd_mod1=0, P_agc_use_sd_mod2=0, P_agc_freq_pwm_div=5, P_agc_inv_pwm1=0, P_agc_inv_pwm2=0,
  794. * P_agc_inh_dc_rv_est=0, P_agc_time_est=3, P_agc_freeze=0, P_agc_nb_est=2, P_agc_write=0 */
  795. (0 << 15) | (0 << 14) | (5 << 11) | (0 << 10) | (0 << 9) | (0 << 8) | (3 << 5) | (0 << 4) | (2 << 1) | (0 << 0), // setup
  796. 712, // inv_gain
  797. 41, // time_stabiliz
  798. 0, // alpha_level
  799. 118, // thlock
  800. 0, // wbd_inv
  801. 4095, // wbd_ref
  802. 0, // wbd_sel
  803. 0, // wbd_alpha
  804. 42598, // agc1_max
  805. 16384, // agc1_min
  806. 42598, // agc2_max
  807. 0, // agc2_min
  808. 0, // agc1_pt1
  809. 137, // agc1_pt2
  810. 255, // agc1_pt3
  811. 0, // agc1_slope1
  812. 255, // agc1_slope2
  813. 0, // agc2_pt1
  814. 0, // agc2_pt2
  815. 0, // agc2_slope1
  816. 41, // agc2_slope2
  817. 15, // alpha_mant
  818. 25, // alpha_exp
  819. 28, // beta_mant
  820. 48, // beta_exp
  821. 0, // perform_agc_softsplit
  822. };
  823. static struct dibx000_bandwidth_config stk7700p_pll_config = {
  824. 60000, 30000, // internal, sampling
  825. 1, 8, 3, 1, 0, // pll_cfg: prediv, ratio, range, reset, bypass
  826. 0, 0, 1, 1, 0, // misc: refdiv, bypclk_div, IO_CLK_en_core, ADClkSrc, modulo
  827. (3 << 14) | (1 << 12) | (524 << 0), // sad_cfg: refsel, sel, freq_15k
  828. 60258167, // ifreq
  829. 20452225, // timf
  830. 30000000, // xtal
  831. };
  832. static struct dib7000m_config stk7700p_dib7000m_config = {
  833. .dvbt_mode = 1,
  834. .output_mpeg2_in_188_bytes = 1,
  835. .quartz_direct = 1,
  836. .agc_config_count = 1,
  837. .agc = &stk7700p_7000m_mt2060_agc_config,
  838. .bw = &stk7700p_pll_config,
  839. .gpio_dir = DIB7000M_GPIO_DEFAULT_DIRECTIONS,
  840. .gpio_val = DIB7000M_GPIO_DEFAULT_VALUES,
  841. .gpio_pwm_pos = DIB7000M_GPIO_DEFAULT_PWM_POS,
  842. };
  843. static struct dib7000p_config stk7700p_dib7000p_config = {
  844. .output_mpeg2_in_188_bytes = 1,
  845. .agc_config_count = 1,
  846. .agc = &stk7700p_7000p_mt2060_agc_config,
  847. .bw = &stk7700p_pll_config,
  848. .gpio_dir = DIB7000M_GPIO_DEFAULT_DIRECTIONS,
  849. .gpio_val = DIB7000M_GPIO_DEFAULT_VALUES,
  850. .gpio_pwm_pos = DIB7000M_GPIO_DEFAULT_PWM_POS,
  851. };
  852. static int stk7700p_frontend_attach(struct dvb_usb_adapter *adap)
  853. {
  854. struct dib0700_state *st = adap->dev->priv;
  855. /* unless there is no real power management in DVB - we leave the device on GPIO6 */
  856. dib0700_set_gpio(adap->dev, GPIO10, GPIO_OUT, 0);
  857. dib0700_set_gpio(adap->dev, GPIO6, GPIO_OUT, 0); msleep(50);
  858. dib0700_set_gpio(adap->dev, GPIO6, GPIO_OUT, 1); msleep(10);
  859. dib0700_set_gpio(adap->dev, GPIO9, GPIO_OUT, 1);
  860. dib0700_set_gpio(adap->dev, GPIO10, GPIO_OUT, 0); msleep(10);
  861. dib0700_ctrl_clock(adap->dev, 72, 1);
  862. dib0700_set_gpio(adap->dev, GPIO10, GPIO_OUT, 1); msleep(100);
  863. dib0700_set_gpio(adap->dev, GPIO0, GPIO_OUT, 1);
  864. st->mt2060_if1[0] = 1220;
  865. if (dib7000pc_detection(&adap->dev->i2c_adap)) {
  866. adap->fe = dvb_attach(dib7000p_attach, &adap->dev->i2c_adap, 18, &stk7700p_dib7000p_config);
  867. st->is_dib7000pc = 1;
  868. } else
  869. adap->fe = dvb_attach(dib7000m_attach, &adap->dev->i2c_adap, 18, &stk7700p_dib7000m_config);
  870. return adap->fe == NULL ? -ENODEV : 0;
  871. }
  872. static struct mt2060_config stk7700p_mt2060_config = {
  873. 0x60
  874. };
  875. static int stk7700p_tuner_attach(struct dvb_usb_adapter *adap)
  876. {
  877. struct i2c_adapter *prim_i2c = &adap->dev->i2c_adap;
  878. struct dib0700_state *st = adap->dev->priv;
  879. struct i2c_adapter *tun_i2c;
  880. s8 a;
  881. int if1=1220;
  882. if (adap->dev->udev->descriptor.idVendor == cpu_to_le16(USB_VID_HAUPPAUGE) &&
  883. adap->dev->udev->descriptor.idProduct == cpu_to_le16(USB_PID_HAUPPAUGE_NOVA_T_STICK)) {
  884. if (!eeprom_read(prim_i2c,0x58,&a)) if1=1220+a;
  885. }
  886. if (st->is_dib7000pc)
  887. tun_i2c = dib7000p_get_i2c_master(adap->fe, DIBX000_I2C_INTERFACE_TUNER, 1);
  888. else
  889. tun_i2c = dib7000m_get_i2c_master(adap->fe, DIBX000_I2C_INTERFACE_TUNER, 1);
  890. return dvb_attach(mt2060_attach, adap->fe, tun_i2c, &stk7700p_mt2060_config,
  891. if1) == NULL ? -ENODEV : 0;
  892. }
  893. /* DIB7070 generic */
  894. static struct dibx000_agc_config dib7070_agc_config = {
  895. BAND_UHF | BAND_VHF | BAND_LBAND | BAND_SBAND,
  896. /* P_agc_use_sd_mod1=0, P_agc_use_sd_mod2=0, P_agc_freq_pwm_div=5, P_agc_inv_pwm1=0, P_agc_inv_pwm2=0,
  897. * P_agc_inh_dc_rv_est=0, P_agc_time_est=3, P_agc_freeze=0, P_agc_nb_est=5, P_agc_write=0 */
  898. (0 << 15) | (0 << 14) | (5 << 11) | (0 << 10) | (0 << 9) | (0 << 8) | (3 << 5) | (0 << 4) | (5 << 1) | (0 << 0), // setup
  899. 600, // inv_gain
  900. 10, // time_stabiliz
  901. 0, // alpha_level
  902. 118, // thlock
  903. 0, // wbd_inv
  904. 3530, // wbd_ref
  905. 1, // wbd_sel
  906. 5, // wbd_alpha
  907. 65535, // agc1_max
  908. 0, // agc1_min
  909. 65535, // agc2_max
  910. 0, // agc2_min
  911. 0, // agc1_pt1
  912. 40, // agc1_pt2
  913. 183, // agc1_pt3
  914. 206, // agc1_slope1
  915. 255, // agc1_slope2
  916. 72, // agc2_pt1
  917. 152, // agc2_pt2
  918. 88, // agc2_slope1
  919. 90, // agc2_slope2
  920. 17, // alpha_mant
  921. 27, // alpha_exp
  922. 23, // beta_mant
  923. 51, // beta_exp
  924. 0, // perform_agc_softsplit
  925. };
  926. static int dib7070_tuner_reset(struct dvb_frontend *fe, int onoff)
  927. {
  928. return dib7000p_set_gpio(fe, 8, 0, !onoff);
  929. }
  930. static int dib7070_tuner_sleep(struct dvb_frontend *fe, int onoff)
  931. {
  932. return dib7000p_set_gpio(fe, 9, 0, onoff);
  933. }
  934. static struct dib0070_config dib7070p_dib0070_config[2] = {
  935. {
  936. .i2c_address = DEFAULT_DIB0070_I2C_ADDRESS,
  937. .reset = dib7070_tuner_reset,
  938. .sleep = dib7070_tuner_sleep,
  939. .clock_khz = 12000,
  940. .clock_pad_drive = 4
  941. }, {
  942. .i2c_address = DEFAULT_DIB0070_I2C_ADDRESS,
  943. .reset = dib7070_tuner_reset,
  944. .sleep = dib7070_tuner_sleep,
  945. .clock_khz = 12000,
  946. }
  947. };
  948. static int dib7070_set_param_override(struct dvb_frontend *fe, struct dvb_frontend_parameters *fep)
  949. {
  950. struct dvb_usb_adapter *adap = fe->dvb->priv;
  951. struct dib0700_adapter_state *state = adap->priv;
  952. u16 offset;
  953. u8 band = BAND_OF_FREQUENCY(fep->frequency/1000);
  954. switch (band) {
  955. case BAND_VHF: offset = 950; break;
  956. case BAND_UHF:
  957. default: offset = 550; break;
  958. }
  959. deb_info("WBD for DiB7000P: %d\n", offset + dib0070_wbd_offset(fe));
  960. dib7000p_set_wbd_ref(fe, offset + dib0070_wbd_offset(fe));
  961. return state->set_param_save(fe, fep);
  962. }
  963. static int dib7070p_tuner_attach(struct dvb_usb_adapter *adap)
  964. {
  965. struct dib0700_adapter_state *st = adap->priv;
  966. struct i2c_adapter *tun_i2c = dib7000p_get_i2c_master(adap->fe, DIBX000_I2C_INTERFACE_TUNER, 1);
  967. if (adap->id == 0) {
  968. if (dvb_attach(dib0070_attach, adap->fe, tun_i2c, &dib7070p_dib0070_config[0]) == NULL)
  969. return -ENODEV;
  970. } else {
  971. if (dvb_attach(dib0070_attach, adap->fe, tun_i2c, &dib7070p_dib0070_config[1]) == NULL)
  972. return -ENODEV;
  973. }
  974. st->set_param_save = adap->fe->ops.tuner_ops.set_params;
  975. adap->fe->ops.tuner_ops.set_params = dib7070_set_param_override;
  976. return 0;
  977. }
  978. static struct dibx000_bandwidth_config dib7070_bw_config_12_mhz = {
  979. 60000, 15000, // internal, sampling
  980. 1, 20, 3, 1, 0, // pll_cfg: prediv, ratio, range, reset, bypass
  981. 0, 0, 1, 1, 2, // misc: refdiv, bypclk_div, IO_CLK_en_core, ADClkSrc, modulo
  982. (3 << 14) | (1 << 12) | (524 << 0), // sad_cfg: refsel, sel, freq_15k
  983. (0 << 25) | 0, // ifreq = 0.000000 MHz
  984. 20452225, // timf
  985. 12000000, // xtal_hz
  986. };
  987. static struct dib7000p_config dib7070p_dib7000p_config = {
  988. .output_mpeg2_in_188_bytes = 1,
  989. .agc_config_count = 1,
  990. .agc = &dib7070_agc_config,
  991. .bw = &dib7070_bw_config_12_mhz,
  992. .tuner_is_baseband = 1,
  993. .spur_protect = 1,
  994. .gpio_dir = DIB7000P_GPIO_DEFAULT_DIRECTIONS,
  995. .gpio_val = DIB7000P_GPIO_DEFAULT_VALUES,
  996. .gpio_pwm_pos = DIB7000P_GPIO_DEFAULT_PWM_POS,
  997. .hostbus_diversity = 1,
  998. };
  999. /* STK7070P */
  1000. static int stk7070p_frontend_attach(struct dvb_usb_adapter *adap)
  1001. {
  1002. struct usb_device_descriptor *p = &adap->dev->udev->descriptor;
  1003. if (p->idVendor == cpu_to_le16(USB_VID_PINNACLE) &&
  1004. p->idProduct == cpu_to_le16(USB_PID_PINNACLE_PCTV72E))
  1005. dib0700_set_gpio(adap->dev, GPIO6, GPIO_OUT, 0);
  1006. else
  1007. dib0700_set_gpio(adap->dev, GPIO6, GPIO_OUT, 1);
  1008. msleep(10);
  1009. dib0700_set_gpio(adap->dev, GPIO9, GPIO_OUT, 1);
  1010. dib0700_set_gpio(adap->dev, GPIO4, GPIO_OUT, 1);
  1011. dib0700_set_gpio(adap->dev, GPIO7, GPIO_OUT, 1);
  1012. dib0700_set_gpio(adap->dev, GPIO10, GPIO_OUT, 0);
  1013. dib0700_ctrl_clock(adap->dev, 72, 1);
  1014. msleep(10);
  1015. dib0700_set_gpio(adap->dev, GPIO10, GPIO_OUT, 1);
  1016. msleep(10);
  1017. dib0700_set_gpio(adap->dev, GPIO0, GPIO_OUT, 1);
  1018. dib7000p_i2c_enumeration(&adap->dev->i2c_adap, 1, 18,
  1019. &dib7070p_dib7000p_config);
  1020. adap->fe = dvb_attach(dib7000p_attach, &adap->dev->i2c_adap, 0x80,
  1021. &dib7070p_dib7000p_config);
  1022. return adap->fe == NULL ? -ENODEV : 0;
  1023. }
  1024. /* STK7070PD */
  1025. static struct dib7000p_config stk7070pd_dib7000p_config[2] = {
  1026. {
  1027. .output_mpeg2_in_188_bytes = 1,
  1028. .agc_config_count = 1,
  1029. .agc = &dib7070_agc_config,
  1030. .bw = &dib7070_bw_config_12_mhz,
  1031. .tuner_is_baseband = 1,
  1032. .spur_protect = 1,
  1033. .gpio_dir = DIB7000P_GPIO_DEFAULT_DIRECTIONS,
  1034. .gpio_val = DIB7000P_GPIO_DEFAULT_VALUES,
  1035. .gpio_pwm_pos = DIB7000P_GPIO_DEFAULT_PWM_POS,
  1036. .hostbus_diversity = 1,
  1037. }, {
  1038. .output_mpeg2_in_188_bytes = 1,
  1039. .agc_config_count = 1,
  1040. .agc = &dib7070_agc_config,
  1041. .bw = &dib7070_bw_config_12_mhz,
  1042. .tuner_is_baseband = 1,
  1043. .spur_protect = 1,
  1044. .gpio_dir = DIB7000P_GPIO_DEFAULT_DIRECTIONS,
  1045. .gpio_val = DIB7000P_GPIO_DEFAULT_VALUES,
  1046. .gpio_pwm_pos = DIB7000P_GPIO_DEFAULT_PWM_POS,
  1047. .hostbus_diversity = 1,
  1048. }
  1049. };
  1050. static int stk7070pd_frontend_attach0(struct dvb_usb_adapter *adap)
  1051. {
  1052. dib0700_set_gpio(adap->dev, GPIO6, GPIO_OUT, 1);
  1053. msleep(10);
  1054. dib0700_set_gpio(adap->dev, GPIO9, GPIO_OUT, 1);
  1055. dib0700_set_gpio(adap->dev, GPIO4, GPIO_OUT, 1);
  1056. dib0700_set_gpio(adap->dev, GPIO7, GPIO_OUT, 1);
  1057. dib0700_set_gpio(adap->dev, GPIO10, GPIO_OUT, 0);
  1058. dib0700_ctrl_clock(adap->dev, 72, 1);
  1059. msleep(10);
  1060. dib0700_set_gpio(adap->dev, GPIO10, GPIO_OUT, 1);
  1061. msleep(10);
  1062. dib0700_set_gpio(adap->dev, GPIO0, GPIO_OUT, 1);
  1063. dib7000p_i2c_enumeration(&adap->dev->i2c_adap, 2, 18, stk7070pd_dib7000p_config);
  1064. adap->fe = dvb_attach(dib7000p_attach, &adap->dev->i2c_adap, 0x80, &stk7070pd_dib7000p_config[0]);
  1065. return adap->fe == NULL ? -ENODEV : 0;
  1066. }
  1067. static int stk7070pd_frontend_attach1(struct dvb_usb_adapter *adap)
  1068. {
  1069. adap->fe = dvb_attach(dib7000p_attach, &adap->dev->i2c_adap, 0x82, &stk7070pd_dib7000p_config[1]);
  1070. return adap->fe == NULL ? -ENODEV : 0;
  1071. }
  1072. /* S5H1411 */
  1073. static struct s5h1411_config pinnacle_801e_config = {
  1074. .output_mode = S5H1411_PARALLEL_OUTPUT,
  1075. .gpio = S5H1411_GPIO_OFF,
  1076. .mpeg_timing = S5H1411_MPEGTIMING_NONCONTINOUS_NONINVERTING_CLOCK,
  1077. .qam_if = S5H1411_IF_44000,
  1078. .vsb_if = S5H1411_IF_44000,
  1079. .inversion = S5H1411_INVERSION_OFF,
  1080. .status_mode = S5H1411_DEMODLOCKING
  1081. };
  1082. /* Pinnacle PCTV HD Pro 801e GPIOs map:
  1083. GPIO0 - currently unknown
  1084. GPIO1 - xc5000 tuner reset
  1085. GPIO2 - CX25843 sleep
  1086. GPIO3 - currently unknown
  1087. GPIO4 - currently unknown
  1088. GPIO6 - currently unknown
  1089. GPIO7 - currently unknown
  1090. GPIO9 - currently unknown
  1091. GPIO10 - CX25843 reset
  1092. */
  1093. static int s5h1411_frontend_attach(struct dvb_usb_adapter *adap)
  1094. {
  1095. struct dib0700_state *st = adap->dev->priv;
  1096. /* Make use of the new i2c functions from FW 1.20 */
  1097. st->fw_use_new_i2c_api = 1;
  1098. /* The s5h1411 requires the dib0700 to not be in master mode */
  1099. st->disable_streaming_master_mode = 1;
  1100. /* All msleep values taken from Windows USB trace */
  1101. dib0700_set_gpio(adap->dev, GPIO0, GPIO_OUT, 0);
  1102. dib0700_set_gpio(adap->dev, GPIO3, GPIO_OUT, 0);
  1103. dib0700_set_gpio(adap->dev, GPIO6, GPIO_OUT, 1);
  1104. msleep(400);
  1105. dib0700_set_gpio(adap->dev, GPIO10, GPIO_OUT, 0);
  1106. msleep(60);
  1107. dib0700_set_gpio(adap->dev, GPIO10, GPIO_OUT, 1);
  1108. msleep(30);
  1109. dib0700_set_gpio(adap->dev, GPIO0, GPIO_OUT, 1);
  1110. dib0700_set_gpio(adap->dev, GPIO9, GPIO_OUT, 1);
  1111. dib0700_set_gpio(adap->dev, GPIO4, GPIO_OUT, 1);
  1112. dib0700_set_gpio(adap->dev, GPIO7, GPIO_OUT, 1);
  1113. dib0700_set_gpio(adap->dev, GPIO2, GPIO_OUT, 0);
  1114. msleep(30);
  1115. /* Put the CX25843 to sleep for now since we're in digital mode */
  1116. dib0700_set_gpio(adap->dev, GPIO2, GPIO_OUT, 1);
  1117. /* GPIOs are initialized, do the attach */
  1118. adap->fe = dvb_attach(s5h1411_attach, &pinnacle_801e_config,
  1119. &adap->dev->i2c_adap);
  1120. return adap->fe == NULL ? -ENODEV : 0;
  1121. }
  1122. static int dib0700_xc5000_tuner_callback(void *priv, int component,
  1123. int command, int arg)
  1124. {
  1125. struct dvb_usb_adapter *adap = priv;
  1126. if (command == XC5000_TUNER_RESET) {
  1127. /* Reset the tuner */
  1128. dib0700_set_gpio(adap->dev, GPIO1, GPIO_OUT, 0);
  1129. msleep(330); /* from Windows USB trace */
  1130. dib0700_set_gpio(adap->dev, GPIO1, GPIO_OUT, 1);
  1131. msleep(330); /* from Windows USB trace */
  1132. } else {
  1133. err("xc5000: unknown tuner callback command: %d\n", command);
  1134. return -EINVAL;
  1135. }
  1136. return 0;
  1137. }
  1138. static struct xc5000_config s5h1411_xc5000_tunerconfig = {
  1139. .i2c_address = 0x64,
  1140. .if_khz = 5380,
  1141. };
  1142. static int xc5000_tuner_attach(struct dvb_usb_adapter *adap)
  1143. {
  1144. /* FIXME: generalize & move to common area */
  1145. adap->fe->callback = dib0700_xc5000_tuner_callback;
  1146. return dvb_attach(xc5000_attach, adap->fe, &adap->dev->i2c_adap,
  1147. &s5h1411_xc5000_tunerconfig)
  1148. == NULL ? -ENODEV : 0;
  1149. }
  1150. /* DVB-USB and USB stuff follows */
  1151. struct usb_device_id dib0700_usb_id_table[] = {
  1152. /* 0 */ { USB_DEVICE(USB_VID_DIBCOM, USB_PID_DIBCOM_STK7700P) },
  1153. { USB_DEVICE(USB_VID_DIBCOM, USB_PID_DIBCOM_STK7700P_PC) },
  1154. { USB_DEVICE(USB_VID_HAUPPAUGE, USB_PID_HAUPPAUGE_NOVA_T_500) },
  1155. { USB_DEVICE(USB_VID_HAUPPAUGE, USB_PID_HAUPPAUGE_NOVA_T_500_2) },
  1156. { USB_DEVICE(USB_VID_HAUPPAUGE, USB_PID_HAUPPAUGE_NOVA_T_STICK) },
  1157. /* 5 */ { USB_DEVICE(USB_VID_AVERMEDIA, USB_PID_AVERMEDIA_VOLAR) },
  1158. { USB_DEVICE(USB_VID_COMPRO, USB_PID_COMPRO_VIDEOMATE_U500) },
  1159. { USB_DEVICE(USB_VID_UNIWILL, USB_PID_UNIWILL_STK7700P) },
  1160. { USB_DEVICE(USB_VID_LEADTEK, USB_PID_WINFAST_DTV_DONGLE_STK7700P) },
  1161. { USB_DEVICE(USB_VID_HAUPPAUGE, USB_PID_HAUPPAUGE_NOVA_T_STICK_2) },
  1162. /* 10 */{ USB_DEVICE(USB_VID_AVERMEDIA, USB_PID_AVERMEDIA_VOLAR_2) },
  1163. { USB_DEVICE(USB_VID_PINNACLE, USB_PID_PINNACLE_PCTV2000E) },
  1164. { USB_DEVICE(USB_VID_TERRATEC,
  1165. USB_PID_TERRATEC_CINERGY_DT_XS_DIVERSITY) },
  1166. { USB_DEVICE(USB_VID_HAUPPAUGE, USB_PID_HAUPPAUGE_NOVA_TD_STICK) },
  1167. { USB_DEVICE(USB_VID_DIBCOM, USB_PID_DIBCOM_STK7700D) },
  1168. /* 15 */{ USB_DEVICE(USB_VID_DIBCOM, USB_PID_DIBCOM_STK7070P) },
  1169. { USB_DEVICE(USB_VID_PINNACLE, USB_PID_PINNACLE_PCTV_DVB_T_FLASH) },
  1170. { USB_DEVICE(USB_VID_DIBCOM, USB_PID_DIBCOM_STK7070PD) },
  1171. { USB_DEVICE(USB_VID_PINNACLE,
  1172. USB_PID_PINNACLE_PCTV_DUAL_DIVERSITY_DVB_T) },
  1173. { USB_DEVICE(USB_VID_COMPRO, USB_PID_COMPRO_VIDEOMATE_U500_PC) },
  1174. /* 20 */{ USB_DEVICE(USB_VID_AVERMEDIA, USB_PID_AVERMEDIA_EXPRESS) },
  1175. { USB_DEVICE(USB_VID_GIGABYTE, USB_PID_GIGABYTE_U7000) },
  1176. { USB_DEVICE(USB_VID_ULTIMA_ELECTRONIC, USB_PID_ARTEC_T14BR) },
  1177. { USB_DEVICE(USB_VID_ASUS, USB_PID_ASUS_U3000) },
  1178. { USB_DEVICE(USB_VID_ASUS, USB_PID_ASUS_U3100) },
  1179. /* 25 */{ USB_DEVICE(USB_VID_HAUPPAUGE, USB_PID_HAUPPAUGE_NOVA_T_STICK_3) },
  1180. { USB_DEVICE(USB_VID_HAUPPAUGE, USB_PID_HAUPPAUGE_MYTV_T) },
  1181. { USB_DEVICE(USB_VID_TERRATEC, USB_PID_TERRATEC_CINERGY_HT_USB_XE) },
  1182. { USB_DEVICE(USB_VID_PINNACLE, USB_PID_PINNACLE_EXPRESSCARD_320CX) },
  1183. { USB_DEVICE(USB_VID_PINNACLE, USB_PID_PINNACLE_PCTV72E) },
  1184. /* 30 */{ USB_DEVICE(USB_VID_PINNACLE, USB_PID_PINNACLE_PCTV73E) },
  1185. { USB_DEVICE(USB_VID_YUAN, USB_PID_YUAN_EC372S) },
  1186. { USB_DEVICE(USB_VID_TERRATEC, USB_PID_TERRATEC_CINERGY_HT_EXPRESS) },
  1187. { USB_DEVICE(USB_VID_TERRATEC, USB_PID_TERRATEC_CINERGY_T_XXS) },
  1188. { USB_DEVICE(USB_VID_LEADTEK, USB_PID_WINFAST_DTV_DONGLE_STK7700P_2) },
  1189. /* 35 */{ USB_DEVICE(USB_VID_HAUPPAUGE, USB_PID_HAUPPAUGE_NOVA_TD_STICK_52009) },
  1190. { USB_DEVICE(USB_VID_HAUPPAUGE, USB_PID_HAUPPAUGE_NOVA_T_500_3) },
  1191. { USB_DEVICE(USB_VID_GIGABYTE, USB_PID_GIGABYTE_U8000) },
  1192. { USB_DEVICE(USB_VID_YUAN, USB_PID_YUAN_STK7700PH) },
  1193. { USB_DEVICE(USB_VID_ASUS, USB_PID_ASUS_U3000H) },
  1194. /* 40 */{ USB_DEVICE(USB_VID_PINNACLE, USB_PID_PINNACLE_PCTV801E) },
  1195. { USB_DEVICE(USB_VID_PINNACLE, USB_PID_PINNACLE_PCTV801E_SE) },
  1196. { 0 } /* Terminating entry */
  1197. };
  1198. MODULE_DEVICE_TABLE(usb, dib0700_usb_id_table);
  1199. #define DIB0700_DEFAULT_DEVICE_PROPERTIES \
  1200. .caps = DVB_USB_IS_AN_I2C_ADAPTER, \
  1201. .usb_ctrl = DEVICE_SPECIFIC, \
  1202. .firmware = "dvb-usb-dib0700-1.20.fw", \
  1203. .download_firmware = dib0700_download_firmware, \
  1204. .no_reconnect = 1, \
  1205. .size_of_priv = sizeof(struct dib0700_state), \
  1206. .i2c_algo = &dib0700_i2c_algo, \
  1207. .identify_state = dib0700_identify_state
  1208. #define DIB0700_DEFAULT_STREAMING_CONFIG(ep) \
  1209. .streaming_ctrl = dib0700_streaming_ctrl, \
  1210. .stream = { \
  1211. .type = USB_BULK, \
  1212. .count = 4, \
  1213. .endpoint = ep, \
  1214. .u = { \
  1215. .bulk = { \
  1216. .buffersize = 39480, \
  1217. } \
  1218. } \
  1219. }
  1220. struct dvb_usb_device_properties dib0700_devices[] = {
  1221. {
  1222. DIB0700_DEFAULT_DEVICE_PROPERTIES,
  1223. .num_adapters = 1,
  1224. .adapter = {
  1225. {
  1226. .frontend_attach = stk7700p_frontend_attach,
  1227. .tuner_attach = stk7700p_tuner_attach,
  1228. DIB0700_DEFAULT_STREAMING_CONFIG(0x02),
  1229. },
  1230. },
  1231. .num_device_descs = 8,
  1232. .devices = {
  1233. { "DiBcom STK7700P reference design",
  1234. { &dib0700_usb_id_table[0], &dib0700_usb_id_table[1] },
  1235. { NULL },
  1236. },
  1237. { "Hauppauge Nova-T Stick",
  1238. { &dib0700_usb_id_table[4], &dib0700_usb_id_table[9], NULL },
  1239. { NULL },
  1240. },
  1241. { "AVerMedia AVerTV DVB-T Volar",
  1242. { &dib0700_usb_id_table[5], &dib0700_usb_id_table[10] },
  1243. { NULL },
  1244. },
  1245. { "Compro Videomate U500",
  1246. { &dib0700_usb_id_table[6], &dib0700_usb_id_table[19] },
  1247. { NULL },
  1248. },
  1249. { "Uniwill STK7700P based (Hama and others)",
  1250. { &dib0700_usb_id_table[7], NULL },
  1251. { NULL },
  1252. },
  1253. { "Leadtek Winfast DTV Dongle (STK7700P based)",
  1254. { &dib0700_usb_id_table[8], &dib0700_usb_id_table[34] },
  1255. { NULL },
  1256. },
  1257. { "AVerMedia AVerTV DVB-T Express",
  1258. { &dib0700_usb_id_table[20] },
  1259. { NULL },
  1260. },
  1261. { "Gigabyte U7000",
  1262. { &dib0700_usb_id_table[21], NULL },
  1263. { NULL },
  1264. }
  1265. },
  1266. .rc_interval = DEFAULT_RC_INTERVAL,
  1267. .rc_key_map = dib0700_rc_keys,
  1268. .rc_key_map_size = ARRAY_SIZE(dib0700_rc_keys),
  1269. .rc_query = dib0700_rc_query
  1270. }, { DIB0700_DEFAULT_DEVICE_PROPERTIES,
  1271. .num_adapters = 2,
  1272. .adapter = {
  1273. {
  1274. .frontend_attach = bristol_frontend_attach,
  1275. .tuner_attach = bristol_tuner_attach,
  1276. DIB0700_DEFAULT_STREAMING_CONFIG(0x02),
  1277. }, {
  1278. .frontend_attach = bristol_frontend_attach,
  1279. .tuner_attach = bristol_tuner_attach,
  1280. DIB0700_DEFAULT_STREAMING_CONFIG(0x03),
  1281. }
  1282. },
  1283. .num_device_descs = 1,
  1284. .devices = {
  1285. { "Hauppauge Nova-T 500 Dual DVB-T",
  1286. { &dib0700_usb_id_table[2], &dib0700_usb_id_table[3], NULL },
  1287. { NULL },
  1288. },
  1289. },
  1290. .rc_interval = DEFAULT_RC_INTERVAL,
  1291. .rc_key_map = dib0700_rc_keys,
  1292. .rc_key_map_size = ARRAY_SIZE(dib0700_rc_keys),
  1293. .rc_query = dib0700_rc_query
  1294. }, { DIB0700_DEFAULT_DEVICE_PROPERTIES,
  1295. .num_adapters = 2,
  1296. .adapter = {
  1297. {
  1298. .frontend_attach = stk7700d_frontend_attach,
  1299. .tuner_attach = stk7700d_tuner_attach,
  1300. DIB0700_DEFAULT_STREAMING_CONFIG(0x02),
  1301. }, {
  1302. .frontend_attach = stk7700d_frontend_attach,
  1303. .tuner_attach = stk7700d_tuner_attach,
  1304. DIB0700_DEFAULT_STREAMING_CONFIG(0x03),
  1305. }
  1306. },
  1307. .num_device_descs = 4,
  1308. .devices = {
  1309. { "Pinnacle PCTV 2000e",
  1310. { &dib0700_usb_id_table[11], NULL },
  1311. { NULL },
  1312. },
  1313. { "Terratec Cinergy DT XS Diversity",
  1314. { &dib0700_usb_id_table[12], NULL },
  1315. { NULL },
  1316. },
  1317. { "Hauppauge Nova-TD Stick/Elgato Eye-TV Diversity",
  1318. { &dib0700_usb_id_table[13], NULL },
  1319. { NULL },
  1320. },
  1321. { "DiBcom STK7700D reference design",
  1322. { &dib0700_usb_id_table[14], NULL },
  1323. { NULL },
  1324. }
  1325. },
  1326. .rc_interval = DEFAULT_RC_INTERVAL,
  1327. .rc_key_map = dib0700_rc_keys,
  1328. .rc_key_map_size = ARRAY_SIZE(dib0700_rc_keys),
  1329. .rc_query = dib0700_rc_query
  1330. }, { DIB0700_DEFAULT_DEVICE_PROPERTIES,
  1331. .num_adapters = 1,
  1332. .adapter = {
  1333. {
  1334. .frontend_attach = stk7700P2_frontend_attach,
  1335. .tuner_attach = stk7700d_tuner_attach,
  1336. DIB0700_DEFAULT_STREAMING_CONFIG(0x02),
  1337. },
  1338. },
  1339. .num_device_descs = 2,
  1340. .devices = {
  1341. { "ASUS My Cinema U3000 Mini DVBT Tuner",
  1342. { &dib0700_usb_id_table[23], NULL },
  1343. { NULL },
  1344. },
  1345. { "Yuan EC372S",
  1346. { &dib0700_usb_id_table[31], NULL },
  1347. { NULL },
  1348. }
  1349. },
  1350. .rc_interval = DEFAULT_RC_INTERVAL,
  1351. .rc_key_map = dib0700_rc_keys,
  1352. .rc_key_map_size = ARRAY_SIZE(dib0700_rc_keys),
  1353. .rc_query = dib0700_rc_query
  1354. }, { DIB0700_DEFAULT_DEVICE_PROPERTIES,
  1355. .num_adapters = 1,
  1356. .adapter = {
  1357. {
  1358. .frontend_attach = stk7070p_frontend_attach,
  1359. .tuner_attach = dib7070p_tuner_attach,
  1360. DIB0700_DEFAULT_STREAMING_CONFIG(0x02),
  1361. .size_of_priv = sizeof(struct dib0700_adapter_state),
  1362. },
  1363. },
  1364. .num_device_descs = 9,
  1365. .devices = {
  1366. { "DiBcom STK7070P reference design",
  1367. { &dib0700_usb_id_table[15], NULL },
  1368. { NULL },
  1369. },
  1370. { "Pinnacle PCTV DVB-T Flash Stick",
  1371. { &dib0700_usb_id_table[16], NULL },
  1372. { NULL },
  1373. },
  1374. { "Artec T14BR DVB-T",
  1375. { &dib0700_usb_id_table[22], NULL },
  1376. { NULL },
  1377. },
  1378. { "ASUS My Cinema U3100 Mini DVBT Tuner",
  1379. { &dib0700_usb_id_table[24], NULL },
  1380. { NULL },
  1381. },
  1382. { "Hauppauge Nova-T Stick",
  1383. { &dib0700_usb_id_table[25], NULL },
  1384. { NULL },
  1385. },
  1386. { "Hauppauge Nova-T MyTV.t",
  1387. { &dib0700_usb_id_table[26], NULL },
  1388. { NULL },
  1389. },
  1390. { "Pinnacle PCTV 72e",
  1391. { &dib0700_usb_id_table[29], NULL },
  1392. { NULL },
  1393. },
  1394. { "Pinnacle PCTV 73e",
  1395. { &dib0700_usb_id_table[30], NULL },
  1396. { NULL },
  1397. },
  1398. { "Terratec Cinergy T USB XXS",
  1399. { &dib0700_usb_id_table[33], NULL },
  1400. { NULL },
  1401. },
  1402. },
  1403. .rc_interval = DEFAULT_RC_INTERVAL,
  1404. .rc_key_map = dib0700_rc_keys,
  1405. .rc_key_map_size = ARRAY_SIZE(dib0700_rc_keys),
  1406. .rc_query = dib0700_rc_query
  1407. }, { DIB0700_DEFAULT_DEVICE_PROPERTIES,
  1408. .num_adapters = 2,
  1409. .adapter = {
  1410. {
  1411. .frontend_attach = stk7070pd_frontend_attach0,
  1412. .tuner_attach = dib7070p_tuner_attach,
  1413. DIB0700_DEFAULT_STREAMING_CONFIG(0x02),
  1414. .size_of_priv = sizeof(struct dib0700_adapter_state),
  1415. }, {
  1416. .frontend_attach = stk7070pd_frontend_attach1,
  1417. .tuner_attach = dib7070p_tuner_attach,
  1418. DIB0700_DEFAULT_STREAMING_CONFIG(0x03),
  1419. .size_of_priv = sizeof(struct dib0700_adapter_state),
  1420. }
  1421. },
  1422. .num_device_descs = 4,
  1423. .devices = {
  1424. { "DiBcom STK7070PD reference design",
  1425. { &dib0700_usb_id_table[17], NULL },
  1426. { NULL },
  1427. },
  1428. { "Pinnacle PCTV Dual DVB-T Diversity Stick",
  1429. { &dib0700_usb_id_table[18], NULL },
  1430. { NULL },
  1431. },
  1432. { "Hauppauge Nova-TD Stick (52009)",
  1433. { &dib0700_usb_id_table[35], NULL },
  1434. { NULL },
  1435. },
  1436. { "Hauppauge Nova-TD-500 (84xxx)",
  1437. { &dib0700_usb_id_table[36], NULL },
  1438. { NULL },
  1439. }
  1440. }
  1441. }, { DIB0700_DEFAULT_DEVICE_PROPERTIES,
  1442. .num_adapters = 1,
  1443. .adapter = {
  1444. {
  1445. .frontend_attach = stk7700ph_frontend_attach,
  1446. .tuner_attach = stk7700ph_tuner_attach,
  1447. DIB0700_DEFAULT_STREAMING_CONFIG(0x02),
  1448. .size_of_priv = sizeof(struct
  1449. dib0700_adapter_state),
  1450. },
  1451. },
  1452. .num_device_descs = 5,
  1453. .devices = {
  1454. { "Terratec Cinergy HT USB XE",
  1455. { &dib0700_usb_id_table[27], NULL },
  1456. { NULL },
  1457. },
  1458. { "Pinnacle Expresscard 320cx",
  1459. { &dib0700_usb_id_table[28], NULL },
  1460. { NULL },
  1461. },
  1462. { "Terratec Cinergy HT Express",
  1463. { &dib0700_usb_id_table[32], NULL },
  1464. { NULL },
  1465. },
  1466. { "Gigabyte U8000-RH",
  1467. { &dib0700_usb_id_table[37], NULL },
  1468. { NULL },
  1469. },
  1470. { "YUAN High-Tech STK7700PH",
  1471. { &dib0700_usb_id_table[38], NULL },
  1472. { NULL },
  1473. },
  1474. { "Asus My Cinema-U3000Hybrid",
  1475. { &dib0700_usb_id_table[39], NULL },
  1476. { NULL },
  1477. },
  1478. },
  1479. .rc_interval = DEFAULT_RC_INTERVAL,
  1480. .rc_key_map = dib0700_rc_keys,
  1481. .rc_key_map_size = ARRAY_SIZE(dib0700_rc_keys),
  1482. .rc_query = dib0700_rc_query
  1483. }, { DIB0700_DEFAULT_DEVICE_PROPERTIES,
  1484. .num_adapters = 1,
  1485. .adapter = {
  1486. {
  1487. .frontend_attach = s5h1411_frontend_attach,
  1488. .tuner_attach = xc5000_tuner_attach,
  1489. DIB0700_DEFAULT_STREAMING_CONFIG(0x02),
  1490. .size_of_priv = sizeof(struct
  1491. dib0700_adapter_state),
  1492. },
  1493. },
  1494. .num_device_descs = 2,
  1495. .devices = {
  1496. { "Pinnacle PCTV HD Pro USB Stick",
  1497. { &dib0700_usb_id_table[40], NULL },
  1498. { NULL },
  1499. },
  1500. { "Pinnacle PCTV HD USB Stick",
  1501. { &dib0700_usb_id_table[41], NULL },
  1502. { NULL },
  1503. },
  1504. },
  1505. .rc_interval = DEFAULT_RC_INTERVAL,
  1506. .rc_key_map = dib0700_rc_keys,
  1507. .rc_key_map_size = ARRAY_SIZE(dib0700_rc_keys),
  1508. .rc_query = dib0700_rc_query
  1509. },
  1510. };
  1511. int dib0700_device_count = ARRAY_SIZE(dib0700_devices);