⭐ 欢迎来到虫虫下载站! | 📦 资源下载 📁 资源专辑 ℹ️ 关于我们
⭐ 虫虫下载站

📄 audio_windows_sink.cc

📁 这是用python语言写的一个数字广播的信号处理工具包。利用它
💻 CC
字号:
/* -*- c++ -*- *//** Copyright 2004 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_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>static const double CHUNK_TIME = 0.1;	//0.001;           // 100 ms// FIXME these should query some kind of user preferencestatic std::stringdefault_device_name (){  return "WAVE_MAPPER";}audio_windows_sink::audio_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];}audio_windows_sink::~audio_windows_sink (){  /* Free the callback Event */  CloseHandle (d_wave_write_event);  waveOutClose (d_h_waveout);  delete[]d_buffer;}audio_windows_sink_sptraudio_windows_make_sink (int sampling_freq, const std::string dev){  return audio_windows_sink_sptr (new				  audio_windows_sink (sampling_freq, dev));}intaudio_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;}intaudio_windows_sink::string_to_int (const std::string & s){  int i;  std::istringstream (s) >> i;  return i;}				//ToInt()intaudio_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;}intaudio_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;}

⌨️ 快捷键说明

复制代码 Ctrl + C
搜索代码 Ctrl + F
全屏模式 F11
切换主题 Ctrl + Shift + D
显示快捷键 ?
增大字号 Ctrl + =
减小字号 Ctrl + -