au88x0_pcm.c 16 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548
  1. /*
  2. * This program is free software; you can redistribute it and/or modify
  3. * it under the terms of the GNU General Public License as published by
  4. * the Free Software Foundation; either version 2 of the License, or
  5. * (at your option) any later version.
  6. *
  7. * This program is distributed in the hope that it will be useful,
  8. * but WITHOUT ANY WARRANTY; without even the implied warranty of
  9. * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  10. * GNU Library General Public License for more details.
  11. *
  12. * You should have received a copy of the GNU General Public License
  13. * along with this program; if not, write to the Free Software
  14. * Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
  15. */
  16. /*
  17. * Vortex PCM ALSA driver.
  18. *
  19. * Supports ADB and WT DMA. Unfortunately, WT channels do not run yet.
  20. * It remains stuck,and DMA transfers do not happen.
  21. */
  22. #include <sound/asoundef.h>
  23. #include <sound/driver.h>
  24. #include <linux/time.h>
  25. #include <sound/core.h>
  26. #include <sound/pcm.h>
  27. #include <sound/pcm_params.h>
  28. #include "au88x0.h"
  29. #define VORTEX_PCM_TYPE(x) (x->name[40])
  30. /* hardware definition */
  31. static snd_pcm_hardware_t snd_vortex_playback_hw_adb = {
  32. .info =
  33. (SNDRV_PCM_INFO_MMAP | /* SNDRV_PCM_INFO_RESUME | */
  34. SNDRV_PCM_INFO_PAUSE | SNDRV_PCM_INFO_INTERLEAVED |
  35. SNDRV_PCM_INFO_MMAP_VALID),
  36. .formats =
  37. SNDRV_PCM_FMTBIT_S16_LE | SNDRV_PCM_FMTBIT_U8 |
  38. SNDRV_PCM_FMTBIT_MU_LAW | SNDRV_PCM_FMTBIT_A_LAW,
  39. .rates = SNDRV_PCM_RATE_CONTINUOUS,
  40. .rate_min = 5000,
  41. .rate_max = 48000,
  42. .channels_min = 1,
  43. #ifdef CHIP_AU8830
  44. .channels_max = 4,
  45. #else
  46. .channels_max = 2,
  47. #endif
  48. .buffer_bytes_max = 0x10000,
  49. .period_bytes_min = 0x1,
  50. .period_bytes_max = 0x1000,
  51. .periods_min = 2,
  52. .periods_max = 32,
  53. };
  54. #ifndef CHIP_AU8820
  55. static snd_pcm_hardware_t snd_vortex_playback_hw_a3d = {
  56. .info =
  57. (SNDRV_PCM_INFO_MMAP | /* SNDRV_PCM_INFO_RESUME | */
  58. SNDRV_PCM_INFO_PAUSE | SNDRV_PCM_INFO_INTERLEAVED |
  59. SNDRV_PCM_INFO_MMAP_VALID),
  60. .formats =
  61. SNDRV_PCM_FMTBIT_S16_LE | SNDRV_PCM_FMTBIT_U8 |
  62. SNDRV_PCM_FMTBIT_MU_LAW | SNDRV_PCM_FMTBIT_A_LAW,
  63. .rates = SNDRV_PCM_RATE_CONTINUOUS,
  64. .rate_min = 5000,
  65. .rate_max = 48000,
  66. .channels_min = 1,
  67. .channels_max = 1,
  68. .buffer_bytes_max = 0x10000,
  69. .period_bytes_min = 0x100,
  70. .period_bytes_max = 0x1000,
  71. .periods_min = 2,
  72. .periods_max = 64,
  73. };
  74. #endif
  75. static snd_pcm_hardware_t snd_vortex_playback_hw_spdif = {
  76. .info =
  77. (SNDRV_PCM_INFO_MMAP | /* SNDRV_PCM_INFO_RESUME | */
  78. SNDRV_PCM_INFO_PAUSE | SNDRV_PCM_INFO_INTERLEAVED |
  79. SNDRV_PCM_INFO_MMAP_VALID),
  80. .formats =
  81. SNDRV_PCM_FMTBIT_S16_LE | SNDRV_PCM_FMTBIT_U8 |
  82. SNDRV_PCM_FMTBIT_IEC958_SUBFRAME_LE | SNDRV_PCM_FMTBIT_MU_LAW |
  83. SNDRV_PCM_FMTBIT_A_LAW,
  84. .rates =
  85. SNDRV_PCM_RATE_32000 | SNDRV_PCM_RATE_44100 | SNDRV_PCM_RATE_48000,
  86. .rate_min = 32000,
  87. .rate_max = 48000,
  88. .channels_min = 1,
  89. .channels_max = 2,
  90. .buffer_bytes_max = 0x10000,
  91. .period_bytes_min = 0x100,
  92. .period_bytes_max = 0x1000,
  93. .periods_min = 2,
  94. .periods_max = 64,
  95. };
  96. #ifndef CHIP_AU8810
  97. static snd_pcm_hardware_t snd_vortex_playback_hw_wt = {
  98. .info = (SNDRV_PCM_INFO_MMAP |
  99. SNDRV_PCM_INFO_INTERLEAVED |
  100. SNDRV_PCM_INFO_BLOCK_TRANSFER | SNDRV_PCM_INFO_MMAP_VALID),
  101. .formats = SNDRV_PCM_FMTBIT_S16_LE,
  102. .rates = SNDRV_PCM_RATE_8000_48000 | SNDRV_PCM_RATE_CONTINUOUS, // SNDRV_PCM_RATE_48000,
  103. .rate_min = 8000,
  104. .rate_max = 48000,
  105. .channels_min = 1,
  106. .channels_max = 2,
  107. .buffer_bytes_max = 0x10000,
  108. .period_bytes_min = 0x0400,
  109. .period_bytes_max = 0x1000,
  110. .periods_min = 2,
  111. .periods_max = 64,
  112. };
  113. #endif
  114. /* open callback */
  115. static int snd_vortex_pcm_open(snd_pcm_substream_t * substream)
  116. {
  117. vortex_t *vortex = snd_pcm_substream_chip(substream);
  118. snd_pcm_runtime_t *runtime = substream->runtime;
  119. int err;
  120. /* Force equal size periods */
  121. if ((err =
  122. snd_pcm_hw_constraint_integer(runtime,
  123. SNDRV_PCM_HW_PARAM_PERIODS)) < 0)
  124. return err;
  125. /* Avoid PAGE_SIZE boundary to fall inside of a period. */
  126. if ((err =
  127. snd_pcm_hw_constraint_pow2(runtime, 0,
  128. SNDRV_PCM_HW_PARAM_PERIOD_BYTES)) < 0)
  129. return err;
  130. if (VORTEX_PCM_TYPE(substream->pcm) != VORTEX_PCM_WT) {
  131. #ifndef CHIP_AU8820
  132. if (VORTEX_PCM_TYPE(substream->pcm) == VORTEX_PCM_A3D) {
  133. runtime->hw = snd_vortex_playback_hw_a3d;
  134. }
  135. #endif
  136. if (VORTEX_PCM_TYPE(substream->pcm) == VORTEX_PCM_SPDIF) {
  137. runtime->hw = snd_vortex_playback_hw_spdif;
  138. switch (vortex->spdif_sr) {
  139. case 32000:
  140. runtime->hw.rates = SNDRV_PCM_RATE_32000;
  141. break;
  142. case 44100:
  143. runtime->hw.rates = SNDRV_PCM_RATE_44100;
  144. break;
  145. case 48000:
  146. runtime->hw.rates = SNDRV_PCM_RATE_48000;
  147. break;
  148. }
  149. }
  150. if (VORTEX_PCM_TYPE(substream->pcm) == VORTEX_PCM_ADB
  151. || VORTEX_PCM_TYPE(substream->pcm) == VORTEX_PCM_I2S)
  152. runtime->hw = snd_vortex_playback_hw_adb;
  153. substream->runtime->private_data = NULL;
  154. }
  155. #ifndef CHIP_AU8810
  156. else {
  157. runtime->hw = snd_vortex_playback_hw_wt;
  158. substream->runtime->private_data = NULL;
  159. }
  160. #endif
  161. return 0;
  162. }
  163. /* close callback */
  164. static int snd_vortex_pcm_close(snd_pcm_substream_t * substream)
  165. {
  166. //vortex_t *chip = snd_pcm_substream_chip(substream);
  167. stream_t *stream = (stream_t *) substream->runtime->private_data;
  168. // the hardware-specific codes will be here
  169. if (stream != NULL) {
  170. stream->substream = NULL;
  171. stream->nr_ch = 0;
  172. }
  173. substream->runtime->private_data = NULL;
  174. return 0;
  175. }
  176. /* hw_params callback */
  177. static int
  178. snd_vortex_pcm_hw_params(snd_pcm_substream_t * substream,
  179. snd_pcm_hw_params_t * hw_params)
  180. {
  181. vortex_t *chip = snd_pcm_substream_chip(substream);
  182. stream_t *stream = (stream_t *) (substream->runtime->private_data);
  183. snd_pcm_sgbuf_t *sgbuf;
  184. int err;
  185. // Alloc buffer memory.
  186. err =
  187. snd_pcm_lib_malloc_pages(substream, params_buffer_bytes(hw_params));
  188. if (err < 0) {
  189. printk(KERN_ERR "Vortex: pcm page alloc failed!\n");
  190. return err;
  191. }
  192. //sgbuf = (snd_pcm_sgbuf_t *) substream->runtime->dma_private;
  193. sgbuf = snd_pcm_substream_sgbuf(substream);
  194. /*
  195. printk(KERN_INFO "Vortex: periods %d, period_bytes %d, channels = %d\n", params_periods(hw_params),
  196. params_period_bytes(hw_params), params_channels(hw_params));
  197. */
  198. spin_lock_irq(&chip->lock);
  199. // Make audio routes and config buffer DMA.
  200. if (VORTEX_PCM_TYPE(substream->pcm) != VORTEX_PCM_WT) {
  201. int dma, type = VORTEX_PCM_TYPE(substream->pcm);
  202. /* Dealloc any routes. */
  203. if (stream != NULL)
  204. vortex_adb_allocroute(chip, stream->dma,
  205. stream->nr_ch, stream->dir,
  206. stream->type);
  207. /* Alloc routes. */
  208. dma =
  209. vortex_adb_allocroute(chip, -1,
  210. params_channels(hw_params),
  211. substream->stream, type);
  212. if (dma < 0)
  213. return dma;
  214. stream = substream->runtime->private_data = &chip->dma_adb[dma];
  215. stream->substream = substream;
  216. /* Setup Buffers. */
  217. vortex_adbdma_setbuffers(chip, dma, sgbuf,
  218. params_period_bytes(hw_params),
  219. params_periods(hw_params));
  220. }
  221. #ifndef CHIP_AU8810
  222. else {
  223. /* if (stream != NULL)
  224. vortex_wt_allocroute(chip, substream->number, 0); */
  225. vortex_wt_allocroute(chip, substream->number,
  226. params_channels(hw_params));
  227. stream = substream->runtime->private_data =
  228. &chip->dma_wt[substream->number];
  229. stream->dma = substream->number;
  230. stream->substream = substream;
  231. vortex_wtdma_setbuffers(chip, substream->number, sgbuf,
  232. params_period_bytes(hw_params),
  233. params_periods(hw_params));
  234. }
  235. #endif
  236. spin_unlock_irq(&chip->lock);
  237. return 0;
  238. }
  239. /* hw_free callback */
  240. static int snd_vortex_pcm_hw_free(snd_pcm_substream_t * substream)
  241. {
  242. vortex_t *chip = snd_pcm_substream_chip(substream);
  243. stream_t *stream = (stream_t *) (substream->runtime->private_data);
  244. spin_lock_irq(&chip->lock);
  245. // Delete audio routes.
  246. if (VORTEX_PCM_TYPE(substream->pcm) != VORTEX_PCM_WT) {
  247. if (stream != NULL)
  248. vortex_adb_allocroute(chip, stream->dma,
  249. stream->nr_ch, stream->dir,
  250. stream->type);
  251. }
  252. #ifndef CHIP_AU8810
  253. else {
  254. if (stream != NULL)
  255. vortex_wt_allocroute(chip, stream->dma, 0);
  256. }
  257. #endif
  258. substream->runtime->private_data = NULL;
  259. spin_unlock_irq(&chip->lock);
  260. return snd_pcm_lib_free_pages(substream);
  261. }
  262. /* prepare callback */
  263. static int snd_vortex_pcm_prepare(snd_pcm_substream_t * substream)
  264. {
  265. vortex_t *chip = snd_pcm_substream_chip(substream);
  266. snd_pcm_runtime_t *runtime = substream->runtime;
  267. stream_t *stream = (stream_t *) substream->runtime->private_data;
  268. int dma = stream->dma, fmt, dir;
  269. // set up the hardware with the current configuration.
  270. if (substream->stream == SNDRV_PCM_STREAM_PLAYBACK)
  271. dir = 1;
  272. else
  273. dir = 0;
  274. fmt = vortex_alsafmt_aspfmt(runtime->format);
  275. spin_lock_irq(&chip->lock);
  276. if (VORTEX_PCM_TYPE(substream->pcm) != VORTEX_PCM_WT) {
  277. vortex_adbdma_setmode(chip, dma, 1, dir, fmt, 0 /*? */ ,
  278. 0);
  279. vortex_adbdma_setstartbuffer(chip, dma, 0);
  280. if (VORTEX_PCM_TYPE(substream->pcm) != VORTEX_PCM_SPDIF)
  281. vortex_adb_setsrc(chip, dma, runtime->rate, dir);
  282. }
  283. #ifndef CHIP_AU8810
  284. else {
  285. vortex_wtdma_setmode(chip, dma, 1, fmt, 0, 0);
  286. // FIXME: Set rate (i guess using vortex_wt_writereg() somehow).
  287. vortex_wtdma_setstartbuffer(chip, dma, 0);
  288. }
  289. #endif
  290. spin_unlock_irq(&chip->lock);
  291. return 0;
  292. }
  293. /* trigger callback */
  294. static int snd_vortex_pcm_trigger(snd_pcm_substream_t * substream, int cmd)
  295. {
  296. vortex_t *chip = snd_pcm_substream_chip(substream);
  297. stream_t *stream = (stream_t *) substream->runtime->private_data;
  298. int dma = stream->dma;
  299. spin_lock(&chip->lock);
  300. switch (cmd) {
  301. case SNDRV_PCM_TRIGGER_START:
  302. // do something to start the PCM engine
  303. //printk(KERN_INFO "vortex: start %d\n", dma);
  304. stream->fifo_enabled = 1;
  305. if (VORTEX_PCM_TYPE(substream->pcm) != VORTEX_PCM_WT) {
  306. vortex_adbdma_resetup(chip, dma);
  307. vortex_adbdma_startfifo(chip, dma);
  308. }
  309. #ifndef CHIP_AU8810
  310. else {
  311. printk(KERN_INFO "vortex: wt start %d\n", dma);
  312. vortex_wtdma_startfifo(chip, dma);
  313. }
  314. #endif
  315. break;
  316. case SNDRV_PCM_TRIGGER_STOP:
  317. // do something to stop the PCM engine
  318. //printk(KERN_INFO "vortex: stop %d\n", dma);
  319. stream->fifo_enabled = 0;
  320. if (VORTEX_PCM_TYPE(substream->pcm) != VORTEX_PCM_WT)
  321. vortex_adbdma_pausefifo(chip, dma);
  322. //vortex_adbdma_stopfifo(chip, dma);
  323. #ifndef CHIP_AU8810
  324. else {
  325. printk(KERN_INFO "vortex: wt stop %d\n", dma);
  326. vortex_wtdma_stopfifo(chip, dma);
  327. }
  328. #endif
  329. break;
  330. case SNDRV_PCM_TRIGGER_PAUSE_PUSH:
  331. //printk(KERN_INFO "vortex: pause %d\n", dma);
  332. if (VORTEX_PCM_TYPE(substream->pcm) != VORTEX_PCM_WT)
  333. vortex_adbdma_pausefifo(chip, dma);
  334. #ifndef CHIP_AU8810
  335. else
  336. vortex_wtdma_pausefifo(chip, dma);
  337. #endif
  338. break;
  339. case SNDRV_PCM_TRIGGER_PAUSE_RELEASE:
  340. //printk(KERN_INFO "vortex: resume %d\n", dma);
  341. if (VORTEX_PCM_TYPE(substream->pcm) != VORTEX_PCM_WT)
  342. vortex_adbdma_resumefifo(chip, dma);
  343. #ifndef CHIP_AU8810
  344. else
  345. vortex_wtdma_resumefifo(chip, dma);
  346. #endif
  347. break;
  348. default:
  349. spin_unlock(&chip->lock);
  350. return -EINVAL;
  351. }
  352. spin_unlock(&chip->lock);
  353. return 0;
  354. }
  355. /* pointer callback */
  356. static snd_pcm_uframes_t snd_vortex_pcm_pointer(snd_pcm_substream_t * substream)
  357. {
  358. vortex_t *chip = snd_pcm_substream_chip(substream);
  359. stream_t *stream = (stream_t *) substream->runtime->private_data;
  360. int dma = stream->dma;
  361. snd_pcm_uframes_t current_ptr = 0;
  362. spin_lock(&chip->lock);
  363. if (VORTEX_PCM_TYPE(substream->pcm) != VORTEX_PCM_WT)
  364. current_ptr = vortex_adbdma_getlinearpos(chip, dma);
  365. #ifndef CHIP_AU8810
  366. else
  367. current_ptr = vortex_wtdma_getlinearpos(chip, dma);
  368. #endif
  369. //printk(KERN_INFO "vortex: pointer = 0x%x\n", current_ptr);
  370. spin_unlock(&chip->lock);
  371. return (bytes_to_frames(substream->runtime, current_ptr));
  372. }
  373. /* Page callback. */
  374. /*
  375. static struct page *snd_pcm_sgbuf_ops_page(snd_pcm_substream_t *substream, unsigned long offset) {
  376. }
  377. */
  378. /* operators */
  379. static snd_pcm_ops_t snd_vortex_playback_ops = {
  380. .open = snd_vortex_pcm_open,
  381. .close = snd_vortex_pcm_close,
  382. .ioctl = snd_pcm_lib_ioctl,
  383. .hw_params = snd_vortex_pcm_hw_params,
  384. .hw_free = snd_vortex_pcm_hw_free,
  385. .prepare = snd_vortex_pcm_prepare,
  386. .trigger = snd_vortex_pcm_trigger,
  387. .pointer = snd_vortex_pcm_pointer,
  388. .page = snd_pcm_sgbuf_ops_page,
  389. };
  390. /*
  391. * definitions of capture are omitted here...
  392. */
  393. static char *vortex_pcm_prettyname[VORTEX_PCM_LAST] = {
  394. "AU88x0 ADB",
  395. "AU88x0 SPDIF",
  396. "AU88x0 A3D",
  397. "AU88x0 WT",
  398. "AU88x0 I2S",
  399. };
  400. static char *vortex_pcm_name[VORTEX_PCM_LAST] = {
  401. "adb",
  402. "spdif",
  403. "a3d",
  404. "wt",
  405. "i2s",
  406. };
  407. /* SPDIF kcontrol */
  408. static int snd_vortex_spdif_info(snd_kcontrol_t *kcontrol, snd_ctl_elem_info_t * uinfo)
  409. {
  410. uinfo->type = SNDRV_CTL_ELEM_TYPE_IEC958;
  411. uinfo->count = 1;
  412. return 0;
  413. }
  414. static int snd_vortex_spdif_mask_get(snd_kcontrol_t * kcontrol, snd_ctl_elem_value_t * ucontrol)
  415. {
  416. ucontrol->value.iec958.status[0] = 0xff;
  417. ucontrol->value.iec958.status[1] = 0xff;
  418. ucontrol->value.iec958.status[2] = 0xff;
  419. ucontrol->value.iec958.status[3] = IEC958_AES3_CON_FS;
  420. return 0;
  421. }
  422. static int snd_vortex_spdif_get(snd_kcontrol_t * kcontrol, snd_ctl_elem_value_t * ucontrol)
  423. {
  424. vortex_t *vortex = snd_kcontrol_chip(kcontrol);
  425. ucontrol->value.iec958.status[0] = 0x00;
  426. ucontrol->value.iec958.status[1] = IEC958_AES1_CON_ORIGINAL|IEC958_AES1_CON_DIGDIGCONV_ID;
  427. ucontrol->value.iec958.status[2] = 0x00;
  428. switch (vortex->spdif_sr) {
  429. case 32000: ucontrol->value.iec958.status[3] = IEC958_AES3_CON_FS_32000; break;
  430. case 44100: ucontrol->value.iec958.status[3] = IEC958_AES3_CON_FS_44100; break;
  431. case 48000: ucontrol->value.iec958.status[3] = IEC958_AES3_CON_FS_48000; break;
  432. }
  433. return 0;
  434. }
  435. static int snd_vortex_spdif_put(snd_kcontrol_t * kcontrol, snd_ctl_elem_value_t * ucontrol)
  436. {
  437. vortex_t *vortex = snd_kcontrol_chip(kcontrol);
  438. int spdif_sr = 48000;
  439. switch (ucontrol->value.iec958.status[3] & IEC958_AES3_CON_FS) {
  440. case IEC958_AES3_CON_FS_32000: spdif_sr = 32000; break;
  441. case IEC958_AES3_CON_FS_44100: spdif_sr = 44100; break;
  442. case IEC958_AES3_CON_FS_48000: spdif_sr = 48000; break;
  443. }
  444. if (spdif_sr == vortex->spdif_sr)
  445. return 0;
  446. vortex->spdif_sr = spdif_sr;
  447. vortex_spdif_init(vortex, vortex->spdif_sr, 1);
  448. return 1;
  449. }
  450. /* spdif controls */
  451. static snd_kcontrol_new_t snd_vortex_mixer_spdif[] __devinitdata = {
  452. {
  453. .iface = SNDRV_CTL_ELEM_IFACE_PCM,
  454. .name = SNDRV_CTL_NAME_IEC958("",PLAYBACK,DEFAULT),
  455. .info = snd_vortex_spdif_info,
  456. .get = snd_vortex_spdif_get,
  457. .put = snd_vortex_spdif_put,
  458. },
  459. {
  460. .access = SNDRV_CTL_ELEM_ACCESS_READ,
  461. .iface = SNDRV_CTL_ELEM_IFACE_PCM,
  462. .name = SNDRV_CTL_NAME_IEC958("",PLAYBACK,CON_MASK),
  463. .info = snd_vortex_spdif_info,
  464. .get = snd_vortex_spdif_mask_get
  465. },
  466. };
  467. /* create a pcm device */
  468. static int __devinit snd_vortex_new_pcm(vortex_t * chip, int idx, int nr)
  469. {
  470. snd_pcm_t *pcm;
  471. snd_kcontrol_t *kctl;
  472. int i;
  473. int err, nr_capt;
  474. if ((chip == 0) || (idx < 0) || (idx > VORTEX_PCM_LAST))
  475. return -ENODEV;
  476. /* idx indicates which kind of PCM device. ADB, SPDIF, I2S and A3D share the
  477. * same dma engine. WT uses it own separate dma engine whcih cant capture. */
  478. if (idx == VORTEX_PCM_ADB)
  479. nr_capt = nr;
  480. else
  481. nr_capt = 0;
  482. if ((err =
  483. snd_pcm_new(chip->card, vortex_pcm_prettyname[idx], idx, nr,
  484. nr_capt, &pcm)) < 0)
  485. return err;
  486. strcpy(pcm->name, vortex_pcm_name[idx]);
  487. chip->pcm[idx] = pcm;
  488. // This is an evil hack, but it saves a lot of duplicated code.
  489. VORTEX_PCM_TYPE(pcm) = idx;
  490. pcm->private_data = chip;
  491. /* set operators */
  492. snd_pcm_set_ops(pcm, SNDRV_PCM_STREAM_PLAYBACK,
  493. &snd_vortex_playback_ops);
  494. if (idx == VORTEX_PCM_ADB)
  495. snd_pcm_set_ops(pcm, SNDRV_PCM_STREAM_CAPTURE,
  496. &snd_vortex_playback_ops);
  497. /* pre-allocation of Scatter-Gather buffers */
  498. snd_pcm_lib_preallocate_pages_for_all(pcm, SNDRV_DMA_TYPE_DEV_SG,
  499. snd_dma_pci_data(chip->pci_dev),
  500. 0x10000, 0x10000);
  501. if (VORTEX_PCM_TYPE(pcm) == VORTEX_PCM_SPDIF) {
  502. for (i = 0; i < ARRAY_SIZE(snd_vortex_mixer_spdif); i++) {
  503. kctl = snd_ctl_new1(&snd_vortex_mixer_spdif[i], chip);
  504. if (!kctl)
  505. return -ENOMEM;
  506. if ((err = snd_ctl_add(chip->card, kctl)) < 0)
  507. return err;
  508. }
  509. }
  510. return 0;
  511. }