diff options
Diffstat (limited to 'gr-audio/lib/windows/windows_sink.cc')
-rw-r--r-- | gr-audio/lib/windows/windows_sink.cc | 314 |
1 files changed, 314 insertions, 0 deletions
diff --git a/gr-audio/lib/windows/windows_sink.cc b/gr-audio/lib/windows/windows_sink.cc new file mode 100644 index 0000000000..f372bd0e75 --- /dev/null +++ b/gr-audio/lib/windows/windows_sink.cc @@ -0,0 +1,314 @@ +/* -*- c++ -*- */ +/* +* Copyright 2004-2011,2013 Free Software Foundation, Inc. +* +* This file is part of GNU Radio +* +* GNU Radio is free software; you can redistribute it and/or modify +* it under the terms of the GNU General Public License as published by +* the Free Software Foundation; either version 3, or (at your option) +* any later version. +* +* GNU Radio is distributed in the hope that it will be useful, +* but WITHOUT ANY WARRANTY; without even the implied warranty of +* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +* GNU General Public License for more details. +* +* You should have received a copy of the GNU General Public License +* along with GNU Radio; see the file COPYING. If not, write to +* the Free Software Foundation, Inc., 51 Franklin Street, +* Boston, MA 02110-1301, USA. +*/ + +#ifdef HAVE_CONFIG_H +#include "config.h" +#endif + +#include "audio_registry.h" +#include <windows_sink.h> +#include <gr_io_signature.h> +#include <sys/types.h> +#include <sys/stat.h> +#include <fcntl.h> +#include <unistd.h> +#include <stdio.h> +#include <iostream> +#include <stdexcept> +#include <string> +#include <sstream> + +namespace gr { + namespace audio { + + AUDIO_REGISTER_SINK(REG_PRIO_HIGH, windows)(int sampling_rate, + const std::string &device_name, + bool) + { + return sink::sptr + (new windows_sink(sampling_rate, device_name)); + } + + static const double CHUNK_TIME = 0.1; //0.001; // 100 ms + + // FIXME these should query some kind of user preference + + static std::string + default_device_name() + { + return "WAVE_MAPPER"; + } + + windows_sink::windows_sink(int sampling_freq, const std::string device_name) + : gr_sync_block("audio_windows_sink", + gr_make_io_signature(1, 2, sizeof(float)), + gr_make_io_signature(0, 0, 0)), + d_sampling_freq(sampling_freq), + d_device_name(device_name.empty() ? default_device_name() : device_name), + d_fd(-1), d_buffer(0), d_chunk_size(0) + { + d_wave_write_event = CreateEvent(NULL, FALSE, FALSE, NULL); + if(open_waveout_device() < 0) { + //fprintf(stderr, "audio_windows_sink:open_waveout_device() failed\n"); + perror("audio_windows_sink:open_waveout_device() failed\n"); + throw + std::runtime_error ("audio_windows_sink:open_waveout_device() failed"); + } + + d_chunk_size = (int)(d_sampling_freq * CHUNK_TIME); + set_output_multiple(d_chunk_size); + + d_buffer = new short[d_chunk_size * 2]; + } + + windows_sink::~windows_sink() + { + /* Free the callback Event */ + CloseHandle(d_wave_write_event); + waveOutClose(d_h_waveout); + delete [] d_buffer; + } + + int + windows_sink::work(int noutput_items, + gr_vector_const_void_star & input_items, + gr_vector_void_star & output_items) + { + const float *f0, *f1; + bool playtestsound = false; + if(playtestsound) { + // dummy + f0 = (const float*)input_items[0]; + + for(int i = 0; i < noutput_items; i += d_chunk_size) { + for(int j = 0; j < d_chunk_size; j++) { + d_buffer[2*j + 0] = (short)(sin(2.0 * 3.1415926535897932384626 * + (float)j * 1000.0 / (float)d_sampling_freq) * + 8192 + 0); //+32767 + d_buffer[2*j + 1] = d_buffer[2*j + 0]; + } + f0 += d_chunk_size; + if(write_waveout + ((HPSTR)d_buffer, 2*d_chunk_size * sizeof(short)) < 0) { + fprintf(stderr, "audio_windows_sink: write failed\n"); + perror("audio_windows_sink: write failed"); + } + } + // break; + } + else { + switch(input_items.size ()) { + case 1: // mono input + f0 = (const float*)input_items[0]; + + for(int i = 0; i < noutput_items; i += d_chunk_size) { + for(int j = 0; j < d_chunk_size; j++) { + d_buffer[2*j + 0] = (short)(f0[j] * 32767); + d_buffer[2*j + 1] = (short)(f0[j] * 32767); + } + f0 += d_chunk_size; + if(write_waveout + ((HPSTR)d_buffer, 2*d_chunk_size * sizeof(short)) < 0) { + //fprintf(stderr, "audio_windows_sink: write failed\n"); + perror("audio_windows_sink: write failed"); + } + } + break; + + case 2: // stereo input + f0 = (const float*)input_items[0]; + f1 = (const float*)input_items[1]; + + for(int i = 0; i < noutput_items; i += d_chunk_size) { + for(int j = 0; j < d_chunk_size; j++) { + d_buffer[2*j + 0] = (short)(f0[j] * 32767); + d_buffer[2*j + 1] = (short)(f1[j] * 32767); + } + f0 += d_chunk_size; + f1 += d_chunk_size; + if(write_waveout + ((HPSTR)d_buffer, 2*d_chunk_size * sizeof(short)) < 0) { + //fprintf(stderr, "audio_windows_sink: write failed\n"); + perror("audio_windows_sink: write failed"); + } + } + break; + } + } + return noutput_items; + } + + int + windows_sink::string_to_int(const std::string & s) + { + int i; + std::istringstream (s) >> i; + return i; + } //ToInt() + + int + windows_sink::open_waveout_device(void) + { + UINT /*UINT_PTR */ u_device_id; + + /** Identifier of the waveform-audio output device to open. It + can be either a device identifier or a handle of an open + waveform-audio input device. You can use the following flag + instead of a device identifier. + * + * Value Meaning + * WAVE_MAPPER The function selects a waveform-audio output + * device capable of playing the given format. + */ + if(d_device_name.empty () || default_device_name () == d_device_name) + u_device_id = WAVE_MAPPER; + else + u_device_id = (UINT) string_to_int (d_device_name); + // Open a waveform device for output using event callback. + + unsigned long result; + //HWAVEOUT outHandle; + WAVEFORMATEX wave_format; + + /* Initialize the WAVEFORMATEX for 16-bit, 44KHz, stereo */ + wave_format.wFormatTag = WAVE_FORMAT_PCM; + wave_format.nChannels = 2; + wave_format.nSamplesPerSec = d_sampling_freq; //44100; + wave_format.wBitsPerSample = 16; + wave_format.nBlockAlign = + wave_format.nChannels * (wave_format.wBitsPerSample / 8); + wave_format.nAvgBytesPerSec = + wave_format.nSamplesPerSec * wave_format.nBlockAlign; + wave_format.cbSize = 0; + + /* Open the (preferred) Digital Audio Out device. */ + result = waveOutOpen(&d_h_waveout, WAVE_MAPPER, + &wave_format, + (DWORD_PTR)d_wave_write_event, + 0, CALLBACK_EVENT | WAVE_ALLOWSYNC); + //|WAVE_FORMAT_DIRECT | CALLBACK_EVENT| WAVE_ALLOWSYNC + + if(result) { + //fprintf(stderr, "audio_windows_sink: Failed to open waveform output device.\n"); + perror("audio_windows_sink: Failed to open waveform output device."); + //LocalUnlock(hFormat); + //LocalFree(hFormat); + //mmioClose(hmmio, 0); + return -1; + } + + // + // Do not Swallow the "open" event. + // + //WaitForSingleObject(d_wave_write_event, INFINITE); + + // Allocate and lock memory for the header. + + d_h_wave_hdr = GlobalAlloc(GMEM_MOVEABLE | GMEM_SHARE, + (DWORD)sizeof(WAVEHDR)); + if(d_h_wave_hdr == NULL) { + //GlobalUnlock(hData); + //GlobalFree(hData); + //fprintf(stderr, "audio_windows_sink: Not enough memory for header.\n"); + perror("audio_windows_sink: Not enough memory for header."); + return -1; + } + + d_lp_wave_hdr = (LPWAVEHDR)GlobalLock(d_h_wave_hdr); + if(d_lp_wave_hdr == NULL) { + //GlobalUnlock(hData); + //GlobalFree(hData); + //fprintf(stderr, "audio_windows_sink: Failed to lock memory for header.\n"); + perror("audio_windows_sink: Failed to lock memory for header."); + return -1; + } + //d_lp_wave_hdr->dwFlags = WHDR_DONE; + return 0; + } + + int + windows_sink::write_waveout(HPSTR lp_data, DWORD dw_data_size) + { + UINT w_result; + int teller = 100; + // After allocation, set up and prepare header. + /*while ((d_lp_wave_hdr->dwFlags & WHDR_DONE)==0 && teller>0) { + teller--; + Sleep(1); + } */ + // Wait until previous wave write completes (first event is the open event). + WaitForSingleObject(d_wave_write_event, 100); // INFINITE + d_lp_wave_hdr->lpData = lp_data; + d_lp_wave_hdr->dwBufferLength = dw_data_size; + d_lp_wave_hdr->dwFlags = 0L; + /* Clear the WHDR_DONE bit (which the driver set last time that + this WAVEHDR was sent via waveOutWrite and was played). Some + drivers need this to be cleared */ + //d_lp_wave_hdr->dwFlags &= ~WHDR_DONE; + + d_lp_wave_hdr->dwLoops = 0L; + w_result = + waveOutPrepareHeader(d_h_waveout, d_lp_wave_hdr, sizeof(WAVEHDR)); + if(w_result != 0) { + //GlobalUnlock(hData); + //GlobalFree(hData); + //fprintf(stderr, "audio_windows_sink: Failed to waveOutPrepareHeader. error %i\n",w_result); + perror("audio_windows_sink: Failed to waveOutPrepareHeader"); + } + // Now the data block can be sent to the output device. The + // waveOutWrite function returns immediately and waveform + // data is sent to the output device in the background. + //while(!readyforplayback) Sleep(1); + //readyforplayback=false; + + w_result = waveOutWrite(d_h_waveout, d_lp_wave_hdr, sizeof(WAVEHDR)); + if(w_result != 0) { + //GlobalUnlock(hData); + //GlobalFree(hData); + //fprintf(stderr, "audio_windows_sink: Failed to write block to device.error %i\n",w_result); + perror("audio_windows_sink: Failed to write block to device"); + switch(w_result) { + case MMSYSERR_INVALHANDLE: + fprintf(stderr, "Specified device handle is invalid.\n"); + break; + case MMSYSERR_NODRIVER: + fprintf(stderr, " No device driver is present.\n"); + break; + case MMSYSERR_NOMEM: + fprintf(stderr, " Unable to allocate or lock memory.\n"); + break; + case WAVERR_UNPREPARED: + fprintf(stderr, + " The data block pointed to by the pwh parameter hasn't been prepared.\n"); + break; + default: + fprintf(stderr, "Unknown error %i\n", w_result); + } + waveOutUnprepareHeader(d_h_waveout, d_lp_wave_hdr, sizeof(WAVEHDR)); + return -1; + } + //WaitForSingleObject(d_wave_write_event, INFINITE); + return 0; + } + + } /* namespace audio */ +} /* namespace gr */ |