mirror of
				https://github.com/cookiengineer/audacity
				synced 2025-10-26 07:13:49 +01:00 
			
		
		
		
	
		
			
				
	
	
		
			822 lines
		
	
	
		
			27 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			822 lines
		
	
	
		
			27 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| /* sndwrite.c -- write sounds to files */
 | |
| 
 | |
| #include "stdlib.h"
 | |
| #include "switches.h"
 | |
| #include "string.h"
 | |
| #ifdef UNIX
 | |
| #include "sys/types.h"
 | |
| #endif
 | |
| #ifdef WINDOWS
 | |
| #include <io.h>
 | |
| #endif
 | |
| #include <stdio.h>
 | |
| /* include sound.h first because it includes math.h 
 | |
|  * which declares abs(). cext.h #defines abs()!
 | |
|  * sound.h depends on xlisp.h
 | |
|  */
 | |
| #include "xlisp.h"
 | |
| #include "sound.h"
 | |
| #include "cext.h"
 | |
| #include "userio.h"
 | |
| #include "falloc.h"
 | |
| #include "sndfmt.h"
 | |
| #include "sndwrite.h"
 | |
| #include "extern.h"
 | |
| #include "sndfile.h"
 | |
| 
 | |
| #ifdef UNIX
 | |
| #include "sys/file.h"
 | |
| /* #include <sys/stat.h>*/
 | |
| #include <netinet/in.h>
 | |
| #else
 | |
| #ifdef MACINTOSH
 | |
| #include <unistd.h>
 | |
| #include <stat.h>
 | |
| #define L_SET SEEK_SET
 | |
| #define L_INCR SEEK_CUR
 | |
| #endif
 | |
| #endif
 | |
| 
 | |
| /* Previously, Nyquist would wrap samples that 
 | |
|  * overflowed -- this produces horrible output, 
 | |
|  * but makes it really easy to detect clipping,
 | |
|  * which I found helpful in my own work and good 
 | |
|  * for students too since the effect is impossible
 | |
|  * to ignore. Now that Nyquist is doing IO to
 | |
|  * libraries that clip, we're going to artificially
 | |
|  * generate the wrapping here. This is floating point
 | |
|  * wrapping, so +1.0 does not wrap (it would if it
 | |
|  * were an integer since the maximum sample value for
 | |
|  * 16-bit data is a bit less than 1.) Since this is extra
 | |
|  * overhead, I'm trying to be a bit clever by using
 | |
|  * the compare to max_sample to eliminate compares
 | |
|  * for clipping in the common case.
 | |
|  * 
 | |
|  * INPUTS: max_sample -- initially 0.0
 | |
|  *         threshold -- initially 0.0
 | |
|  *         s -- the value of the current sample
 | |
|  *         x -- if s has to be wrapped, put the value here
 | |
|  */
 | |
| #define COMPUTE_MAXIMUM_AND_WRAP(x) \
 | |
|     if (s > threshold) { \
 | |
|         if (s > max_sample) { \
 | |
|             max_sample = s; \
 | |
|             threshold = min(1.0, s); \
 | |
|         } \
 | |
|         if (s > 1.0) { \
 | |
|             s = fmod(s + 1.0, 2.0) - 1.0; \
 | |
|             (x) = s; \
 | |
|         } \
 | |
|     } else if (s < -threshold) { \
 | |
|         if (s < -max_sample) { \
 | |
|             max_sample = -s; \
 | |
|             threshold = min(1.0, -s); \
 | |
|         } \
 | |
|         if (s < -1.0) { \
 | |
|             s = -(fmod(-s + 1.0, 2.0) - 1.0); \
 | |
|             (x) = s; \
 | |
|         } \
 | |
|     }
 | |
| // the s < -threshold case is tricky: 
 | |
| //    flip the signal, do the wrap, flip again
 | |
| //    in order to pass positive values to fmod
 | |
| 
 | |
| 
 | |
| /* When not using PCM encodings, we do not wrap
 | |
|  * samples -- therefore float sample formats do 
 | |
|  * not wrap or clip when written to sound files
 | |
|  */
 | |
| #define COMPUTE_MAXIMUM() \
 | |
|         if (s > max_sample) { \
 | |
|             max_sample = s; \
 | |
|         } else if (s < -max_sample) { \
 | |
|             max_sample = -s; \
 | |
|         }
 | |
| 
 | |
| 
 | |
| /* jlh Changed these to the <> format, so it will be sought for in the
 | |
|    include path */
 | |
| #include <portaudio.h>
 | |
| /* #include "exitpa.h" jlh obsolete? or windows only? PortAudio 2007
 | |
|                        has no such thing.  Is it PA 19 or PA 2.0??? */
 | |
| 
 | |
| long flush_count = 0; /* how many samples to write to finish */
 | |
| 
 | |
| #define D if (0) 
 | |
| 
 | |
| int sndwrite_trace = 0;	/* debugging */
 | |
| 
 | |
| static int portaudio_initialized = false; /* is PortAudio initialized? */
 | |
| 
 | |
