tas3001c.c 21 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849
  1. /*
  2. * Driver for the i2c/i2s based TA3004 sound chip used
  3. * on some Apple hardware. Also known as "snapper".
  4. *
  5. * Tobias Sargeant <tobias.sargeant@bigpond.com>
  6. * Based upon, tas3001c.c by Christopher C. Chimelis <chris@debian.org>:
  7. *
  8. * TODO:
  9. * -----
  10. * * Enable control over input line 2 (is this connected?)
  11. * * Implement sleep support (at least mute everything and
  12. * * set gains to minimum during sleep)
  13. * * Look into some of Darwin's tweaks regarding the mute
  14. * * lines (delays & different behaviour on some HW)
  15. *
  16. */
  17. #include <linux/module.h>
  18. #include <linux/slab.h>
  19. #include <linux/proc_fs.h>
  20. #include <linux/ioport.h>
  21. #include <linux/sysctl.h>
  22. #include <linux/types.h>
  23. #include <linux/i2c.h>
  24. #include <linux/init.h>
  25. #include <linux/soundcard.h>
  26. #include <linux/workqueue.h>
  27. #include <asm/uaccess.h>
  28. #include <asm/errno.h>
  29. #include <asm/io.h>
  30. #include <asm/prom.h>
  31. #include "dmasound.h"
  32. #include "tas_common.h"
  33. #include "tas3001c.h"
  34. #include "tas_ioctl.h"
  35. #define TAS3001C_BIQUAD_FILTER_COUNT 6
  36. #define TAS3001C_BIQUAD_CHANNEL_COUNT 2
  37. #define VOL_DEFAULT (100 * 4 / 5)
  38. #define INPUT_DEFAULT (100 * 4 / 5)
  39. #define BASS_DEFAULT (100 / 2)
  40. #define TREBLE_DEFAULT (100 / 2)
  41. struct tas3001c_data_t {
  42. struct tas_data_t super;
  43. int device_id;
  44. int output_id;
  45. int speaker_id;
  46. struct tas_drce_t drce_state;
  47. struct work_struct change;
  48. };
  49. static const union tas_biquad_t
  50. tas3001c_eq_unity={
  51. .buf = { 0x100000, 0x000000, 0x000000, 0x000000, 0x000000 }
  52. };
  53. static inline unsigned char db_to_regval(short db) {
  54. int r=0;
  55. r=(db+0x59a0) / 0x60;
  56. if (r < 0x91) return 0x91;
  57. if (r > 0xef) return 0xef;
  58. return r;
  59. }
  60. static inline short quantize_db(short db) {
  61. return db_to_regval(db) * 0x60 - 0x59a0;
  62. }
  63. static inline int
  64. register_width(enum tas3001c_reg_t r)
  65. {
  66. switch(r) {
  67. case TAS3001C_REG_MCR:
  68. case TAS3001C_REG_TREBLE:
  69. case TAS3001C_REG_BASS:
  70. return 1;
  71. case TAS3001C_REG_DRC:
  72. return 2;
  73. case TAS3001C_REG_MIXER1:
  74. case TAS3001C_REG_MIXER2:
  75. return 3;
  76. case TAS3001C_REG_VOLUME:
  77. return 6;
  78. case TAS3001C_REG_LEFT_BIQUAD0:
  79. case TAS3001C_REG_LEFT_BIQUAD1:
  80. case TAS3001C_REG_LEFT_BIQUAD2:
  81. case TAS3001C_REG_LEFT_BIQUAD3:
  82. case TAS3001C_REG_LEFT_BIQUAD4:
  83. case TAS3001C_REG_LEFT_BIQUAD5:
  84. case TAS3001C_REG_LEFT_BIQUAD6:
  85. case TAS3001C_REG_RIGHT_BIQUAD0:
  86. case TAS3001C_REG_RIGHT_BIQUAD1:
  87. case TAS3001C_REG_RIGHT_BIQUAD2:
  88. case TAS3001C_REG_RIGHT_BIQUAD3:
  89. case TAS3001C_REG_RIGHT_BIQUAD4:
  90. case TAS3001C_REG_RIGHT_BIQUAD5:
  91. case TAS3001C_REG_RIGHT_BIQUAD6:
  92. return 15;
  93. default:
  94. return 0;
  95. }
  96. }
  97. static int
  98. tas3001c_write_register( struct tas3001c_data_t *self,
  99. enum tas3001c_reg_t reg_num,
  100. char *data,
  101. uint write_mode)
  102. {
  103. if (reg_num==TAS3001C_REG_MCR ||
  104. reg_num==TAS3001C_REG_BASS ||
  105. reg_num==TAS3001C_REG_TREBLE) {
  106. return tas_write_byte_register(&self->super,
  107. (uint)reg_num,
  108. *data,
  109. write_mode);
  110. } else {
  111. return tas_write_register(&self->super,
  112. (uint)reg_num,
  113. register_width(reg_num),
  114. data,
  115. write_mode);
  116. }
  117. }
  118. static int
  119. tas3001c_sync_register( struct tas3001c_data_t *self,
  120. enum tas3001c_reg_t reg_num)
  121. {
  122. if (reg_num==TAS3001C_REG_MCR ||
  123. reg_num==TAS3001C_REG_BASS ||
  124. reg_num==TAS3001C_REG_TREBLE) {
  125. return tas_sync_byte_register(&self->super,
  126. (uint)reg_num,
  127. register_width(reg_num));
  128. } else {
  129. return tas_sync_register(&self->super,
  130. (uint)reg_num,
  131. register_width(reg_num));
  132. }
  133. }
  134. static int
  135. tas3001c_read_register( struct tas3001c_data_t *self,
  136. enum tas3001c_reg_t reg_num,
  137. char *data,
  138. uint write_mode)
  139. {
  140. return tas_read_register(&self->super,
  141. (uint)reg_num,
  142. register_width(reg_num),
  143. data);
  144. }
  145. static inline int
  146. tas3001c_fast_load(struct tas3001c_data_t *self, int fast)
  147. {
  148. if (fast)
  149. self->super.shadow[TAS3001C_REG_MCR][0] |= 0x80;
  150. else
  151. self->super.shadow[TAS3001C_REG_MCR][0] &= 0x7f;
  152. return tas3001c_sync_register(self,TAS3001C_REG_MCR);
  153. }
  154. static uint
  155. tas3001c_supported_mixers(struct tas3001c_data_t *self)
  156. {
  157. return SOUND_MASK_VOLUME |
  158. SOUND_MASK_PCM |
  159. SOUND_MASK_ALTPCM |
  160. SOUND_MASK_TREBLE |
  161. SOUND_MASK_BASS;
  162. }
  163. static int
  164. tas3001c_mixer_is_stereo(struct tas3001c_data_t *self,int mixer)
  165. {
  166. switch(mixer) {
  167. case SOUND_MIXER_VOLUME:
  168. return 1;
  169. default:
  170. return 0;
  171. }
  172. }
  173. static uint
  174. tas3001c_stereo_mixers(struct tas3001c_data_t *self)
  175. {
  176. uint r=tas3001c_supported_mixers(self);
  177. uint i;
  178. for (i=1; i<SOUND_MIXER_NRDEVICES; i++)
  179. if (r&(1<<i) && !tas3001c_mixer_is_stereo(self,i))
  180. r &= ~(1<<i);
  181. return r;
  182. }
  183. static int
  184. tas3001c_get_mixer_level(struct tas3001c_data_t *self,int mixer,uint *level)
  185. {
  186. if (!self)
  187. return -1;
  188. *level=self->super.mixer[mixer];
  189. return 0;
  190. }
  191. static int
  192. tas3001c_set_mixer_level(struct tas3001c_data_t *self,int mixer,uint level)
  193. {
  194. int rc;
  195. tas_shadow_t *shadow;
  196. uint temp;
  197. uint offset=0;
  198. if (!self)
  199. return -1;
  200. shadow=self->super.shadow;
  201. if (!tas3001c_mixer_is_stereo(self,mixer))
  202. level = tas_mono_to_stereo(level);
  203. switch(mixer) {
  204. case SOUND_MIXER_VOLUME:
  205. temp = tas3001c_gain.master[level&0xff];
  206. shadow[TAS3001C_REG_VOLUME][0] = (temp >> 16) & 0xff;
  207. shadow[TAS3001C_REG_VOLUME][1] = (temp >> 8) & 0xff;
  208. shadow[TAS3001C_REG_VOLUME][2] = (temp >> 0) & 0xff;
  209. temp = tas3001c_gain.master[(level>>8)&0xff];
  210. shadow[TAS3001C_REG_VOLUME][3] = (temp >> 16) & 0xff;
  211. shadow[TAS3001C_REG_VOLUME][4] = (temp >> 8) & 0xff;
  212. shadow[TAS3001C_REG_VOLUME][5] = (temp >> 0) & 0xff;
  213. rc = tas3001c_sync_register(self,TAS3001C_REG_VOLUME);
  214. break;
  215. case SOUND_MIXER_ALTPCM:
  216. /* tas3001c_fast_load(self, 1); */
  217. level = tas_mono_to_stereo(level);
  218. temp = tas3001c_gain.mixer[level&0xff];
  219. shadow[TAS3001C_REG_MIXER2][offset+0] = (temp >> 16) & 0xff;
  220. shadow[TAS3001C_REG_MIXER2][offset+1] = (temp >> 8) & 0xff;
  221. shadow[TAS3001C_REG_MIXER2][offset+2] = (temp >> 0) & 0xff;
  222. rc = tas3001c_sync_register(self,TAS3001C_REG_MIXER2);
  223. /* tas3001c_fast_load(self, 0); */
  224. break;
  225. case SOUND_MIXER_PCM:
  226. /* tas3001c_fast_load(self, 1); */
  227. level = tas_mono_to_stereo(level);
  228. temp = tas3001c_gain.mixer[level&0xff];
  229. shadow[TAS3001C_REG_MIXER1][offset+0] = (temp >> 16) & 0xff;
  230. shadow[TAS3001C_REG_MIXER1][offset+1] = (temp >> 8) & 0xff;
  231. shadow[TAS3001C_REG_MIXER1][offset+2] = (temp >> 0) & 0xff;
  232. rc = tas3001c_sync_register(self,TAS3001C_REG_MIXER1);
  233. /* tas3001c_fast_load(self, 0); */
  234. break;
  235. case SOUND_MIXER_TREBLE:
  236. temp = tas3001c_gain.treble[level&0xff];
  237. shadow[TAS3001C_REG_TREBLE][0]=temp&0xff;
  238. rc = tas3001c_sync_register(self,TAS3001C_REG_TREBLE);
  239. break;
  240. case SOUND_MIXER_BASS:
  241. temp = tas3001c_gain.bass[level&0xff];
  242. shadow[TAS3001C_REG_BASS][0]=temp&0xff;
  243. rc = tas3001c_sync_register(self,TAS3001C_REG_BASS);
  244. break;
  245. default:
  246. rc = -1;
  247. break;
  248. }
  249. if (rc < 0)
  250. return rc;
  251. self->super.mixer[mixer]=level;
  252. return 0;
  253. }
  254. static int
  255. tas3001c_leave_sleep(struct tas3001c_data_t *self)
  256. {
  257. unsigned char mcr = (1<<6)+(2<<4)+(2<<2);
  258. if (!self)
  259. return -1;
  260. /* Make sure something answers on the i2c bus */
  261. if (tas3001c_write_register(self, TAS3001C_REG_MCR, &mcr,
  262. WRITE_NORMAL|FORCE_WRITE) < 0)
  263. return -1;
  264. tas3001c_fast_load(self, 1);
  265. (void)tas3001c_sync_register(self,TAS3001C_REG_RIGHT_BIQUAD0);
  266. (void)tas3001c_sync_register(self,TAS3001C_REG_RIGHT_BIQUAD1);
  267. (void)tas3001c_sync_register(self,TAS3001C_REG_RIGHT_BIQUAD2);
  268. (void)tas3001c_sync_register(self,TAS3001C_REG_RIGHT_BIQUAD3);
  269. (void)tas3001c_sync_register(self,TAS3001C_REG_RIGHT_BIQUAD4);
  270. (void)tas3001c_sync_register(self,TAS3001C_REG_RIGHT_BIQUAD5);
  271. (void)tas3001c_sync_register(self,TAS3001C_REG_LEFT_BIQUAD0);
  272. (void)tas3001c_sync_register(self,TAS3001C_REG_LEFT_BIQUAD1);
  273. (void)tas3001c_sync_register(self,TAS3001C_REG_LEFT_BIQUAD2);
  274. (void)tas3001c_sync_register(self,TAS3001C_REG_LEFT_BIQUAD3);
  275. (void)tas3001c_sync_register(self,TAS3001C_REG_LEFT_BIQUAD4);
  276. (void)tas3001c_sync_register(self,TAS3001C_REG_LEFT_BIQUAD5);
  277. tas3001c_fast_load(self, 0);
  278. (void)tas3001c_sync_register(self,TAS3001C_REG_BASS);
  279. (void)tas3001c_sync_register(self,TAS3001C_REG_TREBLE);
  280. (void)tas3001c_sync_register(self,TAS3001C_REG_MIXER1);
  281. (void)tas3001c_sync_register(self,TAS3001C_REG_MIXER2);
  282. (void)tas3001c_sync_register(self,TAS3001C_REG_VOLUME);
  283. return 0;
  284. }
  285. static int
  286. tas3001c_enter_sleep(struct tas3001c_data_t *self)
  287. {
  288. /* Stub for now, but I have the details on low-power mode */
  289. if (!self)
  290. return -1;
  291. return 0;
  292. }
  293. static int
  294. tas3001c_sync_biquad( struct tas3001c_data_t *self,
  295. u_int channel,
  296. u_int filter)
  297. {
  298. enum tas3001c_reg_t reg;
  299. if (channel >= TAS3001C_BIQUAD_CHANNEL_COUNT ||
  300. filter >= TAS3001C_BIQUAD_FILTER_COUNT) return -EINVAL;
  301. reg=( channel ? TAS3001C_REG_RIGHT_BIQUAD0 : TAS3001C_REG_LEFT_BIQUAD0 ) + filter;
  302. return tas3001c_sync_register(self,reg);
  303. }
  304. static int
  305. tas3001c_write_biquad_shadow( struct tas3001c_data_t *self,
  306. u_int channel,
  307. u_int filter,
  308. const union tas_biquad_t *biquad)
  309. {
  310. tas_shadow_t *shadow=self->super.shadow;
  311. enum tas3001c_reg_t reg;
  312. if (channel >= TAS3001C_BIQUAD_CHANNEL_COUNT ||
  313. filter >= TAS3001C_BIQUAD_FILTER_COUNT) return -EINVAL;
  314. reg=( channel ? TAS3001C_REG_RIGHT_BIQUAD0 : TAS3001C_REG_LEFT_BIQUAD0 ) + filter;
  315. SET_4_20(shadow[reg], 0,biquad->coeff.b0);
  316. SET_4_20(shadow[reg], 3,biquad->coeff.b1);
  317. SET_4_20(shadow[reg], 6,biquad->coeff.b2);
  318. SET_4_20(shadow[reg], 9,biquad->coeff.a1);
  319. SET_4_20(shadow[reg],12,biquad->coeff.a2);
  320. return 0;
  321. }
  322. static int
  323. tas3001c_write_biquad( struct tas3001c_data_t *self,
  324. u_int channel,
  325. u_int filter,
  326. const union tas_biquad_t *biquad)
  327. {
  328. int rc;
  329. rc=tas3001c_write_biquad_shadow(self, channel, filter, biquad);
  330. if (rc < 0) return rc;
  331. return tas3001c_sync_biquad(self, channel, filter);
  332. }
  333. static int
  334. tas3001c_write_biquad_list( struct tas3001c_data_t *self,
  335. u_int filter_count,
  336. u_int flags,
  337. struct tas_biquad_ctrl_t *biquads)
  338. {
  339. int i;
  340. int rc;
  341. if (flags & TAS_BIQUAD_FAST_LOAD) tas3001c_fast_load(self,1);
  342. for (i=0; i<filter_count; i++) {
  343. rc=tas3001c_write_biquad(self,
  344. biquads[i].channel,
  345. biquads[i].filter,
  346. &biquads[i].data);
  347. if (rc < 0) break;
  348. }
  349. if (flags & TAS_BIQUAD_FAST_LOAD) {
  350. tas3001c_fast_load(self,0);
  351. (void)tas3001c_sync_register(self,TAS3001C_REG_BASS);
  352. (void)tas3001c_sync_register(self,TAS3001C_REG_TREBLE);
  353. (void)tas3001c_sync_register(self,TAS3001C_REG_MIXER1);
  354. (void)tas3001c_sync_register(self,TAS3001C_REG_MIXER2);
  355. (void)tas3001c_sync_register(self,TAS3001C_REG_VOLUME);
  356. }
  357. return rc;
  358. }
  359. static int
  360. tas3001c_read_biquad( struct tas3001c_data_t *self,
  361. u_int channel,
  362. u_int filter,
  363. union tas_biquad_t *biquad)
  364. {
  365. tas_shadow_t *shadow=self->super.shadow;
  366. enum tas3001c_reg_t reg;
  367. if (channel >= TAS3001C_BIQUAD_CHANNEL_COUNT ||
  368. filter >= TAS3001C_BIQUAD_FILTER_COUNT) return -EINVAL;
  369. reg=( channel ? TAS3001C_REG_RIGHT_BIQUAD0 : TAS3001C_REG_LEFT_BIQUAD0 ) + filter;
  370. biquad->coeff.b0=GET_4_20(shadow[reg], 0);
  371. biquad->coeff.b1=GET_4_20(shadow[reg], 3);
  372. biquad->coeff.b2=GET_4_20(shadow[reg], 6);
  373. biquad->coeff.a1=GET_4_20(shadow[reg], 9);
  374. biquad->coeff.a2=GET_4_20(shadow[reg],12);
  375. return 0;
  376. }
  377. static int
  378. tas3001c_eq_rw( struct tas3001c_data_t *self,
  379. u_int cmd,
  380. u_long arg)
  381. {
  382. int rc;
  383. struct tas_biquad_ctrl_t biquad;
  384. void __user *argp = (void __user *)arg;
  385. if (copy_from_user(&biquad, argp, sizeof(struct tas_biquad_ctrl_t))) {
  386. return -EFAULT;
  387. }
  388. if (cmd & SIOC_IN) {
  389. rc=tas3001c_write_biquad(self, biquad.channel, biquad.filter, &biquad.data);
  390. if (rc != 0) return rc;
  391. }
  392. if (cmd & SIOC_OUT) {
  393. rc=tas3001c_read_biquad(self, biquad.channel, biquad.filter, &biquad.data);
  394. if (rc != 0) return rc;
  395. if (copy_to_user(argp, &biquad, sizeof(struct tas_biquad_ctrl_t))) {
  396. return -EFAULT;
  397. }
  398. }
  399. return 0;
  400. }
  401. static int
  402. tas3001c_eq_list_rw( struct tas3001c_data_t *self,
  403. u_int cmd,
  404. u_long arg)
  405. {
  406. int rc;
  407. int filter_count;
  408. int flags;
  409. int i,j;
  410. char sync_required[2][6];
  411. struct tas_biquad_ctrl_t biquad;
  412. struct tas_biquad_ctrl_list_t __user *argp = (void __user *)arg;
  413. memset(sync_required,0,sizeof(sync_required));
  414. if (copy_from_user(&filter_count, &argp->filter_count, sizeof(int)))
  415. return -EFAULT;
  416. if (copy_from_user(&flags, &argp->flags, sizeof(int)))
  417. return -EFAULT;
  418. if (cmd & SIOC_IN) {
  419. }
  420. for (i=0; i < filter_count; i++) {
  421. if (copy_from_user(&biquad, &argp->biquads[i],
  422. sizeof(struct tas_biquad_ctrl_t))) {
  423. return -EFAULT;
  424. }
  425. if (cmd & SIOC_IN) {
  426. sync_required[biquad.channel][biquad.filter]=1;
  427. rc=tas3001c_write_biquad_shadow(self, biquad.channel, biquad.filter, &biquad.data);
  428. if (rc != 0) return rc;
  429. }
  430. if (cmd & SIOC_OUT) {
  431. rc=tas3001c_read_biquad(self, biquad.channel, biquad.filter, &biquad.data);
  432. if (rc != 0) return rc;
  433. if (copy_to_user(&argp->biquads[i], &biquad,
  434. sizeof(struct tas_biquad_ctrl_t))) {
  435. return -EFAULT;
  436. }
  437. }
  438. }
  439. if (cmd & SIOC_IN) {
  440. if (flags & TAS_BIQUAD_FAST_LOAD) tas3001c_fast_load(self,1);
  441. for (i=0; i<2; i++) {
  442. for (j=0; j<6; j++) {
  443. if (sync_required[i][j]) {
  444. rc=tas3001c_sync_biquad(self, i, j);
  445. if (rc < 0) return rc;
  446. }
  447. }
  448. }
  449. if (flags & TAS_BIQUAD_FAST_LOAD) {
  450. tas3001c_fast_load(self,0);
  451. /* now we need to set up the mixers again,
  452. because leaving fast mode resets them. */
  453. (void)tas3001c_sync_register(self,TAS3001C_REG_BASS);
  454. (void)tas3001c_sync_register(self,TAS3001C_REG_TREBLE);
  455. (void)tas3001c_sync_register(self,TAS3001C_REG_MIXER1);
  456. (void)tas3001c_sync_register(self,TAS3001C_REG_MIXER2);
  457. (void)tas3001c_sync_register(self,TAS3001C_REG_VOLUME);
  458. }
  459. }
  460. return 0;
  461. }
  462. static int
  463. tas3001c_update_drce( struct tas3001c_data_t *self,
  464. int flags,
  465. struct tas_drce_t *drce)
  466. {
  467. tas_shadow_t *shadow;
  468. shadow=self->super.shadow;
  469. shadow[TAS3001C_REG_DRC][1] = 0xc1;
  470. if (flags & TAS_DRCE_THRESHOLD) {
  471. self->drce_state.threshold=quantize_db(drce->threshold);
  472. shadow[TAS3001C_REG_DRC][2] = db_to_regval(self->drce_state.threshold);
  473. }
  474. if (flags & TAS_DRCE_ENABLE) {
  475. self->drce_state.enable = drce->enable;
  476. }
  477. if (!self->drce_state.enable) {
  478. shadow[TAS3001C_REG_DRC][0] = 0xf0;
  479. }
  480. #ifdef DEBUG_DRCE
  481. printk("DRCE IOCTL: set [ ENABLE:%x THRESH:%x\n",
  482. self->drce_state.enable,
  483. self->drce_state.threshold);
  484. printk("DRCE IOCTL: reg [ %02x %02x ]\n",
  485. (unsigned char)shadow[TAS3001C_REG_DRC][0],
  486. (unsigned char)shadow[TAS3001C_REG_DRC][1]);
  487. #endif
  488. return tas3001c_sync_register(self, TAS3001C_REG_DRC);
  489. }
  490. static int
  491. tas3001c_drce_rw( struct tas3001c_data_t *self,
  492. u_int cmd,
  493. u_long arg)
  494. {
  495. int rc;
  496. struct tas_drce_ctrl_t drce_ctrl;
  497. void __user *argp = (void __user *)arg;
  498. if (copy_from_user(&drce_ctrl, argp, sizeof(struct tas_drce_ctrl_t)))
  499. return -EFAULT;
  500. #ifdef DEBUG_DRCE
  501. printk("DRCE IOCTL: input [ FLAGS:%x ENABLE:%x THRESH:%x\n",
  502. drce_ctrl.flags,
  503. drce_ctrl.data.enable,
  504. drce_ctrl.data.threshold);
  505. #endif
  506. if (cmd & SIOC_IN) {
  507. rc = tas3001c_update_drce(self, drce_ctrl.flags, &drce_ctrl.data);
  508. if (rc < 0)
  509. return rc;
  510. }
  511. if (cmd & SIOC_OUT) {
  512. if (drce_ctrl.flags & TAS_DRCE_ENABLE)
  513. drce_ctrl.data.enable = self->drce_state.enable;
  514. if (drce_ctrl.flags & TAS_DRCE_THRESHOLD)
  515. drce_ctrl.data.threshold = self->drce_state.threshold;
  516. if (copy_to_user(argp, &drce_ctrl,
  517. sizeof(struct tas_drce_ctrl_t))) {
  518. return -EFAULT;
  519. }
  520. }
  521. return 0;
  522. }
  523. static void
  524. tas3001c_update_device_parameters(struct tas3001c_data_t *self)
  525. {
  526. int i,j;
  527. if (!self) return;
  528. if (self->output_id == TAS_OUTPUT_HEADPHONES) {
  529. tas3001c_fast_load(self, 1);
  530. for (i=0; i<TAS3001C_BIQUAD_CHANNEL_COUNT; i++) {
  531. for (j=0; j<TAS3001C_BIQUAD_FILTER_COUNT; j++) {
  532. tas3001c_write_biquad(self, i, j, &tas3001c_eq_unity);
  533. }
  534. }
  535. tas3001c_fast_load(self, 0);
  536. (void)tas3001c_sync_register(self,TAS3001C_REG_BASS);
  537. (void)tas3001c_sync_register(self,TAS3001C_REG_TREBLE);
  538. (void)tas3001c_sync_register(self,TAS3001C_REG_MIXER1);
  539. (void)tas3001c_sync_register(self,TAS3001C_REG_MIXER2);
  540. (void)tas3001c_sync_register(self,TAS3001C_REG_VOLUME);
  541. return;
  542. }
  543. for (i=0; tas3001c_eq_prefs[i]; i++) {
  544. struct tas_eq_pref_t *eq = tas3001c_eq_prefs[i];
  545. if (eq->device_id == self->device_id &&
  546. (eq->output_id == 0 || eq->output_id == self->output_id) &&
  547. (eq->speaker_id == 0 || eq->speaker_id == self->speaker_id)) {
  548. tas3001c_update_drce(self, TAS_DRCE_ALL, eq->drce);
  549. tas3001c_write_biquad_list(self, eq->filter_count, TAS_BIQUAD_FAST_LOAD, eq->biquads);
  550. break;
  551. }
  552. }
  553. }
  554. static void
  555. tas3001c_device_change_handler(struct work_struct *work)
  556. {
  557. struct tas3001c_data_t *self;
  558. self = container_of(work, struct tas3001c_data_t, change);
  559. tas3001c_update_device_parameters(self);
  560. }
  561. static int
  562. tas3001c_output_device_change( struct tas3001c_data_t *self,
  563. int device_id,
  564. int output_id,
  565. int speaker_id)
  566. {
  567. self->device_id=device_id;
  568. self->output_id=output_id;
  569. self->speaker_id=speaker_id;
  570. schedule_work(&self->change);
  571. return 0;
  572. }
  573. static int
  574. tas3001c_device_ioctl( struct tas3001c_data_t *self,
  575. u_int cmd,
  576. u_long arg)
  577. {
  578. uint __user *argp = (void __user *)arg;
  579. switch (cmd) {
  580. case TAS_READ_EQ:
  581. case TAS_WRITE_EQ:
  582. return tas3001c_eq_rw(self, cmd, arg);
  583. case TAS_READ_EQ_LIST:
  584. case TAS_WRITE_EQ_LIST:
  585. return tas3001c_eq_list_rw(self, cmd, arg);
  586. case TAS_READ_EQ_FILTER_COUNT:
  587. put_user(TAS3001C_BIQUAD_FILTER_COUNT, argp);
  588. return 0;
  589. case TAS_READ_EQ_CHANNEL_COUNT:
  590. put_user(TAS3001C_BIQUAD_CHANNEL_COUNT, argp);
  591. return 0;
  592. case TAS_READ_DRCE:
  593. case TAS_WRITE_DRCE:
  594. return tas3001c_drce_rw(self, cmd, arg);
  595. case TAS_READ_DRCE_CAPS:
  596. put_user(TAS_DRCE_ENABLE | TAS_DRCE_THRESHOLD, argp);
  597. return 0;
  598. case TAS_READ_DRCE_MIN:
  599. case TAS_READ_DRCE_MAX: {
  600. struct tas_drce_ctrl_t drce_ctrl;
  601. if (copy_from_user(&drce_ctrl, argp,
  602. sizeof(struct tas_drce_ctrl_t))) {
  603. return -EFAULT;
  604. }
  605. if (drce_ctrl.flags & TAS_DRCE_THRESHOLD) {
  606. if (cmd == TAS_READ_DRCE_MIN) {
  607. drce_ctrl.data.threshold=-36<<8;
  608. } else {
  609. drce_ctrl.data.threshold=-6<<8;
  610. }
  611. }
  612. if (copy_to_user(argp, &drce_ctrl,
  613. sizeof(struct tas_drce_ctrl_t))) {
  614. return -EFAULT;
  615. }
  616. }
  617. }
  618. return -EINVAL;
  619. }
  620. static int
  621. tas3001c_init_mixer(struct tas3001c_data_t *self)
  622. {
  623. unsigned char mcr = (1<<6)+(2<<4)+(2<<2);
  624. /* Make sure something answers on the i2c bus */
  625. if (tas3001c_write_register(self, TAS3001C_REG_MCR, &mcr,
  626. WRITE_NORMAL|FORCE_WRITE) < 0)
  627. return -1;
  628. tas3001c_fast_load(self, 1);
  629. (void)tas3001c_sync_register(self,TAS3001C_REG_RIGHT_BIQUAD0);
  630. (void)tas3001c_sync_register(self,TAS3001C_REG_RIGHT_BIQUAD1);
  631. (void)tas3001c_sync_register(self,TAS3001C_REG_RIGHT_BIQUAD2);
  632. (void)tas3001c_sync_register(self,TAS3001C_REG_RIGHT_BIQUAD3);
  633. (void)tas3001c_sync_register(self,TAS3001C_REG_RIGHT_BIQUAD4);
  634. (void)tas3001c_sync_register(self,TAS3001C_REG_RIGHT_BIQUAD5);
  635. (void)tas3001c_sync_register(self,TAS3001C_REG_RIGHT_BIQUAD6);
  636. (void)tas3001c_sync_register(self,TAS3001C_REG_LEFT_BIQUAD0);
  637. (void)tas3001c_sync_register(self,TAS3001C_REG_LEFT_BIQUAD1);
  638. (void)tas3001c_sync_register(self,TAS3001C_REG_LEFT_BIQUAD2);
  639. (void)tas3001c_sync_register(self,TAS3001C_REG_LEFT_BIQUAD3);
  640. (void)tas3001c_sync_register(self,TAS3001C_REG_LEFT_BIQUAD4);
  641. (void)tas3001c_sync_register(self,TAS3001C_REG_LEFT_BIQUAD5);
  642. (void)tas3001c_sync_register(self,TAS3001C_REG_LEFT_BIQUAD6);
  643. tas3001c_fast_load(self, 0);
  644. tas3001c_set_mixer_level(self, SOUND_MIXER_VOLUME, VOL_DEFAULT<<8 | VOL_DEFAULT);
  645. tas3001c_set_mixer_level(self, SOUND_MIXER_PCM, INPUT_DEFAULT<<8 | INPUT_DEFAULT);
  646. tas3001c_set_mixer_level(self, SOUND_MIXER_ALTPCM, 0);
  647. tas3001c_set_mixer_level(self, SOUND_MIXER_BASS, BASS_DEFAULT);
  648. tas3001c_set_mixer_level(self, SOUND_MIXER_TREBLE, TREBLE_DEFAULT);
  649. return 0;
  650. }
  651. static int
  652. tas3001c_uninit_mixer(struct tas3001c_data_t *self)
  653. {
  654. tas3001c_set_mixer_level(self, SOUND_MIXER_VOLUME, 0);
  655. tas3001c_set_mixer_level(self, SOUND_MIXER_PCM, 0);
  656. tas3001c_set_mixer_level(self, SOUND_MIXER_ALTPCM, 0);
  657. tas3001c_set_mixer_level(self, SOUND_MIXER_BASS, 0);
  658. tas3001c_set_mixer_level(self, SOUND_MIXER_TREBLE, 0);
  659. return 0;
  660. }
  661. static int
  662. tas3001c_init(struct i2c_client *client)
  663. {
  664. struct tas3001c_data_t *self;
  665. size_t sz = sizeof(*self) + (TAS3001C_REG_MAX*sizeof(tas_shadow_t));
  666. int i, j;
  667. self = kzalloc(sz, GFP_KERNEL);
  668. if (!self)
  669. return -ENOMEM;
  670. self->super.client = client;
  671. self->super.shadow = (tas_shadow_t *)(self+1);
  672. self->output_id = TAS_OUTPUT_HEADPHONES;
  673. dev_set_drvdata(&client->dev, self);
  674. for (i = 0; i < TAS3001C_BIQUAD_CHANNEL_COUNT; i++)
  675. for (j = 0; j < TAS3001C_BIQUAD_FILTER_COUNT; j++)
  676. tas3001c_write_biquad_shadow(self, i, j,
  677. &tas3001c_eq_unity);
  678. INIT_WORK(&self->change, tas3001c_device_change_handler);
  679. return 0;
  680. }
  681. static void
  682. tas3001c_uninit(struct tas3001c_data_t *self)
  683. {
  684. tas3001c_uninit_mixer(self);
  685. kfree(self);
  686. }
  687. struct tas_driver_hooks_t tas3001c_hooks = {
  688. .init = (tas_hook_init_t)tas3001c_init,
  689. .post_init = (tas_hook_post_init_t)tas3001c_init_mixer,
  690. .uninit = (tas_hook_uninit_t)tas3001c_uninit,
  691. .get_mixer_level = (tas_hook_get_mixer_level_t)tas3001c_get_mixer_level,
  692. .set_mixer_level = (tas_hook_set_mixer_level_t)tas3001c_set_mixer_level,
  693. .enter_sleep = (tas_hook_enter_sleep_t)tas3001c_enter_sleep,
  694. .leave_sleep = (tas_hook_leave_sleep_t)tas3001c_leave_sleep,
  695. .supported_mixers = (tas_hook_supported_mixers_t)tas3001c_supported_mixers,
  696. .mixer_is_stereo = (tas_hook_mixer_is_stereo_t)tas3001c_mixer_is_stereo,
  697. .stereo_mixers = (tas_hook_stereo_mixers_t)tas3001c_stereo_mixers,
  698. .output_device_change = (tas_hook_output_device_change_t)tas3001c_output_device_change,
  699. .device_ioctl = (tas_hook_device_ioctl_t)tas3001c_device_ioctl
  700. };