xref: /openbmc/linux/sound/pci/emu10k1/emuproc.c (revision 6e91a93d1e7417e5f700fb1d10de994d1539de8e)
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;
1731da177e4SLinus Torvalds 	unsigned int val, val1;
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);
1831da177e4SLinus Torvalds 	snd_iprintf(buffer, "\n");
1841da177e4SLinus Torvalds 	snd_iprintf(buffer, "Effect Send Routing   :\n");
1851da177e4SLinus Torvalds 	for (idx = 0; idx < NUM_G; idx++) {
1861da177e4SLinus Torvalds 		val = emu->audigy ?
1871da177e4SLinus Torvalds 			snd_emu10k1_ptr_read(emu, A_FXRT1, idx) :
1881da177e4SLinus Torvalds 			snd_emu10k1_ptr_read(emu, FXRT, idx);
1891da177e4SLinus Torvalds 		val1 = emu->audigy ?
1901da177e4SLinus Torvalds 			snd_emu10k1_ptr_read(emu, A_FXRT2, idx) :
1911da177e4SLinus Torvalds 			0;
1921da177e4SLinus Torvalds 		if (emu->audigy) {
1931da177e4SLinus Torvalds 			snd_iprintf(buffer, "Ch%i: A=%i, B=%i, C=%i, D=%i, ",
1941da177e4SLinus Torvalds 				idx,
1951da177e4SLinus Torvalds 				val & 0x3f,
1961da177e4SLinus Torvalds 				(val >> 8) & 0x3f,
1971da177e4SLinus Torvalds 				(val >> 16) & 0x3f,
1981da177e4SLinus Torvalds 				(val >> 24) & 0x3f);
1991da177e4SLinus Torvalds 			snd_iprintf(buffer, "E=%i, F=%i, G=%i, H=%i\n",
2001da177e4SLinus Torvalds 				val1 & 0x3f,
2011da177e4SLinus Torvalds 				(val1 >> 8) & 0x3f,
2021da177e4SLinus Torvalds 				(val1 >> 16) & 0x3f,
2031da177e4SLinus Torvalds 				(val1 >> 24) & 0x3f);
2041da177e4SLinus Torvalds 		} else {
2051da177e4SLinus Torvalds 			snd_iprintf(buffer, "Ch%i: A=%i, B=%i, C=%i, D=%i\n",
2061da177e4SLinus Torvalds 				idx,
2071da177e4SLinus Torvalds 				(val >> 16) & 0x0f,
2081da177e4SLinus Torvalds 				(val >> 20) & 0x0f,
2091da177e4SLinus Torvalds 				(val >> 24) & 0x0f,
2101da177e4SLinus Torvalds 				(val >> 28) & 0x0f);
2111da177e4SLinus Torvalds 		}
2121da177e4SLinus Torvalds 	}
2131da177e4SLinus Torvalds 	snd_iprintf(buffer, "\nCaptured FX Outputs   :\n");
2141da177e4SLinus Torvalds 	for (idx = 0; idx < nefx; idx++) {
2151da177e4SLinus Torvalds 		if (emu->efx_voices_mask[idx/32] & (1 << (idx%32)))
2161da177e4SLinus Torvalds 			snd_iprintf(buffer, "  Output %02i [%s]\n", idx, outputs[idx]);
2171da177e4SLinus Torvalds 	}
2181da177e4SLinus Torvalds 	snd_iprintf(buffer, "\nAll FX Outputs        :\n");
2191da177e4SLinus Torvalds 	for (idx = 0; idx < (emu->audigy ? 64 : 32); idx++)
2201da177e4SLinus Torvalds 		snd_iprintf(buffer, "  Output %02i [%s]\n", idx, outputs[idx]);
221df34140aSJames Courtier-Dutton }
222df34140aSJames Courtier-Dutton 
223eb4698f3STakashi Iwai static void snd_emu10k1_proc_spdif_read(struct snd_info_entry *entry,
224eb4698f3STakashi Iwai 				  struct snd_info_buffer *buffer)
225df34140aSJames Courtier-Dutton {
226eb4698f3STakashi Iwai 	struct snd_emu10k1 *emu = entry->private_data;
227f93abe51SJames Courtier-Dutton 	u32 value;
228f93abe51SJames Courtier-Dutton 	u32 value2;
229f93abe51SJames Courtier-Dutton 	u32 rate;
230f93abe51SJames Courtier-Dutton 
231190d2c46SJames Courtier-Dutton 	if (emu->card_capabilities->emu_model) {
2326f3609f8SOswald Buddenhagen 		if (!emu->card_capabilities->no_adat) {
233f93abe51SJames Courtier-Dutton 			snd_emu1010_fpga_read(emu, 0x38, &value);
234f93abe51SJames Courtier-Dutton 			if ((value & 0x1) == 0) {
235f93abe51SJames Courtier-Dutton 				snd_emu1010_fpga_read(emu, 0x2a, &value);
236f93abe51SJames Courtier-Dutton 				snd_emu1010_fpga_read(emu, 0x2b, &value2);
237f93abe51SJames Courtier-Dutton 				rate = 0x1770000 / (((value << 5) | value2)+1);
238f93abe51SJames Courtier-Dutton 				snd_iprintf(buffer, "ADAT Locked : %u\n", rate);
239f93abe51SJames Courtier-Dutton 			} else {
240f93abe51SJames Courtier-Dutton 				snd_iprintf(buffer, "ADAT Unlocked\n");
241f93abe51SJames Courtier-Dutton 			}
2426f3609f8SOswald Buddenhagen 		}
243f93abe51SJames Courtier-Dutton 		snd_emu1010_fpga_read(emu, 0x20, &value);
244f93abe51SJames Courtier-Dutton 		if ((value & 0x4) == 0) {
245f93abe51SJames Courtier-Dutton 			snd_emu1010_fpga_read(emu, 0x28, &value);
246f93abe51SJames Courtier-Dutton 			snd_emu1010_fpga_read(emu, 0x29, &value2);
247f93abe51SJames Courtier-Dutton 			rate = 0x1770000 / (((value << 5) | value2)+1);
248f93abe51SJames Courtier-Dutton 			snd_iprintf(buffer, "SPDIF Locked : %d\n", rate);
249f93abe51SJames Courtier-Dutton 		} else {
250f93abe51SJames Courtier-Dutton 			snd_iprintf(buffer, "SPDIF Unlocked\n");
251f93abe51SJames Courtier-Dutton 		}
252f93abe51SJames Courtier-Dutton 	} else {
253df34140aSJames Courtier-Dutton 		snd_emu10k1_proc_spdif_status(emu, buffer, "CD-ROM S/PDIF In", CDCS, CDSRCS);
254df34140aSJames Courtier-Dutton 		snd_emu10k1_proc_spdif_status(emu, buffer, "Optical or Coax S/PDIF In", GPSCS, GPSRCS);
255f93abe51SJames Courtier-Dutton 	}
256df34140aSJames Courtier-Dutton #if 0
2571da177e4SLinus Torvalds 	val = snd_emu10k1_ptr_read(emu, ZVSRCS, 0);
2581da177e4SLinus Torvalds 	snd_iprintf(buffer, "\nZoomed Video\n");
2591da177e4SLinus Torvalds 	snd_iprintf(buffer, "Rate Locked           : %s\n", val & SRCS_RATELOCKED ? "on" : "off");
2601da177e4SLinus Torvalds 	snd_iprintf(buffer, "Estimated Sample Rate : 0x%x\n", val & SRCS_ESTSAMPLERATE);
261df34140aSJames Courtier-Dutton #endif
2621da177e4SLinus Torvalds }
2631da177e4SLinus Torvalds 
264eb4698f3STakashi Iwai static void snd_emu10k1_proc_rates_read(struct snd_info_entry *entry,
265eb4698f3STakashi Iwai 				  struct snd_info_buffer *buffer)
266001f7589SJames Courtier-Dutton {
2676fddce26STakashi Iwai 	static const int samplerate[8] = { 44100, 48000, 96000, 192000, 4, 5, 6, 7 };
268eb4698f3STakashi Iwai 	struct snd_emu10k1 *emu = entry->private_data;
269001f7589SJames Courtier-Dutton 	unsigned int val, tmp, n;
270001f7589SJames Courtier-Dutton 	val = snd_emu10k1_ptr20_read(emu, CAPTURE_RATE_STATUS, 0);
271001f7589SJames Courtier-Dutton 	for (n = 0; n < 4; n++) {
272001f7589SJames Courtier-Dutton 		tmp = val >> (16 + (n*4));
273001f7589SJames Courtier-Dutton 		if (tmp & 0x8) snd_iprintf(buffer, "Channel %d: Rate=%d\n", n, samplerate[tmp & 0x7]);
274001f7589SJames Courtier-Dutton 		else snd_iprintf(buffer, "Channel %d: No input\n", n);
275001f7589SJames Courtier-Dutton 	}
276001f7589SJames Courtier-Dutton }
277001f7589SJames Courtier-Dutton 
278*6e91a93dSOswald Buddenhagen struct emu10k1_reg_entry {
279*6e91a93dSOswald Buddenhagen 	unsigned short base, size;
280*6e91a93dSOswald Buddenhagen 	const char *name;
281*6e91a93dSOswald Buddenhagen };
282*6e91a93dSOswald Buddenhagen 
283*6e91a93dSOswald Buddenhagen static const struct emu10k1_reg_entry sblive_reg_entries[] = {
284*6e91a93dSOswald Buddenhagen 	{    0, 0x10, "FXBUS" },
285*6e91a93dSOswald Buddenhagen 	{ 0x10, 0x10, "EXTIN" },
286*6e91a93dSOswald Buddenhagen 	{ 0x20, 0x10, "EXTOUT" },
287*6e91a93dSOswald Buddenhagen 	{ 0x30, 0x10, "FXBUS2" },
288*6e91a93dSOswald Buddenhagen 	{ 0x40, 0x20, NULL },  // Constants
289*6e91a93dSOswald Buddenhagen 	{ 0x100, 0x100, "GPR" },
290*6e91a93dSOswald Buddenhagen 	{ 0x200, 0x80, "ITRAM_DATA" },
291*6e91a93dSOswald Buddenhagen 	{ 0x280, 0x20, "ETRAM_DATA" },
292*6e91a93dSOswald Buddenhagen 	{ 0x300, 0x80, "ITRAM_ADDR" },
293*6e91a93dSOswald Buddenhagen 	{ 0x380, 0x20, "ETRAM_ADDR" },
294*6e91a93dSOswald Buddenhagen 	{ 0x400, 0, NULL }
295*6e91a93dSOswald Buddenhagen };
296*6e91a93dSOswald Buddenhagen 
297*6e91a93dSOswald Buddenhagen static const struct emu10k1_reg_entry audigy_reg_entries[] = {
298*6e91a93dSOswald Buddenhagen 	{    0, 0x40, "FXBUS" },
299*6e91a93dSOswald Buddenhagen 	{ 0x40, 0x10, "EXTIN" },
300*6e91a93dSOswald Buddenhagen 	{ 0x50, 0x10, "P16VIN" },
301*6e91a93dSOswald Buddenhagen 	{ 0x60, 0x20, "EXTOUT" },
302*6e91a93dSOswald Buddenhagen 	{ 0x80, 0x20, "FXBUS2" },
303*6e91a93dSOswald Buddenhagen 	{ 0xa0, 0x10, "EMU32OUTH" },
304*6e91a93dSOswald Buddenhagen 	{ 0xb0, 0x10, "EMU32OUTL" },
305*6e91a93dSOswald Buddenhagen 	{ 0xc0, 0x20, NULL },  // Constants
306*6e91a93dSOswald Buddenhagen 	// This can't be quite right - overlap.
307*6e91a93dSOswald Buddenhagen 	//{ 0x100, 0xc0, "ITRAM_CTL" },
308*6e91a93dSOswald Buddenhagen 	//{ 0x1c0, 0x40, "ETRAM_CTL" },
309*6e91a93dSOswald Buddenhagen 	{ 0x160, 0x20, "A3_EMU32IN" },
310*6e91a93dSOswald Buddenhagen 	{ 0x1e0, 0x20, "A3_EMU32OUT" },
311*6e91a93dSOswald Buddenhagen 	{ 0x200, 0xc0, "ITRAM_DATA" },
312*6e91a93dSOswald Buddenhagen 	{ 0x2c0, 0x40, "ETRAM_DATA" },
313*6e91a93dSOswald Buddenhagen 	{ 0x300, 0xc0, "ITRAM_ADDR" },
314*6e91a93dSOswald Buddenhagen 	{ 0x3c0, 0x40, "ETRAM_ADDR" },
315*6e91a93dSOswald Buddenhagen 	{ 0x400, 0x200, "GPR" },
316*6e91a93dSOswald Buddenhagen 	{ 0x600, 0, NULL }
317*6e91a93dSOswald Buddenhagen };
318*6e91a93dSOswald Buddenhagen 
319*6e91a93dSOswald Buddenhagen static const char * const emu10k1_const_entries[] = {
320*6e91a93dSOswald Buddenhagen 	"C_00000000",
321*6e91a93dSOswald Buddenhagen 	"C_00000001",
322*6e91a93dSOswald Buddenhagen 	"C_00000002",
323*6e91a93dSOswald Buddenhagen 	"C_00000003",
324*6e91a93dSOswald Buddenhagen 	"C_00000004",
325*6e91a93dSOswald Buddenhagen 	"C_00000008",
326*6e91a93dSOswald Buddenhagen 	"C_00000010",
327*6e91a93dSOswald Buddenhagen 	"C_00000020",
328*6e91a93dSOswald Buddenhagen 	"C_00000100",
329*6e91a93dSOswald Buddenhagen 	"C_00010000",
330*6e91a93dSOswald Buddenhagen 	"C_00000800",
331*6e91a93dSOswald Buddenhagen 	"C_10000000",
332*6e91a93dSOswald Buddenhagen 	"C_20000000",
333*6e91a93dSOswald Buddenhagen 	"C_40000000",
334*6e91a93dSOswald Buddenhagen 	"C_80000000",
335*6e91a93dSOswald Buddenhagen 	"C_7fffffff",
336*6e91a93dSOswald Buddenhagen 	"C_ffffffff",
337*6e91a93dSOswald Buddenhagen 	"C_fffffffe",
338*6e91a93dSOswald Buddenhagen 	"C_c0000000",
339*6e91a93dSOswald Buddenhagen 	"C_4f1bbcdc",
340*6e91a93dSOswald Buddenhagen 	"C_5a7ef9db",
341*6e91a93dSOswald Buddenhagen 	"C_00100000",
342*6e91a93dSOswald Buddenhagen 	"GPR_ACCU",
343*6e91a93dSOswald Buddenhagen 	"GPR_COND",
344*6e91a93dSOswald Buddenhagen 	"GPR_NOISE0",
345*6e91a93dSOswald Buddenhagen 	"GPR_NOISE1",
346*6e91a93dSOswald Buddenhagen 	"GPR_IRQ",
347*6e91a93dSOswald Buddenhagen 	"GPR_DBAC",
348*6e91a93dSOswald Buddenhagen 	"GPR_DBACE",
349*6e91a93dSOswald Buddenhagen 	"???",
350*6e91a93dSOswald Buddenhagen };
351*6e91a93dSOswald Buddenhagen 
352*6e91a93dSOswald Buddenhagen static int disasm_emu10k1_reg(char *buffer,
353*6e91a93dSOswald Buddenhagen 			      const struct emu10k1_reg_entry *entries,
354*6e91a93dSOswald Buddenhagen 			      unsigned reg, const char *pfx)
355*6e91a93dSOswald Buddenhagen {
356*6e91a93dSOswald Buddenhagen 	for (int i = 0; ; i++) {
357*6e91a93dSOswald Buddenhagen 		unsigned base = entries[i].base;
358*6e91a93dSOswald Buddenhagen 		unsigned size = entries[i].size;
359*6e91a93dSOswald Buddenhagen 		if (!size)
360*6e91a93dSOswald Buddenhagen 			return sprintf(buffer, "%s0x%03x", pfx, reg);
361*6e91a93dSOswald Buddenhagen 		if (reg >= base && reg < base + size) {
362*6e91a93dSOswald Buddenhagen 			const char *name = entries[i].name;
363*6e91a93dSOswald Buddenhagen 			reg -= base;
364*6e91a93dSOswald Buddenhagen 			if (name)
365*6e91a93dSOswald Buddenhagen 				return sprintf(buffer, "%s%s(%u)", pfx, name, reg);
366*6e91a93dSOswald Buddenhagen 			return sprintf(buffer, "%s%s", pfx, emu10k1_const_entries[reg]);
367*6e91a93dSOswald Buddenhagen 		}
368*6e91a93dSOswald Buddenhagen 	}
369*6e91a93dSOswald Buddenhagen }
370*6e91a93dSOswald Buddenhagen 
371*6e91a93dSOswald Buddenhagen static int disasm_sblive_reg(char *buffer, unsigned reg, const char *pfx)
372*6e91a93dSOswald Buddenhagen {
373*6e91a93dSOswald Buddenhagen 	return disasm_emu10k1_reg(buffer, sblive_reg_entries, reg, pfx);
374*6e91a93dSOswald Buddenhagen }
375*6e91a93dSOswald Buddenhagen 
376*6e91a93dSOswald Buddenhagen static int disasm_audigy_reg(char *buffer, unsigned reg, const char *pfx)
377*6e91a93dSOswald Buddenhagen {
378*6e91a93dSOswald Buddenhagen 	return disasm_emu10k1_reg(buffer, audigy_reg_entries, reg, pfx);
379*6e91a93dSOswald Buddenhagen }
380*6e91a93dSOswald Buddenhagen 
381eb4698f3STakashi Iwai static void snd_emu10k1_proc_acode_read(struct snd_info_entry *entry,
382eb4698f3STakashi Iwai 				        struct snd_info_buffer *buffer)
3831da177e4SLinus Torvalds {
3841da177e4SLinus Torvalds 	u32 pc;
385eb4698f3STakashi Iwai 	struct snd_emu10k1 *emu = entry->private_data;
386*6e91a93dSOswald Buddenhagen 	static const char * const insns[16] = {
387*6e91a93dSOswald Buddenhagen 		"MAC0", "MAC1", "MAC2", "MAC3", "MACINT0", "MACINT1", "ACC3", "MACMV",
388*6e91a93dSOswald Buddenhagen 		"ANDXOR", "TSTNEG", "LIMITGE", "LIMITLT", "LOG", "EXP", "INTERP", "SKIP",
389*6e91a93dSOswald Buddenhagen 	};
390*6e91a93dSOswald Buddenhagen 	static const char spaces[] = "                              ";
391*6e91a93dSOswald Buddenhagen 	const int nspaces = sizeof(spaces) - 1;
3921da177e4SLinus Torvalds 
3931da177e4SLinus Torvalds 	snd_iprintf(buffer, "FX8010 Instruction List '%s'\n", emu->fx8010.name);
3941da177e4SLinus Torvalds 	snd_iprintf(buffer, "  Code dump      :\n");
3951da177e4SLinus Torvalds 	for (pc = 0; pc < (emu->audigy ? 1024 : 512); pc++) {
3961da177e4SLinus Torvalds 		u32 low, high;
397*6e91a93dSOswald Buddenhagen 		int len;
398*6e91a93dSOswald Buddenhagen 		char buf[100];
399*6e91a93dSOswald Buddenhagen 		char *bufp = buf;
4001da177e4SLinus Torvalds 
4011da177e4SLinus Torvalds 		low = snd_emu10k1_efx_read(emu, pc * 2);
4021da177e4SLinus Torvalds 		high = snd_emu10k1_efx_read(emu, pc * 2 + 1);
403*6e91a93dSOswald Buddenhagen 		if (emu->audigy) {
404*6e91a93dSOswald Buddenhagen 			bufp += sprintf(bufp, "    %-7s  ", insns[(high >> 24) & 0x0f]);
405*6e91a93dSOswald Buddenhagen 			bufp += disasm_audigy_reg(bufp, (high >> 12) & 0x7ff, "");
406*6e91a93dSOswald Buddenhagen 			bufp += disasm_audigy_reg(bufp, (high >> 0) & 0x7ff, ", ");
407*6e91a93dSOswald Buddenhagen 			bufp += disasm_audigy_reg(bufp, (low >> 12) & 0x7ff, ", ");
408*6e91a93dSOswald Buddenhagen 			bufp += disasm_audigy_reg(bufp, (low >> 0) & 0x7ff, ", ");
409*6e91a93dSOswald Buddenhagen 		} else {
410*6e91a93dSOswald Buddenhagen 			bufp += sprintf(bufp, "    %-7s  ", insns[(high >> 20) & 0x0f]);
411*6e91a93dSOswald Buddenhagen 			bufp += disasm_sblive_reg(bufp, (high >> 10) & 0x3ff, "");
412*6e91a93dSOswald Buddenhagen 			bufp += disasm_sblive_reg(bufp, (high >> 0) & 0x3ff, ", ");
413*6e91a93dSOswald Buddenhagen 			bufp += disasm_sblive_reg(bufp, (low >> 10) & 0x3ff, ", ");
414*6e91a93dSOswald Buddenhagen 			bufp += disasm_sblive_reg(bufp, (low >> 0) & 0x3ff, ", ");
415*6e91a93dSOswald Buddenhagen 		}
416*6e91a93dSOswald Buddenhagen 		len = (int)(ptrdiff_t)(bufp - buf);
417*6e91a93dSOswald Buddenhagen 		snd_iprintf(buffer, "%s %s /* 0x%04x: 0x%08x%08x */\n",
418*6e91a93dSOswald Buddenhagen 			    buf, &spaces[nspaces - clamp(65 - len, 0, nspaces)],
419*6e91a93dSOswald Buddenhagen 			    pc, high, low);
4201da177e4SLinus Torvalds 	}
4211da177e4SLinus Torvalds }
4221da177e4SLinus Torvalds 
4231da177e4SLinus Torvalds #define TOTAL_SIZE_GPR		(0x100*4)
4241da177e4SLinus Torvalds #define A_TOTAL_SIZE_GPR	(0x200*4)
4251da177e4SLinus Torvalds #define TOTAL_SIZE_TANKMEM_DATA	(0xa0*4)
4261da177e4SLinus Torvalds #define TOTAL_SIZE_TANKMEM_ADDR (0xa0*4)
4271da177e4SLinus Torvalds #define A_TOTAL_SIZE_TANKMEM_DATA (0x100*4)
4281da177e4SLinus Torvalds #define A_TOTAL_SIZE_TANKMEM_ADDR (0x100*4)
4291da177e4SLinus Torvalds #define TOTAL_SIZE_CODE		(0x200*8)
4301da177e4SLinus Torvalds #define A_TOTAL_SIZE_CODE	(0x400*8)
4311da177e4SLinus Torvalds 
43224e4a121STakashi Iwai static ssize_t snd_emu10k1_fx8010_read(struct snd_info_entry *entry,
433eb4698f3STakashi Iwai 				       void *file_private_data,
4341da177e4SLinus Torvalds 				       struct file *file, char __user *buf,
43524e4a121STakashi Iwai 				       size_t count, loff_t pos)
4361da177e4SLinus Torvalds {
437eb4698f3STakashi Iwai 	struct snd_emu10k1 *emu = entry->private_data;
4381da177e4SLinus Torvalds 	unsigned int offset;
4391da177e4SLinus Torvalds 	int tram_addr = 0;
440d97e1b78STakashi Iwai 	unsigned int *tmp;
441d97e1b78STakashi Iwai 	long res;
442d97e1b78STakashi Iwai 	unsigned int idx;
4431da177e4SLinus Torvalds 
4441da177e4SLinus Torvalds 	if (!strcmp(entry->name, "fx8010_tram_addr")) {
4451da177e4SLinus Torvalds 		offset = TANKMEMADDRREGBASE;
4461da177e4SLinus Torvalds 		tram_addr = 1;
4471da177e4SLinus Torvalds 	} else if (!strcmp(entry->name, "fx8010_tram_data")) {
4481da177e4SLinus Torvalds 		offset = TANKMEMDATAREGBASE;
4491da177e4SLinus Torvalds 	} else if (!strcmp(entry->name, "fx8010_code")) {
4501da177e4SLinus Torvalds 		offset = emu->audigy ? A_MICROCODEBASE : MICROCODEBASE;
4511da177e4SLinus Torvalds 	} else {
4521da177e4SLinus Torvalds 		offset = emu->audigy ? A_FXGPREGBASE : FXGPREGBASE;
4531da177e4SLinus Torvalds 	}
454d97e1b78STakashi Iwai 
455d97e1b78STakashi Iwai 	tmp = kmalloc(count + 8, GFP_KERNEL);
456d97e1b78STakashi Iwai 	if (!tmp)
4571da177e4SLinus Torvalds 		return -ENOMEM;
458d97e1b78STakashi Iwai 	for (idx = 0; idx < ((pos & 3) + count + 3) >> 2; idx++) {
459d97e1b78STakashi Iwai 		unsigned int val;
460d97e1b78STakashi Iwai 		val = snd_emu10k1_ptr_read(emu, offset + idx + (pos >> 2), 0);
4611da177e4SLinus Torvalds 		if (tram_addr && emu->audigy) {
462d97e1b78STakashi Iwai 			val >>= 11;
463d97e1b78STakashi Iwai 			val |= snd_emu10k1_ptr_read(emu, 0x100 + idx + (pos >> 2), 0) << 20;
4641da177e4SLinus Torvalds 		}
465d97e1b78STakashi Iwai 		tmp[idx] = val;
466d97e1b78STakashi Iwai 	}
467d97e1b78STakashi Iwai 	if (copy_to_user(buf, ((char *)tmp) + (pos & 3), count))
468d97e1b78STakashi Iwai 		res = -EFAULT;
469d97e1b78STakashi Iwai 	else
470d97e1b78STakashi Iwai 		res = count;
4711da177e4SLinus Torvalds 	kfree(tmp);
4721da177e4SLinus Torvalds 	return res;
4731da177e4SLinus Torvalds }
4741da177e4SLinus Torvalds 
475eb4698f3STakashi Iwai static void snd_emu10k1_proc_voices_read(struct snd_info_entry *entry,
476eb4698f3STakashi Iwai 				  struct snd_info_buffer *buffer)
4771da177e4SLinus Torvalds {
478eb4698f3STakashi Iwai 	struct snd_emu10k1 *emu = entry->private_data;
479eb4698f3STakashi Iwai 	struct snd_emu10k1_voice *voice;
4801da177e4SLinus Torvalds 	int idx;
481b840f8d8SOswald Buddenhagen 	static const char * const types[] = {
482b840f8d8SOswald Buddenhagen 		"Unused", "EFX", "EFX IRQ", "PCM", "PCM IRQ", "Synth"
483b840f8d8SOswald Buddenhagen 	};
484b840f8d8SOswald Buddenhagen 	static_assert(ARRAY_SIZE(types) == EMU10K1_NUM_TYPES);
4851da177e4SLinus Torvalds 
486a915d604SOswald Buddenhagen 	snd_iprintf(buffer, "ch\tdirty\tlast\tuse\n");
4871da177e4SLinus Torvalds 	for (idx = 0; idx < NUM_G; idx++) {
4881da177e4SLinus Torvalds 		voice = &emu->voices[idx];
489a915d604SOswald Buddenhagen 		snd_iprintf(buffer, "%i\t%u\t%u\t%s\n",
4901da177e4SLinus Torvalds 			idx,
49182a9fa6eSOswald Buddenhagen 			voice->dirty,
492a915d604SOswald Buddenhagen 			voice->last,
493b840f8d8SOswald Buddenhagen 			types[voice->use]);
4941da177e4SLinus Torvalds 	}
4951da177e4SLinus Torvalds }
4961da177e4SLinus Torvalds 
4971da177e4SLinus Torvalds #ifdef CONFIG_SND_DEBUG
4989f4bd5ddSJames Courtier-Dutton static void snd_emu_proc_emu1010_reg_read(struct snd_info_entry *entry,
4999f4bd5ddSJames Courtier-Dutton 				     struct snd_info_buffer *buffer)
5009f4bd5ddSJames Courtier-Dutton {
5019f4bd5ddSJames Courtier-Dutton 	struct snd_emu10k1 *emu = entry->private_data;
5024677df07SHarvey Harrison 	u32 value;
5039f4bd5ddSJames Courtier-Dutton 	int i;
5049f4bd5ddSJames Courtier-Dutton 	snd_iprintf(buffer, "EMU1010 Registers:\n\n");
5059f4bd5ddSJames Courtier-Dutton 
50690fd5ce5SJames Courtier-Dutton 	for(i = 0; i < 0x40; i+=1) {
507f93abe51SJames Courtier-Dutton 		snd_emu1010_fpga_read(emu, i, &value);
508f93abe51SJames Courtier-Dutton 		snd_iprintf(buffer, "%02X: %08X, %02X\n", i, value, (value >> 8) & 0x7f);
5099f4bd5ddSJames Courtier-Dutton 	}
5109f4bd5ddSJames Courtier-Dutton }
5119f4bd5ddSJames Courtier-Dutton 
512eb4698f3STakashi Iwai static void snd_emu_proc_io_reg_read(struct snd_info_entry *entry,
513eb4698f3STakashi Iwai 				     struct snd_info_buffer *buffer)
5141da177e4SLinus Torvalds {
515eb4698f3STakashi Iwai 	struct snd_emu10k1 *emu = entry->private_data;
5161da177e4SLinus Torvalds 	unsigned long value;
5171da177e4SLinus Torvalds 	int i;
5181da177e4SLinus Torvalds 	snd_iprintf(buffer, "IO Registers:\n\n");
5191da177e4SLinus Torvalds 	for(i = 0; i < 0x40; i+=4) {
5201da177e4SLinus Torvalds 		value = inl(emu->port + i);
5211da177e4SLinus Torvalds 		snd_iprintf(buffer, "%02X: %08lX\n", i, value);
5221da177e4SLinus Torvalds 	}
5231da177e4SLinus Torvalds }
5241da177e4SLinus Torvalds 
525eb4698f3STakashi Iwai static void snd_emu_proc_io_reg_write(struct snd_info_entry *entry,
526eb4698f3STakashi Iwai                                       struct snd_info_buffer *buffer)
5271da177e4SLinus Torvalds {
528eb4698f3STakashi Iwai 	struct snd_emu10k1 *emu = entry->private_data;
5291da177e4SLinus Torvalds 	char line[64];
5301da177e4SLinus Torvalds 	u32 reg, val;
5311da177e4SLinus Torvalds 	while (!snd_info_get_line(buffer, line, sizeof(line))) {
5321da177e4SLinus Torvalds 		if (sscanf(line, "%x %x", &reg, &val) != 2)
5331da177e4SLinus Torvalds 			continue;
53484ed1a19SRoel Kluin 		if (reg < 0x40 && val <= 0xffffffff) {
5351da177e4SLinus Torvalds 			outl(val, emu->port + (reg & 0xfffffffc));
5361da177e4SLinus Torvalds 		}
5371da177e4SLinus Torvalds 	}
5381da177e4SLinus Torvalds }
5391da177e4SLinus Torvalds 
540eb4698f3STakashi Iwai static unsigned int snd_ptr_read(struct snd_emu10k1 * emu,
5411da177e4SLinus Torvalds 				 unsigned int iobase,
5421da177e4SLinus Torvalds 				 unsigned int reg,
5431da177e4SLinus Torvalds 				 unsigned int chn)
5441da177e4SLinus Torvalds {
5451da177e4SLinus Torvalds 	unsigned long flags;
5461da177e4SLinus Torvalds 	unsigned int regptr, val;
5471da177e4SLinus Torvalds 
5481da177e4SLinus Torvalds 	regptr = (reg << 16) | chn;
5491da177e4SLinus Torvalds 
5501da177e4SLinus Torvalds 	spin_lock_irqsave(&emu->emu_lock, flags);
5511da177e4SLinus Torvalds 	outl(regptr, emu->port + iobase + PTR);
5521da177e4SLinus Torvalds 	val = inl(emu->port + iobase + DATA);
5531da177e4SLinus Torvalds 	spin_unlock_irqrestore(&emu->emu_lock, flags);
5541da177e4SLinus Torvalds 	return val;
5551da177e4SLinus Torvalds }
5561da177e4SLinus Torvalds 
557eb4698f3STakashi Iwai static void snd_ptr_write(struct snd_emu10k1 *emu,
5581da177e4SLinus Torvalds 			  unsigned int iobase,
5591da177e4SLinus Torvalds 			  unsigned int reg,
5601da177e4SLinus Torvalds 			  unsigned int chn,
5611da177e4SLinus Torvalds 			  unsigned int data)
5621da177e4SLinus Torvalds {
5631da177e4SLinus Torvalds 	unsigned int regptr;
5641da177e4SLinus Torvalds 	unsigned long flags;
5651da177e4SLinus Torvalds 
5661da177e4SLinus Torvalds 	regptr = (reg << 16) | chn;
5671da177e4SLinus Torvalds 
5681da177e4SLinus Torvalds 	spin_lock_irqsave(&emu->emu_lock, flags);
5691da177e4SLinus Torvalds 	outl(regptr, emu->port + iobase + PTR);
5701da177e4SLinus Torvalds 	outl(data, emu->port + iobase + DATA);
5711da177e4SLinus Torvalds 	spin_unlock_irqrestore(&emu->emu_lock, flags);
5721da177e4SLinus Torvalds }
5731da177e4SLinus Torvalds 
5741da177e4SLinus Torvalds 
575eb4698f3STakashi Iwai static void snd_emu_proc_ptr_reg_read(struct snd_info_entry *entry,
576eb4698f3STakashi Iwai 				      struct snd_info_buffer *buffer, int iobase, int offset, int length, int voices)
5771da177e4SLinus Torvalds {
578eb4698f3STakashi Iwai 	struct snd_emu10k1 *emu = entry->private_data;
5791da177e4SLinus Torvalds 	unsigned long value;
5801da177e4SLinus Torvalds 	int i,j;
5816ef7e861SJames Courtier-Dutton 	if (offset+length > 0xa0) {
5821da177e4SLinus Torvalds 		snd_iprintf(buffer, "Input values out of range\n");
5831da177e4SLinus Torvalds 		return;
5841da177e4SLinus Torvalds 	}
5851da177e4SLinus Torvalds 	snd_iprintf(buffer, "Registers 0x%x\n", iobase);
5861da177e4SLinus Torvalds 	for(i = offset; i < offset+length; i++) {
5871da177e4SLinus Torvalds 		snd_iprintf(buffer, "%02X: ",i);
5881da177e4SLinus Torvalds 		for (j = 0; j < voices; j++) {
589e922da40SOswald Buddenhagen 			value = snd_ptr_read(emu, iobase, i, j);
5901da177e4SLinus Torvalds 			snd_iprintf(buffer, "%08lX ", value);
5911da177e4SLinus Torvalds 		}
5921da177e4SLinus Torvalds 		snd_iprintf(buffer, "\n");
5931da177e4SLinus Torvalds 	}
5941da177e4SLinus Torvalds }
5951da177e4SLinus Torvalds 
596eb4698f3STakashi Iwai static void snd_emu_proc_ptr_reg_write(struct snd_info_entry *entry,
59767ff2addSOswald Buddenhagen 				       struct snd_info_buffer *buffer,
59867ff2addSOswald Buddenhagen 				       int iobase, int length, int voices)
5991da177e4SLinus Torvalds {
600eb4698f3STakashi Iwai 	struct snd_emu10k1 *emu = entry->private_data;
6011da177e4SLinus Torvalds 	char line[64];
6021da177e4SLinus Torvalds 	unsigned int reg, channel_id , val;
6031da177e4SLinus Torvalds 	while (!snd_info_get_line(buffer, line, sizeof(line))) {
6041da177e4SLinus Torvalds 		if (sscanf(line, "%x %x %x", &reg, &channel_id, &val) != 3)
6051da177e4SLinus Torvalds 			continue;
60667ff2addSOswald Buddenhagen 		if (reg < length && channel_id < voices)
6071da177e4SLinus Torvalds 			snd_ptr_write(emu, iobase, reg, channel_id, val);
6081da177e4SLinus Torvalds 	}
6091da177e4SLinus Torvalds }
6101da177e4SLinus Torvalds 
611eb4698f3STakashi Iwai static void snd_emu_proc_ptr_reg_write00(struct snd_info_entry *entry,
612eb4698f3STakashi Iwai 					 struct snd_info_buffer *buffer)
6131da177e4SLinus Torvalds {
61467ff2addSOswald Buddenhagen 	snd_emu_proc_ptr_reg_write(entry, buffer, 0, 0x80, 64);
6151da177e4SLinus Torvalds }
6161da177e4SLinus Torvalds 
617eb4698f3STakashi Iwai static void snd_emu_proc_ptr_reg_write20(struct snd_info_entry *entry,
618eb4698f3STakashi Iwai 					 struct snd_info_buffer *buffer)
6191da177e4SLinus Torvalds {
62067ff2addSOswald Buddenhagen 	struct snd_emu10k1 *emu = entry->private_data;
62167ff2addSOswald Buddenhagen 	snd_emu_proc_ptr_reg_write(entry, buffer, 0x20,
62267ff2addSOswald Buddenhagen 				   emu->card_capabilities->ca0108_chip ? 0xa0 : 0x80, 4);
6231da177e4SLinus Torvalds }
6241da177e4SLinus Torvalds 
6251da177e4SLinus Torvalds 
626eb4698f3STakashi Iwai static void snd_emu_proc_ptr_reg_read00a(struct snd_info_entry *entry,
627eb4698f3STakashi Iwai 					 struct snd_info_buffer *buffer)
6281da177e4SLinus Torvalds {
6291da177e4SLinus Torvalds 	snd_emu_proc_ptr_reg_read(entry, buffer, 0, 0, 0x40, 64);
6301da177e4SLinus Torvalds }
6311da177e4SLinus Torvalds 
632eb4698f3STakashi Iwai static void snd_emu_proc_ptr_reg_read00b(struct snd_info_entry *entry,
633eb4698f3STakashi Iwai 					 struct snd_info_buffer *buffer)
6341da177e4SLinus Torvalds {
6351da177e4SLinus Torvalds 	snd_emu_proc_ptr_reg_read(entry, buffer, 0, 0x40, 0x40, 64);
6361da177e4SLinus Torvalds }
6371da177e4SLinus Torvalds 
638eb4698f3STakashi Iwai static void snd_emu_proc_ptr_reg_read20a(struct snd_info_entry *entry,
639eb4698f3STakashi Iwai 					 struct snd_info_buffer *buffer)
6401da177e4SLinus Torvalds {
6411da177e4SLinus Torvalds 	snd_emu_proc_ptr_reg_read(entry, buffer, 0x20, 0, 0x40, 4);
6421da177e4SLinus Torvalds }
6431da177e4SLinus Torvalds 
644eb4698f3STakashi Iwai static void snd_emu_proc_ptr_reg_read20b(struct snd_info_entry *entry,
645eb4698f3STakashi Iwai 					 struct snd_info_buffer *buffer)
6461da177e4SLinus Torvalds {
6471da177e4SLinus Torvalds 	snd_emu_proc_ptr_reg_read(entry, buffer, 0x20, 0x40, 0x40, 4);
6481da177e4SLinus Torvalds }
6496ef7e861SJames Courtier-Dutton 
650eb4698f3STakashi Iwai static void snd_emu_proc_ptr_reg_read20c(struct snd_info_entry *entry,
651eb4698f3STakashi Iwai 					 struct snd_info_buffer * buffer)
6526ef7e861SJames Courtier-Dutton {
6536ef7e861SJames Courtier-Dutton 	snd_emu_proc_ptr_reg_read(entry, buffer, 0x20, 0x80, 0x20, 4);
6546ef7e861SJames Courtier-Dutton }
6551da177e4SLinus Torvalds #endif
6561da177e4SLinus Torvalds 
657d25ff268STakashi Iwai static const struct snd_info_entry_ops snd_emu10k1_proc_ops_fx8010 = {
6581da177e4SLinus Torvalds 	.read = snd_emu10k1_fx8010_read,
6591da177e4SLinus Torvalds };
6601da177e4SLinus Torvalds 
661e23e7a14SBill Pemberton int snd_emu10k1_proc_init(struct snd_emu10k1 *emu)
6621da177e4SLinus Torvalds {
663eb4698f3STakashi Iwai 	struct snd_info_entry *entry;
6641da177e4SLinus Torvalds #ifdef CONFIG_SND_DEBUG
665190d2c46SJames Courtier-Dutton 	if (emu->card_capabilities->emu_model) {
66647f2769bSTakashi Iwai 		snd_card_ro_proc_new(emu->card, "emu1010_regs",
66747f2769bSTakashi Iwai 				     emu, snd_emu_proc_emu1010_reg_read);
6689f4bd5ddSJames Courtier-Dutton 	}
66947f2769bSTakashi Iwai 	snd_card_rw_proc_new(emu->card, "io_regs", emu,
67047f2769bSTakashi Iwai 			     snd_emu_proc_io_reg_read,
67147f2769bSTakashi Iwai 			     snd_emu_proc_io_reg_write);
67247f2769bSTakashi Iwai 	snd_card_rw_proc_new(emu->card, "ptr_regs00a", emu,
67347f2769bSTakashi Iwai 			     snd_emu_proc_ptr_reg_read00a,
67447f2769bSTakashi Iwai 			     snd_emu_proc_ptr_reg_write00);
67547f2769bSTakashi Iwai 	snd_card_rw_proc_new(emu->card, "ptr_regs00b", emu,
67647f2769bSTakashi Iwai 			     snd_emu_proc_ptr_reg_read00b,
67747f2769bSTakashi Iwai 			     snd_emu_proc_ptr_reg_write00);
678219153c6SOswald Buddenhagen 	if (!emu->card_capabilities->emu_model &&
679219153c6SOswald Buddenhagen 	    (emu->card_capabilities->ca0151_chip || emu->card_capabilities->ca0108_chip)) {
68047f2769bSTakashi Iwai 		snd_card_rw_proc_new(emu->card, "ptr_regs20a", emu,
68147f2769bSTakashi Iwai 				     snd_emu_proc_ptr_reg_read20a,
68247f2769bSTakashi Iwai 				     snd_emu_proc_ptr_reg_write20);
68347f2769bSTakashi Iwai 		snd_card_rw_proc_new(emu->card, "ptr_regs20b", emu,
68447f2769bSTakashi Iwai 				     snd_emu_proc_ptr_reg_read20b,
68547f2769bSTakashi Iwai 				     snd_emu_proc_ptr_reg_write20);
686219153c6SOswald Buddenhagen 		if (emu->card_capabilities->ca0108_chip)
68747f2769bSTakashi Iwai 			snd_card_rw_proc_new(emu->card, "ptr_regs20c", emu,
68847f2769bSTakashi Iwai 					     snd_emu_proc_ptr_reg_read20c,
68947f2769bSTakashi Iwai 					     snd_emu_proc_ptr_reg_write20);
690219153c6SOswald Buddenhagen 	}
6911da177e4SLinus Torvalds #endif
6921da177e4SLinus Torvalds 
69347f2769bSTakashi Iwai 	snd_card_ro_proc_new(emu->card, "emu10k1", emu, snd_emu10k1_proc_read);
6941da177e4SLinus Torvalds 
69547f2769bSTakashi Iwai 	if (emu->card_capabilities->emu10k2_chip)
69647f2769bSTakashi Iwai 		snd_card_ro_proc_new(emu->card, "spdif-in", emu,
69747f2769bSTakashi Iwai 				     snd_emu10k1_proc_spdif_read);
69847f2769bSTakashi Iwai 	if (emu->card_capabilities->ca0151_chip)
69947f2769bSTakashi Iwai 		snd_card_ro_proc_new(emu->card, "capture-rates", emu,
70047f2769bSTakashi Iwai 				     snd_emu10k1_proc_rates_read);
701df34140aSJames Courtier-Dutton 
70247f2769bSTakashi Iwai 	snd_card_ro_proc_new(emu->card, "voices", emu,
70347f2769bSTakashi Iwai 			     snd_emu10k1_proc_voices_read);
7041da177e4SLinus Torvalds 
7051da177e4SLinus Torvalds 	if (! snd_card_proc_new(emu->card, "fx8010_gpr", &entry)) {
7061da177e4SLinus Torvalds 		entry->content = SNDRV_INFO_CONTENT_DATA;
7071da177e4SLinus Torvalds 		entry->private_data = emu;
7086a73cf46SJoe Perches 		entry->mode = S_IFREG | 0444 /*| S_IWUSR*/;
7091da177e4SLinus Torvalds 		entry->size = emu->audigy ? A_TOTAL_SIZE_GPR : TOTAL_SIZE_GPR;
7101da177e4SLinus Torvalds 		entry->c.ops = &snd_emu10k1_proc_ops_fx8010;
7111da177e4SLinus Torvalds 	}
7121da177e4SLinus Torvalds 	if (! snd_card_proc_new(emu->card, "fx8010_tram_data", &entry)) {
7131da177e4SLinus Torvalds 		entry->content = SNDRV_INFO_CONTENT_DATA;
7141da177e4SLinus Torvalds 		entry->private_data = emu;
7156a73cf46SJoe Perches 		entry->mode = S_IFREG | 0444 /*| S_IWUSR*/;
7161da177e4SLinus Torvalds 		entry->size = emu->audigy ? A_TOTAL_SIZE_TANKMEM_DATA : TOTAL_SIZE_TANKMEM_DATA ;
7171da177e4SLinus Torvalds 		entry->c.ops = &snd_emu10k1_proc_ops_fx8010;
7181da177e4SLinus Torvalds 	}
7191da177e4SLinus Torvalds 	if (! snd_card_proc_new(emu->card, "fx8010_tram_addr", &entry)) {
7201da177e4SLinus Torvalds 		entry->content = SNDRV_INFO_CONTENT_DATA;
7211da177e4SLinus Torvalds 		entry->private_data = emu;
7226a73cf46SJoe Perches 		entry->mode = S_IFREG | 0444 /*| S_IWUSR*/;
7231da177e4SLinus Torvalds 		entry->size = emu->audigy ? A_TOTAL_SIZE_TANKMEM_ADDR : TOTAL_SIZE_TANKMEM_ADDR ;
7241da177e4SLinus Torvalds 		entry->c.ops = &snd_emu10k1_proc_ops_fx8010;
7251da177e4SLinus Torvalds 	}
7261da177e4SLinus Torvalds 	if (! snd_card_proc_new(emu->card, "fx8010_code", &entry)) {
7271da177e4SLinus Torvalds 		entry->content = SNDRV_INFO_CONTENT_DATA;
7281da177e4SLinus Torvalds 		entry->private_data = emu;
7296a73cf46SJoe Perches 		entry->mode = S_IFREG | 0444 /*| S_IWUSR*/;
7301da177e4SLinus Torvalds 		entry->size = emu->audigy ? A_TOTAL_SIZE_CODE : TOTAL_SIZE_CODE;
7311da177e4SLinus Torvalds 		entry->c.ops = &snd_emu10k1_proc_ops_fx8010;
7321da177e4SLinus Torvalds 	}
73347f2769bSTakashi Iwai 	snd_card_ro_proc_new(emu->card, "fx8010_acode", emu,
73447f2769bSTakashi Iwai 			     snd_emu10k1_proc_acode_read);
7351da177e4SLinus Torvalds 	return 0;
7361da177e4SLinus Torvalds }
737