| void portaudio_exit()
 | |
| {
 | |
|     if (portaudio_initialized) {
 | |
|         Pa_Terminate();
 | |
|     }
 | |
| }
 | |
| 
 | |
| 
 | |
| sample_type sound_save_sound(LVAL s_as_lval, long n, SF_INFO *sf_info, SNDFILE *snd_file,
 | |
|                              float *buf, long *ntotal, PaStream *audio_stream);
 | |
| 
 | |
| sample_type sound_save_array(LVAL sa, long n, SF_INFO *sf_info, SNDFILE *snd_file,
 | |
|                              float *buf, long *ntotal, PaStream *audio_stream);
 | |
| 
 | |
| unsigned char st_linear_to_ulaw(int sample);/* jlh not used anywhere */
 | |
| 
 | |
| 
 | |
| typedef struct {
 | |
|     sound_type sound;
 | |
|     long cnt;
 | |
|     sample_block_values_type ptr;
 | |
|     double scale;
 | |
|     int terminated;
 | |
| } sound_state_node, *sound_state_type;
 | |
| 
 | |
| 
 | |
| static int portaudio_error(PaError err, char *problem)
 | |
| {
 | |
|     char msgbuffer[256];
 | |
|     if (err != paNoError) {
 | |
|         sprintf(msgbuffer, "%s, error %ld, %s.", problem, err, 
 | |
|                             Pa_GetErrorText(err));
 | |
|         xlerrprint("warning", NULL, msgbuffer, s_unbound);
 | |
|         return true;
 | |
|     }
 | |
|     return false;
 | |
| }
 | |
| 
 | |
| 
 | |
| LVAL prepare_audio(LVAL play, SF_INFO *sf_info, PaStream **audio_stream)
 | |
