blob: a915a1affcde029637e10da78ab967e2661e6554 [file] [log] [blame]
#! /bin/sh
#
# Turtle Beach MultiSound Driver Notes
# -- Andrew Veliath <andrewtv@usa.net>
#
# Last update: September 10, 1998
# Corresponding msnd driver: 0.8.3
#
# ** This file is a README (top part) and shell archive (bottom part).
# The corresponding archived utility sources can be unpacked by
# running `sh MultiSound' (the utilities are only needed for the
# Pinnacle and Fiji cards). **
#
#
# -=-=- Getting Firmware -=-=-
# ~~~~~~~~~~~~~~~~~~~~~~~~~~~~
#
# See the section `Obtaining and Creating Firmware Files' in this
# document for instructions on obtaining the necessary firmware
# files.
#
#
# Supported Features
# ~~~~~~~~~~~~~~~~~~
#
# Currently, full-duplex digital audio (/dev/dsp only, /dev/audio is
# not currently available) and mixer functionality (/dev/mixer) are
# supported (memory mapped digital audio is not yet supported).
# Digital transfers and monitoring can be done as well if you have
# the digital daughterboard (see the section on using the S/PDIF port
# for more information).
#
# Support for the Turtle Beach MultiSound Hurricane architecture is
# composed of the following modules (these can also operate compiled
# into the kernel):
#
# snd-msnd-lib - MultiSound base (requires snd)
#
# snd-msnd-classic - Base audio/mixer support for Classic, Monetery and
# Tahiti cards
#
# snd-msnd-pinnacle - Base audio/mixer support for Pinnacle and Fiji cards
#
#
# Important Notes - Read Before Using
# ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
#
# The firmware files are not included (may change in future). You
# must obtain these images from Turtle Beach (they are included in
# the MultiSound Development Kits), and place them in /etc/sound for
# example, and give the full paths in the Linux configuration. If
# you are compiling in support for the MultiSound driver rather than
# using it as a module, these firmware files must be accessible
# during kernel compilation.
#
# Please note these files must be binary files, not assembler. See
# the section later in this document for instructions to obtain these
# files.
#
#
# Configuring Card Resources
# ~~~~~~~~~~~~~~~~~~~~~~~~~~
#
# ** This section is very important, as your card may not work at all
# or your machine may crash if you do not do this correctly. **
#
# * Classic/Monterey/Tahiti
#
# These cards are configured through the driver snd-msnd-classic. You must
# know the io port, then the driver will select the irq and memory resources
# on the card. It is up to you to know if these are free locations or now,
# a conflict can lock the machine up.
#
# * Pinnacle/Fiji
#
# The Pinnacle and Fiji cards have an extra config port, either
# 0x250, 0x260 or 0x270. This port can be disabled to have the card
# configured strictly through PnP, however you lose the ability to
# access the IDE controller and joystick devices on this card when
# using PnP. The included pinnaclecfg program in this shell archive
# can be used to configure the card in non-PnP mode, and in PnP mode
# you can use isapnptools. These are described briefly here.
#
# pinnaclecfg is not required; you can use the snd-msnd-pinnacle module
# to fully configure the card as well. However, pinnaclecfg can be
# used to change the resource values of a particular device after the
# snd-msnd-pinnacle module has been loaded. If you are compiling the
# driver into the kernel, you must set these values during compile
# time, however other peripheral resource values can be changed with
# the pinnaclecfg program after the kernel is loaded.
#
#
# *** PnP mode
#
# Use pnpdump to obtain a sample configuration if you can; I was able
# to obtain one with the command `pnpdump 1 0x203' -- this may vary
# for you (running pnpdump by itself did not work for me). Then,
# edit this file and use isapnp to uncomment and set the card values.
# Use these values when inserting the snd-msnd-pinnacle module. Using
# this method, you can set the resources for the DSP and the Kurzweil
# synth (Pinnacle). Since Linux does not directly support PnP
# devices, you may have difficulty when using the card in PnP mode
# when it the driver is compiled into the kernel. Using non-PnP mode
# is preferable in this case.
#
# Here is an example mypinnacle.conf for isapnp that sets the card to
# io base 0x210, irq 5 and mem 0xd8000, and also sets the Kurzweil
# synth to 0x330 and irq 9 (may need editing for your system):
#
# (READPORT 0x0203)
# (CSN 2)
# (IDENTIFY *)
#
# # DSP
# (CONFIGURE BVJ0440/-1 (LD 0
# (INT 0 (IRQ 5 (MODE +E))) (IO 0 (BASE 0x0210)) (MEM 0 (BASE 0x0d8000))
# (ACT Y)))
#
# # Kurzweil Synth (Pinnacle Only)
# (CONFIGURE BVJ0440/-1 (LD 1
# (IO 0 (BASE 0x0330)) (INT 0 (IRQ 9 (MODE +E)))
# (ACT Y)))
#
# (WAITFORKEY)
#
#
# *** Non-PnP mode
#
# The second way is by running the card in non-PnP mode. This
# actually has some advantages in that you can access some other
# devices on the card, such as the joystick and IDE controller. To
# configure the card, unpack this shell archive and build the
# pinnaclecfg program. Using this program, you can assign the
# resource values to the card's devices, or disable the devices. As
# an alternative to using pinnaclecfg, you can specify many of the
# configuration values when loading the snd-msnd-pinnacle module (or
# during kernel configuration when compiling the driver into the
# kernel).
#
# If you specify cfg=0x250 for the snd-msnd-pinnacle module, it
# automatically configure the card to the given io, irq and memory
# values using that config port (the config port is jumper selectable
# on the card to 0x250, 0x260 or 0x270).
#
# See the `snd-msnd-pinnacle Additional Options' section below for more
# information on these parameters (also, if you compile the driver
# directly into the kernel, these extra parameters can be useful
# here).
#
#
# ** It is very easy to cause problems in your machine if you choose a
# resource value which is incorrect. **
#
#
# Examples
# ~~~~~~~~
#
# * MultiSound Classic/Monterey/Tahiti:
#
# modprobe snd
# insmod snd-msnd-lib
# insmod snd-msnd-classic io=0x290 irq=7 mem=0xd0000
#
# * MultiSound Pinnacle in PnP mode:
#
# modprobe snd
# insmod snd-msnd-lib
# isapnp mypinnacle.conf
# insmod snd-msnd-pinnacle io=0x210 irq=5 mem=0xd8000 <-- match mypinnacle.conf values
#
# * MultiSound Pinnacle in non-PnP mode (replace 0x250 with your configuration port,
# one of 0x250, 0x260 or 0x270):
#
# modprobe snd
# insmod snd-msnd-lib
# insmod snd-msnd-pinnacle cfg=0x250 io=0x290 irq=5 mem=0xd0000
#
# * To use the MPU-compatible Kurzweil synth on the Pinnacle in PnP
# mode, add the following (assumes you did `isapnp mypinnacle.conf'):
#
# insmod snd
# insmod mpu401 io=0x330 irq=9 <-- match mypinnacle.conf values
#
# * To use the MPU-compatible Kurzweil synth on the Pinnacle in non-PnP
# mode, add the following. Note how we first configure the peripheral's
# resources, _then_ install a Linux driver for it:
#
# insmod snd
# pinnaclecfg 0x250 mpu 0x330 9
# insmod mpu401 io=0x330 irq=9
#
# -- OR you can use the following sequence without pinnaclecfg in non-PnP mode:
#
# modprobe snd
# insmod snd-msnd-lib
# insmod snd-msnd-pinnacle cfg=0x250 io=0x290 irq=5 mem=0xd0000 mpu_io=0x330 mpu_irq=9
# insmod snd
# insmod mpu401 io=0x330 irq=9
#
# * To setup the joystick port on the Pinnacle in non-PnP mode (though
# you have to find the actual Linux joystick driver elsewhere), you
# can use pinnaclecfg:
#
# pinnaclecfg 0x250 joystick 0x200
#
# -- OR you can configure this using snd-msnd-pinnacle with the following:
#
# modprobe snd
# insmod snd-msnd-lib
# insmod snd-msnd-pinnacle cfg=0x250 io=0x290 irq=5 mem=0xd0000 joystick_io=0x200
#
#
# snd-msnd-classic, snd-msnd-pinnacle Required Options
# ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
#
# If the following options are not given, the module will not load.
# Examine the kernel message log for informative error messages.
# WARNING--probing isn't supported so try to make sure you have the
# correct shared memory area, otherwise you may experience problems.
#
# io I/O base of DSP, e.g. io=0x210
# irq IRQ number, e.g. irq=5
# mem Shared memory area, e.g. mem=0xd8000
#
#
# snd-msnd-classic, snd-msnd-pinnacle Additional Options
# ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
#
# fifosize The digital audio FIFOs, in kilobytes. If not
# specified, the default will be used. Increasing
# this value will reduce the chance of a FIFO
# underflow at the expense of increasing overall
# latency. For example, fifosize=512 will
# allocate 512kB read and write FIFOs (1MB total).
# While this may reduce dropouts, a heavy machine
# load will undoubtedly starve the FIFO of data
# and you will eventually get dropouts. One
# option is to alter the scheduling priority of
# the playback process, using `nice' or some form
# of POSIX soft real-time scheduling.
#
# calibrate_signal Setting this to one calibrates the ADCs to the
# signal, zero calibrates to the card (defaults
# to zero).
#
#
# snd-msnd-pinnacle Additional Options
# ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
#
# digital Specify digital=1 to enable the S/PDIF input
# if you have the digital daughterboard
# adapter. This will enable access to the
# DIGITAL1 input for the soundcard in the mixer.
# Some mixer programs might have trouble setting
# the DIGITAL1 source as an input. If you have
# trouble, you can try the setdigital.c program
# at the bottom of this document.
#
# cfg Non-PnP configuration port for the Pinnacle
# and Fiji (typically 0x250, 0x260 or 0x270,
# depending on the jumper configuration). If
# this option is omitted, then it is assumed
# that the card is in PnP mode, and that the
# specified DSP resource values are already
# configured with PnP (i.e. it won't attempt to
# do any sort of configuration).
#
# When the Pinnacle is in non-PnP mode, you can use the following
# options to configure particular devices. If a full specification
# for a device is not given, then the device is not configured. Note
# that you still must use a Linux driver for any of these devices
# once their resources are setup (such as the Linux joystick driver,
# or the MPU401 driver from OSS for the Kurzweil synth).
#
# mpu_io I/O port of MPU (on-board Kurzweil synth)
# mpu_irq IRQ of MPU (on-board Kurzweil synth)
# ide_io0 First I/O port of IDE controller
# ide_io1 Second I/O port of IDE controller
# ide_irq IRQ IDE controller
# joystick_io I/O port of joystick
#
#
# Obtaining and Creating Firmware Files
# ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
#
# For the Classic/Tahiti/Monterey
# ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
#
# Download to /tmp and unzip the following file from Turtle Beach:
#
# ftp://ftp.voyetra.com/pub/tbs/msndcl/msndvkit.zip
#
# When unzipped, unzip the file named MsndFiles.zip. Then copy the
# following firmware files to /etc/sound (note the file renaming):
#
# cp DSPCODE/MSNDINIT.BIN /etc/sound/msndinit.bin
# cp DSPCODE/MSNDPERM.REB /etc/sound/msndperm.bin
#
# When configuring the Linux kernel, specify /etc/sound/msndinit.bin and
# /etc/sound/msndperm.bin for the two firmware files (Linux kernel
# versions older than 2.2 do not ask for firmware paths, and are
# hardcoded to /etc/sound).
#
# If you are compiling the driver into the kernel, these files must
# be accessible during compilation, but will not be needed later.
# The files must remain, however, if the driver is used as a module.
#
#
# For the Pinnacle/Fiji
# ~~~~~~~~~~~~~~~~~~~~~
#
# Download to /tmp and unzip the following file from Turtle Beach (be
# sure to use the entire URL; some have had trouble navigating to the
# URL):
#
# ftp://ftp.voyetra.com/pub/tbs/pinn/pnddk100.zip
#
# Unpack this shell archive, and run make in the created directory
# (you need a C compiler and flex to build the utilities). This
# should give you the executables conv, pinnaclecfg and setdigital.
# conv is only used temporarily here to create the firmware files,
# while pinnaclecfg is used to configure the Pinnacle or Fiji card in
# non-PnP mode, and setdigital can be used to set the S/PDIF input on
# the mixer (pinnaclecfg and setdigital should be copied to a
# convenient place, possibly run during system initialization).
#
# To generating the firmware files with the `conv' program, we create
# the binary firmware files by doing the following conversion
# (assuming the archive unpacked into a directory named PINNDDK):
#
# ./conv < PINNDDK/dspcode/pndspini.asm > /etc/sound/pndspini.bin
# ./conv < PINNDDK/dspcode/pndsperm.asm > /etc/sound/pndsperm.bin
#
# The conv (and conv.l) program is not needed after conversion and can
# be safely deleted. Then, when configuring the Linux kernel, specify
# /etc/sound/pndspini.bin and /etc/sound/pndsperm.bin for the two
# firmware files (Linux kernel versions older than 2.2 do not ask for
# firmware paths, and are hardcoded to /etc/sound).
#
# If you are compiling the driver into the kernel, these files must
# be accessible during compilation, but will not be needed later.
# The files must remain, however, if the driver is used as a module.
#
#
# Using Digital I/O with the S/PDIF Port
# ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
#
# If you have a Pinnacle or Fiji with the digital daughterboard and
# want to set it as the input source, you can use this program if you
# have trouble trying to do it with a mixer program (be sure to
# insert the module with the digital=1 option, or say Y to the option
# during compiled-in kernel operation). Upon selection of the S/PDIF
# port, you should be able monitor and record from it.
#
# There is something to note about using the S/PDIF port. Digital
# timing is taken from the digital signal, so if a signal is not
# connected to the port and it is selected as recording input, you
# will find PCM playback to be distorted in playback rate. Also,
# attempting to record at a sampling rate other than the DAT rate may
# be problematic (i.e. trying to record at 8000Hz when the DAT signal
# is 44100Hz). If you have a problem with this, set the recording
# input to analog if you need to record at a rate other than that of
# the DAT rate.
#
#
# -- Shell archive attached below, just run `sh MultiSound' to extract.
# Contains Pinnacle/Fiji utilities to convert firmware, configure
# in non-PnP mode, and select the DIGITAL1 input for the mixer.
#
#
#!/bin/sh
# This is a shell archive (produced by GNU sharutils 4.2).
# To extract the files from this archive, save it to some FILE, remove
# everything before the `!/bin/sh' line above, then type `sh FILE'.
#
# Made on 1998-12-04 10:07 EST by <andrewtv@ztransform.velsoft.com>.
# Source directory was `/home/andrewtv/programming/pinnacle/pinnacle'.
#
# Existing files will *not* be overwritten unless `-c' is specified.
#
# This shar contains:
# length mode name
# ------ ---------- ------------------------------------------
# 2064 -rw-rw-r-- MultiSound.d/setdigital.c
# 10224 -rw-rw-r-- MultiSound.d/pinnaclecfg.c
# 106 -rw-rw-r-- MultiSound.d/Makefile
# 146 -rw-rw-r-- MultiSound.d/conv.l
# 1491 -rw-rw-r-- MultiSound.d/msndreset.c
#
save_IFS="${IFS}"
IFS="${IFS}:"
gettext_dir=FAILED
locale_dir=FAILED
first_param="$1"
for dir in $PATH
do
if test "$gettext_dir" = FAILED && test -f $dir/gettext \
&& ($dir/gettext --version >/dev/null 2>&1)
then
set `$dir/gettext --version 2>&1`
if test "$3" = GNU
then
gettext_dir=$dir
fi
fi
if test "$locale_dir" = FAILED && test -f $dir/shar \
&& ($dir/shar --print-text-domain-dir >/dev/null 2>&1)
then
locale_dir=`$dir/shar --print-text-domain-dir`
fi
done
IFS="$save_IFS"
if test "$locale_dir" = FAILED || test "$gettext_dir" = FAILED
then
echo=echo
else
TEXTDOMAINDIR=$locale_dir
export TEXTDOMAINDIR
TEXTDOMAIN=sharutils
export TEXTDOMAIN
echo="$gettext_dir/gettext -s"
fi
touch -am 1231235999 $$.touch >/dev/null 2>&1
if test ! -f 1231235999 && test -f $$.touch; then
shar_touch=touch
else
shar_touch=:
echo
$echo 'WARNING: not restoring timestamps. Consider getting and'
$echo "installing GNU \`touch', distributed in GNU File Utilities..."
echo
fi
rm -f 1231235999 $$.touch
#
if mkdir _sh01426; then
$echo 'x -' 'creating lock directory'
else
$echo 'failed to create lock directory'
exit 1
fi
# ============= MultiSound.d/setdigital.c ==============
if test ! -d 'MultiSound.d'; then
$echo 'x -' 'creating directory' 'MultiSound.d'
mkdir 'MultiSound.d'
fi
if test -f 'MultiSound.d/setdigital.c' && test "$first_param" != -c; then
$echo 'x -' SKIPPING 'MultiSound.d/setdigital.c' '(file already exists)'
else
$echo 'x -' extracting 'MultiSound.d/setdigital.c' '(text)'
sed 's/^X//' << 'SHAR_EOF' > 'MultiSound.d/setdigital.c' &&
/*********************************************************************
X *
X * setdigital.c - sets the DIGITAL1 input for a mixer
X *
X * Copyright (C) 1998 Andrew Veliath
X *
X * This program is free software; you can redistribute it and/or modify
X * it under the terms of the GNU General Public License as published by
X * the Free Software Foundation; either version 2 of the License, or
X * (at your option) any later version.
X *
X * This program is distributed in the hope that it will be useful,
X * but WITHOUT ANY WARRANTY; without even the implied warranty of
X * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
X * GNU General Public License for more details.
X *
X * You should have received a copy of the GNU General Public License
X * along with this program; if not, write to the Free Software
X * Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
X *
X ********************************************************************/
X
#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <fcntl.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <sys/ioctl.h>
#include <sys/soundcard.h>
X
int main(int argc, char *argv[])
{
X int fd;
X unsigned long recmask, recsrc;
X
X if (argc != 2) {
X fprintf(stderr, "usage: setdigital <mixer device>\n");
X exit(1);
X }
X
X if ((fd = open(argv[1], O_RDWR)) < 0) {
X perror(argv[1]);
X exit(1);
X }
X
X if (ioctl(fd, SOUND_MIXER_READ_RECMASK, &recmask) < 0) {
X fprintf(stderr, "error: ioctl read recording mask failed\n");
X perror("ioctl");
X close(fd);
X exit(1);
X }
X
X if (!(recmask & SOUND_MASK_DIGITAL1)) {
X fprintf(stderr, "error: cannot find DIGITAL1 device in mixer\n");
X close(fd);
X exit(1);
X }
X
X if (ioctl(fd, SOUND_MIXER_READ_RECSRC, &recsrc) < 0) {
X fprintf(stderr, "error: ioctl read recording source failed\n");
X perror("ioctl");
X close(fd);
X exit(1);
X }
X
X recsrc |= SOUND_MASK_DIGITAL1;
X
X if (ioctl(fd, SOUND_MIXER_WRITE_RECSRC, &recsrc) < 0) {
X fprintf(stderr, "error: ioctl write recording source failed\n");
X perror("ioctl");
X close(fd);
X exit(1);
X }
X
X close(fd);
X
X return 0;
}
SHAR_EOF
$shar_touch -am 1204092598 'MultiSound.d/setdigital.c' &&
chmod 0664 'MultiSound.d/setdigital.c' ||
$echo 'restore of' 'MultiSound.d/setdigital.c' 'failed'
if ( md5sum --help 2>&1 | grep 'sage: md5sum \[' ) >/dev/null 2>&1 \
&& ( md5sum --version 2>&1 | grep -v 'textutils 1.12' ) >/dev/null; then
md5sum -c << SHAR_EOF >/dev/null 2>&1 \
|| $echo 'MultiSound.d/setdigital.c:' 'MD5 check failed'
e87217fc3e71288102ba41fd81f71ec4 MultiSound.d/setdigital.c
SHAR_EOF
else
shar_count="`LC_ALL= LC_CTYPE= LANG= wc -c < 'MultiSound.d/setdigital.c'`"
test 2064 -eq "$shar_count" ||
$echo 'MultiSound.d/setdigital.c:' 'original size' '2064,' 'current size' "$shar_count!"
fi
fi
# ============= MultiSound.d/pinnaclecfg.c ==============
if test -f 'MultiSound.d/pinnaclecfg.c' && test "$first_param" != -c; then
$echo 'x -' SKIPPING 'MultiSound.d/pinnaclecfg.c' '(file already exists)'
else
$echo 'x -' extracting 'MultiSound.d/pinnaclecfg.c' '(text)'
sed 's/^X//' << 'SHAR_EOF' > 'MultiSound.d/pinnaclecfg.c' &&
/*********************************************************************
X *
X * pinnaclecfg.c - Pinnacle/Fiji Device Configuration Program
X *
X * This is for NON-PnP mode only. For PnP mode, use isapnptools.
X *
X * This is Linux-specific, and must be run with root permissions.
X *
X * Part of the Turtle Beach MultiSound Sound Card Driver for Linux
X *
X * Copyright (C) 1998 Andrew Veliath
X *
X * This program is free software; you can redistribute it and/or modify
X * it under the terms of the GNU General Public License as published by
X * the Free Software Foundation; either version 2 of the License, or
X * (at your option) any later version.
X *
X * This program is distributed in the hope that it will be useful,
X * but WITHOUT ANY WARRANTY; without even the implied warranty of
X * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
X * GNU General Public License for more details.
X *
X * You should have received a copy of the GNU General Public License
X * along with this program; if not, write to the Free Software
X * Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
X *
X ********************************************************************/
X
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <errno.h>
#include <unistd.h>
#include <asm/types.h>
#include <sys/io.h>
X
#define IREG_LOGDEVICE 0x07
#define IREG_ACTIVATE 0x30
#define LD_ACTIVATE 0x01
#define LD_DISACTIVATE 0x00
#define IREG_EECONTROL 0x3F
#define IREG_MEMBASEHI 0x40
#define IREG_MEMBASELO 0x41
#define IREG_MEMCONTROL 0x42
#define IREG_MEMRANGEHI 0x43
#define IREG_MEMRANGELO 0x44
#define MEMTYPE_8BIT 0x00
#define MEMTYPE_16BIT 0x02
#define MEMTYPE_RANGE 0x00
#define MEMTYPE_HIADDR 0x01
#define IREG_IO0_BASEHI 0x60
#define IREG_IO0_BASELO 0x61
#define IREG_IO1_BASEHI 0x62
#define IREG_IO1_BASELO 0x63
#define IREG_IRQ_NUMBER 0x70
#define IREG_IRQ_TYPE 0x71
#define IRQTYPE_HIGH 0x02
#define IRQTYPE_LOW 0x00
#define IRQTYPE_LEVEL 0x01
#define IRQTYPE_EDGE 0x00
X
#define HIBYTE(w) ((BYTE)(((WORD)(w) >> 8) & 0xFF))
#define LOBYTE(w) ((BYTE)(w))
#define MAKEWORD(low,hi) ((WORD)(((BYTE)(low))|(((WORD)((BYTE)(hi)))<<8)))
X
typedef __u8 BYTE;
typedef __u16 USHORT;
typedef __u16 WORD;
X
static int config_port = -1;
X
static int msnd_write_cfg(int cfg, int reg, int value)
{
X outb(reg, cfg);
X outb(value, cfg + 1);
X if (value != inb(cfg + 1)) {
X fprintf(stderr, "error: msnd_write_cfg: I/O error\n");
X return -EIO;
X }
X return 0;
}
X
static int msnd_read_cfg(int cfg, int reg)
{
X outb(reg, cfg);
X return inb(cfg + 1);
}
X
static int msnd_write_cfg_io0(int cfg, int num, WORD io)
{
X if (msnd_write_cfg(cfg, IREG_LOGDEVICE, num))
X return -EIO;
X if (msnd_write_cfg(cfg, IREG_IO0_BASEHI, HIBYTE(io)))
X return -EIO;
X if (msnd_write_cfg(cfg, IREG_IO0_BASELO, LOBYTE(io)))
X return -EIO;
X return 0;
}
X
static int msnd_read_cfg_io0(int cfg, int num, WORD *io)
{
X if (msnd_write_cfg(cfg, IREG_LOGDEVICE, num))
X return -EIO;
X
X *io = MAKEWORD(msnd_read_cfg(cfg, IREG_IO0_BASELO),
X msnd_read_cfg(cfg, IREG_IO0_BASEHI));
X
X return 0;
}
X
static int msnd_write_cfg_io1(int cfg, int num, WORD io)
{
X if (msnd_write_cfg(cfg, IREG_LOGDEVICE, num))
X return -EIO;
X if (msnd_write_cfg(cfg, IREG_IO1_BASEHI, HIBYTE(io)))
X return -EIO;
X if (msnd_write_cfg(cfg, IREG_IO1_BASELO, LOBYTE(io)))
X return -EIO;
X return 0;
}
X
static int msnd_read_cfg_io1(int cfg, int num, WORD *io)
{
X if (msnd_write_cfg(cfg, IREG_LOGDEVICE, num))
X return -EIO;
X
X *io = MAKEWORD(msnd_read_cfg(cfg, IREG_IO1_BASELO),
X msnd_read_cfg(cfg, IREG_IO1_BASEHI));
X
X return 0;
}
X
static int msnd_write_cfg_irq(int cfg, int num, WORD irq)
{
X if (msnd_write_cfg(cfg, IREG_LOGDEVICE, num))
X return -EIO;
X if (msnd_write_cfg(cfg, IREG_IRQ_NUMBER, LOBYTE(irq)))
X return -EIO;
X if (msnd_write_cfg(cfg, IREG_IRQ_TYPE, IRQTYPE_EDGE))
X return -EIO;
X return 0;
}
X
static int msnd_read_cfg_irq(int cfg, int num, WORD *irq)
{
X if (msnd_write_cfg(cfg, IREG_LOGDEVICE, num))
X return -EIO;
X
X *irq = msnd_read_cfg(cfg, IREG_IRQ_NUMBER);
X
X return 0;
}
X
static int msnd_write_cfg_mem(int cfg, int num, int mem)
{
X WORD wmem;
X
X mem >>= 8;
X mem &= 0xfff;
X wmem = (WORD)mem;
X if (msnd_write_cfg(cfg, IREG_LOGDEVICE, num))
X return -EIO;
X if (msnd_write_cfg(cfg, IREG_MEMBASEHI, HIBYTE(wmem)))
X return -EIO;
X if (msnd_write_cfg(cfg, IREG_MEMBASELO, LOBYTE(wmem)))
X return -EIO;
X if (wmem && msnd_write_cfg(cfg, IREG_MEMCONTROL, (MEMTYPE_HIADDR | MEMTYPE_16BIT)))
X return -EIO;
X return 0;
}
X
static int msnd_read_cfg_mem(int cfg, int num, int *mem)
{
X if (msnd_write_cfg(cfg, IREG_LOGDEVICE, num))
X return -EIO;
X
X *mem = MAKEWORD(msnd_read_cfg(cfg, IREG_MEMBASELO),
X msnd_read_cfg(cfg, IREG_MEMBASEHI));
X *mem <<= 8;
X
X return 0;
}
X
static int msnd_activate_logical(int cfg, int num)
{
X if (msnd_write_cfg(cfg, IREG_LOGDEVICE, num))
X return -EIO;
X if (msnd_write_cfg(cfg, IREG_ACTIVATE, LD_ACTIVATE))
X return -EIO;
X return 0;
}
X
static int msnd_write_cfg_logical(int cfg, int num, WORD io0, WORD io1, WORD irq, int mem)
{
X if (msnd_write_cfg(cfg, IREG_LOGDEVICE, num))
X return -EIO;
X if (msnd_write_cfg_io0(cfg, num, io0))
X return -EIO;
X if (msnd_write_cfg_io1(cfg, num, io1))
X return -EIO;
X if (msnd_write_cfg_irq(cfg, num, irq))
X return -EIO;
X if (msnd_write_cfg_mem(cfg, num, mem))
X return -EIO;
X if (msnd_activate_logical(cfg, num))
X return -EIO;
X return 0;
}
X
static int msnd_read_cfg_logical(int cfg, int num, WORD *io0, WORD *io1, WORD *irq, int *mem)
{
X if (msnd_write_cfg(cfg, IREG_LOGDEVICE, num))
X return -EIO;
X if (msnd_read_cfg_io0(cfg, num, io0))
X return -EIO;
X if (msnd_read_cfg_io1(cfg, num, io1))
X return -EIO;
X if (msnd_read_cfg_irq(cfg, num, irq))
X return -EIO;
X if (msnd_read_cfg_mem(cfg, num, mem))
X return -EIO;
X return 0;
}
X
static void usage(void)
{
X fprintf(stderr,
X "\n"
X "pinnaclecfg 1.0\n"
X "\n"
X "usage: pinnaclecfg <config port> [device config]\n"
X "\n"
X "This is for use with the card in NON-PnP mode only.\n"
X "\n"
X "Available devices (not all available for Fiji):\n"
X "\n"
X " Device Description\n"
X " -------------------------------------------------------------------\n"
X " reset Reset all devices (i.e. disable)\n"
X " show Display current device configurations\n"
X "\n"
X " dsp <io> <irq> <mem> Audio device\n"
X " mpu <io> <irq> Internal Kurzweil synth\n"
X " ide <io0> <io1> <irq> On-board IDE controller\n"
X " joystick <io> Joystick port\n"
X "\n");
X exit(1);
}
X
static int cfg_reset(void)
{
X int i;
X
X for (i = 0; i < 4; ++i)
X msnd_write_cfg_logical(config_port, i, 0, 0, 0, 0);
X
X return 0;
}
X
static int cfg_show(void)
{
X int i;
X int count = 0;
X
X for (i = 0; i < 4; ++i) {
X WORD io0, io1, irq;
X int mem;
X msnd_read_cfg_logical(config_port, i, &io0, &io1, &irq, &mem);
X switch (i) {
X case 0:
X if (io0 || irq || mem) {
X printf("dsp 0x%x %d 0x%x\n", io0, irq, mem);
X ++count;
X }
X break;
X case 1:
X if (io0 || irq) {
X printf("mpu 0x%x %d\n", io0, irq);
X ++count;
X }
X break;
X case 2:
X if (io0 || io1 || irq) {
X printf("ide 0x%x 0x%x %d\n", io0, io1, irq);
X ++count;
X }
X break;
X case 3:
X if (io0) {
X printf("joystick 0x%x\n", io0);
X ++count;
X }
X break;
X }
X }
X
X if (count == 0)
X fprintf(stderr, "no devices configured\n");
X
X return 0;
}
X
static int cfg_dsp(int argc, char *argv[])
{
X int io, irq, mem;
X
X if (argc < 3 ||
X sscanf(argv[0], "0x%x", &io) != 1 ||
X sscanf(argv[1], "%d", &irq) != 1 ||
X sscanf(argv[2], "0x%x", &mem) != 1)
X usage();
X
X if (!(io == 0x290 ||
X io == 0x260 ||
X io == 0x250 ||
X io == 0x240 ||
X io == 0x230 ||
X io == 0x220 ||
X io == 0x210 ||
X io == 0x3e0)) {
X fprintf(stderr, "error: io must be one of "
X "210, 220, 230, 240, 250, 260, 290, or 3E0\n");
X usage();
X }
X
X if (!(irq == 5 ||
X irq == 7 ||
X irq == 9 ||
X irq == 10 ||
X irq == 11 ||
X irq == 12)) {
X fprintf(stderr, "error: irq must be one of "
X "5, 7, 9, 10, 11 or 12\n");
X usage();
X }
X
X if (!(mem == 0xb0000 ||
X mem == 0xc8000 ||
X mem == 0xd0000 ||
X mem == 0xd8000 ||
X mem == 0xe0000 ||
X mem == 0xe8000)) {
X fprintf(stderr, "error: mem must be one of "
X "0xb0000, 0xc8000, 0xd0000, 0xd8000, 0xe0000 or 0xe8000\n");
X usage();
X }
X
X return msnd_write_cfg_logical(config_port, 0, io, 0, irq, mem);
}
X
static int cfg_mpu(int argc, char *argv[])
{
X int io, irq;
X
X if (argc < 2 ||
X sscanf(argv[0], "0x%x", &io) != 1 ||
X sscanf(argv[1], "%d", &irq) != 1)
X usage();
X
X return msnd_write_cfg_logical(config_port, 1, io, 0, irq, 0);
}
X
static int cfg_ide(int argc, char *argv[])
{
X int io0, io1, irq;
X
X if (argc < 3 ||
X sscanf(argv[0], "0x%x", &io0) != 1 ||
X sscanf(argv[0], "0x%x", &io1) != 1 ||
X sscanf(argv[1], "%d", &irq) != 1)
X usage();
X
X return msnd_write_cfg_logical(config_port, 2, io0, io1, irq, 0);
}
X
static int cfg_joystick(int argc, char *argv[])
{
X int io;
X
X if (argc < 1 ||
X sscanf(argv[0], "0x%x", &io) != 1)
X usage();
X
X return msnd_write_cfg_logical(config_port, 3, io, 0, 0, 0);
}
X
int main(int argc, char *argv[])
{
X char *device;
X int rv = 0;
X
X --argc; ++argv;
X
X if (argc < 2)
X usage();
X
X sscanf(argv[0], "0x%x", &config_port);
X if (config_port != 0x250 && config_port != 0x260 && config_port != 0x270) {
X fprintf(stderr, "error: <config port> must be 0x250, 0x260 or 0x270\n");
X exit(1);
X }
X if (ioperm(config_port, 2, 1)) {
X perror("ioperm");
X fprintf(stderr, "note: pinnaclecfg must be run as root\n");
X exit(1);
X }
X device = argv[1];
X
X argc -= 2; argv += 2;
X
X if (strcmp(device, "reset") == 0)
X rv = cfg_reset();
X else if (strcmp(device, "show") == 0)
X rv = cfg_show();
X else if (strcmp(device, "dsp") == 0)
X rv = cfg_dsp(argc, argv);
X else if (strcmp(device, "mpu") == 0)
X rv = cfg_mpu(argc, argv);
X else if (strcmp(device, "ide") == 0)
X rv = cfg_ide(argc, argv);
X else if (strcmp(device, "joystick") == 0)
X rv = cfg_joystick(argc, argv);
X else {
X fprintf(stderr, "error: unknown device %s\n", device);
X usage();
X }
X
X if (rv)
X fprintf(stderr, "error: device configuration failed\n");
X
X return 0;
}
SHAR_EOF
$shar_touch -am 1204092598 'MultiSound.d/pinnaclecfg.c' &&
chmod 0664 'MultiSound.d/pinnaclecfg.c' ||
$echo 'restore of' 'MultiSound.d/pinnaclecfg.c' 'failed'
if ( md5sum --help 2>&1 | grep 'sage: md5sum \[' ) >/dev/null 2>&1 \
&& ( md5sum --version 2>&1 | grep -v 'textutils 1.12' ) >/dev/null; then
md5sum -c << SHAR_EOF >/dev/null 2>&1 \
|| $echo 'MultiSound.d/pinnaclecfg.c:' 'MD5 check failed'
366bdf27f0db767a3c7921d0a6db20fe MultiSound.d/pinnaclecfg.c
SHAR_EOF
else
shar_count="`LC_ALL= LC_CTYPE= LANG= wc -c < 'MultiSound.d/pinnaclecfg.c'`"
test 10224 -eq "$shar_count" ||
$echo 'MultiSound.d/pinnaclecfg.c:' 'original size' '10224,' 'current size' "$shar_count!"
fi
fi
# ============= MultiSound.d/Makefile ==============
if test -f 'MultiSound.d/Makefile' && test "$first_param" != -c; then
$echo 'x -' SKIPPING 'MultiSound.d/Makefile' '(file already exists)'
else
$echo 'x -' extracting 'MultiSound.d/Makefile' '(text)'
sed 's/^X//' << 'SHAR_EOF' > 'MultiSound.d/Makefile' &&
CC = gcc
CFLAGS = -O
PROGS = setdigital msndreset pinnaclecfg conv
X
all: $(PROGS)
X
clean:
X rm -f $(PROGS)
SHAR_EOF
$shar_touch -am 1204092398 'MultiSound.d/Makefile' &&
chmod 0664 'MultiSound.d/Makefile' ||
$echo 'restore of' 'MultiSound.d/Makefile' 'failed'
if ( md5sum --help 2>&1 | grep 'sage: md5sum \[' ) >/dev/null 2>&1 \
&& ( md5sum --version 2>&1 | grep -v 'textutils 1.12' ) >/dev/null; then
md5sum -c << SHAR_EOF >/dev/null 2>&1 \
|| $echo 'MultiSound.d/Makefile:' 'MD5 check failed'
76ca8bb44e3882edcf79c97df6c81845 MultiSound.d/Makefile
SHAR_EOF
else
shar_count="`LC_ALL= LC_CTYPE= LANG= wc -c < 'MultiSound.d/Makefile'`"
test 106 -eq "$shar_count" ||
$echo 'MultiSound.d/Makefile:' 'original size' '106,' 'current size' "$shar_count!"
fi
fi
# ============= MultiSound.d/conv.l ==============
if test -f 'MultiSound.d/conv.l' && test "$first_param" != -c; then
$echo 'x -' SKIPPING 'MultiSound.d/conv.l' '(file already exists)'
else
$echo 'x -' extracting 'MultiSound.d/conv.l' '(text)'
sed 's/^X//' << 'SHAR_EOF' > 'MultiSound.d/conv.l' &&
%%
[ \n\t,\r]
\;.*
DB
[0-9A-Fa-f]+H { int n; sscanf(yytext, "%xH", &n); printf("%c", n); }
%%
int yywrap() { return 1; }
void main() { yylex(); }
SHAR_EOF
$shar_touch -am 0828231798 'MultiSound.d/conv.l' &&
chmod 0664 'MultiSound.d/conv.l' ||
$echo 'restore of' 'MultiSound.d/conv.l' 'failed'
if ( md5sum --help 2>&1 | grep 'sage: md5sum \[' ) >/dev/null 2>&1 \
&& ( md5sum --version 2>&1 | grep -v 'textutils 1.12' ) >/dev/null; then
md5sum -c << SHAR_EOF >/dev/null 2>&1 \
|| $echo 'MultiSound.d/conv.l:' 'MD5 check failed'
d2411fc32cd71a00dcdc1f009e858dd2 MultiSound.d/conv.l
SHAR_EOF
else
shar_count="`LC_ALL= LC_CTYPE= LANG= wc -c < 'MultiSound.d/conv.l'`"
test 146 -eq "$shar_count" ||
$echo 'MultiSound.d/conv.l:' 'original size' '146,' 'current size' "$shar_count!"
fi
fi
# ============= MultiSound.d/msndreset.c ==============
if test -f 'MultiSound.d/msndreset.c' && test "$first_param" != -c; then
$echo 'x -' SKIPPING 'MultiSound.d/msndreset.c' '(file already exists)'
else
$echo 'x -' extracting 'MultiSound.d/msndreset.c' '(text)'
sed 's/^X//' << 'SHAR_EOF' > 'MultiSound.d/msndreset.c' &&
/*********************************************************************
X *
X * msndreset.c - resets the MultiSound card
X *
X * Copyright (C) 1998 Andrew Veliath
X *
X * This program is free software; you can redistribute it and/or modify
X * it under the terms of the GNU General Public License as published by
X * the Free Software Foundation; either version 2 of the License, or
X * (at your option) any later version.
X *
X * This program is distributed in the hope that it will be useful,
X * but WITHOUT ANY WARRANTY; without even the implied warranty of
X * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
X * GNU General Public License for more details.
X *
X * You should have received a copy of the GNU General Public License
X * along with this program; if not, write to the Free Software
X * Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
X *
X ********************************************************************/
X
#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <fcntl.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <sys/ioctl.h>
#include <sys/soundcard.h>
X
int main(int argc, char *argv[])
{
X int fd;
X
X if (argc != 2) {
X fprintf(stderr, "usage: msndreset <mixer device>\n");
X exit(1);
X }
X
X if ((fd = open(argv[1], O_RDWR)) < 0) {
X perror(argv[1]);
X exit(1);
X }
X
X if (ioctl(fd, SOUND_MIXER_PRIVATE1, 0) < 0) {
X fprintf(stderr, "error: msnd ioctl reset failed\n");
X perror("ioctl");
X close(fd);
X exit(1);
X }
X
X close(fd);
X
X return 0;
}
SHAR_EOF
$shar_touch -am 1204100698 'MultiSound.d/msndreset.c' &&
chmod 0664 'MultiSound.d/msndreset.c' ||
$echo 'restore of' 'MultiSound.d/msndreset.c' 'failed'
if ( md5sum --help 2>&1 | grep 'sage: md5sum \[' ) >/dev/null 2>&1 \
&& ( md5sum --version 2>&1 | grep -v 'textutils 1.12' ) >/dev/null; then
md5sum -c << SHAR_EOF >/dev/null 2>&1 \
|| $echo 'MultiSound.d/msndreset.c:' 'MD5 check failed'
c52f876521084e8eb25e12e01dcccb8a MultiSound.d/msndreset.c
SHAR_EOF
else
shar_count="`LC_ALL= LC_CTYPE= LANG= wc -c < 'MultiSound.d/msndreset.c'`"
test 1491 -eq "$shar_count" ||
$echo 'MultiSound.d/msndreset.c:' 'original size' '1491,' 'current size' "$shar_count!"
fi
fi
rm -fr _sh01426
exit 0