xref: /openbmc/linux/sound/pci/emu10k1/emuproc.c (revision 6ab13291ba82e6f0c8778cb45726dffffb9205f5)
11a59d1b8SThomas Gleixner // SPDX-License-Identifier: GPL-2.0-or-later
21da177e4SLinus Torvalds /*
3c1017a4cSJaroslav Kysela  *  Copyright (c) by Jaroslav Kysela <perex@perex.cz>
41da177e4SLinus Torvalds  *                   Creative Labs, Inc.
51da177e4SLinus Torvalds  *  Routines for control of EMU10K1 chips / proc interface routines
61da177e4SLinus Torvalds  *
79f4bd5ddSJames Courtier-Dutton  *  Copyright (c) by James Courtier-Dutton <James@superbug.co.uk>
89f4bd5ddSJames Courtier-Dutton  *  	Added EMU 1010 support.
99f4bd5ddSJames Courtier-Dutton  *
101da177e4SLinus Torvalds  *  BUGS:
111da177e4SLinus Torvalds  *    --
121da177e4SLinus Torvalds  *
131da177e4SLinus Torvalds  *  TODO:
141da177e4SLinus Torvalds  *    --
151da177e4SLinus Torvalds  */
161da177e4SLinus Torvalds 
171da177e4SLinus Torvalds #include <linux/slab.h>
181da177e4SLinus Torvalds #include <linux/init.h>
191da177e4SLinus Torvalds #include <sound/core.h>
201da177e4SLinus Torvalds #include <sound/emu10k1.h>
21001f7589SJames Courtier-Dutton #include "p16v.h"
221da177e4SLinus Torvalds 
23eb4698f3STakashi Iwai static void snd_emu10k1_proc_spdif_status(struct snd_emu10k1 * emu,
24eb4698f3STakashi Iwai 					  struct snd_info_buffer *buffer,
251da177e4SLinus Torvalds 					  char *title,
261da177e4SLinus Torvalds 					  int status_reg,
271da177e4SLinus Torvalds 					  int rate_reg)
281da177e4SLinus Torvalds {
296fddce26STakashi Iwai 	static const char * const clkaccy[4] = { "1000ppm", "50ppm", "variable", "unknown" };
306fddce26STakashi Iwai 	static const int samplerate[16] = { 44100, 1, 48000, 32000, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15 };
316fddce26STakashi Iwai 	static const char * const channel[16] = { "unspec", "left", "right", "3", "4", "5", "6", "7", "8", "9", "10", "11", "12", "13", "14", "15" };
326fddce26STakashi Iwai 	static const char * const emphasis[8] = { "none", "50/15 usec 2 channel", "2", "3", "4", "5", "6", "7" };
331da177e4SLinus Torvalds 	unsigned int status, rate = 0;
341da177e4SLinus Torvalds 
351da177e4SLinus Torvalds 	status = snd_emu10k1_ptr_read(emu, status_reg, 0);
361da177e4SLinus Torvalds 
371da177e4SLinus Torvalds 	snd_iprintf(buffer, "\n%s\n", title);
381da177e4SLinus Torvalds 
39df34140aSJames Courtier-Dutton 	if (status != 0xffffffff) {
401da177e4SLinus Torvalds 		snd_iprintf(buffer, "Professional Mode     : %s\n", (status & SPCS_PROFESSIONAL) ? "yes" : "no");
411da177e4SLinus Torvalds 		snd_iprintf(buffer, "Not Audio Data        : %s\n", (status & SPCS_NOTAUDIODATA) ? "yes" : "no");
421da177e4SLinus Torvalds 		snd_iprintf(buffer, "Copyright             : %s\n", (status & SPCS_COPYRIGHT) ? "yes" : "no");
431da177e4SLinus Torvalds 		snd_iprintf(buffer, "Emphasis              : %s\n", emphasis[(status & SPCS_EMPHASISMASK) >> 3]);
441da177e4SLinus Torvalds 		snd_iprintf(buffer, "Mode                  : %i\n", (status & SPCS_MODEMASK) >> 6);
451da177e4SLinus Torvalds 		snd_iprintf(buffer, "Category Code         : 0x%x\n", (status & SPCS_CATEGORYCODEMASK) >> 8);
461da177e4SLinus Torvalds 		snd_iprintf(buffer, "Generation Status     : %s\n", status & SPCS_GENERATIONSTATUS ? "original" : "copy");
471da177e4SLinus Torvalds 		snd_iprintf(buffer, "Source Mask           : %i\n", (status & SPCS_SOURCENUMMASK) >> 16);
481da177e4SLinus Torvalds 		snd_iprintf(buffer, "Channel Number        : %s\n", channel[(status & SPCS_CHANNELNUMMASK) >> 20]);
491da177e4SLinus Torvalds 		snd_iprintf(buffer, "Sample Rate           : %iHz\n", samplerate[(status & SPCS_SAMPLERATEMASK) >> 24]);
501da177e4SLinus Torvalds 		snd_iprintf(buffer, "Clock Accuracy        : %s\n", clkaccy[(status & SPCS_CLKACCYMASK) >> 28]);
511da177e4SLinus Torvalds 
521da177e4SLinus Torvalds 		if (rate_reg > 0) {
53df34140aSJames Courtier-Dutton 			rate = snd_emu10k1_ptr_read(emu, rate_reg, 0);
54001f7589SJames Courtier-Dutton 			snd_iprintf(buffer, "S/PDIF Valid          : %s\n", rate & SRCS_SPDIFVALID ? "on" : "off");
551da177e4SLinus Torvalds 			snd_iprintf(buffer, "S/PDIF Locked         : %s\n", rate & SRCS_SPDIFLOCKED ? "on" : "off");
561da177e4SLinus Torvalds 			snd_iprintf(buffer, "Rate Locked           : %s\n", rate & SRCS_RATELOCKED ? "on" : "off");
57df34140aSJames Courtier-Dutton 			/* From ((Rate * 48000 ) / 262144); */
58df34140aSJames Courtier-Dutton 			snd_iprintf(buffer, "Estimated Sample Rate : %d\n", ((rate & 0xFFFFF ) * 375) >> 11);
591da177e4SLinus Torvalds 		}
60df34140aSJames Courtier-Dutton 	} else {
61df34140aSJames Courtier-Dutton 		snd_iprintf(buffer, "No signal detected.\n");
62df34140aSJames Courtier-Dutton 	}
63df34140aSJames Courtier-Dutton 
641da177e4SLinus Torvalds }
651da177e4SLinus Torvalds 
66eb4698f3STakashi Iwai static void snd_emu10k1_proc_read(struct snd_info_entry *entry,
67eb4698f3STakashi Iwai 				  struct snd_info_buffer *buffer)
681da177e4SLinus Torvalds {
691da177e4SLinus Torvalds 	/* FIXME - output names are in emufx.c too */
706fddce26STakashi Iwai 	static const char * const creative_outs[32] = {
711da177e4SLinus Torvalds 		/* 00 */ "AC97 Left",
721da177e4SLinus Torvalds 		/* 01 */ "AC97 Right",
731da177e4SLinus Torvalds 		/* 02 */ "Optical IEC958 Left",
741da177e4SLinus Torvalds 		/* 03 */ "Optical IEC958 Right",
751da177e4SLinus Torvalds 		/* 04 */ "Center",
761da177e4SLinus Torvalds 		/* 05 */ "LFE",
771da177e4SLinus Torvalds 		/* 06 */ "Headphone Left",
781da177e4SLinus Torvalds 		/* 07 */ "Headphone Right",
791da177e4SLinus Torvalds 		/* 08 */ "Surround Left",
801da177e4SLinus Torvalds 		/* 09 */ "Surround Right",
811da177e4SLinus Torvalds 		/* 10 */ "PCM Capture Left",
821da177e4SLinus Torvalds 		/* 11 */ "PCM Capture Right",
831da177e4SLinus Torvalds 		/* 12 */ "MIC Capture",
841da177e4SLinus Torvalds 		/* 13 */ "AC97 Surround Left",
851da177e4SLinus Torvalds 		/* 14 */ "AC97 Surround Right",
861da177e4SLinus Torvalds 		/* 15 */ "???",
871da177e4SLinus Torvalds 		/* 16 */ "???",
881da177e4SLinus Torvalds 		/* 17 */ "Analog Center",
891da177e4SLinus Torvalds 		/* 18 */ "Analog LFE",
901da177e4SLinus Torvalds 		/* 19 */ "???",
911da177e4SLinus Torvalds 		/* 20 */ "???",
921da177e4SLinus Torvalds 		/* 21 */ "???",
931da177e4SLinus Torvalds 		/* 22 */ "???",
941da177e4SLinus Torvalds 		/* 23 */ "???",
951da177e4SLinus Torvalds 		/* 24 */ "???",
961da177e4SLinus Torvalds 		/* 25 */ "???",
971da177e4SLinus Torvalds 		/* 26 */ "???",
981da177e4SLinus Torvalds 		/* 27 */ "???",
991da177e4SLinus Torvalds 		/* 28 */ "???",
1001da177e4SLinus Torvalds 		/* 29 */ "???",
1011da177e4SLinus Torvalds 		/* 30 */ "???",
1021da177e4SLinus Torvalds 		/* 31 */ "???"
1031da177e4SLinus Torvalds 	};
1041da177e4SLinus Torvalds 
1056fddce26STakashi Iwai 	static const char * const audigy_outs[64] = {
1061da177e4SLinus Torvalds 		/* 00 */ "Digital Front Left",
1071da177e4SLinus Torvalds 		/* 01 */ "Digital Front Right",
1081da177e4SLinus Torvalds 		/* 02 */ "Digital Center",
1091da177e4SLinus Torvalds 		/* 03 */ "Digital LEF",
1101da177e4SLinus Torvalds 		/* 04 */ "Headphone Left",
1111da177e4SLinus Torvalds 		/* 05 */ "Headphone Right",
1121da177e4SLinus Torvalds 		/* 06 */ "Digital Rear Left",
1131da177e4SLinus Torvalds 		/* 07 */ "Digital Rear Right",
1141da177e4SLinus Torvalds 		/* 08 */ "Front Left",
1151da177e4SLinus Torvalds 		/* 09 */ "Front Right",
1161da177e4SLinus Torvalds 		/* 10 */ "Center",
1171da177e4SLinus Torvalds 		/* 11 */ "LFE",
1181da177e4SLinus Torvalds 		/* 12 */ "???",
1191da177e4SLinus Torvalds 		/* 13 */ "???",
1201da177e4SLinus Torvalds 		/* 14 */ "Rear Left",
1211da177e4SLinus Torvalds 		/* 15 */ "Rear Right",
1221da177e4SLinus Torvalds 		/* 16 */ "AC97 Front Left",
1231da177e4SLinus Torvalds 		/* 17 */ "AC97 Front Right",
124bf8b47feSColin Ian King 		/* 18 */ "ADC Capture Left",
1251da177e4SLinus Torvalds 		/* 19 */ "ADC Capture Right",
1261da177e4SLinus Torvalds 		/* 20 */ "???",
1271da177e4SLinus Torvalds 		/* 21 */ "???",
1281da177e4SLinus Torvalds 		/* 22 */ "???",
1291da177e4SLinus Torvalds 		/* 23 */ "???",
1301da177e4SLinus Torvalds 		/* 24 */ "???",
1311da177e4SLinus Torvalds 		/* 25 */ "???",
1321da177e4SLinus Torvalds 		/* 26 */ "???",
1331da177e4SLinus Torvalds 		/* 27 */ "???",
1341da177e4SLinus Torvalds 		/* 28 */ "???",
1351da177e4SLinus Torvalds 		/* 29 */ "???",
1361da177e4SLinus Torvalds 		/* 30 */ "???",
1371da177e4SLinus Torvalds 		/* 31 */ "???",
1381da177e4SLinus Torvalds 		/* 32 */ "FXBUS2_0",
1391da177e4SLinus Torvalds 		/* 33 */ "FXBUS2_1",
1401da177e4SLinus Torvalds 		/* 34 */ "FXBUS2_2",
1411da177e4SLinus Torvalds 		/* 35 */ "FXBUS2_3",
1421da177e4SLinus Torvalds 		/* 36 */ "FXBUS2_4",
1431da177e4SLinus Torvalds 		/* 37 */ "FXBUS2_5",
1441da177e4SLinus Torvalds 		/* 38 */ "FXBUS2_6",
1451da177e4SLinus Torvalds 		/* 39 */ "FXBUS2_7",
1461da177e4SLinus Torvalds 		/* 40 */ "FXBUS2_8",
1471da177e4SLinus Torvalds 		/* 41 */ "FXBUS2_9",
1481da177e4SLinus Torvalds 		/* 42 */ "FXBUS2_10",
1491da177e4SLinus Torvalds 		/* 43 */ "FXBUS2_11",
1501da177e4SLinus Torvalds 		/* 44 */ "FXBUS2_12",
1511da177e4SLinus Torvalds 		/* 45 */ "FXBUS2_13",
1521da177e4SLinus Torvalds 		/* 46 */ "FXBUS2_14",
1531da177e4SLinus Torvalds 		/* 47 */ "FXBUS2_15",
1541da177e4SLinus Torvalds 		/* 48 */ "FXBUS2_16",
1551da177e4SLinus Torvalds 		/* 49 */ "FXBUS2_17",
1561da177e4SLinus Torvalds 		/* 50 */ "FXBUS2_18",
1571da177e4SLinus Torvalds 		/* 51 */ "FXBUS2_19",
1581da177e4SLinus Torvalds 		/* 52 */ "FXBUS2_20",
1591da177e4SLinus Torvalds 		/* 53 */ "FXBUS2_21",
1601da177e4SLinus Torvalds 		/* 54 */ "FXBUS2_22",
1611da177e4SLinus Torvalds 		/* 55 */ "FXBUS2_23",
1621da177e4SLinus Torvalds 		/* 56 */ "FXBUS2_24",
1631da177e4SLinus Torvalds 		/* 57 */ "FXBUS2_25",
1641da177e4SLinus Torvalds 		/* 58 */ "FXBUS2_26",
1651da177e4SLinus Torvalds 		/* 59 */ "FXBUS2_27",
1661da177e4SLinus Torvalds 		/* 60 */ "FXBUS2_28",
1671da177e4SLinus Torvalds 		/* 61 */ "FXBUS2_29",
1681da177e4SLinus Torvalds 		/* 62 */ "FXBUS2_30",
1691da177e4SLinus Torvalds 		/* 63 */ "FXBUS2_31"
1701da177e4SLinus Torvalds 	};
1711da177e4SLinus Torvalds 
172eb4698f3STakashi Iwai 	struct snd_emu10k1 *emu = entry->private_data;
173ad326d4aSOswald Buddenhagen 	unsigned int val, val1, ptrx, psst, dsl, snda;
1741da177e4SLinus Torvalds 	int nefx = emu->audigy ? 64 : 32;
1756fddce26STakashi Iwai 	const char * const *outputs = emu->audigy ? audigy_outs : creative_outs;
1761da177e4SLinus Torvalds 	int idx;
1771da177e4SLinus Torvalds 
1781da177e4SLinus Torvalds 	snd_iprintf(buffer, "EMU10K1\n\n");
1791da177e4SLinus Torvalds 	snd_iprintf(buffer, "Card                  : %s\n",
1802b637da5SLee Revell 		    emu->audigy ? "Audigy" : (emu->card_capabilities->ecard ? "EMU APS" : "Creative"));
1811da177e4SLinus Torvalds 	snd_iprintf(buffer, "Internal TRAM (words) : 0x%x\n", emu->fx8010.itram_size);
1821da177e4SLinus Torvalds 	snd_iprintf(buffer, "External TRAM (words) : 0x%x\n", (int)emu->fx8010.etram_pages.bytes / 2);
183ad326d4aSOswald Buddenhagen 
184ad326d4aSOswald Buddenhagen 	snd_iprintf(buffer, "\nEffect Send Routing & Amounts:\n");
1851da177e4SLinus Torvalds 	for (idx = 0; idx < NUM_G; idx++) {
186ad326d4aSOswald Buddenhagen 		ptrx = snd_emu10k1_ptr_read(emu, PTRX, idx);
187ad326d4aSOswald Buddenhagen 		psst = snd_emu10k1_ptr_read(emu, PSST, idx);
188ad326d4aSOswald Buddenhagen 		dsl = snd_emu10k1_ptr_read(emu, DSL, idx);
1891da177e4SLinus Torvalds 		if (emu->audigy) {
190ad326d4aSOswald Buddenhagen 			val = snd_emu10k1_ptr_read(emu, A_FXRT1, idx);
191ad326d4aSOswald Buddenhagen 			val1 = snd_emu10k1_ptr_read(emu, A_FXRT2, idx);
192ad326d4aSOswald Buddenhagen 			snda = snd_emu10k1_ptr_read(emu, A_SENDAMOUNTS, idx);
193ad326d4aSOswald Buddenhagen 			snd_iprintf(buffer, "Ch%-2i: A=%2i:%02x, B=%2i:%02x, C=%2i:%02x, D=%2i:%02x, ",
1941da177e4SLinus Torvalds 				idx,
195ad326d4aSOswald Buddenhagen 				val & 0x3f, REG_VAL_GET(PTRX_FXSENDAMOUNT_A, ptrx),
196ad326d4aSOswald Buddenhagen 				(val >> 8) & 0x3f, REG_VAL_GET(PTRX_FXSENDAMOUNT_B, ptrx),
197ad326d4aSOswald Buddenhagen 				(val >> 16) & 0x3f, REG_VAL_GET(PSST_FXSENDAMOUNT_C, psst),
198ad326d4aSOswald Buddenhagen 				(val >> 24) & 0x3f, REG_VAL_GET(DSL_FXSENDAMOUNT_D, dsl));
199ad326d4aSOswald Buddenhagen 			snd_iprintf(buffer, "E=%2i:%02x, F=%2i:%02x, G=%2i:%02x, H=%2i:%02x\n",
200ad326d4aSOswald Buddenhagen 				val1 & 0x3f, (snda >> 24) & 0xff,
201ad326d4aSOswald Buddenhagen 				(val1 >> 8) & 0x3f, (snda >> 16) & 0xff,
202ad326d4aSOswald Buddenhagen 				(val1 >> 16) & 0x3f, (snda >> 8) & 0xff,
203ad326d4aSOswald Buddenhagen 				(val1 >> 24) & 0x3f, snda & 0xff);
2041da177e4SLinus Torvalds 		} else {
205ad326d4aSOswald Buddenhagen 			val = snd_emu10k1_ptr_read(emu, FXRT, idx);
206ad326d4aSOswald Buddenhagen 			snd_iprintf(buffer, "Ch%-2i: A=%2i:%02x, B=%2i:%02x, C=%2i:%02x, D=%2i:%02x\n",
2071da177e4SLinus Torvalds 				idx,
208ad326d4aSOswald Buddenhagen 				(val >> 16) & 0x0f, REG_VAL_GET(PTRX_FXSENDAMOUNT_A, ptrx),
209ad326d4aSOswald Buddenhagen 				(val >> 20) & 0x0f, REG_VAL_GET(PTRX_FXSENDAMOUNT_B, ptrx),
210ad326d4aSOswald Buddenhagen 				(val >> 24) & 0x0f, REG_VAL_GET(PSST_FXSENDAMOUNT_C, psst),
211ad326d4aSOswald Buddenhagen 				(val >> 28) & 0x0f, REG_VAL_GET(DSL_FXSENDAMOUNT_D, dsl));
2121da177e4SLinus Torvalds 		}
2131da177e4SLinus Torvalds 	}
2141da177e4SLinus Torvalds 	snd_iprintf(buffer, "\nCaptured FX Outputs   :\n");
2151da177e4SLinus Torvalds 	for (idx = 0; idx < nefx; idx++) {
2161da177e4SLinus Torvalds 		if (emu->efx_voices_mask[idx/32] & (1 << (idx%32)))
2171da177e4SLinus Torvalds 			snd_iprintf(buffer, "  Output %02i [%s]\n", idx, outputs[idx]);
2181da177e4SLinus Torvalds 	}
2191da177e4SLinus Torvalds 	snd_iprintf(buffer, "\nAll FX Outputs        :\n");
2201da177e4SLinus Torvalds 	for (idx = 0; idx < (emu->audigy ? 64 : 32); idx++)
2211da177e4SLinus Torvalds 		snd_iprintf(buffer, "  Output %02i [%s]\n", idx, outputs[idx]);
222df34140aSJames Courtier-Dutton }
223df34140aSJames Courtier-Dutton 
224eb4698f3STakashi Iwai static void snd_emu10k1_proc_spdif_read(struct snd_info_entry *entry,
225eb4698f3STakashi Iwai 				  struct snd_info_buffer *buffer)
226df34140aSJames Courtier-Dutton {
227eb4698f3STakashi Iwai 	struct snd_emu10k1 *emu = entry->private_data;
228f93abe51SJames Courtier-Dutton 	u32 value;
229f93abe51SJames Courtier-Dutton 	u32 value2;
230f93abe51SJames Courtier-Dutton 	u32 rate;
231f93abe51SJames Courtier-Dutton 
232190d2c46SJames Courtier-Dutton 	if (emu->card_capabilities->emu_model) {
2336f3609f8SOswald Buddenhagen 		if (!emu->card_capabilities->no_adat) {
234f93abe51SJames Courtier-Dutton 			snd_emu1010_fpga_read(emu, 0x38, &value);
235f93abe51SJames Courtier-Dutton 			if ((value & 0x1) == 0) {
236f93abe51SJames Courtier-Dutton 				snd_emu1010_fpga_read(emu, 0x2a, &value);
237f93abe51SJames Courtier-Dutton 				snd_emu1010_fpga_read(emu, 0x2b, &value2);
238f93abe51SJames Courtier-Dutton 				rate = 0x1770000 / (((value << 5) | value2)+1);
239f93abe51SJames Courtier-Dutton 				snd_iprintf(buffer, "ADAT Locked : %u\n", rate);
240f93abe51SJames Courtier-Dutton 			} else {
241f93abe51SJames Courtier-Dutton 				snd_iprintf(buffer, "ADAT Unlocked\n");
242f93abe51SJames Courtier-Dutton 			}
2436f3609f8SOswald Buddenhagen 		}
244f93abe51SJames Courtier-Dutton 		snd_emu1010_fpga_read(emu, 0x20, &value);
245f93abe51SJames Courtier-Dutton 		if ((value & 0x4) == 0) {
246f93abe51SJames Courtier-Dutton 			snd_emu1010_fpga_read(emu, 0x28, &value);
247f93abe51SJames Courtier-Dutton 			snd_emu1010_fpga_read(emu, 0x29, &value2);
248f93abe51SJames Courtier-Dutton 			rate = 0x1770000 / (((value << 5) | value2)+1);
249f93abe51SJames Courtier-Dutton 			snd_iprintf(buffer, "SPDIF Locked : %d\n", rate);
250f93abe51SJames Courtier-Dutton 		} else {
251f93abe51SJames Courtier-Dutton 			snd_iprintf(buffer, "SPDIF Unlocked\n");
252f93abe51SJames Courtier-Dutton 		}
253f93abe51SJames Courtier-Dutton 	} else {
254df34140aSJames Courtier-Dutton 		snd_emu10k1_proc_spdif_status(emu, buffer, "CD-ROM S/PDIF In", CDCS, CDSRCS);
255df34140aSJames Courtier-Dutton 		snd_emu10k1_proc_spdif_status(emu, buffer, "Optical or Coax S/PDIF In", GPSCS, GPSRCS);
256f93abe51SJames Courtier-Dutton 	}
257df34140aSJames Courtier-Dutton #if 0
2581da177e4SLinus Torvalds 	val = snd_emu10k1_ptr_read(emu, ZVSRCS, 0);
2591da177e4SLinus Torvalds 	snd_iprintf(buffer, "\nZoomed Video\n");
2601da177e4SLinus Torvalds 	snd_iprintf(buffer, "Rate Locked           : %s\n", val & SRCS_RATELOCKED ? "on" : "off");
2611da177e4SLinus Torvalds 	snd_iprintf(buffer, "Estimated Sample Rate : 0x%x\n", val & SRCS_ESTSAMPLERATE);
262df34140aSJames Courtier-Dutton #endif
2631da177e4SLinus Torvalds }
2641da177e4SLinus Torvalds 
265eb4698f3STakashi Iwai static void snd_emu10k1_proc_rates_read(struct snd_info_entry *entry,
266eb4698f3STakashi Iwai 				  struct snd_info_buffer *buffer)
267001f7589SJames Courtier-Dutton {
2686fddce26STakashi Iwai 	static const int samplerate[8] = { 44100, 48000, 96000, 192000, 4, 5, 6, 7 };
269eb4698f3STakashi Iwai 	struct snd_emu10k1 *emu = entry->private_data;
270001f7589SJames Courtier-Dutton 	unsigned int val, tmp, n;
271001f7589SJames Courtier-Dutton 	val = snd_emu10k1_ptr20_read(emu, CAPTURE_RATE_STATUS, 0);
272001f7589SJames Courtier-Dutton 	for (n = 0; n < 4; n++) {
273001f7589SJames Courtier-Dutton 		tmp = val >> (16 + (n*4));
274001f7589SJames Courtier-Dutton 		if (tmp & 0x8) snd_iprintf(buffer, "Channel %d: Rate=%d\n", n, samplerate[tmp & 0x7]);
275001f7589SJames Courtier-Dutton 		else snd_iprintf(buffer, "Channel %d: No input\n", n);
276001f7589SJames Courtier-Dutton 	}
277001f7589SJames Courtier-Dutton }
278001f7589SJames Courtier-Dutton 
2796e91a93dSOswald Buddenhagen struct emu10k1_reg_entry {
2806e91a93dSOswald Buddenhagen 	unsigned short base, size;
2816e91a93dSOswald Buddenhagen 	const char *name;
2826e91a93dSOswald Buddenhagen };
2836e91a93dSOswald Buddenhagen 
2846e91a93dSOswald Buddenhagen static const struct emu10k1_reg_entry sblive_reg_entries[] = {
2856e91a93dSOswald Buddenhagen 	{    0, 0x10, "FXBUS" },
2866e91a93dSOswald Buddenhagen 	{ 0x10, 0x10, "EXTIN" },
2876e91a93dSOswald Buddenhagen 	{ 0x20, 0x10, "EXTOUT" },
2886e91a93dSOswald Buddenhagen 	{ 0x30, 0x10, "FXBUS2" },
2896e91a93dSOswald Buddenhagen 	{ 0x40, 0x20, NULL },  // Constants
2906e91a93dSOswald Buddenhagen 	{ 0x100, 0x100, "GPR" },
2916e91a93dSOswald Buddenhagen 	{ 0x200, 0x80, "ITRAM_DATA" },
2926e91a93dSOswald Buddenhagen 	{ 0x280, 0x20, "ETRAM_DATA" },
2936e91a93dSOswald Buddenhagen 	{ 0x300, 0x80, "ITRAM_ADDR" },
2946e91a93dSOswald Buddenhagen 	{ 0x380, 0x20, "ETRAM_ADDR" },
2956e91a93dSOswald Buddenhagen 	{ 0x400, 0, NULL }
2966e91a93dSOswald Buddenhagen };
2976e91a93dSOswald Buddenhagen 
2986e91a93dSOswald Buddenhagen static const struct emu10k1_reg_entry audigy_reg_entries[] = {
2996e91a93dSOswald Buddenhagen 	{    0, 0x40, "FXBUS" },
3006e91a93dSOswald Buddenhagen 	{ 0x40, 0x10, "EXTIN" },
3016e91a93dSOswald Buddenhagen 	{ 0x50, 0x10, "P16VIN" },
3026e91a93dSOswald Buddenhagen 	{ 0x60, 0x20, "EXTOUT" },
3036e91a93dSOswald Buddenhagen 	{ 0x80, 0x20, "FXBUS2" },
3046e91a93dSOswald Buddenhagen 	{ 0xa0, 0x10, "EMU32OUTH" },
3056e91a93dSOswald Buddenhagen 	{ 0xb0, 0x10, "EMU32OUTL" },
3066e91a93dSOswald Buddenhagen 	{ 0xc0, 0x20, NULL },  // Constants
3076e91a93dSOswald Buddenhagen 	// This can't be quite right - overlap.
3086e91a93dSOswald Buddenhagen 	//{ 0x100, 0xc0, "ITRAM_CTL" },
3096e91a93dSOswald Buddenhagen 	//{ 0x1c0, 0x40, "ETRAM_CTL" },
3106e91a93dSOswald Buddenhagen 	{ 0x160, 0x20, "A3_EMU32IN" },
3116e91a93dSOswald Buddenhagen 	{ 0x1e0, 0x20, "A3_EMU32OUT" },
3126e91a93dSOswald Buddenhagen 	{ 0x200, 0xc0, "ITRAM_DATA" },
3136e91a93dSOswald Buddenhagen 	{ 0x2c0, 0x40, "ETRAM_DATA" },
3146e91a93dSOswald Buddenhagen 	{ 0x300, 0xc0, "ITRAM_ADDR" },
3156e91a93dSOswald Buddenhagen 	{ 0x3c0, 0x40, "ETRAM_ADDR" },
3166e91a93dSOswald Buddenhagen 	{ 0x400, 0x200, "GPR" },
3176e91a93dSOswald Buddenhagen 	{ 0x600, 0, NULL }
3186e91a93dSOswald Buddenhagen };
3196e91a93dSOswald Buddenhagen 
3206e91a93dSOswald Buddenhagen static const char * const emu10k1_const_entries[] = {
3216e91a93dSOswald Buddenhagen 	"C_00000000",
3226e91a93dSOswald Buddenhagen 	"C_00000001",
3236e91a93dSOswald Buddenhagen 	"C_00000002",
3246e91a93dSOswald Buddenhagen 	"C_00000003",
3256e91a93dSOswald Buddenhagen 	"C_00000004",
3266e91a93dSOswald Buddenhagen 	"C_00000008",
3276e91a93dSOswald Buddenhagen 	"C_00000010",
3286e91a93dSOswald Buddenhagen 	"C_00000020",
3296e91a93dSOswald Buddenhagen 	"C_00000100",
3306e91a93dSOswald Buddenhagen 	"C_00010000",
3316e91a93dSOswald Buddenhagen 	"C_00000800",
3326e91a93dSOswald Buddenhagen 	"C_10000000",
3336e91a93dSOswald Buddenhagen 	"C_20000000",
3346e91a93dSOswald Buddenhagen 	"C_40000000",
3356e91a93dSOswald Buddenhagen 	"C_80000000",
3366e91a93dSOswald Buddenhagen 	"C_7fffffff",
3376e91a93dSOswald Buddenhagen 	"C_ffffffff",
3386e91a93dSOswald Buddenhagen 	"C_fffffffe",
3396e91a93dSOswald Buddenhagen 	"C_c0000000",
3406e91a93dSOswald Buddenhagen 	"C_4f1bbcdc",
3416e91a93dSOswald Buddenhagen 	"C_5a7ef9db",
3426e91a93dSOswald Buddenhagen 	"C_00100000",
3436e91a93dSOswald Buddenhagen 	"GPR_ACCU",
3446e91a93dSOswald Buddenhagen 	"GPR_COND",
3456e91a93dSOswald Buddenhagen 	"GPR_NOISE0",
3466e91a93dSOswald Buddenhagen 	"GPR_NOISE1",
3476e91a93dSOswald Buddenhagen 	"GPR_IRQ",
3486e91a93dSOswald Buddenhagen 	"GPR_DBAC",
3496e91a93dSOswald Buddenhagen 	"GPR_DBACE",
3506e91a93dSOswald Buddenhagen 	"???",
3516e91a93dSOswald Buddenhagen };
3526e91a93dSOswald Buddenhagen 
3536e91a93dSOswald Buddenhagen static int disasm_emu10k1_reg(char *buffer,
3546e91a93dSOswald Buddenhagen 			      const struct emu10k1_reg_entry *entries,
3556e91a93dSOswald Buddenhagen 			      unsigned reg, const char *pfx)
3566e91a93dSOswald Buddenhagen {
3576e91a93dSOswald Buddenhagen 	for (int i = 0; ; i++) {
3586e91a93dSOswald Buddenhagen 		unsigned base = entries[i].base;
3596e91a93dSOswald Buddenhagen 		unsigned size = entries[i].size;
3606e91a93dSOswald Buddenhagen 		if (!size)
3616e91a93dSOswald Buddenhagen 			return sprintf(buffer, "%s0x%03x", pfx, reg);
3626e91a93dSOswald Buddenhagen 		if (reg >= base && reg < base + size) {
3636e91a93dSOswald Buddenhagen 			const char *name = entries[i].name;
3646e91a93dSOswald Buddenhagen 			reg -= base;
3656e91a93dSOswald Buddenhagen 			if (name)
3666e91a93dSOswald Buddenhagen 				return sprintf(buffer, "%s%s(%u)", pfx, name, reg);
3676e91a93dSOswald Buddenhagen 			return sprintf(buffer, "%s%s", pfx, emu10k1_const_entries[reg]);
3686e91a93dSOswald Buddenhagen 		}
3696e91a93dSOswald Buddenhagen 	}
3706e91a93dSOswald Buddenhagen }
3716e91a93dSOswald Buddenhagen 
3726e91a93dSOswald Buddenhagen static int disasm_sblive_reg(char *buffer, unsigned reg, const char *pfx)
3736e91a93dSOswald Buddenhagen {
3746e91a93dSOswald Buddenhagen 	return disasm_emu10k1_reg(buffer, sblive_reg_entries, reg, pfx);
3756e91a93dSOswald Buddenhagen }
3766e91a93dSOswald Buddenhagen 
3776e91a93dSOswald Buddenhagen static int disasm_audigy_reg(char *buffer, unsigned reg, const char *pfx)
3786e91a93dSOswald Buddenhagen {
3796e91a93dSOswald Buddenhagen 	return disasm_emu10k1_reg(buffer, audigy_reg_entries, reg, pfx);
3806e91a93dSOswald Buddenhagen }
3816e91a93dSOswald Buddenhagen 
382eb4698f3STakashi Iwai static void snd_emu10k1_proc_acode_read(struct snd_info_entry *entry,
383eb4698f3STakashi Iwai 				        struct snd_info_buffer *buffer)
3841da177e4SLinus Torvalds {
3851da177e4SLinus Torvalds 	u32 pc;
386eb4698f3STakashi Iwai 	struct snd_emu10k1 *emu = entry->private_data;
3876e91a93dSOswald Buddenhagen 	static const char * const insns[16] = {
3886e91a93dSOswald Buddenhagen 		"MAC0", "MAC1", "MAC2", "MAC3", "MACINT0", "MACINT1", "ACC3", "MACMV",
3896e91a93dSOswald Buddenhagen 		"ANDXOR", "TSTNEG", "LIMITGE", "LIMITLT", "LOG", "EXP", "INTERP", "SKIP",
3906e91a93dSOswald Buddenhagen 	};
3916e91a93dSOswald Buddenhagen 	static const char spaces[] = "                              ";
3926e91a93dSOswald Buddenhagen 	const int nspaces = sizeof(spaces) - 1;
3931da177e4SLinus Torvalds 
3941da177e4SLinus Torvalds 	snd_iprintf(buffer, "FX8010 Instruction List '%s'\n", emu->fx8010.name);
3951da177e4SLinus Torvalds 	snd_iprintf(buffer, "  Code dump      :\n");
3961da177e4SLinus Torvalds 	for (pc = 0; pc < (emu->audigy ? 1024 : 512); pc++) {
3971da177e4SLinus Torvalds 		u32 low, high;
3986e91a93dSOswald Buddenhagen 		int len;
3996e91a93dSOswald Buddenhagen 		char buf[100];
4006e91a93dSOswald Buddenhagen 		char *bufp = buf;
4011da177e4SLinus Torvalds 
4021da177e4SLinus Torvalds 		low = snd_emu10k1_efx_read(emu, pc * 2);
4031da177e4SLinus Torvalds 		high = snd_emu10k1_efx_read(emu, pc * 2 + 1);
4046e91a93dSOswald Buddenhagen 		if (emu->audigy) {
4056e91a93dSOswald Buddenhagen 			bufp += sprintf(bufp, "    %-7s  ", insns[(high >> 24) & 0x0f]);
4066e91a93dSOswald Buddenhagen 			bufp += disasm_audigy_reg(bufp, (high >> 12) & 0x7ff, "");
4076e91a93dSOswald Buddenhagen 			bufp += disasm_audigy_reg(bufp, (high >> 0) & 0x7ff, ", ");
4086e91a93dSOswald Buddenhagen 			bufp += disasm_audigy_reg(bufp, (low >> 12) & 0x7ff, ", ");
4096e91a93dSOswald Buddenhagen 			bufp += disasm_audigy_reg(bufp, (low >> 0) & 0x7ff, ", ");
4106e91a93dSOswald Buddenhagen 		} else {
4116e91a93dSOswald Buddenhagen 			bufp += sprintf(bufp, "    %-7s  ", insns[(high >> 20) & 0x0f]);
4126e91a93dSOswald Buddenhagen 			bufp += disasm_sblive_reg(bufp, (high >> 10) & 0x3ff, "");
4136e91a93dSOswald Buddenhagen 			bufp += disasm_sblive_reg(bufp, (high >> 0) & 0x3ff, ", ");
4146e91a93dSOswald Buddenhagen 			bufp += disasm_sblive_reg(bufp, (low >> 10) & 0x3ff, ", ");
4156e91a93dSOswald Buddenhagen 			bufp += disasm_sblive_reg(bufp, (low >> 0) & 0x3ff, ", ");
4166e91a93dSOswald Buddenhagen 		}
4176e91a93dSOswald Buddenhagen 		len = (int)(ptrdiff_t)(bufp - buf);
4186e91a93dSOswald Buddenhagen 		snd_iprintf(buffer, "%s %s /* 0x%04x: 0x%08x%08x */\n",
4196e91a93dSOswald Buddenhagen 			    buf, &spaces[nspaces - clamp(65 - len, 0, nspaces)],
4206e91a93dSOswald Buddenhagen 			    pc, high, low);
4211da177e4SLinus Torvalds 	}
4221da177e4SLinus Torvalds }
4231da177e4SLinus Torvalds 
4241da177e4SLinus Torvalds #define TOTAL_SIZE_GPR		(0x100*4)
4251da177e4SLinus Torvalds #define A_TOTAL_SIZE_GPR	(0x200*4)
4261da177e4SLinus Torvalds #define TOTAL_SIZE_TANKMEM_DATA	(0xa0*4)
4271da177e4SLinus Torvalds #define TOTAL_SIZE_TANKMEM_ADDR (0xa0*4)
4281da177e4SLinus Torvalds #define A_TOTAL_SIZE_TANKMEM_DATA (0x100*4)
4291da177e4SLinus Torvalds #define A_TOTAL_SIZE_TANKMEM_ADDR (0x100*4)
4301da177e4SLinus Torvalds #define TOTAL_SIZE_CODE		(0x200*8)
4311da177e4SLinus Torvalds #define A_TOTAL_SIZE_CODE	(0x400*8)
4321da177e4SLinus Torvalds 
43324e4a121STakashi Iwai static ssize_t snd_emu10k1_fx8010_read(struct snd_info_entry *entry,
434eb4698f3STakashi Iwai 				       void *file_private_data,
4351da177e4SLinus Torvalds 				       struct file *file, char __user *buf,
43624e4a121STakashi Iwai 				       size_t count, loff_t pos)
4371da177e4SLinus Torvalds {
438eb4698f3STakashi Iwai 	struct snd_emu10k1 *emu = entry->private_data;
4391da177e4SLinus Torvalds 	unsigned int offset;
4401da177e4SLinus Torvalds 	int tram_addr = 0;
441d97e1b78STakashi Iwai 	unsigned int *tmp;
442d97e1b78STakashi Iwai 	long res;
443d97e1b78STakashi Iwai 	unsigned int idx;
4441da177e4SLinus Torvalds 
4451da177e4SLinus Torvalds 	if (!strcmp(entry->name, "fx8010_tram_addr")) {
4461da177e4SLinus Torvalds 		offset = TANKMEMADDRREGBASE;
4471da177e4SLinus Torvalds 		tram_addr = 1;
4481da177e4SLinus Torvalds 	} else if (!strcmp(entry->name, "fx8010_tram_data")) {
4491da177e4SLinus Torvalds 		offset = TANKMEMDATAREGBASE;
4501da177e4SLinus Torvalds 	} else if (!strcmp(entry->name, "fx8010_code")) {
4511da177e4SLinus Torvalds 		offset = emu->audigy ? A_MICROCODEBASE : MICROCODEBASE;
4521da177e4SLinus Torvalds 	} else {
4531da177e4SLinus Torvalds 		offset = emu->audigy ? A_FXGPREGBASE : FXGPREGBASE;
4541da177e4SLinus Torvalds 	}
455d97e1b78STakashi Iwai 
456d97e1b78STakashi Iwai 	tmp = kmalloc(count + 8, GFP_KERNEL);
457d97e1b78STakashi Iwai 	if (!tmp)
4581da177e4SLinus Torvalds 		return -ENOMEM;
459d97e1b78STakashi Iwai 	for (idx = 0; idx < ((pos & 3) + count + 3) >> 2; idx++) {
460d97e1b78STakashi Iwai 		unsigned int val;
461d97e1b78STakashi Iwai 		val = snd_emu10k1_ptr_read(emu, offset + idx + (pos >> 2), 0);
4621da177e4SLinus Torvalds 		if (tram_addr && emu->audigy) {
463d97e1b78STakashi Iwai 			val >>= 11;
464d97e1b78STakashi Iwai 			val |= snd_emu10k1_ptr_read(emu, 0x100 + idx + (pos >> 2), 0) << 20;
4651da177e4SLinus Torvalds 		}
466d97e1b78STakashi Iwai 		tmp[idx] = val;
467d97e1b78STakashi Iwai 	}
468d97e1b78STakashi Iwai 	if (copy_to_user(buf, ((char *)tmp) + (pos & 3), count))
469d97e1b78STakashi Iwai 		res = -EFAULT;
470d97e1b78STakashi Iwai 	else
471d97e1b78STakashi Iwai 		res = count;
4721da177e4SLinus Torvalds 	kfree(tmp);
4731da177e4SLinus Torvalds 	return res;
4741da177e4SLinus Torvalds }
4751da177e4SLinus Torvalds 
476eb4698f3STakashi Iwai static void snd_emu10k1_proc_voices_read(struct snd_info_entry *entry,
477eb4698f3STakashi Iwai 				  struct snd_info_buffer *buffer)
4781da177e4SLinus Torvalds {
479eb4698f3STakashi Iwai 	struct snd_emu10k1 *emu = entry->private_data;
480eb4698f3STakashi Iwai 	struct snd_emu10k1_voice *voice;
4811da177e4SLinus Torvalds 	int idx;
482b840f8d8SOswald Buddenhagen 	static const char * const types[] = {
483b840f8d8SOswald Buddenhagen 		"Unused", "EFX", "EFX IRQ", "PCM", "PCM IRQ", "Synth"
484b840f8d8SOswald Buddenhagen 	};
485b840f8d8SOswald Buddenhagen 	static_assert(ARRAY_SIZE(types) == EMU10K1_NUM_TYPES);
4861da177e4SLinus Torvalds 
487a915d604SOswald Buddenhagen 	snd_iprintf(buffer, "ch\tdirty\tlast\tuse\n");
4881da177e4SLinus Torvalds 	for (idx = 0; idx < NUM_G; idx++) {
4891da177e4SLinus Torvalds 		voice = &emu->voices[idx];
490a915d604SOswald Buddenhagen 		snd_iprintf(buffer, "%i\t%u\t%u\t%s\n",
4911da177e4SLinus Torvalds 			idx,
49282a9fa6eSOswald Buddenhagen 			voice->dirty,
493a915d604SOswald Buddenhagen 			voice->last,
494b840f8d8SOswald Buddenhagen 			types[voice->use]);
4951da177e4SLinus Torvalds 	}
4961da177e4SLinus Torvalds }
4971da177e4SLinus Torvalds 
4981da177e4SLinus Torvalds #ifdef CONFIG_SND_DEBUG
499*6ab13291SOswald Buddenhagen 
500*6ab13291SOswald Buddenhagen static void snd_emu_proc_emu1010_link_read(struct snd_emu10k1 *emu,
501*6ab13291SOswald Buddenhagen 					   struct snd_info_buffer *buffer,
502*6ab13291SOswald Buddenhagen 					   u32 dst)
503*6ab13291SOswald Buddenhagen {
504*6ab13291SOswald Buddenhagen 	u32 src = snd_emu1010_fpga_link_dst_src_read(emu, dst);
505*6ab13291SOswald Buddenhagen 	snd_iprintf(buffer, "%04x: %04x\n", dst, src);
506*6ab13291SOswald Buddenhagen }
507*6ab13291SOswald Buddenhagen 
5089f4bd5ddSJames Courtier-Dutton static void snd_emu_proc_emu1010_reg_read(struct snd_info_entry *entry,
5099f4bd5ddSJames Courtier-Dutton 				     struct snd_info_buffer *buffer)
5109f4bd5ddSJames Courtier-Dutton {
5119f4bd5ddSJames Courtier-Dutton 	struct snd_emu10k1 *emu = entry->private_data;
5124677df07SHarvey Harrison 	u32 value;
5139f4bd5ddSJames Courtier-Dutton 	int i;
5149f4bd5ddSJames Courtier-Dutton 	snd_iprintf(buffer, "EMU1010 Registers:\n\n");
5159f4bd5ddSJames Courtier-Dutton 
51690fd5ce5SJames Courtier-Dutton 	for(i = 0; i < 0x40; i+=1) {
517f93abe51SJames Courtier-Dutton 		snd_emu1010_fpga_read(emu, i, &value);
518*6ab13291SOswald Buddenhagen 		snd_iprintf(buffer, "%02x: %02x\n", i, value);
519*6ab13291SOswald Buddenhagen 	}
520*6ab13291SOswald Buddenhagen 
521*6ab13291SOswald Buddenhagen 	snd_iprintf(buffer, "\nEMU1010 Routes:\n\n");
522*6ab13291SOswald Buddenhagen 
523*6ab13291SOswald Buddenhagen 	for (i = 0; i < 16; i++)  // To Alice2/Tina[2] via EMU32
524*6ab13291SOswald Buddenhagen 		snd_emu_proc_emu1010_link_read(emu, buffer, i);
525*6ab13291SOswald Buddenhagen 	if (emu->card_capabilities->emu_model != EMU_MODEL_EMU0404)
526*6ab13291SOswald Buddenhagen 		for (i = 0; i < 32; i++)  // To Dock via EDI
527*6ab13291SOswald Buddenhagen 			snd_emu_proc_emu1010_link_read(emu, buffer, 0x100 + i);
528*6ab13291SOswald Buddenhagen 	if (emu->card_capabilities->emu_model != EMU_MODEL_EMU1616)
529*6ab13291SOswald Buddenhagen 		for (i = 0; i < 8; i++)  // To Hamoa/local
530*6ab13291SOswald Buddenhagen 			snd_emu_proc_emu1010_link_read(emu, buffer, 0x200 + i);
531*6ab13291SOswald Buddenhagen 	for (i = 0; i < 8; i++)  // To Hamoa/Mana/local
532*6ab13291SOswald Buddenhagen 		snd_emu_proc_emu1010_link_read(emu, buffer, 0x300 + i);
533*6ab13291SOswald Buddenhagen 	if (emu->card_capabilities->emu_model == EMU_MODEL_EMU1616) {
534*6ab13291SOswald Buddenhagen 		for (i = 0; i < 16; i++)  // To Tina2 via EMU32
535*6ab13291SOswald Buddenhagen 			snd_emu_proc_emu1010_link_read(emu, buffer, 0x400 + i);
536*6ab13291SOswald Buddenhagen 	} else if (emu->card_capabilities->emu_model != EMU_MODEL_EMU0404) {
537*6ab13291SOswald Buddenhagen 		for (i = 0; i < 8; i++)  // To Hana ADAT
538*6ab13291SOswald Buddenhagen 			snd_emu_proc_emu1010_link_read(emu, buffer, 0x400 + i);
539*6ab13291SOswald Buddenhagen 		if (emu->card_capabilities->emu_model == EMU_MODEL_EMU1010B) {
540*6ab13291SOswald Buddenhagen 			for (i = 0; i < 16; i++)  // To Tina via EMU32
541*6ab13291SOswald Buddenhagen 				snd_emu_proc_emu1010_link_read(emu, buffer, 0x500 + i);
542*6ab13291SOswald Buddenhagen 		} else {
543*6ab13291SOswald Buddenhagen 			// To Alice2 via I2S
544*6ab13291SOswald Buddenhagen 			snd_emu_proc_emu1010_link_read(emu, buffer, 0x500);
545*6ab13291SOswald Buddenhagen 			snd_emu_proc_emu1010_link_read(emu, buffer, 0x501);
546*6ab13291SOswald Buddenhagen 			snd_emu_proc_emu1010_link_read(emu, buffer, 0x600);
547*6ab13291SOswald Buddenhagen 			snd_emu_proc_emu1010_link_read(emu, buffer, 0x601);
548*6ab13291SOswald Buddenhagen 			snd_emu_proc_emu1010_link_read(emu, buffer, 0x700);
549*6ab13291SOswald Buddenhagen 			snd_emu_proc_emu1010_link_read(emu, buffer, 0x701);
550*6ab13291SOswald Buddenhagen 		}
5519f4bd5ddSJames Courtier-Dutton 	}
5529f4bd5ddSJames Courtier-Dutton }
5539f4bd5ddSJames Courtier-Dutton 
554eb4698f3STakashi Iwai static void snd_emu_proc_io_reg_read(struct snd_info_entry *entry,
555eb4698f3STakashi Iwai 				     struct snd_info_buffer *buffer)
5561da177e4SLinus Torvalds {
557eb4698f3STakashi Iwai 	struct snd_emu10k1 *emu = entry->private_data;
5581da177e4SLinus Torvalds 	unsigned long value;
5591da177e4SLinus Torvalds 	int i;
5601da177e4SLinus Torvalds 	snd_iprintf(buffer, "IO Registers:\n\n");
5611da177e4SLinus Torvalds 	for(i = 0; i < 0x40; i+=4) {
5621da177e4SLinus Torvalds 		value = inl(emu->port + i);
5631da177e4SLinus Torvalds 		snd_iprintf(buffer, "%02X: %08lX\n", i, value);
5641da177e4SLinus Torvalds 	}
5651da177e4SLinus Torvalds }
5661da177e4SLinus Torvalds 
567eb4698f3STakashi Iwai static void snd_emu_proc_io_reg_write(struct snd_info_entry *entry,
568eb4698f3STakashi Iwai                                       struct snd_info_buffer *buffer)
5691da177e4SLinus Torvalds {
570eb4698f3STakashi Iwai 	struct snd_emu10k1 *emu = entry->private_data;
5711da177e4SLinus Torvalds 	char line[64];
5721da177e4SLinus Torvalds 	u32 reg, val;
5731da177e4SLinus Torvalds 	while (!snd_info_get_line(buffer, line, sizeof(line))) {
5741da177e4SLinus Torvalds 		if (sscanf(line, "%x %x", &reg, &val) != 2)
5751da177e4SLinus Torvalds 			continue;
57684ed1a19SRoel Kluin 		if (reg < 0x40 && val <= 0xffffffff) {
5771da177e4SLinus Torvalds 			outl(val, emu->port + (reg & 0xfffffffc));
5781da177e4SLinus Torvalds 		}
5791da177e4SLinus Torvalds 	}
5801da177e4SLinus Torvalds }
5811da177e4SLinus Torvalds 
582eb4698f3STakashi Iwai static unsigned int snd_ptr_read(struct snd_emu10k1 * emu,
5831da177e4SLinus Torvalds 				 unsigned int iobase,
5841da177e4SLinus Torvalds 				 unsigned int reg,
5851da177e4SLinus Torvalds 				 unsigned int chn)
5861da177e4SLinus Torvalds {
5871da177e4SLinus Torvalds 	unsigned long flags;
5881da177e4SLinus Torvalds 	unsigned int regptr, val;
5891da177e4SLinus Torvalds 
5901da177e4SLinus Torvalds 	regptr = (reg << 16) | chn;
5911da177e4SLinus Torvalds 
5921da177e4SLinus Torvalds 	spin_lock_irqsave(&emu->emu_lock, flags);
5931da177e4SLinus Torvalds 	outl(regptr, emu->port + iobase + PTR);
5941da177e4SLinus Torvalds 	val = inl(emu->port + iobase + DATA);
5951da177e4SLinus Torvalds 	spin_unlock_irqrestore(&emu->emu_lock, flags);
5961da177e4SLinus Torvalds 	return val;
5971da177e4SLinus Torvalds }
5981da177e4SLinus Torvalds 
599eb4698f3STakashi Iwai static void snd_ptr_write(struct snd_emu10k1 *emu,
6001da177e4SLinus Torvalds 			  unsigned int iobase,
6011da177e4SLinus Torvalds 			  unsigned int reg,
6021da177e4SLinus Torvalds 			  unsigned int chn,
6031da177e4SLinus Torvalds 			  unsigned int data)
6041da177e4SLinus Torvalds {
6051da177e4SLinus Torvalds 	unsigned int regptr;
6061da177e4SLinus Torvalds 	unsigned long flags;
6071da177e4SLinus Torvalds 
6081da177e4SLinus Torvalds 	regptr = (reg << 16) | chn;
6091da177e4SLinus Torvalds 
6101da177e4SLinus Torvalds 	spin_lock_irqsave(&emu->emu_lock, flags);
6111da177e4SLinus Torvalds 	outl(regptr, emu->port + iobase + PTR);
6121da177e4SLinus Torvalds 	outl(data, emu->port + iobase + DATA);
6131da177e4SLinus Torvalds 	spin_unlock_irqrestore(&emu->emu_lock, flags);
6141da177e4SLinus Torvalds }
6151da177e4SLinus Torvalds 
6161da177e4SLinus Torvalds 
617eb4698f3STakashi Iwai static void snd_emu_proc_ptr_reg_read(struct snd_info_entry *entry,
618eb4698f3STakashi Iwai 				      struct snd_info_buffer *buffer, int iobase, int offset, int length, int voices)
6191da177e4SLinus Torvalds {
620eb4698f3STakashi Iwai 	struct snd_emu10k1 *emu = entry->private_data;
6211da177e4SLinus Torvalds 	unsigned long value;
6221da177e4SLinus Torvalds 	int i,j;
6236ef7e861SJames Courtier-Dutton 	if (offset+length > 0xa0) {
6241da177e4SLinus Torvalds 		snd_iprintf(buffer, "Input values out of range\n");
6251da177e4SLinus Torvalds 		return;
6261da177e4SLinus Torvalds 	}
6271da177e4SLinus Torvalds 	snd_iprintf(buffer, "Registers 0x%x\n", iobase);
6281da177e4SLinus Torvalds 	for(i = offset; i < offset+length; i++) {
6291da177e4SLinus Torvalds 		snd_iprintf(buffer, "%02X: ",i);
6301da177e4SLinus Torvalds 		for (j = 0; j < voices; j++) {
631e922da40SOswald Buddenhagen 			value = snd_ptr_read(emu, iobase, i, j);
6321da177e4SLinus Torvalds 			snd_iprintf(buffer, "%08lX ", value);
6331da177e4SLinus Torvalds 		}
6341da177e4SLinus Torvalds 		snd_iprintf(buffer, "\n");
6351da177e4SLinus Torvalds 	}
6361da177e4SLinus Torvalds }
6371da177e4SLinus Torvalds 
638eb4698f3STakashi Iwai static void snd_emu_proc_ptr_reg_write(struct snd_info_entry *entry,
63967ff2addSOswald Buddenhagen 				       struct snd_info_buffer *buffer,
64067ff2addSOswald Buddenhagen 				       int iobase, int length, int voices)
6411da177e4SLinus Torvalds {
642eb4698f3STakashi Iwai 	struct snd_emu10k1 *emu = entry->private_data;
6431da177e4SLinus Torvalds 	char line[64];
6441da177e4SLinus Torvalds 	unsigned int reg, channel_id , val;
6451da177e4SLinus Torvalds 	while (!snd_info_get_line(buffer, line, sizeof(line))) {
6461da177e4SLinus Torvalds 		if (sscanf(line, "%x %x %x", &reg, &channel_id, &val) != 3)
6471da177e4SLinus Torvalds 			continue;
64867ff2addSOswald Buddenhagen 		if (reg < length && channel_id < voices)
6491da177e4SLinus Torvalds 			snd_ptr_write(emu, iobase, reg, channel_id, val);
6501da177e4SLinus Torvalds 	}
6511da177e4SLinus Torvalds }
6521da177e4SLinus Torvalds 
653eb4698f3STakashi Iwai static void snd_emu_proc_ptr_reg_write00(struct snd_info_entry *entry,
654eb4698f3STakashi Iwai 					 struct snd_info_buffer *buffer)
6551da177e4SLinus Torvalds {
65667ff2addSOswald Buddenhagen 	snd_emu_proc_ptr_reg_write(entry, buffer, 0, 0x80, 64);
6571da177e4SLinus Torvalds }
6581da177e4SLinus Torvalds 
659eb4698f3STakashi Iwai static void snd_emu_proc_ptr_reg_write20(struct snd_info_entry *entry,
660eb4698f3STakashi Iwai 					 struct snd_info_buffer *buffer)
6611da177e4SLinus Torvalds {
66267ff2addSOswald Buddenhagen 	struct snd_emu10k1 *emu = entry->private_data;
66367ff2addSOswald Buddenhagen 	snd_emu_proc_ptr_reg_write(entry, buffer, 0x20,
66467ff2addSOswald Buddenhagen 				   emu->card_capabilities->ca0108_chip ? 0xa0 : 0x80, 4);
6651da177e4SLinus Torvalds }
6661da177e4SLinus Torvalds 
6671da177e4SLinus Torvalds 
668eb4698f3STakashi Iwai static void snd_emu_proc_ptr_reg_read00a(struct snd_info_entry *entry,
669eb4698f3STakashi Iwai 					 struct snd_info_buffer *buffer)
6701da177e4SLinus Torvalds {
6711da177e4SLinus Torvalds 	snd_emu_proc_ptr_reg_read(entry, buffer, 0, 0, 0x40, 64);
6721da177e4SLinus Torvalds }
6731da177e4SLinus Torvalds 
674eb4698f3STakashi Iwai static void snd_emu_proc_ptr_reg_read00b(struct snd_info_entry *entry,
675eb4698f3STakashi Iwai 					 struct snd_info_buffer *buffer)
6761da177e4SLinus Torvalds {
6771da177e4SLinus Torvalds 	snd_emu_proc_ptr_reg_read(entry, buffer, 0, 0x40, 0x40, 64);
6781da177e4SLinus Torvalds }
6791da177e4SLinus Torvalds 
680eb4698f3STakashi Iwai static void snd_emu_proc_ptr_reg_read20a(struct snd_info_entry *entry,
681eb4698f3STakashi Iwai 					 struct snd_info_buffer *buffer)
6821da177e4SLinus Torvalds {
6831da177e4SLinus Torvalds 	snd_emu_proc_ptr_reg_read(entry, buffer, 0x20, 0, 0x40, 4);
6841da177e4SLinus Torvalds }
6851da177e4SLinus Torvalds 
686eb4698f3STakashi Iwai static void snd_emu_proc_ptr_reg_read20b(struct snd_info_entry *entry,
687eb4698f3STakashi Iwai 					 struct snd_info_buffer *buffer)
6881da177e4SLinus Torvalds {
6891da177e4SLinus Torvalds 	snd_emu_proc_ptr_reg_read(entry, buffer, 0x20, 0x40, 0x40, 4);
6901da177e4SLinus Torvalds }
6916ef7e861SJames Courtier-Dutton 
692eb4698f3STakashi Iwai static void snd_emu_proc_ptr_reg_read20c(struct snd_info_entry *entry,
693eb4698f3STakashi Iwai 					 struct snd_info_buffer * buffer)
6946ef7e861SJames Courtier-Dutton {
6956ef7e861SJames Courtier-Dutton 	snd_emu_proc_ptr_reg_read(entry, buffer, 0x20, 0x80, 0x20, 4);
6966ef7e861SJames Courtier-Dutton }
6971da177e4SLinus Torvalds #endif
6981da177e4SLinus Torvalds 
699d25ff268STakashi Iwai static const struct snd_info_entry_ops snd_emu10k1_proc_ops_fx8010 = {
7001da177e4SLinus Torvalds 	.read = snd_emu10k1_fx8010_read,
7011da177e4SLinus Torvalds };
7021da177e4SLinus Torvalds 
703e23e7a14SBill Pemberton int snd_emu10k1_proc_init(struct snd_emu10k1 *emu)
7041da177e4SLinus Torvalds {
705eb4698f3STakashi Iwai 	struct snd_info_entry *entry;
7061da177e4SLinus Torvalds #ifdef CONFIG_SND_DEBUG
707190d2c46SJames Courtier-Dutton 	if (emu->card_capabilities->emu_model) {
70847f2769bSTakashi Iwai 		snd_card_ro_proc_new(emu->card, "emu1010_regs",
70947f2769bSTakashi Iwai 				     emu, snd_emu_proc_emu1010_reg_read);
7109f4bd5ddSJames Courtier-Dutton 	}
71147f2769bSTakashi Iwai 	snd_card_rw_proc_new(emu->card, "io_regs", emu,
71247f2769bSTakashi Iwai 			     snd_emu_proc_io_reg_read,
71347f2769bSTakashi Iwai 			     snd_emu_proc_io_reg_write);
71447f2769bSTakashi Iwai 	snd_card_rw_proc_new(emu->card, "ptr_regs00a", emu,
71547f2769bSTakashi Iwai 			     snd_emu_proc_ptr_reg_read00a,
71647f2769bSTakashi Iwai 			     snd_emu_proc_ptr_reg_write00);
71747f2769bSTakashi Iwai 	snd_card_rw_proc_new(emu->card, "ptr_regs00b", emu,
71847f2769bSTakashi Iwai 			     snd_emu_proc_ptr_reg_read00b,
71947f2769bSTakashi Iwai 			     snd_emu_proc_ptr_reg_write00);
720219153c6SOswald Buddenhagen 	if (!emu->card_capabilities->emu_model &&
721219153c6SOswald Buddenhagen 	    (emu->card_capabilities->ca0151_chip || emu->card_capabilities->ca0108_chip)) {
72247f2769bSTakashi Iwai 		snd_card_rw_proc_new(emu->card, "ptr_regs20a", emu,
72347f2769bSTakashi Iwai 				     snd_emu_proc_ptr_reg_read20a,
72447f2769bSTakashi Iwai 				     snd_emu_proc_ptr_reg_write20);
72547f2769bSTakashi Iwai 		snd_card_rw_proc_new(emu->card, "ptr_regs20b", emu,
72647f2769bSTakashi Iwai 				     snd_emu_proc_ptr_reg_read20b,
72747f2769bSTakashi Iwai 				     snd_emu_proc_ptr_reg_write20);
728219153c6SOswald Buddenhagen 		if (emu->card_capabilities->ca0108_chip)
72947f2769bSTakashi Iwai 			snd_card_rw_proc_new(emu->card, "ptr_regs20c", emu,
73047f2769bSTakashi Iwai 					     snd_emu_proc_ptr_reg_read20c,
73147f2769bSTakashi Iwai 					     snd_emu_proc_ptr_reg_write20);
732219153c6SOswald Buddenhagen 	}
7331da177e4SLinus Torvalds #endif
7341da177e4SLinus Torvalds 
73547f2769bSTakashi Iwai 	snd_card_ro_proc_new(emu->card, "emu10k1", emu, snd_emu10k1_proc_read);
7361da177e4SLinus Torvalds 
73747f2769bSTakashi Iwai 	if (emu->card_capabilities->emu10k2_chip)
73847f2769bSTakashi Iwai 		snd_card_ro_proc_new(emu->card, "spdif-in", emu,
73947f2769bSTakashi Iwai 				     snd_emu10k1_proc_spdif_read);
74047f2769bSTakashi Iwai 	if (emu->card_capabilities->ca0151_chip)
74147f2769bSTakashi Iwai 		snd_card_ro_proc_new(emu->card, "capture-rates", emu,
74247f2769bSTakashi Iwai 				     snd_emu10k1_proc_rates_read);
743df34140aSJames Courtier-Dutton 
74447f2769bSTakashi Iwai 	snd_card_ro_proc_new(emu->card, "voices", emu,
74547f2769bSTakashi Iwai 			     snd_emu10k1_proc_voices_read);
7461da177e4SLinus Torvalds 
7471da177e4SLinus Torvalds 	if (! snd_card_proc_new(emu->card, "fx8010_gpr", &entry)) {
7481da177e4SLinus Torvalds 		entry->content = SNDRV_INFO_CONTENT_DATA;
7491da177e4SLinus Torvalds 		entry->private_data = emu;
7506a73cf46SJoe Perches 		entry->mode = S_IFREG | 0444 /*| S_IWUSR*/;
7511da177e4SLinus Torvalds 		entry->size = emu->audigy ? A_TOTAL_SIZE_GPR : TOTAL_SIZE_GPR;
7521da177e4SLinus Torvalds 		entry->c.ops = &snd_emu10k1_proc_ops_fx8010;
7531da177e4SLinus Torvalds 	}
7541da177e4SLinus Torvalds 	if (! snd_card_proc_new(emu->card, "fx8010_tram_data", &entry)) {
7551da177e4SLinus Torvalds 		entry->content = SNDRV_INFO_CONTENT_DATA;
7561da177e4SLinus Torvalds 		entry->private_data = emu;
7576a73cf46SJoe Perches 		entry->mode = S_IFREG | 0444 /*| S_IWUSR*/;
7581da177e4SLinus Torvalds 		entry->size = emu->audigy ? A_TOTAL_SIZE_TANKMEM_DATA : TOTAL_SIZE_TANKMEM_DATA ;
7591da177e4SLinus Torvalds 		entry->c.ops = &snd_emu10k1_proc_ops_fx8010;
7601da177e4SLinus Torvalds 	}
7611da177e4SLinus Torvalds 	if (! snd_card_proc_new(emu->card, "fx8010_tram_addr", &entry)) {
7621da177e4SLinus Torvalds 		entry->content = SNDRV_INFO_CONTENT_DATA;
7631da177e4SLinus Torvalds 		entry->private_data = emu;
7646a73cf46SJoe Perches 		entry->mode = S_IFREG | 0444 /*| S_IWUSR*/;
7651da177e4SLinus Torvalds 		entry->size = emu->audigy ? A_TOTAL_SIZE_TANKMEM_ADDR : TOTAL_SIZE_TANKMEM_ADDR ;
7661da177e4SLinus Torvalds 		entry->c.ops = &snd_emu10k1_proc_ops_fx8010;
7671da177e4SLinus Torvalds 	}
7681da177e4SLinus Torvalds 	if (! snd_card_proc_new(emu->card, "fx8010_code", &entry)) {
7691da177e4SLinus Torvalds 		entry->content = SNDRV_INFO_CONTENT_DATA;
7701da177e4SLinus Torvalds 		entry->private_data = emu;
7716a73cf46SJoe Perches 		entry->mode = S_IFREG | 0444 /*| S_IWUSR*/;
7721da177e4SLinus Torvalds 		entry->size = emu->audigy ? A_TOTAL_SIZE_CODE : TOTAL_SIZE_CODE;
7731da177e4SLinus Torvalds 		entry->c.ops = &snd_emu10k1_proc_ops_fx8010;
7741da177e4SLinus Torvalds 	}
77547f2769bSTakashi Iwai 	snd_card_ro_proc_new(emu->card, "fx8010_acode", emu,
77647f2769bSTakashi Iwai 			     snd_emu10k1_proc_acode_read);
7771da177e4SLinus Torvalds 	return 0;
7781da177e4SLinus Torvalds }
779