| {
 | |
|     PaStreamParameters output_parameters;
 | |
|     int i;
 | |
|     int num_devices;
 | |
|     const PaDeviceInfo *device_info;
 | |
|     const PaHostApiInfo *host_info;
 | |
| 
 | |
|     if (!portaudio_initialized) {
 | |
|         if (portaudio_error(Pa_Initialize(), 
 | |
|                             "could not initialize portaudio")) {
 | |
|             return NIL;
 | |
|         }
 | |
|         portaudio_initialized = TRUE;
 | |
|     }
 | |
|         
 | |
|     output_parameters.device = Pa_GetDefaultOutputDevice(); 
 | |
|     output_parameters.channelCount = sf_info->channels;
 | |
|     output_parameters.sampleFormat = paFloat32;
 | |
|     output_parameters.hostApiSpecificStreamInfo = NULL;
 | |
|     /* remember that Nyquist has to do GC */
 | |
|     output_parameters.suggestedLatency = sound_latency;
 | |
| 
 | |
|     // Initialize the audio stream for output
 | |
|     // If this is Linux, prefer to open ALSA device
 | |
|     num_devices = Pa_GetDeviceCount();
 | |
|     for (i = 0; i < num_devices; i++) {
 | |
|         device_info = Pa_GetDeviceInfo(i);
 | |
|         host_info = Pa_GetHostApiInfo(device_info->hostApi);
 | |
|         if (host_info->type == paALSA) {
 | |
|             output_parameters.device = i;
 | |
|             break;
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     if (portaudio_error(
 | |
|          Pa_OpenStream(audio_stream, NULL /* input */, &output_parameters,
 | |
|                    sf_info->samplerate, max_sample_block_len, 
 | |
|                    paClipOff, NULL /* callback */, NULL /* userdata */),
 | |
|          "could not open audio")) {
 | |
|         return NIL;
 | |
|     }
 | |
|     flush_count = (long) (sf_info->samplerate * (sound_latency + 0.2));
 | |
| 
 | |
|     if (portaudio_error(Pa_StartStream(*audio_stream), 
 | |
|                         "could not start audio")) {
 | |
|         return NIL;
 | |
|     }
 | |
| 
 | |
|     return play;
 | |
| }
 | |
| 
 | |
| 
 | |
| /* finish_audio -- flush the remaining samples, then close */
 | |
| /**/
 | |
| void finish_audio(PaStream *audio_stream)
 | |
| {
 | |
|     /* portaudio_error(Pa_StopStream(audio_stream), "could not stop stream"); */
 | |
|     /* write Latency frames of audio to make sure all samples are played */
 | |
|     float zero[MAX_SND_CHANNELS * 16];
 | |
|     int i;
 | |
|     for (i = 0; i < MAX_SND_CHANNELS * 16; i++) zero[i] = 0.0F;
 | |
|     while (flush_count > 0) {
 | |
|         Pa_WriteStream(audio_stream, zero, 16);
 | |
|         flush_count -= 16;
 | |
|     }
 | |
|     portaudio_error(Pa_CloseStream(audio_stream), "could not close audio");
 | |
| }
 | |
| 
 | |
| long lookup_format(long format, long mode, long bits, long swap)
 | |
| {
 | |
|     long sf_mode;
 | |
|     long sf_format;
 | |
| 
 | |
|     switch (format) {
 | |
|     case SND_HEAD_NONE: return 0; break; // get info from file
 | |
|     case SND_HEAD_AIFF: sf_format = SF_FORMAT_AIFF; break;
 | |
|     case SND_HEAD_IRCAM: sf_format = SF_FORMAT_IRCAM; break;
 | |
|     case SND_HEAD_NEXT: sf_format = SF_FORMAT_AU; break;
 | |
|     case SND_HEAD_WAVE: sf_format = SF_FORMAT_WAV; break;
 | |
|     case SND_HEAD_PAF: sf_format = SF_FORMAT_PAF; break;
 | |
|     case SND_HEAD_SVX: sf_format = SF_FORMAT_SVX; break;
 | |
|     case SND_HEAD_NIST: sf_format = SF_FORMAT_NIST; break;
 | |
|     case SND_HEAD_VOC: sf_format = SF_FORMAT_VOC; break;
 | |
|     case SND_HEAD_W64: sf_format = SF_FORMAT_W64; break;
 | |
|     case SND_HEAD_MAT4: sf_format = SF_FORMAT_MAT4; break;
 | |
|     case SND_HEAD_MAT5: sf_format = SF_FORMAT_MAT5; break;
 | |
|     case SND_HEAD_PVF: sf_format = SF_FORMAT_PVF; break;
 | |
|     case SND_HEAD_XI: sf_format = SF_FORMAT_XI; break;
 | |
|     case SND_HEAD_HTK: sf_format = SF_FORMAT_HTK; break;
 | |
|     case SND_HEAD_SDS: sf_format = SF_FORMAT_SDS; break;
 | |
|     case SND_HEAD_AVR: sf_format = SF_FORMAT_AVR; break;
 | |
|     case SND_HEAD_SD2: sf_format = SF_FORMAT_SD2; break;
 | |
|     case SND_HEAD_FLAC: sf_format = SF_FORMAT_FLAC; break;
 | |
|     case SND_HEAD_CAF: sf_format = SF_FORMAT_CAF; break;
 | |
|     case SND_HEAD_RAW: 
 | |
|         sf_format = SF_FORMAT_RAW; 
 | |
| #ifdef XL_BIG_ENDIAN
 | |
|         sf_format |= (swap ? SF_ENDIAN_LITTLE : SF_ENDIAN_BIG);
 | |
| #endif
 | |
| #ifdef XL_LITTLE_ENDIAN
 | |
|         sf_format |= (swap ? SF_ENDIAN_LITTLE : SF_ENDIAN_LITTLE);
 | |
| #endif        
 | |
|         break;
 | |
|     default: 
 | |
|         sf_format = SF_FORMAT_WAV; 
 | |
|         nyquist_printf("s-save: unrecognized format, using SND_HEAD_WAVE\n");
 | |
|         break;
 | |
|     }
 | |
| 
 | |
|     switch (mode) {
 | |
|     case SND_MODE_ADPCM: sf_mode = SF_FORMAT_IMA_ADPCM; break;
 | |
|     case SND_MODE_UPCM: 
 | |
|         if (bits <= 8) {
 | |
|             sf_mode = SF_FORMAT_PCM_U8; break;
 | |
|         } else {
 | |
|             nyquist_printf("s-save: SND_MODE_UPCM is for 8-bit samples only, "
 | |
|                            "using PCM instead\n");
 | |
|         } /* no break here, fall through to SND_MODE_PCM... */
 | |
|     default:
 | |
|         nyquist_printf("s-save: unrecognized mode (%d), using PCM\n",
 | |
|                        mode);
 | |
|         /* no break, fall through as SND_MODE_PCM */
 | |
|     case SND_MODE_PCM: 
 | |
|         if (bits <= 8) sf_mode = SF_FORMAT_PCM_S8;
 | |
|         else if (bits <= 16) sf_mode = SF_FORMAT_PCM_16;
 | |
|         else if (bits <= 24) sf_mode = SF_FORMAT_PCM_24;
 | |
|         else if (bits <= 32) sf_mode = SF_FORMAT_PCM_32;
 | |
|         else {
 | |
|             sf_mode = SF_FORMAT_PCM_16;
 | |
|             nyquist_printf(
 | |
|                     "s-save: bad bits parameter (%d), using 16-bit PCM\n",
 | |
|                     bits);
 | |
|         }
 | |
|         break;
 | |
|     case SND_MODE_ULAW: sf_mode = SF_FORMAT_ULAW; break;
 | |
|     case SND_MODE_ALAW: sf_mode = SF_FORMAT_ALAW; break;
 | |
|     case SND_MODE_FLOAT: sf_mode = SF_FORMAT_FLOAT; break;
 | |
|     case SND_MODE_DOUBLE: sf_mode = SF_FORMAT_DOUBLE; break;
 | |
|     case SND_MODE_UNKNOWN: sf_mode = SF_FORMAT_PCM_16; break;
 | |
|     case SND_MODE_GSM610: sf_mode = SF_FORMAT_GSM610; break;
 | |
|     case SND_MODE_DWVW: 
 | |
|         if (bits <= 12) sf_mode = SF_FORMAT_DWVW_12;
 | |
|         else if (bits <= 16) sf_mode = SF_FORMAT_DWVW_16;
 | |
|         else if (bits <= 24) sf_mode = SF_FORMAT_DWVW_24;
 | |
|         else sf_mode = SF_FORMAT_DWVW_N;
 | |
|         break;
 | |
|     case SND_MODE_DPCM:
 | |
|         if (bits <= 8) sf_mode = SF_FORMAT_DPCM_8;
 | |
|         else if (bits <= 16) sf_mode = SF_FORMAT_DPCM_16;
 | |
|         else {
 | |
|             sf_mode = SF_FORMAT_DPCM_16;
 | |
|             nyquist_printf(
 | |
|                     "s-save: bad bits parameter (%d), using 16-bit DPCM\n",
 | |
|                     bits);
 | |
|         }
 | |
|         break;
 | |
|     case SND_MODE_MSADPCM: sf_mode = SF_FORMAT_MS_ADPCM; break;
 | |
|     }
 | |
|     return sf_format | sf_mode;
 | |
| }
 | |
| 
 | |
| 
 | |
| double sound_save(
 | |
|   LVAL snd_expr,
 | |
|   long n,
 | |
|   unsigned char *filename,
 | |
|   long format,
 | |
|   long mode,
 | |
|   long bits,
 | |
|   long swap,
 | |
|   double *sr,
 | |
|   long *nchans,
 | |
|   double *duration,
 | |
|   LVAL play)
 | |
| {
 | |
|     LVAL result;
 | |
|     float *buf;
 | |
|     long ntotal;
 | |
|     double max_sample;
 | |
|     SNDFILE *sndfile = NULL;
 | |
|     SF_INFO sf_info;
 | |
|     PaStream *audio_stream = NULL;
 | |
|     
 | |
|     gc();
 | |
|     
 | |
|     memset(&sf_info, 0, sizeof(sf_info));
 | |
|     sf_info.format = lookup_format(format, mode, bits, swap);
 | |
| 
 | |
|     result = xleval(snd_expr);
 | |
|     /* BE CAREFUL - DO NOT ALLOW GC TO RUN WHILE RESULT IS UNPROTECTED */
 | |
|     if (vectorp(result)) {
 | |
|         /* make sure all elements are of type a_sound */
 | |
|         long i = getsize(result);
 | |
|         *nchans = sf_info.channels = i;
 | |
|         while (i > 0) {
 | |
|             i--;
 | |
|             if (!exttypep(getelement(result, i), a_sound)) {
 | |
|                 xlerror("sound_save: array has non-sound element",
 | |
|                          result);
 | |
|             }
 | |
|         }
 | |
|         /* assume all are the same: */
 | |
|         *sr = sf_info.samplerate = ROUND(getsound(getelement(result, 0))->sr); 
 | |
| 
 | |
|         /* note: if filename is "", then don't write file; therefore,
 | |
|          * write the file if (filename[0])
 | |
|          */ 
 | |
|         if (filename[0]) {
 | |
|             sndfile = sf_open((char *) filename, SFM_WRITE, &sf_info);
 | |
|             if (sndfile) {
 | |
|                 /* use proper scale factor: 8000 vs 7FFF */
 | |
|                 sf_command(sndfile, SFC_SET_CLIPPING, NULL, SF_TRUE);
 | |
|             }
 | |
|         }
 | |
|         
 | |
|         if (play) 
 | |
|             play = prepare_audio(play, &sf_info, &audio_stream);
 | |
| 
 | |
|         if ((buf = (float *) malloc(max_sample_block_len * sf_info.channels *
 | |
|                                     sizeof(float))) == NULL) {
 | |
|             xlabort("snd_save -- couldn't allocate memory");
 | |
|         }
 | |
| 
 | |
|         max_sample = sound_save_array(result, n, &sf_info, sndfile, 
 | |
|                                       buf, &ntotal, audio_stream);
 | |
|         *duration = ntotal / *sr;
 | |
|         if (sndfile) sf_close(sndfile);
 | |
|         if (play != NIL) finish_audio(audio_stream);
 | |
|     } else if (exttypep(result, a_sound)) {
 | |
|         *nchans = sf_info.channels = 1;
 | |
|         sf_info.samplerate = ROUND((getsound(result))->sr);
 | |
|         *sr = sf_info.samplerate;
 | |
|         if (filename[0]) {
 | |
|             sndfile = sf_open((char *) filename, SFM_WRITE, &sf_info);
 | |
|             if (sndfile) {
 | |
|                 /* use proper scale factor: 8000 vs 7FFF */
 | |
|                 sf_command(sndfile, SFC_SET_CLIPPING, NULL, SF_TRUE);
 | |
|             }
 | |
|         }
 | |
|         if (play)
 | |
|             play = prepare_audio(play, &sf_info, &audio_stream);
 | |
| 
 | |
|         if ((buf = (float *) malloc(max_sample_block_len * 
 | |
|                                     sizeof(float))) == NULL) {
 | |
|             xlabort("snd_save -- couldn't allocate memory");
 | |
|         }
 | |
| 
 | |
|         max_sample = sound_save_sound(result, n, &sf_info, sndfile,
 | |
|                                       buf, &ntotal, audio_stream);
 | |
|         *duration = ntotal / *sr;
 | |
|         if (sndfile) sf_close(sndfile);
 | |
|         if (play != NIL) finish_audio(audio_stream);
 | |
|     } else {
 | |
|         xlerror("sound_save: expression did not return a sound",
 | |
|                  result);
 | |
|         max_sample = 0.0;
 | |
|     }
 | |
|     free(buf);
 | |
|     return max_sample;
 | |
| }
 | |
| 
 | |
| 
 | |
| /* open_for_write -- helper function for sound_overwrite */
 | |
| /*
 | |
|  * if the format is RAW, then fill in sf_info according to 
 | |
|  * sound sample rate and channels. Otherwise, open the file
 | |
|  * and see if the sample rate and channele match.
 | |
|  */
 | |
| SNDFILE *open_for_write(unsigned char *filename, long direction,
 | |
|                         long format, SF_INFO *sf_info, int channels,
 | |
|                         long srate, double offset, float **buf)
 | |
| /* channels and srate are based on the sound we're writing to the file */
 | |
| {
 | |
|     SNDFILE *sndfile;
 | |
|     sf_count_t frames; // frame count passed into sf_seek
 | |
|     char error[140];   // error messages are formatted here
 | |
|     sf_count_t rslt;   // frame count returned from sf_seek
 | |
| 
 | |
|     if (format == SND_HEAD_RAW) {
 | |
|         sf_info->channels = channels;
 | |
|         sf_info->samplerate = srate;
 | |
|     } else {
 | |
|         sf_info->format = 0;
 | |
|     }
 | |
|     sndfile = sf_open((const char *) filename, direction, sf_info); 
 | |
| 
 | |
|     if (!sndfile) {
 | |
|         sprintf(error, "snd_overwrite: cannot open file %s", filename);
 | |
|         xlabort(error);
 | |
|     }
 | |
|     /* use proper scale factor: 8000 vs 7FFF */
 | |
|     sf_command(sndfile, SFC_SET_CLIPPING, NULL, SF_TRUE);
 | |
|     
 | |
|     frames = round(offset * sf_info->samplerate);
 | |
|     rslt = sf_seek(sndfile, frames, SEEK_SET);
 | |
|     if (rslt < 0) {
 | |
|         sprintf(error, "snd_overwrite: cannot seek to frame %lld of %s",
 | |
|                 frames, filename);
 | |
|         xlabort(error);
 | |
|     }
 | |
|     if (sf_info->channels != channels) {
 | |
|         sprintf(error, "%s%ld%s%ld%s", 
 | |
|                 "snd_overwrite: number of channels in sound (",
 | |
|                 channels,
 | |
|                 ") does not match\n    number of channels in file (",
 | |
|                 sf_info->channels, ")");
 | |
|         sf_close(sndfile);
 | |
|         xlabort(error);
 | |
|     }
 | |
| 
 | |
|     if (sf_info->samplerate != srate) {
 | |
|         sprintf(error, "%s%g%s%ld%s",
 | |
|                 "snd_overwrite: sample rate in sound (",
 | |
|                 srate,
 | |
|                 ") does not match\n    sample rate in file (",
 | |
|                 sf_info->samplerate,
 | |
|                 ")"); 
 | |
|         sf_close(sndfile);
 | |
|         xlabort(error);
 | |
|     }
 | |
|         
 | |
|     if ((*buf = (float *) malloc(max_sample_block_len * channels *
 | |
|                                  sizeof(float))) == NULL) {
 | |
|         xlabort("snd_overwrite: couldn't allocate memory");
 | |
|     }
 | |
|     return sndfile;
 | |
| }
 | |
| 
 | |
| 
 | |
| double sound_overwrite(
 | |
|   LVAL snd_expr,
 | |
|   long n,
 | |
|   unsigned char *filename,
 | |
|   double offset_secs,
 | |
|   long format,
 | |
|   long mode,
 | |
|   long bits,
 | |
|   long swap,
 | |
|   double *duration)
 | |
| {
 | |
|     LVAL result;       // the SOUND to be evaluated
 | |
|     SF_INFO sf_info;   // info about the sound file
 | |
|     double max_sample; // return value
 | |
|     long ntotal;       // how many samples were overwritten
 | |
|     /*
 | |
|     long flags;
 | |
|     */
 | |
|     // first check if sound file exists, do not create new file
 | |
|     FILE *file = fopen((char *) filename, "rb");
 | |
|     // if not then fail
 | |
|     if (!file) {
 | |
|         *duration = 0;
 | |
|         return 0.0;
 | |
|     } else {
 | |
|         fclose(file);
 | |
|     }
 | |
|     memset(&sf_info, 0, sizeof(sf_info));
 | |
|     sf_info.format = lookup_format(format, mode, bits, swap);
 | |
|     result = xleval(snd_expr);
 | |
|     /* BE CAREFUL - DO NOT ALLOW GC TO RUN WHILE RESULT IS UNPROTECTED */
 | |
|     if (vectorp(result)) {
 | |
|         SNDFILE *sndfile;  // opened sound file 
 | |
|         float *buf; // buffer for samples read in from sound file
 | |
|         /* make sure all elements are of type a_sound */
 | |
|         long i = getsize(result);
 | |
|         while (i > 0) {
 | |
|             i--;
 | |
|             if (!exttypep(getelement(result, i), a_sound)) {
 | |
|                 xlerror("sound_save: array has non-sound element",
 | |
|                          result);
 | |
|             }
 | |
|         }
 | |
|         sndfile = open_for_write(filename, SFM_RDWR, format, &sf_info, i,
 | |
|                                  ROUND(getsound(getelement(result, 0))->sr),
 | |
|                                  offset_secs, &buf);
 | |
| 
 | |
|         max_sample = sound_save_array(result, n, &sf_info, sndfile, 
 | |
|                                       buf, &ntotal, NULL);
 | |
|         *duration = ntotal / (double) sf_info.samplerate;
 | |
|         free(buf);
 | |
|         sf_close(sndfile);
 | |
|     } else if (exttypep(result, a_sound)) {
 | |
|         SNDFILE *sndfile;  // opened sound file 
 | |
|         float *buf; // buffer for samples read in from sound file
 | |
|         sndfile = open_for_write(filename, SFM_RDWR, format, &sf_info, 1, 
 | |
|                                  ROUND(getsound(result)->sr), 
 | |
|                                  offset_secs, &buf);
 | |
|         max_sample = sound_save_sound(result, n, &sf_info, sndfile, buf, 
 | |
|                                       &ntotal, NULL);
 | |
|         *duration = ntotal / (double) sf_info.samplerate;
 | |
|         free(buf);
 | |
|         sf_close(sndfile);
 | |
|     } else {
 | |
|         xlerror("sound_save: expression did not return a sound",
 | |
|                  result);
 | |
|         max_sample = 0.0;
 | |
|     }
 | |
|     return max_sample;
 | |
| }
 | |
| 
 | |
| int is_pcm(SF_INFO *sf_info)
 | |
| {
 | |
|     long subtype = sf_info->format & SF_FORMAT_SUBMASK;
 | |
|     return (subtype == SF_FORMAT_PCM_S8 || subtype == SF_FORMAT_PCM_16 ||
 | |
|             subtype == SF_FORMAT_PCM_24 || subtype == SF_FORMAT_PCM_32);
 | |
| }
 | |
| 
 | |
| 
 | |
| sample_type sound_save_sound(LVAL s_as_lval, long n, SF_INFO *sf_info, 
 | |
|         SNDFILE *sndfile, float *buf, long *ntotal, PaStream *audio_stream)
 | |
| {
 | |
|     long blocklen;
 | |
|     sound_type s;
 | |
|     int i;
 | |
|     sample_type *samps;
 | |
|     long debug_unit;    /* print messages at intervals of this many samples */
 | |
|     long debug_count;   /* next point at which to print a message */
 | |
|     sample_type max_sample = 0.0F;
 | |
|     sample_type threshold = 0.0F;
 | |
|     /* jlh    cvtfn_type cvtfn; */
 | |
|     *ntotal = 0;
 | |
| 
 | |
|     /* if snd_expr was simply a symbol, then s now points to
 | |
|         a shared sound_node.  If we read samples from it, then
 | |
|         the sound bound to the symbol will be destroyed, so
 | |
|         copy it first.  If snd_expr was a real expression that
 | |
|         computed a new value, then the next garbage collection
 | |
|         will reclaim the sound_node.  We need to make the new
 | |
|         sound reachable by the garbage collector to that any
 | |
|         lisp data reachable from the sound do not get collected.
 | |
|         To make the sound reachable, we need to allocate a node,
 | |
|         and the GC might run, so we need to protect the OLD s
 | |
|         but then make it unreachable.
 | |
|         We will let the GC collect the sound in the end.
 | |
|     */
 | |
|     xlprot1(s_as_lval);
 | |
|     s = sound_copy(getsound(s_as_lval));
 | |
|     s_as_lval = cvsound(s);	/* destroys only ref. to original */
 | |
| 
 | |
|     /* for debugging */
 | |
| /*    printing_this_sound = s;*/
 | |
| 
 | |
| 
 | |
|     debug_unit = debug_count = (long) max(sf_info->samplerate, 10000.0);
 | |
| 
 | |
|     sound_frames = 0;
 | |
|     sound_srate = sf_info->samplerate;
 | |
| 
 | |
|     while (n > 0) {
 | |
|         long togo;
 | |
|         sample_block_type sampblock = sound_get_next(s, &blocklen);
 | |
|         oscheck();
 | |
| #ifdef SNAPSHOTS
 | |
|         stdputstr(".");
 | |
|         if (sound_created_flag) {
 | |
|             stdputstr("SNAPSHOT: ");
 | |
|             sound_print_tree(printing_this_sound);
 | |
|             sound_created_flag = false;
 | |
|         }
 | |
|         fflush(stdout);
 | |
| #endif
 | |
|         if (sampblock == zero_block || blocklen == 0) {
 | |
|             break;
 | |
|         }
 | |
|         togo = min(blocklen, n);
 | |
|         if (s->scale != 1) { /* copy/scale samples into buf */
 | |
|             for (i = 0; i < togo; i++) {
 | |
|                 buf[i] = s->scale * sampblock->samples[i];
 | |
|             }
 | |
|             samps = buf;
 | |
|         } else {
 | |
|             samps = sampblock->samples;
 | |
|         }
 | |
|         if (is_pcm(sf_info)) {
 | |
|             for (i = 0; i < togo; i++) {
 | |
|                 sample_type s = samps[i];
 | |
|                 COMPUTE_MAXIMUM_AND_WRAP(samps[i]);
 | |
|             }
 | |
|         } else {
 | |
|             for (i = 0; i < togo; i++) {
 | |
|                 sample_type s = samps[i];
 | |
|                 COMPUTE_MAXIMUM();
 | |
|             }
 | |
|         }
 | |
|         if (sndfile) {
 | |
|             sf_writef_float(sndfile, samps, togo);
 | |
|         }
 | |
|         if (audio_stream) {
 | |
|             Pa_WriteStream(audio_stream, samps, togo);
 | |
|             sound_frames += togo;
 | |
|         }
 | |
| 
 | |
|         n -= togo;
 | |
|         *ntotal += togo;
 | |
|         if (*ntotal > debug_count) {
 | |
|             gprintf(TRANS, " %ld ", *ntotal);
 | |
|             fflush(stdout);
 | |
|             debug_count += debug_unit;
 | |
|         }
 | |
|     }
 | |
|     gprintf(TRANS, "\ntotal samples: %ld\n", *ntotal);
 | |
|     xlpop();
 | |
|     return max_sample;
 | |
| }
 | |
| 
 | |
| 
 | |
| sample_type sound_save_array(LVAL sa, long n, SF_INFO *sf_info, 
 | |
|         SNDFILE *sndfile, float *buf, long *ntotal, PaStream *audio_stream)
 | |
| {
 | |
|     long i, chans;
 | |
|     float *float_bufp;
 | |
|     sound_state_type state;
 | |
|     double start_time = HUGE_VAL;
 | |
|     LVAL sa_copy;
 | |
|     long debug_unit;    /* print messages at intervals of this many samples */
 | |
|     long debug_count;   /* next point at which to print a message */
 | |
|     sample_type max_sample = 0.0F;
 | |
|     sample_type threshold = 0.0F;
 | |
|     /*    cvtfn_type cvtfn; jlh */
 | |
| 
 | |
|     *ntotal = 0;
 | |
| 
 | |
|     /* THE ALGORITHM: first merge floating point samples from N channels
 | |
|      * into consecutive multi-channel frames in buf.  Then, treat buf
 | |
|      * as just one channel and use one of the cvt_to_* functions to
 | |
|      * convert the data IN PLACE in the buffer (this is ok because the
 | |
|      * converted data will never take more space than the original 32-bit
 | |
|      * floats, so the converted data will not overwrite any floats before
 | |
|      * the floats are converted
 | |
|      */
 | |
| 
 | |
|     /* if snd_expr was simply a symbol, then sa now points to
 | |
|         a shared sound_node.  If we read samples from it, then
 | |
|         the sounds bound to the symbol will be destroyed, so
 | |
|         copy it first.  If snd_expr was a real expression that
 | |
|         computed a new value, then the next garbage collection
 | |
|         will reclaim the sound array.  See also sound_save_sound()
 | |
|     */
 | |
| 
 | |
|     chans = getsize(sa);
 | |
|     if (chans > MAX_SND_CHANNELS) {
 | |
|         xlerror("sound_save: too many channels", sa);
 | |
|         free(buf);
 | |
|         sf_close(sndfile);
 | |
|     }
 | |
|     xlprot1(sa);
 | |
|     sa_copy = newvector(chans);
 | |
|     xlprot1(sa_copy);
 | |
| 
 | |
|     /* Why do we copy the array into an xlisp array instead of just
 | |
|      * the state[i] array? Because some of these sounds may reference
 | |
|      * the lisp heap. We must put the sounds in an xlisp array so that
 | |
|      * the gc will find and mark them. xlprot1(sa_copy) makes the array
 | |
|      * visible to gc.
 | |
|      */
 | |
|     for (i = 0; i < chans; i++) {
 | |
|         sound_type s = getsound(getelement(sa, i));
 | |
|         setelement(sa_copy, i, cvsound(sound_copy(s)));
 | |
|     }
 | |
|     sa = sa_copy;	/* destroy original reference to allow GC */
 | |
| 
 | |
|     state = (sound_state_type) malloc(sizeof(sound_state_node) * chans);
 | |
|     for (i = 0; i < chans; i++) {
 | |
|         state[i].sound = getsound(getelement(sa, i));
 | |
|         state[i].scale = state[i].sound->scale;
 | |
| D       nyquist_printf("save scale factor %ld = %g\n", i, state[i].scale);
 | |
|         state[i].terminated = false;
 | |
|         state[i].cnt = 0;   /* force a fetch */
 | |
|         start_time = min(start_time, state[i].sound->t0);
 | |
|     }
 | |
| 
 | |
|     for (i = 0; i < chans; i++) {
 | |
|         if (state[i].sound->t0 > start_time)
 | |
|             sound_prepend_zeros(state[i].sound, start_time);
 | |
|     }
 | |
| 
 | |
|     debug_unit = debug_count = (long) max(sf_info->samplerate, 10000.0);
 | |
| 
 | |
|     sound_frames = 0;
 | |
|     sound_srate = sf_info->samplerate;
 | |
|     while (n > 0) {
 | |
|         /* keep the following information for each sound:
 | |
|             has it terminated?
 | |
|             pointer to samples
 | |
|             number of samples remaining in block
 | |
|            scan to find the minimum remaining samples and
 | |
|            output that many in an inner loop.  Stop outer
 | |
|            loop if all sounds have terminated
 | |
|          */
 | |
|         int terminated = true;
 | |
|         int togo = n;
 | |
|         int j;
 | |
| 
 | |
|         oscheck();
 | |
| 
 | |
|         for (i = 0; i < chans; i++) {
 | |
|             if (state[i].cnt == 0) {
 | |
|                 if (sndwrite_trace) {
 | |
|                     nyquist_printf("CALLING SOUND_GET_NEXT ON CHANNEL %ld (%x)\n",
 | |
| 				   i, (unsigned long) state[i].sound); /* jlh 64 bit issue */
 | |
|                     sound_print_tree(state[i].sound);
 | |
|                 }
 | |
|                 state[i].ptr = sound_get_next(state[i].sound,
 | |
|                                    &(state[i].cnt))->samples;
 | |
|                 if (sndwrite_trace) {
 | |
|                     nyquist_printf("RETURNED FROM CALL TO SOUND_GET_NEXT ON CHANNEL %ld\n", i);
 | |
|                 }
 | |
|                 if (state[i].ptr == zero_block->samples) {
 | |
|                     state[i].terminated = true;
 | |
|                 }
 | |
|             }
 | |
|             if (!state[i].terminated) terminated = false;
 | |
|             togo = min(togo, state[i].cnt);
 | |
|         }
 | |
| 
 | |
|         if (terminated) break;
 | |
| 
 | |
|         float_bufp = (float *) buf;
 | |
|         if (is_pcm(sf_info)) {
 | |
|             for (j = 0; j < togo; j++) {
 | |
|                 for (i = 0; i < chans; i++) {
 | |
|                     float s = (float) (*(state[i].ptr++) * (float) state[i].scale);
 | |
|                     COMPUTE_MAXIMUM_AND_WRAP(s);
 | |
|                     *float_bufp++ = s;
 | |
|                 }
 | |
|             }
 | |
|         } else {
 | |
|             for (j = 0; j < togo; j++) {
 | |
|                 for (i = 0; i < chans; i++) {
 | |
|                     float s = (float) (*(state[i].ptr++) * (float) state[i].scale);
 | |
|                     COMPUTE_MAXIMUM();
 | |
|                     *float_bufp++ = s;
 | |
|                 }
 | |
|             }
 | |
|         }
 | |
|         /* Here we have interleaved floats. Before converting to the sound
 | |
|            file format, call PortAudio to play them. */
 | |
|         if (audio_stream) {
 | |
|             PaError err = Pa_WriteStream(audio_stream, buf, togo);
 | |
|             if (err) {
 | |
|                 printf("Pa_WriteStream error %d\n", err);
 | |
|             }
 | |
|             sound_frames += togo;
 | |
|         }
 | |
|         if (sndfile) sf_writef_float(sndfile, buf, togo);
 | |
| 
 | |
|         n -= togo;
 | |
|         for (i = 0; i < chans; i++) {
 | |
|             state[i].cnt -= togo;
 | |
|         }
 | |
|         *ntotal += togo;
 | |
|         if (*ntotal > debug_count) {
 | |
|             gprintf(TRANS, " %ld ", *ntotal);
 | |
|             fflush(stdout);
 | |
|             debug_count += debug_unit;
 | |
|         }
 | |
|     }
 | |
|     gprintf(TRANS, "total samples: %ld x %ld channels\n",
 | |
|            *ntotal, chans);
 | |
| 
 | |
|     /* references to sounds are shared by sa_copy and state[].
 | |
|      * here, we dispose of state[], allowing GC to do the
 | |
|      * sound_unref call that frees the sounds. (Freeing them now
 | |
|      * would be a bug.)
 | |
|      */
 | |
|     free(state);
 | |
|     xlpop();
 | |
|     return max_sample;
 | |
| }
 | |
| 
 | |
| 
 |