xref: /OK3568_Linux_fs/kernel/sound/isa/gus/gus_irq.c (revision 4882a59341e53eb6f0b4789bf948001014eff981)
1*4882a593Smuzhiyun // SPDX-License-Identifier: GPL-2.0-or-later
2*4882a593Smuzhiyun /*
3*4882a593Smuzhiyun  *  Routine for IRQ handling from GF1/InterWave chip
4*4882a593Smuzhiyun  *  Copyright (c) by Jaroslav Kysela <perex@perex.cz>
5*4882a593Smuzhiyun  */
6*4882a593Smuzhiyun 
7*4882a593Smuzhiyun #include <sound/core.h>
8*4882a593Smuzhiyun #include <sound/info.h>
9*4882a593Smuzhiyun #include <sound/gus.h>
10*4882a593Smuzhiyun 
11*4882a593Smuzhiyun #ifdef CONFIG_SND_DEBUG
12*4882a593Smuzhiyun #define STAT_ADD(x)	((x)++)
13*4882a593Smuzhiyun #else
14*4882a593Smuzhiyun #define STAT_ADD(x)	while (0) { ; }
15*4882a593Smuzhiyun #endif
16*4882a593Smuzhiyun 
snd_gus_interrupt(int irq,void * dev_id)17*4882a593Smuzhiyun irqreturn_t snd_gus_interrupt(int irq, void *dev_id)
18*4882a593Smuzhiyun {
19*4882a593Smuzhiyun 	struct snd_gus_card * gus = dev_id;
20*4882a593Smuzhiyun 	unsigned char status;
21*4882a593Smuzhiyun 	int loop = 100;
22*4882a593Smuzhiyun 	int handled = 0;
23*4882a593Smuzhiyun 
24*4882a593Smuzhiyun __again:
25*4882a593Smuzhiyun 	status = inb(gus->gf1.reg_irqstat);
26*4882a593Smuzhiyun 	if (status == 0)
27*4882a593Smuzhiyun 		return IRQ_RETVAL(handled);
28*4882a593Smuzhiyun 	handled = 1;
29*4882a593Smuzhiyun 	/* snd_printk(KERN_DEBUG "IRQ: status = 0x%x\n", status); */
30*4882a593Smuzhiyun 	if (status & 0x02) {
31*4882a593Smuzhiyun 		STAT_ADD(gus->gf1.interrupt_stat_midi_in);
32*4882a593Smuzhiyun 		if (gus->gf1.interrupt_handler_midi_in)
33*4882a593Smuzhiyun 			gus->gf1.interrupt_handler_midi_in(gus);
34*4882a593Smuzhiyun 	}
35*4882a593Smuzhiyun 	if (status & 0x01) {
36*4882a593Smuzhiyun 		STAT_ADD(gus->gf1.interrupt_stat_midi_out);
37*4882a593Smuzhiyun 		if (gus->gf1.interrupt_handler_midi_out)
38*4882a593Smuzhiyun 			gus->gf1.interrupt_handler_midi_out(gus);
39*4882a593Smuzhiyun 	}
40*4882a593Smuzhiyun 	if (status & (0x20 | 0x40)) {
41*4882a593Smuzhiyun 		unsigned int already, _current_;
42*4882a593Smuzhiyun 		unsigned char voice_status, voice;
43*4882a593Smuzhiyun 		struct snd_gus_voice *pvoice;
44*4882a593Smuzhiyun 
45*4882a593Smuzhiyun 		already = 0;
46*4882a593Smuzhiyun 		while (((voice_status = snd_gf1_i_read8(gus, SNDRV_GF1_GB_VOICES_IRQ)) & 0xc0) != 0xc0) {
47*4882a593Smuzhiyun 			voice = voice_status & 0x1f;
48*4882a593Smuzhiyun 			_current_ = 1 << voice;
49*4882a593Smuzhiyun 			if (already & _current_)
50*4882a593Smuzhiyun 				continue;	/* multi request */
51*4882a593Smuzhiyun 			already |= _current_;	/* mark request */
52*4882a593Smuzhiyun #if 0
53*4882a593Smuzhiyun 			printk(KERN_DEBUG "voice = %i, voice_status = 0x%x, "
54*4882a593Smuzhiyun 			       "voice_verify = %i\n",
55*4882a593Smuzhiyun 			       voice, voice_status, inb(GUSP(gus, GF1PAGE)));
56*4882a593Smuzhiyun #endif
57*4882a593Smuzhiyun 			pvoice = &gus->gf1.voices[voice];
58*4882a593Smuzhiyun 			if (pvoice->use) {
59*4882a593Smuzhiyun 				if (!(voice_status & 0x80)) {	/* voice position IRQ */
60*4882a593Smuzhiyun 					STAT_ADD(pvoice->interrupt_stat_wave);
61*4882a593Smuzhiyun 					pvoice->handler_wave(gus, pvoice);
62*4882a593Smuzhiyun 				}
63*4882a593Smuzhiyun 				if (!(voice_status & 0x40)) {	/* volume ramp IRQ */
64*4882a593Smuzhiyun 					STAT_ADD(pvoice->interrupt_stat_volume);
65*4882a593Smuzhiyun 					pvoice->handler_volume(gus, pvoice);
66*4882a593Smuzhiyun 				}
67*4882a593Smuzhiyun 			} else {
68*4882a593Smuzhiyun 				STAT_ADD(gus->gf1.interrupt_stat_voice_lost);
69*4882a593Smuzhiyun 				snd_gf1_i_ctrl_stop(gus, SNDRV_GF1_VB_ADDRESS_CONTROL);
70*4882a593Smuzhiyun 				snd_gf1_i_ctrl_stop(gus, SNDRV_GF1_VB_VOLUME_CONTROL);
71*4882a593Smuzhiyun 			}
72*4882a593Smuzhiyun 		}
73*4882a593Smuzhiyun 	}
74*4882a593Smuzhiyun 	if (status & 0x04) {
75*4882a593Smuzhiyun 		STAT_ADD(gus->gf1.interrupt_stat_timer1);
76*4882a593Smuzhiyun 		if (gus->gf1.interrupt_handler_timer1)
77*4882a593Smuzhiyun 			gus->gf1.interrupt_handler_timer1(gus);
78*4882a593Smuzhiyun 	}
79*4882a593Smuzhiyun 	if (status & 0x08) {
80*4882a593Smuzhiyun 		STAT_ADD(gus->gf1.interrupt_stat_timer2);
81*4882a593Smuzhiyun 		if (gus->gf1.interrupt_handler_timer2)
82*4882a593Smuzhiyun 			gus->gf1.interrupt_handler_timer2(gus);
83*4882a593Smuzhiyun 	}
84*4882a593Smuzhiyun 	if (status & 0x80) {
85*4882a593Smuzhiyun 		if (snd_gf1_i_look8(gus, SNDRV_GF1_GB_DRAM_DMA_CONTROL) & 0x40) {
86*4882a593Smuzhiyun 			STAT_ADD(gus->gf1.interrupt_stat_dma_write);
87*4882a593Smuzhiyun 			if (gus->gf1.interrupt_handler_dma_write)
88*4882a593Smuzhiyun 				gus->gf1.interrupt_handler_dma_write(gus);
89*4882a593Smuzhiyun 		}
90*4882a593Smuzhiyun 		if (snd_gf1_i_look8(gus, SNDRV_GF1_GB_REC_DMA_CONTROL) & 0x40) {
91*4882a593Smuzhiyun 			STAT_ADD(gus->gf1.interrupt_stat_dma_read);
92*4882a593Smuzhiyun 			if (gus->gf1.interrupt_handler_dma_read)
93*4882a593Smuzhiyun 				gus->gf1.interrupt_handler_dma_read(gus);
94*4882a593Smuzhiyun 		}
95*4882a593Smuzhiyun 	}
96*4882a593Smuzhiyun 	if (--loop > 0)
97*4882a593Smuzhiyun 		goto __again;
98*4882a593Smuzhiyun 	return IRQ_NONE;
99*4882a593Smuzhiyun }
100*4882a593Smuzhiyun 
101*4882a593Smuzhiyun #ifdef CONFIG_SND_DEBUG
snd_gus_irq_info_read(struct snd_info_entry * entry,struct snd_info_buffer * buffer)102*4882a593Smuzhiyun static void snd_gus_irq_info_read(struct snd_info_entry *entry,
103*4882a593Smuzhiyun 				  struct snd_info_buffer *buffer)
104*4882a593Smuzhiyun {
105*4882a593Smuzhiyun 	struct snd_gus_card *gus;
106*4882a593Smuzhiyun 	struct snd_gus_voice *pvoice;
107*4882a593Smuzhiyun 	int idx;
108*4882a593Smuzhiyun 
109*4882a593Smuzhiyun 	gus = entry->private_data;
110*4882a593Smuzhiyun 	snd_iprintf(buffer, "midi out = %u\n", gus->gf1.interrupt_stat_midi_out);
111*4882a593Smuzhiyun 	snd_iprintf(buffer, "midi in = %u\n", gus->gf1.interrupt_stat_midi_in);
112*4882a593Smuzhiyun 	snd_iprintf(buffer, "timer1 = %u\n", gus->gf1.interrupt_stat_timer1);
113*4882a593Smuzhiyun 	snd_iprintf(buffer, "timer2 = %u\n", gus->gf1.interrupt_stat_timer2);
114*4882a593Smuzhiyun 	snd_iprintf(buffer, "dma write = %u\n", gus->gf1.interrupt_stat_dma_write);
115*4882a593Smuzhiyun 	snd_iprintf(buffer, "dma read = %u\n", gus->gf1.interrupt_stat_dma_read);
116*4882a593Smuzhiyun 	snd_iprintf(buffer, "voice lost = %u\n", gus->gf1.interrupt_stat_voice_lost);
117*4882a593Smuzhiyun 	for (idx = 0; idx < 32; idx++) {
118*4882a593Smuzhiyun 		pvoice = &gus->gf1.voices[idx];
119*4882a593Smuzhiyun 		snd_iprintf(buffer, "voice %i: wave = %u, volume = %u\n",
120*4882a593Smuzhiyun 					idx,
121*4882a593Smuzhiyun 					pvoice->interrupt_stat_wave,
122*4882a593Smuzhiyun 					pvoice->interrupt_stat_volume);
123*4882a593Smuzhiyun 	}
124*4882a593Smuzhiyun }
125*4882a593Smuzhiyun 
snd_gus_irq_profile_init(struct snd_gus_card * gus)126*4882a593Smuzhiyun void snd_gus_irq_profile_init(struct snd_gus_card *gus)
127*4882a593Smuzhiyun {
128*4882a593Smuzhiyun 	snd_card_ro_proc_new(gus->card, "gusirq", gus, snd_gus_irq_info_read);
129*4882a593Smuzhiyun }
130*4882a593Smuzhiyun 
131*4882a593Smuzhiyun #endif
132