• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 /*
2  *  Copyright (c) by Jaroslav Kysela <perex@perex.cz>
3  *     and (c) 1999 Steve Ratcliffe <steve@parabola.demon.co.uk>
4  *  Copyright (C) 1999-2000 Takashi Iwai <tiwai@suse.de>
5  *
6  *  Routines for control of EMU8000 chip
7  *
8  *   This program is free software; you can redistribute it and/or modify
9  *   it under the terms of the GNU General Public License as published by
10  *   the Free Software Foundation; either version 2 of the License, or
11  *   (at your option) any later version.
12  *
13  *   This program is distributed in the hope that it will be useful,
14  *   but WITHOUT ANY WARRANTY; without even the implied warranty of
15  *   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
16  *   GNU General Public License for more details.
17  *
18  *   You should have received a copy of the GNU General Public License
19  *   along with this program; if not, write to the Free Software
20  *   Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307 USA
21  */
22 
23 #include <linux/wait.h>
24 #include <linux/sched.h>
25 #include <linux/slab.h>
26 #include <linux/ioport.h>
27 #include <linux/export.h>
28 #include <linux/delay.h>
29 #include <sound/core.h>
30 #include <sound/emu8000.h>
31 #include <sound/emu8000_reg.h>
32 #include <asm/io.h>
33 #include <asm/uaccess.h>
34 #include <linux/init.h>
35 #include <sound/control.h>
36 #include <sound/initval.h>
37 
38 /*
39  * emu8000 register controls
40  */
41 
42 /*
43  * The following routines read and write registers on the emu8000.  They
44  * should always be called via the EMU8000*READ/WRITE macros and never
45  * directly.  The macros handle the port number and command word.
46  */
47 /* Write a word */
snd_emu8000_poke(struct snd_emu8000 * emu,unsigned int port,unsigned int reg,unsigned int val)48 void snd_emu8000_poke(struct snd_emu8000 *emu, unsigned int port, unsigned int reg, unsigned int val)
49 {
50 	unsigned long flags;
51 	spin_lock_irqsave(&emu->reg_lock, flags);
52 	if (reg != emu->last_reg) {
53 		outw((unsigned short)reg, EMU8000_PTR(emu)); /* Set register */
54 		emu->last_reg = reg;
55 	}
56 	outw((unsigned short)val, port); /* Send data */
57 	spin_unlock_irqrestore(&emu->reg_lock, flags);
58 }
59 
60 /* Read a word */
snd_emu8000_peek(struct snd_emu8000 * emu,unsigned int port,unsigned int reg)61 unsigned short snd_emu8000_peek(struct snd_emu8000 *emu, unsigned int port, unsigned int reg)
62 {
63 	unsigned short res;
64 	unsigned long flags;
65 	spin_lock_irqsave(&emu->reg_lock, flags);
66 	if (reg != emu->last_reg) {
67 		outw((unsigned short)reg, EMU8000_PTR(emu)); /* Set register */
68 		emu->last_reg = reg;
69 	}
70 	res = inw(port);	/* Read data */
71 	spin_unlock_irqrestore(&emu->reg_lock, flags);
72 	return res;
73 }
74 
75 /* Write a double word */
snd_emu8000_poke_dw(struct snd_emu8000 * emu,unsigned int port,unsigned int reg,unsigned int val)76 void snd_emu8000_poke_dw(struct snd_emu8000 *emu, unsigned int port, unsigned int reg, unsigned int val)
77 {
78 	unsigned long flags;
79 	spin_lock_irqsave(&emu->reg_lock, flags);
80 	if (reg != emu->last_reg) {
81 		outw((unsigned short)reg, EMU8000_PTR(emu)); /* Set register */
82 		emu->last_reg = reg;
83 	}
84 	outw((unsigned short)val, port); /* Send low word of data */
85 	outw((unsigned short)(val>>16), port+2); /* Send high word of data */
86 	spin_unlock_irqrestore(&emu->reg_lock, flags);
87 }
88 
89 /* Read a double word */
snd_emu8000_peek_dw(struct snd_emu8000 * emu,unsigned int port,unsigned int reg)90 unsigned int snd_emu8000_peek_dw(struct snd_emu8000 *emu, unsigned int port, unsigned int reg)
91 {
92 	unsigned short low;
93 	unsigned int res;
94 	unsigned long flags;
95 	spin_lock_irqsave(&emu->reg_lock, flags);
96 	if (reg != emu->last_reg) {
97 		outw((unsigned short)reg, EMU8000_PTR(emu)); /* Set register */
98 		emu->last_reg = reg;
99 	}
100 	low = inw(port);	/* Read low word of data */
101 	res = low + (inw(port+2) << 16);
102 	spin_unlock_irqrestore(&emu->reg_lock, flags);
103 	return res;
104 }
105 
106 /*
107  * Set up / close a channel to be used for DMA.
108  */
109 /*exported*/ void
snd_emu8000_dma_chan(struct snd_emu8000 * emu,int ch,int mode)110 snd_emu8000_dma_chan(struct snd_emu8000 *emu, int ch, int mode)
111 {
112 	unsigned right_bit = (mode & EMU8000_RAM_RIGHT) ? 0x01000000 : 0;
113 	mode &= EMU8000_RAM_MODE_MASK;
114 	if (mode == EMU8000_RAM_CLOSE) {
115 		EMU8000_CCCA_WRITE(emu, ch, 0);
116 		EMU8000_DCYSUSV_WRITE(emu, ch, 0x807F);
117 		return;
118 	}
119 	EMU8000_DCYSUSV_WRITE(emu, ch, 0x80);
120 	EMU8000_VTFT_WRITE(emu, ch, 0);
121 	EMU8000_CVCF_WRITE(emu, ch, 0);
122 	EMU8000_PTRX_WRITE(emu, ch, 0x40000000);
123 	EMU8000_CPF_WRITE(emu, ch, 0x40000000);
124 	EMU8000_PSST_WRITE(emu, ch, 0);
125 	EMU8000_CSL_WRITE(emu, ch, 0);
126 	if (mode == EMU8000_RAM_WRITE) /* DMA write */
127 		EMU8000_CCCA_WRITE(emu, ch, 0x06000000 | right_bit);
128 	else	   /* DMA read */
129 		EMU8000_CCCA_WRITE(emu, ch, 0x04000000 | right_bit);
130 }
131 
132 /*
133  */
134 static void
snd_emu8000_read_wait(struct snd_emu8000 * emu)135 snd_emu8000_read_wait(struct snd_emu8000 *emu)
136 {
137 	while ((EMU8000_SMALR_READ(emu) & 0x80000000) != 0) {
138 		schedule_timeout_interruptible(1);
139 		if (signal_pending(current))
140 			break;
141 	}
142 }
143 
144 /*
145  */
146 static void
snd_emu8000_write_wait(struct snd_emu8000 * emu)147 snd_emu8000_write_wait(struct snd_emu8000 *emu)
148 {
149 	while ((EMU8000_SMALW_READ(emu) & 0x80000000) != 0) {
150 		schedule_timeout_interruptible(1);
151 		if (signal_pending(current))
152 			break;
153 	}
154 }
155 
156 /*
157  * detect a card at the given port
158  */
159 static int
snd_emu8000_detect(struct snd_emu8000 * emu)160 snd_emu8000_detect(struct snd_emu8000 *emu)
161 {
162 	/* Initialise */
163 	EMU8000_HWCF1_WRITE(emu, 0x0059);
164 	EMU8000_HWCF2_WRITE(emu, 0x0020);
165 	EMU8000_HWCF3_WRITE(emu, 0x0000);
166 	/* Check for a recognisable emu8000 */
167 	/*
168 	if ((EMU8000_U1_READ(emu) & 0x000f) != 0x000c)
169 		return -ENODEV;
170 		*/
171 	if ((EMU8000_HWCF1_READ(emu) & 0x007e) != 0x0058)
172 		return -ENODEV;
173 	if ((EMU8000_HWCF2_READ(emu) & 0x0003) != 0x0003)
174 		return -ENODEV;
175 
176 	snd_printdd("EMU8000 [0x%lx]: Synth chip found\n",
177                     emu->port1);
178 	return 0;
179 }
180 
181 
182 /*
183  * intiailize audio channels
184  */
185 static void
init_audio(struct snd_emu8000 * emu)186 init_audio(struct snd_emu8000 *emu)
187 {
188 	int ch;
189 
190 	/* turn off envelope engines */
191 	for (ch = 0; ch < EMU8000_CHANNELS; ch++)
192 		EMU8000_DCYSUSV_WRITE(emu, ch, 0x80);
193 
194 	/* reset all other parameters to zero */
195 	for (ch = 0; ch < EMU8000_CHANNELS; ch++) {
196 		EMU8000_ENVVOL_WRITE(emu, ch, 0);
197 		EMU8000_ENVVAL_WRITE(emu, ch, 0);
198 		EMU8000_DCYSUS_WRITE(emu, ch, 0);
199 		EMU8000_ATKHLDV_WRITE(emu, ch, 0);
200 		EMU8000_LFO1VAL_WRITE(emu, ch, 0);
201 		EMU8000_ATKHLD_WRITE(emu, ch, 0);
202 		EMU8000_LFO2VAL_WRITE(emu, ch, 0);
203 		EMU8000_IP_WRITE(emu, ch, 0);
204 		EMU8000_IFATN_WRITE(emu, ch, 0);
205 		EMU8000_PEFE_WRITE(emu, ch, 0);
206 		EMU8000_FMMOD_WRITE(emu, ch, 0);
207 		EMU8000_TREMFRQ_WRITE(emu, ch, 0);
208 		EMU8000_FM2FRQ2_WRITE(emu, ch, 0);
209 		EMU8000_PTRX_WRITE(emu, ch, 0);
210 		EMU8000_VTFT_WRITE(emu, ch, 0);
211 		EMU8000_PSST_WRITE(emu, ch, 0);
212 		EMU8000_CSL_WRITE(emu, ch, 0);
213 		EMU8000_CCCA_WRITE(emu, ch, 0);
214 	}
215 
216 	for (ch = 0; ch < EMU8000_CHANNELS; ch++) {
217 		EMU8000_CPF_WRITE(emu, ch, 0);
218 		EMU8000_CVCF_WRITE(emu, ch, 0);
219 	}
220 }
221 
222 
223 /*
224  * initialize DMA address
225  */
226 static void
init_dma(struct snd_emu8000 * emu)227 init_dma(struct snd_emu8000 *emu)
228 {
229 	EMU8000_SMALR_WRITE(emu, 0);
230 	EMU8000_SMARR_WRITE(emu, 0);
231 	EMU8000_SMALW_WRITE(emu, 0);
232 	EMU8000_SMARW_WRITE(emu, 0);
233 }
234 
235 /*
236  * initialization arrays; from ADIP
237  */
238 static unsigned short init1[128] = {
239 	0x03ff, 0x0030,  0x07ff, 0x0130, 0x0bff, 0x0230,  0x0fff, 0x0330,
240 	0x13ff, 0x0430,  0x17ff, 0x0530, 0x1bff, 0x0630,  0x1fff, 0x0730,
241 	0x23ff, 0x0830,  0x27ff, 0x0930, 0x2bff, 0x0a30,  0x2fff, 0x0b30,
242 	0x33ff, 0x0c30,  0x37ff, 0x0d30, 0x3bff, 0x0e30,  0x3fff, 0x0f30,
243 
244 	0x43ff, 0x0030,  0x47ff, 0x0130, 0x4bff, 0x0230,  0x4fff, 0x0330,
245 	0x53ff, 0x0430,  0x57ff, 0x0530, 0x5bff, 0x0630,  0x5fff, 0x0730,
246 	0x63ff, 0x0830,  0x67ff, 0x0930, 0x6bff, 0x0a30,  0x6fff, 0x0b30,
247 	0x73ff, 0x0c30,  0x77ff, 0x0d30, 0x7bff, 0x0e30,  0x7fff, 0x0f30,
248 
249 	0x83ff, 0x0030,  0x87ff, 0x0130, 0x8bff, 0x0230,  0x8fff, 0x0330,
250 	0x93ff, 0x0430,  0x97ff, 0x0530, 0x9bff, 0x0630,  0x9fff, 0x0730,
251 	0xa3ff, 0x0830,  0xa7ff, 0x0930, 0xabff, 0x0a30,  0xafff, 0x0b30,
252 	0xb3ff, 0x0c30,  0xb7ff, 0x0d30, 0xbbff, 0x0e30,  0xbfff, 0x0f30,
253 
254 	0xc3ff, 0x0030,  0xc7ff, 0x0130, 0xcbff, 0x0230,  0xcfff, 0x0330,
255 	0xd3ff, 0x0430,  0xd7ff, 0x0530, 0xdbff, 0x0630,  0xdfff, 0x0730,
256 	0xe3ff, 0x0830,  0xe7ff, 0x0930, 0xebff, 0x0a30,  0xefff, 0x0b30,
257 	0xf3ff, 0x0c30,  0xf7ff, 0x0d30, 0xfbff, 0x0e30,  0xffff, 0x0f30,
258 };
259 
260 static unsigned short init2[128] = {
261 	0x03ff, 0x8030, 0x07ff, 0x8130, 0x0bff, 0x8230, 0x0fff, 0x8330,
262 	0x13ff, 0x8430, 0x17ff, 0x8530, 0x1bff, 0x8630, 0x1fff, 0x8730,
263 	0x23ff, 0x8830, 0x27ff, 0x8930, 0x2bff, 0x8a30, 0x2fff, 0x8b30,
264 	0x33ff, 0x8c30, 0x37ff, 0x8d30, 0x3bff, 0x8e30, 0x3fff, 0x8f30,
265 
266 	0x43ff, 0x8030, 0x47ff, 0x8130, 0x4bff, 0x8230, 0x4fff, 0x8330,
267 	0x53ff, 0x8430, 0x57ff, 0x8530, 0x5bff, 0x8630, 0x5fff, 0x8730,
268 	0x63ff, 0x8830, 0x67ff, 0x8930, 0x6bff, 0x8a30, 0x6fff, 0x8b30,
269 	0x73ff, 0x8c30, 0x77ff, 0x8d30, 0x7bff, 0x8e30, 0x7fff, 0x8f30,
270 
271 	0x83ff, 0x8030, 0x87ff, 0x8130, 0x8bff, 0x8230, 0x8fff, 0x8330,
272 	0x93ff, 0x8430, 0x97ff, 0x8530, 0x9bff, 0x8630, 0x9fff, 0x8730,
273 	0xa3ff, 0x8830, 0xa7ff, 0x8930, 0xabff, 0x8a30, 0xafff, 0x8b30,
274 	0xb3ff, 0x8c30, 0xb7ff, 0x8d30, 0xbbff, 0x8e30, 0xbfff, 0x8f30,
275 
276 	0xc3ff, 0x8030, 0xc7ff, 0x8130, 0xcbff, 0x8230, 0xcfff, 0x8330,
277 	0xd3ff, 0x8430, 0xd7ff, 0x8530, 0xdbff, 0x8630, 0xdfff, 0x8730,
278 	0xe3ff, 0x8830, 0xe7ff, 0x8930, 0xebff, 0x8a30, 0xefff, 0x8b30,
279 	0xf3ff, 0x8c30, 0xf7ff, 0x8d30, 0xfbff, 0x8e30, 0xffff, 0x8f30,
280 };
281 
282 static unsigned short init3[128] = {
283 	0x0C10, 0x8470, 0x14FE, 0xB488, 0x167F, 0xA470, 0x18E7, 0x84B5,
284 	0x1B6E, 0x842A, 0x1F1D, 0x852A, 0x0DA3, 0x8F7C, 0x167E, 0xF254,
285 	0x0000, 0x842A, 0x0001, 0x852A, 0x18E6, 0x8BAA, 0x1B6D, 0xF234,
286 	0x229F, 0x8429, 0x2746, 0x8529, 0x1F1C, 0x86E7, 0x229E, 0xF224,
287 
288 	0x0DA4, 0x8429, 0x2C29, 0x8529, 0x2745, 0x87F6, 0x2C28, 0xF254,
289 	0x383B, 0x8428, 0x320F, 0x8528, 0x320E, 0x8F02, 0x1341, 0xF264,
290 	0x3EB6, 0x8428, 0x3EB9, 0x8528, 0x383A, 0x8FA9, 0x3EB5, 0xF294,
291 	0x3EB7, 0x8474, 0x3EBA, 0x8575, 0x3EB8, 0xC4C3, 0x3EBB, 0xC5C3,
292 
293 	0x0000, 0xA404, 0x0001, 0xA504, 0x141F, 0x8671, 0x14FD, 0x8287,
294 	0x3EBC, 0xE610, 0x3EC8, 0x8C7B, 0x031A, 0x87E6, 0x3EC8, 0x86F7,
295 	0x3EC0, 0x821E, 0x3EBE, 0xD208, 0x3EBD, 0x821F, 0x3ECA, 0x8386,
296 	0x3EC1, 0x8C03, 0x3EC9, 0x831E, 0x3ECA, 0x8C4C, 0x3EBF, 0x8C55,
297 
298 	0x3EC9, 0xC208, 0x3EC4, 0xBC84, 0x3EC8, 0x8EAD, 0x3EC8, 0xD308,
299 	0x3EC2, 0x8F7E, 0x3ECB, 0x8219, 0x3ECB, 0xD26E, 0x3EC5, 0x831F,
300 	0x3EC6, 0xC308, 0x3EC3, 0xB2FF, 0x3EC9, 0x8265, 0x3EC9, 0x8319,
301 	0x1342, 0xD36E, 0x3EC7, 0xB3FF, 0x0000, 0x8365, 0x1420, 0x9570,
302 };
303 
304 static unsigned short init4[128] = {
305 	0x0C10, 0x8470, 0x14FE, 0xB488, 0x167F, 0xA470, 0x18E7, 0x84B5,
306 	0x1B6E, 0x842A, 0x1F1D, 0x852A, 0x0DA3, 0x0F7C, 0x167E, 0x7254,
307 	0x0000, 0x842A, 0x0001, 0x852A, 0x18E6, 0x0BAA, 0x1B6D, 0x7234,
308 	0x229F, 0x8429, 0x2746, 0x8529, 0x1F1C, 0x06E7, 0x229E, 0x7224,
309 
310 	0x0DA4, 0x8429, 0x2C29, 0x8529, 0x2745, 0x07F6, 0x2C28, 0x7254,
311 	0x383B, 0x8428, 0x320F, 0x8528, 0x320E, 0x0F02, 0x1341, 0x7264,
312 	0x3EB6, 0x8428, 0x3EB9, 0x8528, 0x383A, 0x0FA9, 0x3EB5, 0x7294,
313 	0x3EB7, 0x8474, 0x3EBA, 0x8575, 0x3EB8, 0x44C3, 0x3EBB, 0x45C3,
314 
315 	0x0000, 0xA404, 0x0001, 0xA504, 0x141F, 0x0671, 0x14FD, 0x0287,
316 	0x3EBC, 0xE610, 0x3EC8, 0x0C7B, 0x031A, 0x07E6, 0x3EC8, 0x86F7,
317 	0x3EC0, 0x821E, 0x3EBE, 0xD208, 0x3EBD, 0x021F, 0x3ECA, 0x0386,
318 	0x3EC1, 0x0C03, 0x3EC9, 0x031E, 0x3ECA, 0x8C4C, 0x3EBF, 0x0C55,
319 
320 	0x3EC9, 0xC208, 0x3EC4, 0xBC84, 0x3EC8, 0x0EAD, 0x3EC8, 0xD308,
321 	0x3EC2, 0x8F7E, 0x3ECB, 0x0219, 0x3ECB, 0xD26E, 0x3EC5, 0x031F,
322 	0x3EC6, 0xC308, 0x3EC3, 0x32FF, 0x3EC9, 0x0265, 0x3EC9, 0x8319,
323 	0x1342, 0xD36E, 0x3EC7, 0x33FF, 0x0000, 0x8365, 0x1420, 0x9570,
324 };
325 
326 /* send an initialization array
327  * Taken from the oss driver, not obvious from the doc how this
328  * is meant to work
329  */
330 static void
send_array(struct snd_emu8000 * emu,unsigned short * data,int size)331 send_array(struct snd_emu8000 *emu, unsigned short *data, int size)
332 {
333 	int i;
334 	unsigned short *p;
335 
336 	p = data;
337 	for (i = 0; i < size; i++, p++)
338 		EMU8000_INIT1_WRITE(emu, i, *p);
339 	for (i = 0; i < size; i++, p++)
340 		EMU8000_INIT2_WRITE(emu, i, *p);
341 	for (i = 0; i < size; i++, p++)
342 		EMU8000_INIT3_WRITE(emu, i, *p);
343 	for (i = 0; i < size; i++, p++)
344 		EMU8000_INIT4_WRITE(emu, i, *p);
345 }
346 
347 
348 /*
349  * Send initialization arrays to start up, this just follows the
350  * initialisation sequence in the adip.
351  */
352 static void
init_arrays(struct snd_emu8000 * emu)353 init_arrays(struct snd_emu8000 *emu)
354 {
355 	send_array(emu, init1, ARRAY_SIZE(init1)/4);
356 
357 	msleep((1024 * 1000) / 44100); /* wait for 1024 clocks */
358 	send_array(emu, init2, ARRAY_SIZE(init2)/4);
359 	send_array(emu, init3, ARRAY_SIZE(init3)/4);
360 
361 	EMU8000_HWCF4_WRITE(emu, 0);
362 	EMU8000_HWCF5_WRITE(emu, 0x83);
363 	EMU8000_HWCF6_WRITE(emu, 0x8000);
364 
365 	send_array(emu, init4, ARRAY_SIZE(init4)/4);
366 }
367 
368 
369 #define UNIQUE_ID1	0xa5b9
370 #define UNIQUE_ID2	0x9d53
371 
372 /*
373  * Size the onboard memory.
374  * This is written so as not to need arbitrary delays after the write. It
375  * seems that the only way to do this is to use the one channel and keep
376  * reallocating between read and write.
377  */
378 static void
size_dram(struct snd_emu8000 * emu)379 size_dram(struct snd_emu8000 *emu)
380 {
381 	int i, size, detected_size;
382 
383 	if (emu->dram_checked)
384 		return;
385 
386 	size = 0;
387 	detected_size = 0;
388 
389 	/* write out a magic number */
390 	snd_emu8000_dma_chan(emu, 0, EMU8000_RAM_WRITE);
391 	snd_emu8000_dma_chan(emu, 1, EMU8000_RAM_READ);
392 	EMU8000_SMALW_WRITE(emu, EMU8000_DRAM_OFFSET);
393 	EMU8000_SMLD_WRITE(emu, UNIQUE_ID1);
394 	snd_emu8000_init_fm(emu); /* This must really be here and not 2 lines back even */
395 
396 	while (size < EMU8000_MAX_DRAM) {
397 
398 		size += 512 * 1024;  /* increment 512kbytes */
399 
400 		/* Write a unique data on the test address.
401 		 * if the address is out of range, the data is written on
402 		 * 0x200000(=EMU8000_DRAM_OFFSET).  Then the id word is
403 		 * changed by this data.
404 		 */
405 		/*snd_emu8000_dma_chan(emu, 0, EMU8000_RAM_WRITE);*/
406 		EMU8000_SMALW_WRITE(emu, EMU8000_DRAM_OFFSET + (size>>1));
407 		EMU8000_SMLD_WRITE(emu, UNIQUE_ID2);
408 		snd_emu8000_write_wait(emu);
409 
410 		/*
411 		 * read the data on the just written DRAM address
412 		 * if not the same then we have reached the end of ram.
413 		 */
414 		/*snd_emu8000_dma_chan(emu, 0, EMU8000_RAM_READ);*/
415 		EMU8000_SMALR_WRITE(emu, EMU8000_DRAM_OFFSET + (size>>1));
416 		/*snd_emu8000_read_wait(emu);*/
417 		EMU8000_SMLD_READ(emu); /* discard stale data  */
418 		if (EMU8000_SMLD_READ(emu) != UNIQUE_ID2)
419 			break; /* no memory at this address */
420 		snd_emu8000_read_wait(emu);
421 
422 		/*
423 		 * If it is the same it could be that the address just
424 		 * wraps back to the beginning; so check to see if the
425 		 * initial value has been overwritten.
426 		 */
427 		EMU8000_SMALR_WRITE(emu, EMU8000_DRAM_OFFSET);
428 		EMU8000_SMLD_READ(emu); /* discard stale data  */
429 		if (EMU8000_SMLD_READ(emu) != UNIQUE_ID1)
430 			break; /* we must have wrapped around */
431 		snd_emu8000_read_wait(emu);
432 
433 		/* Otherwise, it's valid memory. */
434 		detected_size = size + 512 * 1024;
435 	}
436 
437 	/* Distinguish 512 KiB from 0. */
438 	if (detected_size == 0) {
439 		snd_emu8000_read_wait(emu);
440 		EMU8000_SMALR_WRITE(emu, EMU8000_DRAM_OFFSET);
441 		EMU8000_SMLD_READ(emu); /* discard stale data  */
442 		if (EMU8000_SMLD_READ(emu) == UNIQUE_ID1)
443 			detected_size = 512 * 1024;
444 	}
445 
446 	/* wait until FULL bit in SMAxW register is false */
447 	for (i = 0; i < 10000; i++) {
448 		if ((EMU8000_SMALW_READ(emu) & 0x80000000) == 0)
449 			break;
450 		schedule_timeout_interruptible(1);
451 		if (signal_pending(current))
452 			break;
453 	}
454 	snd_emu8000_dma_chan(emu, 0, EMU8000_RAM_CLOSE);
455 	snd_emu8000_dma_chan(emu, 1, EMU8000_RAM_CLOSE);
456 
457 	snd_printdd("EMU8000 [0x%lx]: %d Kb on-board memory detected\n",
458 		    emu->port1, detected_size/1024);
459 
460 	emu->mem_size = detected_size;
461 	emu->dram_checked = 1;
462 }
463 
464 
465 /*
466  * Initiailise the FM section.  You have to do this to use sample RAM
467  * and therefore lose 2 voices.
468  */
469 /*exported*/ void
snd_emu8000_init_fm(struct snd_emu8000 * emu)470 snd_emu8000_init_fm(struct snd_emu8000 *emu)
471 {
472 	unsigned long flags;
473 
474 	/* Initialize the last two channels for DRAM refresh and producing
475 	   the reverb and chorus effects for Yamaha OPL-3 synthesizer */
476 
477 	/* 31: FM left channel, 0xffffe0-0xffffe8 */
478 	EMU8000_DCYSUSV_WRITE(emu, 30, 0x80);
479 	EMU8000_PSST_WRITE(emu, 30, 0xFFFFFFE0); /* full left */
480 	EMU8000_CSL_WRITE(emu, 30, 0x00FFFFE8 | (emu->fm_chorus_depth << 24));
481 	EMU8000_PTRX_WRITE(emu, 30, (emu->fm_reverb_depth << 8));
482 	EMU8000_CPF_WRITE(emu, 30, 0);
483 	EMU8000_CCCA_WRITE(emu, 30, 0x00FFFFE3);
484 
485 	/* 32: FM right channel, 0xfffff0-0xfffff8 */
486 	EMU8000_DCYSUSV_WRITE(emu, 31, 0x80);
487 	EMU8000_PSST_WRITE(emu, 31, 0x00FFFFF0); /* full right */
488 	EMU8000_CSL_WRITE(emu, 31, 0x00FFFFF8 | (emu->fm_chorus_depth << 24));
489 	EMU8000_PTRX_WRITE(emu, 31, (emu->fm_reverb_depth << 8));
490 	EMU8000_CPF_WRITE(emu, 31, 0x8000);
491 	EMU8000_CCCA_WRITE(emu, 31, 0x00FFFFF3);
492 
493 	snd_emu8000_poke((emu), EMU8000_DATA0(emu), EMU8000_CMD(1, (30)), 0);
494 
495 	spin_lock_irqsave(&emu->reg_lock, flags);
496 	while (!(inw(EMU8000_PTR(emu)) & 0x1000))
497 		;
498 	while ((inw(EMU8000_PTR(emu)) & 0x1000))
499 		;
500 	spin_unlock_irqrestore(&emu->reg_lock, flags);
501 	snd_emu8000_poke((emu), EMU8000_DATA0(emu), EMU8000_CMD(1, (30)), 0x4828);
502 	/* this is really odd part.. */
503 	outb(0x3C, EMU8000_PTR(emu));
504 	outb(0, EMU8000_DATA1(emu));
505 
506 	/* skew volume & cutoff */
507 	EMU8000_VTFT_WRITE(emu, 30, 0x8000FFFF);
508 	EMU8000_VTFT_WRITE(emu, 31, 0x8000FFFF);
509 }
510 
511 
512 /*
513  * The main initialization routine.
514  */
515 static void
snd_emu8000_init_hw(struct snd_emu8000 * emu)516 snd_emu8000_init_hw(struct snd_emu8000 *emu)
517 {
518 	int i;
519 
520 	emu->last_reg = 0xffff; /* reset the last register index */
521 
522 	/* initialize hardware configuration */
523 	EMU8000_HWCF1_WRITE(emu, 0x0059);
524 	EMU8000_HWCF2_WRITE(emu, 0x0020);
525 
526 	/* disable audio; this seems to reduce a clicking noise a bit.. */
527 	EMU8000_HWCF3_WRITE(emu, 0);
528 
529 	/* initialize audio channels */
530 	init_audio(emu);
531 
532 	/* initialize DMA */
533 	init_dma(emu);
534 
535 	/* initialize init arrays */
536 	init_arrays(emu);
537 
538 	/*
539 	 * Initialize the FM section of the AWE32, this is needed
540 	 * for DRAM refresh as well
541 	 */
542 	snd_emu8000_init_fm(emu);
543 
544 	/* terminate all voices */
545 	for (i = 0; i < EMU8000_DRAM_VOICES; i++)
546 		EMU8000_DCYSUSV_WRITE(emu, 0, 0x807F);
547 
548 	/* check DRAM memory size */
549 	size_dram(emu);
550 
551 	/* enable audio */
552 	EMU8000_HWCF3_WRITE(emu, 0x4);
553 
554 	/* set equzlier, chorus and reverb modes */
555 	snd_emu8000_update_equalizer(emu);
556 	snd_emu8000_update_chorus_mode(emu);
557 	snd_emu8000_update_reverb_mode(emu);
558 }
559 
560 
561 /*----------------------------------------------------------------
562  * Bass/Treble Equalizer
563  *----------------------------------------------------------------*/
564 
565 static unsigned short bass_parm[12][3] = {
566 	{0xD26A, 0xD36A, 0x0000}, /* -12 dB */
567 	{0xD25B, 0xD35B, 0x0000}, /*  -8 */
568 	{0xD24C, 0xD34C, 0x0000}, /*  -6 */
569 	{0xD23D, 0xD33D, 0x0000}, /*  -4 */
570 	{0xD21F, 0xD31F, 0x0000}, /*  -2 */
571 	{0xC208, 0xC308, 0x0001}, /*   0 (HW default) */
572 	{0xC219, 0xC319, 0x0001}, /*  +2 */
573 	{0xC22A, 0xC32A, 0x0001}, /*  +4 */
574 	{0xC24C, 0xC34C, 0x0001}, /*  +6 */
575 	{0xC26E, 0xC36E, 0x0001}, /*  +8 */
576 	{0xC248, 0xC384, 0x0002}, /* +10 */
577 	{0xC26A, 0xC36A, 0x0002}, /* +12 dB */
578 };
579 
580 static unsigned short treble_parm[12][9] = {
581 	{0x821E, 0xC26A, 0x031E, 0xC36A, 0x021E, 0xD208, 0x831E, 0xD308, 0x0001}, /* -12 dB */
582 	{0x821E, 0xC25B, 0x031E, 0xC35B, 0x021E, 0xD208, 0x831E, 0xD308, 0x0001},
583 	{0x821E, 0xC24C, 0x031E, 0xC34C, 0x021E, 0xD208, 0x831E, 0xD308, 0x0001},
584 	{0x821E, 0xC23D, 0x031E, 0xC33D, 0x021E, 0xD208, 0x831E, 0xD308, 0x0001},
585 	{0x821E, 0xC21F, 0x031E, 0xC31F, 0x021E, 0xD208, 0x831E, 0xD308, 0x0001},
586 	{0x821E, 0xD208, 0x031E, 0xD308, 0x021E, 0xD208, 0x831E, 0xD308, 0x0002},
587 	{0x821E, 0xD208, 0x031E, 0xD308, 0x021D, 0xD219, 0x831D, 0xD319, 0x0002},
588 	{0x821E, 0xD208, 0x031E, 0xD308, 0x021C, 0xD22A, 0x831C, 0xD32A, 0x0002},
589 	{0x821E, 0xD208, 0x031E, 0xD308, 0x021A, 0xD24C, 0x831A, 0xD34C, 0x0002},
590 	{0x821E, 0xD208, 0x031E, 0xD308, 0x0219, 0xD26E, 0x8319, 0xD36E, 0x0002}, /* +8 (HW default) */
591 	{0x821D, 0xD219, 0x031D, 0xD319, 0x0219, 0xD26E, 0x8319, 0xD36E, 0x0002},
592 	{0x821C, 0xD22A, 0x031C, 0xD32A, 0x0219, 0xD26E, 0x8319, 0xD36E, 0x0002}  /* +12 dB */
593 };
594 
595 
596 /*
597  * set Emu8000 digital equalizer; from 0 to 11 [-12dB - 12dB]
598  */
599 /*exported*/ void
snd_emu8000_update_equalizer(struct snd_emu8000 * emu)600 snd_emu8000_update_equalizer(struct snd_emu8000 *emu)
601 {
602 	unsigned short w;
603 	int bass = emu->bass_level;
604 	int treble = emu->treble_level;
605 
606 	if (bass < 0 || bass > 11 || treble < 0 || treble > 11)
607 		return;
608 	EMU8000_INIT4_WRITE(emu, 0x01, bass_parm[bass][0]);
609 	EMU8000_INIT4_WRITE(emu, 0x11, bass_parm[bass][1]);
610 	EMU8000_INIT3_WRITE(emu, 0x11, treble_parm[treble][0]);
611 	EMU8000_INIT3_WRITE(emu, 0x13, treble_parm[treble][1]);
612 	EMU8000_INIT3_WRITE(emu, 0x1b, treble_parm[treble][2]);
613 	EMU8000_INIT4_WRITE(emu, 0x07, treble_parm[treble][3]);
614 	EMU8000_INIT4_WRITE(emu, 0x0b, treble_parm[treble][4]);
615 	EMU8000_INIT4_WRITE(emu, 0x0d, treble_parm[treble][5]);
616 	EMU8000_INIT4_WRITE(emu, 0x17, treble_parm[treble][6]);
617 	EMU8000_INIT4_WRITE(emu, 0x19, treble_parm[treble][7]);
618 	w = bass_parm[bass][2] + treble_parm[treble][8];
619 	EMU8000_INIT4_WRITE(emu, 0x15, (unsigned short)(w + 0x0262));
620 	EMU8000_INIT4_WRITE(emu, 0x1d, (unsigned short)(w + 0x8362));
621 }
622 
623 
624 /*----------------------------------------------------------------
625  * Chorus mode control
626  *----------------------------------------------------------------*/
627 
628 /*
629  * chorus mode parameters
630  */
631 #define SNDRV_EMU8000_CHORUS_1		0
632 #define	SNDRV_EMU8000_CHORUS_2		1
633 #define	SNDRV_EMU8000_CHORUS_3		2
634 #define	SNDRV_EMU8000_CHORUS_4		3
635 #define	SNDRV_EMU8000_CHORUS_FEEDBACK	4
636 #define	SNDRV_EMU8000_CHORUS_FLANGER	5
637 #define	SNDRV_EMU8000_CHORUS_SHORTDELAY	6
638 #define	SNDRV_EMU8000_CHORUS_SHORTDELAY2	7
639 #define SNDRV_EMU8000_CHORUS_PREDEFINED	8
640 /* user can define chorus modes up to 32 */
641 #define SNDRV_EMU8000_CHORUS_NUMBERS	32
642 
643 struct soundfont_chorus_fx {
644 	unsigned short feedback;	/* feedback level (0xE600-0xE6FF) */
645 	unsigned short delay_offset;	/* delay (0-0x0DA3) [1/44100 sec] */
646 	unsigned short lfo_depth;	/* LFO depth (0xBC00-0xBCFF) */
647 	unsigned int delay;	/* right delay (0-0xFFFFFFFF) [1/256/44100 sec] */
648 	unsigned int lfo_freq;		/* LFO freq LFO freq (0-0xFFFFFFFF) */
649 };
650 
651 /* 5 parameters for each chorus mode; 3 x 16bit, 2 x 32bit */
652 static char chorus_defined[SNDRV_EMU8000_CHORUS_NUMBERS];
653 static struct soundfont_chorus_fx chorus_parm[SNDRV_EMU8000_CHORUS_NUMBERS] = {
654 	{0xE600, 0x03F6, 0xBC2C ,0x00000000, 0x0000006D}, /* chorus 1 */
655 	{0xE608, 0x031A, 0xBC6E, 0x00000000, 0x0000017C}, /* chorus 2 */
656 	{0xE610, 0x031A, 0xBC84, 0x00000000, 0x00000083}, /* chorus 3 */
657 	{0xE620, 0x0269, 0xBC6E, 0x00000000, 0x0000017C}, /* chorus 4 */
658 	{0xE680, 0x04D3, 0xBCA6, 0x00000000, 0x0000005B}, /* feedback */
659 	{0xE6E0, 0x044E, 0xBC37, 0x00000000, 0x00000026}, /* flanger */
660 	{0xE600, 0x0B06, 0xBC00, 0x0006E000, 0x00000083}, /* short delay */
661 	{0xE6C0, 0x0B06, 0xBC00, 0x0006E000, 0x00000083}, /* short delay + feedback */
662 };
663 
664 /*exported*/ int
snd_emu8000_load_chorus_fx(struct snd_emu8000 * emu,int mode,const void __user * buf,long len)665 snd_emu8000_load_chorus_fx(struct snd_emu8000 *emu, int mode, const void __user *buf, long len)
666 {
667 	struct soundfont_chorus_fx rec;
668 	if (mode < SNDRV_EMU8000_CHORUS_PREDEFINED || mode >= SNDRV_EMU8000_CHORUS_NUMBERS) {
669 		snd_printk(KERN_WARNING "invalid chorus mode %d for uploading\n", mode);
670 		return -EINVAL;
671 	}
672 	if (len < (long)sizeof(rec) || copy_from_user(&rec, buf, sizeof(rec)))
673 		return -EFAULT;
674 	chorus_parm[mode] = rec;
675 	chorus_defined[mode] = 1;
676 	return 0;
677 }
678 
679 /*exported*/ void
snd_emu8000_update_chorus_mode(struct snd_emu8000 * emu)680 snd_emu8000_update_chorus_mode(struct snd_emu8000 *emu)
681 {
682 	int effect = emu->chorus_mode;
683 	if (effect < 0 || effect >= SNDRV_EMU8000_CHORUS_NUMBERS ||
684 	    (effect >= SNDRV_EMU8000_CHORUS_PREDEFINED && !chorus_defined[effect]))
685 		return;
686 	EMU8000_INIT3_WRITE(emu, 0x09, chorus_parm[effect].feedback);
687 	EMU8000_INIT3_WRITE(emu, 0x0c, chorus_parm[effect].delay_offset);
688 	EMU8000_INIT4_WRITE(emu, 0x03, chorus_parm[effect].lfo_depth);
689 	EMU8000_HWCF4_WRITE(emu, chorus_parm[effect].delay);
690 	EMU8000_HWCF5_WRITE(emu, chorus_parm[effect].lfo_freq);
691 	EMU8000_HWCF6_WRITE(emu, 0x8000);
692 	EMU8000_HWCF7_WRITE(emu, 0x0000);
693 }
694 
695 /*----------------------------------------------------------------
696  * Reverb mode control
697  *----------------------------------------------------------------*/
698 
699 /*
700  * reverb mode parameters
701  */
702 #define	SNDRV_EMU8000_REVERB_ROOM1	0
703 #define SNDRV_EMU8000_REVERB_ROOM2	1
704 #define	SNDRV_EMU8000_REVERB_ROOM3	2
705 #define	SNDRV_EMU8000_REVERB_HALL1	3
706 #define	SNDRV_EMU8000_REVERB_HALL2	4
707 #define	SNDRV_EMU8000_REVERB_PLATE	5
708 #define	SNDRV_EMU8000_REVERB_DELAY	6
709 #define	SNDRV_EMU8000_REVERB_PANNINGDELAY 7
710 #define SNDRV_EMU8000_REVERB_PREDEFINED	8
711 /* user can define reverb modes up to 32 */
712 #define SNDRV_EMU8000_REVERB_NUMBERS	32
713 
714 struct soundfont_reverb_fx {
715 	unsigned short parms[28];
716 };
717 
718 /* reverb mode settings; write the following 28 data of 16 bit length
719  *   on the corresponding ports in the reverb_cmds array
720  */
721 static char reverb_defined[SNDRV_EMU8000_CHORUS_NUMBERS];
722 static struct soundfont_reverb_fx reverb_parm[SNDRV_EMU8000_REVERB_NUMBERS] = {
723 {{  /* room 1 */
724 	0xB488, 0xA450, 0x9550, 0x84B5, 0x383A, 0x3EB5, 0x72F4,
725 	0x72A4, 0x7254, 0x7204, 0x7204, 0x7204, 0x4416, 0x4516,
726 	0xA490, 0xA590, 0x842A, 0x852A, 0x842A, 0x852A, 0x8429,
727 	0x8529, 0x8429, 0x8529, 0x8428, 0x8528, 0x8428, 0x8528,
728 }},
729 {{  /* room 2 */
730 	0xB488, 0xA458, 0x9558, 0x84B5, 0x383A, 0x3EB5, 0x7284,
731 	0x7254, 0x7224, 0x7224, 0x7254, 0x7284, 0x4448, 0x4548,
732 	0xA440, 0xA540, 0x842A, 0x852A, 0x842A, 0x852A, 0x8429,
733 	0x8529, 0x8429, 0x8529, 0x8428, 0x8528, 0x8428, 0x8528,
734 }},
735 {{  /* room 3 */
736 	0xB488, 0xA460, 0x9560, 0x84B5, 0x383A, 0x3EB5, 0x7284,
737 	0x7254, 0x7224, 0x7224, 0x7254, 0x7284, 0x4416, 0x4516,
738 	0xA490, 0xA590, 0x842C, 0x852C, 0x842C, 0x852C, 0x842B,
739 	0x852B, 0x842B, 0x852B, 0x842A, 0x852A, 0x842A, 0x852A,
740 }},
741 {{  /* hall 1 */
742 	0xB488, 0xA470, 0x9570, 0x84B5, 0x383A, 0x3EB5, 0x7284,
743 	0x7254, 0x7224, 0x7224, 0x7254, 0x7284, 0x4448, 0x4548,
744 	0xA440, 0xA540, 0x842B, 0x852B, 0x842B, 0x852B, 0x842A,
745 	0x852A, 0x842A, 0x852A, 0x8429, 0x8529, 0x8429, 0x8529,
746 }},
747 {{  /* hall 2 */
748 	0xB488, 0xA470, 0x9570, 0x84B5, 0x383A, 0x3EB5, 0x7254,
749 	0x7234, 0x7224, 0x7254, 0x7264, 0x7294, 0x44C3, 0x45C3,
750 	0xA404, 0xA504, 0x842A, 0x852A, 0x842A, 0x852A, 0x8429,
751 	0x8529, 0x8429, 0x8529, 0x8428, 0x8528, 0x8428, 0x8528,
752 }},
753 {{  /* plate */
754 	0xB4FF, 0xA470, 0x9570, 0x84B5, 0x383A, 0x3EB5, 0x7234,
755 	0x7234, 0x7234, 0x7234, 0x7234, 0x7234, 0x4448, 0x4548,
756 	0xA440, 0xA540, 0x842A, 0x852A, 0x842A, 0x852A, 0x8429,
757 	0x8529, 0x8429, 0x8529, 0x8428, 0x8528, 0x8428, 0x8528,
758 }},
759 {{  /* delay */
760 	0xB4FF, 0xA470, 0x9500, 0x84B5, 0x333A, 0x39B5, 0x7204,
761 	0x7204, 0x7204, 0x7204, 0x7204, 0x72F4, 0x4400, 0x4500,
762 	0xA4FF, 0xA5FF, 0x8420, 0x8520, 0x8420, 0x8520, 0x8420,
763 	0x8520, 0x8420, 0x8520, 0x8420, 0x8520, 0x8420, 0x8520,
764 }},
765 {{  /* panning delay */
766 	0xB4FF, 0xA490, 0x9590, 0x8474, 0x333A, 0x39B5, 0x7204,
767 	0x7204, 0x7204, 0x7204, 0x7204, 0x72F4, 0x4400, 0x4500,
768 	0xA4FF, 0xA5FF, 0x8420, 0x8520, 0x8420, 0x8520, 0x8420,
769 	0x8520, 0x8420, 0x8520, 0x8420, 0x8520, 0x8420, 0x8520,
770 }},
771 };
772 
773 enum { DATA1, DATA2 };
774 #define AWE_INIT1(c)	EMU8000_CMD(2,c), DATA1
775 #define AWE_INIT2(c)	EMU8000_CMD(2,c), DATA2
776 #define AWE_INIT3(c)	EMU8000_CMD(3,c), DATA1
777 #define AWE_INIT4(c)	EMU8000_CMD(3,c), DATA2
778 
779 static struct reverb_cmd_pair {
780 	unsigned short cmd, port;
781 } reverb_cmds[28] = {
782   {AWE_INIT1(0x03)}, {AWE_INIT1(0x05)}, {AWE_INIT4(0x1F)}, {AWE_INIT1(0x07)},
783   {AWE_INIT2(0x14)}, {AWE_INIT2(0x16)}, {AWE_INIT1(0x0F)}, {AWE_INIT1(0x17)},
784   {AWE_INIT1(0x1F)}, {AWE_INIT2(0x07)}, {AWE_INIT2(0x0F)}, {AWE_INIT2(0x17)},
785   {AWE_INIT2(0x1D)}, {AWE_INIT2(0x1F)}, {AWE_INIT3(0x01)}, {AWE_INIT3(0x03)},
786   {AWE_INIT1(0x09)}, {AWE_INIT1(0x0B)}, {AWE_INIT1(0x11)}, {AWE_INIT1(0x13)},
787   {AWE_INIT1(0x19)}, {AWE_INIT1(0x1B)}, {AWE_INIT2(0x01)}, {AWE_INIT2(0x03)},
788   {AWE_INIT2(0x09)}, {AWE_INIT2(0x0B)}, {AWE_INIT2(0x11)}, {AWE_INIT2(0x13)},
789 };
790 
791 /*exported*/ int
snd_emu8000_load_reverb_fx(struct snd_emu8000 * emu,int mode,const void __user * buf,long len)792 snd_emu8000_load_reverb_fx(struct snd_emu8000 *emu, int mode, const void __user *buf, long len)
793 {
794 	struct soundfont_reverb_fx rec;
795 
796 	if (mode < SNDRV_EMU8000_REVERB_PREDEFINED || mode >= SNDRV_EMU8000_REVERB_NUMBERS) {
797 		snd_printk(KERN_WARNING "invalid reverb mode %d for uploading\n", mode);
798 		return -EINVAL;
799 	}
800 	if (len < (long)sizeof(rec) || copy_from_user(&rec, buf, sizeof(rec)))
801 		return -EFAULT;
802 	reverb_parm[mode] = rec;
803 	reverb_defined[mode] = 1;
804 	return 0;
805 }
806 
807 /*exported*/ void
snd_emu8000_update_reverb_mode(struct snd_emu8000 * emu)808 snd_emu8000_update_reverb_mode(struct snd_emu8000 *emu)
809 {
810 	int effect = emu->reverb_mode;
811 	int i;
812 
813 	if (effect < 0 || effect >= SNDRV_EMU8000_REVERB_NUMBERS ||
814 	    (effect >= SNDRV_EMU8000_REVERB_PREDEFINED && !reverb_defined[effect]))
815 		return;
816 	for (i = 0; i < 28; i++) {
817 		int port;
818 		if (reverb_cmds[i].port == DATA1)
819 			port = EMU8000_DATA1(emu);
820 		else
821 			port = EMU8000_DATA2(emu);
822 		snd_emu8000_poke(emu, port, reverb_cmds[i].cmd, reverb_parm[effect].parms[i]);
823 	}
824 }
825 
826 
827 /*----------------------------------------------------------------
828  * mixer interface
829  *----------------------------------------------------------------*/
830 
831 /*
832  * bass/treble
833  */
mixer_bass_treble_info(struct snd_kcontrol * kcontrol,struct snd_ctl_elem_info * uinfo)834 static int mixer_bass_treble_info(struct snd_kcontrol *kcontrol, struct snd_ctl_elem_info *uinfo)
835 {
836 	uinfo->type = SNDRV_CTL_ELEM_TYPE_INTEGER;
837 	uinfo->count = 1;
838 	uinfo->value.integer.min = 0;
839 	uinfo->value.integer.max = 11;
840 	return 0;
841 }
842 
mixer_bass_treble_get(struct snd_kcontrol * kcontrol,struct snd_ctl_elem_value * ucontrol)843 static int mixer_bass_treble_get(struct snd_kcontrol *kcontrol, struct snd_ctl_elem_value *ucontrol)
844 {
845 	struct snd_emu8000 *emu = snd_kcontrol_chip(kcontrol);
846 
847 	ucontrol->value.integer.value[0] = kcontrol->private_value ? emu->treble_level : emu->bass_level;
848 	return 0;
849 }
850 
mixer_bass_treble_put(struct snd_kcontrol * kcontrol,struct snd_ctl_elem_value * ucontrol)851 static int mixer_bass_treble_put(struct snd_kcontrol *kcontrol, struct snd_ctl_elem_value *ucontrol)
852 {
853 	struct snd_emu8000 *emu = snd_kcontrol_chip(kcontrol);
854 	unsigned long flags;
855 	int change;
856 	unsigned short val1;
857 
858 	val1 = ucontrol->value.integer.value[0] % 12;
859 	spin_lock_irqsave(&emu->control_lock, flags);
860 	if (kcontrol->private_value) {
861 		change = val1 != emu->treble_level;
862 		emu->treble_level = val1;
863 	} else {
864 		change = val1 != emu->bass_level;
865 		emu->bass_level = val1;
866 	}
867 	spin_unlock_irqrestore(&emu->control_lock, flags);
868 	snd_emu8000_update_equalizer(emu);
869 	return change;
870 }
871 
872 static struct snd_kcontrol_new mixer_bass_control =
873 {
874 	.iface = SNDRV_CTL_ELEM_IFACE_MIXER,
875 	.name = "Synth Tone Control - Bass",
876 	.info = mixer_bass_treble_info,
877 	.get = mixer_bass_treble_get,
878 	.put = mixer_bass_treble_put,
879 	.private_value = 0,
880 };
881 
882 static struct snd_kcontrol_new mixer_treble_control =
883 {
884 	.iface = SNDRV_CTL_ELEM_IFACE_MIXER,
885 	.name = "Synth Tone Control - Treble",
886 	.info = mixer_bass_treble_info,
887 	.get = mixer_bass_treble_get,
888 	.put = mixer_bass_treble_put,
889 	.private_value = 1,
890 };
891 
892 /*
893  * chorus/reverb mode
894  */
mixer_chorus_reverb_info(struct snd_kcontrol * kcontrol,struct snd_ctl_elem_info * uinfo)895 static int mixer_chorus_reverb_info(struct snd_kcontrol *kcontrol, struct snd_ctl_elem_info *uinfo)
896 {
897 	uinfo->type = SNDRV_CTL_ELEM_TYPE_INTEGER;
898 	uinfo->count = 1;
899 	uinfo->value.integer.min = 0;
900 	uinfo->value.integer.max = kcontrol->private_value ? (SNDRV_EMU8000_CHORUS_NUMBERS-1) : (SNDRV_EMU8000_REVERB_NUMBERS-1);
901 	return 0;
902 }
903 
mixer_chorus_reverb_get(struct snd_kcontrol * kcontrol,struct snd_ctl_elem_value * ucontrol)904 static int mixer_chorus_reverb_get(struct snd_kcontrol *kcontrol, struct snd_ctl_elem_value *ucontrol)
905 {
906 	struct snd_emu8000 *emu = snd_kcontrol_chip(kcontrol);
907 
908 	ucontrol->value.integer.value[0] = kcontrol->private_value ? emu->chorus_mode : emu->reverb_mode;
909 	return 0;
910 }
911 
mixer_chorus_reverb_put(struct snd_kcontrol * kcontrol,struct snd_ctl_elem_value * ucontrol)912 static int mixer_chorus_reverb_put(struct snd_kcontrol *kcontrol, struct snd_ctl_elem_value *ucontrol)
913 {
914 	struct snd_emu8000 *emu = snd_kcontrol_chip(kcontrol);
915 	unsigned long flags;
916 	int change;
917 	unsigned short val1;
918 
919 	spin_lock_irqsave(&emu->control_lock, flags);
920 	if (kcontrol->private_value) {
921 		val1 = ucontrol->value.integer.value[0] % SNDRV_EMU8000_CHORUS_NUMBERS;
922 		change = val1 != emu->chorus_mode;
923 		emu->chorus_mode = val1;
924 	} else {
925 		val1 = ucontrol->value.integer.value[0] % SNDRV_EMU8000_REVERB_NUMBERS;
926 		change = val1 != emu->reverb_mode;
927 		emu->reverb_mode = val1;
928 	}
929 	spin_unlock_irqrestore(&emu->control_lock, flags);
930 	if (change) {
931 		if (kcontrol->private_value)
932 			snd_emu8000_update_chorus_mode(emu);
933 		else
934 			snd_emu8000_update_reverb_mode(emu);
935 	}
936 	return change;
937 }
938 
939 static struct snd_kcontrol_new mixer_chorus_mode_control =
940 {
941 	.iface = SNDRV_CTL_ELEM_IFACE_MIXER,
942 	.name = "Chorus Mode",
943 	.info = mixer_chorus_reverb_info,
944 	.get = mixer_chorus_reverb_get,
945 	.put = mixer_chorus_reverb_put,
946 	.private_value = 1,
947 };
948 
949 static struct snd_kcontrol_new mixer_reverb_mode_control =
950 {
951 	.iface = SNDRV_CTL_ELEM_IFACE_MIXER,
952 	.name = "Reverb Mode",
953 	.info = mixer_chorus_reverb_info,
954 	.get = mixer_chorus_reverb_get,
955 	.put = mixer_chorus_reverb_put,
956 	.private_value = 0,
957 };
958 
959 /*
960  * FM OPL3 chorus/reverb depth
961  */
mixer_fm_depth_info(struct snd_kcontrol * kcontrol,struct snd_ctl_elem_info * uinfo)962 static int mixer_fm_depth_info(struct snd_kcontrol *kcontrol, struct snd_ctl_elem_info *uinfo)
963 {
964 	uinfo->type = SNDRV_CTL_ELEM_TYPE_INTEGER;
965 	uinfo->count = 1;
966 	uinfo->value.integer.min = 0;
967 	uinfo->value.integer.max = 255;
968 	return 0;
969 }
970 
mixer_fm_depth_get(struct snd_kcontrol * kcontrol,struct snd_ctl_elem_value * ucontrol)971 static int mixer_fm_depth_get(struct snd_kcontrol *kcontrol, struct snd_ctl_elem_value *ucontrol)
972 {
973 	struct snd_emu8000 *emu = snd_kcontrol_chip(kcontrol);
974 
975 	ucontrol->value.integer.value[0] = kcontrol->private_value ? emu->fm_chorus_depth : emu->fm_reverb_depth;
976 	return 0;
977 }
978 
mixer_fm_depth_put(struct snd_kcontrol * kcontrol,struct snd_ctl_elem_value * ucontrol)979 static int mixer_fm_depth_put(struct snd_kcontrol *kcontrol, struct snd_ctl_elem_value *ucontrol)
980 {
981 	struct snd_emu8000 *emu = snd_kcontrol_chip(kcontrol);
982 	unsigned long flags;
983 	int change;
984 	unsigned short val1;
985 
986 	val1 = ucontrol->value.integer.value[0] % 256;
987 	spin_lock_irqsave(&emu->control_lock, flags);
988 	if (kcontrol->private_value) {
989 		change = val1 != emu->fm_chorus_depth;
990 		emu->fm_chorus_depth = val1;
991 	} else {
992 		change = val1 != emu->fm_reverb_depth;
993 		emu->fm_reverb_depth = val1;
994 	}
995 	spin_unlock_irqrestore(&emu->control_lock, flags);
996 	if (change)
997 		snd_emu8000_init_fm(emu);
998 	return change;
999 }
1000 
1001 static struct snd_kcontrol_new mixer_fm_chorus_depth_control =
1002 {
1003 	.iface = SNDRV_CTL_ELEM_IFACE_MIXER,
1004 	.name = "FM Chorus Depth",
1005 	.info = mixer_fm_depth_info,
1006 	.get = mixer_fm_depth_get,
1007 	.put = mixer_fm_depth_put,
1008 	.private_value = 1,
1009 };
1010 
1011 static struct snd_kcontrol_new mixer_fm_reverb_depth_control =
1012 {
1013 	.iface = SNDRV_CTL_ELEM_IFACE_MIXER,
1014 	.name = "FM Reverb Depth",
1015 	.info = mixer_fm_depth_info,
1016 	.get = mixer_fm_depth_get,
1017 	.put = mixer_fm_depth_put,
1018 	.private_value = 0,
1019 };
1020 
1021 
1022 static struct snd_kcontrol_new *mixer_defs[EMU8000_NUM_CONTROLS] = {
1023 	&mixer_bass_control,
1024 	&mixer_treble_control,
1025 	&mixer_chorus_mode_control,
1026 	&mixer_reverb_mode_control,
1027 	&mixer_fm_chorus_depth_control,
1028 	&mixer_fm_reverb_depth_control,
1029 };
1030 
1031 /*
1032  * create and attach mixer elements for WaveTable treble/bass controls
1033  */
1034 static int
snd_emu8000_create_mixer(struct snd_card * card,struct snd_emu8000 * emu)1035 snd_emu8000_create_mixer(struct snd_card *card, struct snd_emu8000 *emu)
1036 {
1037 	int i, err = 0;
1038 
1039 	if (snd_BUG_ON(!emu || !card))
1040 		return -EINVAL;
1041 
1042 	spin_lock_init(&emu->control_lock);
1043 
1044 	memset(emu->controls, 0, sizeof(emu->controls));
1045 	for (i = 0; i < EMU8000_NUM_CONTROLS; i++) {
1046 		if ((err = snd_ctl_add(card, emu->controls[i] = snd_ctl_new1(mixer_defs[i], emu))) < 0)
1047 			goto __error;
1048 	}
1049 	return 0;
1050 
1051 __error:
1052 	for (i = 0; i < EMU8000_NUM_CONTROLS; i++) {
1053 		down_write(&card->controls_rwsem);
1054 		if (emu->controls[i])
1055 			snd_ctl_remove(card, emu->controls[i]);
1056 		up_write(&card->controls_rwsem);
1057 	}
1058 	return err;
1059 }
1060 
1061 
1062 /*
1063  * free resources
1064  */
snd_emu8000_free(struct snd_emu8000 * hw)1065 static int snd_emu8000_free(struct snd_emu8000 *hw)
1066 {
1067 	release_and_free_resource(hw->res_port1);
1068 	release_and_free_resource(hw->res_port2);
1069 	release_and_free_resource(hw->res_port3);
1070 	kfree(hw);
1071 	return 0;
1072 }
1073 
1074 /*
1075  */
snd_emu8000_dev_free(struct snd_device * device)1076 static int snd_emu8000_dev_free(struct snd_device *device)
1077 {
1078 	struct snd_emu8000 *hw = device->device_data;
1079 	return snd_emu8000_free(hw);
1080 }
1081 
1082 /*
1083  * initialize and register emu8000 synth device.
1084  */
1085 int
snd_emu8000_new(struct snd_card * card,int index,long port,int seq_ports,struct snd_seq_device ** awe_ret)1086 snd_emu8000_new(struct snd_card *card, int index, long port, int seq_ports,
1087 		struct snd_seq_device **awe_ret)
1088 {
1089 	struct snd_seq_device *awe;
1090 	struct snd_emu8000 *hw;
1091 	int err;
1092 	static struct snd_device_ops ops = {
1093 		.dev_free = snd_emu8000_dev_free,
1094 	};
1095 
1096 	if (awe_ret)
1097 		*awe_ret = NULL;
1098 
1099 	if (seq_ports <= 0)
1100 		return 0;
1101 
1102 	hw = kzalloc(sizeof(*hw), GFP_KERNEL);
1103 	if (hw == NULL)
1104 		return -ENOMEM;
1105 	spin_lock_init(&hw->reg_lock);
1106 	hw->index = index;
1107 	hw->port1 = port;
1108 	hw->port2 = port + 0x400;
1109 	hw->port3 = port + 0x800;
1110 	if (!(hw->res_port1 = request_region(hw->port1, 4, "Emu8000-1")) ||
1111 	    !(hw->res_port2 = request_region(hw->port2, 4, "Emu8000-2")) ||
1112 	    !(hw->res_port3 = request_region(hw->port3, 4, "Emu8000-3"))) {
1113 		snd_printk(KERN_ERR "sbawe: can't grab ports 0x%lx, 0x%lx, 0x%lx\n", hw->port1, hw->port2, hw->port3);
1114 		snd_emu8000_free(hw);
1115 		return -EBUSY;
1116 	}
1117 	hw->mem_size = 0;
1118 	hw->card = card;
1119 	hw->seq_ports = seq_ports;
1120 	hw->bass_level = 5;
1121 	hw->treble_level = 9;
1122 	hw->chorus_mode = 2;
1123 	hw->reverb_mode = 4;
1124 	hw->fm_chorus_depth = 0;
1125 	hw->fm_reverb_depth = 0;
1126 
1127 	if (snd_emu8000_detect(hw) < 0) {
1128 		snd_emu8000_free(hw);
1129 		return -ENODEV;
1130 	}
1131 
1132 	snd_emu8000_init_hw(hw);
1133 	if ((err = snd_emu8000_create_mixer(card, hw)) < 0) {
1134 		snd_emu8000_free(hw);
1135 		return err;
1136 	}
1137 
1138 	if ((err = snd_device_new(card, SNDRV_DEV_CODEC, hw, &ops)) < 0) {
1139 		snd_emu8000_free(hw);
1140 		return err;
1141 	}
1142 #if defined(CONFIG_SND_SEQUENCER) || (defined(MODULE) && defined(CONFIG_SND_SEQUENCER_MODULE))
1143 	if (snd_seq_device_new(card, index, SNDRV_SEQ_DEV_ID_EMU8000,
1144 			       sizeof(struct snd_emu8000*), &awe) >= 0) {
1145 		strcpy(awe->name, "EMU-8000");
1146 		*(struct snd_emu8000 **)SNDRV_SEQ_DEVICE_ARGPTR(awe) = hw;
1147 	}
1148 #else
1149 	awe = NULL;
1150 #endif
1151 	if (awe_ret)
1152 		*awe_ret = awe;
1153 
1154 	return 0;
1155 }
1156 
1157 
1158 /*
1159  * exported stuff
1160  */
1161 
1162 EXPORT_SYMBOL(snd_emu8000_poke);
1163 EXPORT_SYMBOL(snd_emu8000_peek);
1164 EXPORT_SYMBOL(snd_emu8000_poke_dw);
1165 EXPORT_SYMBOL(snd_emu8000_peek_dw);
1166 EXPORT_SYMBOL(snd_emu8000_dma_chan);
1167 EXPORT_SYMBOL(snd_emu8000_init_fm);
1168 EXPORT_SYMBOL(snd_emu8000_load_chorus_fx);
1169 EXPORT_SYMBOL(snd_emu8000_load_reverb_fx);
1170 EXPORT_SYMBOL(snd_emu8000_update_chorus_mode);
1171 EXPORT_SYMBOL(snd_emu8000_update_reverb_mode);
1172 EXPORT_SYMBOL(snd_emu8000_update_equalizer);
1173