summaryrefslogtreecommitdiffstats
path: root/sys/i386/isa/sound
diff options
context:
space:
mode:
Diffstat (limited to 'sys/i386/isa/sound')
-rw-r--r--sys/i386/isa/sound/CHANGELOG22
-rw-r--r--sys/i386/isa/sound/RELNOTES.Linux47
-rw-r--r--sys/i386/isa/sound/adlib_card.c10
-rw-r--r--sys/i386/isa/sound/audio.c20
-rw-r--r--sys/i386/isa/sound/dev_table.c21
-rw-r--r--sys/i386/isa/sound/dev_table.h4
-rw-r--r--sys/i386/isa/sound/dmabuf.c81
-rw-r--r--sys/i386/isa/sound/gus_card.c10
-rw-r--r--sys/i386/isa/sound/gus_hw.h15
-rw-r--r--sys/i386/isa/sound/gus_midi.c10
-rw-r--r--sys/i386/isa/sound/gus_vol.c63
-rw-r--r--sys/i386/isa/sound/gus_wave.c1211
-rw-r--r--sys/i386/isa/sound/midibuf.c12
-rw-r--r--sys/i386/isa/sound/mpu401.c79
-rw-r--r--sys/i386/isa/sound/opl3.c52
-rw-r--r--sys/i386/isa/sound/pas.h6
-rw-r--r--sys/i386/isa/sound/pas2_card.c18
-rw-r--r--sys/i386/isa/sound/pas2_midi.c16
-rw-r--r--sys/i386/isa/sound/pas2_mixer.c14
-rw-r--r--sys/i386/isa/sound/pas2_pcm.c29
-rw-r--r--sys/i386/isa/sound/patmgr.c10
-rw-r--r--sys/i386/isa/sound/sb16_dsp.c86
-rw-r--r--sys/i386/isa/sound/sb16_midi.c65
-rw-r--r--sys/i386/isa/sound/sb_card.c10
-rw-r--r--sys/i386/isa/sound/sb_dsp.c131
-rw-r--r--sys/i386/isa/sound/sb_midi.c42
-rw-r--r--sys/i386/isa/sound/sb_mixer.c89
-rw-r--r--sys/i386/isa/sound/sb_mixer.h40
-rw-r--r--sys/i386/isa/sound/sequencer.c36
-rw-r--r--sys/i386/isa/sound/sound_calls.h7
-rw-r--r--sys/i386/isa/sound/sound_switch.c11
31 files changed, 1302 insertions, 965 deletions
diff --git a/sys/i386/isa/sound/CHANGELOG b/sys/i386/isa/sound/CHANGELOG
index 78168dd..6a9bef1 100644
--- a/sys/i386/isa/sound/CHANGELOG
+++ b/sys/i386/isa/sound/CHANGELOG
@@ -1,6 +1,26 @@
-Changelog for version 2.4
+Changelog for version 2.5
-------------------------
+Since 2.5-beta2
+- Some fine tuning to the GUS v3.7 mixer code.
+- Fixed speed limits for the plain SB (1.0 to 2.0).
+
+Since 2.5-beta
+- Fixed OPL-3 detection with SB. Caused problems with PAS16.
+- GUS v3.7 mixer support.
+
+Since 2.4
+- Mixer support for Sound Galaxy NX Pro (define __SGNXPRO__ on your local.h).
+- Fixed truncated sound on /dev/dsp when the device is closed.
+- Linear volume mode for GUS
+- Pitch bends larger than +/- 2 octaves.
+- MIDI recording for SB and SB Pro. (Untested).
+- Some other fixes.
+- SB16 MIDI and DSP drivers only initialized if SB16 actually installed.
+- Implemented better detection for OPL-3. This should be usefull if you
+ have an old SB Pro (the non-OPL-3 one) or a SB 2.0 clone which has a OPL-3.
+- SVR4.2 support by Ian Hartas. Initial ALPHA TEST version (untested).
+
Since 2.3b
- Fixed bug which made it impossible to make long recordings to disk.
Recording was not restarted after a buffer overflow situation.
diff --git a/sys/i386/isa/sound/RELNOTES.Linux b/sys/i386/isa/sound/RELNOTES.Linux
index bd47ede..ea57d0a 100644
--- a/sys/i386/isa/sound/RELNOTES.Linux
+++ b/sys/i386/isa/sound/RELNOTES.Linux
@@ -1,5 +1,15 @@
-Release notes for the Linux Sound Driver 2.4
+Release notes for the Linux Sound Driver 2.5
--------------------------------------------
+There is also a version called 2.5-beta floating around the net. This
+version contains some fixes after it. Mainly to the SB and GUS code.
+
+CAUTION! The SVR4.2 port has not been tested much. Backup your system
+ carefully before trying it.
+
+This is mainly a bug fix release. There are couple of new things such as
+linear volume mode for GUS and MIDI recording for SB 2.0 and SB Pro.
+Also this version supports the mixer of GUS v3.7. (Support for GUS MAX and
+the 16-bit daughtercard is coming sooner or later).
NOTE! The sound driver is a part of the Linux kernel distribution also.
Check that your kernel doesn't have more recent version than this
@@ -17,17 +27,13 @@ time with the /dev/dsp. Also the WaveBlaster daughter board is supported.
No support for the ASP chip yet (the ASP chip can be installed but it's
not used by the driver).
-You will need the snd-util-2.4.tar.gz and snd-data-0.1.tar.Z
+You will need the snd-util-2.5.tar.gz and snd-data-0.1.tar.Z
packages to use this driver. They should be in the same
ftp site or BBS from where you got this driver. For
example at nic.funet.fi:pub/OS/Linux/*.
-There is a new version of the tracker program available (tracker-3_19.lzh) but
-I don't know where it is available. The tracker 3.10 has bugs and it don't work
-without some fixes. Look at the README of the snd-util-2.3.
-
If you are looking for the installation instructions, please
-look at linux/Readme.
+look at $OS/Readme.
This version supports the following soundcards:
GUS, SoundBlaster, SB Pro, SB16, Pro Audio Spectrum 16 and AdLib.
@@ -220,21 +226,28 @@ This driver contains code by several contributors. In addition several other
persons have given usefull suggestions. The following is a list of major
contributors. (I could have forgotten some names.)
- Craig Metz 1/2 of the PAS16 Mixer and PCM support
- Rob Hooft Volume computation algorithm for the FM synth.
- Mika Liljeberg uLaw encoding and decoding routines
- Greg Lee Volume computation algorithm for the GUS and
- lot's of valuable suggestions.
- Andy Warner ISC port
- Jim Lowe FreeBSD port
+ Craig Metz 1/2 of the PAS16 Mixer and PCM support
+ Rob Hooft Volume computation algorithm for the FM synth.
+ Mika Liljeberg uLaw encoding and decoding routines
+ Greg Lee Volume computation algorithm for the GUS and
+ lot's of valuable suggestions.
+ Andy Warner Initial ISC port
+ Jim Lowe Initial FreeBSD port
Anders Baekgaard Bughunting and valuable suggestions.
- Joerg Schubert SB16 DSP support.
- Andrew Robinson Improvements to the GUS driver
+ Joerg Schubert SB16 DSP support.
+ Andrew Robinson Improvements to the GUS driver
+ Megens SA MIDI recording for SB and SB Pro.
+ Mikael Nordqvist Linear volume support for GUS.
+ Ian Hartas SVR4.2 port
+ Markus Aroharju and
+ Risto Kankkunen Major contributions to the mixer support
+ of GUS v3.7.
+ Hunyue Yau Sound Galaxy NX Pro mixer support.
Regards,
Hannu Savolainen
-hsavolai@cs.helsinki.fi
+hannu@voxware.pp.fi, Hannu.Savolainen@Helsinki.fi
Snail mail: Hannu Savolainen
Pallaksentie 4 A 2
diff --git a/sys/i386/isa/sound/adlib_card.c b/sys/i386/isa/sound/adlib_card.c
index 3b7df5e..6365069 100644
--- a/sys/i386/isa/sound/adlib_card.c
+++ b/sys/i386/isa/sound/adlib_card.c
@@ -1,10 +1,10 @@
/*
* sound/adlib_card.c
- *
+ *
* Detection routine for the AdLib card.
- *
+ *
* Copyright by Hannu Savolainen 1993
- *
+ *
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions are
* met: 1. Redistributions of source code must retain the above copyright
@@ -12,7 +12,7 @@
* Redistributions in binary form must reproduce the above copyright notice,
* this list of conditions and the following disclaimer in the documentation
* and/or other materials provided with the distribution.
- *
+ *
* THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND ANY
* EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
* WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
@@ -24,7 +24,7 @@
* LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
* OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
* SUCH DAMAGE.
- *
+ *
*/
#include "sound_config.h"
diff --git a/sys/i386/isa/sound/audio.c b/sys/i386/isa/sound/audio.c
index 84c1f6c..f27f9d5 100644
--- a/sys/i386/isa/sound/audio.c
+++ b/sys/i386/isa/sound/audio.c
@@ -1,10 +1,10 @@
/*
* sound/audio.c
- *
+ *
* Device file manager for /dev/audio
- *
+ *
* Copyright by Hannu Savolainen 1993
- *
+ *
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions are
* met: 1. Redistributions of source code must retain the above copyright
@@ -12,7 +12,7 @@
* Redistributions in binary form must reproduce the above copyright notice,
* this list of conditions and the following disclaimer in the documentation
* and/or other materials provided with the distribution.
- *
+ *
* THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND ANY
* EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
* WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
@@ -24,7 +24,7 @@
* LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
* OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
* SUCH DAMAGE.
- *
+ *
*/
#include "sound_config.h"
@@ -116,9 +116,9 @@ translate_bytes (const void *table, void *buff, unsigned long n)
"1:\tlodsb\n\t"
"xlatb\n\t"
"stosb\n\t"
-"loop 1b\n\t":
-: "b" ((long) table), "c" (n), "D" ((long) buff), "S" ((long) buff)
-: "bx", "cx", "di", "si", "ax");
+ "loop 1b\n\t":
+ :"b" ((long) table), "c" (n), "D" ((long) buff), "S" ((long) buff)
+ :"bx", "cx", "di", "si", "ax");
}
#endif
@@ -183,7 +183,7 @@ audio_write (int dev, struct fileinfo *file, snd_rw_buf * buf, int count)
/* This just allows interrupts while the conversion is running */
__asm__ ("sti");
#endif
- translate_bytes (ulaw_dsp, &wr_dma_buf[dev][wr_buff_ptr[dev]], l);
+ translate_bytes (ulaw_dsp, (unsigned char *) &wr_dma_buf[dev][wr_buff_ptr[dev]], l);
}
c -= l;
@@ -244,7 +244,7 @@ audio_read (int dev, struct fileinfo *file, snd_rw_buf * buf, int count)
__asm__ ("sti");
#endif
- translate_bytes (dsp_ulaw, dmabuf, l);
+ translate_bytes (dsp_ulaw, (unsigned char *) dmabuf, l);
}
COPY_TO_USER (buf, p, dmabuf, l);
diff --git a/sys/i386/isa/sound/dev_table.c b/sys/i386/isa/sound/dev_table.c
index 5f6fc7b..7f7cae1 100644
--- a/sys/i386/isa/sound/dev_table.c
+++ b/sys/i386/isa/sound/dev_table.c
@@ -1,10 +1,10 @@
/*
* sound/dev_table.c
- *
+ *
* Device call tables.
- *
+ *
* Copyright by Hannu Savolainen 1993
- *
+ *
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions are
* met: 1. Redistributions of source code must retain the above copyright
@@ -12,7 +12,7 @@
* Redistributions in binary form must reproduce the above copyright notice,
* this list of conditions and the following disclaimer in the documentation
* and/or other materials provided with the distribution.
- *
+ *
* THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND ANY
* EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
* WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
@@ -24,7 +24,7 @@
* LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
* OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
* SUCH DAMAGE.
- *
+ *
*/
#define _DEV_TABLE_C_
@@ -116,12 +116,12 @@ sndtable_get_cardcount (void)
}
#ifdef linux
-void
+void
sound_setup (char *str, int *ints)
{
int i, n = sizeof (supported_drivers) / sizeof (struct card_info);
-/*
+ /*
* First disable all drivers
*/
@@ -130,7 +130,7 @@ sound_setup (char *str, int *ints)
if (ints[0] == 0 || ints[1] == 0)
return;
-/*
+ /*
* Then enable them one by time
*/
@@ -169,8 +169,9 @@ sound_setup (char *str, int *ints)
}
}
}
+
#else
-void
+void
sound_chconf (int card_type, int ioaddr, int irq, int dma)
{
int i, n = sizeof (supported_drivers) / sizeof (struct card_info);
@@ -193,6 +194,7 @@ sound_chconf (int card_type, int ioaddr, int irq, int dma)
supported_drivers[ptr].config.dma = dma;
}
}
+
#endif
struct address_info *
@@ -211,4 +213,5 @@ sound_getconf (int card_type)
return &supported_drivers[ptr].config;
}
+
#endif
diff --git a/sys/i386/isa/sound/dev_table.h b/sys/i386/isa/sound/dev_table.h
index 63e4776..4b656ba 100644
--- a/sys/i386/isa/sound/dev_table.h
+++ b/sys/i386/isa/sound/dev_table.h
@@ -58,6 +58,9 @@ struct generic_midi_info{
struct audio_operations {
char name[32];
+ int flags;
+#define NOTHING_SPECIAL 0
+#define NEEDS_RESTART 1
int (*open) (int dev, int mode);
void (*close) (int dev);
void (*output_block) (int dev, unsigned long buf,
@@ -96,6 +99,7 @@ struct synth_operations {
void (*aftertouch) (int dev, int voice, int pressure);
void (*controller) (int dev, int voice, int ctrl_num, int value);
void (*panning) (int dev, int voice, int value);
+ void (*volume_method) (int dev, int mode);
int (*pmgr_interface) (int dev, struct patmgr_info *info);
};
diff --git a/sys/i386/isa/sound/dmabuf.c b/sys/i386/isa/sound/dmabuf.c
index 4b75e40..851a70a1 100644
--- a/sys/i386/isa/sound/dmabuf.c
+++ b/sys/i386/isa/sound/dmabuf.c
@@ -1,10 +1,10 @@
/*
* sound/dmabuf.c
- *
+ *
* The DMA buffer manager for digitized voice applications
- *
+ *
* Copyright by Hannu Savolainen 1993
- *
+ *
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions are
* met: 1. Redistributions of source code must retain the above copyright
@@ -12,7 +12,7 @@
* Redistributions in binary form must reproduce the above copyright notice,
* this list of conditions and the following disclaimer in the documentation
* and/or other materials provided with the distribution.
- *
+ *
* THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND ANY
* EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
* WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
@@ -24,7 +24,7 @@
* LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
* OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
* SUCH DAMAGE.
- *
+ *
*/
#include "sound_config.h"
@@ -49,10 +49,6 @@
#define DMODE_OUTPUT 1
#define DMODE_INPUT 2
-extern int sb_dsp_ok;
-extern struct audio_operations sb_dsp_operations;
-static int force_reset = 0;
-
DEFINE_WAIT_QUEUES (dev_sleeper[MAX_DSP_DEV], dev_sleep_flag[MAX_DSP_DEV]);
static int dma_mode[MAX_DSP_DEV] =
@@ -91,7 +87,7 @@ static int bufferalloc_done[MAX_DSP_DEV] =
* Logical buffers for each devices
*/
-static int dev_nbufs[MAX_DSP_DEV]; /* # of logical buffers ( >=
+static int dev_nbufs[MAX_DSP_DEV]; /* # of logical buffers ( >=
* sound_buffcounts[dev] */
static int dev_counts[MAX_DSP_DEV][MAX_SUB_BUFFERS];
@@ -243,9 +239,6 @@ dma_reset (int dev)
int retval;
unsigned long flags;
- if (!force_reset && sb_dsp_ok && dsp_devs[dev] == &sb_dsp_operations)
- return; /* We don't need this code for SB */
-
DISABLE_INTR (flags);
dsp_devs[dev]->reset (dev);
dsp_devs[dev]->close (dev);
@@ -262,23 +255,18 @@ static int
dma_sync (int dev)
{
unsigned long flags;
- unsigned long time;
- int timed_out;
if (dma_mode[dev] == DMODE_OUTPUT)
{
DISABLE_INTR (flags);
- timed_out = 0;
- time = GET_TIME ();
-
while ((!(PROCESS_ABORTING (dev_sleeper[dev], dev_sleep_flag[dev]) ||
- dmabuf_interrupted[dev]) && !timed_out)
+ dmabuf_interrupted[dev]))
&& dev_qlen[dev])
{
DO_SLEEP (dev_sleeper[dev], dev_sleep_flag[dev], 10 * HZ);
- if ((GET_TIME () - time) > (10 * HZ))
- timed_out = 1;
+ if (TIMED_OUT (dev_sleeper[dev], dev_sleep_flag[dev]))
+ return dev_qlen[dev];
}
RESTORE_INTR (flags);
@@ -445,9 +433,7 @@ DMAbuf_ioctl (int dev, unsigned int cmd, unsigned int arg, int local)
switch (cmd)
{
case SNDCTL_DSP_RESET:
- force_reset = 1;
dma_reset (dev);
- force_reset = 0;
return 0;
break;
@@ -494,6 +480,7 @@ DMAbuf_ioctl (int dev, unsigned int cmd, unsigned int arg, int local)
return dsp_devs[dev]->ioctl (dev, cmd, arg, local);
}
+ /* NOTREACHED */
return RET_ERROR (EIO);
}
@@ -576,7 +563,8 @@ DMAbuf_start_output (int dev, int buff_no, int l)
dev_counts[dev][dev_qtail[dev]] = l;
- dev_needs_restart[dev] = (l != dev_buffsize[dev]);
+ dev_needs_restart[dev] = (l != dev_buffsize[dev]) &&
+ (sound_dma_automode[dev] || dsp_devs[dev]->flags & NEEDS_RESTART);
dev_qtail[dev] = (dev_qtail[dev] + 1) % dev_nbufs[dev];
@@ -615,11 +603,11 @@ DMAbuf_start_dma (int dev, unsigned long physaddr, int count, int dma_mode)
disable_dma (chan);
clear_dma_ff (chan);
set_dma_mode (chan, dma_mode | DMA_AUTOINIT);
- set_dma_addr (chan, (caddr_t)snd_raw_buf_phys[dev][0]);
- set_dma_count (chan, (unsigned)sound_buffsizes[dev]);
- enable_dma ((unsigned)chan);
+ set_dma_addr (chan, snd_raw_buf_phys[dev][0]);
+ set_dma_count (chan, sound_buffsizes[dev]);
+ enable_dma (chan);
RESTORE_INTR (flags);
-#else
+#else /* linux */
#ifdef __386BSD__
printk ("sound: Invalid DMA mode for device %d\n", dev);
@@ -628,24 +616,24 @@ DMAbuf_start_dma (int dev, unsigned long physaddr, int count, int dma_mode)
(caddr_t)snd_raw_buf_phys[dev][0],
sound_buffsizes[dev],
chan);
-#else
-#if defined(ISC) || defined(SCO)
+#else /* __386BSD__ */
+#if defined(ISC) || defined(SCO) || defined(SVR42)
#ifndef DMAMODE_AUTO
printk ("sound: Invalid DMA mode for device %d\n", dev);
-#endif
+#endif /* DMAMODE_AUTO */
dma_param (chan, ((dma_mode == DMA_MODE_READ) ? DMA_Rdmode : DMA_Wrmode)
#ifdef DMAMODE_AUTO
| DMAMODE_AUTO
-#endif
+#endif /* DMAMODE_AUTO */
,
snd_raw_buf_phys[dev][0], count);
dma_enable (chan);
-#else
+#else /* SYSV */
#error This routine is not valid for this OS.
-#endif
-#endif
+#endif /* SYSV */
+#endif /* __386BSD__ */
-#endif
+#endif /* linux */
}
else
{
@@ -658,24 +646,24 @@ DMAbuf_start_dma (int dev, unsigned long physaddr, int count, int dma_mode)
set_dma_count (chan, count);
enable_dma (chan);
RESTORE_INTR (flags);
-#else
+#else /* linux */
#ifdef __386BSD__
isa_dmastart ((dma_mode == DMA_MODE_READ) ? B_READ : B_WRITE,
(caddr_t)physaddr,
count,
chan);
-#else
+#else /* __386BSD__ */
-#if defined(ISC) || defined(SCO)
+#if defined(ISC) || defined(SCO) || defined(SVR42)
dma_param (chan, ((dma_mode == DMA_MODE_READ) ? DMA_Rdmode : DMA_Wrmode),
physaddr, count);
dma_enable (chan);
-#else
+#else /* SYSV */
#error This routine is not valid for this OS.
-#endif /* !ISC */
-#endif
+#endif /* SYSV */
+#endif /* __386BSD__ */
-#endif
+#endif /* linux */
}
return count;
@@ -724,7 +712,8 @@ DMAbuf_outputintr (int dev, int underrun_flag)
{
dev_underrun[dev]++;
dsp_devs[dev]->halt_xfer (dev);
- dev_needs_restart[dev] = 1;
+ dev_needs_restart[dev] = (sound_dma_automode[dev] ||
+ dsp_devs[dev]->flags & NEEDS_RESTART);
}
DISABLE_INTR (flags);
@@ -750,7 +739,7 @@ DMAbuf_inputintr (int dev)
dev_underrun[dev]++;
dsp_devs[dev]->halt_xfer (dev);
dev_active[dev] = 0;
- dev_needs_restart[dev] = 1;
+ dev_needs_restart[dev] = sound_dma_automode[dev];
}
else
{
@@ -810,7 +799,7 @@ DMAbuf_reset_dma (int chan)
/*
* The sound_mem_init() is called by mem_init() immediately after mem_map is
* initialized and before free_page_list is created.
- *
+ *
* This routine allocates DMA buffers at the end of available physical memory (
* <16M) and marks pages reserved at mem_map.
*/
diff --git a/sys/i386/isa/sound/gus_card.c b/sys/i386/isa/sound/gus_card.c
index 29cc3d5..c7cfc0a 100644
--- a/sys/i386/isa/sound/gus_card.c
+++ b/sys/i386/isa/sound/gus_card.c
@@ -1,10 +1,10 @@
/*
* sound/gus_card.c
- *
+ *
* Detection routine for the Gravis Ultrasound.
- *
+ *
* Copyright by Hannu Savolainen 1993
- *
+ *
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions are
* met: 1. Redistributions of source code must retain the above copyright
@@ -12,7 +12,7 @@
* Redistributions in binary form must reproduce the above copyright notice,
* this list of conditions and the following disclaimer in the documentation
* and/or other materials provided with the distribution.
- *
+ *
* THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND ANY
* EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
* WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
@@ -24,7 +24,7 @@
* LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
* OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
* SUCH DAMAGE.
- *
+ *
*/
#include "sound_config.h"
diff --git a/sys/i386/isa/sound/gus_hw.h b/sys/i386/isa/sound/gus_hw.h
index 48233e7..f97a0b8 100644
--- a/sys/i386/isa/sound/gus_hw.h
+++ b/sys/i386/isa/sound/gus_hw.h
@@ -24,6 +24,8 @@
#define u_Command (gus_base + 0x103)
#define u_DataLo (gus_base + 0x104)
#define u_DataHi (gus_base + 0x105)
+#define u_MixData (gus_base + 0x106) /* Rev. 3.7+ mixing */
+#define u_MixSelect (gus_base + 0x506) /* registers. */
#define u_IrqStatus u_Status
# define MIDI_TX_IRQ 0x01 /* pending MIDI xmit IRQ */
# define MIDI_RX_IRQ 0x02 /* pending MIDI recv IRQ */
@@ -32,4 +34,17 @@
# define WAVETABLE_IRQ 0x20 /* pending wavetable IRQ */
# define ENVELOPE_IRQ 0x40 /* pending volume envelope IRQ */
# define DMA_TC_IRQ 0x80 /* pending dma tc IRQ */
+
+#define ICS2101 1
+# define ICS_MIXDEVS 6
+# define DEV_MIC 0
+# define DEV_LINE 1
+# define DEV_CD 2
+# define DEV_GF1 3
+# define DEV_UNUSED 4
+# define DEV_VOL 5
+
+# define CHN_LEFT 0
+# define CHN_RIGHT 1
+#define CS4231 2
#define u_DRAMIO (gus_base + 0x107)
diff --git a/sys/i386/isa/sound/gus_midi.c b/sys/i386/isa/sound/gus_midi.c
index 5e06f7f..935c5c9 100644
--- a/sys/i386/isa/sound/gus_midi.c
+++ b/sys/i386/isa/sound/gus_midi.c
@@ -1,10 +1,10 @@
/*
* sound/gus2_midi.c
- *
+ *
* The low level driver for the GUS Midi Interface.
- *
+ *
* Copyright by Hannu Savolainen 1993
- *
+ *
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions are
* met: 1. Redistributions of source code must retain the above copyright
@@ -12,7 +12,7 @@
* Redistributions in binary form must reproduce the above copyright notice,
* this list of conditions and the following disclaimer in the documentation
* and/or other materials provided with the distribution.
- *
+ *
* THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND ANY
* EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
* WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
@@ -24,7 +24,7 @@
* LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
* OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
* SUCH DAMAGE.
- *
+ *
*/
#include "sound_config.h"
diff --git a/sys/i386/isa/sound/gus_vol.c b/sys/i386/isa/sound/gus_vol.c
index 6e56d54..055a117 100644
--- a/sys/i386/isa/sound/gus_vol.c
+++ b/sys/i386/isa/sound/gus_vol.c
@@ -1,17 +1,18 @@
-/*
+/*
* gus_vol.c - Compute volume for GUS.
- *
+ *
* Greg Lee 1993.
*/
#include "sound_config.h"
#ifndef EXCLUDE_GUS
+#include "gus_linearvol.h"
#define GUS_VOLUME gus_wave_volume
extern int gus_wave_volume;
-/*
+/*
* Calculate gus volume from note velocity, main volume, expression, and
* intrinsic patch volume given in patch library. Expression is multiplied
* in, so it emphasizes differences in note velocity, while main volume is
@@ -20,7 +21,7 @@ extern int gus_wave_volume;
* to expression controller messages, if they were found to be used for
* dynamic volume adjustments, so here, main volume can be assumed to be
* constant throughout a song.)
- *
+ *
* Intrinsic patch volume is added in, but if over 64 is also multiplied in, so
* we can give a big boost to very weak voices like nylon guitar and the
* basses. The normal value is 64. Strings are assigned lower values.
@@ -31,27 +32,27 @@ gus_adagio_vol (int vel, int mainv, int xpn, int voicev)
int i, m, n, x;
- /*
+ /*
* A voice volume of 64 is considered neutral, so adjust the main volume if
* something other than this neutral value was assigned in the patch
* library.
*/
x = 256 + 6 * (voicev - 64);
- /*
- * Boost expression by voice volume above neutral.
+ /*
+ * Boost expression by voice volume above neutral.
*/
if (voicev > 65)
xpn += voicev - 64;
xpn += (voicev - 64) / 2;
- /*
- * Combine multiplicative and level components.
+ /*
+ * Combine multiplicative and level components.
*/
x = vel * xpn * 6 + (voicev / 4) * x;
#ifdef GUS_VOLUME
- /*
+ /*
* Further adjustment by installation-specific master volume control
* (default 60).
*/
@@ -72,7 +73,7 @@ gus_adagio_vol (int vel, int mainv, int xpn, int voicev)
else if (x >= 65535)
return ((15 << 8) | 255);
- /*
+ /*
* Convert to gus's logarithmic form with 4 bit exponent i and 8 bit
* mantissa m.
*/
@@ -89,14 +90,14 @@ gus_adagio_vol (int vel, int mainv, int xpn, int voicev)
n >>= 1;
i++;
}
- /*
+ /*
* Mantissa is part of linear volume not expressed in exponent. (This is
* not quite like real logs -- I wonder if it's right.)
*/
m = x - (1 << i);
- /*
- * Adjust mantissa to 8 bits.
+ /*
+ * Adjust mantissa to 8 bits.
*/
if (m > 0)
{
@@ -109,4 +110,38 @@ gus_adagio_vol (int vel, int mainv, int xpn, int voicev)
return ((i << 8) + m);
}
+/*
+ * Volume-values are interpreted as linear values. Volume is based on the
+ * value supplied with SEQ_START_NOTE(), channel main volume (if compiled in)
+ * and the volume set by the mixer-device (default 60%).
+ */
+
+unsigned short
+gus_linear_vol (int vol, int mainvol)
+{
+ int mixer_mainvol;
+
+ if (vol <= 0)
+ vol = 0;
+ else if (vol >= 127)
+ vol = 127;
+
+#ifdef GUS_VOLUME
+ mixer_mainvol = GUS_VOLUME;
+#else
+ mixer_mainvol = 100;
+#endif
+
+#ifdef GUS_USE_CHN_MAIN_VOLUME
+ if (mainvol <= 0)
+ mainvol = 0;
+ else if (mainvol >= 127)
+ mainvol = 127;
+#else
+ mainvol = 128;
+#endif
+
+ return gus_linearvol[(((vol * mainvol) / 128) * mixer_mainvol) / 100];
+}
+
#endif
diff --git a/sys/i386/isa/sound/gus_wave.c b/sys/i386/isa/sound/gus_wave.c
index db5ad81..948dd407 100644
--- a/sys/i386/isa/sound/gus_wave.c
+++ b/sys/i386/isa/sound/gus_wave.c
@@ -1,11 +1,10 @@
-
-/*
+/*
* sound/gus_wave.c
- *
+ *
* Driver for the Gravis UltraSound wave table synth.
- *
+ *
* Copyright by Hannu Savolainen 1993
- *
+ *
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions are
* met: 1. Redistributions of source code must retain the above copyright
@@ -13,7 +12,7 @@
* Redistributions in binary form must reproduce the above copyright notice,
* this list of conditions and the following disclaimer in the documentation
* and/or other materials provided with the distribution.
- *
+ *
* THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND ANY
* EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
* WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
@@ -25,13 +24,11 @@
* LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
* OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
* SUCH DAMAGE.
- *
+ *
*/
#include "sound_config.h"
-#ifdef linux
-#include <linux/ultrasound.h>
-#elif __FreeBSD__
+#ifdef __FreeBSD__
#include <machine/ultrasound.h>
#else
#include "ultrasound.h"
@@ -67,7 +64,7 @@ struct voice_info
unsigned char env_rate[6];
unsigned char env_offset[6];
- /*
+ /*
* Volume computation parameters for gus_adagio_vol()
*/
int main_vol, expression_vol, patch_vol;
@@ -94,22 +91,21 @@ static int gus_line_vol = 100, gus_mic_vol = 0;
static int gus_recmask = SOUND_MASK_MIC;
static int recording_active = 0;
-#define VOL_METHOD_ADAGIO 1
int gus_wave_volume = 60;
int gus_pcm_volume = 80;
static unsigned char mix_image = 0x00;
-/*
+/*
* Current version of this driver doesn't allow synth and PCM functions
* at the same time. The active_device specifies the active driver
*/
static int active_device = 0;
-#define GUS_DEV_WAVE 1 /*
+#define GUS_DEV_WAVE 1 /*
* * * Wave table synth */
-#define GUS_DEV_PCM_DONE 2 /*
+#define GUS_DEV_PCM_DONE 2 /*
* * * PCM device, transfer done */
-#define GUS_DEV_PCM_CONTINUE 3 /*
+#define GUS_DEV_PCM_CONTINUE 3 /*
* * * PCM device, transfer the
* second * * * chn */
@@ -119,31 +115,31 @@ static int gus_sampling_bits;
DEFINE_WAIT_QUEUE (dram_sleeper, dram_sleep_flag);
-/*
+/*
* Variables and buffers for PCM output
*/
-#define MAX_PCM_BUFFERS (32*MAX_REALTIME_FACTOR) /*
+#define MAX_PCM_BUFFERS (32*MAX_REALTIME_FACTOR) /*
* * * Don't
- * * * change
- *
+ * * * change
+ *
*/
-static int pcm_bsize, /*
- * Current blocksize
+static int pcm_bsize, /*
+ * Current blocksize
*/
- pcm_nblk, /*
- * Current # of blocks
+ pcm_nblk, /*
+ * Current # of blocks
*/
- pcm_banksize; /*
+ pcm_banksize; /*
* * * * # bytes allocated for channels */
-static int pcm_datasize[MAX_PCM_BUFFERS]; /*
+static int pcm_datasize[MAX_PCM_BUFFERS]; /*
* * * * Actual # of bytes
* in blk * */
-static volatile int pcm_head, pcm_tail, pcm_qlen; /*
+static volatile int pcm_head, pcm_tail, pcm_qlen; /*
* * * * DRAM queue
@@ -160,62 +156,62 @@ struct voice_info voices[32];
static int freq_div_table[] =
{
- 44100, /*
- * 14
+ 44100, /*
+ * 14
*/
- 41160, /*
- * 15
+ 41160, /*
+ * 15
*/
- 38587, /*
- * 16
+ 38587, /*
+ * 16
*/
- 36317, /*
- * 17
+ 36317, /*
+ * 17
*/
- 34300, /*
- * 18
+ 34300, /*
+ * 18
*/
- 32494, /*
- * 19
+ 32494, /*
+ * 19
*/
- 30870, /*
- * 20
+ 30870, /*
+ * 20
*/
- 29400, /*
- * 21
+ 29400, /*
+ * 21
*/
- 28063, /*
- * 22
+ 28063, /*
+ * 22
*/
- 26843, /*
- * 23
+ 26843, /*
+ * 23
*/
- 25725, /*
- * 24
+ 25725, /*
+ * 24
*/
- 24696, /*
- * 25
+ 24696, /*
+ * 25
*/
- 23746, /*
- * 26
+ 23746, /*
+ * 26
*/
- 22866, /*
- * 27
+ 22866, /*
+ * 27
*/
- 22050, /*
- * 28
+ 22050, /*
+ * 28
*/
- 21289, /*
- * 29
+ 21289, /*
+ * 29
*/
- 20580, /*
- * 30
+ 20580, /*
+ * 30
*/
- 19916, /*
- * 31
+ 19916, /*
+ * 31
*/
- 19293 /*
- * 32
+ 19293 /*
+ * 32
*/
};
@@ -234,13 +230,14 @@ static struct synth_info gus_info =
static void gus_poke (long addr, unsigned char data);
static void compute_and_set_volume (int voice, int volume, int ramp_time);
extern unsigned short gus_adagio_vol (int vel, int mainv, int xpn, int voicev);
+extern unsigned short gus_linear_vol (int vol, int mainvol);
static void compute_volume (int voice, int volume);
static void do_volume_irq (int voice);
static void set_input_volumes (void);
-#define INSTANT_RAMP -1 /*
+#define INSTANT_RAMP -1 /*
* * * Dont use ramping */
-#define FAST_RAMP 0 /*
+#define FAST_RAMP 0 /*
* * * Fastest possible ramp */
static void
@@ -255,8 +252,8 @@ reset_sample_memory (void)
for (i = 0; i < 32; i++)
patch_map[i] = -1;
- gus_poke (0, 0); /*
- * Put silence here
+ gus_poke (0, 0); /*
+ * Put silence here
*/
gus_poke (1, 0);
@@ -392,7 +389,7 @@ gus_write_addr (int reg, unsigned long address, int is16bit)
if (is16bit)
{
- /*
+ /*
* Special processing required for 16 bit patches
*/
@@ -404,6 +401,10 @@ gus_write_addr (int reg, unsigned long address, int is16bit)
gus_write16 (reg, (unsigned short) ((address >> 7) & 0xffff));
gus_write16 (reg + 1, (unsigned short) ((address << 9) & 0xffff));
+ /* Could writing twice fix problems with GUS_VOICE_POS() ? Lets try... */
+ gus_delay ();
+ gus_write16 (reg, (unsigned short) ((address >> 7) & 0xffff));
+ gus_write16 (reg + 1, (unsigned short) ((address << 9) & 0xffff));
}
static void
@@ -447,13 +448,13 @@ gus_voice_mode (unsigned int m)
{
unsigned char mode = (unsigned char) (m & 0xff);
- gus_write8 (0x00, (gus_read8 (0x00) & 0x03) | (mode & 0xfc)); /*
- * Don't
- * start
- * or
+ gus_write8 (0x00, (gus_read8 (0x00) & 0x03) | (mode & 0xfc)); /*
+ * Don't
+ * start
+ * or
* stop
* *
- * voice
+ * voice
*/
gus_delay ();
gus_write8 (0x00, (gus_read8 (0x00) & 0x03) | (mode & 0xfc));
@@ -474,8 +475,8 @@ gus_voice_freq (unsigned long freq)
static void
gus_voice_volume (unsigned int vol)
{
- gus_write8 (0x0d, 0x03); /*
- * Stop ramp before setting volume
+ gus_write8 (0x0d, 0x03); /*
+ * Stop ramp before setting volume
*/
gus_write16 (0x09, (unsigned short) (vol << 4));
}
@@ -514,13 +515,13 @@ gus_ramp_mode (unsigned int m)
{
unsigned char mode = (unsigned char) (m & 0xff);
- gus_write8 (0x0d, (gus_read8 (0x0d) & 0x03) | (mode & 0xfc)); /*
- * Don't
- * start
- * or
+ gus_write8 (0x0d, (gus_read8 (0x0d) & 0x03) | (mode & 0xfc)); /*
+ * Don't
+ * start
+ * or
* stop
* *
- * ramping
+ * ramping
*/
gus_delay ();
gus_write8 (0x0d, (gus_read8 (0x0d) & 0x03) | (mode & 0xfc));
@@ -554,14 +555,14 @@ gus_voice_init (int voice)
DISABLE_INTR (flags);
gus_select_voice (voice);
gus_voice_volume (0);
- gus_write_addr (0x0a, 0, 0); /*
- * Set current position to 0
+ gus_write_addr (0x0a, 0, 0); /*
+ * Set current position to 0
*/
- gus_write8 (0x00, 0x03); /*
- * Voice off
+ gus_write8 (0x00, 0x03); /*
+ * Voice off
*/
- gus_write8 (0x0d, 0x03); /*
- * Ramping off
+ gus_write8 (0x0d, 0x03); /*
+ * Ramping off
*/
RESTORE_INTR (flags);
@@ -594,18 +595,22 @@ step_envelope (int voice)
{
unsigned vol, prev_vol, phase;
unsigned char rate;
+ long int flags;
if (voices[voice].mode & WAVE_SUSTAIN_ON && voices[voice].env_phase == 2)
{
+ DISABLE_INTR (flags);
+ gus_select_voice (voice);
gus_rampoff ();
- return; /*
- * Sustain
+ RESTORE_INTR (flags);
+ return; /*
+ * Sustain
*/
}
if (voices[voice].env_phase >= 5)
{
- /*
+ /*
* Shoot the voice off
*/
@@ -614,25 +619,30 @@ step_envelope (int voice)
}
prev_vol = voices[voice].current_volume;
- gus_voice_volume (prev_vol);
phase = ++voices[voice].env_phase;
-
compute_volume (voice, voices[voice].midi_volume);
-
vol = voices[voice].initial_volume * voices[voice].env_offset[phase] / 255;
rate = voices[voice].env_rate[phase];
- gus_write8 (0x06, rate); /*
- * Ramping rate
+
+ DISABLE_INTR (flags);
+ gus_select_voice (voice);
+
+ gus_voice_volume (prev_vol);
+
+
+ gus_write8 (0x06, rate); /*
+ * Ramping rate
*/
voices[voice].volume_irq_mode = VMODE_ENVELOPE;
- if (((vol - prev_vol) / 64) == 0) /*
- * No significant volume change
+ if (((vol - prev_vol) / 64) == 0) /*
+ * No significant volume change
*/
{
- step_envelope (voice); /*
- * Continue with the next phase
+ RESTORE_INTR (flags);
+ step_envelope (voice); /*
+ * Continue with the next phase
*/
return;
}
@@ -642,8 +652,8 @@ step_envelope (int voice)
if (vol >= (4096 - 64))
vol = 4096 - 65;
gus_ramp_range (0, vol);
- gus_rampon (0x20); /*
- * Increasing, irq
+ gus_rampon (0x20); /*
+ * Increasing, irq
*/
}
else
@@ -651,11 +661,12 @@ step_envelope (int voice)
if (vol <= 64)
vol = 65;
gus_ramp_range (vol, 4030);
- gus_rampon (0x60); /*
- * Decreasing, irq
+ gus_rampon (0x60); /*
+ * Decreasing, irq
*/
}
voices[voice].current_volume = vol;
+ RESTORE_INTR (flags);
}
static void
@@ -668,25 +679,26 @@ init_envelope (int voice)
}
static void
-start_release (int voice)
+start_release (int voice, long int flags)
{
if (gus_read8 (0x00) & 0x03)
- return; /*
- * Voice already stopped
+ return; /*
+ * Voice already stopped
*/
- voices[voice].env_phase = 2; /*
- * Will be incremented by step_envelope
+ voices[voice].env_phase = 2; /*
+ * Will be incremented by step_envelope
*/
voices[voice].current_volume =
voices[voice].initial_volume =
- gus_read16 (0x09) >> 4; /*
- * Get current volume
+ gus_read16 (0x09) >> 4; /*
+ * Get current volume
*/
voices[voice].mode &= ~WAVE_SUSTAIN_ON;
gus_rampoff ();
+ RESTORE_INTR (flags);
step_envelope (voice);
}
@@ -694,32 +706,37 @@ static void
gus_voice_fade (int voice)
{
int instr_no = sample_map[voice], is16bits;
+ long int flags;
+
+ DISABLE_INTR (flags);
+ gus_select_voice (voice);
if (instr_no < 0 || instr_no > MAX_SAMPLE)
{
- gus_write8 (0x00, 0x03); /*
- * Hard stop
+ gus_write8 (0x00, 0x03); /*
+ * Hard stop
*/
+ RESTORE_INTR (flags);
return;
}
- is16bits = (samples[instr_no].mode & WAVE_16_BITS) ? 1 : 0; /*
+ is16bits = (samples[instr_no].mode & WAVE_16_BITS) ? 1 : 0; /*
* 8 or 16
* bit
- * samples
+ * samples
*/
if (voices[voice].mode & WAVE_ENVELOPES)
{
- start_release (voice);
+ start_release (voice, flags);
return;
}
- /*
+ /*
* Ramp the volume down but not too quickly.
*/
- if ((gus_read16 (0x09) >> 4) < 100) /*
- * Get current volume
+ if ((gus_read16 (0x09) >> 4) < 100) /*
+ * Get current volume
*/
{
gus_voice_off ();
@@ -730,10 +747,11 @@ gus_voice_fade (int voice)
gus_ramp_range (65, 4030);
gus_ramp_rate (2, 4);
- gus_rampon (0x40 | 0x20); /*
- * Down, once, irq
+ gus_rampon (0x40 | 0x20); /*
+ * Down, once, irq
*/
voices[voice].volume_irq_mode = VMODE_HALT;
+ RESTORE_INTR (flags);
}
static void
@@ -748,24 +766,24 @@ gus_reset (void)
for (i = 0; i < 32; i++)
{
- gus_voice_init (i); /*
- * Turn voice off
+ gus_voice_init (i); /*
+ * Turn voice off
*/
gus_voice_init2 (i);
}
- INB (u_Status); /*
- * Touch the status register
+ INB (u_Status); /*
+ * Touch the status register
*/
- gus_look8 (0x41); /*
- * Clear any pending DMA IRQs
+ gus_look8 (0x41); /*
+ * Clear any pending DMA IRQs
*/
- gus_look8 (0x49); /*
- * Clear any pending sample IRQs
+ gus_look8 (0x49); /*
+ * Clear any pending sample IRQs
*/
- gus_read8 (0x0f); /*
- * Clear pending IRQs
+ gus_read8 (0x0f); /*
+ * Clear pending IRQs
*/
}
@@ -774,7 +792,7 @@ static void
gus_initialize (void)
{
unsigned long flags;
- unsigned char dma_image, irq_image, tmp;
+ register unsigned char dma_image, irq_image, tmp;
static unsigned char gus_irq_map[16] =
{0, 0, 1, 3, 0, 2, 0, 4, 0, 0, 0, 5, 6, 0, 0, 7};
@@ -784,74 +802,74 @@ gus_initialize (void)
DISABLE_INTR (flags);
- gus_write8 (0x4c, 0); /*
- * Reset GF1
+ gus_write8 (0x4c, 0); /*
+ * Reset GF1
*/
gus_delay ();
gus_delay ();
- gus_write8 (0x4c, 1); /*
- * Release Reset
+ gus_write8 (0x4c, 1); /*
+ * Release Reset
*/
gus_delay ();
gus_delay ();
- /*
+ /*
* Clear all interrupts
*/
- gus_write8 (0x41, 0); /*
- * DMA control
+ gus_write8 (0x41, 0); /*
+ * DMA control
*/
- gus_write8 (0x45, 0); /*
- * Timer control
+ gus_write8 (0x45, 0); /*
+ * Timer control
*/
- gus_write8 (0x49, 0); /*
- * Sample control
+ gus_write8 (0x49, 0); /*
+ * Sample control
*/
gus_select_max_voices (24);
- INB (u_Status); /*
- * Touch the status register
+ INB (u_Status); /*
+ * Touch the status register
*/
- gus_look8 (0x41); /*
- * Clear any pending DMA IRQs
+ gus_look8 (0x41); /*
+ * Clear any pending DMA IRQs
*/
- gus_look8 (0x49); /*
- * Clear any pending sample IRQs
+ gus_look8 (0x49); /*
+ * Clear any pending sample IRQs
*/
- gus_read8 (0x0f); /*
- * Clear pending IRQs
+ gus_read8 (0x0f); /*
+ * Clear pending IRQs
*/
- gus_reset (); /*
- * Resets all voices
+ gus_reset (); /*
+ * Resets all voices
*/
- gus_look8 (0x41); /*
- * Clear any pending DMA IRQs
+ gus_look8 (0x41); /*
+ * Clear any pending DMA IRQs
*/
- gus_look8 (0x49); /*
- * Clear any pending sample IRQs
+ gus_look8 (0x49); /*
+ * Clear any pending sample IRQs
*/
- gus_read8 (0x0f); /*
- * Clear pending IRQs
+ gus_read8 (0x0f); /*
+ * Clear pending IRQs
*/
- gus_write8 (0x4c, 7); /*
- * Master reset | DAC enable | IRQ enable
+ gus_write8 (0x4c, 7); /*
+ * Master reset | DAC enable | IRQ enable
*/
- /*
+ /*
* Set up for Digital ASIC
*/
OUTB (0x05, gus_base + 0x0f);
- mix_image |= 0x02; /*
- * Disable line out
+ mix_image |= 0x02; /*
+ * Disable line out
*/
OUTB (mix_image, u_Mixer);
@@ -859,11 +877,11 @@ gus_initialize (void)
OUTB (0x00, gus_base + 0x0f);
- /*
+ /*
* Now set up the DMA and IRQ interface
- *
+ *
* The GUS supports two IRQs and two DMAs.
- *
+ *
* Just one DMA channel is used. This prevents simultaneous ADC and DAC.
* Adding this support requires significant changes to the dmabuf.c, dsp.c
* and audio.c also.
@@ -874,79 +892,79 @@ gus_initialize (void)
if (!tmp)
printk ("Warning! GUS IRQ not selected\n");
irq_image |= tmp;
- irq_image |= 0x40; /*
- * Combine IRQ1 (GF1) and IRQ2 (Midi)
+ irq_image |= 0x40; /*
+ * Combine IRQ1 (GF1) and IRQ2 (Midi)
*/
- dma_image = 0x40; /*
- * Combine DMA1 (DRAM) and IRQ2 (ADC)
+ dma_image = 0x40; /*
+ * Combine DMA1 (DRAM) and IRQ2 (ADC)
*/
tmp = gus_dma_map[gus_dma];
if (!tmp)
printk ("Warning! GUS DMA not selected\n");
dma_image |= tmp;
- /*
+ /*
* For some reason the IRQ and DMA addresses must be written twice
*/
- /*
- * Doing it first time
+ /*
+ * Doing it first time
*/
- OUTB (mix_image, u_Mixer); /*
- * Select DMA control
+ OUTB (mix_image, u_Mixer); /*
+ * Select DMA control
*/
- OUTB (dma_image | 0x80, u_IRQDMAControl); /*
- * Set DMA address
+ OUTB (dma_image | 0x80, u_IRQDMAControl); /*
+ * Set DMA address
*/
- OUTB (mix_image | 0x40, u_Mixer); /*
- * Select IRQ control
+ OUTB (mix_image | 0x40, u_Mixer); /*
+ * Select IRQ control
*/
- OUTB (irq_image, u_IRQDMAControl); /*
- * Set IRQ address
+ OUTB (irq_image, u_IRQDMAControl); /*
+ * Set IRQ address
*/
- /*
- * Doing it second time
+ /*
+ * Doing it second time
*/
- OUTB (mix_image, u_Mixer); /*
- * Select DMA control
+ OUTB (mix_image, u_Mixer); /*
+ * Select DMA control
*/
- OUTB (dma_image, u_IRQDMAControl); /*
- * Set DMA address
+ OUTB (dma_image, u_IRQDMAControl); /*
+ * Set DMA address
*/
- OUTB (mix_image | 0x40, u_Mixer); /*
- * Select IRQ control
+ OUTB (mix_image | 0x40, u_Mixer); /*
+ * Select IRQ control
*/
- OUTB (irq_image, u_IRQDMAControl); /*
- * Set IRQ address
+ OUTB (irq_image, u_IRQDMAControl); /*
+ * Set IRQ address
*/
- gus_select_voice (0); /*
- * This disables writes to IRQ/DMA reg
+ gus_select_voice (0); /*
+ * This disables writes to IRQ/DMA reg
*/
- mix_image &= ~0x02; /*
- * Enable line out
+ mix_image &= ~0x02; /*
+ * Enable line out
*/
- mix_image |= 0x08; /*
- * Enable IRQ
+ mix_image |= 0x08; /*
+ * Enable IRQ
*/
- OUTB (mix_image, u_Mixer); /*
- * Turn mixer channels on
+ OUTB (mix_image, u_Mixer); /*
+ * Turn mixer channels on
* Note! Mic in is left off.
*/
- gus_select_voice (0); /*
- * This disables writes to IRQ/DMA reg
+ gus_select_voice (0); /*
+ * This disables writes to IRQ/DMA reg
*/
- gusintr (0); /*
- * Serve pending interrupts
+ gusintr (0); /*
+ * Serve pending interrupts
*/
RESTORE_INTR (flags);
}
@@ -1054,13 +1072,13 @@ guswave_set_instr (int dev, int voice, int instr_no)
if (sample_no < 0)
{
printk ("GUS: Undefined patch %d for voice %d\n", instr_no, voice);
- return RET_ERROR (EINVAL); /*
- * Patch not defined
+ return RET_ERROR (EINVAL);/*
+ * Patch not defined
*/
}
- if (sample_ptrs[sample_no] == -1) /*
- * Sample not loaded
+ if (sample_ptrs[sample_no] == -1) /*
+ * Sample not loaded
*/
{
printk ("GUS: Sample #%d not loaded for patch %d (voice %d)\n", sample_no, instr_no, voice);
@@ -1083,13 +1101,15 @@ guswave_kill_note (int dev, int voice, int velocity)
DISABLE_INTR (flags);
if (voices[voice].volume_irq_mode == VMODE_START_NOTE)
- voices[voice].kill_pending = 1;
+ {
+ voices[voice].kill_pending = 1;
+ RESTORE_INTR (flags);
+ }
else
{
- gus_select_voice (voice);
+ RESTORE_INTR (flags);
gus_voice_fade (voice);
}
- RESTORE_INTR (flags);
return 0;
}
@@ -1100,16 +1120,16 @@ guswave_aftertouch (int dev, int voice, int pressure)
short lo_limit, hi_limit;
unsigned long flags;
- return; /*
- * Currently disabled
+ return; /*
+ * Currently disabled
*/
if (voice < 0 || voice > 31)
return;
if (voices[voice].mode & WAVE_ENVELOPES && voices[voice].env_phase != 2)
- return; /*
- * Don't mix with envelopes
+ return; /*
+ * Don't mix with envelopes
*/
if (pressure < 32)
@@ -1117,8 +1137,8 @@ guswave_aftertouch (int dev, int voice, int pressure)
DISABLE_INTR (flags);
gus_select_voice (voice);
gus_rampoff ();
- compute_and_set_volume (voice, 255, 0); /*
- * Back to original volume
+ compute_and_set_volume (voice, 255, 0); /*
+ * Back to original volume
*/
RESTORE_INTR (flags);
return;
@@ -1138,8 +1158,8 @@ guswave_aftertouch (int dev, int voice, int pressure)
}
gus_ramp_range (lo_limit, hi_limit);
gus_ramp_rate (3, 8);
- gus_rampon (0x58); /*
- * Bidirectional, Down, Loop
+ gus_rampon (0x58); /*
+ * Bidirectional, Down, Loop
*/
RESTORE_INTR (flags);
}
@@ -1152,6 +1172,13 @@ guswave_panning (int dev, int voice, int value)
}
static void
+guswave_volume_method (int dev, int mode)
+{
+ if (mode == VOL_METHOD_LINEAR || mode == VOL_METHOD_ADAGIO)
+ volume_method = mode;
+}
+
+static void
compute_volume (int voice, int volume)
{
if (volume < 128)
@@ -1166,6 +1193,11 @@ compute_volume (int voice, int volume)
voices[voice].patch_vol);
break;
+ case VOL_METHOD_LINEAR: /* Totally ignores patch-volume and expression */
+ voices[voice].initial_volume =
+ gus_linear_vol (volume, voices[voice].main_vol);
+ break;
+
default:
voices[voice].initial_volume = volume_base +
(voices[voice].midi_volume * volume_scale);
@@ -1181,16 +1213,16 @@ compute_and_set_volume (int voice, int volume, int ramp_time)
int current, target, rate;
unsigned long flags;
+ compute_volume (voice, volume);
+ voices[voice].current_volume = voices[voice].initial_volume;
+
DISABLE_INTR (flags);
-/*
+ /*
* CAUTION! Interrupts disabled. Enable them before returning
*/
gus_select_voice (voice);
- compute_volume (voice, volume);
- voices[voice].current_volume = voices[voice].initial_volume;
-
current = gus_read16 (0x09) >> 4;
target = voices[voice].initial_volume;
@@ -1208,8 +1240,8 @@ compute_and_set_volume (int voice, int volume, int ramp_time)
rate = 16;
gus_ramp_rate (0, rate);
- if ((target - current) / 64 == 0) /*
- * Too close
+ if ((target - current) / 64 == 0) /*
+ * Too close
*/
{
gus_rampoff ();
@@ -1223,8 +1255,8 @@ compute_and_set_volume (int voice, int volume, int ramp_time)
if (target > (4095 - 65))
target = 4095 - 65;
gus_ramp_range (current, target);
- gus_rampon (0x00); /*
- * Ramp up, once, no irq
+ gus_rampon (0x00); /*
+ * Ramp up, once, no irq
*/
}
else
@@ -1233,8 +1265,8 @@ compute_and_set_volume (int voice, int volume, int ramp_time)
target = 65;
gus_ramp_range (target, current);
- gus_rampon (0x40); /*
- * Ramp down, once, no irq
+ gus_rampon (0x40); /*
+ * Ramp down, once, no irq
*/
}
RESTORE_INTR (flags);
@@ -1248,14 +1280,14 @@ dynamic_volume_change (int voice)
DISABLE_INTR (flags);
gus_select_voice (voice);
- status = gus_read8 (0x00); /*
- * Voice status
+ status = gus_read8 (0x00); /*
+ * Voice status
*/
RESTORE_INTR (flags);
if (status & 0x03)
- return; /*
- * Voice not started
+ return; /*
+ * Voice not started
*/
if (!(voices[voice].mode & WAVE_ENVELOPES))
@@ -1264,19 +1296,19 @@ dynamic_volume_change (int voice)
return;
}
- /*
+ /*
* Voice is running and has envelopes.
*/
DISABLE_INTR (flags);
gus_select_voice (voice);
- status = gus_read8 (0x0d); /*
- * Ramping status
+ status = gus_read8 (0x0d); /*
+ * Ramping status
*/
RESTORE_INTR (flags);
- if (status & 0x03) /*
- * Sustain phase?
+ if (status & 0x03) /*
+ * Sustain phase?
*/
{
compute_and_set_volume (voice, voices[voice].midi_volume, 1);
@@ -1288,11 +1320,11 @@ dynamic_volume_change (int voice)
compute_volume (voice, voices[voice].midi_volume);
-#if 0 /*
+#if 0 /*
* * * Is this really required */
voices[voice].current_volume =
- gus_read16 (0x09) >> 4; /*
- * Get current volume
+ gus_read16 (0x09) >> 4; /*
+ * Get current volume
*/
voices[voice].env_phase--;
@@ -1334,10 +1366,12 @@ guswave_controller (int dev, int voice, int ctrl_num, int value)
value /= 128;
#endif
case CTRL_EXPRESSION:
- volume_method = VOL_METHOD_ADAGIO;
- voices[voice].expression_vol = value;
- if (voices[voice].volume_irq_mode != VMODE_START_NOTE)
- dynamic_volume_change (voice);
+ if (volume_method == VOL_METHOD_ADAGIO)
+ {
+ voices[voice].expression_vol = value;
+ if (voices[voice].volume_irq_mode != VMODE_START_NOTE)
+ dynamic_volume_change (voice);
+ }
break;
#ifdef FUTURE_VERSION
@@ -1350,14 +1384,13 @@ guswave_controller (int dev, int voice, int ctrl_num, int value)
#endif
case CTRL_MAIN_VOLUME:
- volume_method = VOL_METHOD_ADAGIO;
voices[voice].main_vol = value;
if (voices[voice].volume_irq_mode != VMODE_START_NOTE)
dynamic_volume_change (voice);
break;
- default: /*
- * Ignore
+ default: /*
+ * Ignore
*/
break;
}
@@ -1402,7 +1435,7 @@ guswave_start_note2 (int dev, int voice, int note_num, int volume)
note_freq = note_to_freq (note_num);
- /*
+ /*
* Find a sample within a patch so that the note_freq is between low_note
* and high_note.
*/
@@ -1423,8 +1456,8 @@ guswave_start_note2 (int dev, int voice, int note_num, int volume)
if (samples[samplep].low_note <= note_freq && note_freq <= samples[samplep].high_note)
sample = samplep;
else
- samplep = samples[samplep].key; /*
- * Follow link
+ samplep = samples[samplep].key; /*
+ * Follow link
*/
}
if (sample == -1)
@@ -1433,15 +1466,15 @@ guswave_start_note2 (int dev, int voice, int note_num, int volume)
if (sample == -1)
{
printk ("GUS: Patch %d not defined for note %d\n", patch, note_num);
- return 0; /*
- * Should play default patch ???
+ return 0; /*
+ * Should play default patch ???
*/
}
- is16bits = (samples[sample].mode & WAVE_16_BITS) ? 1 : 0; /*
+ is16bits = (samples[sample].mode & WAVE_16_BITS) ? 1 : 0; /*
* 8 or 16
* bit
- * samples
+ * samples
*/
voices[voice].mode = samples[sample].mode;
voices[voice].patch_vol = samples[sample].volume;
@@ -1459,8 +1492,8 @@ guswave_start_note2 (int dev, int voice, int note_num, int volume)
sample_map[voice] = sample;
- base_note = samples[sample].base_note / 100; /*
- * To avoid overflows
+ base_note = samples[sample].base_note / 100; /*
+ * To avoid overflows
*/
note_freq /= 100;
@@ -1468,7 +1501,7 @@ guswave_start_note2 (int dev, int voice, int note_num, int volume)
voices[voice].orig_freq = freq;
- /*
+ /*
* Since the pitch bender may have been set before playing the note, we
* have to calculate the bending now.
*/
@@ -1485,8 +1518,8 @@ guswave_start_note2 (int dev, int voice, int note_num, int volume)
if (samples[sample].mode & WAVE_16_BITS)
{
- mode |= 0x04; /*
- * 16 bits
+ mode |= 0x04; /*
+ * 16 bits
*/
if ((sample_ptrs[sample] >> 18) !=
((sample_ptrs[sample] + samples[sample].len) >> 18))
@@ -1499,10 +1532,13 @@ guswave_start_note2 (int dev, int voice, int note_num, int volume)
DISABLE_INTR (flags);
gus_select_voice (voice);
- gus_voice_off (); /*
- * It may still be running
+ gus_voice_off (); /*
+ * It may still be running
*/
gus_rampoff ();
+
+ RESTORE_INTR (flags);
+
if (voices[voice].mode & WAVE_ENVELOPES)
{
compute_volume (voice, volume);
@@ -1511,6 +1547,9 @@ guswave_start_note2 (int dev, int voice, int note_num, int volume)
else
compute_and_set_volume (voice, volume, 0);
+ DISABLE_INTR (flags);
+ gus_select_voice (voice);
+
if (samples[sample].mode & WAVE_LOOP_BACK)
gus_write_addr (0x0a, sample_ptrs[sample] + samples[sample].len -
voices[voice].offset_pending, is16bits); /* Sample
@@ -1521,13 +1560,13 @@ guswave_start_note2 (int dev, int voice, int note_num, int volume)
if (samples[sample].mode & WAVE_LOOPING)
{
- mode |= 0x08; /*
- * Looping on
+ mode |= 0x08; /*
+ * Looping on
*/
if (samples[sample].mode & WAVE_BIDIR_LOOP)
- mode |= 0x10; /*
- * Bidirectional looping on
+ mode |= 0x10; /*
+ * Bidirectional looping on
*/
if (samples[sample].mode & WAVE_LOOP_BACK)
@@ -1538,35 +1577,35 @@ guswave_start_note2 (int dev, int voice, int note_num, int volume)
mode |= 0x40;
}
- gus_write_addr (0x02, sample_ptrs[sample] + samples[sample].loop_start, is16bits); /*
- * Loop
- * start
- * location
+ gus_write_addr (0x02, sample_ptrs[sample] + samples[sample].loop_start, is16bits); /*
+ * Loop
+ * start
+ * location
*/
- gus_write_addr (0x04, sample_ptrs[sample] + samples[sample].loop_end, is16bits); /*
- * Loop
- * end
- * location
+ gus_write_addr (0x04, sample_ptrs[sample] + samples[sample].loop_end, is16bits); /*
+ * Loop
+ * end
+ * location
*/
}
else
{
- mode |= 0x20; /*
- * Loop irq at the end
+ mode |= 0x20; /*
+ * Loop irq at the end
*/
- voices[voice].loop_irq_mode = LMODE_FINISH; /*
+ voices[voice].loop_irq_mode = LMODE_FINISH; /*
* Ramp it down at
- * the * end
+ * the * end
*/
voices[voice].loop_irq_parm = 1;
- gus_write_addr (0x02, sample_ptrs[sample], is16bits); /*
- * Loop start
- * location
+ gus_write_addr (0x02, sample_ptrs[sample], is16bits); /*
+ * Loop start
+ * location
*/
- gus_write_addr (0x04, sample_ptrs[sample] + samples[sample].len - 1, is16bits); /*
- * Loop
- * end
- * location
+ gus_write_addr (0x04, sample_ptrs[sample] + samples[sample].len - 1, is16bits); /*
+ * Loop
+ * end
+ * location
*/
}
gus_voice_freq (freq);
@@ -1577,9 +1616,9 @@ guswave_start_note2 (int dev, int voice, int note_num, int volume)
return 0;
}
-/*
+/*
* * New guswave_start_note by Andrew J. Robinson attempts to minimize
- * clicking * when the note playing on the voice is changed. It uses volume
+ * clicking * when the note playing on the voice is changed. It uses volume
* ramping. */
static int
@@ -1595,14 +1634,17 @@ guswave_start_note (int dev, int voice, int note_num, int volume)
if (voices[voice].volume_irq_mode == VMODE_START_NOTE)
voices[voice].volume_pending = volume;
else
- ret_val = guswave_start_note2 (dev, voice, note_num, volume);
+ {
+ RESTORE_INTR (flags);
+ ret_val = guswave_start_note2 (dev, voice, note_num, volume);
+ }
}
else
{
gus_select_voice (voice);
mode = gus_read8 (0x00);
if (mode & 0x20)
- gus_write8 (0x00, mode & 0xdf); /* No interrupt! */
+ gus_write8 (0x00, mode & 0xdf); /* No interrupt! */
voices[voice].offset_pending = 0;
voices[voice].kill_pending = 0;
@@ -1611,9 +1653,11 @@ guswave_start_note (int dev, int voice, int note_num, int volume)
if (voices[voice].sample_pending >= 0)
{
+ RESTORE_INTR (flags);
guswave_set_instr (voices[voice].dev_pending, voice,
voices[voice].sample_pending);
voices[voice].sample_pending = -1;
+ DISABLE_INTR (flags);
}
if ((mode & 0x01) || ((gus_read16 (0x09) >> 4) < 2065))
@@ -1629,11 +1673,11 @@ guswave_start_note (int dev, int voice, int note_num, int volume)
gus_rampoff ();
gus_ramp_range (2000, 4065);
- gus_ramp_rate (0, 63); /* Fastest possible rate */
+ gus_ramp_rate (0, 63);/* Fastest possible rate */
gus_rampon (0x20 | 0x40); /* Ramp down, once, irq */
+ RESTORE_INTR (flags);
}
}
- RESTORE_INTR (flags);
return ret_val;
}
@@ -1691,10 +1735,10 @@ guswave_load_patch (int dev, int format, snd_rw_buf * addr,
unsigned long blk_size, blk_end, left, src_offs, target;
- sizeof_patch = (long) &patch.data[0] - (long) &patch; /*
+ sizeof_patch = (long) &patch.data[0] - (long) &patch; /*
* Size of
* the header
- * * info
+ * * info
*/
if (format != GUS_PATCH)
@@ -1717,7 +1761,7 @@ guswave_load_patch (int dev, int format, snd_rw_buf * addr,
return RET_ERROR (ENOSPC);
}
- /*
+ /*
* Copy the header from user space but ignore the first bytes which have
* been transferred already.
*/
@@ -1760,15 +1804,15 @@ guswave_load_patch (int dev, int format, snd_rw_buf * addr,
}
}
- free_mem_ptr = (free_mem_ptr + 31) & ~31; /*
- * Alignment 32 bytes
+ free_mem_ptr = (free_mem_ptr + 31) & ~31; /*
+ * Alignment 32 bytes
*/
#define GUS_BANK_SIZE (256*1024)
if (patch.mode & WAVE_16_BITS)
{
- /*
+ /*
* 16 bit samples must fit one 256k bank.
*/
if (patch.len >= GUS_BANK_SIZE)
@@ -1780,16 +1824,16 @@ guswave_load_patch (int dev, int format, snd_rw_buf * addr,
if ((free_mem_ptr / GUS_BANK_SIZE) !=
((free_mem_ptr + patch.len) / GUS_BANK_SIZE))
{
- unsigned long tmp_mem = /*
- * Align to 256K*N
+ unsigned long tmp_mem = /*
+ * Align to 256K*N
*/
((free_mem_ptr / GUS_BANK_SIZE) + 1) * GUS_BANK_SIZE;
if ((tmp_mem + patch.len) > gus_mem_size)
return RET_ERROR (ENOSPC);
- free_mem_ptr = tmp_mem; /*
- * This leaves unusable memory
+ free_mem_ptr = tmp_mem; /*
+ * This leaves unusable memory
*/
}
}
@@ -1799,8 +1843,8 @@ guswave_load_patch (int dev, int format, snd_rw_buf * addr,
sample_ptrs[free_sample] = free_mem_ptr;
- /*
- * Tremolo is not possible with envelopes
+ /*
+ * Tremolo is not possible with envelopes
*/
if (patch.mode & WAVE_ENVELOPES)
@@ -1808,14 +1852,14 @@ guswave_load_patch (int dev, int format, snd_rw_buf * addr,
memcpy ((char *) &samples[free_sample], &patch, sizeof_patch);
- /*
+ /*
* Link this_one sample to the list of samples for patch 'instr'.
*/
samples[free_sample].key = patch_table[instr];
patch_table[instr] = free_sample;
- /*
+ /*
* Use DMA to transfer the wave data to the DRAM
*/
@@ -1823,22 +1867,22 @@ guswave_load_patch (int dev, int format, snd_rw_buf * addr,
src_offs = 0;
target = free_mem_ptr;
- while (left) /*
- * Not all moved
+ while (left) /*
+ * Not all moved
*/
{
blk_size = sound_buffsizes[gus_devnum];
if (blk_size > left)
blk_size = left;
- /*
+ /*
* DMA cannot cross 256k bank boundaries. Check for that.
*/
blk_end = target + blk_size;
if ((target >> 18) != (blk_end >> 18))
- { /*
- * Have to split the block
+ { /*
+ * Have to split the block
*/
blk_end &= ~(256 * 1024 - 1);
@@ -1846,7 +1890,7 @@ guswave_load_patch (int dev, int format, snd_rw_buf * addr,
}
#if defined(GUS_NO_DMA) || defined(GUS_PATCH_NO_DMA)
- /*
+ /*
* For some reason the DMA is not possible. We have to use PIO.
*/
{
@@ -1859,20 +1903,20 @@ guswave_load_patch (int dev, int format, snd_rw_buf * addr,
if (patch.mode & WAVE_UNSIGNED)
if (!(patch.mode & WAVE_16_BITS) || (i & 0x01))
- data ^= 0x80; /*
- * Convert to signed
+ data ^= 0x80; /*
+ * Convert to signed
*/
gus_poke (target + i, data);
}
}
-#else /*
+#else /*
* * * GUS_NO_DMA */
{
unsigned long address, hold_address;
unsigned char dma_command;
unsigned long flags;
- /*
+ /*
* OK, move now. First in and then out.
*/
@@ -1881,13 +1925,13 @@ guswave_load_patch (int dev, int format, snd_rw_buf * addr,
blk_size);
DISABLE_INTR (flags); /******** INTERRUPTS DISABLED NOW ********/
- gus_write8 (0x41, 0); /*
- * Disable GF1 DMA
+ gus_write8 (0x41, 0); /*
+ * Disable GF1 DMA
*/
DMAbuf_start_dma (gus_devnum, snd_raw_buf_phys[gus_devnum][0],
blk_size, DMA_MODE_WRITE);
- /*
+ /*
* Set the DRAM address for the wave data
*/
@@ -1901,35 +1945,35 @@ guswave_load_patch (int dev, int format, snd_rw_buf * addr,
address |= (hold_address & 0x000c0000L);
}
- gus_write16 (0x42, (address >> 4) & 0xffff); /*
- * DRAM DMA address
+ gus_write16 (0x42, (address >> 4) & 0xffff); /*
+ * DRAM DMA address
*/
- /*
+ /*
* Start the DMA transfer
*/
- dma_command = 0x21; /*
- * IRQ enable, DMA start
+ dma_command = 0x21; /*
+ * IRQ enable, DMA start
*/
if (patch.mode & WAVE_UNSIGNED)
- dma_command |= 0x80; /*
- * Invert MSB
+ dma_command |= 0x80; /*
+ * Invert MSB
*/
if (patch.mode & WAVE_16_BITS)
- dma_command |= 0x40; /*
- * 16 bit _DATA_
+ dma_command |= 0x40; /*
+ * 16 bit _DATA_
*/
if (sound_dsp_dmachan[gus_devnum] > 3)
- dma_command |= 0x04; /*
- * 16 bit DMA channel
+ dma_command |= 0x04; /*
+ * 16 bit DMA channel
*/
- gus_write8 (0x41, dma_command); /*
- * Let's go luteet (=bugs)
+ gus_write8 (0x41, dma_command); /*
+ * Let's go luteet (=bugs)
*/
- /*
+ /*
* Sleep here until the DRAM DMA done interrupt is served
*/
active_device = GUS_DEV_WAVE;
@@ -1939,10 +1983,10 @@ guswave_load_patch (int dev, int format, snd_rw_buf * addr,
printk ("GUS: DMA Transfer timed out\n");
RESTORE_INTR (flags);
}
-#endif /*
+#endif /*
* * * GUS_NO_DMA */
- /*
+ /*
* Now the next part
*/
@@ -1950,8 +1994,8 @@ guswave_load_patch (int dev, int format, snd_rw_buf * addr,
src_offs += blk_size;
target += blk_size;
- gus_write8 (0x41, 0); /*
- * Stop DMA
+ gus_write8 (0x41, 0); /*
+ * Stop DMA
*/
}
@@ -1997,8 +2041,8 @@ guswave_hw_control (int dev, unsigned char *event)
case _GUS_VOICEON:
DISABLE_INTR (flags);
gus_select_voice (voice);
- p1 &= ~0x20; /*
- * Disable intr
+ p1 &= ~0x20; /*
+ * Disable intr
*/
gus_voice_on (p1);
RESTORE_INTR (flags);
@@ -2012,17 +2056,14 @@ guswave_hw_control (int dev, unsigned char *event)
break;
case _GUS_VOICEFADE:
- DISABLE_INTR (flags);
- gus_select_voice (voice);
gus_voice_fade (voice);
- RESTORE_INTR (flags);
break;
case _GUS_VOICEMODE:
DISABLE_INTR (flags);
gus_select_voice (voice);
- p1 &= ~0x20; /*
- * Disable intr
+ p1 &= ~0x20; /*
+ * Disable intr
*/
gus_voice_mode (p1);
RESTORE_INTR (flags);
@@ -2049,8 +2090,8 @@ guswave_hw_control (int dev, unsigned char *event)
RESTORE_INTR (flags);
break;
- case _GUS_VOICEVOL2: /*
- * Just update the voice value
+ case _GUS_VOICEVOL2: /*
+ * Just update the voice value
*/
voices[voice].initial_volume =
voices[voice].current_volume = p1;
@@ -2058,8 +2099,8 @@ guswave_hw_control (int dev, unsigned char *event)
case _GUS_RAMPRANGE:
if (voices[voice].mode & WAVE_ENVELOPES)
- break; /*
- * NO-NO
+ break; /*
+ * NO-NO
*/
DISABLE_INTR (flags);
gus_select_voice (voice);
@@ -2069,8 +2110,8 @@ guswave_hw_control (int dev, unsigned char *event)
case _GUS_RAMPRATE:
if (voices[voice].mode & WAVE_ENVELOPES)
- break; /*
- * NO-NO
+ break; /*
+ * NO-NO
*/
DISABLE_INTR (flags);
gus_select_voice (voice);
@@ -2080,13 +2121,13 @@ guswave_hw_control (int dev, unsigned char *event)
case _GUS_RAMPMODE:
if (voices[voice].mode & WAVE_ENVELOPES)
- break; /*
- * NO-NO
+ break; /*
+ * NO-NO
*/
DISABLE_INTR (flags);
gus_select_voice (voice);
- p1 &= ~0x20; /*
- * Disable intr
+ p1 &= ~0x20; /*
+ * Disable intr
*/
gus_ramp_mode (p1);
RESTORE_INTR (flags);
@@ -2094,13 +2135,13 @@ guswave_hw_control (int dev, unsigned char *event)
case _GUS_RAMPON:
if (voices[voice].mode & WAVE_ENVELOPES)
- break; /*
- * NO-NO
+ break; /*
+ * NO-NO
*/
DISABLE_INTR (flags);
gus_select_voice (voice);
- p1 &= ~0x20; /*
- * Disable intr
+ p1 &= ~0x20; /*
+ * Disable intr
*/
gus_rampon (p1);
RESTORE_INTR (flags);
@@ -2108,8 +2149,8 @@ guswave_hw_control (int dev, unsigned char *event)
case _GUS_RAMPOFF:
if (voices[voice].mode & WAVE_ENVELOPES)
- break; /*
- * NO-NO
+ break; /*
+ * NO-NO
*/
DISABLE_INTR (flags);
gus_select_voice (voice);
@@ -2218,8 +2259,8 @@ gus_sampling_ioctl (int dev, unsigned int cmd, unsigned int arg, int local)
return gus_sampling_bits;
return IOCTL_OUT (arg, gus_sampling_bits);
- case SOUND_PCM_WRITE_FILTER: /*
- * NOT YET IMPLEMENTED
+ case SOUND_PCM_WRITE_FILTER: /*
+ * NOT YET IMPLEMENTED
*/
return IOCTL_OUT (arg, RET_ERROR (EINVAL));
break;
@@ -2319,14 +2360,14 @@ play_next_pcm_block (void)
for (chn = 0; chn < gus_sampling_channels; chn++)
{
mode[chn] = 0x00;
- ramp_mode[chn] = 0x03; /*
- * Ramping and rollover off
+ ramp_mode[chn] = 0x03; /*
+ * Ramping and rollover off
*/
if (chn == 0)
{
- mode[chn] |= 0x20; /*
- * Loop irq
+ mode[chn] |= 0x20; /*
+ * Loop irq
*/
voices[chn].loop_irq_mode = LMODE_PCM;
}
@@ -2334,8 +2375,8 @@ play_next_pcm_block (void)
if (gus_sampling_bits != 8)
{
is16bits = 1;
- mode[chn] |= 0x04; /*
- * 16 bit data
+ mode[chn] |= 0x04; /*
+ * 16 bit data
*/
}
else
@@ -2344,22 +2385,22 @@ play_next_pcm_block (void)
dram_loc = this_one * pcm_bsize;
dram_loc += chn * pcm_banksize;
- if (this_one == (pcm_nblk - 1)) /*
- * Last of the DRAM buffers
+ if (this_one == (pcm_nblk - 1)) /*
+ * Last of the DRAM buffers
*/
{
- mode[chn] |= 0x08; /*
- * Enable loop
+ mode[chn] |= 0x08; /*
+ * Enable loop
*/
- ramp_mode[chn] = 0x03; /*
- * Disable rollover
+ ramp_mode[chn] = 0x03;/*
+ * Disable rollover
*/
}
else
{
if (chn == 0)
- ramp_mode[chn] = 0x04; /*
- * Enable rollover bit
+ ramp_mode[chn] = 0x04; /*
+ * Enable rollover bit
*/
}
@@ -2368,89 +2409,89 @@ play_next_pcm_block (void)
gus_voice_freq (speed);
if (gus_sampling_channels == 1)
- gus_voice_balance (7); /*
- * mono
+ gus_voice_balance (7); /*
+ * mono
*/
else if (chn == 0)
- gus_voice_balance (0); /*
- * left
+ gus_voice_balance (0); /*
+ * left
*/
else
- gus_voice_balance (15); /*
- * right
+ gus_voice_balance (15); /*
+ * right
*/
- if (!pcm_active) /*
- * Voice not started yet
+ if (!pcm_active) /*
+ * Voice not started yet
*/
{
- /*
+ /*
* The playback was not started yet (or there has been a pause).
* Start the voice (again) and ask for a rollover irq at the end of
* this_one block. If this_one one is last of the buffers, use just
* the normal loop with irq.
*/
- gus_voice_off (); /*
- * It could already be running
+ gus_voice_off (); /*
+ * It could already be running
*/
gus_rampoff ();
gus_voice_volume (1530 + (25 * gus_pcm_volume));
gus_ramp_range (65, 1530 + (25 * gus_pcm_volume));
- gus_write_addr (0x0a, dram_loc, is16bits); /*
- * Starting position
+ gus_write_addr (0x0a, dram_loc, is16bits); /*
+ * Starting position
*/
- gus_write_addr (0x02, chn * pcm_banksize, is16bits); /*
- * Loop start
- * location
+ gus_write_addr (0x02, chn * pcm_banksize, is16bits); /*
+ * Loop start
+ * location
*/
if (chn != 0)
gus_write_addr (0x04, pcm_banksize + (pcm_bsize * pcm_nblk),
- is16bits); /*
- * Loop end location
+ is16bits); /*
+ * Loop end location
*/
}
if (chn == 0)
- gus_write_addr (0x04, dram_loc + pcm_datasize[this_one], is16bits); /*
- * Loop
- * end
- * location
+ gus_write_addr (0x04, dram_loc + pcm_datasize[this_one], is16bits); /*
+ * Loop
+ * end
+ * location
*/
else
- mode[chn] |= 0x08; /*
- * Enable loop
+ mode[chn] |= 0x08; /*
+ * Enable loop
*/
if (pcm_datasize[this_one] != pcm_bsize)
{
- /*
- * Incomplete block. Possibly the last one.
+ /*
+ * Incomplete block. Possibly the last one.
*/
if (chn == 0)
{
- mode[chn] &= ~0x08; /*
- * Disable loop
+ mode[chn] &= ~0x08; /*
+ * Disable loop
*/
- mode[chn] |= 0x20; /*
- * Enable loop IRQ
+ mode[chn] |= 0x20;/*
+ * Enable loop IRQ
*/
voices[0].loop_irq_mode = LMODE_PCM_STOP;
- ramp_mode[chn] = 0x03; /*
- * No rollover bit
+ ramp_mode[chn] = 0x03; /*
+ * No rollover bit
*/
}
else
{
- gus_write_addr (0x04, dram_loc + pcm_datasize[this_one], is16bits); /*
- * Loop
- * end
- * location
+ gus_write_addr (0x04, dram_loc + pcm_datasize[this_one], is16bits); /*
+ * Loop
+ * end
+ * location
*/
- mode[chn] &= ~0x08; /*
- * Disable loop
+ mode[chn] &= ~0x08; /*
+ * Disable loop
*/
}
}
@@ -2474,11 +2515,11 @@ static void
gus_transfer_output_block (int dev, unsigned long buf,
int total_count, int intrflag, int chn)
{
- /*
+ /*
* This routine transfers one block of audio data to the DRAM. In mono mode
* it's called just once. When in stereo mode, this_one routine is called
* once for both channels.
- *
+ *
* The left/mono channel data is transferred to the beginning of dram and the
* right data to the area pointed by gus_page_size.
*/
@@ -2505,8 +2546,8 @@ gus_transfer_output_block (int dev, unsigned long buf,
else
this_one = pcm_current_block;
- gus_write8 (0x41, 0); /*
- * Disable GF1 DMA
+ gus_write8 (0x41, 0); /*
+ * Disable GF1 DMA
*/
DMAbuf_start_dma (dev, buf + (chn * count), count, DMA_MODE_WRITE);
@@ -2521,38 +2562,38 @@ gus_transfer_output_block (int dev, unsigned long buf,
address |= (hold_address & 0x000c0000L);
}
- gus_write16 (0x42, (address >> 4) & 0xffff); /*
- * DRAM DMA address
+ gus_write16 (0x42, (address >> 4) & 0xffff); /*
+ * DRAM DMA address
*/
- dma_command = 0x21; /*
- * IRQ enable, DMA start
+ dma_command = 0x21; /*
+ * IRQ enable, DMA start
*/
if (gus_sampling_bits != 8)
- dma_command |= 0x40; /*
- * 16 bit _DATA_
+ dma_command |= 0x40; /*
+ * 16 bit _DATA_
*/
else
- dma_command |= 0x80; /*
- * Invert MSB
+ dma_command |= 0x80; /*
+ * Invert MSB
*/
if (sound_dsp_dmachan[dev] > 3)
- dma_command |= 0x04; /*
- * 16 bit DMA channel
+ dma_command |= 0x04; /*
+ * 16 bit DMA channel
*/
- gus_write8 (0x41, dma_command); /*
- * Kick on
+ gus_write8 (0x41, dma_command); /*
+ * Kick on
*/
- if (chn == (gus_sampling_channels - 1)) /*
- * Last channel
+ if (chn == (gus_sampling_channels - 1)) /*
+ * Last channel
*/
{
- /*
- * Last (right or mono) channel data
+ /*
+ * Last (right or mono) channel data
*/
active_device = GUS_DEV_PCM_DONE;
if (!pcm_active && (pcm_qlen > 2 || count < pcm_bsize))
@@ -2560,7 +2601,7 @@ gus_transfer_output_block (int dev, unsigned long buf,
play_next_pcm_block ();
}
}
- else /*
+ else /*
* * * Left channel data. The right channel
* is * * * transferred after DMA interrupt */
active_device = GUS_DEV_PCM_CONTINUE;
@@ -2590,20 +2631,20 @@ gus_sampling_start_input (int dev, unsigned long buf, int count,
DMAbuf_start_dma (dev, buf, count, DMA_MODE_READ);
- mode = 0xa0; /*
- * DMA IRQ enable, invert MSB
+ mode = 0xa0; /*
+ * DMA IRQ enable, invert MSB
*/
if (sound_dsp_dmachan[dev] > 3)
- mode |= 0x04; /*
- * 16 bit DMA channel
+ mode |= 0x04; /*
+ * 16 bit DMA channel
*/
if (gus_sampling_channels > 1)
- mode |= 0x02; /*
- * Stereo
+ mode |= 0x02; /*
+ * Stereo
*/
- mode |= 0x01; /*
- * DMA enable
+ mode |= 0x01; /*
+ * DMA enable
*/
gus_write8 (0x49, mode);
@@ -2618,8 +2659,8 @@ gus_sampling_prepare_for_input (int dev, int bsize, int bcount)
rate = (9878400 / (gus_sampling_speed + 2)) / 16;
- gus_write8 (0x48, rate & 0xff); /*
- * Set sampling frequency
+ gus_write8 (0x48, rate & 0xff); /*
+ * Set sampling frequency
*/
if (gus_sampling_bits != 8)
@@ -2722,6 +2763,7 @@ gus_copy_from_user (int dev, char *localbuf, int localoffs,
static struct audio_operations gus_sampling_operations =
{
"Gravis UltraSound",
+ NEEDS_RESTART,
gus_sampling_open,
gus_sampling_close,
gus_sampling_output_block,
@@ -2751,6 +2793,7 @@ guswave_bender (int dev, int voice, int value)
gus_voice_freq (freq);
RESTORE_INTR (flags);
}
+
#endif
static int
@@ -2782,8 +2825,8 @@ guswave_patchmgr (int dev, struct patmgr_info *rec)
while (ptr >= 0 && ptr < free_sample)
{
rec->data.data8[i]++;
- ptr = samples[ptr].key; /*
- * Follow link
+ ptr = samples[ptr].key; /*
+ * Follow link
*/
}
}
@@ -2799,8 +2842,8 @@ guswave_patchmgr (int dev, struct patmgr_info *rec)
while (ptr >= 0 && ptr < free_sample)
{
rec->data.data32[n++] = ptr;
- ptr = samples[ptr].key; /*
- * Follow link
+ ptr = samples[ptr].key; /*
+ * Follow link
*/
}
}
@@ -2821,11 +2864,11 @@ guswave_patchmgr (int dev, struct patmgr_info *rec)
pat = (struct patch_info *) rec->data.data8;
- pat->key = GUS_PATCH; /*
- * Restore patch type
+ pat->key = GUS_PATCH; /*
+ * Restore patch type
*/
- rec->parm1 = sample_ptrs[ptr]; /*
- * DRAM address
+ rec->parm1 = sample_ptrs[ptr]; /*
+ * DRAM address
*/
rec->parm2 = sizeof (struct patch_info);
}
@@ -2842,13 +2885,13 @@ guswave_patchmgr (int dev, struct patmgr_info *rec)
pat = (struct patch_info *) rec->data.data8;
- if (pat->len > samples[ptr].len) /*
- * Cannot expand sample
+ if (pat->len > samples[ptr].len) /*
+ * Cannot expand sample
*/
return RET_ERROR (EINVAL);
- pat->key = samples[ptr].key; /*
- * Ensure the link is correct
+ pat->key = samples[ptr].key; /*
+ * Ensure the link is correct
*/
memcpy ((char *) &samples[ptr], rec->data.data8,
@@ -2859,8 +2902,8 @@ guswave_patchmgr (int dev, struct patmgr_info *rec)
return 0;
break;
- case PM_READ_PATCH: /*
- * Returns a block of wave data from the DRAM
+ case PM_READ_PATCH: /*
+ * Returns a block of wave data from the DRAM
*/
{
int sample = rec->parm1;
@@ -2872,12 +2915,12 @@ guswave_patchmgr (int dev, struct patmgr_info *rec)
return RET_ERROR (EINVAL);
if (offs < 0 || offs >= samples[sample].len)
- return RET_ERROR (EINVAL); /*
- * Invalid offset
+ return RET_ERROR (EINVAL); /*
+ * Invalid offset
*/
- n = samples[sample].len - offs; /*
- * Nr of bytes left
+ n = samples[sample].len - offs; /*
+ * Nr of bytes left
*/
if (l > n)
@@ -2887,25 +2930,25 @@ guswave_patchmgr (int dev, struct patmgr_info *rec)
l = sizeof (rec->data.data8);
if (l <= 0)
- return RET_ERROR (EINVAL); /*
- * Was there a bug?
+ return RET_ERROR (EINVAL); /*
+ * Was there a bug?
*/
- offs += sample_ptrs[sample]; /*
- * Begin offsess + offset to DRAM
+ offs += sample_ptrs[sample]; /*
+ * Begin offsess + offset to DRAM
*/
for (n = 0; n < l; n++)
rec->data.data8[n] = gus_peek (offs++);
- rec->parm1 = n; /*
- * Nr of bytes copied
+ rec->parm1 = n; /*
+ * Nr of bytes copied
*/
}
return 0;
break;
- case PM_WRITE_PATCH: /*
- * Writes a block of wave data to the DRAM
+ case PM_WRITE_PATCH: /*
+ * Writes a block of wave data to the DRAM
*/
{
int sample = rec->parm1;
@@ -2917,12 +2960,12 @@ guswave_patchmgr (int dev, struct patmgr_info *rec)
return RET_ERROR (EINVAL);
if (offs < 0 || offs >= samples[sample].len)
- return RET_ERROR (EINVAL); /*
- * Invalid offset
+ return RET_ERROR (EINVAL); /*
+ * Invalid offset
*/
- n = samples[sample].len - offs; /*
- * Nr of bytes left
+ n = samples[sample].len - offs; /*
+ * Nr of bytes left
*/
if (l > n)
@@ -2932,18 +2975,18 @@ guswave_patchmgr (int dev, struct patmgr_info *rec)
l = sizeof (rec->data.data8);
if (l <= 0)
- return RET_ERROR (EINVAL); /*
- * Was there a bug?
+ return RET_ERROR (EINVAL); /*
+ * Was there a bug?
*/
- offs += sample_ptrs[sample]; /*
- * Begin offsess + offset to DRAM
+ offs += sample_ptrs[sample]; /*
+ * Begin offsess + offset to DRAM
*/
for (n = 0; n < l; n++)
gus_poke (offs++, rec->data.data8[n]);
- rec->parm1 = n; /*
- * Nr of bytes copied
+ rec->parm1 = n; /*
+ * Nr of bytes copied
*/
}
return 0;
@@ -2974,6 +3017,7 @@ static struct synth_operations guswave_operations =
guswave_aftertouch,
guswave_controller,
guswave_panning,
+ guswave_volume_method,
guswave_patchmgr,
#ifdef FUTURE_VERSION
guswave_bender
@@ -2988,7 +3032,7 @@ set_input_volumes (void)
DISABLE_INTR (flags);
-/*
+ /*
* Enable channels having vol > 10%
* Note! bit 0x01 means line in DISABLED while 0x04 means
* mic in ENABLED.
@@ -3000,7 +3044,7 @@ set_input_volumes (void)
if (recording_active)
{
-/*
+ /*
* Disable channel, if not selected for recording
*/
if (!(gus_recmask & SOUND_MASK_LINE))
@@ -3016,8 +3060,8 @@ set_input_volumes (void)
RESTORE_INTR (flags);
}
-static int
-gus_mixer_ioctl (int dev, unsigned int cmd, unsigned int arg)
+int
+gus_default_mixer_ioctl (int dev, unsigned int cmd, unsigned int arg)
{
#define MIX_DEVS (SOUND_MASK_MIC|SOUND_MASK_LINE| \
SOUND_MASK_SYNTH|SOUND_MASK_PCM)
@@ -3085,9 +3129,9 @@ gus_mixer_ioctl (int dev, unsigned int cmd, unsigned int arg)
if (active_device == GUS_DEV_WAVE)
for (voice = 0; voice < nr_voices; voice++)
- dynamic_volume_change (voice); /*
+ dynamic_volume_change (voice); /*
* Apply the new
- * volume
+ * volume
*/
return IOCTL_OUT (arg, gus_wave_volume | (gus_wave_volume << 8));
@@ -3098,8 +3142,8 @@ gus_mixer_ioctl (int dev, unsigned int cmd, unsigned int arg)
return RET_ERROR (EINVAL);
}
else
- switch (cmd & 0xff) /*
- * Return parameters
+ switch (cmd & 0xff) /*
+ * Return parameters
*/
{
@@ -3149,13 +3193,97 @@ gus_mixer_ioctl (int dev, unsigned int cmd, unsigned int arg)
static struct mixer_operations gus_mixer_operations =
{
- gus_mixer_ioctl
+ gus_default_mixer_ioctl
};
+static long
+gus_default_mixer_init (long mem_start)
+{
+ if (num_mixers < MAX_MIXER_DEV) /*
+ * Don't install if there is another
+ * mixer
+ */
+ mixer_devs[num_mixers++] = &gus_mixer_operations;
+
+ return mem_start;
+}
+
long
gus_wave_init (long mem_start, int irq, int dma)
{
- printk ("snd4: <Gravis UltraSound %dk>", (int) gus_mem_size / 1024);
+ unsigned long flags;
+ unsigned char val;
+ char *model_num = "2.4";
+ int gus_type = 0x24; /* 2.4 */
+ int mixer_type = 0;
+
+ /*
+ * Try to identify the GUS model.
+ *
+ * Versions < 3.6 don't have the digital ASIC. Try to probe it first.
+ */
+
+ DISABLE_INTR (flags);
+ OUTB (0x20, gus_base + 0x0f);
+ val = INB (gus_base + 0x0f);
+ RESTORE_INTR (flags);
+
+ if (val != 0xff && (val & 0x06)) /* Should be 0x02? */
+ {
+ /*
+ * It has the digital ASIC so the card is at least v3.4.
+ * Next try to detect the true model.
+ */
+
+ val = INB (u_MixSelect);
+
+ /*
+ * Value 255 means pre-3.7 which don't have mixer.
+ * Values 5 thru 9 mean v3.7 which has a ICS2101 mixer.
+ * 10 and above is GUS MAX which has the CS4231 codec/mixer.
+ *
+ * Sorry. No GUS max support yet but it should be available
+ * soon after the SDK for GUS MAX is available.
+ */
+
+ if (val == 255 || val < 5)
+ {
+ model_num = "3.4";
+ gus_type = 0x34;
+ }
+ else if (val < 10)
+ {
+ model_num = "3.7";
+ gus_type = 0x37;
+ mixer_type = ICS2101;
+ }
+ else
+ {
+ model_num = "MAX";
+ gus_type = 0x40;
+ mixer_type = CS4231;
+ }
+ }
+ else
+ {
+ /*
+ * ASIC not detected so the card must be 2.2 or 2.4.
+ * There could still be the 16-bit/mixer daughter card.
+ * It has the same codec/mixer than MAX.
+ * At this time there is no support for it but it will appear soon.
+ */
+ }
+
+
+#ifdef __FreeBSD__
+ printk ("snd4: <Gravis UltraSound %s (%dk)>", model_num, (int) gus_mem_size / 1024);
+#else
+ printk (" <Gravis UltraSound %s (%dk)>", model_num, (int) gus_mem_size / 1024);
+#endif
+
+#ifndef SCO
+ sprintf (gus_info.name, "Gravis UltraSound %s (%dk)", model_num, (int) gus_mem_size / 1024);
+#endif
if (irq < 0 || irq > 15)
{
@@ -3188,18 +3316,28 @@ gus_wave_init (long mem_start, int irq, int dma)
{
dsp_devs[gus_devnum = num_dspdevs++] = &gus_sampling_operations;
sound_dsp_dmachan[gus_devnum] = dma;
- sound_buffcounts[gus_devnum] = DSP_BUFFCOUNT;
+ sound_buffcounts[gus_devnum] = 1;
sound_buffsizes[gus_devnum] = DSP_BUFFSIZE;
sound_dma_automode[gus_devnum] = 0;
}
else
printk ("GUS: Too many PCM devices available\n");
- if (num_mixers < MAX_MIXER_DEV) /*
- * Don't install if there is another
- * mixer
- */
- mixer_devs[num_mixers++] = &gus_mixer_operations;
+ /*
+ * Mixer dependent initialization.
+ */
+
+ switch (mixer_type)
+ {
+ case ICS2101:
+ gus_line_vol=gus_mic_vol=gus_wave_volume = gus_pcm_volume = 100;
+ return ics2101_mixer_init (mem_start);
+
+ case CS4231:
+ /* Available soon */
+ default:
+ return gus_default_mixer_init (mem_start);
+ }
return mem_start;
}
@@ -3215,8 +3353,8 @@ do_loop_irq (int voice)
gus_select_voice (voice);
tmp = gus_read8 (0x00);
- tmp &= ~0x20; /*
- * Disable wave IRQ for this_one voice
+ tmp &= ~0x20; /*
+ * Disable wave IRQ for this_one voice
*/
gus_write8 (0x00, tmp);
@@ -3227,12 +3365,12 @@ do_loop_irq (int voice)
switch (mode)
{
- case LMODE_FINISH: /*
- * Final loop finished, shoot volume down
+ case LMODE_FINISH: /*
+ * Final loop finished, shoot volume down
*/
- if ((gus_read16 (0x09) >> 4) < 100) /*
- * Get current volume
+ if ((gus_read16 (0x09) >> 4) < 100) /*
+ * Get current volume
*/
{
gus_voice_off ();
@@ -3241,18 +3379,18 @@ do_loop_irq (int voice)
break;
}
gus_ramp_range (65, 4065);
- gus_ramp_rate (0, 63); /*
- * Fastest possible rate
+ gus_ramp_rate (0, 63); /*
+ * Fastest possible rate
*/
- gus_rampon (0x20 | 0x40); /*
- * Ramp down, once, irq
+ gus_rampon (0x20 | 0x40); /*
+ * Ramp down, once, irq
*/
voices[voice].volume_irq_mode = VMODE_HALT;
break;
case LMODE_PCM_STOP:
- pcm_active = 0; /*
- * Requires extensive processing
+ pcm_active = 0; /*
+ * Requires extensive processing
*/
case LMODE_PCM:
{
@@ -3265,8 +3403,8 @@ do_loop_irq (int voice)
play_next_pcm_block ();
}
else
- { /*
- * Out of data. Just stop the voice
+ { /*
+ * Out of data. Just stop the voice
*/
gus_voice_off ();
gus_rampoff ();
@@ -3297,8 +3435,8 @@ do_volume_irq (int voice)
gus_select_voice (voice);
tmp = gus_read8 (0x0d);
- tmp &= ~0x20; /*
- * Disable volume ramp IRQ
+ tmp &= ~0x20; /*
+ * Disable volume ramp IRQ
*/
gus_write8 (0x0d, tmp);
@@ -3308,18 +3446,21 @@ do_volume_irq (int voice)
switch (mode)
{
- case VMODE_HALT: /*
- * Decay phase finished
+ case VMODE_HALT: /*
+ * Decay phase finished
*/
+ RESTORE_INTR (flags);
gus_voice_init (voice);
break;
case VMODE_ENVELOPE:
gus_rampoff ();
+ RESTORE_INTR (flags);
step_envelope (voice);
break;
case VMODE_START_NOTE:
+ RESTORE_INTR (flags);
guswave_start_note2 (voices[voice].dev_pending, voice,
voices[voice].note_pending, voices[voice].volume_pending);
if (voices[voice].kill_pending)
@@ -3334,8 +3475,6 @@ do_volume_irq (int voice)
default:;
}
-
- RESTORE_INTR (flags);
}
void
@@ -3348,37 +3487,37 @@ gus_voice_irq (void)
while (1)
{
- src = gus_read8 (0x0f); /*
- * Get source info
+ src = gus_read8 (0x0f); /*
+ * Get source info
*/
voice = src & 0x1f;
src &= 0xc0;
if (src == (0x80 | 0x40))
- return; /*
- * No interrupt
+ return; /*
+ * No interrupt
*/
voice_bit = 1 << voice;
- if (!(src & 0x80)) /*
- * Wave IRQ pending
+ if (!(src & 0x80)) /*
+ * Wave IRQ pending
*/
- if (!(wave_ignore & voice_bit) && voice < nr_voices) /*
+ if (!(wave_ignore & voice_bit) && voice < nr_voices) /*
* Not done
- * yet
+ * yet
*/
{
wave_ignore |= voice_bit;
do_loop_irq (voice);
}
- if (!(src & 0x40)) /*
- * Volume IRQ pending
+ if (!(src & 0x40)) /*
+ * Volume IRQ pending
*/
- if (!(volume_ignore & voice_bit) && voice < nr_voices) /*
+ if (!(volume_ignore & voice_bit) && voice < nr_voices) /*
* Not done
- * yet
+ * yet
*/
{
volume_ignore |= voice_bit;
@@ -3392,11 +3531,11 @@ guswave_dma_irq (void)
{
unsigned char status;
- status = gus_look8 (0x41); /*
- * Get DMA IRQ Status
+ status = gus_look8 (0x41); /*
+ * Get DMA IRQ Status
*/
- if (status & 0x40) /*
- * DMA Irq pending
+ if (status & 0x40) /*
+ * DMA Irq pending
*/
switch (active_device)
{
@@ -3421,11 +3560,11 @@ guswave_dma_irq (void)
default:;
}
- status = gus_look8 (0x49); /*
- * Get Sampling IRQ Status
+ status = gus_look8 (0x49); /*
+ * Get Sampling IRQ Status
*/
- if (status & 0x40) /*
- * Sampling Irq pending
+ if (status & 0x40) /*
+ * Sampling Irq pending
*/
{
DMAbuf_inputintr (gus_devnum);
diff --git a/sys/i386/isa/sound/midibuf.c b/sys/i386/isa/sound/midibuf.c
index 54b7266..7dadb3f 100644
--- a/sys/i386/isa/sound/midibuf.c
+++ b/sys/i386/isa/sound/midibuf.c
@@ -1,12 +1,12 @@
/*
* sound/midibuf.c
- *
+ *
* Device file manager for /dev/midi
- *
+ *
* NOTE! This part of the driver is currently just a stub.
- *
+ *
* Copyright by Hannu Savolainen 1993
- *
+ *
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions are
* met: 1. Redistributions of source code must retain the above copyright
@@ -14,7 +14,7 @@
* Redistributions in binary form must reproduce the above copyright notice,
* this list of conditions and the following disclaimer in the documentation
* and/or other materials provided with the distribution.
- *
+ *
* THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND ANY
* EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
* WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
@@ -26,7 +26,7 @@
* LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
* OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
* SUCH DAMAGE.
- *
+ *
*/
#include "sound_config.h"
diff --git a/sys/i386/isa/sound/mpu401.c b/sys/i386/isa/sound/mpu401.c
index 449b6be..38ba486 100644
--- a/sys/i386/isa/sound/mpu401.c
+++ b/sys/i386/isa/sound/mpu401.c
@@ -1,12 +1,12 @@
/*
* sound/mpu401.c
- *
+ *
* The low level driver for Roland MPU-401 compatible Midi cards.
- *
+ *
* This version supports just the DUMB UART mode.
- *
+ *
* Copyright by Hannu Savolainen 1993
- *
+ *
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions are
* met: 1. Redistributions of source code must retain the above copyright
@@ -14,7 +14,7 @@
* Redistributions in binary form must reproduce the above copyright notice,
* this list of conditions and the following disclaimer in the documentation
* and/or other materials provided with the distribution.
- *
+ *
* THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND ANY
* EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
* WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
@@ -26,7 +26,7 @@
* LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
* OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
* SUCH DAMAGE.
- *
+ *
*/
#include "sound_config.h"
@@ -35,7 +35,7 @@
#if !defined(EXCLUDE_MPU401) && !defined(EXCLUDE_MIDI)
-#define DATAPORT (mpu401_base) /* MPU-401 Data I/O Port on IBM */
+#define DATAPORT (mpu401_base)/* MPU-401 Data I/O Port on IBM */
#define COMDPORT (mpu401_base+1) /* MPU-401 Command Port on IBM */
#define STATPORT (mpu401_base+1) /* MPU-401 Status Port on IBM */
@@ -61,58 +61,16 @@ static int my_dev;
static int reset_mpu401 (void);
static void (*midi_input_intr) (int dev, unsigned char data);
-static void
-mpu401_input_loop (void)
-{
- int count;
-
- count = 10;
-
- while (count) /* Not timed out */
- if (input_avail ())
- {
- unsigned char c = mpu401_read ();
-
- count = 100;
-
- if (mpu401_opened & OPEN_READ)
- midi_input_intr (my_dev, c);
- }
- else
- while (!input_avail () && count)
- count--;
-}
-
void
mpuintr (int unit)
{
- if (input_avail ())
- mpu401_input_loop ();
-}
-
-/*
- * It looks like there is no input interrupts in the UART mode. Let's try
- * polling.
- */
-
-static void
-poll_mpu401 (unsigned long dummy)
-{
- unsigned long flags;
-
- DEFINE_TIMER (mpu401_timer, poll_mpu401);
-
- if (!(mpu401_opened & OPEN_READ))
- return; /* No longer required */
-
- DISABLE_INTR (flags);
-
- if (input_avail ())
- mpu401_input_loop ();
-
- ACTIVATE_TIMER (mpu401_timer, poll_mpu401, 1); /* Come back later */
+ while (input_avail ())
+ {
+ unsigned char c = mpu401_read ();
- RESTORE_INTR (flags);
+ if (mpu401_opened & OPEN_READ)
+ midi_input_intr (my_dev, c);
+ }
}
static int
@@ -127,11 +85,10 @@ mpu401_open (int dev, int mode,
return RET_ERROR (EBUSY);
}
- mpu401_input_loop ();
+ mpuintr (0);
midi_input_intr = input;
mpu401_opened = mode;
- poll_mpu401 (0); /* Enable input polling */
return 0;
}
@@ -155,7 +112,7 @@ mpu401_out (int dev, unsigned char midi_byte)
DISABLE_INTR (flags);
if (input_avail ())
- mpu401_input_loop ();
+ mpuintr (0);
RESTORE_INTR (flags);
@@ -250,7 +207,11 @@ attach_mpu401 (long mem_start, struct address_info *hw_config)
RESTORE_INTR (flags);
+#ifdef __FreeBSD__
printk ("snd5: <Roland MPU-401>");
+#else
+ printk (" <Roland MPU-401>");
+#endif
my_dev = num_midis;
mpu401_dev = num_midis;
@@ -291,7 +252,7 @@ reset_mpu401 (void)
mpu401_opened = 0;
if (ok)
- mpu401_input_loop (); /* Flush input before enabling interrupts */
+ mpuintr (0); /* Flush input before enabling interrupts */
RESTORE_INTR (flags);
diff --git a/sys/i386/isa/sound/opl3.c b/sys/i386/isa/sound/opl3.c
index 58327a3..6e3dcca 100644
--- a/sys/i386/isa/sound/opl3.c
+++ b/sys/i386/isa/sound/opl3.c
@@ -1,10 +1,10 @@
/*
* sound/opl3.c
- *
+ *
* A low level driver for Yamaha YM3812 and OPL-3 -chips
- *
+ *
* Copyright by Hannu Savolainen 1993
- *
+ *
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions are
* met: 1. Redistributions of source code must retain the above copyright
@@ -12,7 +12,7 @@
* Redistributions in binary form must reproduce the above copyright notice,
* this list of conditions and the following disclaimer in the documentation
* and/or other materials provided with the distribution.
- *
+ *
* THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND ANY
* EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
* WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
@@ -24,7 +24,7 @@
* LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
* OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
* SUCH DAMAGE.
- *
+ *
*/
/* Major improvements to the FM handling 30AUG92 by Rob Hooft, */
@@ -81,6 +81,9 @@ static unsigned char connection_mask = 0x00;
void
enable_opl3_mode (int left, int right, int both)
{
+ if (opl3_enabled)
+ return;
+
opl3_enabled = 1;
left_address = left;
right_address = right;
@@ -168,10 +171,10 @@ opl3_detect (int ioaddr)
* This function returns 1 if the FM chicp is present at the given I/O port
* The detection algorithm plays with the timer built in the FM chip and
* looks for a change in the status register.
- *
+ *
* Note! The timers of the FM chip are not connected to AdLib (and compatible)
* boards.
- *
+ *
* Note2! The chip is initialized if detected.
*/
@@ -197,7 +200,7 @@ opl3_detect (int ioaddr)
return 0; /* Should be 0x00 */
}
- opl3_command (ioaddr, TIMER1_REGISTER, 0xff); /* Set timer 1 to 0xff */
+ opl3_command (ioaddr, TIMER1_REGISTER, 0xff); /* Set timer 1 to 0xff */
opl3_command (ioaddr, TIMER_CONTROL_REGISTER,
TIMER2_MASK | TIMER1_START); /* Unmask and start timer 1 */
@@ -305,9 +308,9 @@ char fm_volume_table[128] =
-24, -23, -21, -20, -19, -18, -18, -17, /* 8 - 15 */
-16, -15, -15, -14, -13, -13, -12, -12, /* 16 - 23 */
-11, -11, -10, -10, -10, -9, -9, -8, /* 24 - 31 */
- -8, -8, -7, -7, -7, -6, -6, -6, /* 32 - 39 */
- -5, -5, -5, -5, -4, -4, -4, -4, /* 40 - 47 */
- -3, -3, -3, -3, -2, -2, -2, -2, /* 48 - 55 */
+ -8, -8, -7, -7, -7, -6, -6, -6,/* 32 - 39 */
+ -5, -5, -5, -5, -4, -4, -4, -4,/* 40 - 47 */
+ -3, -3, -3, -3, -2, -2, -2, -2,/* 48 - 55 */
-2, -1, -1, -1, -1, 0, 0, 0, /* 56 - 63 */
0, 0, 0, 1, 1, 1, 1, 1, /* 64 - 71 */
1, 2, 2, 2, 2, 2, 2, 2, /* 72 - 79 */
@@ -373,8 +376,8 @@ set_voice_volume (int voice, int volume)
calc_vol (&vol2, volume);
}
- opl3_command (map->ioaddr, KSL_LEVEL + map->op[0], vol1); /* Modulator volume */
- opl3_command (map->ioaddr, KSL_LEVEL + map->op[1], vol2); /* Carrier volume */
+ opl3_command (map->ioaddr, KSL_LEVEL + map->op[0], vol1); /* Modulator volume */
+ opl3_command (map->ioaddr, KSL_LEVEL + map->op[1], vol2); /* Carrier volume */
}
else
{ /* 4 OP voice */
@@ -414,7 +417,7 @@ set_voice_volume (int voice, int volume)
calc_vol (&vol4, volume);
break;
- default: /* Why ?? */ ;
+ default:/* Why ?? */ ;
}
opl3_command (map->ioaddr, KSL_LEVEL + map->op[0], vol1);
@@ -747,6 +750,11 @@ opl3_panning (int dev, int voice, int pressure)
{
}
+static void
+opl3_volume_method (int dev, int mode)
+{
+}
+
#define SET_VIBRATO(cell) { \
tmp = instr->operators[(cell-1)+(((cell-1)/2)*OFFS_4OP)]; \
if (pressure > 110) \
@@ -851,9 +859,8 @@ opl3_controller (int dev, int voice, int ctrl_num, int value)
data = fnum & 0xff; /* Least significant bits of fnumber */
opl3_command (map->ioaddr, FNUM_LOW + map->voice_num, data);
- data = 0x20 | ((block & 0x7) << 2) | ((fnum >> 8) & 0x3);
- /* KEYON|OCTAVE|MS bits * of f-num */
-
+ data = 0x20 | ((block & 0x7) << 2) | ((fnum >> 8) & 0x3); /* KEYON|OCTAVE|MS bits
+ * of f-num */
voices[voice].keyon_byte = data;
opl3_command (map->ioaddr, KEYON_BLOCK + map->voice_num, data);
break;
@@ -887,6 +894,7 @@ static struct synth_operations opl3_operations =
opl3_aftertouch,
opl3_controller,
opl3_panning,
+ opl3_volume_method,
opl3_patchmgr
};
@@ -895,7 +903,7 @@ opl3_init (long mem_start)
{
int i;
- PERMANENT_MALLOC(struct sbi_instrument *, instrmap,
+ PERMANENT_MALLOC (struct sbi_instrument *, instrmap,
SBFM_MAXINSTR * sizeof (*instrmap), mem_start);
synth_devs[num_synths++] = &opl3_operations;
@@ -903,7 +911,11 @@ opl3_init (long mem_start)
opl3_ok = 1;
if (opl3_enabled)
{
+#ifdef __FreeBSD__
printk ("snd1: <Yamaha OPL-3 FM>");
+#else
+ printk (" <Yamaha OPL-3 FM>");
+#endif
fm_model = 2;
nr_voices = 18;
fm_info.nr_drums = 0;
@@ -925,7 +937,11 @@ opl3_init (long mem_start)
}
else
{
+#ifdef __FreeBSD__
printk ("snd1: <Yamaha 2-OP FM>");
+#else
+ printk (" <Yamaha 2-OP FM>");
+#endif
fm_model = 1;
nr_voices = 9;
fm_info.nr_drums = 0;
diff --git a/sys/i386/isa/sound/pas.h b/sys/i386/isa/sound/pas.h
index 797c8fd..9902e03 100644
--- a/sys/i386/isa/sound/pas.h
+++ b/sys/i386/isa/sound/pas.h
@@ -138,11 +138,11 @@
char I_C_2_PCM_DMA_translate[] = /* R W PCM PCM DMA channel value translations */
{ 4, 1, 2, 3, 0, 5, 6, 7 };
char I_C_3_PCM_IRQ_translate[] = /* R W PCM PCM IRQ level value translation */
- { 0, 0, 1, 2, 3, 4, 5, 6, 0, 0, 7, 8, 9, 0, 10, 11 };
+ { 0, 0, 1, 2, 3, 4, 5, 6, 0, 1, 7, 8, 9, 0, 10, 11 };
char E_C_MPU401_IRQ_translate[] = /* R W MIDI MPU401 emulation IRQ value translation */
- { 0x00, 0x00, 0x01, 0x02, 0x00, 0x03, 0x00, 0x04, 0x00, 0x00, 0x05, 0x06, 0x07 };
+ { 0x00, 0x00, 0x01, 0x02, 0x00, 0x03, 0x00, 0x04, 0x00, 0x01, 0x05, 0x06, 0x07 };
char E_C_SB_IRQ_translate[] = /* R W PCM SB emulation IRQ translate */
- { 0x00, 0x00, 0x08, 0x10, 0x00, 0x18, 0x00, 0x20, 0x00, 0x00, 0x28, 0x30, 0x38, 0, 0 };
+ { 0x00, 0x00, 0x08, 0x10, 0x00, 0x18, 0x00, 0x20, 0x00, 0x08, 0x28, 0x30, 0x38, 0, 0 };
char E_C_SB_DMA_translate[] = /* R W PCM SB emulation DMA translate */
{ 0x00, 0x40, 0x80, 0xC0, 0, 0, 0, 0 };
char O_M_1_to_card[] = /* R W Control Translate (OM1 & 0x0f) to card type */
diff --git a/sys/i386/isa/sound/pas2_card.c b/sys/i386/isa/sound/pas2_card.c
index c89dd33..a4791e8 100644
--- a/sys/i386/isa/sound/pas2_card.c
+++ b/sys/i386/isa/sound/pas2_card.c
@@ -2,11 +2,11 @@
#define SND_SA_INTERRUPT
/*
* sound/pas2_card.c
- *
+ *
* Detection routine for the Pro Audio Spectrum cards.
- *
+ *
* Copyright by Hannu Savolainen 1993
- *
+ *
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions are
* met: 1. Redistributions of source code must retain the above copyright
@@ -14,7 +14,7 @@
* Redistributions in binary form must reproduce the above copyright notice,
* this list of conditions and the following disclaimer in the documentation
* and/or other materials provided with the distribution.
- *
+ *
* THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND ANY
* EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
* WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
@@ -26,7 +26,7 @@
* LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
* OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
* SUCH DAMAGE.
- *
+ *
*/
#include "sound_config.h"
@@ -79,7 +79,7 @@ pasintr (int unused)
int status;
status = pas_read (INTERRUPT_STATUS);
- pas_write (status, INTERRUPT_STATUS); /* Clear interrupt */
+ pas_write (status, INTERRUPT_STATUS); /* Clear interrupt */
if (status & I_S_PCM_SAMPLE_BUFFER_IRQ)
{
@@ -197,7 +197,7 @@ config_pas_hw (struct address_info *hw_config)
}
}
-/*
+ /*
* This fixes the timing problems of the PAS due to the Symphony chipset
* as per Media Vision. Only define this if your PAS doesn't work correctly.
*/
@@ -317,7 +317,11 @@ attach_pas_card (long mem_start, struct address_info *hw_config)
if ((pas_model = O_M_1_to_card[pas_read (OPERATION_MODE_1) & 0x0f]))
{
+#ifdef __FreeBSD__
printk ("snd3: <%s rev %d>", pas_model_names[(int) pas_model], pas_read (BOARD_REV_ID));
+#else
+ printk (" <%s rev %d>", pas_model_names[(int) pas_model], pas_read (BOARD_REV_ID));
+#endif
}
if (config_pas_hw (hw_config))
diff --git a/sys/i386/isa/sound/pas2_midi.c b/sys/i386/isa/sound/pas2_midi.c
index 5620d72..4a07b0b 100644
--- a/sys/i386/isa/sound/pas2_midi.c
+++ b/sys/i386/isa/sound/pas2_midi.c
@@ -1,10 +1,10 @@
/*
* sound/pas2_midi.c
- *
+ *
* The low level driver for the PAS Midi Interface.
- *
+ *
* Copyright by Hannu Savolainen 1993
- *
+ *
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions are
* met: 1. Redistributions of source code must retain the above copyright
@@ -12,7 +12,7 @@
* Redistributions in binary form must reproduce the above copyright notice,
* this list of conditions and the following disclaimer in the documentation
* and/or other materials provided with the distribution.
- *
+ *
* THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND ANY
* EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
* WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
@@ -24,7 +24,7 @@
* LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
* OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
* SUCH DAMAGE.
- *
+ *
*/
#include "sound_config.h"
@@ -79,7 +79,7 @@ pas_midi_open (int dev, int mode,
if (mode == OPEN_READ || mode == OPEN_READWRITE)
{
- ctrl |= M_C_ENA_INPUT_IRQ; /* Enable input */
+ ctrl |= M_C_ENA_INPUT_IRQ;/* Enable input */
input_opened = 1;
}
@@ -122,7 +122,7 @@ dump_to_midi (unsigned char midi_byte)
fifo_space = ((x = pas_read (MIDI_FIFO_STATUS)) >> 4) & 0x0f;
- if (fifo_space == 15 || (fifo_space < 2 && ofifo_bytes > 13)) /* Fifo full */
+ if (fifo_space == 15 || (fifo_space < 2 && ofifo_bytes > 13)) /* Fifo full */
{
return 0; /* Upper layer will call again */
}
@@ -287,7 +287,7 @@ pas_midi_interrupt (void)
ofifo_bytes = 100;
}
- pas_write (stat, MIDI_STATUS); /* Acknowledge interrupts */
+ pas_write (stat, MIDI_STATUS);/* Acknowledge interrupts */
}
#endif
diff --git a/sys/i386/isa/sound/pas2_mixer.c b/sys/i386/isa/sound/pas2_mixer.c
index 0ec6753..fd3e106 100644
--- a/sys/i386/isa/sound/pas2_mixer.c
+++ b/sys/i386/isa/sound/pas2_mixer.c
@@ -2,11 +2,11 @@
/*
* sound/pas2_mixer.c
- *
+ *
* Mixer routines for the Pro Audio Spectrum cards.
- *
+ *
* Copyright by Hannu Savolainen 1993
- *
+ *
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions are
* met: 1. Redistributions of source code must retain the above copyright
@@ -14,7 +14,7 @@
* Redistributions in binary form must reproduce the above copyright notice,
* this list of conditions and the following disclaimer in the documentation
* and/or other materials provided with the distribution.
- *
+ *
* THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND ANY
* EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
* WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
@@ -26,7 +26,7 @@
* LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
* OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
* SUCH DAMAGE.
- *
+ *
*/
#include "sound_config.h"
@@ -39,7 +39,7 @@
extern int translat_code;
-static int rec_devices = (SOUND_MASK_MIC); /* Default recording source */
+static int rec_devices = (SOUND_MASK_MIC); /* Default recording source */
static int mode_control = 0;
#define POSSIBLE_RECORDING_DEVICES (SOUND_MASK_SYNTH | SOUND_MASK_SPEAKER | SOUND_MASK_LINE | SOUND_MASK_MIC | \
@@ -331,7 +331,7 @@ mixer_get_levels (struct sb_mixer_levels *user_l)
struct sb_mixer_levels l;
l.master.r = ((((levels[SOUND_MIXER_VOLUME] >> 8) & 0x7f) * 15) + 50) / 100; /* Master */
- l.master.l = (((levels[SOUND_MIXER_VOLUME] & 0x7f) * 15) + 50) / 100; /* Master */
+ l.master.l = (((levels[SOUND_MIXER_VOLUME] & 0x7f) * 15) + 50) / 100; /* Master */
l.line.r = ((getmixer (SOUND_MIXER_LINE, P_M_MV508_RIGHT) * 15) + 50) / 100; /* Line */
l.line.l = ((getmixer (SOUND_MIXER_LINE, P_M_MV508_LEFT) * 15) + 50) / 100;
diff --git a/sys/i386/isa/sound/pas2_pcm.c b/sys/i386/isa/sound/pas2_pcm.c
index f8b2238..06742d5 100644
--- a/sys/i386/isa/sound/pas2_pcm.c
+++ b/sys/i386/isa/sound/pas2_pcm.c
@@ -1,11 +1,11 @@
#define _PAS2_PCM_C_
/*
* sound/pas2_pcm.c
- *
+ *
* The low level driver for the Pro Audio Spectrum ADC/DAC.
- *
+ *
* Copyright by Hannu Savolainen 1993
- *
+ *
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions are
* met: 1. Redistributions of source code must retain the above copyright
@@ -13,7 +13,7 @@
* Redistributions in binary form must reproduce the above copyright notice,
* this list of conditions and the following disclaimer in the documentation
* and/or other materials provided with the distribution.
- *
+ *
* THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND ANY
* EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
* WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
@@ -25,7 +25,7 @@
* LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
* OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
* SUCH DAMAGE.
- *
+ *
*/
#include "sound_config.h"
@@ -100,7 +100,7 @@ pcm_set_channels (int arg)
pas_write (pas_read (PCM_CONTROL) ^ P_C_PCM_MONO, PCM_CONTROL);
pcm_channels = arg;
- pcm_set_speed (pcm_speed); /* The speed must be reinitialized */
+ pcm_set_speed (pcm_speed);/* The speed must be reinitialized */
}
return pcm_channels;
@@ -337,14 +337,15 @@ pas_pcm_prepare_for_output (int dev, int bsize, int bcount)
static struct audio_operations pas_pcm_operations =
{
"Pro Audio Spectrum",
- pas_pcm_open, /* */
- pas_pcm_close, /* */
- pas_pcm_output_block, /* */
- pas_pcm_start_input, /* */
- pas_pcm_ioctl, /* */
- pas_pcm_prepare_for_input, /* */
- pas_pcm_prepare_for_output, /* */
- pas_pcm_reset, /* */
+ NOTHING_SPECIAL,
+ pas_pcm_open,
+ pas_pcm_close,
+ pas_pcm_output_block,
+ pas_pcm_start_input,
+ pas_pcm_ioctl,
+ pas_pcm_prepare_for_input,
+ pas_pcm_prepare_for_output,
+ pas_pcm_reset,
pas_pcm_reset, /* halt_xfer */
NULL, /* has_output_drained */
NULL /* copy_from_user */
diff --git a/sys/i386/isa/sound/patmgr.c b/sys/i386/isa/sound/patmgr.c
index f5697ae..042d42d 100644
--- a/sys/i386/isa/sound/patmgr.c
+++ b/sys/i386/isa/sound/patmgr.c
@@ -1,10 +1,10 @@
/*
* sound/patmgr.c
- *
+ *
* The patch maneger interface for the /dev/sequencer
- *
+ *
* Copyright by Hannu Savolainen 1993
- *
+ *
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions are
* met: 1. Redistributions of source code must retain the above copyright
@@ -12,7 +12,7 @@
* Redistributions in binary form must reproduce the above copyright notice,
* this list of conditions and the following disclaimer in the documentation
* and/or other materials provided with the distribution.
- *
+ *
* THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND ANY
* EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
* WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
@@ -24,7 +24,7 @@
* LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
* OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
* SUCH DAMAGE.
- *
+ *
*/
#define PATMGR_C
diff --git a/sys/i386/isa/sound/sb16_dsp.c b/sys/i386/isa/sound/sb16_dsp.c
index d85359b..a755712 100644
--- a/sys/i386/isa/sound/sb16_dsp.c
+++ b/sys/i386/isa/sound/sb16_dsp.c
@@ -1,12 +1,12 @@
/*
* sound/sb16_dsp.c
- *
+ *
* The low level driver for the SoundBlaster DSP chip.
- *
+ *
* (C) 1993 J. Schubert (jsb@sth.ruhr-uni-bochum.de)
*
* based on SB-driver by (C) Hannu Savolainen
- *
+ *
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions are
* met: 1. Redistributions of source code must retain the above copyright
@@ -14,7 +14,7 @@
* Redistributions in binary form must reproduce the above copyright notice,
* this list of conditions and the following disclaimer in the documentation
* and/or other materials provided with the distribution.
- *
+ *
* THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND ANY
* EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
* WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
@@ -26,7 +26,7 @@
* LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
* OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
* SUCH DAMAGE.
- *
+ *
*/
#define DEB(x)
@@ -40,9 +40,9 @@
#if defined(CONFIGURE_SOUNDCARD) && !defined(EXCLUDE_SB16) && !defined(EXCLUDE_SB) && !defined(EXCLUDE_AUDIO) && !defined(EXCLUDE_SBPRO)
-extern int sbc_base;
+extern int sbc_base, sbc_minor, sbc_major;
-static int sb16_dsp_ok = 0; /* Set to 1 after successful initialization */
+static int sb16_dsp_ok = 0;/* Set to 1 after successful initialization */
static int dsp_16bit = 0;
static int dsp_stereo = 0;
static int dsp_current_speed = 8000; /*DSP_DEFAULT_SPEED; */
@@ -74,6 +74,7 @@ int sb_reset_dsp (void);
static struct audio_operations sb16_dsp_operations =
{
"SoundBlaster 16",
+ NOTHING_SPECIAL,
sb16_dsp_open,
sb16_dsp_close,
sb16_dsp_output_block,
@@ -87,7 +88,7 @@ static struct audio_operations sb16_dsp_operations =
NULL
};
-static int
+static int
sb_dsp_command01 (unsigned char val)
{
int i = 1 << 16;
@@ -98,33 +99,32 @@ sb_dsp_command01 (unsigned char val)
return sb_dsp_command (val);
}
-static int
-wait_data_avail (int t)
+static int
+wait_data_avail (unsigned long t)
{
int loopc = 5000000;
- unsigned long tt;
- tt = t + GET_TIME ();
+ t += GET_TIME ();
do
{
if (INB (DSP_DATA_AVAIL) & 0x80)
return 1;
}
- while (--loopc && GET_TIME () < tt);
+ while (--loopc && GET_TIME () < t);
printk ("!data_avail l=%d\n", loopc);
return 0;
}
-static int
+static int
read_dsp (int t)
{
- if (!wait_data_avail (t))
+ if (!wait_data_avail ((unsigned long) t))
return -1;
else
return INB (DSP_READ);
}
-static int
+static int
dsp_ini2 (void)
{
#if 0
@@ -141,6 +141,7 @@ dsp_ini2 (void)
#endif
return 0;
}
+
/*
static char *dsp_getmessage(unsigned char command,int maxn)
{
@@ -148,7 +149,7 @@ dsp_ini2 (void)
int n=0;
sb_dsp_command(command);
- while(n<maxn && wait_data_avail(2)) {
+ while(n<maxn && wait_data_avail(2L)) {
buff[++n]=INB(DSP_READ);
if(!buff[n])
break;
@@ -171,7 +172,7 @@ dsp_ini2 (void)
printk("\n");
}
*/
-static int
+static int
dsp_set_speed (int mode)
{
DEB (printk ("dsp_set_speed(%d)\n", mode));
@@ -186,7 +187,7 @@ dsp_set_speed (int mode)
return mode;
}
-static int
+static int
dsp_set_stereo (int mode)
{
DEB (printk ("dsp_set_stereo(%d)\n", mode));
@@ -196,7 +197,7 @@ dsp_set_stereo (int mode)
return mode;
}
-static int
+static int
dsp_set_bits (int arg)
{
DEB (printk ("dsp_set_bits(%d)\n", arg));
@@ -523,36 +524,24 @@ set_irq_hw (int level)
long
sb16_dsp_init (long mem_start, struct address_info *hw_config)
{
- int i, major, minor;
-
- major = minor = 0;
- sb_dsp_command (0xe1); /* Get version */
-
- for (i = 1000; i; i--)
- {
- if (INB (DSP_DATA_AVAIL) & 0x80)
- { /* wait for Data Ready */
- if (major == 0)
- major = INB (DSP_READ);
- else
- {
- minor = INB (DSP_READ);
- break;
- }
- }
- }
+ if (sbc_major < 4)
+ return mem_start;
#ifndef SCO
- sprintf (sb16_dsp_operations.name, "SoundBlaster 16 %d.%d", major, minor);
+ sprintf (sb16_dsp_operations.name, "SoundBlaster 16 %d.%d", sbc_major, sbc_minor);
#endif
+#ifdef __FreeBSD__
printk ("snd6: <%s>", sb16_dsp_operations.name);
+#else
+ printk (" <%s>", sb16_dsp_operations.name);
+#endif
if (num_dspdevs < MAX_DSP_DEV)
{
dsp_devs[my_dev = num_dspdevs++] = &sb16_dsp_operations;
sound_dsp_dmachan[my_dev] = hw_config->dma;
- sound_buffcounts[my_dev] = DSP_BUFFCOUNT;
+ sound_buffcounts[my_dev] = 1;
sound_buffsizes[my_dev] = DSP_BUFFSIZE;
sound_dma_automode[my_dev] = 1;
}
@@ -576,9 +565,6 @@ sb16_dsp_detect (struct address_info *hw_config)
return 0;
}
- if (sbc_base != hw_config->io_base)
- printk ("Warning! SB16 I/O != SB I/O\n");
-
/* sb_setmixer(OPSW,0xf);
if(sb_getmixer(OPSW)!=0xf)
return 0; */
@@ -586,13 +572,6 @@ sb16_dsp_detect (struct address_info *hw_config)
if (!sb_reset_dsp ())
return 0;
- if (hw_config->irq != sb_config->irq)
- {
- printk ("SB16 Error: Invalid IRQ number %d/%d\n",
- sb_config->irq, hw_config->irq);
- return 0;
- }
-
if (hw_config->dma < 4)
if (hw_config->dma != sb_config->dma)
{
@@ -603,12 +582,12 @@ sb16_dsp_detect (struct address_info *hw_config)
dma16 = hw_config->dma;
dma8 = sb_config->dma;
- set_irq_hw (hw_config->irq);
+ set_irq_hw (sb_config->irq);
sb_setmixer (DMA_NR, (1 << hw_config->dma) | (1 << sb_config->dma));
- DEB (printk ("SoundBlaster 16: IRQ %d DMA %d OK\n", hw_config->irq, hw_config->dma));
+ DEB (printk ("SoundBlaster 16: IRQ %d DMA %d OK\n", sb_config->irq, hw_config->dma));
-/*
+ /*
dsp_showmessage(0xe3,99);
*/
sb16_dsp_ok = 1;
@@ -639,4 +618,5 @@ sb16_dsp_interrupt (int unused)
printk ("SoundBlaster: Unexpected interrupt\n");
}
}
+
#endif
diff --git a/sys/i386/isa/sound/sb16_midi.c b/sys/i386/isa/sound/sb16_midi.c
index f7a61a6..39808c8 100644
--- a/sys/i386/isa/sound/sb16_midi.c
+++ b/sys/i386/isa/sound/sb16_midi.c
@@ -1,10 +1,10 @@
/*
* sound/sb16_midi.c
- *
+ *
* The low level driver for the MPU-401 UART emulation of the SB16.
- *
+ *
* Copyright by Hannu Savolainen 1993
- *
+ *
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions are
* met: 1. Redistributions of source code must retain the above copyright
@@ -12,7 +12,7 @@
* Redistributions in binary form must reproduce the above copyright notice,
* this list of conditions and the following disclaimer in the documentation
* and/or other materials provided with the distribution.
- *
+ *
* THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND ANY
* EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
* WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
@@ -24,7 +24,7 @@
* LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
* OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
* SUCH DAMAGE.
- *
+ *
*/
#include "sound_config.h"
@@ -58,26 +58,19 @@ static int my_dev;
static int reset_sb16midi (void);
static void (*midi_input_intr) (int dev, unsigned char data);
+extern int sbc_major;
+
static void
sb16midi_input_loop (void)
{
- int count;
-
- count = 10;
-
- while (count) /* Not timed out */
- if (input_avail ())
- {
- unsigned char c = sb16midi_read ();
- count = 100;
+ while (input_avail ())
+ {
+ unsigned char c = sb16midi_read ();
- if (sb16midi_opened & OPEN_READ)
- midi_input_intr (my_dev, c);
- }
- else
- while (!input_avail () && count)
- count--;
+ if (sb16midi_opened & OPEN_READ)
+ midi_input_intr (my_dev, c);
+ }
}
void
@@ -87,31 +80,6 @@ sb16midiintr (int unit)
sb16midi_input_loop ();
}
-/*
- * It looks like there is no input interrupts in the UART mode. Let's try
- * polling.
- */
-
-static void
-poll_sb16midi (unsigned long dummy)
-{
- unsigned long flags;
-
- DEFINE_TIMER (sb16midi_timer, poll_sb16midi);
-
- if (!(sb16midi_opened & OPEN_READ))
- return; /* No longer required */
-
- DISABLE_INTR (flags);
-
- if (input_avail ())
- sb16midi_input_loop ();
-
- ACTIVATE_TIMER (sb16midi_timer, poll_sb16midi, 1); /* Come back later */
-
- RESTORE_INTR (flags);
-}
-
static int
sb16midi_open (int dev, int mode,
void (*input) (int dev, unsigned char data),
@@ -127,7 +95,6 @@ sb16midi_open (int dev, int mode,
midi_input_intr = input;
sb16midi_opened = mode;
- poll_sb16midi (0); /* Enable input polling */
return 0;
}
@@ -245,7 +212,11 @@ attach_sb16midi (long mem_start, struct address_info *hw_config)
RESTORE_INTR (flags);
+#ifdef __FreeBSD__
printk ("snd7: <SoundBlaster MPU-401>");
+#else
+ printk (" <SoundBlaster MPU-401>");
+#endif
my_dev = num_midis;
midi_devs[num_midis++] = &sb16midi_operations;
@@ -299,6 +270,8 @@ probe_sb16midi (struct address_info *hw_config)
int ok = 0;
sb16midi_base = hw_config->io_base;
+ if (sbc_major < 4)
+ return 0; /* SB16 not detected */
if (sb_get_irq () < 0)
return 0;
diff --git a/sys/i386/isa/sound/sb_card.c b/sys/i386/isa/sound/sb_card.c
index ef90b4b..f7588e1 100644
--- a/sys/i386/isa/sound/sb_card.c
+++ b/sys/i386/isa/sound/sb_card.c
@@ -1,10 +1,10 @@
/*
* sound/sb_card.c
- *
+ *
* Detection routine for the SoundBlaster cards.
- *
+ *
* Copyright by Hannu Savolainen 1993
- *
+ *
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions are
* met: 1. Redistributions of source code must retain the above copyright
@@ -12,7 +12,7 @@
* Redistributions in binary form must reproduce the above copyright notice,
* this list of conditions and the following disclaimer in the documentation
* and/or other materials provided with the distribution.
- *
+ *
* THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND ANY
* EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
* WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
@@ -24,7 +24,7 @@
* LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
* OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
* SUCH DAMAGE.
- *
+ *
*/
#include "sound_config.h"
diff --git a/sys/i386/isa/sound/sb_dsp.c b/sys/i386/isa/sound/sb_dsp.c
index 9f0bfb2..17fb4b7 100644
--- a/sys/i386/isa/sound/sb_dsp.c
+++ b/sys/i386/isa/sound/sb_dsp.c
@@ -25,6 +25,10 @@
* OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
* SUCH DAMAGE.
*
+ * Modified:
+ * Hunyue Yau Jan 6 1994
+ * Added code to support Sound Galaxy NX Pro
+ *
*/
#include "sound_config.h"
@@ -37,6 +41,7 @@
int sbc_base = 0;
static int sbc_irq = 0;
+static int open_mode=0;
/*
* The DSP channel can be used either for input or output. Variable
@@ -49,7 +54,8 @@ static int sbc_irq = 0;
int sb_dsp_ok = 0; /* Set to 1 after successful initialization */
static int midi_disabled = 0;
int sb_dsp_highspeed = 0;
-static int major = 1, minor = 0; /* DSP version */
+int sbc_major = 1;
+int sbc_minor = 0; /* DSP version */
static int dsp_stereo = 0;
static int dsp_current_speed = DSP_DEFAULT_SPEED;
static int sb16 = 0;
@@ -64,7 +70,6 @@ volatile int sb_irq_mode = IMODE_NONE; /* IMODE_INPUT, IMODE_OUTPUT
* or IMODE_NONE */
static volatile int irq_ok = 0;
-int sb_dsp_model = 1; /* 1=SB, 2=SB Pro */
int sb_duplex_midi = 0;
static int my_dev = 0;
@@ -84,7 +89,7 @@ sb_dsp_command (unsigned char val)
int i;
unsigned long limit;
- limit = GET_TIME () + HZ/10; /* The timeout is 0.1 secods */
+ limit = GET_TIME () + HZ / 10;/* The timeout is 0.1 secods */
/*
* Note! the i<500000 is an emergency exit. The sb_dsp_command() is sometimes
@@ -111,7 +116,7 @@ sb_dsp_command (unsigned char val)
void
sbintr (int unit)
{
- int status, data;
+ int status;
#ifndef EXCLUDE_SBPRO
if (sb16)
@@ -134,7 +139,7 @@ sbintr (int unit)
}
#endif
- status = INB (DSP_DATA_AVAIL); /* Clear interrupt */
+ status = INB (DSP_DATA_AVAIL);/* Clear interrupt */
if (sb_intr_active)
switch (sb_irq_mode)
@@ -156,10 +161,7 @@ sbintr (int unit)
break;
case IMODE_MIDI:
- printk ("+");
- data = INB (DSP_READ);
- printk ("%x", data);
-
+ sb_midi_interrupt (unit);
break;
default:
@@ -234,17 +236,31 @@ dsp_speed (int speed)
{
unsigned char tconst;
unsigned long flags;
+ int max_speed = 44100;
if (speed < 4000)
speed = 4000;
- if (speed > 44100)
- speed = 44100; /* Invalid speed */
+ /*
+ * Older SB models don't support higher speeds than 22050.
+ */
- if (sb_dsp_model == 1 && speed > 22050)
- speed = 22050;
- /* SB Classic doesn't support higher speed */
+ if (sbc_major < 2 ||
+ (sbc_major == 2 && sbc_minor == 0))
+ max_speed = 22050;
+ /*
+ * SB models earlier than SB Pro have low limit for the input speed.
+ */
+ if (open_mode != OPEN_WRITE) /* Recording is possible */
+ if (sbc_major < 3) /* Limited input speed with these cards */
+ if (sbc_major == 2 && sbc_minor > 0)
+ max_speed = 15000;
+ else
+ max_speed = 13000;
+
+ if (speed > max_speed)
+ speed = max_speed; /* Invalid speed */
if (dsp_stereo && speed > 22050)
speed = 22050;
@@ -285,7 +301,7 @@ dsp_speed (int speed)
tconst = (256 - ((1000000 + speed / 2) / speed)) & 0xff;
DISABLE_INTR (flags);
- if (sb_dsp_command (0x40)) /* Set time constant */
+ if (sb_dsp_command (0x40))/* Set time constant */
sb_dsp_command (tconst);
RESTORE_INTR (flags);
@@ -308,7 +324,7 @@ dsp_set_stereo (int mode)
#ifdef EXCLUDE_SBPRO
return 0;
#else
- if (sb_dsp_model < 3 || sb16)
+ if (sbc_major < 3 || sb16)
return 0; /* Sorry no stereo */
if (mode && sb_midi_busy)
@@ -341,11 +357,11 @@ sb_dsp_output_block (int dev, unsigned long buf, int count,
if (sb_dsp_highspeed)
{
DISABLE_INTR (flags);
- if (sb_dsp_command (0x48)) /* High speed size */
+ if (sb_dsp_command (0x48))/* High speed size */
{
sb_dsp_command ((unsigned char) (count & 0xff));
sb_dsp_command ((unsigned char) ((count >> 8) & 0xff));
- sb_dsp_command (0x91); /* High speed 8 bit DAC */
+ sb_dsp_command (0x91);/* High speed 8 bit DAC */
}
else
printk ("SB Error: Unable to start (high speed) DAC\n");
@@ -354,7 +370,7 @@ sb_dsp_output_block (int dev, unsigned long buf, int count,
else
{
DISABLE_INTR (flags);
- if (sb_dsp_command (0x14)) /* 8-bit DAC (DMA) */
+ if (sb_dsp_command (0x14))/* 8-bit DAC (DMA) */
{
sb_dsp_command ((unsigned char) (count & 0xff));
sb_dsp_command ((unsigned char) ((count >> 8) & 0xff));
@@ -387,11 +403,11 @@ sb_dsp_start_input (int dev, unsigned long buf, int count, int intrflag,
if (sb_dsp_highspeed)
{
DISABLE_INTR (flags);
- if (sb_dsp_command (0x48)) /* High speed size */
+ if (sb_dsp_command (0x48))/* High speed size */
{
sb_dsp_command ((unsigned char) (count & 0xff));
sb_dsp_command ((unsigned char) ((count >> 8) & 0xff));
- sb_dsp_command (0x99); /* High speed 8 bit ADC */
+ sb_dsp_command (0x99);/* High speed 8 bit ADC */
}
else
printk ("SB Error: Unable to start (high speed) ADC\n");
@@ -400,7 +416,7 @@ sb_dsp_start_input (int dev, unsigned long buf, int count, int intrflag,
else
{
DISABLE_INTR (flags);
- if (sb_dsp_command (0x24)) /* 8-bit ADC (DMA) */
+ if (sb_dsp_command (0x24))/* 8-bit ADC (DMA) */
{
sb_dsp_command ((unsigned char) (count & 0xff));
sb_dsp_command ((unsigned char) ((count >> 8) & 0xff));
@@ -425,7 +441,7 @@ sb_dsp_prepare_for_input (int dev, int bsize, int bcount)
dsp_cleanup ();
dsp_speaker (OFF);
- if (major == 3) /* SB Pro */
+ if (sbc_major == 3) /* SB Pro */
{
if (dsp_stereo)
sb_dsp_command (0xa8);
@@ -445,7 +461,7 @@ sb_dsp_prepare_for_output (int dev, int bsize, int bcount)
dsp_speaker (ON);
#ifndef EXCLUDE_SBPRO
- if (major == 3) /* SB Pro */
+ if (sbc_major == 3) /* SB Pro */
{
sb_mixer_set_stereo (dsp_stereo);
dsp_speed (dsp_current_speed); /* Speed must be recalculated if #channels
@@ -534,6 +550,7 @@ sb_dsp_open (int dev, int mode)
sb_irq_mode = IMODE_NONE;
sb_dsp_busy = 1;
+ open_mode = mode;
return 0;
}
@@ -547,6 +564,7 @@ sb_dsp_close (int dev)
dsp_speaker (OFF);
sb_dsp_busy = 0;
sb_dsp_highspeed = 0;
+ open_mode = 0;
}
static int
@@ -588,7 +606,7 @@ sb_dsp_ioctl (int dev, unsigned int cmd, unsigned int arg, int local)
case SOUND_PCM_READ_BITS:
if (local)
return 8;
- return IOCTL_OUT (arg, 8); /* Only 8 bits/sample supported */
+ return IOCTL_OUT (arg, 8);/* Only 8 bits/sample supported */
break;
case SOUND_PCM_WRITE_FILTER:
@@ -611,6 +629,7 @@ sb_dsp_reset (int dev)
DISABLE_INTR (flags);
sb_reset_dsp ();
+ dsp_speed (dsp_current_speed);
dsp_cleanup ();
RESTORE_INTR (flags);
@@ -636,9 +655,10 @@ sb_dsp_detect (struct address_info *hw_config)
static char card_name[32] = "SoundBlaster";
#ifndef EXCLUDE_AUDIO
-struct audio_operations sb_dsp_operations =
+static struct audio_operations sb_dsp_operations =
{
"SoundBlaster",
+ NOTHING_SPECIAL,
sb_dsp_open,
sb_dsp_close,
sb_dsp_output_block,
@@ -658,62 +678,77 @@ long
sb_dsp_init (long mem_start, struct address_info *hw_config)
{
int i;
+ int prostat = 0;
- major = minor = 0;
+ sbc_major = sbc_minor = 0;
sb_dsp_command (0xe1); /* Get version */
for (i = 1000; i; i--)
{
if (INB (DSP_DATA_AVAIL) & 0x80)
{ /* wait for Data Ready */
- if (major == 0)
- major = INB (DSP_READ);
+ if (sbc_major == 0)
+ sbc_major = INB (DSP_READ);
else
{
- minor = INB (DSP_READ);
+ sbc_minor = INB (DSP_READ);
break;
}
}
}
- if (major == 2 || major == 3)
+ if (sbc_major == 2 || sbc_major == 3) /* SB 2.0 or SB Pro */
sb_duplex_midi = 1;
- if (major == 4)
+ if (sbc_major == 4)
sb16 = 1;
- sb_dsp_model = major;
-
#ifndef EXCLUDE_SBPRO
- if (major >= 3)
- sb_mixer_init (major);
+ if (sbc_major >= 3 ||
+ (sbc_major == 2 && sbc_minor == 1)) /* Sound Galaxy ??? */
+ prostat = sb_mixer_init (sbc_major);
#endif
#ifndef EXCLUDE_YM3812
- if (major > 3 || (major == 3 && minor > 0)) /* SB Pro2 or later */
- {
- enable_opl3_mode (OPL3_LEFT, OPL3_RIGHT, OPL3_BOTH);
- }
+ if (sbc_major > 3 ||
+ (sbc_major == 3 && INB (0x388) == 0x00)) /* Non OPL-3 should return 0x06 */
+ enable_opl3_mode (OPL3_LEFT, OPL3_RIGHT, OPL3_BOTH);
#endif
-#ifndef SCO
- if (major >= 3)
+ if (sbc_major >= 3)
{
+#ifndef SCO
+ if (prostat)
+ {
#ifndef EXCLUDE_AUDIO
- sprintf (sb_dsp_operations.name, "SoundBlaster Pro %d.%d", major, minor);
+ sprintf (sb_dsp_operations.name, "Sound Galaxy NX Pro %d.%d", sbc_major, sbc_minor);
+#endif
+ sprintf (card_name, "Sound Galaxy NX Pro %d.%d", sbc_major, sbc_minor);
+ }
+ else
+ {
+#ifndef EXCLUDE_AUDIO
+ sprintf (sb_dsp_operations.name, "SoundBlaster Pro %d.%d", sbc_major, sbc_minor);
+#endif
+ sprintf (card_name, "SoundBlaster Pro %d.%d", sbc_major, sbc_minor);
+ }
#endif
- sprintf (card_name, "SoundBlaster Pro %d.%d", major, minor);
}
else
{
+#ifndef SCO
#ifndef EXCLUDE_AUDIO
- sprintf (sb_dsp_operations.name, "SoundBlaster %d.%d", major, minor);
+ sprintf (sb_dsp_operations.name, "SoundBlaster %d.%d", sbc_major, sbc_minor);
#endif
- sprintf (card_name, "SoundBlaster %d.%d", major, minor);
- }
+ sprintf (card_name, "SoundBlaster %d.%d", sbc_major, sbc_minor);
#endif
+ }
+#ifdef __FreeBSD__
printk ("snd2: <%s>", card_name);
+#else
+ printk (" <%s>", card_name);
+#endif
#ifndef EXCLUDE_AUDIO
#if !defined(EXCLUDE_SB16) && !defined(EXCLUDE_SBPRO)
@@ -734,7 +769,7 @@ sb_dsp_init (long mem_start, struct address_info *hw_config)
#ifndef EXCLUDE_MIDI
if (!midi_disabled && !sb16) /* Midi don't work in the SB emulation mode
* of PAS, SB16 has better midi interface */
- sb_midi_init (major);
+ sb_midi_init (sbc_major);
#endif
sb_dsp_ok = 1;
diff --git a/sys/i386/isa/sound/sb_midi.c b/sys/i386/isa/sound/sb_midi.c
index 2b0ab52..fed19ab 100644
--- a/sys/i386/isa/sound/sb_midi.c
+++ b/sys/i386/isa/sound/sb_midi.c
@@ -1,10 +1,10 @@
/*
* sound/sb_dsp.c
- *
+ *
* The low level driver for the SoundBlaster DS chips.
- *
+ *
* Copyright by Hannu Savolainen 1993
- *
+ *
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions are
* met: 1. Redistributions of source code must retain the above copyright
@@ -12,7 +12,7 @@
* Redistributions in binary form must reproduce the above copyright notice,
* this list of conditions and the following disclaimer in the documentation
* and/or other materials provided with the distribution.
- *
+ *
* THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND ANY
* EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
* WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
@@ -24,7 +24,7 @@
* LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
* OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
* SUCH DAMAGE.
- *
+ *
*/
#include "sound_config.h"
@@ -49,12 +49,16 @@ extern int sb_midi_busy; /* 1 if the process has output to MIDI */
extern int sb_dsp_busy;
extern int sb_dsp_highspeed;
-extern volatile int sb_irq_mode; /* IMODE_INPUT, IMODE_OUTPUT
+extern volatile int sb_irq_mode;/* IMODE_INPUT, IMODE_OUTPUT
* or IMODE_NONE */
-extern int sb_dsp_model; /* 1=SB, 2=SB Pro */
extern int sb_duplex_midi;
extern int sb_intr_active;
+extern int sbc_base;
+
+static int input_opened = 0;
+static void (*midi_input_intr) (int dev, unsigned char data);
+static int my_dev = 0;
static int
sb_midi_open (int dev, int mode,
@@ -73,7 +77,7 @@ sb_midi_open (int dev, int mode,
if (mode != OPEN_WRITE && !sb_duplex_midi)
{
if (num_midis == 1)
- printk ("SoundBlaster: Midi input not currently supported\n");
+ printk ("SoundBlaster: MIDI input not supported with plain SB\n");
return RET_ERROR (EPERM);
}
@@ -97,6 +101,9 @@ sb_midi_open (int dev, int mode,
sb_reset_dsp ();
+ if (!sb_dsp_command (0xf2)) /* This is undodumented, isn't it */
+ return RET_ERROR (EIO); /* be nice to DSP */
+
if (!sb_dsp_command (0x35))
return RET_ERROR (EIO); /* Enter the UART mode */
sb_intr_active = 1;
@@ -106,6 +113,9 @@ sb_midi_open (int dev, int mode,
sb_reset_dsp ();
return 0; /* IRQ not free */
}
+ input_opened = 1;
+ my_dev = dev;
+ midi_input_intr = input;
}
sb_midi_busy = 1;
@@ -123,6 +133,7 @@ sb_midi_close (int dev)
}
sb_intr_active = 0;
sb_midi_busy = 0;
+ input_opened = 0;
}
static int
@@ -175,6 +186,21 @@ sb_midi_ioctl (int dev, unsigned cmd, unsigned arg)
return RET_ERROR (EPERM);
}
+void
+sb_midi_interrupt (int dummy)
+{
+ unsigned long flags;
+ unsigned char data;
+
+ DISABLE_INTR (flags);
+
+ data = INB (DSP_READ);
+ if (input_opened)
+ midi_input_intr (my_dev, data);
+
+ RESTORE_INTR (flags);
+}
+
static struct midi_operations sb_midi_operations =
{
{"SoundBlaster", 0, 0, SNDCARD_SB},
diff --git a/sys/i386/isa/sound/sb_mixer.c b/sys/i386/isa/sound/sb_mixer.c
index 089a2af..39b97ca 100644
--- a/sys/i386/isa/sound/sb_mixer.c
+++ b/sys/i386/isa/sound/sb_mixer.c
@@ -1,11 +1,11 @@
/*
* sound/sb_mixer.c
- *
+ *
* The low level mixer driver for the SoundBlaster Pro and SB16 cards.
- *
+ *
* Copyright by Hannu Savolainen 1993
- *
+ *
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions are
* met: 1. Redistributions of source code must retain the above copyright
@@ -13,7 +13,7 @@
* Redistributions in binary form must reproduce the above copyright notice,
* this list of conditions and the following disclaimer in the documentation
* and/or other materials provided with the distribution.
- *
+ *
* THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND ANY
* EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
* WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
@@ -25,7 +25,11 @@
* LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
* OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
* SUCH DAMAGE.
- *
+ *
+ * Modified:
+ * Hunyue Yau Jan 6 1994
+ * Added code to support the Sound Galaxy NX Pro mixer.
+ *
*/
#include "sound_config.h"
@@ -87,9 +91,21 @@ sb_mixer_set_stereo (int mode)
| (mode ? STEREO_DAC : MONO_DAC)));
}
+/*
+ * Returns:
+ * 0 No mixer detected.
+ * 1 Only a plain Sound Blaster Pro style mixer detected.
+ * 2 The Sound Galaxy NX Pro mixer detected.
+ */
static int
detect_mixer (void)
{
+#ifdef __SGNXPRO__
+ int oldbass, oldtreble;
+
+#endif
+ int retcode = 1;
+
/*
* Detect the mixer by changing parameters of two volume channels. If the
* values read back match with the values written, the mixer is there (is
@@ -103,7 +119,30 @@ detect_mixer (void)
if (sb_getmixer (VOC_VOL) != 0x33)
return 0;
- return 1;
+#ifdef __SGNXPRO__
+ /* Attempt to detect the SG NX Pro by check for valid bass/treble
+ * registers.
+ */
+ oldbass = sb_getmixer (BASS_LVL);
+ oldtreble = sb_getmixer (TREBLE_LVL);
+
+ sb_setmixer (BASS_LVL, 0xaa);
+ sb_setmixer (TREBLE_LVL, 0x55);
+
+ if ((sb_getmixer (BASS_LVL) != 0xaa) ||
+ (sb_getmixer (TREBLE_LVL) != 0x55))
+ {
+ retcode = 1; /* 1 == Only SB Pro detected */
+ }
+ else
+ retcode = 2; /* 2 == SG NX Pro detected */
+ /* Restore register in either case since SG NX Pro has EEPROM with
+ * 'preferred' values stored.
+ */
+ sb_setmixer (BASS_LVL, oldbass);
+ sb_setmixer (TREBLE_LVL, oldtreble);
+#endif
+ return retcode;
}
static void
@@ -320,13 +359,24 @@ sb_mixer_reset (void)
set_recmask (SOUND_MASK_MIC);
}
-void
+/*
+ * Returns a code depending on whether a SG NX Pro was detected.
+ * 0 == Plain SB 16 or SB Pro
+ * 1 == SG NX Pro detected.
+ *
+ * Used to update message.
+ */
+int
sb_mixer_init (int major_model)
{
+ int mixerstat;
+
sb_setmixer (0x00, 0); /* Reset mixer */
- if (!detect_mixer ())
- return; /* No mixer. Why? */
+ mixerstat = detect_mixer ();
+
+ if (!mixerstat)
+ return 0; /* No mixer. Why? */
mixer_initialized = 1;
mixer_model = major_model;
@@ -335,9 +385,21 @@ sb_mixer_init (int major_model)
{
case 3:
mixer_caps = SOUND_CAP_EXCL_INPUT;
- supported_devices = SBPRO_MIXER_DEVICES;
- supported_rec_devices = SBPRO_RECORDING_DEVICES;
- iomap = &sbpro_mix;
+#ifdef __SGNXPRO__
+ if (mixerstat == 2)
+ { /* A SGNXPRO was detected */
+ supported_devices = SGNXPRO_MIXER_DEVICES;
+ supported_rec_devices = SGNXPRO_RECORDING_DEVICES;
+ iomap = &sgnxpro_mix;
+ }
+ else
+#endif
+ { /* Otherwise plain SB Pro */
+ supported_devices = SBPRO_MIXER_DEVICES;
+ supported_rec_devices = SBPRO_RECORDING_DEVICES;
+ iomap = &sbpro_mix;
+ }
+
break;
case 4:
@@ -349,11 +411,12 @@ sb_mixer_init (int major_model)
default:
printk ("SB Warning: Unsupported mixer type\n");
- return;
+ return 0;
}
mixer_devs[num_mixers++] = &sb_mixer_operations;
sb_mixer_reset ();
+ return (mixerstat == 2);
}
#endif
diff --git a/sys/i386/isa/sound/sb_mixer.h b/sys/i386/isa/sound/sb_mixer.h
index 304af642..4caf773 100644
--- a/sys/i386/isa/sound/sb_mixer.h
+++ b/sys/i386/isa/sound/sb_mixer.h
@@ -24,13 +24,29 @@
* LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
* OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
* SUCH DAMAGE.
+ *
+ * Modified:
+ * Hunyue Yau Jan 6 1994
+ * Added defines for the Sound Galaxy NX Pro mixer.
*
*/
+
#define SBPRO_RECORDING_DEVICES (SOUND_MASK_LINE | SOUND_MASK_MIC | SOUND_MASK_CD)
+/* Same as SB Pro, unless I find otherwise */
+#define SGNXPRO_RECORDING_DEVICES SBPRO_RECORDING_DEVICES
+
#define SBPRO_MIXER_DEVICES (SOUND_MASK_SYNTH | SOUND_MASK_PCM | SOUND_MASK_LINE | SOUND_MASK_MIC | \
SOUND_MASK_CD | SOUND_MASK_VOLUME)
+/* SG NX Pro has treble and bass settings on the mixer. The 'speaker'
+ * channel is the COVOX/DisneySoundSource emulation volume control
+ * on the mixer. It does NOT control speaker volume. Should have own
+ * mask eventually?
+ */
+#define SGNXPRO_MIXER_DEVICES (SBPRO_MIXER_DEVICES|SOUND_MASK_BASS| \
+ SOUND_MASK_TREBLE|SOUND_MASK_SPEAKER )
+
#define SB16_RECORDING_DEVICES (SOUND_MASK_SYNTH | SOUND_MASK_LINE | SOUND_MASK_MIC | \
SOUND_MASK_CD)
@@ -62,6 +78,13 @@
#define IRQ_STAT 0x82
#define OPSW 0x3c
+/*
+ * Additional registers on the SG NX Pro
+ */
+#define COVOX_VOL 0x42
+#define TREBLE_LVL 0x44
+#define BASS_LVL 0x46
+
#define FREQ_HI (1 << 3)/* Use High-frequency ANFI filters */
#define FREQ_LOW 0 /* Use Low-frequency ANFI filters */
#define FILT_ON 0 /* Yes, 0 to turn it on, 1 for off */
@@ -108,6 +131,23 @@ MIX_ENT(SOUND_MIXER_ALTPCM, 0x00, 0, 0, 0x00, 0, 0),
MIX_ENT(SOUND_MIXER_RECLEV, 0x00, 0, 0, 0x00, 0, 0)
};
+#ifdef __SGNXPRO__
+mixer_tab sgnxpro_mix = {
+MIX_ENT(SOUND_MIXER_VOLUME, 0x22, 7, 4, 0x22, 3, 4),
+MIX_ENT(SOUND_MIXER_BASS, 0x46, 2, 3, 0x00, 0, 0),
+MIX_ENT(SOUND_MIXER_TREBLE, 0x44, 2, 3, 0x00, 0, 0),
+MIX_ENT(SOUND_MIXER_SYNTH, 0x26, 7, 4, 0x26, 3, 4),
+MIX_ENT(SOUND_MIXER_PCM, 0x04, 7, 4, 0x04, 3, 4),
+MIX_ENT(SOUND_MIXER_SPEAKER, 0x42, 2, 3, 0x00, 0, 0),
+MIX_ENT(SOUND_MIXER_LINE, 0x2e, 7, 4, 0x2e, 3, 4),
+MIX_ENT(SOUND_MIXER_MIC, 0x0a, 2, 3, 0x00, 0, 0),
+MIX_ENT(SOUND_MIXER_CD, 0x28, 7, 4, 0x28, 3, 4),
+MIX_ENT(SOUND_MIXER_IMIX, 0x00, 0, 0, 0x00, 0, 0),
+MIX_ENT(SOUND_MIXER_ALTPCM, 0x00, 0, 0, 0x00, 0, 0),
+MIX_ENT(SOUND_MIXER_RECLEV, 0x00, 0, 0, 0x00, 0, 0)
+};
+#endif
+
mixer_tab sb16_mix = {
MIX_ENT(SOUND_MIXER_VOLUME, 0x30, 7, 5, 0x31, 7, 5),
MIX_ENT(SOUND_MIXER_BASS, 0x46, 7, 4, 0x47, 7, 4),
diff --git a/sys/i386/isa/sound/sequencer.c b/sys/i386/isa/sound/sequencer.c
index e61e445..9885473 100644
--- a/sys/i386/isa/sound/sequencer.c
+++ b/sys/i386/isa/sound/sequencer.c
@@ -1,10 +1,10 @@
/*
* sound/sequencer.c
- *
+ *
* The sequencer personality manager.
- *
+ *
* Copyright by Hannu Savolainen 1993
- *
+ *
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions are
* met: 1. Redistributions of source code must retain the above copyright
@@ -12,7 +12,7 @@
* Redistributions in binary form must reproduce the above copyright notice,
* this list of conditions and the following disclaimer in the documentation
* and/or other materials provided with the distribution.
- *
+ *
* THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND ANY
* EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
* WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
@@ -24,7 +24,7 @@
* LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
* OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
* SUCH DAMAGE.
- *
+ *
*/
#define SEQUENCER_C
@@ -46,7 +46,7 @@ static int midi_opened[MAX_MIDI_DEV] =
static int midi_written[MAX_MIDI_DEV] =
{0};
-unsigned long seq_time = 0; /* Reference point for the timer */
+unsigned long seq_time = 0; /* Reference point for the timer */
#include "tuning.h"
@@ -87,6 +87,9 @@ sequencer_read (int dev, struct fileinfo *file, snd_rw_buf * buf, int count)
{
if (!iqlen)
{
+ if (c != count) /* Some data has been received */
+ return count - c; /* Return what we have */
+
DO_SLEEP (midi_sleeper, midi_sleep_flag, 0);
if (!iqlen)
@@ -327,6 +330,10 @@ extended_event (unsigned char *q)
synth_devs[dev]->controller (dev, q[3], q[4], *(short *) &q[5]);
break;
+ case SEQ_VOLMODE:
+ synth_devs[dev]->volume_method (dev, q[3]);
+ break;
+
default:
return RET_ERROR (EINVAL);
}
@@ -688,7 +695,7 @@ seq_reset (void)
{
midi_outc (i,
(unsigned char) (0xb0 + (chn & 0xff))); /* Channel msg */
- midi_outc (i, 0x7b); /* All notes off */
+ midi_outc (i, 0x7b);/* All notes off */
midi_outc (i, 0); /* Dummy parameter */
}
@@ -1049,7 +1056,7 @@ unsigned long
compute_finetune (unsigned long base_freq, int bend, int range)
{
unsigned long amount;
- int negative, semitones, cents;
+ int negative, semitones, cents, multiplier = 1;
if (!bend)
return base_freq;
@@ -1073,13 +1080,20 @@ compute_finetune (unsigned long base_freq, int bend, int range)
if (bend > range)
bend = range;
- if (bend > 2399)
- bend = 2399;
+ /*
+ if (bend > 2399)
+ bend = 2399;
+ */
+ while (bend > 2399)
+ {
+ multiplier *= 4;
+ bend -= 2400;
+ }
semitones = bend / 100;
cents = bend % 100;
- amount = semitone_tuning[semitones] * cent_tuning[cents] / 10000;
+ amount = semitone_tuning[semitones] * multiplier * cent_tuning[cents] / 10000;
if (negative)
return (base_freq * 10000) / amount; /* Bend down */
diff --git a/sys/i386/isa/sound/sound_calls.h b/sys/i386/isa/sound/sound_calls.h
index 97fb769..abc8200 100644
--- a/sys/i386/isa/sound/sound_calls.h
+++ b/sys/i386/isa/sound/sound_calls.h
@@ -129,12 +129,13 @@ int probe_sb16midi(struct address_info *hw_config);
/* From sb_midi.c */
void sb_midi_init(int model);
+void sb_midi_interrupt(int dummy);
/* From sb_mixer.c */
void sb_setmixer (unsigned int port, unsigned int value);
int sb_getmixer (unsigned int port);
void sb_mixer_set_stereo(int mode);
-void sb_mixer_init(int major_model);
+int sb_mixer_init(int major_model);
/* From opl3.c */
int opl3_detect (int ioaddr);
@@ -181,6 +182,7 @@ unsigned char gus_read8 (int reg);
void gus_write8(int reg, unsigned int data);
void guswave_dma_irq(void);
void gus_delay(void);
+int gus_default_mixer_ioctl (int dev, unsigned int cmd, unsigned int arg);
/* From gus_midi.c */
long gus_midi_init(long mem_start);
@@ -201,3 +203,6 @@ int pmgr_write (int dev, struct fileinfo *file, snd_rw_buf * buf, int count);
int pmgr_access(int dev, struct patmgr_info *rec);
int pmgr_inform(int dev, int event, unsigned long parm1, unsigned long parm2,
unsigned long parm3, unsigned long parm4);
+
+/* From ics2101.c */
+long ics2101_mixer_init(long mem_start);
diff --git a/sys/i386/isa/sound/sound_switch.c b/sys/i386/isa/sound/sound_switch.c
index 52120d5..68c7575 100644
--- a/sys/i386/isa/sound/sound_switch.c
+++ b/sys/i386/isa/sound/sound_switch.c
@@ -1,10 +1,10 @@
/*
* sound/sound_switch.c
- *
+ *
* The system call switch
- *
+ *
* Copyright by Hannu Savolainen 1993
- *
+ *
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions are
* met: 1. Redistributions of source code must retain the above copyright
@@ -12,7 +12,7 @@
* Redistributions in binary form must reproduce the above copyright notice,
* this list of conditions and the following disclaimer in the documentation
* and/or other materials provided with the distribution.
- *
+ *
* THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND ANY
* EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
* WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
@@ -24,7 +24,7 @@
* LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
* OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
* SUCH DAMAGE.
- *
+ *
*/
#include "sound_config.h"
@@ -441,4 +441,5 @@ sound_ioctl_sw (int dev, struct fileinfo *file,
return RET_ERROR (EPERM);
}
+
#endif
OpenPOWER on IntegriCloud