Loading...
1/*
2 * Aztech AZT1605/AZT2316 Driver
3 * Copyright (C) 2007,2010 Rene Herman
4 *
5 * This program is free software: you can redistribute it and/or modify
6 * it under the terms of the GNU General Public License as published by
7 * the Free Software Foundation, either version 2 of the License, or
8 * (at your option) any later version.
9 *
10 * This program is distributed in the hope that it will be useful,
11 * but WITHOUT ANY WARRANTY; without even the implied warranty of
12 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
13 * GNU General Public License for more details.
14 *
15 * You should have received a copy of the GNU General Public License
16 * along with this program. If not, see <http://www.gnu.org/licenses/>.
17 *
18 */
19
20#include <linux/kernel.h>
21#include <linux/module.h>
22#include <linux/isa.h>
23#include <linux/delay.h>
24#include <linux/io.h>
25#include <asm/processor.h>
26#include <sound/core.h>
27#include <sound/initval.h>
28#include <sound/wss.h>
29#include <sound/mpu401.h>
30#include <sound/opl3.h>
31
32MODULE_DESCRIPTION(CRD_NAME);
33MODULE_AUTHOR("Rene Herman");
34MODULE_LICENSE("GPL");
35
36static int index[SNDRV_CARDS] = SNDRV_DEFAULT_IDX;
37static char *id[SNDRV_CARDS] = SNDRV_DEFAULT_STR;
38static bool enable[SNDRV_CARDS] = SNDRV_DEFAULT_ENABLE;
39
40module_param_array(index, int, NULL, 0444);
41MODULE_PARM_DESC(index, "Index value for " CRD_NAME " soundcard.");
42module_param_array(id, charp, NULL, 0444);
43MODULE_PARM_DESC(id, "ID string for " CRD_NAME " soundcard.");
44module_param_array(enable, bool, NULL, 0444);
45MODULE_PARM_DESC(enable, "Enable " CRD_NAME " soundcard.");
46
47static long port[SNDRV_CARDS] = SNDRV_DEFAULT_PORT;
48static long wss_port[SNDRV_CARDS] = SNDRV_DEFAULT_PORT;
49static long mpu_port[SNDRV_CARDS] = SNDRV_DEFAULT_PORT;
50static long fm_port[SNDRV_CARDS] = SNDRV_DEFAULT_PORT;
51static int irq[SNDRV_CARDS] = SNDRV_DEFAULT_IRQ;
52static int mpu_irq[SNDRV_CARDS] = SNDRV_DEFAULT_IRQ;
53static int dma1[SNDRV_CARDS] = SNDRV_DEFAULT_DMA;
54static int dma2[SNDRV_CARDS] = SNDRV_DEFAULT_DMA;
55
56module_param_array(port, long, NULL, 0444);
57MODULE_PARM_DESC(port, "Port # for " CRD_NAME " driver.");
58module_param_array(wss_port, long, NULL, 0444);
59MODULE_PARM_DESC(wss_port, "WSS port # for " CRD_NAME " driver.");
60module_param_array(mpu_port, long, NULL, 0444);
61MODULE_PARM_DESC(mpu_port, "MPU-401 port # for " CRD_NAME " driver.");
62module_param_array(fm_port, long, NULL, 0444);
63MODULE_PARM_DESC(fm_port, "FM port # for " CRD_NAME " driver.");
64module_param_array(irq, int, NULL, 0444);
65MODULE_PARM_DESC(irq, "IRQ # for " CRD_NAME " driver.");
66module_param_array(mpu_irq, int, NULL, 0444);
67MODULE_PARM_DESC(mpu_irq, "MPU-401 IRQ # for " CRD_NAME " driver.");
68module_param_array(dma1, int, NULL, 0444);
69MODULE_PARM_DESC(dma1, "Playback DMA # for " CRD_NAME " driver.");
70module_param_array(dma2, int, NULL, 0444);
71MODULE_PARM_DESC(dma2, "Capture DMA # for " CRD_NAME " driver.");
72
73/*
74 * Generic SB DSP support routines
75 */
76
77#define DSP_PORT_RESET 0x6
78#define DSP_PORT_READ 0xa
79#define DSP_PORT_COMMAND 0xc
80#define DSP_PORT_STATUS 0xc
81#define DSP_PORT_DATA_AVAIL 0xe
82
83#define DSP_SIGNATURE 0xaa
84
85#define DSP_COMMAND_GET_VERSION 0xe1
86
87static int dsp_get_byte(void __iomem *port, u8 *val)
88{
89 int loops = 1000;
90
91 while (!(ioread8(port + DSP_PORT_DATA_AVAIL) & 0x80)) {
92 if (!loops--)
93 return -EIO;
94 cpu_relax();
95 }
96 *val = ioread8(port + DSP_PORT_READ);
97 return 0;
98}
99
100static int dsp_reset(void __iomem *port)
101{
102 u8 val;
103
104 iowrite8(1, port + DSP_PORT_RESET);
105 udelay(10);
106 iowrite8(0, port + DSP_PORT_RESET);
107
108 if (dsp_get_byte(port, &val) < 0 || val != DSP_SIGNATURE)
109 return -ENODEV;
110
111 return 0;
112}
113
114static int dsp_command(void __iomem *port, u8 cmd)
115{
116 int loops = 1000;
117
118 while (ioread8(port + DSP_PORT_STATUS) & 0x80) {
119 if (!loops--)
120 return -EIO;
121 cpu_relax();
122 }
123 iowrite8(cmd, port + DSP_PORT_COMMAND);
124 return 0;
125}
126
127static int dsp_get_version(void __iomem *port, u8 *major, u8 *minor)
128{
129 int err;
130
131 err = dsp_command(port, DSP_COMMAND_GET_VERSION);
132 if (err < 0)
133 return err;
134
135 err = dsp_get_byte(port, major);
136 if (err < 0)
137 return err;
138
139 err = dsp_get_byte(port, minor);
140 if (err < 0)
141 return err;
142
143 return 0;
144}
145
146/*
147 * Generic WSS support routines
148 */
149
150#define WSS_CONFIG_DMA_0 (1 << 0)
151#define WSS_CONFIG_DMA_1 (2 << 0)
152#define WSS_CONFIG_DMA_3 (3 << 0)
153#define WSS_CONFIG_DUPLEX (1 << 2)
154#define WSS_CONFIG_IRQ_7 (1 << 3)
155#define WSS_CONFIG_IRQ_9 (2 << 3)
156#define WSS_CONFIG_IRQ_10 (3 << 3)
157#define WSS_CONFIG_IRQ_11 (4 << 3)
158
159#define WSS_PORT_CONFIG 0
160#define WSS_PORT_SIGNATURE 3
161
162#define WSS_SIGNATURE 4
163
164static int wss_detect(void __iomem *wss_port)
165{
166 if ((ioread8(wss_port + WSS_PORT_SIGNATURE) & 0x3f) != WSS_SIGNATURE)
167 return -ENODEV;
168
169 return 0;
170}
171
172static void wss_set_config(void __iomem *wss_port, u8 wss_config)
173{
174 iowrite8(wss_config, wss_port + WSS_PORT_CONFIG);
175}
176
177/*
178 * Aztech Sound Galaxy specifics
179 */
180
181#define GALAXY_PORT_CONFIG 1024
182#define CONFIG_PORT_SET 4
183
184#define DSP_COMMAND_GALAXY_8 8
185#define GALAXY_COMMAND_GET_TYPE 5
186
187#define DSP_COMMAND_GALAXY_9 9
188#define GALAXY_COMMAND_WSSMODE 0
189#define GALAXY_COMMAND_SB8MODE 1
190
191#define GALAXY_MODE_WSS GALAXY_COMMAND_WSSMODE
192#define GALAXY_MODE_SB8 GALAXY_COMMAND_SB8MODE
193
194struct snd_galaxy {
195 void __iomem *port;
196 void __iomem *config_port;
197 void __iomem *wss_port;
198 u32 config;
199 struct resource *res_port;
200 struct resource *res_config_port;
201 struct resource *res_wss_port;
202};
203
204static u32 config[SNDRV_CARDS];
205static u8 wss_config[SNDRV_CARDS];
206
207static int snd_galaxy_match(struct device *dev, unsigned int n)
208{
209 if (!enable[n])
210 return 0;
211
212 switch (port[n]) {
213 case SNDRV_AUTO_PORT:
214 dev_err(dev, "please specify port\n");
215 return 0;
216 case 0x220:
217 config[n] |= GALAXY_CONFIG_SBA_220;
218 break;
219 case 0x240:
220 config[n] |= GALAXY_CONFIG_SBA_240;
221 break;
222 case 0x260:
223 config[n] |= GALAXY_CONFIG_SBA_260;
224 break;
225 case 0x280:
226 config[n] |= GALAXY_CONFIG_SBA_280;
227 break;
228 default:
229 dev_err(dev, "invalid port %#lx\n", port[n]);
230 return 0;
231 }
232
233 switch (wss_port[n]) {
234 case SNDRV_AUTO_PORT:
235 dev_err(dev, "please specify wss_port\n");
236 return 0;
237 case 0x530:
238 config[n] |= GALAXY_CONFIG_WSS_ENABLE | GALAXY_CONFIG_WSSA_530;
239 break;
240 case 0x604:
241 config[n] |= GALAXY_CONFIG_WSS_ENABLE | GALAXY_CONFIG_WSSA_604;
242 break;
243 case 0xe80:
244 config[n] |= GALAXY_CONFIG_WSS_ENABLE | GALAXY_CONFIG_WSSA_E80;
245 break;
246 case 0xf40:
247 config[n] |= GALAXY_CONFIG_WSS_ENABLE | GALAXY_CONFIG_WSSA_F40;
248 break;
249 default:
250 dev_err(dev, "invalid WSS port %#lx\n", wss_port[n]);
251 return 0;
252 }
253
254 switch (irq[n]) {
255 case SNDRV_AUTO_IRQ:
256 dev_err(dev, "please specify irq\n");
257 return 0;
258 case 7:
259 wss_config[n] |= WSS_CONFIG_IRQ_7;
260 break;
261 case 2:
262 irq[n] = 9;
263 case 9:
264 wss_config[n] |= WSS_CONFIG_IRQ_9;
265 break;
266 case 10:
267 wss_config[n] |= WSS_CONFIG_IRQ_10;
268 break;
269 case 11:
270 wss_config[n] |= WSS_CONFIG_IRQ_11;
271 break;
272 default:
273 dev_err(dev, "invalid IRQ %d\n", irq[n]);
274 return 0;
275 }
276
277 switch (dma1[n]) {
278 case SNDRV_AUTO_DMA:
279 dev_err(dev, "please specify dma1\n");
280 return 0;
281 case 0:
282 wss_config[n] |= WSS_CONFIG_DMA_0;
283 break;
284 case 1:
285 wss_config[n] |= WSS_CONFIG_DMA_1;
286 break;
287 case 3:
288 wss_config[n] |= WSS_CONFIG_DMA_3;
289 break;
290 default:
291 dev_err(dev, "invalid playback DMA %d\n", dma1[n]);
292 return 0;
293 }
294
295 if (dma2[n] == SNDRV_AUTO_DMA || dma2[n] == dma1[n]) {
296 dma2[n] = -1;
297 goto mpu;
298 }
299
300 wss_config[n] |= WSS_CONFIG_DUPLEX;
301 switch (dma2[n]) {
302 case 0:
303 break;
304 case 1:
305 if (dma1[n] == 0)
306 break;
307 default:
308 dev_err(dev, "invalid capture DMA %d\n", dma2[n]);
309 return 0;
310 }
311
312mpu:
313 switch (mpu_port[n]) {
314 case SNDRV_AUTO_PORT:
315 dev_warn(dev, "mpu_port not specified; not using MPU-401\n");
316 mpu_port[n] = -1;
317 goto fm;
318 case 0x300:
319 config[n] |= GALAXY_CONFIG_MPU_ENABLE | GALAXY_CONFIG_MPUA_300;
320 break;
321 case 0x330:
322 config[n] |= GALAXY_CONFIG_MPU_ENABLE | GALAXY_CONFIG_MPUA_330;
323 break;
324 default:
325 dev_err(dev, "invalid MPU port %#lx\n", mpu_port[n]);
326 return 0;
327 }
328
329 switch (mpu_irq[n]) {
330 case SNDRV_AUTO_IRQ:
331 dev_warn(dev, "mpu_irq not specified: using polling mode\n");
332 mpu_irq[n] = -1;
333 break;
334 case 2:
335 mpu_irq[n] = 9;
336 case 9:
337 config[n] |= GALAXY_CONFIG_MPUIRQ_2;
338 break;
339#ifdef AZT1605
340 case 3:
341 config[n] |= GALAXY_CONFIG_MPUIRQ_3;
342 break;
343#endif
344 case 5:
345 config[n] |= GALAXY_CONFIG_MPUIRQ_5;
346 break;
347 case 7:
348 config[n] |= GALAXY_CONFIG_MPUIRQ_7;
349 break;
350#ifdef AZT2316
351 case 10:
352 config[n] |= GALAXY_CONFIG_MPUIRQ_10;
353 break;
354#endif
355 default:
356 dev_err(dev, "invalid MPU IRQ %d\n", mpu_irq[n]);
357 return 0;
358 }
359
360 if (mpu_irq[n] == irq[n]) {
361 dev_err(dev, "cannot share IRQ between WSS and MPU-401\n");
362 return 0;
363 }
364
365fm:
366 switch (fm_port[n]) {
367 case SNDRV_AUTO_PORT:
368 dev_warn(dev, "fm_port not specified: not using OPL3\n");
369 fm_port[n] = -1;
370 break;
371 case 0x388:
372 break;
373 default:
374 dev_err(dev, "illegal FM port %#lx\n", fm_port[n]);
375 return 0;
376 }
377
378 config[n] |= GALAXY_CONFIG_GAME_ENABLE;
379 return 1;
380}
381
382static int galaxy_init(struct snd_galaxy *galaxy, u8 *type)
383{
384 u8 major;
385 u8 minor;
386 int err;
387
388 err = dsp_reset(galaxy->port);
389 if (err < 0)
390 return err;
391
392 err = dsp_get_version(galaxy->port, &major, &minor);
393 if (err < 0)
394 return err;
395
396 if (major != GALAXY_DSP_MAJOR || minor != GALAXY_DSP_MINOR)
397 return -ENODEV;
398
399 err = dsp_command(galaxy->port, DSP_COMMAND_GALAXY_8);
400 if (err < 0)
401 return err;
402
403 err = dsp_command(galaxy->port, GALAXY_COMMAND_GET_TYPE);
404 if (err < 0)
405 return err;
406
407 err = dsp_get_byte(galaxy->port, type);
408 if (err < 0)
409 return err;
410
411 return 0;
412}
413
414static int galaxy_set_mode(struct snd_galaxy *galaxy, u8 mode)
415{
416 int err;
417
418 err = dsp_command(galaxy->port, DSP_COMMAND_GALAXY_9);
419 if (err < 0)
420 return err;
421
422 err = dsp_command(galaxy->port, mode);
423 if (err < 0)
424 return err;
425
426#ifdef AZT1605
427 /*
428 * Needed for MPU IRQ on AZT1605, but AZT2316 loses WSS again
429 */
430 err = dsp_reset(galaxy->port);
431 if (err < 0)
432 return err;
433#endif
434
435 return 0;
436}
437
438static void galaxy_set_config(struct snd_galaxy *galaxy, u32 config)
439{
440 u8 tmp = ioread8(galaxy->config_port + CONFIG_PORT_SET);
441 int i;
442
443 iowrite8(tmp | 0x80, galaxy->config_port + CONFIG_PORT_SET);
444 for (i = 0; i < GALAXY_CONFIG_SIZE; i++) {
445 iowrite8(config, galaxy->config_port + i);
446 config >>= 8;
447 }
448 iowrite8(tmp & 0x7f, galaxy->config_port + CONFIG_PORT_SET);
449 msleep(10);
450}
451
452static void galaxy_config(struct snd_galaxy *galaxy, u32 config)
453{
454 int i;
455
456 for (i = GALAXY_CONFIG_SIZE; i; i--) {
457 u8 tmp = ioread8(galaxy->config_port + i - 1);
458 galaxy->config = (galaxy->config << 8) | tmp;
459 }
460 config |= galaxy->config & GALAXY_CONFIG_MASK;
461 galaxy_set_config(galaxy, config);
462}
463
464static int galaxy_wss_config(struct snd_galaxy *galaxy, u8 wss_config)
465{
466 int err;
467
468 err = wss_detect(galaxy->wss_port);
469 if (err < 0)
470 return err;
471
472 wss_set_config(galaxy->wss_port, wss_config);
473
474 err = galaxy_set_mode(galaxy, GALAXY_MODE_WSS);
475 if (err < 0)
476 return err;
477
478 return 0;
479}
480
481static void snd_galaxy_free(struct snd_card *card)
482{
483 struct snd_galaxy *galaxy = card->private_data;
484
485 if (galaxy->wss_port) {
486 wss_set_config(galaxy->wss_port, 0);
487 ioport_unmap(galaxy->wss_port);
488 release_and_free_resource(galaxy->res_wss_port);
489 }
490 if (galaxy->config_port) {
491 galaxy_set_config(galaxy, galaxy->config);
492 ioport_unmap(galaxy->config_port);
493 release_and_free_resource(galaxy->res_config_port);
494 }
495 if (galaxy->port) {
496 ioport_unmap(galaxy->port);
497 release_and_free_resource(galaxy->res_port);
498 }
499}
500
501static int snd_galaxy_probe(struct device *dev, unsigned int n)
502{
503 struct snd_galaxy *galaxy;
504 struct snd_wss *chip;
505 struct snd_card *card;
506 u8 type;
507 int err;
508
509 err = snd_card_new(dev, index[n], id[n], THIS_MODULE,
510 sizeof(*galaxy), &card);
511 if (err < 0)
512 return err;
513
514 card->private_free = snd_galaxy_free;
515 galaxy = card->private_data;
516
517 galaxy->res_port = request_region(port[n], 16, DRV_NAME);
518 if (!galaxy->res_port) {
519 dev_err(dev, "could not grab ports %#lx-%#lx\n", port[n],
520 port[n] + 15);
521 err = -EBUSY;
522 goto error;
523 }
524 galaxy->port = ioport_map(port[n], 16);
525
526 err = galaxy_init(galaxy, &type);
527 if (err < 0) {
528 dev_err(dev, "did not find a Sound Galaxy at %#lx\n", port[n]);
529 goto error;
530 }
531 dev_info(dev, "Sound Galaxy (type %d) found at %#lx\n", type, port[n]);
532
533 galaxy->res_config_port = request_region(port[n] + GALAXY_PORT_CONFIG,
534 16, DRV_NAME);
535 if (!galaxy->res_config_port) {
536 dev_err(dev, "could not grab ports %#lx-%#lx\n",
537 port[n] + GALAXY_PORT_CONFIG,
538 port[n] + GALAXY_PORT_CONFIG + 15);
539 err = -EBUSY;
540 goto error;
541 }
542 galaxy->config_port = ioport_map(port[n] + GALAXY_PORT_CONFIG, 16);
543
544 galaxy_config(galaxy, config[n]);
545
546 galaxy->res_wss_port = request_region(wss_port[n], 4, DRV_NAME);
547 if (!galaxy->res_wss_port) {
548 dev_err(dev, "could not grab ports %#lx-%#lx\n", wss_port[n],
549 wss_port[n] + 3);
550 err = -EBUSY;
551 goto error;
552 }
553 galaxy->wss_port = ioport_map(wss_port[n], 4);
554
555 err = galaxy_wss_config(galaxy, wss_config[n]);
556 if (err < 0) {
557 dev_err(dev, "could not configure WSS\n");
558 goto error;
559 }
560
561 strcpy(card->driver, DRV_NAME);
562 strcpy(card->shortname, DRV_NAME);
563 sprintf(card->longname, "%s at %#lx/%#lx, irq %d, dma %d/%d",
564 card->shortname, port[n], wss_port[n], irq[n], dma1[n],
565 dma2[n]);
566
567 err = snd_wss_create(card, wss_port[n] + 4, -1, irq[n], dma1[n],
568 dma2[n], WSS_HW_DETECT, 0, &chip);
569 if (err < 0)
570 goto error;
571
572 err = snd_wss_pcm(chip, 0, NULL);
573 if (err < 0)
574 goto error;
575
576 err = snd_wss_mixer(chip);
577 if (err < 0)
578 goto error;
579
580 err = snd_wss_timer(chip, 0, NULL);
581 if (err < 0)
582 goto error;
583
584 if (mpu_port[n] >= 0) {
585 err = snd_mpu401_uart_new(card, 0, MPU401_HW_MPU401,
586 mpu_port[n], 0, mpu_irq[n], NULL);
587 if (err < 0)
588 goto error;
589 }
590
591 if (fm_port[n] >= 0) {
592 struct snd_opl3 *opl3;
593
594 err = snd_opl3_create(card, fm_port[n], fm_port[n] + 2,
595 OPL3_HW_AUTO, 0, &opl3);
596 if (err < 0) {
597 dev_err(dev, "no OPL device at %#lx\n", fm_port[n]);
598 goto error;
599 }
600 err = snd_opl3_timer_new(opl3, 1, 2);
601 if (err < 0)
602 goto error;
603
604 err = snd_opl3_hwdep_new(opl3, 0, 1, NULL);
605 if (err < 0)
606 goto error;
607 }
608
609 err = snd_card_register(card);
610 if (err < 0)
611 goto error;
612
613 dev_set_drvdata(dev, card);
614 return 0;
615
616error:
617 snd_card_free(card);
618 return err;
619}
620
621static int snd_galaxy_remove(struct device *dev, unsigned int n)
622{
623 snd_card_free(dev_get_drvdata(dev));
624 return 0;
625}
626
627static struct isa_driver snd_galaxy_driver = {
628 .match = snd_galaxy_match,
629 .probe = snd_galaxy_probe,
630 .remove = snd_galaxy_remove,
631
632 .driver = {
633 .name = DEV_NAME
634 }
635};
636
637static int __init alsa_card_galaxy_init(void)
638{
639 return isa_register_driver(&snd_galaxy_driver, SNDRV_CARDS);
640}
641
642static void __exit alsa_card_galaxy_exit(void)
643{
644 isa_unregister_driver(&snd_galaxy_driver);
645}
646
647module_init(alsa_card_galaxy_init);
648module_exit(alsa_card_galaxy_exit);
1// SPDX-License-Identifier: GPL-2.0-or-later
2/*
3 * Aztech AZT1605/AZT2316 Driver
4 * Copyright (C) 2007,2010 Rene Herman
5 */
6
7#include <linux/kernel.h>
8#include <linux/module.h>
9#include <linux/isa.h>
10#include <linux/delay.h>
11#include <linux/io.h>
12#include <asm/processor.h>
13#include <sound/core.h>
14#include <sound/initval.h>
15#include <sound/wss.h>
16#include <sound/mpu401.h>
17#include <sound/opl3.h>
18
19MODULE_DESCRIPTION(CRD_NAME);
20MODULE_AUTHOR("Rene Herman");
21MODULE_LICENSE("GPL");
22
23static int index[SNDRV_CARDS] = SNDRV_DEFAULT_IDX;
24static char *id[SNDRV_CARDS] = SNDRV_DEFAULT_STR;
25static bool enable[SNDRV_CARDS] = SNDRV_DEFAULT_ENABLE;
26
27module_param_array(index, int, NULL, 0444);
28MODULE_PARM_DESC(index, "Index value for " CRD_NAME " soundcard.");
29module_param_array(id, charp, NULL, 0444);
30MODULE_PARM_DESC(id, "ID string for " CRD_NAME " soundcard.");
31module_param_array(enable, bool, NULL, 0444);
32MODULE_PARM_DESC(enable, "Enable " CRD_NAME " soundcard.");
33
34static long port[SNDRV_CARDS] = SNDRV_DEFAULT_PORT;
35static long wss_port[SNDRV_CARDS] = SNDRV_DEFAULT_PORT;
36static long mpu_port[SNDRV_CARDS] = SNDRV_DEFAULT_PORT;
37static long fm_port[SNDRV_CARDS] = SNDRV_DEFAULT_PORT;
38static int irq[SNDRV_CARDS] = SNDRV_DEFAULT_IRQ;
39static int mpu_irq[SNDRV_CARDS] = SNDRV_DEFAULT_IRQ;
40static int dma1[SNDRV_CARDS] = SNDRV_DEFAULT_DMA;
41static int dma2[SNDRV_CARDS] = SNDRV_DEFAULT_DMA;
42
43module_param_hw_array(port, long, ioport, NULL, 0444);
44MODULE_PARM_DESC(port, "Port # for " CRD_NAME " driver.");
45module_param_hw_array(wss_port, long, ioport, NULL, 0444);
46MODULE_PARM_DESC(wss_port, "WSS port # for " CRD_NAME " driver.");
47module_param_hw_array(mpu_port, long, ioport, NULL, 0444);
48MODULE_PARM_DESC(mpu_port, "MPU-401 port # for " CRD_NAME " driver.");
49module_param_hw_array(fm_port, long, ioport, NULL, 0444);
50MODULE_PARM_DESC(fm_port, "FM port # for " CRD_NAME " driver.");
51module_param_hw_array(irq, int, irq, NULL, 0444);
52MODULE_PARM_DESC(irq, "IRQ # for " CRD_NAME " driver.");
53module_param_hw_array(mpu_irq, int, irq, NULL, 0444);
54MODULE_PARM_DESC(mpu_irq, "MPU-401 IRQ # for " CRD_NAME " driver.");
55module_param_hw_array(dma1, int, dma, NULL, 0444);
56MODULE_PARM_DESC(dma1, "Playback DMA # for " CRD_NAME " driver.");
57module_param_hw_array(dma2, int, dma, NULL, 0444);
58MODULE_PARM_DESC(dma2, "Capture DMA # for " CRD_NAME " driver.");
59
60/*
61 * Generic SB DSP support routines
62 */
63
64#define DSP_PORT_RESET 0x6
65#define DSP_PORT_READ 0xa
66#define DSP_PORT_COMMAND 0xc
67#define DSP_PORT_STATUS 0xc
68#define DSP_PORT_DATA_AVAIL 0xe
69
70#define DSP_SIGNATURE 0xaa
71
72#define DSP_COMMAND_GET_VERSION 0xe1
73
74static int dsp_get_byte(void __iomem *port, u8 *val)
75{
76 int loops = 1000;
77
78 while (!(ioread8(port + DSP_PORT_DATA_AVAIL) & 0x80)) {
79 if (!loops--)
80 return -EIO;
81 cpu_relax();
82 }
83 *val = ioread8(port + DSP_PORT_READ);
84 return 0;
85}
86
87static int dsp_reset(void __iomem *port)
88{
89 u8 val;
90
91 iowrite8(1, port + DSP_PORT_RESET);
92 udelay(10);
93 iowrite8(0, port + DSP_PORT_RESET);
94
95 if (dsp_get_byte(port, &val) < 0 || val != DSP_SIGNATURE)
96 return -ENODEV;
97
98 return 0;
99}
100
101static int dsp_command(void __iomem *port, u8 cmd)
102{
103 int loops = 1000;
104
105 while (ioread8(port + DSP_PORT_STATUS) & 0x80) {
106 if (!loops--)
107 return -EIO;
108 cpu_relax();
109 }
110 iowrite8(cmd, port + DSP_PORT_COMMAND);
111 return 0;
112}
113
114static int dsp_get_version(void __iomem *port, u8 *major, u8 *minor)
115{
116 int err;
117
118 err = dsp_command(port, DSP_COMMAND_GET_VERSION);
119 if (err < 0)
120 return err;
121
122 err = dsp_get_byte(port, major);
123 if (err < 0)
124 return err;
125
126 err = dsp_get_byte(port, minor);
127 if (err < 0)
128 return err;
129
130 return 0;
131}
132
133/*
134 * Generic WSS support routines
135 */
136
137#define WSS_CONFIG_DMA_0 (1 << 0)
138#define WSS_CONFIG_DMA_1 (2 << 0)
139#define WSS_CONFIG_DMA_3 (3 << 0)
140#define WSS_CONFIG_DUPLEX (1 << 2)
141#define WSS_CONFIG_IRQ_7 (1 << 3)
142#define WSS_CONFIG_IRQ_9 (2 << 3)
143#define WSS_CONFIG_IRQ_10 (3 << 3)
144#define WSS_CONFIG_IRQ_11 (4 << 3)
145
146#define WSS_PORT_CONFIG 0
147#define WSS_PORT_SIGNATURE 3
148
149#define WSS_SIGNATURE 4
150
151static int wss_detect(void __iomem *wss_port)
152{
153 if ((ioread8(wss_port + WSS_PORT_SIGNATURE) & 0x3f) != WSS_SIGNATURE)
154 return -ENODEV;
155
156 return 0;
157}
158
159static void wss_set_config(void __iomem *wss_port, u8 wss_config)
160{
161 iowrite8(wss_config, wss_port + WSS_PORT_CONFIG);
162}
163
164/*
165 * Aztech Sound Galaxy specifics
166 */
167
168#define GALAXY_PORT_CONFIG 1024
169#define CONFIG_PORT_SET 4
170
171#define DSP_COMMAND_GALAXY_8 8
172#define GALAXY_COMMAND_GET_TYPE 5
173
174#define DSP_COMMAND_GALAXY_9 9
175#define GALAXY_COMMAND_WSSMODE 0
176#define GALAXY_COMMAND_SB8MODE 1
177
178#define GALAXY_MODE_WSS GALAXY_COMMAND_WSSMODE
179#define GALAXY_MODE_SB8 GALAXY_COMMAND_SB8MODE
180
181struct snd_galaxy {
182 void __iomem *port;
183 void __iomem *config_port;
184 void __iomem *wss_port;
185 u32 config;
186 struct resource *res_port;
187 struct resource *res_config_port;
188 struct resource *res_wss_port;
189};
190
191static u32 config[SNDRV_CARDS];
192static u8 wss_config[SNDRV_CARDS];
193
194static int snd_galaxy_match(struct device *dev, unsigned int n)
195{
196 if (!enable[n])
197 return 0;
198
199 switch (port[n]) {
200 case SNDRV_AUTO_PORT:
201 dev_err(dev, "please specify port\n");
202 return 0;
203 case 0x220:
204 config[n] |= GALAXY_CONFIG_SBA_220;
205 break;
206 case 0x240:
207 config[n] |= GALAXY_CONFIG_SBA_240;
208 break;
209 case 0x260:
210 config[n] |= GALAXY_CONFIG_SBA_260;
211 break;
212 case 0x280:
213 config[n] |= GALAXY_CONFIG_SBA_280;
214 break;
215 default:
216 dev_err(dev, "invalid port %#lx\n", port[n]);
217 return 0;
218 }
219
220 switch (wss_port[n]) {
221 case SNDRV_AUTO_PORT:
222 dev_err(dev, "please specify wss_port\n");
223 return 0;
224 case 0x530:
225 config[n] |= GALAXY_CONFIG_WSS_ENABLE | GALAXY_CONFIG_WSSA_530;
226 break;
227 case 0x604:
228 config[n] |= GALAXY_CONFIG_WSS_ENABLE | GALAXY_CONFIG_WSSA_604;
229 break;
230 case 0xe80:
231 config[n] |= GALAXY_CONFIG_WSS_ENABLE | GALAXY_CONFIG_WSSA_E80;
232 break;
233 case 0xf40:
234 config[n] |= GALAXY_CONFIG_WSS_ENABLE | GALAXY_CONFIG_WSSA_F40;
235 break;
236 default:
237 dev_err(dev, "invalid WSS port %#lx\n", wss_port[n]);
238 return 0;
239 }
240
241 switch (irq[n]) {
242 case SNDRV_AUTO_IRQ:
243 dev_err(dev, "please specify irq\n");
244 return 0;
245 case 7:
246 wss_config[n] |= WSS_CONFIG_IRQ_7;
247 break;
248 case 2:
249 irq[n] = 9;
250 /* Fall through */
251 case 9:
252 wss_config[n] |= WSS_CONFIG_IRQ_9;
253 break;
254 case 10:
255 wss_config[n] |= WSS_CONFIG_IRQ_10;
256 break;
257 case 11:
258 wss_config[n] |= WSS_CONFIG_IRQ_11;
259 break;
260 default:
261 dev_err(dev, "invalid IRQ %d\n", irq[n]);
262 return 0;
263 }
264
265 switch (dma1[n]) {
266 case SNDRV_AUTO_DMA:
267 dev_err(dev, "please specify dma1\n");
268 return 0;
269 case 0:
270 wss_config[n] |= WSS_CONFIG_DMA_0;
271 break;
272 case 1:
273 wss_config[n] |= WSS_CONFIG_DMA_1;
274 break;
275 case 3:
276 wss_config[n] |= WSS_CONFIG_DMA_3;
277 break;
278 default:
279 dev_err(dev, "invalid playback DMA %d\n", dma1[n]);
280 return 0;
281 }
282
283 if (dma2[n] == SNDRV_AUTO_DMA || dma2[n] == dma1[n]) {
284 dma2[n] = -1;
285 goto mpu;
286 }
287
288 wss_config[n] |= WSS_CONFIG_DUPLEX;
289 switch (dma2[n]) {
290 case 0:
291 break;
292 case 1:
293 if (dma1[n] == 0)
294 break;
295 /* Fall through */
296 default:
297 dev_err(dev, "invalid capture DMA %d\n", dma2[n]);
298 return 0;
299 }
300
301mpu:
302 switch (mpu_port[n]) {
303 case SNDRV_AUTO_PORT:
304 dev_warn(dev, "mpu_port not specified; not using MPU-401\n");
305 mpu_port[n] = -1;
306 goto fm;
307 case 0x300:
308 config[n] |= GALAXY_CONFIG_MPU_ENABLE | GALAXY_CONFIG_MPUA_300;
309 break;
310 case 0x330:
311 config[n] |= GALAXY_CONFIG_MPU_ENABLE | GALAXY_CONFIG_MPUA_330;
312 break;
313 default:
314 dev_err(dev, "invalid MPU port %#lx\n", mpu_port[n]);
315 return 0;
316 }
317
318 switch (mpu_irq[n]) {
319 case SNDRV_AUTO_IRQ:
320 dev_warn(dev, "mpu_irq not specified: using polling mode\n");
321 mpu_irq[n] = -1;
322 break;
323 case 2:
324 mpu_irq[n] = 9;
325 /* Fall through */
326 case 9:
327 config[n] |= GALAXY_CONFIG_MPUIRQ_2;
328 break;
329#ifdef AZT1605
330 case 3:
331 config[n] |= GALAXY_CONFIG_MPUIRQ_3;
332 break;
333#endif
334 case 5:
335 config[n] |= GALAXY_CONFIG_MPUIRQ_5;
336 break;
337 case 7:
338 config[n] |= GALAXY_CONFIG_MPUIRQ_7;
339 break;
340#ifdef AZT2316
341 case 10:
342 config[n] |= GALAXY_CONFIG_MPUIRQ_10;
343 break;
344#endif
345 default:
346 dev_err(dev, "invalid MPU IRQ %d\n", mpu_irq[n]);
347 return 0;
348 }
349
350 if (mpu_irq[n] == irq[n]) {
351 dev_err(dev, "cannot share IRQ between WSS and MPU-401\n");
352 return 0;
353 }
354
355fm:
356 switch (fm_port[n]) {
357 case SNDRV_AUTO_PORT:
358 dev_warn(dev, "fm_port not specified: not using OPL3\n");
359 fm_port[n] = -1;
360 break;
361 case 0x388:
362 break;
363 default:
364 dev_err(dev, "illegal FM port %#lx\n", fm_port[n]);
365 return 0;
366 }
367
368 config[n] |= GALAXY_CONFIG_GAME_ENABLE;
369 return 1;
370}
371
372static int galaxy_init(struct snd_galaxy *galaxy, u8 *type)
373{
374 u8 major;
375 u8 minor;
376 int err;
377
378 err = dsp_reset(galaxy->port);
379 if (err < 0)
380 return err;
381
382 err = dsp_get_version(galaxy->port, &major, &minor);
383 if (err < 0)
384 return err;
385
386 if (major != GALAXY_DSP_MAJOR || minor != GALAXY_DSP_MINOR)
387 return -ENODEV;
388
389 err = dsp_command(galaxy->port, DSP_COMMAND_GALAXY_8);
390 if (err < 0)
391 return err;
392
393 err = dsp_command(galaxy->port, GALAXY_COMMAND_GET_TYPE);
394 if (err < 0)
395 return err;
396
397 err = dsp_get_byte(galaxy->port, type);
398 if (err < 0)
399 return err;
400
401 return 0;
402}
403
404static int galaxy_set_mode(struct snd_galaxy *galaxy, u8 mode)
405{
406 int err;
407
408 err = dsp_command(galaxy->port, DSP_COMMAND_GALAXY_9);
409 if (err < 0)
410 return err;
411
412 err = dsp_command(galaxy->port, mode);
413 if (err < 0)
414 return err;
415
416#ifdef AZT1605
417 /*
418 * Needed for MPU IRQ on AZT1605, but AZT2316 loses WSS again
419 */
420 err = dsp_reset(galaxy->port);
421 if (err < 0)
422 return err;
423#endif
424
425 return 0;
426}
427
428static void galaxy_set_config(struct snd_galaxy *galaxy, u32 config)
429{
430 u8 tmp = ioread8(galaxy->config_port + CONFIG_PORT_SET);
431 int i;
432
433 iowrite8(tmp | 0x80, galaxy->config_port + CONFIG_PORT_SET);
434 for (i = 0; i < GALAXY_CONFIG_SIZE; i++) {
435 iowrite8(config, galaxy->config_port + i);
436 config >>= 8;
437 }
438 iowrite8(tmp & 0x7f, galaxy->config_port + CONFIG_PORT_SET);
439 msleep(10);
440}
441
442static void galaxy_config(struct snd_galaxy *galaxy, u32 config)
443{
444 int i;
445
446 for (i = GALAXY_CONFIG_SIZE; i; i--) {
447 u8 tmp = ioread8(galaxy->config_port + i - 1);
448 galaxy->config = (galaxy->config << 8) | tmp;
449 }
450 config |= galaxy->config & GALAXY_CONFIG_MASK;
451 galaxy_set_config(galaxy, config);
452}
453
454static int galaxy_wss_config(struct snd_galaxy *galaxy, u8 wss_config)
455{
456 int err;
457
458 err = wss_detect(galaxy->wss_port);
459 if (err < 0)
460 return err;
461
462 wss_set_config(galaxy->wss_port, wss_config);
463
464 err = galaxy_set_mode(galaxy, GALAXY_MODE_WSS);
465 if (err < 0)
466 return err;
467
468 return 0;
469}
470
471static void snd_galaxy_free(struct snd_card *card)
472{
473 struct snd_galaxy *galaxy = card->private_data;
474
475 if (galaxy->wss_port) {
476 wss_set_config(galaxy->wss_port, 0);
477 ioport_unmap(galaxy->wss_port);
478 release_and_free_resource(galaxy->res_wss_port);
479 }
480 if (galaxy->config_port) {
481 galaxy_set_config(galaxy, galaxy->config);
482 ioport_unmap(galaxy->config_port);
483 release_and_free_resource(galaxy->res_config_port);
484 }
485 if (galaxy->port) {
486 ioport_unmap(galaxy->port);
487 release_and_free_resource(galaxy->res_port);
488 }
489}
490
491static int snd_galaxy_probe(struct device *dev, unsigned int n)
492{
493 struct snd_galaxy *galaxy;
494 struct snd_wss *chip;
495 struct snd_card *card;
496 u8 type;
497 int err;
498
499 err = snd_card_new(dev, index[n], id[n], THIS_MODULE,
500 sizeof(*galaxy), &card);
501 if (err < 0)
502 return err;
503
504 card->private_free = snd_galaxy_free;
505 galaxy = card->private_data;
506
507 galaxy->res_port = request_region(port[n], 16, DRV_NAME);
508 if (!galaxy->res_port) {
509 dev_err(dev, "could not grab ports %#lx-%#lx\n", port[n],
510 port[n] + 15);
511 err = -EBUSY;
512 goto error;
513 }
514 galaxy->port = ioport_map(port[n], 16);
515
516 err = galaxy_init(galaxy, &type);
517 if (err < 0) {
518 dev_err(dev, "did not find a Sound Galaxy at %#lx\n", port[n]);
519 goto error;
520 }
521 dev_info(dev, "Sound Galaxy (type %d) found at %#lx\n", type, port[n]);
522
523 galaxy->res_config_port = request_region(port[n] + GALAXY_PORT_CONFIG,
524 16, DRV_NAME);
525 if (!galaxy->res_config_port) {
526 dev_err(dev, "could not grab ports %#lx-%#lx\n",
527 port[n] + GALAXY_PORT_CONFIG,
528 port[n] + GALAXY_PORT_CONFIG + 15);
529 err = -EBUSY;
530 goto error;
531 }
532 galaxy->config_port = ioport_map(port[n] + GALAXY_PORT_CONFIG, 16);
533
534 galaxy_config(galaxy, config[n]);
535
536 galaxy->res_wss_port = request_region(wss_port[n], 4, DRV_NAME);
537 if (!galaxy->res_wss_port) {
538 dev_err(dev, "could not grab ports %#lx-%#lx\n", wss_port[n],
539 wss_port[n] + 3);
540 err = -EBUSY;
541 goto error;
542 }
543 galaxy->wss_port = ioport_map(wss_port[n], 4);
544
545 err = galaxy_wss_config(galaxy, wss_config[n]);
546 if (err < 0) {
547 dev_err(dev, "could not configure WSS\n");
548 goto error;
549 }
550
551 strcpy(card->driver, DRV_NAME);
552 strcpy(card->shortname, DRV_NAME);
553 sprintf(card->longname, "%s at %#lx/%#lx, irq %d, dma %d/%d",
554 card->shortname, port[n], wss_port[n], irq[n], dma1[n],
555 dma2[n]);
556
557 err = snd_wss_create(card, wss_port[n] + 4, -1, irq[n], dma1[n],
558 dma2[n], WSS_HW_DETECT, 0, &chip);
559 if (err < 0)
560 goto error;
561
562 err = snd_wss_pcm(chip, 0);
563 if (err < 0)
564 goto error;
565
566 err = snd_wss_mixer(chip);
567 if (err < 0)
568 goto error;
569
570 err = snd_wss_timer(chip, 0);
571 if (err < 0)
572 goto error;
573
574 if (mpu_port[n] >= 0) {
575 err = snd_mpu401_uart_new(card, 0, MPU401_HW_MPU401,
576 mpu_port[n], 0, mpu_irq[n], NULL);
577 if (err < 0)
578 goto error;
579 }
580
581 if (fm_port[n] >= 0) {
582 struct snd_opl3 *opl3;
583
584 err = snd_opl3_create(card, fm_port[n], fm_port[n] + 2,
585 OPL3_HW_AUTO, 0, &opl3);
586 if (err < 0) {
587 dev_err(dev, "no OPL device at %#lx\n", fm_port[n]);
588 goto error;
589 }
590 err = snd_opl3_timer_new(opl3, 1, 2);
591 if (err < 0)
592 goto error;
593
594 err = snd_opl3_hwdep_new(opl3, 0, 1, NULL);
595 if (err < 0)
596 goto error;
597 }
598
599 err = snd_card_register(card);
600 if (err < 0)
601 goto error;
602
603 dev_set_drvdata(dev, card);
604 return 0;
605
606error:
607 snd_card_free(card);
608 return err;
609}
610
611static int snd_galaxy_remove(struct device *dev, unsigned int n)
612{
613 snd_card_free(dev_get_drvdata(dev));
614 return 0;
615}
616
617static struct isa_driver snd_galaxy_driver = {
618 .match = snd_galaxy_match,
619 .probe = snd_galaxy_probe,
620 .remove = snd_galaxy_remove,
621
622 .driver = {
623 .name = DEV_NAME
624 }
625};
626
627module_isa_driver(snd_galaxy_driver, SNDRV_CARDS);