Continued sound documentation update (still not complete).

This commit is contained in:
Volker Ruppert 2015-04-06 08:31:23 +00:00
parent a256414d32
commit e20f69b36e
2 changed files with 165 additions and 95 deletions

View File

@ -1030,11 +1030,15 @@ computer, but the emulation appears to be correct. Even a MOD
player works, although only for lower sampling speeds. player works, although only for lower sampling speeds.
</para> </para>
<para> <para>
The OPL3 chip now also produces output. The source code has been ported from
DOSBox and the output data is polled from the mixer thread.
</para>
<para>
Also, the MIDI data running through the MPU401 ports can be written Also, the MIDI data running through the MPU401 ports can be written
into a SMF, that is the standard midi file. The wave output into a SMF, that is the standard midi file. The wave output
can be written into a VOC file, which has a format defined by can be written into a VOC file, which has a format defined by
Creative Labs. This file format can be converted to WAV by Creative Labs. Output to a WAV file and dual output (device and file
sox for example. at the same time) is now also supported.
</para> </para>
</section> </section>
@ -1045,17 +1049,19 @@ Output to the host sound system is supported on Windows, Linux, FreeBSD, MacOS 9
MacOSX and platforms supported by SDL. MacOSX and platforms supported by SDL.
</para> </para>
<para> <para>
On Linux, the output goes to any file or device. If you have a wavetable synthesizer, On Linux using OSS, the output goes to any file or device. If you have a
midi can go to <filename class="devicefile">/dev/midi00</filename>, otherwise you may need wavetable synthesizer, midi can go to <filename class="devicefile">/dev/midi00</filename>,
a midi interpreter. For example, the midid program from the DosEmu project would work. otherwise you may need a midi interpreter. For example, the midid program from
Wave output should go to <filename class="devicefile">/dev/dsp</filename>. These devices the DosEmu project would work. Wave output should go to <filename class="devicefile">/dev/dsp</filename>.
are assumed to be OSS devices, if they're not some of the ioctl's might fail. These devices are assumed to be OSS devices, if they're not some of the ioctl's
If ALSA is present on Linux and the sound driver is set to <filename>alsa</filename>, might fail. If ALSA is present on Linux and the sound driver is set to
Bochs uses it's default PCM output device and MIDI sequencer. <filename>alsa</filename>, Bochs uses it's default PCM output device and MIDI
sequencer.
</para> </para>
<para> <para>
On Windows, midi and (wave) output go to the midi mapper and the wave mapper, On Windows, midi and wave output go to the midi mapper and the wave mapper,
respectively. A future version might have selectable output devices. respectively. The device ID for the midi is now selectable. A future version
might also have selectable wave output devices.
</para> </para>
<para> <para>
See the next section for more information about the sound lowlevel interface. See the next section for more information about the sound lowlevel interface.
@ -1079,22 +1085,20 @@ running at the same time.
</para> </para>
<para> <para>
BX_SOUND_LOWLEVEL_C is the name of the class used as the "default" sound driver for BX_SOUND_LOWLEVEL_NAME is the name of the driver used as the "default" one for
output. The default value of this setting is the dummy driver with no output. The all features. The default value of this setting is the dummy driver with no output.
configure script usually changes this value. The following are supported at the The configure script usually changes this value. The following are supported at
moment: the moment:
</para> </para>
<programlisting> <programlisting>
bx_sound_linux_c for output to /dev/dsp and /dev/midi00 on Linux, alsa Output for Linux with ALSA PCM and sequencer interface
FreeBSD and maybe other OSes that use the OSS driver. oss Output for Linux, to /dev/dsp and /dev/midi00
bx_sound_windows_c for output to the midi and wave mapper of osx Output for MacOSX midi and wave device
Windows 3.1 and higher. sdl Wave output with SDL/SDL2
bx_sound_alsa_c for output to the default ALSA PCM device and the win Output for Windows midi and wave mappers
default ALSA sequencer. file Wave and midi output to file
bx_sound_osx_c for output on Mac OS 9 and Mac OSX. dummy Dummy functions, no output
bx_sound_sdl_c for wave output on platforms supported by SDL.
bx_sound_output_c for no output at all.
</programlisting> </programlisting>
<para> <para>
@ -1129,32 +1133,34 @@ output routines to their platform. It gives a short outline what services
have to be provided. have to be provided.
</para> </para>
<para> <para>
You should also have a look at the exisiting files, <emphasis>SOUNDMOD.CC</emphasis> You should also have a look at the exisiting files, <emphasis>SOUNDLOW.CC</emphasis>,
and e.g. <emphasis>SOUNDLNX.CC</emphasis> for Linux or <emphasis>SOUNDWIN.CC</emphasis> <emphasis>SOUNDMOD.CC</emphasis> and e.g. <emphasis>SOUNDLNX.CC</emphasis> for Linux
for Windows and their respective header files to get an idea about how these things or <emphasis>SOUNDWIN.CC</emphasis> for Windows and their respective header files
really work. to get an idea about how these things really work.
</para> </para>
<section><title>Files</title> <section><title>Files</title>
<para> <para>
The main include file for a lowlevel sound driver is <emphasis>iodev.h</emphasis>. The main include file for a lowlevel sound driver is <emphasis>iodev.h</emphasis>.
It has all definitions for the system-independent functions that a sound driver It has all definitions for the system-independent functions that a sound driver
uses. The sound driver also needs to include <emphasis>soundmod.h</emphasis> for uses. The sound driver also needs to include <emphasis>soundlow.h</emphasis> for
the definition of the base class <emphasis>bx_sound_output_c</emphasis>. the definitions of the base classes <emphasis>bx_sound_lowlevel_c</emphasis>,
<emphasis>bx_soundlow_waveout_c</emphasis>, <emphasis>bx_soundlow_wavein_c</emphasis>
and <emphasis>bx_soundlow_midiout_c</emphasis>.
</para> </para>
<para> <para>
Additionally, every output driver will have an include file, which should be Additionally, every output driver will have an include file, which should be
included on top of <filename>soundmod.cc</filename> to allow the emulator included on top of <filename>soundmod.cc</filename> to allow the emulator
to use that driver. A soundcard does not need to include the specific driver to use that driver. The code to initialize the object for the selected drivers
header. can be found in that file, so a soundcard emulation does not need to include
the specific driver headers.
</para> </para>
<para> <para>
To actually make the emulator use any specific driver as the default, To actually make the emulator use any specific driver as the default,
<emphasis>BX_SOUND_LOWLEVEL_C</emphasis> has to be set to the name of the <emphasis>BX_SOUND_LOWLEVEL_NAME</emphasis> has to be set to the name of the
respective output class. respective driver.
</para> </para>
<para> <para>
@ -1167,91 +1173,152 @@ and to regenerate the configure script,
</para> </para>
</section> </section>
<section><title>Classes</title> <section><title>Defines and strutures</title>
<para>
<screen>
#define BX_SOUNDLOW_WAVEPACKETSIZE 19200
#define BX_SOUNDLOW_OK 0
#define BX_SOUNDLOW_ERR 1
typedef struct {
Bit16u samplerate;
Bit8u bits;
Bit8u channels;
Bit8u format;
Bit16u volume;
} bx_pcm_param_t;
const bx_pcm_param_t default_pcm_param = {44100, 16, 2, 1};
</screen>
</para>
<para>
The maximum size of a wave data packet, the return values of the lowlevel
functions, the structure for the PCM parameters and the default parameter
set are also important for the sound driver development. They can be found
in the main include file <emphasis>soundlow.h</emphasis>.
</para>
<para>
All lowlevel sound methods called from the device code have to return either
<emphasis>BX_SOUNDLOW_OK</emphasis> if the function was successful, or
<emphasis>BX_SOUNDLOW_ERR</emphasis> if not. If any of the initialization
functions fail, the device emulation should disable the affected feature.
</para>
</section>
<section><title>Classes</title>
<para> <para>
The following classes are involved with the sound lowlevel interface: The following classes are involved with the sound lowlevel interface:
</para> </para>
<itemizedlist> <itemizedlist>
<listitem><para> <listitem><para>
<emphasis>bx_soundmod_ctl_c</emphasis> is a pseudo device that is used to <emphasis>bx_soundmod_ctl_c</emphasis> is a pseudo device that is used to
initialize the sound driver depending on the configuration. In addition to initialize the sound drivers depending on the configuration.
this, it has methods for the PC speaker beep generation and the VOC file
output.
</para></listitem> </para></listitem>
<listitem><para> <listitem><para>
<emphasis>bx_sound_output_c</emphasis> is the base output class. It has all <emphasis>bx_sound_lowlevel_c</emphasis> is the base class of the
the methods used by the soundcard emulation, but only as stubs and does not lowlevel sound support. It has methods to return pointers to the objects for
actually produce any output. These methods are then called by the available services <emphasis>waveout</emphasis>, <emphasis>wavein</emphasis>
the emulator whenever output is necessary. and <emphasis>midiout</emphasis>. The base class returns NULL for all services.
</para></listitem> </para></listitem>
<listitem><para> <listitem><para>
<emphasis>bx_sound_OS_c</emphasis> is derived from <emphasis>bx_sound_dummy_c</emphasis> is derived from <emphasis>bx_sound_lowlevel_c</emphasis>.
<emphasis>bx_sound_output_c</emphasis>. It contains the code to generate It returns vaild pointers for all services, but the output classes are only
output for the <emphasis>OS</emphasis> operating system. implemented as stubs and the <emphasis>wavein</emphasis> service returns silence.
It is not necessary to override all the methods defined in the base class, since This "dummy" driver is used whenever a OS specific driver does not implement
virtual functions are used. The constructor should call the inherited constructor all services.
as usual, even though the current constructor does not do anything yet. </para></listitem>
<listitem><para>
<emphasis>bx_soundlow_waveout_c</emphasis>, <emphasis>bx_soundlow_wavein_c</emphasis>
and <emphasis>bx_soundlow_midiout_c</emphasis> are the base classes for the
services provided by the Bochs lowlevel sound support. Some methods are stubs
and used by the "dummy" sound driver, others are helper methods and used by
the OS specific implementations derived from these base classes.
</para></listitem>
<listitem><para>
<emphasis>bx_sound_OS_c</emphasis> is derived from <emphasis>bx_sound_lowlevel_c</emphasis>.
It returns vaild pointers for all services it implements for the selected
<emphasis>OS</emphasis> (operating system / library) or NULL for services it does
not implement. In the second case the Bochs sound init code falls back to the
"dummy" driver.
</para></listitem> </para></listitem>
</itemizedlist> </itemizedlist>
</section> </section>
<section><title>The base class <emphasis>bx_sound_output_c</emphasis></title> <section><title>The base class <emphasis>bx_sound_lowlevel_c</emphasis></title>
<para> <para>
<screen> <screen>
class bx_sound_lowlevel_c : public logfunctions { class bx_sound_lowlevel_c : public logfunctions {
public: public:
bx_sound_lowlevel_c(); bx_sound_lowlevel_c();
virtual ~bx_sound_lowlevel_c(); virtual ~bx_sound_lowlevel_c();
virtual int get_type() {return BX_SOUNDLOW_DUMMY;} virtual bx_soundlow_waveout_c* get_waveout() {return NULL;}
virtual bx_soundlow_wavein_c* get_wavein() {return NULL;}
virtual bx_soundlow_midiout_c* get_midiout() {return NULL;}
virtual int waveready();
virtual int midiready();
virtual int openmidioutput(const char *mididev);
virtual int sendmidicommand(int delta, int command, int length, Bit8u data[]);
virtual int closemidioutput();
virtual int openwaveoutput(const char *wavedev);
virtual int startwaveplayback(int frequency, int bits, bx_bool stereo, int format);
virtual int sendwavepacket(int length, Bit8u data[]);
virtual int stopwaveplayback();
virtual int closewaveoutput();
virtual int openwaveinput(const char *wavedev, sound_record_handler_t rh);
virtual int startwaverecord(int frequency, int bits, bx_bool stereo, int format);
virtual int getwavepacket(int length, Bit8u data[]);
virtual int stopwaverecord();
virtual int closewaveinput();
static void record_timer_handler(void *);
void record_timer(void);
protected: protected:
int record_timer_index; bx_soundlow_waveout_c *waveout;
int record_packet_size; bx_soundlow_wavein_c *wavein;
sound_record_handler_t record_handler; bx_soundlow_midiout_c *midiout;
}; };
</screen> </screen>
</para>
<para>
The base class for sound lowlevel support is derived from the <emphasis>logfunctions</emphasis> The base class for sound lowlevel support is derived from the <emphasis>logfunctions</emphasis>
class to make the Bochs logging capabilities available in the sound driver code. class to make the Bochs logging capabilities available in the sound driver code.
It contains the framework for wave input (recording) support. The base class is The constructor of this base class only initializes all pointers to NULL and
in use if the "dummy" sound driver is selected or in case no platform or library the destructor deletes the objects if necessary.
specific implementation is available.
</para> </para>
</section> </section>
<section><title>Methods</title> <section><title>The <emphasis>waveout</emphasis> base class <emphasis>bx_soundlow_waveout_c</emphasis></title>
<para> <para>
The following are the methods that the sound driver class has to override. <screen>
All but constructor and destructor have to return either class bx_soundlow_waveout_c : public logfunctions {
<emphasis>BX_SOUNDLOW_OK</emphasis> <emphasis>(0)</emphasis> if the function public:
was successful, or <emphasis>BX_SOUNDLOW_ERR</emphasis> <emphasis>(1)</emphasis> bx_soundlow_waveout_c();
if not. If any of the initialization functions fail, output to that device is virtual ~bx_soundlow_waveout_c();
disabled until the emulator is restarted.
virtual int openwaveoutput(const char *wavedev);
virtual int set_pcm_params(bx_pcm_param_t *param);
virtual int sendwavepacket(int length, Bit8u data[], bx_pcm_param_t *src_param);
virtual int get_packetsize();
virtual int output(int length, Bit8u data[]);
virtual int closewaveoutput();
virtual int register_wave_callback(void *, get_wave_cb_t wd_cb);
virtual void unregister_wave_callback(int callback_id);
virtual bx_bool mixer_common(Bit8u *buffer, int len);
protected:
void convert_pcm_data(Bit8u *src, int srcsize, Bit8u *dst, int dstsize, bx_pcm_param_t *param);
void start_mixer_thread(void);
bx_pcm_param_t emu_pcm_param, real_pcm_param;
int cvt_mult;
int cb_count;
struct {
void *device;
get_wave_cb_t cb;
} get_wave[BX_MAX_WAVE_CALLBACKS];
int pcm_callback_id;
};
</screen>
</para>
<para>
The base class for wave output support is also derived from the
<emphasis>logfunctions</emphasis> class. In addition to wave output methods
used from sound devices, it contains everything required for the mixer thread
feature (register PCM sources, convert data formats, start mixer).
</para>
<para>
&FIXME; ALL SECTIONS BELOW ARE NOT YET UPDATED !
</para> </para>
<section><title>bx_sound_OS_c()</title> <section><title>bx_sound_OS_c()</title>

