From news.kth.se!sunic!pipex!howland.reston.ans.net!math.ohio-state.edu!jussieu.fr!univ-lyon1.fr!ghost.dsi.unimi.it!maya.dei.unipd.it!DEI Fri Oct 28 15:19:02 1994 Path: news.kth.se!sunic!pipex!howland.reston.ans.net!math.ohio-state.edu!jussieu.fr!univ-lyon1.fr!ghost.dsi.unimi.it!maya.dei.unipd.it!DEI From: knight@ugo.dei.unipd.it (Lorenzo Micheletto 260777/IL) Newsgroups: rec.games.programmer Subject: Re: Doom multi-sound programming Date: 27 Oct 1994 10:47:47 +0100 Organization: BBS MAYA - University of Padua, Italy Lines: 88 Message-ID: <38nt03$dun@ugo.dei.unipd.it> References: <783016185snz@oxfordll.demon.co.uk> NNTP-Posting-Host: ugo.dei.unipd.it In article <783016185snz@oxfordll.demon.co.uk>, Matthew Lloyd wrote: > > I've been writing a set of 8-bit Soundblaster DSP routines recently, >and I wish to know how Doom is capable of playing more than one sound >at once. > I understand that one method is to average the samples together before >playing. However, the result is bad quality, low volume samples, although >it does work. > How does Doom play more than one sound without any loss in quality? > > Thanks, > - Matthew > >-- >Matthew Lloyd - Oxford, UK >Internet: mlloyd@oxfordll.demon.co.uk I'm working on something like that, the method is really simple: An 8bit soundblaster (and a "basic" 16bit one) has only one channel for digital sound, to simulate multiple channels you have to mix multiple sound "streams" on the flight. The key for good playback is HOW you mix. First of all, forget to use a timer interrupt (i.e. reprogramming the PIT counter that controls IRQ 0), use DMA instead. As you know, you can send a block of sound data to the SB DAC using dma, so "mix on the flight" 512..4K bytes of sound data from the various "software" sound channels and then send the resulting "mixed" block to the DAC using DMA. Speed is the key to get good sound quality. First of all, you have to mix the "next" block of sound data BEFORE the one currently playing is finished (or you'll get "sound breaks"). Second, remember that reprograming the dma chip takes a little time and this may introduce a discontinuity into the sounds that are playing. Assuming a fixed sound playback rate, mixing data in blocks you add a constant noise (the glitch between a sound block and the next) with a base frequency equal to playback_frequency/samples_in_a_mix_block. For Example: With a playback frequency of 8Khz And mixing 16bit stereo sound data in blocks of 4000 bytes. There are 4000/4 = 1000 samples per block. Thus the "block mix"/"change block" frequency is 8000/1000 = 8Hz. Usually the lower the "block feed" frequency, the better (less added noise) but you have to sinchronize animation and sounds and since if you "play" a new sound on a "soft channel" it starts playing at the next "block feed", you cannot lower the "block feed" frequency too much. Usually a good choice for the "block feed" frequency is 30..40Hz. Once you know the playback frequency PF the sample size SS and the mixaged block feed frequency BF you can calculate the block size as follows: SS*(PF/BF) For example: With a 44000Hz playback frequency 16bit stereo samples (4bytes per sample) and a block feed frequency of 40Hz you need a 4*(44000/40) = 4400 bytes block size. Usually it is better to use blocks of less than 4k if you code under protected mode with paging enabled (so a block fits into a page of memory). Once you resolved the mixing timing problems, comes the choice of the mixage method, the less time you spend mixing sound data the better (less cpu overhead), but usually this result in a loss of sound quality. As far i hear (playing games) the most used method is additive mixing (sound amplitudes added together), and that's the method i use. Cutting sound resolution to 6bit i coded a 4channel mixer that mix like lighting (four PCM samples in a single step) (cutting the sample resolution and using unsigned samples i don't need to check for overflows). The thing will be included in the release 2.00 of the 386Power library ( a set of routines for coding things in 32bit protected mode assembly) Lorenzo Micheletto P.S. The freeware 386P 2.00 will be out in two weeks on x2ftp.oulu.fi there you can already find 386P 1.01 but don't waste time uploading it. 386P 2.00 is very different from the old ugly release 1.01 [ release 2.00 is a lot more powerful and some things has been fully rewritten with lots of docs added]