dw2102.c 36 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244124512461247124812491250125112521253125412551256125712581259126012611262126312641265126612671268126912701271127212731274127512761277127812791280128112821283128412851286128712881289129012911292129312941295129612971298129913001301130213031304130513061307130813091310131113121313131413151316131713181319132013211322132313241325132613271328132913301331133213331334133513361337133813391340134113421343134413451346134713481349135013511352135313541355135613571358135913601361136213631364136513661367136813691370137113721373137413751376137713781379138013811382138313841385138613871388138913901391139213931394139513961397139813991400140114021403140414051406140714081409141014111412141314141415141614171418141914201421142214231424142514261427142814291430143114321433143414351436143714381439144014411442144314441445144614471448144914501451145214531454145514561457145814591460146114621463146414651466146714681469147014711472147314741475147614771478147914801481148214831484148514861487148814891490149114921493149414951496149714981499150015011502150315041505150615071508150915101511151215131514151515161517151815191520
  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 rc_map_dvb_usb_table_table {
  65. struct rc_map_table *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;
  425. int ret = 0;
  426. int len, i, j;
  427. if (!d)
  428. return -ENODEV;
  429. udev = d->udev;
  430. if (mutex_lock_interruptible(&d->i2c_mutex) < 0)
  431. return -EAGAIN;
  432. for (j = 0; j < num; j++) {
  433. switch (msg[j].addr) {
  434. case (DW2102_RC_QUERY): {
  435. u8 ibuf[4];
  436. ret = dw210x_op_rw(d->udev, 0xb8, 0, 0,
  437. ibuf, 4, DW210X_READ_MSG);
  438. memcpy(msg[j].buf, ibuf + 1, 2);
  439. break;
  440. }
  441. case (DW2102_VOLTAGE_CTRL): {
  442. u8 obuf[2];
  443. obuf[0] = 1;
  444. obuf[1] = msg[j].buf[1];/* off-on */
  445. ret = dw210x_op_rw(d->udev, 0x8a, 0, 0,
  446. obuf, 2, DW210X_WRITE_MSG);
  447. obuf[0] = 3;
  448. obuf[1] = msg[j].buf[0];/* 13v-18v */
  449. ret = dw210x_op_rw(d->udev, 0x8a, 0, 0,
  450. obuf, 2, DW210X_WRITE_MSG);
  451. break;
  452. }
  453. /*case 0x55: cx24116
  454. case 0x6a: stv0903
  455. case 0x68: ds3000, stv0903
  456. case 0x60: ts2020, stv6110, stb6100
  457. case 0xa0: eeprom */
  458. default: {
  459. if (msg[j].flags == I2C_M_RD) {
  460. /* read registers */
  461. u8 ibuf[msg[j].len];
  462. ret = dw210x_op_rw(d->udev, 0x91, 0, 0,
  463. ibuf, msg[j].len,
  464. DW210X_READ_MSG);
  465. memcpy(msg[j].buf, ibuf, msg[j].len);
  466. break;
  467. } else if ((msg[j].buf[0] == 0xb0) &&
  468. (msg[j].addr == 0x68)) {
  469. /* write firmware */
  470. u8 obuf[19];
  471. obuf[0] = (msg[j].len > 16 ?
  472. 18 : msg[j].len + 1);
  473. obuf[1] = msg[j].addr << 1;
  474. obuf[2] = msg[j].buf[0];
  475. len = msg[j].len - 1;
  476. i = 1;
  477. do {
  478. memcpy(obuf + 3, msg[j].buf + i,
  479. (len > 16 ? 16 : len));
  480. ret = dw210x_op_rw(d->udev, 0x80, 0, 0,
  481. obuf, (len > 16 ? 16 : len) + 3,
  482. DW210X_WRITE_MSG);
  483. i += 16;
  484. len -= 16;
  485. } while (len > 0);
  486. } else if ((udev->descriptor.idProduct == 0x7500)
  487. && (j < (num - 1))) {
  488. /* write register addr before read */
  489. u8 obuf[msg[j].len + 2];
  490. obuf[0] = msg[j + 1].len;
  491. obuf[1] = (msg[j].addr << 1);
  492. memcpy(obuf + 2, msg[j].buf, msg[j].len);
  493. ret = dw210x_op_rw(d->udev, 0x92, 0, 0,
  494. obuf, msg[j].len + 2,
  495. DW210X_WRITE_MSG);
  496. break;
  497. } else {
  498. /* write registers */
  499. u8 obuf[msg[j].len + 2];
  500. obuf[0] = msg[j].len + 1;
  501. obuf[1] = (msg[j].addr << 1);
  502. memcpy(obuf + 2, msg[j].buf, msg[j].len);
  503. ret = dw210x_op_rw(d->udev,
  504. (num > 1 ? 0x90 : 0x80), 0, 0,
  505. obuf, msg[j].len + 2,
  506. DW210X_WRITE_MSG);
  507. break;
  508. }
  509. break;
  510. }
  511. }
  512. msleep(3);
  513. }
  514. mutex_unlock(&d->i2c_mutex);
  515. return num;
  516. }
  517. static u32 dw210x_i2c_func(struct i2c_adapter *adapter)
  518. {
  519. return I2C_FUNC_I2C;
  520. }
  521. static struct i2c_algorithm dw2102_i2c_algo = {
  522. .master_xfer = dw2102_i2c_transfer,
  523. .functionality = dw210x_i2c_func,
  524. };
  525. static struct i2c_algorithm dw2102_serit_i2c_algo = {
  526. .master_xfer = dw2102_serit_i2c_transfer,
  527. .functionality = dw210x_i2c_func,
  528. };
  529. static struct i2c_algorithm dw2102_earda_i2c_algo = {
  530. .master_xfer = dw2102_earda_i2c_transfer,
  531. .functionality = dw210x_i2c_func,
  532. };
  533. static struct i2c_algorithm dw2104_i2c_algo = {
  534. .master_xfer = dw2104_i2c_transfer,
  535. .functionality = dw210x_i2c_func,
  536. };
  537. static struct i2c_algorithm dw3101_i2c_algo = {
  538. .master_xfer = dw3101_i2c_transfer,
  539. .functionality = dw210x_i2c_func,
  540. };
  541. static struct i2c_algorithm s6x0_i2c_algo = {
  542. .master_xfer = s6x0_i2c_transfer,
  543. .functionality = dw210x_i2c_func,
  544. };
  545. static int dw210x_read_mac_address(struct dvb_usb_device *d, u8 mac[6])
  546. {
  547. int i;
  548. u8 ibuf[] = {0, 0};
  549. u8 eeprom[256], eepromline[16];
  550. for (i = 0; i < 256; i++) {
  551. if (dw210x_op_rw(d->udev, 0xb6, 0xa0 , i, ibuf, 2, DW210X_READ_MSG) < 0) {
  552. err("read eeprom failed.");
  553. return -1;
  554. } else {
  555. eepromline[i%16] = ibuf[0];
  556. eeprom[i] = ibuf[0];
  557. }
  558. if ((i % 16) == 15) {
  559. deb_xfer("%02x: ", i - 15);
  560. debug_dump(eepromline, 16, deb_xfer);
  561. }
  562. }
  563. memcpy(mac, eeprom + 8, 6);
  564. return 0;
  565. };
  566. static int s6x0_read_mac_address(struct dvb_usb_device *d, u8 mac[6])
  567. {
  568. int i, ret;
  569. u8 ibuf[] = { 0 }, obuf[] = { 0 };
  570. u8 eeprom[256], eepromline[16];
  571. struct i2c_msg msg[] = {
  572. {
  573. .addr = 0xa0 >> 1,
  574. .flags = 0,
  575. .buf = obuf,
  576. .len = 1,
  577. }, {
  578. .addr = 0xa0 >> 1,
  579. .flags = I2C_M_RD,
  580. .buf = ibuf,
  581. .len = 1,
  582. }
  583. };
  584. for (i = 0; i < 256; i++) {
  585. obuf[0] = i;
  586. ret = s6x0_i2c_transfer(&d->i2c_adap, msg, 2);
  587. if (ret != 2) {
  588. err("read eeprom failed.");
  589. return -1;
  590. } else {
  591. eepromline[i % 16] = ibuf[0];
  592. eeprom[i] = ibuf[0];
  593. }
  594. if ((i % 16) == 15) {
  595. deb_xfer("%02x: ", i - 15);
  596. debug_dump(eepromline, 16, deb_xfer);
  597. }
  598. }
  599. memcpy(mac, eeprom + 16, 6);
  600. return 0;
  601. };
  602. static int dw210x_set_voltage(struct dvb_frontend *fe, fe_sec_voltage_t voltage)
  603. {
  604. static u8 command_13v[] = {0x00, 0x01};
  605. static u8 command_18v[] = {0x01, 0x01};
  606. static u8 command_off[] = {0x00, 0x00};
  607. struct i2c_msg msg = {
  608. .addr = DW2102_VOLTAGE_CTRL,
  609. .flags = 0,
  610. .buf = command_off,
  611. .len = 2,
  612. };
  613. struct dvb_usb_adapter *udev_adap =
  614. (struct dvb_usb_adapter *)(fe->dvb->priv);
  615. if (voltage == SEC_VOLTAGE_18)
  616. msg.buf = command_18v;
  617. else if (voltage == SEC_VOLTAGE_13)
  618. msg.buf = command_13v;
  619. i2c_transfer(&udev_adap->dev->i2c_adap, &msg, 1);
  620. return 0;
  621. }
  622. static struct stv0299_config sharp_z0194a_config = {
  623. .demod_address = 0x68,
  624. .inittab = sharp_z0194a_inittab,
  625. .mclk = 88000000UL,
  626. .invert = 1,
  627. .skip_reinit = 0,
  628. .lock_output = STV0299_LOCKOUTPUT_1,
  629. .volt13_op0_op1 = STV0299_VOLT13_OP1,
  630. .min_delay_ms = 100,
  631. .set_symbol_rate = sharp_z0194a_set_symbol_rate,
  632. };
  633. static struct cx24116_config dw2104_config = {
  634. .demod_address = 0x55,
  635. .mpg_clk_pos_pol = 0x01,
  636. };
  637. static struct si21xx_config serit_sp1511lhb_config = {
  638. .demod_address = 0x68,
  639. .min_delay_ms = 100,
  640. };
  641. static struct tda10023_config dw3101_tda10023_config = {
  642. .demod_address = 0x0c,
  643. .invert = 1,
  644. };
  645. static struct mt312_config zl313_config = {
  646. .demod_address = 0x0e,
  647. };
  648. static struct ds3000_config dw2104_ds3000_config = {
  649. .demod_address = 0x68,
  650. };
  651. static struct stv0900_config dw2104a_stv0900_config = {
  652. .demod_address = 0x6a,
  653. .demod_mode = 0,
  654. .xtal = 27000000,
  655. .clkmode = 3,/* 0-CLKI, 2-XTALI, else AUTO */
  656. .diseqc_mode = 2,/* 2/3 PWM */
  657. .tun1_maddress = 0,/* 0x60 */
  658. .tun1_adc = 0,/* 2 Vpp */
  659. .path1_mode = 3,
  660. };
  661. static struct stb6100_config dw2104a_stb6100_config = {
  662. .tuner_address = 0x60,
  663. .refclock = 27000000,
  664. };
  665. static struct stv0900_config dw2104_stv0900_config = {
  666. .demod_address = 0x68,
  667. .demod_mode = 0,
  668. .xtal = 8000000,
  669. .clkmode = 3,
  670. .diseqc_mode = 2,
  671. .tun1_maddress = 0,
  672. .tun1_adc = 1,/* 1 Vpp */
  673. .path1_mode = 3,
  674. };
  675. static struct stv6110_config dw2104_stv6110_config = {
  676. .i2c_address = 0x60,
  677. .mclk = 16000000,
  678. .clk_div = 1,
  679. };
  680. static struct stv0900_config prof_7500_stv0900_config = {
  681. .demod_address = 0x6a,
  682. .demod_mode = 0,
  683. .xtal = 27000000,
  684. .clkmode = 3,/* 0-CLKI, 2-XTALI, else AUTO */
  685. .diseqc_mode = 2,/* 2/3 PWM */
  686. .tun1_maddress = 0,/* 0x60 */
  687. .tun1_adc = 0,/* 2 Vpp */
  688. .path1_mode = 3,
  689. .tun1_type = 3,
  690. };
  691. static int dw2104_frontend_attach(struct dvb_usb_adapter *d)
  692. {
  693. struct dvb_tuner_ops *tuner_ops = NULL;
  694. if (demod_probe & 4) {
  695. d->fe = dvb_attach(stv0900_attach, &dw2104a_stv0900_config,
  696. &d->dev->i2c_adap, 0);
  697. if (d->fe != NULL) {
  698. if (dvb_attach(stb6100_attach, d->fe,
  699. &dw2104a_stb6100_config,
  700. &d->dev->i2c_adap)) {
  701. tuner_ops = &d->fe->ops.tuner_ops;
  702. tuner_ops->set_frequency = stb6100_set_freq;
  703. tuner_ops->get_frequency = stb6100_get_freq;
  704. tuner_ops->set_bandwidth = stb6100_set_bandw;
  705. tuner_ops->get_bandwidth = stb6100_get_bandw;
  706. d->fe->ops.set_voltage = dw210x_set_voltage;
  707. info("Attached STV0900+STB6100!\n");
  708. return 0;
  709. }
  710. }
  711. }
  712. if (demod_probe & 2) {
  713. d->fe = dvb_attach(stv0900_attach, &dw2104_stv0900_config,
  714. &d->dev->i2c_adap, 0);
  715. if (d->fe != NULL) {
  716. if (dvb_attach(stv6110_attach, d->fe,
  717. &dw2104_stv6110_config,
  718. &d->dev->i2c_adap)) {
  719. d->fe->ops.set_voltage = dw210x_set_voltage;
  720. info("Attached STV0900+STV6110A!\n");
  721. return 0;
  722. }
  723. }
  724. }
  725. if (demod_probe & 1) {
  726. d->fe = dvb_attach(cx24116_attach, &dw2104_config,
  727. &d->dev->i2c_adap);
  728. if (d->fe != NULL) {
  729. d->fe->ops.set_voltage = dw210x_set_voltage;
  730. info("Attached cx24116!\n");
  731. return 0;
  732. }
  733. }
  734. d->fe = dvb_attach(ds3000_attach, &dw2104_ds3000_config,
  735. &d->dev->i2c_adap);
  736. if (d->fe != NULL) {
  737. d->fe->ops.set_voltage = dw210x_set_voltage;
  738. info("Attached DS3000!\n");
  739. return 0;
  740. }
  741. return -EIO;
  742. }
  743. static struct dvb_usb_device_properties dw2102_properties;
  744. static struct dvb_usb_device_properties dw2104_properties;
  745. static struct dvb_usb_device_properties s6x0_properties;
  746. static int dw2102_frontend_attach(struct dvb_usb_adapter *d)
  747. {
  748. if (dw2102_properties.i2c_algo == &dw2102_serit_i2c_algo) {
  749. /*dw2102_properties.adapter->tuner_attach = NULL;*/
  750. d->fe = dvb_attach(si21xx_attach, &serit_sp1511lhb_config,
  751. &d->dev->i2c_adap);
  752. if (d->fe != NULL) {
  753. d->fe->ops.set_voltage = dw210x_set_voltage;
  754. info("Attached si21xx!\n");
  755. return 0;
  756. }
  757. }
  758. if (dw2102_properties.i2c_algo == &dw2102_earda_i2c_algo) {
  759. d->fe = dvb_attach(stv0288_attach, &earda_config,
  760. &d->dev->i2c_adap);
  761. if (d->fe != NULL) {
  762. if (dvb_attach(stb6000_attach, d->fe, 0x61,
  763. &d->dev->i2c_adap)) {
  764. d->fe->ops.set_voltage = dw210x_set_voltage;
  765. info("Attached stv0288!\n");
  766. return 0;
  767. }
  768. }
  769. }
  770. if (dw2102_properties.i2c_algo == &dw2102_i2c_algo) {
  771. /*dw2102_properties.adapter->tuner_attach = dw2102_tuner_attach;*/
  772. d->fe = dvb_attach(stv0299_attach, &sharp_z0194a_config,
  773. &d->dev->i2c_adap);
  774. if (d->fe != NULL) {
  775. d->fe->ops.set_voltage = dw210x_set_voltage;
  776. info("Attached stv0299!\n");
  777. return 0;
  778. }
  779. }
  780. return -EIO;
  781. }
  782. static int dw3101_frontend_attach(struct dvb_usb_adapter *d)
  783. {
  784. d->fe = dvb_attach(tda10023_attach, &dw3101_tda10023_config,
  785. &d->dev->i2c_adap, 0x48);
  786. if (d->fe != NULL) {
  787. info("Attached tda10023!\n");
  788. return 0;
  789. }
  790. return -EIO;
  791. }
  792. static int s6x0_frontend_attach(struct dvb_usb_adapter *d)
  793. {
  794. d->fe = dvb_attach(mt312_attach, &zl313_config,
  795. &d->dev->i2c_adap);
  796. if (d->fe != NULL) {
  797. if (dvb_attach(zl10039_attach, d->fe, 0x60,
  798. &d->dev->i2c_adap)) {
  799. d->fe->ops.set_voltage = dw210x_set_voltage;
  800. info("Attached zl100313+zl10039!\n");
  801. return 0;
  802. }
  803. }
  804. d->fe = dvb_attach(stv0288_attach, &earda_config,
  805. &d->dev->i2c_adap);
  806. if (d->fe != NULL) {
  807. if (dvb_attach(stb6000_attach, d->fe, 0x61,
  808. &d->dev->i2c_adap)) {
  809. d->fe->ops.set_voltage = dw210x_set_voltage;
  810. info("Attached stv0288+stb6000!\n");
  811. return 0;
  812. }
  813. }
  814. d->fe = dvb_attach(ds3000_attach, &dw2104_ds3000_config,
  815. &d->dev->i2c_adap);
  816. if (d->fe != NULL) {
  817. d->fe->ops.set_voltage = dw210x_set_voltage;
  818. info("Attached ds3000+ds2020!\n");
  819. return 0;
  820. }
  821. return -EIO;
  822. }
  823. static int prof_7500_frontend_attach(struct dvb_usb_adapter *d)
  824. {
  825. d->fe = dvb_attach(stv0900_attach, &prof_7500_stv0900_config,
  826. &d->dev->i2c_adap, 0);
  827. if (d->fe == NULL)
  828. return -EIO;
  829. d->fe->ops.set_voltage = dw210x_set_voltage;
  830. info("Attached STV0900+STB6100A!\n");
  831. return 0;
  832. }
  833. static int dw2102_tuner_attach(struct dvb_usb_adapter *adap)
  834. {
  835. dvb_attach(dvb_pll_attach, adap->fe, 0x60,
  836. &adap->dev->i2c_adap, DVB_PLL_OPERA1);
  837. return 0;
  838. }
  839. static int dw3101_tuner_attach(struct dvb_usb_adapter *adap)
  840. {
  841. dvb_attach(dvb_pll_attach, adap->fe, 0x60,
  842. &adap->dev->i2c_adap, DVB_PLL_TUA6034);
  843. return 0;
  844. }
  845. static struct rc_map_table rc_map_dw210x_table[] = {
  846. { 0xf80a, KEY_Q }, /*power*/
  847. { 0xf80c, KEY_M }, /*mute*/
  848. { 0xf811, KEY_1 },
  849. { 0xf812, KEY_2 },
  850. { 0xf813, KEY_3 },
  851. { 0xf814, KEY_4 },
  852. { 0xf815, KEY_5 },
  853. { 0xf816, KEY_6 },
  854. { 0xf817, KEY_7 },
  855. { 0xf818, KEY_8 },
  856. { 0xf819, KEY_9 },
  857. { 0xf810, KEY_0 },
  858. { 0xf81c, KEY_PAGEUP }, /*ch+*/
  859. { 0xf80f, KEY_PAGEDOWN }, /*ch-*/
  860. { 0xf81a, KEY_O }, /*vol+*/
  861. { 0xf80e, KEY_Z }, /*vol-*/
  862. { 0xf804, KEY_R }, /*rec*/
  863. { 0xf809, KEY_D }, /*fav*/
  864. { 0xf808, KEY_BACKSPACE }, /*rewind*/
  865. { 0xf807, KEY_A }, /*fast*/
  866. { 0xf80b, KEY_P }, /*pause*/
  867. { 0xf802, KEY_ESC }, /*cancel*/
  868. { 0xf803, KEY_G }, /*tab*/
  869. { 0xf800, KEY_UP }, /*up*/
  870. { 0xf81f, KEY_ENTER }, /*ok*/
  871. { 0xf801, KEY_DOWN }, /*down*/
  872. { 0xf805, KEY_C }, /*cap*/
  873. { 0xf806, KEY_S }, /*stop*/
  874. { 0xf840, KEY_F }, /*full*/
  875. { 0xf81e, KEY_W }, /*tvmode*/
  876. { 0xf81b, KEY_B }, /*recall*/
  877. };
  878. static struct rc_map_table rc_map_tevii_table[] = {
  879. { 0xf80a, KEY_POWER },
  880. { 0xf80c, KEY_MUTE },
  881. { 0xf811, KEY_1 },
  882. { 0xf812, KEY_2 },
  883. { 0xf813, KEY_3 },
  884. { 0xf814, KEY_4 },
  885. { 0xf815, KEY_5 },
  886. { 0xf816, KEY_6 },
  887. { 0xf817, KEY_7 },
  888. { 0xf818, KEY_8 },
  889. { 0xf819, KEY_9 },
  890. { 0xf810, KEY_0 },
  891. { 0xf81c, KEY_MENU },
  892. { 0xf80f, KEY_VOLUMEDOWN },
  893. { 0xf81a, KEY_LAST },
  894. { 0xf80e, KEY_OPEN },
  895. { 0xf804, KEY_RECORD },
  896. { 0xf809, KEY_VOLUMEUP },
  897. { 0xf808, KEY_CHANNELUP },
  898. { 0xf807, KEY_PVR },
  899. { 0xf80b, KEY_TIME },
  900. { 0xf802, KEY_RIGHT },
  901. { 0xf803, KEY_LEFT },
  902. { 0xf800, KEY_UP },
  903. { 0xf81f, KEY_OK },
  904. { 0xf801, KEY_DOWN },
  905. { 0xf805, KEY_TUNER },
  906. { 0xf806, KEY_CHANNELDOWN },
  907. { 0xf840, KEY_PLAYPAUSE },
  908. { 0xf81e, KEY_REWIND },
  909. { 0xf81b, KEY_FAVORITES },
  910. { 0xf81d, KEY_BACK },
  911. { 0xf84d, KEY_FASTFORWARD },
  912. { 0xf844, KEY_EPG },
  913. { 0xf84c, KEY_INFO },
  914. { 0xf841, KEY_AB },
  915. { 0xf843, KEY_AUDIO },
  916. { 0xf845, KEY_SUBTITLE },
  917. { 0xf84a, KEY_LIST },
  918. { 0xf846, KEY_F1 },
  919. { 0xf847, KEY_F2 },
  920. { 0xf85e, KEY_F3 },
  921. { 0xf85c, KEY_F4 },
  922. { 0xf852, KEY_F5 },
  923. { 0xf85a, KEY_F6 },
  924. { 0xf856, KEY_MODE },
  925. { 0xf858, KEY_SWITCHVIDEOMODE },
  926. };
  927. static struct rc_map_table rc_map_tbs_table[] = {
  928. { 0xf884, KEY_POWER },
  929. { 0xf894, KEY_MUTE },
  930. { 0xf887, KEY_1 },
  931. { 0xf886, KEY_2 },
  932. { 0xf885, KEY_3 },
  933. { 0xf88b, KEY_4 },
  934. { 0xf88a, KEY_5 },
  935. { 0xf889, KEY_6 },
  936. { 0xf88f, KEY_7 },
  937. { 0xf88e, KEY_8 },
  938. { 0xf88d, KEY_9 },
  939. { 0xf892, KEY_0 },
  940. { 0xf896, KEY_CHANNELUP },
  941. { 0xf891, KEY_CHANNELDOWN },
  942. { 0xf893, KEY_VOLUMEUP },
  943. { 0xf88c, KEY_VOLUMEDOWN },
  944. { 0xf883, KEY_RECORD },
  945. { 0xf898, KEY_PAUSE },
  946. { 0xf899, KEY_OK },
  947. { 0xf89a, KEY_SHUFFLE },
  948. { 0xf881, KEY_UP },
  949. { 0xf890, KEY_LEFT },
  950. { 0xf882, KEY_RIGHT },
  951. { 0xf888, KEY_DOWN },
  952. { 0xf895, KEY_FAVORITES },
  953. { 0xf897, KEY_SUBTITLE },
  954. { 0xf89d, KEY_ZOOM },
  955. { 0xf89f, KEY_EXIT },
  956. { 0xf89e, KEY_MENU },
  957. { 0xf89c, KEY_EPG },
  958. { 0xf880, KEY_PREVIOUS },
  959. { 0xf89b, KEY_MODE }
  960. };
  961. static struct rc_map_dvb_usb_table_table keys_tables[] = {
  962. { rc_map_dw210x_table, ARRAY_SIZE(rc_map_dw210x_table) },
  963. { rc_map_tevii_table, ARRAY_SIZE(rc_map_tevii_table) },
  964. { rc_map_tbs_table, ARRAY_SIZE(rc_map_tbs_table) },
  965. };
  966. static int dw2102_rc_query(struct dvb_usb_device *d, u32 *event, int *state)
  967. {
  968. struct rc_map_table *keymap = d->props.rc.legacy.rc_map_table;
  969. int keymap_size = d->props.rc.legacy.rc_map_size;
  970. u8 key[2];
  971. struct i2c_msg msg = {
  972. .addr = DW2102_RC_QUERY,
  973. .flags = I2C_M_RD,
  974. .buf = key,
  975. .len = 2
  976. };
  977. int i;
  978. /* override keymap */
  979. if ((ir_keymap > 0) && (ir_keymap <= ARRAY_SIZE(keys_tables))) {
  980. keymap = keys_tables[ir_keymap - 1].rc_keys ;
  981. keymap_size = keys_tables[ir_keymap - 1].rc_keys_size;
  982. }
  983. *state = REMOTE_NO_KEY_PRESSED;
  984. if (d->props.i2c_algo->master_xfer(&d->i2c_adap, &msg, 1) == 1) {
  985. for (i = 0; i < keymap_size ; i++) {
  986. if (rc5_data(&keymap[i]) == msg.buf[0]) {
  987. *state = REMOTE_KEY_PRESSED;
  988. *event = keymap[i].keycode;
  989. break;
  990. }
  991. }
  992. if ((*state) == REMOTE_KEY_PRESSED)
  993. deb_rc("%s: found rc key: %x, %x, event: %x\n",
  994. __func__, key[0], key[1], (*event));
  995. else if (key[0] != 0xff)
  996. deb_rc("%s: unknown rc key: %x, %x\n",
  997. __func__, key[0], key[1]);
  998. }
  999. return 0;
  1000. }
  1001. static struct usb_device_id dw2102_table[] = {
  1002. {USB_DEVICE(USB_VID_CYPRESS, USB_PID_DW2102)},
  1003. {USB_DEVICE(USB_VID_CYPRESS, 0x2101)},
  1004. {USB_DEVICE(USB_VID_CYPRESS, USB_PID_DW2104)},
  1005. {USB_DEVICE(0x9022, USB_PID_TEVII_S650)},
  1006. {USB_DEVICE(USB_VID_TERRATEC, USB_PID_CINERGY_S)},
  1007. {USB_DEVICE(USB_VID_CYPRESS, USB_PID_DW3101)},
  1008. {USB_DEVICE(0x9022, USB_PID_TEVII_S630)},
  1009. {USB_DEVICE(0x3011, USB_PID_PROF_1100)},
  1010. {USB_DEVICE(0x9022, USB_PID_TEVII_S660)},
  1011. {USB_DEVICE(0x3034, 0x7500)},
  1012. { }
  1013. };
  1014. MODULE_DEVICE_TABLE(usb, dw2102_table);
  1015. static int dw2102_load_firmware(struct usb_device *dev,
  1016. const struct firmware *frmwr)
  1017. {
  1018. u8 *b, *p;
  1019. int ret = 0, i;
  1020. u8 reset;
  1021. u8 reset16[] = {0, 0, 0, 0, 0, 0, 0};
  1022. const struct firmware *fw;
  1023. const char *fw_2101 = "dvb-usb-dw2101.fw";
  1024. switch (dev->descriptor.idProduct) {
  1025. case 0x2101:
  1026. ret = request_firmware(&fw, fw_2101, &dev->dev);
  1027. if (ret != 0) {
  1028. err(err_str, fw_2101);
  1029. return ret;
  1030. }
  1031. break;
  1032. default:
  1033. fw = frmwr;
  1034. break;
  1035. }
  1036. info("start downloading DW210X firmware");
  1037. p = kmalloc(fw->size, GFP_KERNEL);
  1038. reset = 1;
  1039. /*stop the CPU*/
  1040. dw210x_op_rw(dev, 0xa0, 0x7f92, 0, &reset, 1, DW210X_WRITE_MSG);
  1041. dw210x_op_rw(dev, 0xa0, 0xe600, 0, &reset, 1, DW210X_WRITE_MSG);
  1042. if (p != NULL) {
  1043. memcpy(p, fw->data, fw->size);
  1044. for (i = 0; i < fw->size; i += 0x40) {
  1045. b = (u8 *) p + i;
  1046. if (dw210x_op_rw(dev, 0xa0, i, 0, b , 0x40,
  1047. DW210X_WRITE_MSG) != 0x40) {
  1048. err("error while transferring firmware");
  1049. ret = -EINVAL;
  1050. break;
  1051. }
  1052. }
  1053. /* restart the CPU */
  1054. reset = 0;
  1055. if (ret || dw210x_op_rw(dev, 0xa0, 0x7f92, 0, &reset, 1,
  1056. DW210X_WRITE_MSG) != 1) {
  1057. err("could not restart the USB controller CPU.");
  1058. ret = -EINVAL;
  1059. }
  1060. if (ret || dw210x_op_rw(dev, 0xa0, 0xe600, 0, &reset, 1,
  1061. DW210X_WRITE_MSG) != 1) {
  1062. err("could not restart the USB controller CPU.");
  1063. ret = -EINVAL;
  1064. }
  1065. /* init registers */
  1066. switch (dev->descriptor.idProduct) {
  1067. case USB_PID_PROF_1100:
  1068. s6x0_properties.rc.legacy.rc_map_table = rc_map_tbs_table;
  1069. s6x0_properties.rc.legacy.rc_map_size =
  1070. ARRAY_SIZE(rc_map_tbs_table);
  1071. break;
  1072. case USB_PID_TEVII_S650:
  1073. dw2104_properties.rc.legacy.rc_map_table = rc_map_tevii_table;
  1074. dw2104_properties.rc.legacy.rc_map_size =
  1075. ARRAY_SIZE(rc_map_tevii_table);
  1076. case USB_PID_DW2104:
  1077. reset = 1;
  1078. dw210x_op_rw(dev, 0xc4, 0x0000, 0, &reset, 1,
  1079. DW210X_WRITE_MSG);
  1080. /* break omitted intentionally */
  1081. case USB_PID_DW3101:
  1082. reset = 0;
  1083. dw210x_op_rw(dev, 0xbf, 0x0040, 0, &reset, 0,
  1084. DW210X_WRITE_MSG);
  1085. break;
  1086. case USB_PID_CINERGY_S:
  1087. case USB_PID_DW2102:
  1088. dw210x_op_rw(dev, 0xbf, 0x0040, 0, &reset, 0,
  1089. DW210X_WRITE_MSG);
  1090. dw210x_op_rw(dev, 0xb9, 0x0000, 0, &reset16[0], 2,
  1091. DW210X_READ_MSG);
  1092. /* check STV0299 frontend */
  1093. dw210x_op_rw(dev, 0xb5, 0, 0, &reset16[0], 2,
  1094. DW210X_READ_MSG);
  1095. if ((reset16[0] == 0xa1) || (reset16[0] == 0x80)) {
  1096. dw2102_properties.i2c_algo = &dw2102_i2c_algo;
  1097. dw2102_properties.adapter->tuner_attach = &dw2102_tuner_attach;
  1098. break;
  1099. } else {
  1100. /* check STV0288 frontend */
  1101. reset16[0] = 0xd0;
  1102. reset16[1] = 1;
  1103. reset16[2] = 0;
  1104. dw210x_op_rw(dev, 0xc2, 0, 0, &reset16[0], 3,
  1105. DW210X_WRITE_MSG);
  1106. dw210x_op_rw(dev, 0xc3, 0xd1, 0, &reset16[0], 3,
  1107. DW210X_READ_MSG);
  1108. if (reset16[2] == 0x11) {
  1109. dw2102_properties.i2c_algo = &dw2102_earda_i2c_algo;
  1110. break;
  1111. }
  1112. }
  1113. case 0x2101:
  1114. dw210x_op_rw(dev, 0xbc, 0x0030, 0, &reset16[0], 2,
  1115. DW210X_READ_MSG);
  1116. dw210x_op_rw(dev, 0xba, 0x0000, 0, &reset16[0], 7,
  1117. DW210X_READ_MSG);
  1118. dw210x_op_rw(dev, 0xba, 0x0000, 0, &reset16[0], 7,
  1119. DW210X_READ_MSG);
  1120. dw210x_op_rw(dev, 0xb9, 0x0000, 0, &reset16[0], 2,
  1121. DW210X_READ_MSG);
  1122. break;
  1123. }
  1124. msleep(100);
  1125. kfree(p);
  1126. }
  1127. return ret;
  1128. }
  1129. static struct dvb_usb_device_properties dw2102_properties = {
  1130. .caps = DVB_USB_IS_AN_I2C_ADAPTER,
  1131. .usb_ctrl = DEVICE_SPECIFIC,
  1132. .firmware = "dvb-usb-dw2102.fw",
  1133. .no_reconnect = 1,
  1134. .i2c_algo = &dw2102_serit_i2c_algo,
  1135. .rc.legacy = {
  1136. .rc_map_table = rc_map_dw210x_table,
  1137. .rc_map_size = ARRAY_SIZE(rc_map_dw210x_table),
  1138. .rc_interval = 150,
  1139. .rc_query = dw2102_rc_query,
  1140. },
  1141. .generic_bulk_ctrl_endpoint = 0x81,
  1142. /* parameter for the MPEG2-data transfer */
  1143. .num_adapters = 1,
  1144. .download_firmware = dw2102_load_firmware,
  1145. .read_mac_address = dw210x_read_mac_address,
  1146. .adapter = {
  1147. {
  1148. .frontend_attach = dw2102_frontend_attach,
  1149. .streaming_ctrl = NULL,
  1150. .tuner_attach = NULL,
  1151. .stream = {
  1152. .type = USB_BULK,
  1153. .count = 8,
  1154. .endpoint = 0x82,
  1155. .u = {
  1156. .bulk = {
  1157. .buffersize = 4096,
  1158. }
  1159. }
  1160. },
  1161. }
  1162. },
  1163. .num_device_descs = 3,
  1164. .devices = {
  1165. {"DVBWorld DVB-S 2102 USB2.0",
  1166. {&dw2102_table[0], NULL},
  1167. {NULL},
  1168. },
  1169. {"DVBWorld DVB-S 2101 USB2.0",
  1170. {&dw2102_table[1], NULL},
  1171. {NULL},
  1172. },
  1173. {"TerraTec Cinergy S USB",
  1174. {&dw2102_table[4], NULL},
  1175. {NULL},
  1176. },
  1177. }
  1178. };
  1179. static struct dvb_usb_device_properties dw2104_properties = {
  1180. .caps = DVB_USB_IS_AN_I2C_ADAPTER,
  1181. .usb_ctrl = DEVICE_SPECIFIC,
  1182. .firmware = "dvb-usb-dw2104.fw",
  1183. .no_reconnect = 1,
  1184. .i2c_algo = &dw2104_i2c_algo,
  1185. .rc.legacy = {
  1186. .rc_map_table = rc_map_dw210x_table,
  1187. .rc_map_size = ARRAY_SIZE(rc_map_dw210x_table),
  1188. .rc_interval = 150,
  1189. .rc_query = dw2102_rc_query,
  1190. },
  1191. .generic_bulk_ctrl_endpoint = 0x81,
  1192. /* parameter for the MPEG2-data transfer */
  1193. .num_adapters = 1,
  1194. .download_firmware = dw2102_load_firmware,
  1195. .read_mac_address = dw210x_read_mac_address,
  1196. .adapter = {
  1197. {
  1198. .frontend_attach = dw2104_frontend_attach,
  1199. .streaming_ctrl = NULL,
  1200. /*.tuner_attach = dw2104_tuner_attach,*/
  1201. .stream = {
  1202. .type = USB_BULK,
  1203. .count = 8,
  1204. .endpoint = 0x82,
  1205. .u = {
  1206. .bulk = {
  1207. .buffersize = 4096,
  1208. }
  1209. }
  1210. },
  1211. }
  1212. },
  1213. .num_device_descs = 2,
  1214. .devices = {
  1215. { "DVBWorld DW2104 USB2.0",
  1216. {&dw2102_table[2], NULL},
  1217. {NULL},
  1218. },
  1219. { "TeVii S650 USB2.0",
  1220. {&dw2102_table[3], NULL},
  1221. {NULL},
  1222. },
  1223. }
  1224. };
  1225. static struct dvb_usb_device_properties dw3101_properties = {
  1226. .caps = DVB_USB_IS_AN_I2C_ADAPTER,
  1227. .usb_ctrl = DEVICE_SPECIFIC,
  1228. .firmware = "dvb-usb-dw3101.fw",
  1229. .no_reconnect = 1,
  1230. .i2c_algo = &dw3101_i2c_algo,
  1231. .rc.legacy = {
  1232. .rc_map_table = rc_map_dw210x_table,
  1233. .rc_map_size = ARRAY_SIZE(rc_map_dw210x_table),
  1234. .rc_interval = 150,
  1235. .rc_query = dw2102_rc_query,
  1236. },
  1237. .generic_bulk_ctrl_endpoint = 0x81,
  1238. /* parameter for the MPEG2-data transfer */
  1239. .num_adapters = 1,
  1240. .download_firmware = dw2102_load_firmware,
  1241. .read_mac_address = dw210x_read_mac_address,
  1242. .adapter = {
  1243. {
  1244. .frontend_attach = dw3101_frontend_attach,
  1245. .streaming_ctrl = NULL,
  1246. .tuner_attach = dw3101_tuner_attach,
  1247. .stream = {
  1248. .type = USB_BULK,
  1249. .count = 8,
  1250. .endpoint = 0x82,
  1251. .u = {
  1252. .bulk = {
  1253. .buffersize = 4096,
  1254. }
  1255. }
  1256. },
  1257. }
  1258. },
  1259. .num_device_descs = 1,
  1260. .devices = {
  1261. { "DVBWorld DVB-C 3101 USB2.0",
  1262. {&dw2102_table[5], NULL},
  1263. {NULL},
  1264. },
  1265. }
  1266. };
  1267. static struct dvb_usb_device_properties s6x0_properties = {
  1268. .caps = DVB_USB_IS_AN_I2C_ADAPTER,
  1269. .usb_ctrl = DEVICE_SPECIFIC,
  1270. .firmware = "dvb-usb-s630.fw",
  1271. .no_reconnect = 1,
  1272. .i2c_algo = &s6x0_i2c_algo,
  1273. .rc.legacy = {
  1274. .rc_map_table = rc_map_tevii_table,
  1275. .rc_map_size = ARRAY_SIZE(rc_map_tevii_table),
  1276. .rc_interval = 150,
  1277. .rc_query = dw2102_rc_query,
  1278. },
  1279. .generic_bulk_ctrl_endpoint = 0x81,
  1280. .num_adapters = 1,
  1281. .download_firmware = dw2102_load_firmware,
  1282. .read_mac_address = s6x0_read_mac_address,
  1283. .adapter = {
  1284. {
  1285. .frontend_attach = s6x0_frontend_attach,
  1286. .streaming_ctrl = NULL,
  1287. .tuner_attach = NULL,
  1288. .stream = {
  1289. .type = USB_BULK,
  1290. .count = 8,
  1291. .endpoint = 0x82,
  1292. .u = {
  1293. .bulk = {
  1294. .buffersize = 4096,
  1295. }
  1296. }
  1297. },
  1298. }
  1299. },
  1300. .num_device_descs = 3,
  1301. .devices = {
  1302. {"TeVii S630 USB",
  1303. {&dw2102_table[6], NULL},
  1304. {NULL},
  1305. },
  1306. {"Prof 1100 USB ",
  1307. {&dw2102_table[7], NULL},
  1308. {NULL},
  1309. },
  1310. {"TeVii S660 USB",
  1311. {&dw2102_table[8], NULL},
  1312. {NULL},
  1313. },
  1314. }
  1315. };
  1316. struct dvb_usb_device_properties *p7500;
  1317. static struct dvb_usb_device_description d7500 = {
  1318. "Prof 7500 USB DVB-S2",
  1319. {&dw2102_table[9], NULL},
  1320. {NULL},
  1321. };
  1322. static int dw2102_probe(struct usb_interface *intf,
  1323. const struct usb_device_id *id)
  1324. {
  1325. p7500 = kzalloc(sizeof(struct dvb_usb_device_properties), GFP_KERNEL);
  1326. if (!p7500)
  1327. return -ENOMEM;
  1328. /* copy default structure */
  1329. memcpy(p7500, &s6x0_properties,
  1330. sizeof(struct dvb_usb_device_properties));
  1331. /* fill only different fields */
  1332. p7500->firmware = "dvb-usb-p7500.fw";
  1333. p7500->devices[0] = d7500;
  1334. p7500->rc.legacy.rc_map_table = rc_map_tbs_table;
  1335. p7500->rc.legacy.rc_map_size = ARRAY_SIZE(rc_map_tbs_table);
  1336. p7500->adapter->frontend_attach = prof_7500_frontend_attach;
  1337. if (0 == dvb_usb_device_init(intf, &dw2102_properties,
  1338. THIS_MODULE, NULL, adapter_nr) ||
  1339. 0 == dvb_usb_device_init(intf, &dw2104_properties,
  1340. THIS_MODULE, NULL, adapter_nr) ||
  1341. 0 == dvb_usb_device_init(intf, &dw3101_properties,
  1342. THIS_MODULE, NULL, adapter_nr) ||
  1343. 0 == dvb_usb_device_init(intf, &s6x0_properties,
  1344. THIS_MODULE, NULL, adapter_nr) ||
  1345. 0 == dvb_usb_device_init(intf, p7500,
  1346. THIS_MODULE, NULL, adapter_nr))
  1347. return 0;
  1348. return -ENODEV;
  1349. }
  1350. static struct usb_driver dw2102_driver = {
  1351. .name = "dw2102",
  1352. .probe = dw2102_probe,
  1353. .disconnect = dvb_usb_device_exit,
  1354. .id_table = dw2102_table,
  1355. };
  1356. static int __init dw2102_module_init(void)
  1357. {
  1358. int ret = usb_register(&dw2102_driver);
  1359. if (ret)
  1360. err("usb_register failed. Error number %d", ret);
  1361. return ret;
  1362. }
  1363. static void __exit dw2102_module_exit(void)
  1364. {
  1365. usb_deregister(&dw2102_driver);
  1366. }
  1367. module_init(dw2102_module_init);
  1368. module_exit(dw2102_module_exit);
  1369. MODULE_AUTHOR("Igor M. Liplianin (c) liplianin@me.by");
  1370. MODULE_DESCRIPTION("Driver for DVBWorld DVB-S 2101, 2102, DVB-S2 2104,"
  1371. " DVB-C 3101 USB2.0,"
  1372. " TeVii S600, S630, S650, S660 USB2.0,"
  1373. " Prof 1100, 7500 USB2.0 devices");
  1374. MODULE_VERSION("0.1");
  1375. MODULE_LICENSE("GPL");