dw2102.c 36 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213121412151216121712181219122012211222122312241225122612271228122912301231123212331234123512361237123812391240124112421243124412451246124712481249125012511252125312541255125612571258125912601261126212631264126512661267126812691270127112721273127412751276127712781279128012811282128312841285128612871288128912901291129212931294129512961297129812991300130113021303130413051306130713081309131013111312131313141315131613171318131913201321132213231324132513261327132813291330133113321333133413351336133713381339134013411342134313441345134613471348134913501351135213531354135513561357135813591360136113621363136413651366136713681369137013711372137313741375137613771378137913801381138213831384138513861387138813891390139113921393139413951396139713981399140014011402140314041405140614071408140914101411141214131414141514161417141814191420142114221423142414251426142714281429143014311432143314341435143614371438143914401441144214431444144514461447144814491450145114521453145414551456145714581459146014611462146314641465146614671468146914701471147214731474147514761477147814791480148114821483148414851486148714881489149014911492149314941495149614971498149915001501150215031504150515061507150815091510
  1. /* DVB USB framework compliant Linux driver for the
  2. * DVBWorld DVB-S 2101, 2102, DVB-S2 2104, DVB-C 3101,
  3. * TeVii S600, S630, S650,
  4. * Prof 1100, 7500 Cards
  5. * Copyright (C) 2008,2009 Igor M. Liplianin (liplianin@me.by)
  6. *
  7. * This program is free software; you can redistribute it and/or modify it
  8. * under the terms of the GNU General Public License as published by the
  9. * Free Software Foundation, version 2.
  10. *
  11. * see Documentation/dvb/README.dvb-usb for more information
  12. */
  13. #include "dw2102.h"
  14. #include "si21xx.h"
  15. #include "stv0299.h"
  16. #include "z0194a.h"
  17. #include "stv0288.h"
  18. #include "stb6000.h"
  19. #include "eds1547.h"
  20. #include "cx24116.h"
  21. #include "tda1002x.h"
  22. #include "mt312.h"
  23. #include "zl10039.h"
  24. #include "ds3000.h"
  25. #include "stv0900.h"
  26. #include "stv6110.h"
  27. #include "stb6100.h"
  28. #include "stb6100_proc.h"
  29. #ifndef USB_PID_DW2102
  30. #define USB_PID_DW2102 0x2102
  31. #endif
  32. #ifndef USB_PID_DW2104
  33. #define USB_PID_DW2104 0x2104
  34. #endif
  35. #ifndef USB_PID_DW3101
  36. #define USB_PID_DW3101 0x3101
  37. #endif
  38. #ifndef USB_PID_CINERGY_S
  39. #define USB_PID_CINERGY_S 0x0064
  40. #endif
  41. #ifndef USB_PID_TEVII_S630
  42. #define USB_PID_TEVII_S630 0xd630
  43. #endif
  44. #ifndef USB_PID_TEVII_S650
  45. #define USB_PID_TEVII_S650 0xd650
  46. #endif
  47. #ifndef USB_PID_TEVII_S660
  48. #define USB_PID_TEVII_S660 0xd660
  49. #endif
  50. #ifndef USB_PID_PROF_1100
  51. #define USB_PID_PROF_1100 0xb012
  52. #endif
  53. #define DW210X_READ_MSG 0
  54. #define DW210X_WRITE_MSG 1
  55. #define REG_1F_SYMBOLRATE_BYTE0 0x1f
  56. #define REG_20_SYMBOLRATE_BYTE1 0x20
  57. #define REG_21_SYMBOLRATE_BYTE2 0x21
  58. /* on my own*/
  59. #define DW2102_VOLTAGE_CTRL (0x1800)
  60. #define DW2102_RC_QUERY (0x1a00)
  61. #define err_str "did not find the firmware file. (%s) " \
  62. "Please see linux/Documentation/dvb/ for more details " \
  63. "on firmware-problems."
  64. struct dvb_usb_rc_keys_table {
  65. struct dvb_usb_rc_key *rc_keys;
  66. int rc_keys_size;
  67. };
  68. /* debug */
  69. static int dvb_usb_dw2102_debug;
  70. module_param_named(debug, dvb_usb_dw2102_debug, int, 0644);
  71. MODULE_PARM_DESC(debug, "set debugging level (1=info 2=xfer 4=rc(or-able))."
  72. DVB_USB_DEBUG_STATUS);
  73. /* keymaps */
  74. static int ir_keymap;
  75. module_param_named(keymap, ir_keymap, int, 0644);
  76. MODULE_PARM_DESC(keymap, "set keymap 0=default 1=dvbworld 2=tevii 3=tbs ...");
  77. /* demod probe */
  78. static int demod_probe = 1;
  79. module_param_named(demod, demod_probe, int, 0644);
  80. MODULE_PARM_DESC(demod, "demod to probe (1=cx24116 2=stv0903+stv6110 "
  81. "4=stv0903+stb6100(or-able)).");
  82. DVB_DEFINE_MOD_OPT_ADAPTER_NR(adapter_nr);
  83. static int dw210x_op_rw(struct usb_device *dev, u8 request, u16 value,
  84. u16 index, u8 * data, u16 len, int flags)
  85. {
  86. int ret;
  87. u8 u8buf[len];
  88. unsigned int pipe = (flags == DW210X_READ_MSG) ?
  89. usb_rcvctrlpipe(dev, 0) : usb_sndctrlpipe(dev, 0);
  90. u8 request_type = (flags == DW210X_READ_MSG) ? USB_DIR_IN : USB_DIR_OUT;
  91. if (flags == DW210X_WRITE_MSG)
  92. memcpy(u8buf, data, len);
  93. ret = usb_control_msg(dev, pipe, request, request_type | USB_TYPE_VENDOR,
  94. value, index , u8buf, len, 2000);
  95. if (flags == DW210X_READ_MSG)
  96. memcpy(data, u8buf, len);
  97. return ret;
  98. }
  99. /* I2C */
  100. static int dw2102_i2c_transfer(struct i2c_adapter *adap, struct i2c_msg msg[],
  101. int num)
  102. {
  103. struct dvb_usb_device *d = i2c_get_adapdata(adap);
  104. int i = 0, ret = 0;
  105. u8 buf6[] = {0x2c, 0x05, 0xc0, 0, 0, 0, 0};
  106. u16 value;
  107. if (!d)
  108. return -ENODEV;
  109. if (mutex_lock_interruptible(&d->i2c_mutex) < 0)
  110. return -EAGAIN;
  111. switch (num) {
  112. case 2:
  113. /* read stv0299 register */
  114. value = msg[0].buf[0];/* register */
  115. for (i = 0; i < msg[1].len; i++) {
  116. value = value + i;
  117. ret = dw210x_op_rw(d->udev, 0xb5, value, 0,
  118. buf6, 2, DW210X_READ_MSG);
  119. msg[1].buf[i] = buf6[0];
  120. }
  121. break;
  122. case 1:
  123. switch (msg[0].addr) {
  124. case 0x68:
  125. /* write to stv0299 register */
  126. buf6[0] = 0x2a;
  127. buf6[1] = msg[0].buf[0];
  128. buf6[2] = msg[0].buf[1];
  129. ret = dw210x_op_rw(d->udev, 0xb2, 0, 0,
  130. buf6, 3, DW210X_WRITE_MSG);
  131. break;
  132. case 0x60:
  133. if (msg[0].flags == 0) {
  134. /* write to tuner pll */
  135. buf6[0] = 0x2c;
  136. buf6[1] = 5;
  137. buf6[2] = 0xc0;
  138. buf6[3] = msg[0].buf[0];
  139. buf6[4] = msg[0].buf[1];
  140. buf6[5] = msg[0].buf[2];
  141. buf6[6] = msg[0].buf[3];
  142. ret = dw210x_op_rw(d->udev, 0xb2, 0, 0,
  143. buf6, 7, DW210X_WRITE_MSG);
  144. } else {
  145. /* read from tuner */
  146. ret = dw210x_op_rw(d->udev, 0xb5, 0, 0,
  147. buf6, 1, DW210X_READ_MSG);
  148. msg[0].buf[0] = buf6[0];
  149. }
  150. break;
  151. case (DW2102_RC_QUERY):
  152. ret = dw210x_op_rw(d->udev, 0xb8, 0, 0,
  153. buf6, 2, DW210X_READ_MSG);
  154. msg[0].buf[0] = buf6[0];
  155. msg[0].buf[1] = buf6[1];
  156. break;
  157. case (DW2102_VOLTAGE_CTRL):
  158. buf6[0] = 0x30;
  159. buf6[1] = msg[0].buf[0];
  160. ret = dw210x_op_rw(d->udev, 0xb2, 0, 0,
  161. buf6, 2, DW210X_WRITE_MSG);
  162. break;
  163. }
  164. break;
  165. }
  166. mutex_unlock(&d->i2c_mutex);
  167. return num;
  168. }
  169. static int dw2102_serit_i2c_transfer(struct i2c_adapter *adap,
  170. struct i2c_msg msg[], int num)
  171. {
  172. struct dvb_usb_device *d = i2c_get_adapdata(adap);
  173. int ret = 0;
  174. u8 buf6[] = {0, 0, 0, 0, 0, 0, 0};
  175. if (!d)
  176. return -ENODEV;
  177. if (mutex_lock_interruptible(&d->i2c_mutex) < 0)
  178. return -EAGAIN;
  179. switch (num) {
  180. case 2:
  181. /* read si2109 register by number */
  182. buf6[0] = msg[0].addr << 1;
  183. buf6[1] = msg[0].len;
  184. buf6[2] = msg[0].buf[0];
  185. ret = dw210x_op_rw(d->udev, 0xc2, 0, 0,
  186. buf6, msg[0].len + 2, DW210X_WRITE_MSG);
  187. /* read si2109 register */
  188. ret = dw210x_op_rw(d->udev, 0xc3, 0xd0, 0,
  189. buf6, msg[1].len + 2, DW210X_READ_MSG);
  190. memcpy(msg[1].buf, buf6 + 2, msg[1].len);
  191. break;
  192. case 1:
  193. switch (msg[0].addr) {
  194. case 0x68:
  195. /* write to si2109 register */
  196. buf6[0] = msg[0].addr << 1;
  197. buf6[1] = msg[0].len;
  198. memcpy(buf6 + 2, msg[0].buf, msg[0].len);
  199. ret = dw210x_op_rw(d->udev, 0xc2, 0, 0, buf6,
  200. msg[0].len + 2, DW210X_WRITE_MSG);
  201. break;
  202. case(DW2102_RC_QUERY):
  203. ret = dw210x_op_rw(d->udev, 0xb8, 0, 0,
  204. buf6, 2, DW210X_READ_MSG);
  205. msg[0].buf[0] = buf6[0];
  206. msg[0].buf[1] = buf6[1];
  207. break;
  208. case(DW2102_VOLTAGE_CTRL):
  209. buf6[0] = 0x30;
  210. buf6[1] = msg[0].buf[0];
  211. ret = dw210x_op_rw(d->udev, 0xb2, 0, 0,
  212. buf6, 2, DW210X_WRITE_MSG);
  213. break;
  214. }
  215. break;
  216. }
  217. mutex_unlock(&d->i2c_mutex);
  218. return num;
  219. }
  220. static int dw2102_earda_i2c_transfer(struct i2c_adapter *adap, struct i2c_msg msg[], int num)
  221. {
  222. struct dvb_usb_device *d = i2c_get_adapdata(adap);
  223. int ret = 0;
  224. if (!d)
  225. return -ENODEV;
  226. if (mutex_lock_interruptible(&d->i2c_mutex) < 0)
  227. return -EAGAIN;
  228. switch (num) {
  229. case 2: {
  230. /* read */
  231. /* first write first register number */
  232. u8 ibuf[msg[1].len + 2], obuf[3];
  233. obuf[0] = msg[0].addr << 1;
  234. obuf[1] = msg[0].len;
  235. obuf[2] = msg[0].buf[0];
  236. ret = dw210x_op_rw(d->udev, 0xc2, 0, 0,
  237. obuf, msg[0].len + 2, DW210X_WRITE_MSG);
  238. /* second read registers */
  239. ret = dw210x_op_rw(d->udev, 0xc3, 0xd1 , 0,
  240. ibuf, msg[1].len + 2, DW210X_READ_MSG);
  241. memcpy(msg[1].buf, ibuf + 2, msg[1].len);
  242. break;
  243. }
  244. case 1:
  245. switch (msg[0].addr) {
  246. case 0x68: {
  247. /* write to register */
  248. u8 obuf[msg[0].len + 2];
  249. obuf[0] = msg[0].addr << 1;
  250. obuf[1] = msg[0].len;
  251. memcpy(obuf + 2, msg[0].buf, msg[0].len);
  252. ret = dw210x_op_rw(d->udev, 0xc2, 0, 0,
  253. obuf, msg[0].len + 2, DW210X_WRITE_MSG);
  254. break;
  255. }
  256. case 0x61: {
  257. /* write to tuner */
  258. u8 obuf[msg[0].len + 2];
  259. obuf[0] = msg[0].addr << 1;
  260. obuf[1] = msg[0].len;
  261. memcpy(obuf + 2, msg[0].buf, msg[0].len);
  262. ret = dw210x_op_rw(d->udev, 0xc2, 0, 0,
  263. obuf, msg[0].len + 2, DW210X_WRITE_MSG);
  264. break;
  265. }
  266. case(DW2102_RC_QUERY): {
  267. u8 ibuf[2];
  268. ret = dw210x_op_rw(d->udev, 0xb8, 0, 0,
  269. ibuf, 2, DW210X_READ_MSG);
  270. memcpy(msg[0].buf, ibuf , 2);
  271. break;
  272. }
  273. case(DW2102_VOLTAGE_CTRL): {
  274. u8 obuf[2];
  275. obuf[0] = 0x30;
  276. obuf[1] = msg[0].buf[0];
  277. ret = dw210x_op_rw(d->udev, 0xb2, 0, 0,
  278. obuf, 2, DW210X_WRITE_MSG);
  279. break;
  280. }
  281. }
  282. break;
  283. }
  284. mutex_unlock(&d->i2c_mutex);
  285. return num;
  286. }
  287. static int dw2104_i2c_transfer(struct i2c_adapter *adap, struct i2c_msg msg[], int num)
  288. {
  289. struct dvb_usb_device *d = i2c_get_adapdata(adap);
  290. int ret = 0;
  291. int len, i, j;
  292. if (!d)
  293. return -ENODEV;
  294. if (mutex_lock_interruptible(&d->i2c_mutex) < 0)
  295. return -EAGAIN;
  296. for (j = 0; j < num; j++) {
  297. switch (msg[j].addr) {
  298. case(DW2102_RC_QUERY): {
  299. u8 ibuf[2];
  300. ret = dw210x_op_rw(d->udev, 0xb8, 0, 0,
  301. ibuf, 2, DW210X_READ_MSG);
  302. memcpy(msg[j].buf, ibuf , 2);
  303. break;
  304. }
  305. case(DW2102_VOLTAGE_CTRL): {
  306. u8 obuf[2];
  307. obuf[0] = 0x30;
  308. obuf[1] = msg[j].buf[0];
  309. ret = dw210x_op_rw(d->udev, 0xb2, 0, 0,
  310. obuf, 2, DW210X_WRITE_MSG);
  311. break;
  312. }
  313. /*case 0x55: cx24116
  314. case 0x6a: stv0903
  315. case 0x68: ds3000, stv0903
  316. case 0x60: ts2020, stv6110, stb6100 */
  317. default: {
  318. if (msg[j].flags == I2C_M_RD) {
  319. /* read registers */
  320. u8 ibuf[msg[j].len + 2];
  321. ret = dw210x_op_rw(d->udev, 0xc3,
  322. (msg[j].addr << 1) + 1, 0,
  323. ibuf, msg[j].len + 2,
  324. DW210X_READ_MSG);
  325. memcpy(msg[j].buf, ibuf + 2, msg[j].len);
  326. mdelay(10);
  327. } else if (((msg[j].buf[0] == 0xb0) &&
  328. (msg[j].addr == 0x68)) ||
  329. ((msg[j].buf[0] == 0xf7) &&
  330. (msg[j].addr == 0x55))) {
  331. /* write firmware */
  332. u8 obuf[19];
  333. obuf[0] = msg[j].addr << 1;
  334. obuf[1] = (msg[j].len > 15 ? 17 : msg[j].len);
  335. obuf[2] = msg[j].buf[0];
  336. len = msg[j].len - 1;
  337. i = 1;
  338. do {
  339. memcpy(obuf + 3, msg[j].buf + i,
  340. (len > 16 ? 16 : len));
  341. ret = dw210x_op_rw(d->udev, 0xc2, 0, 0,
  342. obuf, (len > 16 ? 16 : len) + 3,
  343. DW210X_WRITE_MSG);
  344. i += 16;
  345. len -= 16;
  346. } while (len > 0);
  347. } else {
  348. /* write registers */
  349. u8 obuf[msg[j].len + 2];
  350. obuf[0] = msg[j].addr << 1;
  351. obuf[1] = msg[j].len;
  352. memcpy(obuf + 2, msg[j].buf, msg[j].len);
  353. ret = dw210x_op_rw(d->udev, 0xc2, 0, 0,
  354. obuf, msg[j].len + 2,
  355. DW210X_WRITE_MSG);
  356. }
  357. break;
  358. }
  359. }
  360. }
  361. mutex_unlock(&d->i2c_mutex);
  362. return num;
  363. }
  364. static int dw3101_i2c_transfer(struct i2c_adapter *adap, struct i2c_msg msg[],
  365. int num)
  366. {
  367. struct dvb_usb_device *d = i2c_get_adapdata(adap);
  368. int ret = 0, i;
  369. if (!d)
  370. return -ENODEV;
  371. if (mutex_lock_interruptible(&d->i2c_mutex) < 0)
  372. return -EAGAIN;
  373. switch (num) {
  374. case 2: {
  375. /* read */
  376. /* first write first register number */
  377. u8 ibuf[msg[1].len + 2], obuf[3];
  378. obuf[0] = msg[0].addr << 1;
  379. obuf[1] = msg[0].len;
  380. obuf[2] = msg[0].buf[0];
  381. ret = dw210x_op_rw(d->udev, 0xc2, 0, 0,
  382. obuf, msg[0].len + 2, DW210X_WRITE_MSG);
  383. /* second read registers */
  384. ret = dw210x_op_rw(d->udev, 0xc3, 0x19 , 0,
  385. ibuf, msg[1].len + 2, DW210X_READ_MSG);
  386. memcpy(msg[1].buf, ibuf + 2, msg[1].len);
  387. break;
  388. }
  389. case 1:
  390. switch (msg[0].addr) {
  391. case 0x60:
  392. case 0x0c: {
  393. /* write to register */
  394. u8 obuf[msg[0].len + 2];
  395. obuf[0] = msg[0].addr << 1;
  396. obuf[1] = msg[0].len;
  397. memcpy(obuf + 2, msg[0].buf, msg[0].len);
  398. ret = dw210x_op_rw(d->udev, 0xc2, 0, 0,
  399. obuf, msg[0].len + 2, DW210X_WRITE_MSG);
  400. break;
  401. }
  402. case(DW2102_RC_QUERY): {
  403. u8 ibuf[2];
  404. ret = dw210x_op_rw(d->udev, 0xb8, 0, 0,
  405. ibuf, 2, DW210X_READ_MSG);
  406. memcpy(msg[0].buf, ibuf , 2);
  407. break;
  408. }
  409. }
  410. break;
  411. }
  412. for (i = 0; i < num; i++) {
  413. deb_xfer("%02x:%02x: %s ", i, msg[i].addr,
  414. msg[i].flags == 0 ? ">>>" : "<<<");
  415. debug_dump(msg[i].buf, msg[i].len, deb_xfer);
  416. }
  417. mutex_unlock(&d->i2c_mutex);
  418. return num;
  419. }
  420. static int s6x0_i2c_transfer(struct i2c_adapter *adap, struct i2c_msg msg[],
  421. int num)
  422. {
  423. struct dvb_usb_device *d = i2c_get_adapdata(adap);
  424. struct usb_device *udev = d->udev;
  425. int ret = 0;
  426. int len, i, j;
  427. if (!d)
  428. return -ENODEV;
  429. if (mutex_lock_interruptible(&d->i2c_mutex) < 0)
  430. return -EAGAIN;
  431. for (j = 0; j < num; j++) {
  432. switch (msg[j].addr) {
  433. case (DW2102_RC_QUERY): {
  434. u8 ibuf[4];
  435. ret = dw210x_op_rw(d->udev, 0xb8, 0, 0,
  436. ibuf, 4, DW210X_READ_MSG);
  437. memcpy(msg[j].buf, ibuf + 1, 2);
  438. break;
  439. }
  440. case (DW2102_VOLTAGE_CTRL): {
  441. u8 obuf[2];
  442. obuf[0] = 1;
  443. obuf[1] = msg[j].buf[1];/* off-on */
  444. ret = dw210x_op_rw(d->udev, 0x8a, 0, 0,
  445. obuf, 2, DW210X_WRITE_MSG);
  446. obuf[0] = 3;
  447. obuf[1] = msg[j].buf[0];/* 13v-18v */
  448. ret = dw210x_op_rw(d->udev, 0x8a, 0, 0,
  449. obuf, 2, DW210X_WRITE_MSG);
  450. break;
  451. }
  452. /*case 0x55: cx24116
  453. case 0x6a: stv0903
  454. case 0x68: ds3000, stv0903
  455. case 0x60: ts2020, stv6110, stb6100
  456. case 0xa0: eeprom */
  457. default: {
  458. if (msg[j].flags == I2C_M_RD) {
  459. /* read registers */
  460. u8 ibuf[msg[j].len];
  461. ret = dw210x_op_rw(d->udev, 0x91, 0, 0,
  462. ibuf, msg[j].len,
  463. DW210X_READ_MSG);
  464. memcpy(msg[j].buf, ibuf, msg[j].len);
  465. break;
  466. } else if ((msg[j].buf[0] == 0xb0) &&
  467. (msg[j].addr == 0x68)) {
  468. /* write firmware */
  469. u8 obuf[19];
  470. obuf[0] = (msg[j].len > 16 ?
  471. 18 : msg[j].len + 1);
  472. obuf[1] = msg[j].addr << 1;
  473. obuf[2] = msg[j].buf[0];
  474. len = msg[j].len - 1;
  475. i = 1;
  476. do {
  477. memcpy(obuf + 3, msg[j].buf + i,
  478. (len > 16 ? 16 : len));
  479. ret = dw210x_op_rw(d->udev, 0x80, 0, 0,
  480. obuf, (len > 16 ? 16 : len) + 3,
  481. DW210X_WRITE_MSG);
  482. i += 16;
  483. len -= 16;
  484. } while (len > 0);
  485. } else if ((udev->descriptor.idProduct == 0x7500)
  486. && (j < (num - 1))) {
  487. /* write register addr before read */
  488. u8 obuf[msg[j].len + 2];
  489. obuf[0] = msg[j + 1].len;
  490. obuf[1] = (msg[j].addr << 1);
  491. memcpy(obuf + 2, msg[j].buf, msg[j].len);
  492. ret = dw210x_op_rw(d->udev, 0x92, 0, 0,
  493. obuf, msg[j].len + 2,
  494. DW210X_WRITE_MSG);
  495. break;
  496. } else {
  497. /* write registers */
  498. u8 obuf[msg[j].len + 2];
  499. obuf[0] = msg[j].len + 1;
  500. obuf[1] = (msg[j].addr << 1);
  501. memcpy(obuf + 2, msg[j].buf, msg[j].len);
  502. ret = dw210x_op_rw(d->udev,
  503. (num > 1 ? 0x90 : 0x80), 0, 0,
  504. obuf, msg[j].len + 2,
  505. DW210X_WRITE_MSG);
  506. break;
  507. }
  508. break;
  509. }
  510. }
  511. msleep(3);
  512. }
  513. mutex_unlock(&d->i2c_mutex);
  514. return num;
  515. }
  516. static u32 dw210x_i2c_func(struct i2c_adapter *adapter)
  517. {
  518. return I2C_FUNC_I2C;
  519. }
  520. static struct i2c_algorithm dw2102_i2c_algo = {
  521. .master_xfer = dw2102_i2c_transfer,
  522. .functionality = dw210x_i2c_func,
  523. };
  524. static struct i2c_algorithm dw2102_serit_i2c_algo = {
  525. .master_xfer = dw2102_serit_i2c_transfer,
  526. .functionality = dw210x_i2c_func,
  527. };
  528. static struct i2c_algorithm dw2102_earda_i2c_algo = {
  529. .master_xfer = dw2102_earda_i2c_transfer,
  530. .functionality = dw210x_i2c_func,
  531. };
  532. static struct i2c_algorithm dw2104_i2c_algo = {
  533. .master_xfer = dw2104_i2c_transfer,
  534. .functionality = dw210x_i2c_func,
  535. };
  536. static struct i2c_algorithm dw3101_i2c_algo = {
  537. .master_xfer = dw3101_i2c_transfer,
  538. .functionality = dw210x_i2c_func,
  539. };
  540. static struct i2c_algorithm s6x0_i2c_algo = {
  541. .master_xfer = s6x0_i2c_transfer,
  542. .functionality = dw210x_i2c_func,
  543. };
  544. static int dw210x_read_mac_address(struct dvb_usb_device *d, u8 mac[6])
  545. {
  546. int i;
  547. u8 ibuf[] = {0, 0};
  548. u8 eeprom[256], eepromline[16];
  549. for (i = 0; i < 256; i++) {
  550. if (dw210x_op_rw(d->udev, 0xb6, 0xa0 , i, ibuf, 2, DW210X_READ_MSG) < 0) {
  551. err("read eeprom failed.");
  552. return -1;
  553. } else {
  554. eepromline[i%16] = ibuf[0];
  555. eeprom[i] = ibuf[0];
  556. }
  557. if ((i % 16) == 15) {
  558. deb_xfer("%02x: ", i - 15);
  559. debug_dump(eepromline, 16, deb_xfer);
  560. }
  561. }
  562. memcpy(mac, eeprom + 8, 6);
  563. return 0;
  564. };
  565. static int s6x0_read_mac_address(struct dvb_usb_device *d, u8 mac[6])
  566. {
  567. int i, ret;
  568. u8 ibuf[] = { 0 }, obuf[] = { 0 };
  569. u8 eeprom[256], eepromline[16];
  570. struct i2c_msg msg[] = {
  571. {
  572. .addr = 0xa0 >> 1,
  573. .flags = 0,
  574. .buf = obuf,
  575. .len = 1,
  576. }, {
  577. .addr = 0xa0 >> 1,
  578. .flags = I2C_M_RD,
  579. .buf = ibuf,
  580. .len = 1,
  581. }
  582. };
  583. for (i = 0; i < 256; i++) {
  584. obuf[0] = i;
  585. ret = s6x0_i2c_transfer(&d->i2c_adap, msg, 2);
  586. if (ret != 2) {
  587. err("read eeprom failed.");
  588. return -1;
  589. } else {
  590. eepromline[i % 16] = ibuf[0];
  591. eeprom[i] = ibuf[0];
  592. }
  593. if ((i % 16) == 15) {
  594. deb_xfer("%02x: ", i - 15);
  595. debug_dump(eepromline, 16, deb_xfer);
  596. }
  597. }
  598. memcpy(mac, eeprom + 16, 6);
  599. return 0;
  600. };
  601. static int dw210x_set_voltage(struct dvb_frontend *fe, fe_sec_voltage_t voltage)
  602. {
  603. static u8 command_13v[] = {0x00, 0x01};
  604. static u8 command_18v[] = {0x01, 0x01};
  605. static u8 command_off[] = {0x00, 0x00};
  606. struct i2c_msg msg = {
  607. .addr = DW2102_VOLTAGE_CTRL,
  608. .flags = 0,
  609. .buf = command_off,
  610. .len = 2,
  611. };
  612. struct dvb_usb_adapter *udev_adap =
  613. (struct dvb_usb_adapter *)(fe->dvb->priv);
  614. if (voltage == SEC_VOLTAGE_18)
  615. msg.buf = command_18v;
  616. else if (voltage == SEC_VOLTAGE_13)
  617. msg.buf = command_13v;
  618. i2c_transfer(&udev_adap->dev->i2c_adap, &msg, 1);
  619. return 0;
  620. }
  621. static struct stv0299_config sharp_z0194a_config = {
  622. .demod_address = 0x68,
  623. .inittab = sharp_z0194a_inittab,
  624. .mclk = 88000000UL,
  625. .invert = 1,
  626. .skip_reinit = 0,
  627. .lock_output = STV0299_LOCKOUTPUT_1,
  628. .volt13_op0_op1 = STV0299_VOLT13_OP1,
  629. .min_delay_ms = 100,
  630. .set_symbol_rate = sharp_z0194a_set_symbol_rate,
  631. };
  632. static struct cx24116_config dw2104_config = {
  633. .demod_address = 0x55,
  634. .mpg_clk_pos_pol = 0x01,
  635. };
  636. static struct si21xx_config serit_sp1511lhb_config = {
  637. .demod_address = 0x68,
  638. .min_delay_ms = 100,
  639. };
  640. static struct tda10023_config dw3101_tda10023_config = {
  641. .demod_address = 0x0c,
  642. .invert = 1,
  643. };
  644. static struct mt312_config zl313_config = {
  645. .demod_address = 0x0e,
  646. };
  647. static struct ds3000_config dw2104_ds3000_config = {
  648. .demod_address = 0x68,
  649. };
  650. static struct stv0900_config dw2104a_stv0900_config = {
  651. .demod_address = 0x6a,
  652. .demod_mode = 0,
  653. .xtal = 27000000,
  654. .clkmode = 3,/* 0-CLKI, 2-XTALI, else AUTO */
  655. .diseqc_mode = 2,/* 2/3 PWM */
  656. .tun1_maddress = 0,/* 0x60 */
  657. .tun1_adc = 0,/* 2 Vpp */
  658. .path1_mode = 3,
  659. };
  660. static struct stb6100_config dw2104a_stb6100_config = {
  661. .tuner_address = 0x60,
  662. .refclock = 27000000,
  663. };
  664. static struct stv0900_config dw2104_stv0900_config = {
  665. .demod_address = 0x68,
  666. .demod_mode = 0,
  667. .xtal = 8000000,
  668. .clkmode = 3,
  669. .diseqc_mode = 2,
  670. .tun1_maddress = 0,
  671. .tun1_adc = 1,/* 1 Vpp */
  672. .path1_mode = 3,
  673. };
  674. static struct stv6110_config dw2104_stv6110_config = {
  675. .i2c_address = 0x60,
  676. .mclk = 16000000,
  677. .clk_div = 1,
  678. };
  679. static struct stv0900_config prof_7500_stv0900_config = {
  680. .demod_address = 0x6a,
  681. .demod_mode = 0,
  682. .xtal = 27000000,
  683. .clkmode = 3,/* 0-CLKI, 2-XTALI, else AUTO */
  684. .diseqc_mode = 2,/* 2/3 PWM */
  685. .tun1_maddress = 0,/* 0x60 */
  686. .tun1_adc = 0,/* 2 Vpp */
  687. .path1_mode = 3,
  688. .tun1_type = 3,
  689. };
  690. static int dw2104_frontend_attach(struct dvb_usb_adapter *d)
  691. {
  692. struct dvb_tuner_ops *tuner_ops = NULL;
  693. if (demod_probe & 4) {
  694. d->fe = dvb_attach(stv0900_attach, &dw2104a_stv0900_config,
  695. &d->dev->i2c_adap, 0);
  696. if (d->fe != NULL) {
  697. if (dvb_attach(stb6100_attach, d->fe,
  698. &dw2104a_stb6100_config,
  699. &d->dev->i2c_adap)) {
  700. tuner_ops = &d->fe->ops.tuner_ops;
  701. tuner_ops->set_frequency = stb6100_set_freq;
  702. tuner_ops->get_frequency = stb6100_get_freq;
  703. tuner_ops->set_bandwidth = stb6100_set_bandw;
  704. tuner_ops->get_bandwidth = stb6100_get_bandw;
  705. d->fe->ops.set_voltage = dw210x_set_voltage;
  706. info("Attached STV0900+STB6100!\n");
  707. return 0;
  708. }
  709. }
  710. }
  711. if (demod_probe & 2) {
  712. d->fe = dvb_attach(stv0900_attach, &dw2104_stv0900_config,
  713. &d->dev->i2c_adap, 0);
  714. if (d->fe != NULL) {
  715. if (dvb_attach(stv6110_attach, d->fe,
  716. &dw2104_stv6110_config,
  717. &d->dev->i2c_adap)) {
  718. d->fe->ops.set_voltage = dw210x_set_voltage;
  719. info("Attached STV0900+STV6110A!\n");
  720. return 0;
  721. }
  722. }
  723. }
  724. if (demod_probe & 1) {
  725. d->fe = dvb_attach(cx24116_attach, &dw2104_config,
  726. &d->dev->i2c_adap);
  727. if (d->fe != NULL) {
  728. d->fe->ops.set_voltage = dw210x_set_voltage;
  729. info("Attached cx24116!\n");
  730. return 0;
  731. }
  732. }
  733. d->fe = dvb_attach(ds3000_attach, &dw2104_ds3000_config,
  734. &d->dev->i2c_adap);
  735. if (d->fe != NULL) {
  736. d->fe->ops.set_voltage = dw210x_set_voltage;
  737. info("Attached DS3000!\n");
  738. return 0;
  739. }
  740. return -EIO;
  741. }
  742. static struct dvb_usb_device_properties dw2102_properties;
  743. static struct dvb_usb_device_properties dw2104_properties;
  744. static struct dvb_usb_device_properties s6x0_properties;
  745. static int dw2102_frontend_attach(struct dvb_usb_adapter *d)
  746. {
  747. if (dw2102_properties.i2c_algo == &dw2102_serit_i2c_algo) {
  748. /*dw2102_properties.adapter->tuner_attach = NULL;*/
  749. d->fe = dvb_attach(si21xx_attach, &serit_sp1511lhb_config,
  750. &d->dev->i2c_adap);
  751. if (d->fe != NULL) {
  752. d->fe->ops.set_voltage = dw210x_set_voltage;
  753. info("Attached si21xx!\n");
  754. return 0;
  755. }
  756. }
  757. if (dw2102_properties.i2c_algo == &dw2102_earda_i2c_algo) {
  758. d->fe = dvb_attach(stv0288_attach, &earda_config,
  759. &d->dev->i2c_adap);
  760. if (d->fe != NULL) {
  761. if (dvb_attach(stb6000_attach, d->fe, 0x61,
  762. &d->dev->i2c_adap)) {
  763. d->fe->ops.set_voltage = dw210x_set_voltage;
  764. info("Attached stv0288!\n");
  765. return 0;
  766. }
  767. }
  768. }
  769. if (dw2102_properties.i2c_algo == &dw2102_i2c_algo) {
  770. /*dw2102_properties.adapter->tuner_attach = dw2102_tuner_attach;*/
  771. d->fe = dvb_attach(stv0299_attach, &sharp_z0194a_config,
  772. &d->dev->i2c_adap);
  773. if (d->fe != NULL) {
  774. d->fe->ops.set_voltage = dw210x_set_voltage;
  775. info("Attached stv0299!\n");
  776. return 0;
  777. }
  778. }
  779. return -EIO;
  780. }
  781. static int dw3101_frontend_attach(struct dvb_usb_adapter *d)
  782. {
  783. d->fe = dvb_attach(tda10023_attach, &dw3101_tda10023_config,
  784. &d->dev->i2c_adap, 0x48);
  785. if (d->fe != NULL) {
  786. info("Attached tda10023!\n");
  787. return 0;
  788. }
  789. return -EIO;
  790. }
  791. static int s6x0_frontend_attach(struct dvb_usb_adapter *d)
  792. {
  793. d->fe = dvb_attach(mt312_attach, &zl313_config,
  794. &d->dev->i2c_adap);
  795. if (d->fe != NULL) {
  796. if (dvb_attach(zl10039_attach, d->fe, 0x60,
  797. &d->dev->i2c_adap)) {
  798. d->fe->ops.set_voltage = dw210x_set_voltage;
  799. info("Attached zl100313+zl10039!\n");
  800. return 0;
  801. }
  802. }
  803. d->fe = dvb_attach(stv0288_attach, &earda_config,
  804. &d->dev->i2c_adap);
  805. if (d->fe != NULL) {
  806. if (dvb_attach(stb6000_attach, d->fe, 0x61,
  807. &d->dev->i2c_adap)) {
  808. d->fe->ops.set_voltage = dw210x_set_voltage;
  809. info("Attached stv0288+stb6000!\n");
  810. return 0;
  811. }
  812. }
  813. d->fe = dvb_attach(ds3000_attach, &dw2104_ds3000_config,
  814. &d->dev->i2c_adap);
  815. if (d->fe != NULL) {
  816. d->fe->ops.set_voltage = dw210x_set_voltage;
  817. info("Attached ds3000+ds2020!\n");
  818. return 0;
  819. }
  820. return -EIO;
  821. }
  822. static int prof_7500_frontend_attach(struct dvb_usb_adapter *d)
  823. {
  824. d->fe = dvb_attach(stv0900_attach, &prof_7500_stv0900_config,
  825. &d->dev->i2c_adap, 0);
  826. if (d->fe == NULL)
  827. return -EIO;
  828. d->fe->ops.set_voltage = dw210x_set_voltage;
  829. info("Attached STV0900+STB6100A!\n");
  830. return 0;
  831. }
  832. static int dw2102_tuner_attach(struct dvb_usb_adapter *adap)
  833. {
  834. dvb_attach(dvb_pll_attach, adap->fe, 0x60,
  835. &adap->dev->i2c_adap, DVB_PLL_OPERA1);
  836. return 0;
  837. }
  838. static int dw3101_tuner_attach(struct dvb_usb_adapter *adap)
  839. {
  840. dvb_attach(dvb_pll_attach, adap->fe, 0x60,
  841. &adap->dev->i2c_adap, DVB_PLL_TUA6034);
  842. return 0;
  843. }
  844. static struct dvb_usb_rc_key dw210x_rc_keys[] = {
  845. { 0xf80a, KEY_Q }, /*power*/
  846. { 0xf80c, KEY_M }, /*mute*/
  847. { 0xf811, KEY_1 },
  848. { 0xf812, KEY_2 },
  849. { 0xf813, KEY_3 },
  850. { 0xf814, KEY_4 },
  851. { 0xf815, KEY_5 },
  852. { 0xf816, KEY_6 },
  853. { 0xf817, KEY_7 },
  854. { 0xf818, KEY_8 },
  855. { 0xf819, KEY_9 },
  856. { 0xf810, KEY_0 },
  857. { 0xf81c, KEY_PAGEUP }, /*ch+*/
  858. { 0xf80f, KEY_PAGEDOWN }, /*ch-*/
  859. { 0xf81a, KEY_O }, /*vol+*/
  860. { 0xf80e, KEY_Z }, /*vol-*/
  861. { 0xf804, KEY_R }, /*rec*/
  862. { 0xf809, KEY_D }, /*fav*/
  863. { 0xf808, KEY_BACKSPACE }, /*rewind*/
  864. { 0xf807, KEY_A }, /*fast*/
  865. { 0xf80b, KEY_P }, /*pause*/
  866. { 0xf802, KEY_ESC }, /*cancel*/
  867. { 0xf803, KEY_G }, /*tab*/
  868. { 0xf800, KEY_UP }, /*up*/
  869. { 0xf81f, KEY_ENTER }, /*ok*/
  870. { 0xf801, KEY_DOWN }, /*down*/
  871. { 0xf805, KEY_C }, /*cap*/
  872. { 0xf806, KEY_S }, /*stop*/
  873. { 0xf840, KEY_F }, /*full*/
  874. { 0xf81e, KEY_W }, /*tvmode*/
  875. { 0xf81b, KEY_B }, /*recall*/
  876. };
  877. static struct dvb_usb_rc_key tevii_rc_keys[] = {
  878. { 0xf80a, KEY_POWER },
  879. { 0xf80c, KEY_MUTE },
  880. { 0xf811, KEY_1 },
  881. { 0xf812, KEY_2 },
  882. { 0xf813, KEY_3 },
  883. { 0xf814, KEY_4 },
  884. { 0xf815, KEY_5 },
  885. { 0xf816, KEY_6 },
  886. { 0xf817, KEY_7 },
  887. { 0xf818, KEY_8 },
  888. { 0xf819, KEY_9 },
  889. { 0xf810, KEY_0 },
  890. { 0xf81c, KEY_MENU },
  891. { 0xf80f, KEY_VOLUMEDOWN },
  892. { 0xf81a, KEY_LAST },
  893. { 0xf80e, KEY_OPEN },
  894. { 0xf804, KEY_RECORD },
  895. { 0xf809, KEY_VOLUMEUP },
  896. { 0xf808, KEY_CHANNELUP },
  897. { 0xf807, KEY_PVR },
  898. { 0xf80b, KEY_TIME },
  899. { 0xf802, KEY_RIGHT },
  900. { 0xf803, KEY_LEFT },
  901. { 0xf800, KEY_UP },
  902. { 0xf81f, KEY_OK },
  903. { 0xf801, KEY_DOWN },
  904. { 0xf805, KEY_TUNER },
  905. { 0xf806, KEY_CHANNELDOWN },
  906. { 0xf840, KEY_PLAYPAUSE },
  907. { 0xf81e, KEY_REWIND },
  908. { 0xf81b, KEY_FAVORITES },
  909. { 0xf81d, KEY_BACK },
  910. { 0xf84d, KEY_FASTFORWARD },
  911. { 0xf844, KEY_EPG },
  912. { 0xf84c, KEY_INFO },
  913. { 0xf841, KEY_AB },
  914. { 0xf843, KEY_AUDIO },
  915. { 0xf845, KEY_SUBTITLE },
  916. { 0xf84a, KEY_LIST },
  917. { 0xf846, KEY_F1 },
  918. { 0xf847, KEY_F2 },
  919. { 0xf85e, KEY_F3 },
  920. { 0xf85c, KEY_F4 },
  921. { 0xf852, KEY_F5 },
  922. { 0xf85a, KEY_F6 },
  923. { 0xf856, KEY_MODE },
  924. { 0xf858, KEY_SWITCHVIDEOMODE },
  925. };
  926. static struct dvb_usb_rc_key tbs_rc_keys[] = {
  927. { 0xf884, KEY_POWER },
  928. { 0xf894, KEY_MUTE },
  929. { 0xf887, KEY_1 },
  930. { 0xf886, KEY_2 },
  931. { 0xf885, KEY_3 },
  932. { 0xf88b, KEY_4 },
  933. { 0xf88a, KEY_5 },
  934. { 0xf889, KEY_6 },
  935. { 0xf88f, KEY_7 },
  936. { 0xf88e, KEY_8 },
  937. { 0xf88d, KEY_9 },
  938. { 0xf892, KEY_0 },
  939. { 0xf896, KEY_CHANNELUP },
  940. { 0xf891, KEY_CHANNELDOWN },
  941. { 0xf893, KEY_VOLUMEUP },
  942. { 0xf88c, KEY_VOLUMEDOWN },
  943. { 0xf883, KEY_RECORD },
  944. { 0xf898, KEY_PAUSE },
  945. { 0xf899, KEY_OK },
  946. { 0xf89a, KEY_SHUFFLE },
  947. { 0xf881, KEY_UP },
  948. { 0xf890, KEY_LEFT },
  949. { 0xf882, KEY_RIGHT },
  950. { 0xf888, KEY_DOWN },
  951. { 0xf895, KEY_FAVORITES },
  952. { 0xf897, KEY_SUBTITLE },
  953. { 0xf89d, KEY_ZOOM },
  954. { 0xf89f, KEY_EXIT },
  955. { 0xf89e, KEY_MENU },
  956. { 0xf89c, KEY_EPG },
  957. { 0xf880, KEY_PREVIOUS },
  958. { 0xf89b, KEY_MODE }
  959. };
  960. static struct dvb_usb_rc_keys_table keys_tables[] = {
  961. { dw210x_rc_keys, ARRAY_SIZE(dw210x_rc_keys) },
  962. { tevii_rc_keys, ARRAY_SIZE(tevii_rc_keys) },
  963. { tbs_rc_keys, ARRAY_SIZE(tbs_rc_keys) },
  964. };
  965. static int dw2102_rc_query(struct dvb_usb_device *d, u32 *event, int *state)
  966. {
  967. struct dvb_usb_rc_key *keymap = d->props.rc_key_map;
  968. int keymap_size = d->props.rc_key_map_size;
  969. u8 key[2];
  970. struct i2c_msg msg = {
  971. .addr = DW2102_RC_QUERY,
  972. .flags = I2C_M_RD,
  973. .buf = key,
  974. .len = 2
  975. };
  976. int i;
  977. /* override keymap */
  978. if ((ir_keymap > 0) && (ir_keymap <= ARRAY_SIZE(keys_tables))) {
  979. keymap = keys_tables[ir_keymap - 1].rc_keys ;
  980. keymap_size = keys_tables[ir_keymap - 1].rc_keys_size;
  981. }
  982. *state = REMOTE_NO_KEY_PRESSED;
  983. if (d->props.i2c_algo->master_xfer(&d->i2c_adap, &msg, 1) == 1) {
  984. for (i = 0; i < keymap_size ; i++) {
  985. if (rc5_data(&keymap[i]) == msg.buf[0]) {
  986. *state = REMOTE_KEY_PRESSED;
  987. *event = keymap[i].event;
  988. break;
  989. }
  990. }
  991. if ((*state) == REMOTE_KEY_PRESSED)
  992. deb_rc("%s: found rc key: %x, %x, event: %x\n",
  993. __func__, key[0], key[1], (*event));
  994. else if (key[0] != 0xff)
  995. deb_rc("%s: unknown rc key: %x, %x\n",
  996. __func__, key[0], key[1]);
  997. }
  998. return 0;
  999. }
  1000. static struct usb_device_id dw2102_table[] = {
  1001. {USB_DEVICE(USB_VID_CYPRESS, USB_PID_DW2102)},
  1002. {USB_DEVICE(USB_VID_CYPRESS, 0x2101)},
  1003. {USB_DEVICE(USB_VID_CYPRESS, USB_PID_DW2104)},
  1004. {USB_DEVICE(0x9022, USB_PID_TEVII_S650)},
  1005. {USB_DEVICE(USB_VID_TERRATEC, USB_PID_CINERGY_S)},
  1006. {USB_DEVICE(USB_VID_CYPRESS, USB_PID_DW3101)},
  1007. {USB_DEVICE(0x9022, USB_PID_TEVII_S630)},
  1008. {USB_DEVICE(0x3011, USB_PID_PROF_1100)},
  1009. {USB_DEVICE(0x9022, USB_PID_TEVII_S660)},
  1010. {USB_DEVICE(0x3034, 0x7500)},
  1011. { }
  1012. };
  1013. MODULE_DEVICE_TABLE(usb, dw2102_table);
  1014. static int dw2102_load_firmware(struct usb_device *dev,
  1015. const struct firmware *frmwr)
  1016. {
  1017. u8 *b, *p;
  1018. int ret = 0, i;
  1019. u8 reset;
  1020. u8 reset16[] = {0, 0, 0, 0, 0, 0, 0};
  1021. const struct firmware *fw;
  1022. const char *fw_2101 = "dvb-usb-dw2101.fw";
  1023. switch (dev->descriptor.idProduct) {
  1024. case 0x2101:
  1025. ret = request_firmware(&fw, fw_2101, &dev->dev);
  1026. if (ret != 0) {
  1027. err(err_str, fw_2101);
  1028. return ret;
  1029. }
  1030. break;
  1031. default:
  1032. fw = frmwr;
  1033. break;
  1034. }
  1035. info("start downloading DW210X firmware");
  1036. p = kmalloc(fw->size, GFP_KERNEL);
  1037. reset = 1;
  1038. /*stop the CPU*/
  1039. dw210x_op_rw(dev, 0xa0, 0x7f92, 0, &reset, 1, DW210X_WRITE_MSG);
  1040. dw210x_op_rw(dev, 0xa0, 0xe600, 0, &reset, 1, DW210X_WRITE_MSG);
  1041. if (p != NULL) {
  1042. memcpy(p, fw->data, fw->size);
  1043. for (i = 0; i < fw->size; i += 0x40) {
  1044. b = (u8 *) p + i;
  1045. if (dw210x_op_rw(dev, 0xa0, i, 0, b , 0x40,
  1046. DW210X_WRITE_MSG) != 0x40) {
  1047. err("error while transferring firmware");
  1048. ret = -EINVAL;
  1049. break;
  1050. }
  1051. }
  1052. /* restart the CPU */
  1053. reset = 0;
  1054. if (ret || dw210x_op_rw(dev, 0xa0, 0x7f92, 0, &reset, 1,
  1055. DW210X_WRITE_MSG) != 1) {
  1056. err("could not restart the USB controller CPU.");
  1057. ret = -EINVAL;
  1058. }
  1059. if (ret || dw210x_op_rw(dev, 0xa0, 0xe600, 0, &reset, 1,
  1060. DW210X_WRITE_MSG) != 1) {
  1061. err("could not restart the USB controller CPU.");
  1062. ret = -EINVAL;
  1063. }
  1064. /* init registers */
  1065. switch (dev->descriptor.idProduct) {
  1066. case USB_PID_PROF_1100:
  1067. s6x0_properties.rc_key_map = tbs_rc_keys;
  1068. s6x0_properties.rc_key_map_size =
  1069. ARRAY_SIZE(tbs_rc_keys);
  1070. break;
  1071. case USB_PID_TEVII_S650:
  1072. dw2104_properties.rc_key_map = tevii_rc_keys;
  1073. dw2104_properties.rc_key_map_size =
  1074. ARRAY_SIZE(tevii_rc_keys);
  1075. case USB_PID_DW2104:
  1076. reset = 1;
  1077. dw210x_op_rw(dev, 0xc4, 0x0000, 0, &reset, 1,
  1078. DW210X_WRITE_MSG);
  1079. /* break omitted intentionally */
  1080. case USB_PID_DW3101:
  1081. reset = 0;
  1082. dw210x_op_rw(dev, 0xbf, 0x0040, 0, &reset, 0,
  1083. DW210X_WRITE_MSG);
  1084. break;
  1085. case USB_PID_CINERGY_S:
  1086. case USB_PID_DW2102:
  1087. dw210x_op_rw(dev, 0xbf, 0x0040, 0, &reset, 0,
  1088. DW210X_WRITE_MSG);
  1089. dw210x_op_rw(dev, 0xb9, 0x0000, 0, &reset16[0], 2,
  1090. DW210X_READ_MSG);
  1091. /* check STV0299 frontend */
  1092. dw210x_op_rw(dev, 0xb5, 0, 0, &reset16[0], 2,
  1093. DW210X_READ_MSG);
  1094. if ((reset16[0] == 0xa1) || (reset16[0] == 0x80)) {
  1095. dw2102_properties.i2c_algo = &dw2102_i2c_algo;
  1096. dw2102_properties.adapter->tuner_attach = &dw2102_tuner_attach;
  1097. break;
  1098. } else {
  1099. /* check STV0288 frontend */
  1100. reset16[0] = 0xd0;
  1101. reset16[1] = 1;
  1102. reset16[2] = 0;
  1103. dw210x_op_rw(dev, 0xc2, 0, 0, &reset16[0], 3,
  1104. DW210X_WRITE_MSG);
  1105. dw210x_op_rw(dev, 0xc3, 0xd1, 0, &reset16[0], 3,
  1106. DW210X_READ_MSG);
  1107. if (reset16[2] == 0x11) {
  1108. dw2102_properties.i2c_algo = &dw2102_earda_i2c_algo;
  1109. break;
  1110. }
  1111. }
  1112. case 0x2101:
  1113. dw210x_op_rw(dev, 0xbc, 0x0030, 0, &reset16[0], 2,
  1114. DW210X_READ_MSG);
  1115. dw210x_op_rw(dev, 0xba, 0x0000, 0, &reset16[0], 7,
  1116. DW210X_READ_MSG);
  1117. dw210x_op_rw(dev, 0xba, 0x0000, 0, &reset16[0], 7,
  1118. DW210X_READ_MSG);
  1119. dw210x_op_rw(dev, 0xb9, 0x0000, 0, &reset16[0], 2,
  1120. DW210X_READ_MSG);
  1121. break;
  1122. }
  1123. msleep(100);
  1124. kfree(p);
  1125. }
  1126. return ret;
  1127. }
  1128. static struct dvb_usb_device_properties dw2102_properties = {
  1129. .caps = DVB_USB_IS_AN_I2C_ADAPTER,
  1130. .usb_ctrl = DEVICE_SPECIFIC,
  1131. .firmware = "dvb-usb-dw2102.fw",
  1132. .no_reconnect = 1,
  1133. .i2c_algo = &dw2102_serit_i2c_algo,
  1134. .rc_key_map = dw210x_rc_keys,
  1135. .rc_key_map_size = ARRAY_SIZE(dw210x_rc_keys),
  1136. .rc_interval = 150,
  1137. .rc_query = dw2102_rc_query,
  1138. .generic_bulk_ctrl_endpoint = 0x81,
  1139. /* parameter for the MPEG2-data transfer */
  1140. .num_adapters = 1,
  1141. .download_firmware = dw2102_load_firmware,
  1142. .read_mac_address = dw210x_read_mac_address,
  1143. .adapter = {
  1144. {
  1145. .frontend_attach = dw2102_frontend_attach,
  1146. .streaming_ctrl = NULL,
  1147. .tuner_attach = NULL,
  1148. .stream = {
  1149. .type = USB_BULK,
  1150. .count = 8,
  1151. .endpoint = 0x82,
  1152. .u = {
  1153. .bulk = {
  1154. .buffersize = 4096,
  1155. }
  1156. }
  1157. },
  1158. }
  1159. },
  1160. .num_device_descs = 3,
  1161. .devices = {
  1162. {"DVBWorld DVB-S 2102 USB2.0",
  1163. {&dw2102_table[0], NULL},
  1164. {NULL},
  1165. },
  1166. {"DVBWorld DVB-S 2101 USB2.0",
  1167. {&dw2102_table[1], NULL},
  1168. {NULL},
  1169. },
  1170. {"TerraTec Cinergy S USB",
  1171. {&dw2102_table[4], NULL},
  1172. {NULL},
  1173. },
  1174. }
  1175. };
  1176. static struct dvb_usb_device_properties dw2104_properties = {
  1177. .caps = DVB_USB_IS_AN_I2C_ADAPTER,
  1178. .usb_ctrl = DEVICE_SPECIFIC,
  1179. .firmware = "dvb-usb-dw2104.fw",
  1180. .no_reconnect = 1,
  1181. .i2c_algo = &dw2104_i2c_algo,
  1182. .rc_key_map = dw210x_rc_keys,
  1183. .rc_key_map_size = ARRAY_SIZE(dw210x_rc_keys),
  1184. .rc_interval = 150,
  1185. .rc_query = dw2102_rc_query,
  1186. .generic_bulk_ctrl_endpoint = 0x81,
  1187. /* parameter for the MPEG2-data transfer */
  1188. .num_adapters = 1,
  1189. .download_firmware = dw2102_load_firmware,
  1190. .read_mac_address = dw210x_read_mac_address,
  1191. .adapter = {
  1192. {
  1193. .frontend_attach = dw2104_frontend_attach,
  1194. .streaming_ctrl = NULL,
  1195. /*.tuner_attach = dw2104_tuner_attach,*/
  1196. .stream = {
  1197. .type = USB_BULK,
  1198. .count = 8,
  1199. .endpoint = 0x82,
  1200. .u = {
  1201. .bulk = {
  1202. .buffersize = 4096,
  1203. }
  1204. }
  1205. },
  1206. }
  1207. },
  1208. .num_device_descs = 2,
  1209. .devices = {
  1210. { "DVBWorld DW2104 USB2.0",
  1211. {&dw2102_table[2], NULL},
  1212. {NULL},
  1213. },
  1214. { "TeVii S650 USB2.0",
  1215. {&dw2102_table[3], NULL},
  1216. {NULL},
  1217. },
  1218. }
  1219. };
  1220. static struct dvb_usb_device_properties dw3101_properties = {
  1221. .caps = DVB_USB_IS_AN_I2C_ADAPTER,
  1222. .usb_ctrl = DEVICE_SPECIFIC,
  1223. .firmware = "dvb-usb-dw3101.fw",
  1224. .no_reconnect = 1,
  1225. .i2c_algo = &dw3101_i2c_algo,
  1226. .rc_key_map = dw210x_rc_keys,
  1227. .rc_key_map_size = ARRAY_SIZE(dw210x_rc_keys),
  1228. .rc_interval = 150,
  1229. .rc_query = dw2102_rc_query,
  1230. .generic_bulk_ctrl_endpoint = 0x81,
  1231. /* parameter for the MPEG2-data transfer */
  1232. .num_adapters = 1,
  1233. .download_firmware = dw2102_load_firmware,
  1234. .read_mac_address = dw210x_read_mac_address,
  1235. .adapter = {
  1236. {
  1237. .frontend_attach = dw3101_frontend_attach,
  1238. .streaming_ctrl = NULL,
  1239. .tuner_attach = dw3101_tuner_attach,
  1240. .stream = {
  1241. .type = USB_BULK,
  1242. .count = 8,
  1243. .endpoint = 0x82,
  1244. .u = {
  1245. .bulk = {
  1246. .buffersize = 4096,
  1247. }
  1248. }
  1249. },
  1250. }
  1251. },
  1252. .num_device_descs = 1,
  1253. .devices = {
  1254. { "DVBWorld DVB-C 3101 USB2.0",
  1255. {&dw2102_table[5], NULL},
  1256. {NULL},
  1257. },
  1258. }
  1259. };
  1260. static struct dvb_usb_device_properties s6x0_properties = {
  1261. .caps = DVB_USB_IS_AN_I2C_ADAPTER,
  1262. .usb_ctrl = DEVICE_SPECIFIC,
  1263. .firmware = "dvb-usb-s630.fw",
  1264. .no_reconnect = 1,
  1265. .i2c_algo = &s6x0_i2c_algo,
  1266. .rc_key_map = tevii_rc_keys,
  1267. .rc_key_map_size = ARRAY_SIZE(tevii_rc_keys),
  1268. .rc_interval = 150,
  1269. .rc_query = dw2102_rc_query,
  1270. .generic_bulk_ctrl_endpoint = 0x81,
  1271. .num_adapters = 1,
  1272. .download_firmware = dw2102_load_firmware,
  1273. .read_mac_address = s6x0_read_mac_address,
  1274. .adapter = {
  1275. {
  1276. .frontend_attach = s6x0_frontend_attach,
  1277. .streaming_ctrl = NULL,
  1278. .tuner_attach = NULL,
  1279. .stream = {
  1280. .type = USB_BULK,
  1281. .count = 8,
  1282. .endpoint = 0x82,
  1283. .u = {
  1284. .bulk = {
  1285. .buffersize = 4096,
  1286. }
  1287. }
  1288. },
  1289. }
  1290. },
  1291. .num_device_descs = 3,
  1292. .devices = {
  1293. {"TeVii S630 USB",
  1294. {&dw2102_table[6], NULL},
  1295. {NULL},
  1296. },
  1297. {"Prof 1100 USB ",
  1298. {&dw2102_table[7], NULL},
  1299. {NULL},
  1300. },
  1301. {"TeVii S660 USB",
  1302. {&dw2102_table[8], NULL},
  1303. {NULL},
  1304. },
  1305. }
  1306. };
  1307. struct dvb_usb_device_properties *p7500;
  1308. static struct dvb_usb_device_description d7500 = {
  1309. "Prof 7500 USB DVB-S2",
  1310. {&dw2102_table[9], NULL},
  1311. {NULL},
  1312. };
  1313. static int dw2102_probe(struct usb_interface *intf,
  1314. const struct usb_device_id *id)
  1315. {
  1316. p7500 = kzalloc(sizeof(struct dvb_usb_device_properties), GFP_KERNEL);
  1317. if (!p7500)
  1318. return -ENOMEM;
  1319. /* copy default structure */
  1320. memcpy(p7500, &s6x0_properties,
  1321. sizeof(struct dvb_usb_device_properties));
  1322. /* fill only different fields */
  1323. p7500->firmware = "dvb-usb-p7500.fw";
  1324. p7500->devices[0] = d7500;
  1325. p7500->rc_key_map = tbs_rc_keys;
  1326. p7500->rc_key_map_size = ARRAY_SIZE(tbs_rc_keys);
  1327. p7500->adapter->frontend_attach = prof_7500_frontend_attach;
  1328. if (0 == dvb_usb_device_init(intf, &dw2102_properties,
  1329. THIS_MODULE, NULL, adapter_nr) ||
  1330. 0 == dvb_usb_device_init(intf, &dw2104_properties,
  1331. THIS_MODULE, NULL, adapter_nr) ||
  1332. 0 == dvb_usb_device_init(intf, &dw3101_properties,
  1333. THIS_MODULE, NULL, adapter_nr) ||
  1334. 0 == dvb_usb_device_init(intf, &s6x0_properties,
  1335. THIS_MODULE, NULL, adapter_nr) ||
  1336. 0 == dvb_usb_device_init(intf, p7500,
  1337. THIS_MODULE, NULL, adapter_nr))
  1338. return 0;
  1339. return -ENODEV;
  1340. }
  1341. static struct usb_driver dw2102_driver = {
  1342. .name = "dw2102",
  1343. .probe = dw2102_probe,
  1344. .disconnect = dvb_usb_device_exit,
  1345. .id_table = dw2102_table,
  1346. };
  1347. static int __init dw2102_module_init(void)
  1348. {
  1349. int ret = usb_register(&dw2102_driver);
  1350. if (ret)
  1351. err("usb_register failed. Error number %d", ret);
  1352. return ret;
  1353. }
  1354. static void __exit dw2102_module_exit(void)
  1355. {
  1356. usb_deregister(&dw2102_driver);
  1357. }
  1358. module_init(dw2102_module_init);
  1359. module_exit(dw2102_module_exit);
  1360. MODULE_AUTHOR("Igor M. Liplianin (c) liplianin@me.by");
  1361. MODULE_DESCRIPTION("Driver for DVBWorld DVB-S 2101, 2102, DVB-S2 2104,"
  1362. " DVB-C 3101 USB2.0,"
  1363. " TeVii S600, S630, S650, S660 USB2.0,"
  1364. " Prof 1100, 7500 USB2.0 devices");
  1365. MODULE_VERSION("0.1");
  1366. MODULE_LICENSE("GPL");