View File

@ -466,10 +466,13 @@ currently work with.
<row> <row>
<entry>Sound card</entry> <entry>Sound card</entry>
<entry>Yes</entry> <entry>Yes</entry>
<entry>Emulates a Sound Blaster 16 card (ISA, no plug&amp;play) or an ES1370 PCI <entry>Emulates a Sound Blaster 16 card (ISA, no plug&amp;play) or an
card. On Windows, Linux, FreeBSD, MacOS 9, MacOSX and all platforms supported by ES1370 PCI card. The SB16 emulation provides wave input / output, the OPL3
SDL, the output can be sent to the host computer's sound system. For the SB16, chip, the external MIDI port and volume control for wave and FM. The ES1370
see the <ulink url="../development/sb16-emulation-basics.html">developer documentation</ulink> provides wave input / output and volume control for it. On Windows, Linux,
FreeBSD, MacOS 9, MacOSX and all platforms supported by SDL, the output can
be sent to the host computer's sound system. For the SB16, see the
<ulink url="../development/sb16-emulation-basics.html">developer documentation</ulink>
for details. for details.
</entry> </entry>
</row> </row>
@ -2193,16 +2196,16 @@ to explicitly turn it off.
<entry>no</entry> <entry>no</entry>
<entry> <entry>
Enable ES1370 sound emulation. Just like the SB16 option, the available Enable ES1370 sound emulation. Just like the SB16 option, the available
lowlevel sound interface is autodetected and the gameport is turned on. lowlevel sound interfaces are autodetected and the gameport is turned on.
</entry> </entry>
</row> </row>
<row> <row>
<entry>--enable-gameport</entry> <entry>--enable-gameport</entry>
<entry>no</entry> <entry>no</entry>
<entry> <entry>
Enables the standard PC gameport. This option is only necessary if you want to Enables the standard PC gameport. This option is only necessary if you want
have a gameport, but no SB16 (see above). The connection to a real joystick is to have a gameport, but no SB16 or ES1370 (see above). The connection to a
currently supported on Linux and win32 only. real joystick is currently supported on Linux and win32 only.
</entry> </entry>
</row> </row>
<row> <row>
@ -3190,7 +3193,7 @@ These plugins will be loaded by default (if present): 'biosdev', 'extfpuirq',
<para> <para>
These plugins are also supported, but they are usually loaded directly with These plugins are also supported, but they are usually loaded directly with
their bochsrc option: 'e1000', 'es1370', 'ne2k', 'pcidev', 'pcipnic', 'sb16', their bochsrc option: 'e1000', 'es1370', 'ne2k', 'pcidev', 'pcipnic', 'sb16',
'usb_ohci', 'usb_uhci' and 'usb_xhci'. 'usb_ohci', 'usb_uhci', 'usb_xhci' and 'voodoo'.
</para> </para>
</section> </section>