1*4882a593Smuzhiyun /*
2*4882a593Smuzhiyun * Local definitions for the OPL4 driver
3*4882a593Smuzhiyun *
4*4882a593Smuzhiyun * Copyright (c) 2003 by Clemens Ladisch <clemens@ladisch.de>
5*4882a593Smuzhiyun * All rights reserved.
6*4882a593Smuzhiyun *
7*4882a593Smuzhiyun * Redistribution and use in source and binary forms, with or without
8*4882a593Smuzhiyun * modification, are permitted provided that the following conditions
9*4882a593Smuzhiyun * are met:
10*4882a593Smuzhiyun * 1. Redistributions of source code must retain the above copyright
11*4882a593Smuzhiyun * notice, this list of conditions, and the following disclaimer,
12*4882a593Smuzhiyun * without modification.
13*4882a593Smuzhiyun * 2. The name of the author may not be used to endorse or promote products
14*4882a593Smuzhiyun * derived from this software without specific prior written permission.
15*4882a593Smuzhiyun *
16*4882a593Smuzhiyun * Alternatively, this software may be distributed and/or modified under the
17*4882a593Smuzhiyun * terms of the GNU General Public License as published by the Free Software
18*4882a593Smuzhiyun * Foundation; either version 2 of the License, or (at your option) any later
19*4882a593Smuzhiyun * version.
20*4882a593Smuzhiyun *
21*4882a593Smuzhiyun * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND
22*4882a593Smuzhiyun * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
23*4882a593Smuzhiyun * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
24*4882a593Smuzhiyun * ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE FOR
25*4882a593Smuzhiyun * ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
26*4882a593Smuzhiyun * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
27*4882a593Smuzhiyun * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
28*4882a593Smuzhiyun * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
29*4882a593Smuzhiyun * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
30*4882a593Smuzhiyun * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
31*4882a593Smuzhiyun * SUCH DAMAGE.
32*4882a593Smuzhiyun */
33*4882a593Smuzhiyun
34*4882a593Smuzhiyun #ifndef __OPL4_LOCAL_H
35*4882a593Smuzhiyun #define __OPL4_LOCAL_H
36*4882a593Smuzhiyun
37*4882a593Smuzhiyun #include <sound/opl4.h>
38*4882a593Smuzhiyun
39*4882a593Smuzhiyun /*
40*4882a593Smuzhiyun * Register numbers
41*4882a593Smuzhiyun */
42*4882a593Smuzhiyun
43*4882a593Smuzhiyun #define OPL4_REG_TEST0 0x00
44*4882a593Smuzhiyun #define OPL4_REG_TEST1 0x01
45*4882a593Smuzhiyun
46*4882a593Smuzhiyun #define OPL4_REG_MEMORY_CONFIGURATION 0x02
47*4882a593Smuzhiyun #define OPL4_MODE_BIT 0x01
48*4882a593Smuzhiyun #define OPL4_MTYPE_BIT 0x02
49*4882a593Smuzhiyun #define OPL4_TONE_HEADER_MASK 0x1c
50*4882a593Smuzhiyun #define OPL4_DEVICE_ID_MASK 0xe0
51*4882a593Smuzhiyun
52*4882a593Smuzhiyun #define OPL4_REG_MEMORY_ADDRESS_HIGH 0x03
53*4882a593Smuzhiyun #define OPL4_REG_MEMORY_ADDRESS_MID 0x04
54*4882a593Smuzhiyun #define OPL4_REG_MEMORY_ADDRESS_LOW 0x05
55*4882a593Smuzhiyun #define OPL4_REG_MEMORY_DATA 0x06
56*4882a593Smuzhiyun
57*4882a593Smuzhiyun /*
58*4882a593Smuzhiyun * Offsets to the register banks for voices. To get the
59*4882a593Smuzhiyun * register number just add the voice number to the bank offset.
60*4882a593Smuzhiyun *
61*4882a593Smuzhiyun * Wave Table Number low bits (0x08 to 0x1F)
62*4882a593Smuzhiyun */
63*4882a593Smuzhiyun #define OPL4_REG_TONE_NUMBER 0x08
64*4882a593Smuzhiyun
65*4882a593Smuzhiyun /* Wave Table Number high bit, F-Number low bits (0x20 to 0x37) */
66*4882a593Smuzhiyun #define OPL4_REG_F_NUMBER 0x20
67*4882a593Smuzhiyun #define OPL4_TONE_NUMBER_BIT8 0x01
68*4882a593Smuzhiyun #define OPL4_F_NUMBER_LOW_MASK 0xfe
69*4882a593Smuzhiyun
70*4882a593Smuzhiyun /* F-Number high bits, Octave, Pseudo-Reverb (0x38 to 0x4F) */
71*4882a593Smuzhiyun #define OPL4_REG_OCTAVE 0x38
72*4882a593Smuzhiyun #define OPL4_F_NUMBER_HIGH_MASK 0x07
73*4882a593Smuzhiyun #define OPL4_BLOCK_MASK 0xf0
74*4882a593Smuzhiyun #define OPL4_PSEUDO_REVERB_BIT 0x08
75*4882a593Smuzhiyun
76*4882a593Smuzhiyun /* Total Level, Level Direct (0x50 to 0x67) */
77*4882a593Smuzhiyun #define OPL4_REG_LEVEL 0x50
78*4882a593Smuzhiyun #define OPL4_TOTAL_LEVEL_MASK 0xfe
79*4882a593Smuzhiyun #define OPL4_LEVEL_DIRECT_BIT 0x01
80*4882a593Smuzhiyun
81*4882a593Smuzhiyun /* Key On, Damp, LFO RST, CH, Panpot (0x68 to 0x7F) */
82*4882a593Smuzhiyun #define OPL4_REG_MISC 0x68
83*4882a593Smuzhiyun #define OPL4_KEY_ON_BIT 0x80
84*4882a593Smuzhiyun #define OPL4_DAMP_BIT 0x40
85*4882a593Smuzhiyun #define OPL4_LFO_RESET_BIT 0x20
86*4882a593Smuzhiyun #define OPL4_OUTPUT_CHANNEL_BIT 0x10
87*4882a593Smuzhiyun #define OPL4_PAN_POT_MASK 0x0f
88*4882a593Smuzhiyun
89*4882a593Smuzhiyun /* LFO, VIB (0x80 to 0x97) */
90*4882a593Smuzhiyun #define OPL4_REG_LFO_VIBRATO 0x80
91*4882a593Smuzhiyun #define OPL4_LFO_FREQUENCY_MASK 0x38
92*4882a593Smuzhiyun #define OPL4_VIBRATO_DEPTH_MASK 0x07
93*4882a593Smuzhiyun #define OPL4_CHORUS_SEND_MASK 0xc0 /* ML only */
94*4882a593Smuzhiyun
95*4882a593Smuzhiyun /* Attack / Decay 1 rate (0x98 to 0xAF) */
96*4882a593Smuzhiyun #define OPL4_REG_ATTACK_DECAY1 0x98
97*4882a593Smuzhiyun #define OPL4_ATTACK_RATE_MASK 0xf0
98*4882a593Smuzhiyun #define OPL4_DECAY1_RATE_MASK 0x0f
99*4882a593Smuzhiyun
100*4882a593Smuzhiyun /* Decay level / 2 rate (0xB0 to 0xC7) */
101*4882a593Smuzhiyun #define OPL4_REG_LEVEL_DECAY2 0xb0
102*4882a593Smuzhiyun #define OPL4_DECAY_LEVEL_MASK 0xf0
103*4882a593Smuzhiyun #define OPL4_DECAY2_RATE_MASK 0x0f
104*4882a593Smuzhiyun
105*4882a593Smuzhiyun /* Release rate / Rate correction (0xC8 to 0xDF) */
106*4882a593Smuzhiyun #define OPL4_REG_RELEASE_CORRECTION 0xc8
107*4882a593Smuzhiyun #define OPL4_RELEASE_RATE_MASK 0x0f
108*4882a593Smuzhiyun #define OPL4_RATE_INTERPOLATION_MASK 0xf0
109*4882a593Smuzhiyun
110*4882a593Smuzhiyun /* AM (0xE0 to 0xF7) */
111*4882a593Smuzhiyun #define OPL4_REG_TREMOLO 0xe0
112*4882a593Smuzhiyun #define OPL4_TREMOLO_DEPTH_MASK 0x07
113*4882a593Smuzhiyun #define OPL4_REVERB_SEND_MASK 0xe0 /* ML only */
114*4882a593Smuzhiyun
115*4882a593Smuzhiyun /* Mixer */
116*4882a593Smuzhiyun #define OPL4_REG_MIX_CONTROL_FM 0xf8
117*4882a593Smuzhiyun #define OPL4_REG_MIX_CONTROL_PCM 0xf9
118*4882a593Smuzhiyun #define OPL4_MIX_LEFT_MASK 0x07
119*4882a593Smuzhiyun #define OPL4_MIX_RIGHT_MASK 0x38
120*4882a593Smuzhiyun
121*4882a593Smuzhiyun #define OPL4_REG_ATC 0xfa
122*4882a593Smuzhiyun #define OPL4_ATC_BIT 0x01 /* ???, ML only */
123*4882a593Smuzhiyun
124*4882a593Smuzhiyun /* bits in the OPL3 Status register */
125*4882a593Smuzhiyun #define OPL4_STATUS_BUSY 0x01
126*4882a593Smuzhiyun #define OPL4_STATUS_LOAD 0x02
127*4882a593Smuzhiyun
128*4882a593Smuzhiyun
129*4882a593Smuzhiyun #define OPL4_MAX_VOICES 24
130*4882a593Smuzhiyun
131*4882a593Smuzhiyun #define SNDRV_SEQ_DEV_ID_OPL4 "opl4-synth"
132*4882a593Smuzhiyun
133*4882a593Smuzhiyun
134*4882a593Smuzhiyun struct opl4_sound {
135*4882a593Smuzhiyun u16 tone;
136*4882a593Smuzhiyun s16 pitch_offset;
137*4882a593Smuzhiyun u8 key_scaling;
138*4882a593Smuzhiyun s8 panpot;
139*4882a593Smuzhiyun u8 vibrato;
140*4882a593Smuzhiyun u8 tone_attenuate;
141*4882a593Smuzhiyun u8 volume_factor;
142*4882a593Smuzhiyun u8 reg_lfo_vibrato;
143*4882a593Smuzhiyun u8 reg_attack_decay1;
144*4882a593Smuzhiyun u8 reg_level_decay2;
145*4882a593Smuzhiyun u8 reg_release_correction;
146*4882a593Smuzhiyun u8 reg_tremolo;
147*4882a593Smuzhiyun };
148*4882a593Smuzhiyun
149*4882a593Smuzhiyun struct opl4_region {
150*4882a593Smuzhiyun u8 key_min, key_max;
151*4882a593Smuzhiyun struct opl4_sound sound;
152*4882a593Smuzhiyun };
153*4882a593Smuzhiyun
154*4882a593Smuzhiyun struct opl4_region_ptr {
155*4882a593Smuzhiyun int count;
156*4882a593Smuzhiyun const struct opl4_region *regions;
157*4882a593Smuzhiyun };
158*4882a593Smuzhiyun
159*4882a593Smuzhiyun struct opl4_voice {
160*4882a593Smuzhiyun struct list_head list;
161*4882a593Smuzhiyun int number;
162*4882a593Smuzhiyun struct snd_midi_channel *chan;
163*4882a593Smuzhiyun int note;
164*4882a593Smuzhiyun int velocity;
165*4882a593Smuzhiyun const struct opl4_sound *sound;
166*4882a593Smuzhiyun u8 level_direct;
167*4882a593Smuzhiyun u8 reg_f_number;
168*4882a593Smuzhiyun u8 reg_misc;
169*4882a593Smuzhiyun u8 reg_lfo_vibrato;
170*4882a593Smuzhiyun };
171*4882a593Smuzhiyun
172*4882a593Smuzhiyun struct snd_opl4 {
173*4882a593Smuzhiyun unsigned long fm_port;
174*4882a593Smuzhiyun unsigned long pcm_port;
175*4882a593Smuzhiyun struct resource *res_fm_port;
176*4882a593Smuzhiyun struct resource *res_pcm_port;
177*4882a593Smuzhiyun unsigned short hardware;
178*4882a593Smuzhiyun spinlock_t reg_lock;
179*4882a593Smuzhiyun struct snd_card *card;
180*4882a593Smuzhiyun
181*4882a593Smuzhiyun #ifdef CONFIG_SND_PROC_FS
182*4882a593Smuzhiyun struct snd_info_entry *proc_entry;
183*4882a593Smuzhiyun int memory_access;
184*4882a593Smuzhiyun #endif
185*4882a593Smuzhiyun struct mutex access_mutex;
186*4882a593Smuzhiyun
187*4882a593Smuzhiyun #if IS_ENABLED(CONFIG_SND_SEQUENCER)
188*4882a593Smuzhiyun int used;
189*4882a593Smuzhiyun
190*4882a593Smuzhiyun int seq_dev_num;
191*4882a593Smuzhiyun int seq_client;
192*4882a593Smuzhiyun struct snd_seq_device *seq_dev;
193*4882a593Smuzhiyun
194*4882a593Smuzhiyun struct snd_midi_channel_set *chset;
195*4882a593Smuzhiyun struct opl4_voice voices[OPL4_MAX_VOICES];
196*4882a593Smuzhiyun struct list_head off_voices;
197*4882a593Smuzhiyun struct list_head on_voices;
198*4882a593Smuzhiyun #endif
199*4882a593Smuzhiyun };
200*4882a593Smuzhiyun
201*4882a593Smuzhiyun /* opl4_lib.c */
202*4882a593Smuzhiyun void snd_opl4_write(struct snd_opl4 *opl4, u8 reg, u8 value);
203*4882a593Smuzhiyun u8 snd_opl4_read(struct snd_opl4 *opl4, u8 reg);
204*4882a593Smuzhiyun void snd_opl4_read_memory(struct snd_opl4 *opl4, char *buf, int offset, int size);
205*4882a593Smuzhiyun void snd_opl4_write_memory(struct snd_opl4 *opl4, const char *buf, int offset, int size);
206*4882a593Smuzhiyun
207*4882a593Smuzhiyun /* opl4_mixer.c */
208*4882a593Smuzhiyun int snd_opl4_create_mixer(struct snd_opl4 *opl4);
209*4882a593Smuzhiyun
210*4882a593Smuzhiyun #ifdef CONFIG_SND_PROC_FS
211*4882a593Smuzhiyun /* opl4_proc.c */
212*4882a593Smuzhiyun int snd_opl4_create_proc(struct snd_opl4 *opl4);
213*4882a593Smuzhiyun void snd_opl4_free_proc(struct snd_opl4 *opl4);
214*4882a593Smuzhiyun #else
snd_opl4_create_proc(struct snd_opl4 * opl4)215*4882a593Smuzhiyun static inline int snd_opl4_create_proc(struct snd_opl4 *opl4) { return 0; }
snd_opl4_free_proc(struct snd_opl4 * opl4)216*4882a593Smuzhiyun static inline void snd_opl4_free_proc(struct snd_opl4 *opl4) {}
217*4882a593Smuzhiyun #endif
218*4882a593Smuzhiyun
219*4882a593Smuzhiyun /* opl4_seq.c */
220*4882a593Smuzhiyun extern int volume_boost;
221*4882a593Smuzhiyun
222*4882a593Smuzhiyun /* opl4_synth.c */
223*4882a593Smuzhiyun void snd_opl4_synth_reset(struct snd_opl4 *opl4);
224*4882a593Smuzhiyun void snd_opl4_synth_shutdown(struct snd_opl4 *opl4);
225*4882a593Smuzhiyun void snd_opl4_note_on(void *p, int note, int vel, struct snd_midi_channel *chan);
226*4882a593Smuzhiyun void snd_opl4_note_off(void *p, int note, int vel, struct snd_midi_channel *chan);
227*4882a593Smuzhiyun void snd_opl4_terminate_note(void *p, int note, struct snd_midi_channel *chan);
228*4882a593Smuzhiyun void snd_opl4_control(void *p, int type, struct snd_midi_channel *chan);
229*4882a593Smuzhiyun void snd_opl4_sysex(void *p, unsigned char *buf, int len, int parsed, struct snd_midi_channel_set *chset);
230*4882a593Smuzhiyun
231*4882a593Smuzhiyun /* yrw801.c */
232*4882a593Smuzhiyun int snd_yrw801_detect(struct snd_opl4 *opl4);
233*4882a593Smuzhiyun extern const struct opl4_region_ptr snd_yrw801_regions[];
234*4882a593Smuzhiyun
235*4882a593Smuzhiyun #endif /* __OPL4_LOCAL_H */
236