summaryrefslogtreecommitdiff
path: root/gr-audio
diff options
context:
space:
mode:
Diffstat (limited to 'gr-audio')
-rw-r--r--gr-audio/CMakeLists.txt11
-rw-r--r--gr-audio/examples/c++/CMakeLists.txt4
-rw-r--r--gr-audio/examples/c++/dial_tone.cc14
-rw-r--r--gr-audio/examples/grc/cvsd_sweep.grc422
-rw-r--r--gr-audio/examples/grc/dial_tone.grc272
-rwxr-xr-xgr-audio/examples/python/audio_play.py3
-rwxr-xr-xgr-audio/examples/python/audio_to_file.py3
-rwxr-xr-xgr-audio/examples/python/dial_tone.py21
-rwxr-xr-xgr-audio/examples/python/dial_tone_daemon.py21
-rwxr-xr-xgr-audio/examples/python/dial_tone_wav.py15
-rwxr-xr-xgr-audio/examples/python/mono_tone.py16
-rwxr-xr-xgr-audio/examples/python/multi_tone.py24
-rwxr-xr-xgr-audio/examples/python/noise.py3
-rwxr-xr-xgr-audio/examples/python/spectrum_inversion.py3
-rwxr-xr-xgr-audio/examples/python/test_resampler.py26
-rw-r--r--gr-audio/include/audio/CMakeLists.txt (renamed from gr-audio/include/CMakeLists.txt)10
-rw-r--r--gr-audio/include/audio/api.h (renamed from gr-audio/include/gr_audio_api.h)0
-rw-r--r--gr-audio/include/audio/sink.h63
-rw-r--r--gr-audio/include/audio/source.h63
-rw-r--r--gr-audio/include/gr_audio_sink.h57
-rw-r--r--gr-audio/include/gr_audio_source.h57
-rw-r--r--gr-audio/lib/CMakeLists.txt40
-rw-r--r--gr-audio/lib/alsa/alsa_impl.cc (renamed from gr-audio/lib/alsa/gri_alsa.cc)2
-rw-r--r--gr-audio/lib/alsa/alsa_impl.h (renamed from gr-audio/lib/alsa/gri_alsa.h)4
-rw-r--r--gr-audio/lib/alsa/alsa_sink.cc542
-rw-r--r--gr-audio/lib/alsa/alsa_sink.h111
-rw-r--r--gr-audio/lib/alsa/alsa_source.cc514
-rw-r--r--gr-audio/lib/alsa/alsa_source.h114
-rw-r--r--gr-audio/lib/alsa/audio_alsa_sink.cc548
-rw-r--r--gr-audio/lib/alsa/audio_alsa_sink.h105
-rw-r--r--gr-audio/lib/alsa/audio_alsa_source.cc509
-rw-r--r--gr-audio/lib/alsa/audio_alsa_source.h107
-rw-r--r--gr-audio/lib/audio_registry.cc154
-rw-r--r--gr-audio/lib/audio_registry.h63
-rw-r--r--gr-audio/lib/gr_audio_registry.cc132
-rw-r--r--gr-audio/lib/gr_audio_registry.h55
-rw-r--r--gr-audio/lib/jack/audio_jack_sink.cc236
-rw-r--r--gr-audio/lib/jack/audio_jack_sink.h80
-rw-r--r--gr-audio/lib/jack/audio_jack_source.cc237
-rw-r--r--gr-audio/lib/jack/audio_jack_source.h80
-rw-r--r--gr-audio/lib/jack/jack_impl.cc (renamed from gr-audio/lib/jack/gri_jack.cc)2
-rw-r--r--gr-audio/lib/jack/jack_impl.h (renamed from gr-audio/lib/jack/gri_jack.h)6
-rw-r--r--gr-audio/lib/jack/jack_sink.cc241
-rw-r--r--gr-audio/lib/jack/jack_sink.h86
-rw-r--r--gr-audio/lib/jack/jack_source.cc240
-rw-r--r--gr-audio/lib/jack/jack_source.h86
-rw-r--r--gr-audio/lib/oss/audio_oss_sink.cc161
-rw-r--r--gr-audio/lib/oss/audio_oss_source.cc177
-rw-r--r--gr-audio/lib/oss/oss_sink.cc167
-rw-r--r--gr-audio/lib/oss/oss_sink.h (renamed from gr-audio/lib/oss/audio_oss_sink.h)60
-rw-r--r--gr-audio/lib/oss/oss_source.cc186
-rw-r--r--gr-audio/lib/oss/oss_source.h (renamed from gr-audio/lib/oss/audio_oss_source.h)65
-rw-r--r--gr-audio/lib/osx/audio_osx.h72
-rw-r--r--gr-audio/lib/osx/audio_osx_sink.cc404
-rw-r--r--gr-audio/lib/osx/audio_osx_sink.h80
-rw-r--r--gr-audio/lib/osx/audio_osx_source.cc1065
-rw-r--r--gr-audio/lib/osx/audio_osx_source.h116
-rw-r--r--gr-audio/lib/osx/osx_impl.h78
-rw-r--r--gr-audio/lib/osx/osx_sink.cc429
-rw-r--r--gr-audio/lib/osx/osx_sink.h86
-rw-r--r--gr-audio/lib/osx/osx_source.cc1077
-rw-r--r--gr-audio/lib/osx/osx_source.h121
-rw-r--r--gr-audio/lib/portaudio/audio_portaudio_sink.cc362
-rw-r--r--gr-audio/lib/portaudio/audio_portaudio_sink.h86
-rw-r--r--gr-audio/lib/portaudio/audio_portaudio_source.cc374
-rw-r--r--gr-audio/lib/portaudio/audio_portaudio_source.h84
-rw-r--r--gr-audio/lib/portaudio/gri_portaudio.cc111
-rw-r--r--gr-audio/lib/portaudio/portaudio_impl.cc108
-rw-r--r--gr-audio/lib/portaudio/portaudio_impl.h (renamed from gr-audio/lib/portaudio/gri_portaudio.h)18
-rw-r--r--gr-audio/lib/portaudio/portaudio_sink.cc370
-rw-r--r--gr-audio/lib/portaudio/portaudio_sink.h90
-rw-r--r--gr-audio/lib/portaudio/portaudio_source.cc378
-rw-r--r--gr-audio/lib/portaudio/portaudio_source.h89
-rw-r--r--gr-audio/lib/windows/audio_windows_sink.cc323
-rw-r--r--gr-audio/lib/windows/audio_windows_sink.h73
-rw-r--r--gr-audio/lib/windows/audio_windows_source.cc205
-rw-r--r--gr-audio/lib/windows/windows_sink.cc314
-rw-r--r--gr-audio/lib/windows/windows_sink.h75
-rw-r--r--gr-audio/lib/windows/windows_source.cc194
-rw-r--r--gr-audio/lib/windows/windows_source.h (renamed from gr-audio/lib/windows/audio_windows_source.h)61
-rw-r--r--gr-audio/swig/__init__.py9
-rw-r--r--gr-audio/swig/audio_swig.i23
82 files changed, 6719 insertions, 6374 deletions
diff --git a/gr-audio/CMakeLists.txt b/gr-audio/CMakeLists.txt
index 35fd7fc23d..1da6635ec0 100644
--- a/gr-audio/CMakeLists.txt
+++ b/gr-audio/CMakeLists.txt
@@ -31,7 +31,9 @@ GR_REGISTER_COMPONENT("gr-audio" ENABLE_GR_AUDIO
ENABLE_GR_CORE
)
-GR_SET_GLOBAL(GR_AUDIO_INCLUDE_DIRS ${CMAKE_CURRENT_SOURCE_DIR}/include)
+GR_SET_GLOBAL(GR_AUDIO_INCLUDE_DIRS
+ ${CMAKE_CURRENT_SOURCE_DIR}/include
+)
SET(GR_PKG_AUDIO_EXAMPLES_DIR ${GR_PKG_DATA_DIR}/examples/audio)
@@ -84,10 +86,9 @@ CPACK_COMPONENT("audio_swig"
########################################################################
# Add subdirectories
########################################################################
-add_subdirectory(include)
+add_subdirectory(include/audio)
add_subdirectory(lib)
add_subdirectory(doc)
-add_subdirectory(examples/c++)
if(ENABLE_PYTHON)
add_subdirectory(swig)
add_subdirectory(grc)
@@ -95,6 +96,10 @@ if(ENABLE_PYTHON)
add_subdirectory(examples/grc)
endif(ENABLE_PYTHON)
+if(ENABLE_GR_ANALOG)
+ add_subdirectory(examples/c++)
+endif(ENABLE_GR_ANALOG)
+
########################################################################
# Create Pkg Config File
########################################################################
diff --git a/gr-audio/examples/c++/CMakeLists.txt b/gr-audio/examples/c++/CMakeLists.txt
index 0255f6e9b4..ee98f78e72 100644
--- a/gr-audio/examples/c++/CMakeLists.txt
+++ b/gr-audio/examples/c++/CMakeLists.txt
@@ -19,12 +19,14 @@
include_directories(
${GR_AUDIO_INCLUDE_DIRS}
+ ${GR_ANALOG_INCLUDE_DIRS}
${GNURADIO_CORE_INCLUDE_DIRS}
${GRUEL_INCLUDE_DIRS}
${Boost_INCLUDE_DIRS}
)
+
add_executable(dial_tone dial_tone.cc)
-target_link_libraries(dial_tone gnuradio-audio)
+target_link_libraries(dial_tone gnuradio-audio gnuradio-analog)
INSTALL(TARGETS
dial_tone
diff --git a/gr-audio/examples/c++/dial_tone.cc b/gr-audio/examples/c++/dial_tone.cc
index 4cd0ff59cf..f866edfdee 100644
--- a/gr-audio/examples/c++/dial_tone.cc
+++ b/gr-audio/examples/c++/dial_tone.cc
@@ -1,5 +1,5 @@
/*
- * Copyright 2011 Free Software Foundation, Inc.
+ * Copyright 2011,2013 Free Software Foundation, Inc.
*
* This file is part of GNU Radio
*
@@ -38,8 +38,10 @@
// Include header files for each block used in flowgraph
#include <gr_top_block.h>
-#include <gr_sig_source_f.h>
-#include <gr_audio_sink.h>
+#include <analog/sig_source_f.h>
+#include <audio/sink.h>
+
+using namespace gr;
int main(int argc, char **argv)
{
@@ -52,11 +54,11 @@ int main(int argc, char **argv)
gr_top_block_sptr tb = gr_make_top_block("dial_tone");
// Construct a real-valued signal source for each tone, at given sample rate
- gr_sig_source_f_sptr src0 = gr_make_sig_source_f(rate, GR_SIN_WAVE, 350, ampl);
- gr_sig_source_f_sptr src1 = gr_make_sig_source_f(rate, GR_SIN_WAVE, 440, ampl);
+ analog::sig_source_f::sptr src0 = analog::sig_source_f::make(rate, analog::GR_SIN_WAVE, 350, ampl);
+ analog::sig_source_f::sptr src1 = analog::sig_source_f::make(rate, analog::GR_SIN_WAVE, 440, ampl);
// Construct an audio sink to accept audio tones
- audio_sink::sptr sink = audio_make_sink(rate);
+ audio::sink::sptr sink = audio::sink::make(rate);
// Connect output #0 of src0 to input #0 of sink (left channel)
tb->connect(src0, 0, sink, 0);
diff --git a/gr-audio/examples/grc/cvsd_sweep.grc b/gr-audio/examples/grc/cvsd_sweep.grc
index b645b747ab..496fd4f8e7 100644
--- a/gr-audio/examples/grc/cvsd_sweep.grc
+++ b/gr-audio/examples/grc/cvsd_sweep.grc
@@ -1,50 +1,59 @@
<?xml version='1.0' encoding='ASCII'?>
<flow_graph>
- <timestamp>Sat Sep 19 20:30:08 2009</timestamp>
+ <timestamp>Sat Nov 10 15:10:11 2012</timestamp>
<block>
- <key>import</key>
+ <key>options</key>
<param>
<key>id</key>
- <value>import_0</value>
+ <value>cvsd_sweep</value>
</param>
<param>
<key>_enabled</key>
<value>True</value>
</param>
<param>
- <key>import</key>
- <value>import math</value>
+ <key>title</key>
+ <value>CVSD Vocoder Test</value>
</param>
<param>
- <key>_coordinate</key>
- <value>(157, 11)</value>
+ <key>author</key>
+ <value></value>
</param>
<param>
- <key>_rotation</key>
- <value>0</value>
+ <key>description</key>
+ <value></value>
</param>
- </block>
- <block>
- <key>vocoder_cvsd_decode_bf</key>
<param>
- <key>id</key>
- <value>vocoder_cvsd_decode_bf_0</value>
+ <key>window_size</key>
+ <value>1280, 1024</value>
</param>
<param>
- <key>_enabled</key>
+ <key>generate_options</key>
+ <value>wx_gui</value>
+ </param>
+ <param>
+ <key>category</key>
+ <value>Custom</value>
+ </param>
+ <param>
+ <key>run_options</key>
+ <value>prompt</value>
+ </param>
+ <param>
+ <key>run</key>
<value>True</value>
</param>
<param>
- <key>resample</key>
- <value>resample</value>
+ <key>max_nouts</key>
+ <value>0</value>
</param>
<param>
- <key>bw</key>
- <value>bw</value>
+ <key>realtime_scheduling</key>
+ <value></value>
</param>
<param>
<key>_coordinate</key>
- <value>(887, 340)</value>
+ <value>(10, 10)</value>
</param>
<param>
<key>_rotation</key>
@@ -52,42 +61,22 @@
</param>
</block>
<block>
- <key>gr_sig_source_x</key>
+ <key>import</key>
<param>
<key>id</key>
- <value>tri_source</value>
+ <value>import_0</value>
</param>
<param>
<key>_enabled</key>
<value>True</value>
</param>
<param>
- <key>type</key>
- <value>float</value>
- </param>
- <param>
- <key>samp_rate</key>
- <value>audio_rate</value>
- </param>
- <param>
- <key>waveform</key>
- <value>gr.GR_TRI_WAVE</value>
- </param>
- <param>
- <key>freq</key>
- <value>0.05</value>
- </param>
- <param>
- <key>amp</key>
- <value>0.5</value>
- </param>
- <param>
- <key>offset</key>
- <value>0</value>
+ <key>import</key>
+ <value>import math</value>
</param>
<param>
<key>_coordinate</key>
- <value>(44, 316)</value>
+ <value>(157, 11)</value>
</param>
<param>
<key>_rotation</key>
@@ -95,30 +84,26 @@
</param>
</block>
<block>
- <key>gr_throttle</key>
+ <key>vocoder_cvsd_decode_bf</key>
<param>
<key>id</key>
- <value>throttle</value>
+ <value>vocoder_cvsd_decode_bf_0</value>
</param>
<param>
<key>_enabled</key>
<value>True</value>
</param>
<param>
- <key>type</key>
- <value>float</value>
- </param>
- <param>
- <key>samples_per_second</key>
- <value>audio_rate</value>
+ <key>resample</key>
+ <value>resample</value>
</param>
<param>
- <key>vlen</key>
- <value>1</value>
+ <key>bw</key>
+ <value>bw</value>
</param>
<param>
<key>_coordinate</key>
- <value>(238, 348)</value>
+ <value>(887, 340)</value>
</param>
<param>
<key>_rotation</key>
@@ -126,7 +111,7 @@
</param>
</block>
<block>
- <key>gr_vco_f</key>
+ <key>blocks_vco_f</key>
<param>
<key>id</key>
<value>vco</value>
@@ -184,7 +169,7 @@
</param>
</block>
<block>
- <key>gr_packed_to_unpacked_xx</key>
+ <key>blocks_packed_to_unpacked_xx</key>
<param>
<key>id</key>
<value>p2u</value>
@@ -206,6 +191,10 @@
<value>gr.GR_MSB_FIRST</value>
</param>
<param>
+ <key>num_ports</key>
+ <value>1</value>
+ </param>
+ <param>
<key>_coordinate</key>
<value>(648, 415)</value>
</param>
@@ -215,7 +204,7 @@
</param>
</block>
<block>
- <key>gr_char_to_float</key>
+ <key>blocks_char_to_float</key>
<param>
<key>id</key>
<value>c2f</value>
@@ -225,6 +214,14 @@
<value>True</value>
</param>
<param>
+ <key>vlen</key>
+ <value>1</value>
+ </param>
+ <param>
+ <key>scale</key>
+ <value>1</value>
+ </param>
+ <param>
<key>_coordinate</key>
<value>(676, 483)</value>
</param>
@@ -435,6 +432,14 @@
<value>0</value>
</param>
<param>
+ <key>win</key>
+ <value>None</value>
+ </param>
+ <param>
+ <key>win_size</key>
+ <value></value>
+ </param>
+ <param>
<key>grid_pos</key>
<value>0, 0, 1, 1</value>
</param>
@@ -443,6 +448,10 @@
<value>displays, 0</value>
</param>
<param>
+ <key>freqvar</key>
+ <value>None</value>
+ </param>
+ <param>
<key>_coordinate</key>
<value>(415, 97)</value>
</param>
@@ -478,6 +487,10 @@
<value>0</value>
</param>
<param>
+ <key>v_offset</key>
+ <value>0</value>
+ </param>
+ <param>
<key>t_scale</key>
<value>0</value>
</param>
@@ -494,6 +507,10 @@
<value>1</value>
</param>
<param>
+ <key>win_size</key>
+ <value></value>
+ </param>
+ <param>
<key>grid_pos</key>
<value>1, 0, 1, 1</value>
</param>
@@ -502,87 +519,16 @@
<value>displays, 0</value>
</param>
<param>
- <key>_coordinate</key>
- <value>(414, 425)</value>
- </param>
- <param>
- <key>_rotation</key>
- <value>180</value>
- </param>
- </block>
- <block>
- <key>wxgui_fftsink2</key>
- <param>
- <key>id</key>
- <value>enc_fft</value>
- </param>
- <param>
- <key>_enabled</key>
- <value>True</value>
- </param>
- <param>
- <key>type</key>
- <value>float</value>
- </param>
- <param>
- <key>title</key>
- <value>Encoded Spectrum</value>
- </param>
- <param>
- <key>samp_rate</key>
- <value>audio_rate*resample</value>
- </param>
- <param>
- <key>baseband_freq</key>
- <value>0</value>
- </param>
- <param>
- <key>y_per_div</key>
- <value>10</value>
- </param>
- <param>
- <key>y_divs</key>
- <value>8</value>
- </param>
- <param>
- <key>ref_level</key>
- <value>10</value>
- </param>
- <param>
- <key>ref_scale</key>
- <value>2.0</value>
- </param>
- <param>
- <key>fft_size</key>
- <value>1024</value>
- </param>
- <param>
- <key>fft_rate</key>
- <value>30</value>
- </param>
- <param>
- <key>peak_hold</key>
- <value>False</value>
- </param>
- <param>
- <key>average</key>
- <value>False</value>
- </param>
- <param>
- <key>avg_alpha</key>
- <value>0</value>
- </param>
- <param>
- <key>grid_pos</key>
- <value>1, 0, 1, 1</value>
+ <key>trig_mode</key>
+ <value>gr.gr_TRIG_MODE_AUTO</value>
</param>
<param>
- <key>notebook</key>
- <value>displays, 1</value>
+ <key>y_axis_label</key>
+ <value>Counts</value>
</param>
<param>
<key>_coordinate</key>
- <value>(610, 551)</value>
+ <value>(414, 425)</value>
</param>
<param>
<key>_rotation</key>
@@ -616,6 +562,10 @@
<value>0.5</value>
</param>
<param>
+ <key>v_offset</key>
+ <value>0</value>
+ </param>
+ <param>
<key>t_scale</key>
<value>20.0/(audio_rate*resample)</value>
</param>
@@ -632,6 +582,10 @@
<value>1</value>
</param>
<param>
+ <key>win_size</key>
+ <value></value>
+ </param>
+ <param>
<key>grid_pos</key>
<value>0, 0, 1, 1</value>
</param>
@@ -640,6 +594,14 @@
<value>displays, 1</value>
</param>
<param>
+ <key>trig_mode</key>
+ <value>gr.gr_TRIG_MODE_AUTO</value>
+ </param>
+ <param>
+ <key>y_axis_label</key>
+ <value>Counts</value>
+ </param>
+ <param>
<key>_coordinate</key>
<value>(858, 591)</value>
</param>
@@ -711,6 +673,14 @@
<value>0</value>
</param>
<param>
+ <key>win</key>
+ <value>None</value>
+ </param>
+ <param>
+ <key>win_size</key>
+ <value></value>
+ </param>
+ <param>
<key>grid_pos</key>
<value>0, 0, 1, 1</value>
</param>
@@ -719,6 +689,10 @@
<value>displays, 2</value>
</param>
<param>
+ <key>freqvar</key>
+ <value>None</value>
+ </param>
+ <param>
<key>_coordinate</key>
<value>(891, 98)</value>
</param>
@@ -754,6 +728,10 @@
<value>0</value>
</param>
<param>
+ <key>v_offset</key>
+ <value>0</value>
+ </param>
+ <param>
<key>t_scale</key>
<value>0</value>
</param>
@@ -770,6 +748,10 @@
<value>1</value>
</param>
<param>
+ <key>win_size</key>
+ <value></value>
+ </param>
+ <param>
<key>grid_pos</key>
<value>1, 0, 1, 1</value>
</param>
@@ -778,6 +760,14 @@
<value>displays, 2</value>
</param>
<param>
+ <key>trig_mode</key>
+ <value>gr.gr_TRIG_MODE_AUTO</value>
+ </param>
+ <param>
+ <key>y_axis_label</key>
+ <value>Counts</value>
+ </param>
+ <param>
<key>_coordinate</key>
<value>(889, 422)</value>
</param>
@@ -787,56 +777,170 @@
</param>
</block>
<block>
- <key>options</key>
+ <key>blocks_throttle</key>
<param>
<key>id</key>
- <value>cvsd_sweep</value>
+ <value>throttle</value>
</param>
<param>
<key>_enabled</key>
<value>True</value>
</param>
<param>
- <key>title</key>
- <value>CVSD Vocoder Test</value>
+ <key>type</key>
+ <value>float</value>
</param>
<param>
- <key>author</key>
- <value></value>
+ <key>samples_per_second</key>
+ <value>audio_rate</value>
</param>
<param>
- <key>description</key>
- <value></value>
+ <key>vlen</key>
+ <value>1</value>
</param>
<param>
- <key>window_size</key>
- <value>1280, 1024</value>
+ <key>_coordinate</key>
+ <value>(238, 348)</value>
</param>
<param>
- <key>generate_options</key>
- <value>wx_gui</value>
+ <key>_rotation</key>
+ <value>0</value>
</param>
+ </block>
+ <block>
+ <key>analog_sig_source_x</key>
<param>
- <key>category</key>
- <value>Custom</value>
+ <key>id</key>
+ <value>analog_sig_source_x_0</value>
</param>
<param>
- <key>run</key>
+ <key>_enabled</key>
<value>True</value>
</param>
<param>
- <key>realtime_scheduling</key>
- <value></value>
+ <key>type</key>
+ <value>float</value>
+ </param>
+ <param>
+ <key>samp_rate</key>
+ <value>audio_rate</value>
+ </param>
+ <param>
+ <key>waveform</key>
+ <value>analog.GR_TRI_WAVE</value>
+ </param>
+ <param>
+ <key>freq</key>
+ <value>0.05</value>
+ </param>
+ <param>
+ <key>amp</key>
+ <value>0.5</value>
+ </param>
+ <param>
+ <key>offset</key>
+ <value>0</value>
</param>
<param>
<key>_coordinate</key>
- <value>(10, 10)</value>
+ <value>(29, 316)</value>
</param>
<param>
<key>_rotation</key>
<value>0</value>
</param>
</block>
+ <block>
+ <key>wxgui_fftsink2</key>
+ <param>
+ <key>id</key>
+ <value>enc_fft</value>
+ </param>
+ <param>
+ <key>_enabled</key>
+ <value>True</value>
+ </param>
+ <param>
+ <key>type</key>
+ <value>float</value>
+ </param>
+ <param>
+ <key>title</key>
+ <value>Encoded Spectrum</value>
+ </param>
+ <param>
+ <key>samp_rate</key>
+ <value>audio_rate*resample</value>
+ </param>
+ <param>
+ <key>baseband_freq</key>
+ <value>0</value>
+ </param>
+ <param>
+ <key>y_per_div</key>
+ <value>10</value>
+ </param>
+ <param>
+ <key>y_divs</key>
+ <value>8</value>
+ </param>
+ <param>
+ <key>ref_level</key>
+ <value>10</value>
+ </param>
+ <param>
+ <key>ref_scale</key>
+ <value>2.0</value>
+ </param>
+ <param>
+ <key>fft_size</key>
+ <value>1024</value>
+ </param>
+ <param>
+ <key>fft_rate</key>
+ <value>30</value>
+ </param>
+ <param>
+ <key>peak_hold</key>
+ <value>False</value>
+ </param>
+ <param>
+ <key>average</key>
+ <value>False</value>
+ </param>
+ <param>
+ <key>avg_alpha</key>
+ <value>0</value>
+ </param>
+ <param>
+ <key>win</key>
+ <value>None</value>
+ </param>
+ <param>
+ <key>win_size</key>
+ <value></value>
+ </param>
+ <param>
+ <key>grid_pos</key>
+ <value>1, 0, 1, 1</value>
+ </param>
+ <param>
+ <key>notebook</key>
+ <value>displays, 1</value>
+ </param>
+ <param>
+ <key>freqvar</key>
+ <value>None</value>
+ </param>
+ <param>
+ <key>_coordinate</key>
+ <value>(610, 559)</value>
+ </param>
+ <param>
+ <key>_rotation</key>
+ <value>180</value>
+ </param>
+ </block>
<connection>
<source_block_id>vco</source_block_id>
<sink_block_id>orig_fft</sink_block_id>
@@ -844,12 +948,6 @@
<sink_key>0</sink_key>
</connection>
<connection>
- <source_block_id>tri_source</source_block_id>
- <sink_block_id>throttle</sink_block_id>
- <source_key>0</source_key>
- <sink_key>0</sink_key>
- </connection>
- <connection>
<source_block_id>throttle</source_block_id>
<sink_block_id>vco</sink_block_id>
<source_key>0</source_key>
@@ -915,4 +1013,10 @@
<source_key>0</source_key>
<sink_key>0</sink_key>
</connection>
+ <connection>
+ <source_block_id>analog_sig_source_x_0</source_block_id>
+ <sink_block_id>throttle</sink_block_id>
+ <source_key>0</source_key>
+ <sink_key>0</sink_key>
+ </connection>
</flow_graph>
diff --git a/gr-audio/examples/grc/dial_tone.grc b/gr-audio/examples/grc/dial_tone.grc
index ac8cbef279..2503fed640 100644
--- a/gr-audio/examples/grc/dial_tone.grc
+++ b/gr-audio/examples/grc/dial_tone.grc
@@ -1,6 +1,6 @@
<?xml version='1.0' encoding='ASCII'?>
<flow_graph>
- <timestamp>Thu Jul 24 14:27:48 2008</timestamp>
+ <timestamp>Sat Nov 10 15:10:08 2012</timestamp>
<block>
<key>options</key>
<param>
@@ -36,39 +36,24 @@
<value>Custom</value>
</param>
<param>
- <key>_coordinate</key>
- <value>(10, 10)</value>
+ <key>run_options</key>
+ <value>prompt</value>
</param>
<param>
- <key>_rotation</key>
- <value>0</value>
- </param>
- </block>
- <block>
- <key>gr_add_xx</key>
- <param>
- <key>id</key>
- <value>gr_add_xx</value>
- </param>
- <param>
- <key>_enabled</key>
+ <key>run</key>
<value>True</value>
</param>
<param>
- <key>type</key>
- <value>float</value>
- </param>
- <param>
- <key>num_inputs</key>
- <value>3</value>
+ <key>max_nouts</key>
+ <value>0</value>
</param>
<param>
- <key>vlen</key>
- <value>1</value>
+ <key>realtime_scheduling</key>
+ <value></value>
</param>
<param>
<key>_coordinate</key>
- <value>(513, 277)</value>
+ <value>(10, 10)</value>
</param>
<param>
<key>_rotation</key>
@@ -91,7 +76,7 @@
</param>
<param>
<key>device_name</key>
- <value/>
+ <value></value>
</param>
<param>
<key>ok_to_block</key>
@@ -111,34 +96,54 @@
</param>
</block>
<block>
- <key>gr_noise_source_x</key>
+ <key>variable_slider</key>
<param>
<key>id</key>
- <value>gr_noise_source_x</value>
+ <value>ampl</value>
</param>
<param>
<key>_enabled</key>
<value>True</value>
</param>
<param>
- <key>type</key>
- <value>float</value>
+ <key>label</key>
+ <value>Volume</value>
</param>
<param>
- <key>noise_type</key>
- <value>gr.GR_GAUSSIAN</value>
+ <key>value</key>
+ <value>.4</value>
</param>
<param>
- <key>amp</key>
- <value>noise</value>
+ <key>min</key>
+ <value>0</value>
</param>
<param>
- <key>seed</key>
- <value>42</value>
+ <key>max</key>
+ <value>.5</value>
+ </param>
+ <param>
+ <key>num_steps</key>
+ <value>100</value>
+ </param>
+ <param>
+ <key>style</key>
+ <value>wx.SL_HORIZONTAL</value>
+ </param>
+ <param>
+ <key>converver</key>
+ <value>float_converter</value>
+ </param>
+ <param>
+ <key>grid_pos</key>
+ <value>0, 0, 1, 2</value>
+ </param>
+ <param>
+ <key>notebook</key>
+ <value></value>
</param>
<param>
<key>_coordinate</key>
- <value>(238, 380)</value>
+ <value>(634, 413)</value>
</param>
<param>
<key>_rotation</key>
@@ -146,42 +151,54 @@
</param>
</block>
<block>
- <key>gr_sig_source_x</key>
+ <key>variable_slider</key>
<param>
<key>id</key>
- <value>gr_sig_source_x</value>
+ <value>noise</value>
</param>
<param>
<key>_enabled</key>
<value>True</value>
</param>
<param>
- <key>type</key>
- <value>float</value>
+ <key>label</key>
+ <value>Noise</value>
</param>
<param>
- <key>samp_rate</key>
- <value>samp_rate</value>
+ <key>value</key>
+ <value>.005</value>
</param>
<param>
- <key>waveform</key>
- <value>gr.GR_COS_WAVE</value>
+ <key>min</key>
+ <value>0</value>
</param>
<param>
- <key>freq</key>
- <value>440</value>
+ <key>max</key>
+ <value>.2</value>
</param>
<param>
- <key>amp</key>
- <value>ampl</value>
+ <key>num_steps</key>
+ <value>100</value>
</param>
<param>
- <key>offset</key>
- <value>0</value>
+ <key>style</key>
+ <value>wx.SL_HORIZONTAL</value>
+ </param>
+ <param>
+ <key>converver</key>
+ <value>float_converter</value>
+ </param>
+ <param>
+ <key>grid_pos</key>
+ <value>1, 0, 1, 2</value>
+ </param>
+ <param>
+ <key>notebook</key>
+ <value></value>
</param>
<param>
<key>_coordinate</key>
- <value>(240, 208)</value>
+ <value>(443, 412)</value>
</param>
<param>
<key>_rotation</key>
@@ -189,42 +206,53 @@
</param>
</block>
<block>
- <key>gr_sig_source_x</key>
+ <key>variable</key>
<param>
<key>id</key>
- <value>gr_sig_source_x0</value>
+ <value>samp_rate</value>
</param>
<param>
<key>_enabled</key>
<value>True</value>
</param>
<param>
- <key>type</key>
- <value>float</value>
+ <key>value</key>
+ <value>32000</value>
</param>
<param>
- <key>samp_rate</key>
- <value>samp_rate</value>
+ <key>_coordinate</key>
+ <value>(11, 171)</value>
</param>
<param>
- <key>waveform</key>
- <value>gr.GR_COS_WAVE</value>
+ <key>_rotation</key>
+ <value>0</value>
+ </param>
+ </block>
+ <block>
+ <key>blocks_add_xx</key>
+ <param>
+ <key>id</key>
+ <value>blocks_add_xx</value>
</param>
<param>
- <key>freq</key>
- <value>350</value>
+ <key>_enabled</key>
+ <value>True</value>
</param>
<param>
- <key>amp</key>
- <value>ampl</value>
+ <key>type</key>
+ <value>float</value>
</param>
<param>
- <key>offset</key>
- <value>0</value>
+ <key>num_inputs</key>
+ <value>3</value>
+ </param>
+ <param>
+ <key>vlen</key>
+ <value>1</value>
</param>
<param>
<key>_coordinate</key>
- <value>(240, 38)</value>
+ <value>(513, 277)</value>
</param>
<param>
<key>_rotation</key>
@@ -232,46 +260,42 @@
</param>
</block>
<block>
- <key>variable_slider</key>
+ <key>analog_sig_source_x</key>
<param>
<key>id</key>
- <value>ampl</value>
+ <value>analog_sig_source_x_0</value>
</param>
<param>
<key>_enabled</key>
<value>True</value>
</param>
<param>
- <key>label</key>
- <value>Volume</value>
- </param>
- <param>
- <key>value</key>
- <value>.4</value>
+ <key>type</key>
+ <value>float</value>
</param>
<param>
- <key>min</key>
- <value>0</value>
+ <key>samp_rate</key>
+ <value>samp_rate</value>
</param>
<param>
- <key>max</key>
- <value>.5</value>
+ <key>waveform</key>
+ <value>analog.GR_COS_WAVE</value>
</param>
<param>
- <key>num_steps</key>
- <value>100</value>
+ <key>freq</key>
+ <value>350</value>
</param>
<param>
- <key>slider_type</key>
- <value>horizontal</value>
+ <key>amp</key>
+ <value>ampl</value>
</param>
<param>
- <key>grid_pos</key>
- <value>0, 0, 1, 2</value>
+ <key>offset</key>
+ <value>0</value>
</param>
<param>
<key>_coordinate</key>
- <value>(634, 413)</value>
+ <value>(251, 100)</value>
</param>
<param>
<key>_rotation</key>
@@ -279,46 +303,42 @@
</param>
</block>
<block>
- <key>variable_slider</key>
+ <key>analog_sig_source_x</key>
<param>
<key>id</key>
- <value>noise</value>
+ <value>analog_sig_source_x_1</value>
</param>
<param>
<key>_enabled</key>
<value>True</value>
</param>
<param>
- <key>label</key>
- <value>Noise</value>
- </param>
- <param>
- <key>value</key>
- <value>.005</value>
+ <key>type</key>
+ <value>float</value>
</param>
<param>
- <key>min</key>
- <value>0</value>
+ <key>samp_rate</key>
+ <value>samp_rate</value>
</param>
<param>
- <key>max</key>
- <value>.2</value>
+ <key>waveform</key>
+ <value>analog.GR_COS_WAVE</value>
</param>
<param>
- <key>num_steps</key>
- <value>100</value>
+ <key>freq</key>
+ <value>440</value>
</param>
<param>
- <key>slider_type</key>
- <value>horizontal</value>
+ <key>amp</key>
+ <value>ampl</value>
</param>
<param>
- <key>grid_pos</key>
- <value>1, 0, 1, 2</value>
+ <key>offset</key>
+ <value>0</value>
</param>
<param>
<key>_coordinate</key>
- <value>(443, 412)</value>
+ <value>(250, 214)</value>
</param>
<param>
<key>_rotation</key>
@@ -326,22 +346,34 @@
</param>
</block>
<block>
- <key>variable</key>
+ <key>analog_noise_source_x</key>
<param>
<key>id</key>
- <value>samp_rate</value>
+ <value>analog_noise_source_x_0</value>
</param>
<param>
<key>_enabled</key>
<value>True</value>
</param>
<param>
- <key>value</key>
- <value>32000</value>
+ <key>type</key>
+ <value>float</value>
+ </param>
+ <param>
+ <key>noise_type</key>
+ <value>analog.GR_GAUSSIAN</value>
+ </param>
+ <param>
+ <key>amp</key>
+ <value>noise</value>
+ </param>
+ <param>
+ <key>seed</key>
+ <value>-42</value>
</param>
<param>
<key>_coordinate</key>
- <value>(11, 171)</value>
+ <value>(245, 342)</value>
</param>
<param>
<key>_rotation</key>
@@ -349,27 +381,27 @@
</param>
</block>
<connection>
- <source_block_id>gr_sig_source_x0</source_block_id>
- <sink_block_id>gr_add_xx</sink_block_id>
+ <source_block_id>blocks_add_xx</source_block_id>
+ <sink_block_id>audio_sink</sink_block_id>
<source_key>0</source_key>
<sink_key>0</sink_key>
</connection>
<connection>
- <source_block_id>gr_sig_source_x</source_block_id>
- <sink_block_id>gr_add_xx</sink_block_id>
+ <source_block_id>analog_sig_source_x_0</source_block_id>
+ <sink_block_id>blocks_add_xx</sink_block_id>
<source_key>0</source_key>
- <sink_key>1</sink_key>
+ <sink_key>0</sink_key>
</connection>
<connection>
- <source_block_id>gr_noise_source_x</source_block_id>
- <sink_block_id>gr_add_xx</sink_block_id>
+ <source_block_id>analog_sig_source_x_1</source_block_id>
+ <sink_block_id>blocks_add_xx</sink_block_id>
<source_key>0</source_key>
- <sink_key>2</sink_key>
+ <sink_key>1</sink_key>
</connection>
<connection>
- <source_block_id>gr_add_xx</source_block_id>
- <sink_block_id>audio_sink</sink_block_id>
+ <source_block_id>analog_noise_source_x_0</source_block_id>
+ <sink_block_id>blocks_add_xx</sink_block_id>
<source_key>0</source_key>
- <sink_key>0</sink_key>
+ <sink_key>2</sink_key>
</connection>
</flow_graph>
diff --git a/gr-audio/examples/python/audio_play.py b/gr-audio/examples/python/audio_play.py
index 465590f69f..94ea72498d 100755
--- a/gr-audio/examples/python/audio_play.py
+++ b/gr-audio/examples/python/audio_play.py
@@ -22,6 +22,7 @@
from gnuradio import gr
from gnuradio import audio
+from gnuradio import blocks
from gnuradio.eng_option import eng_option
from optparse import OptionParser
@@ -45,7 +46,7 @@ class my_top_block(gr.top_block):
raise SystemExit, 1
sample_rate = int(options.sample_rate)
- src = gr.file_source (gr.sizeof_float, options.filename, options.repeat)
+ src = blocks.file_source (gr.sizeof_float, options.filename, options.repeat)
dst = audio.sink (sample_rate, options.audio_output)
self.connect(src, dst)
diff --git a/gr-audio/examples/python/audio_to_file.py b/gr-audio/examples/python/audio_to_file.py
index 3f7a4f8d1c..201ec90bf1 100755
--- a/gr-audio/examples/python/audio_to_file.py
+++ b/gr-audio/examples/python/audio_to_file.py
@@ -22,6 +22,7 @@
from gnuradio import gr
from gnuradio import audio
+from gnuradio import blocks
from gnuradio.eng_option import eng_option
from optparse import OptionParser
@@ -47,7 +48,7 @@ class my_top_block(gr.top_block):
sample_rate = int(options.sample_rate)
src = audio.source (sample_rate, options.audio_input)
- dst = gr.file_sink (gr.sizeof_float, filename)
+ dst = blocks.file_sink (gr.sizeof_float, filename)
if options.nsamples is None:
self.connect((src, 0), dst)
diff --git a/gr-audio/examples/python/dial_tone.py b/gr-audio/examples/python/dial_tone.py
index 5661d13d48..c55d0d38dd 100755
--- a/gr-audio/examples/python/dial_tone.py
+++ b/gr-audio/examples/python/dial_tone.py
@@ -1,6 +1,6 @@
#!/usr/bin/env python
#
-# Copyright 2004,2005,2007 Free Software Foundation, Inc.
+# Copyright 2004,2005,2007,2012 Free Software Foundation, Inc.
#
# This file is part of GNU Radio
#
@@ -25,6 +25,12 @@ from gnuradio import audio
from gnuradio.eng_option import eng_option
from optparse import OptionParser
+try:
+ from gnuradio import analog
+except ImportError:
+ sys.stderr.write("Error: Program requires gr-analog.\n")
+ sys.exit(1)
+
class my_top_block(gr.top_block):
def __init__(self):
@@ -35,7 +41,7 @@ class my_top_block(gr.top_block):
help="pcm output device name. E.g., hw:0,0 or /dev/dsp")
parser.add_option("-r", "--sample-rate", type="eng_float", default=48000,
help="set sample rate to RATE (48000)")
- (options, args) = parser.parse_args ()
+ (options, args) = parser.parse_args()
if len(args) != 0:
parser.print_help()
raise SystemExit, 1
@@ -43,12 +49,11 @@ class my_top_block(gr.top_block):
sample_rate = int(options.sample_rate)
ampl = 0.1
- src0 = gr.sig_source_f (sample_rate, gr.GR_SIN_WAVE, 350, ampl)
- src1 = gr.sig_source_f (sample_rate, gr.GR_SIN_WAVE, 440, ampl)
- dst = audio.sink (sample_rate, options.audio_output)
- self.connect (src0, (dst, 0))
- self.connect (src1, (dst, 1))
-
+ src0 = analog.sig_source_f(sample_rate, analog.GR_SIN_WAVE, 350, ampl)
+ src1 = analog.sig_source_f(sample_rate, analog.GR_SIN_WAVE, 440, ampl)
+ dst = audio.sink(sample_rate, options.audio_output)
+ self.connect(src0, (dst, 0))
+ self.connect(src1, (dst, 1))
if __name__ == '__main__':
try:
diff --git a/gr-audio/examples/python/dial_tone_daemon.py b/gr-audio/examples/python/dial_tone_daemon.py
index b25baebee2..e4dbd95321 100755
--- a/gr-audio/examples/python/dial_tone_daemon.py
+++ b/gr-audio/examples/python/dial_tone_daemon.py
@@ -1,6 +1,6 @@
#!/usr/bin/env python
#
-# Copyright 2004,2005,2007,2008 Free Software Foundation, Inc.
+# Copyright 2004,2005,2007,2008,2012 Free Software Foundation, Inc.
#
# This file is part of GNU Radio
#
@@ -26,6 +26,12 @@ from gnuradio.eng_option import eng_option
from optparse import OptionParser
import os
+try:
+ from gnuradio import analog
+except ImportError:
+ sys.stderr.write("Error: Program requires gr-analog.\n")
+ sys.exit(1)
+
class my_top_block(gr.top_block):
def __init__(self):
@@ -36,7 +42,7 @@ class my_top_block(gr.top_block):
help="pcm output device name. E.g., hw:0,0 or /dev/dsp")
parser.add_option("-r", "--sample-rate", type="eng_float", default=48000,
help="set sample rate to RATE (48000)")
- (options, args) = parser.parse_args ()
+ (options, args) = parser.parse_args()
if len(args) != 0:
parser.print_help()
raise SystemExit, 1
@@ -44,12 +50,11 @@ class my_top_block(gr.top_block):
sample_rate = int(options.sample_rate)
ampl = 0.1
- src0 = gr.sig_source_f (sample_rate, gr.GR_SIN_WAVE, 350, ampl)
- src1 = gr.sig_source_f (sample_rate, gr.GR_SIN_WAVE, 440, ampl)
- dst = audio.sink (sample_rate, options.audio_output)
- self.connect (src0, (dst, 0))
- self.connect (src1, (dst, 1))
-
+ src0 = analog.sig_source_f(sample_rate, analog.GR_SIN_WAVE, 350, ampl)
+ src1 = analog.sig_source_f(sample_rate, analog.GR_SIN_WAVE, 440, ampl)
+ dst = audio.sink(sample_rate, options.audio_output)
+ self.connect(src0, (dst, 0))
+ self.connect(src1, (dst, 1))
if __name__ == '__main__':
pid = gru.daemonize()
diff --git a/gr-audio/examples/python/dial_tone_wav.py b/gr-audio/examples/python/dial_tone_wav.py
index c06af55b70..91bf744c95 100755
--- a/gr-audio/examples/python/dial_tone_wav.py
+++ b/gr-audio/examples/python/dial_tone_wav.py
@@ -1,6 +1,6 @@
#!/usr/bin/env python
#
-# Copyright 2004,2005,2007,2008 Free Software Foundation, Inc.
+# Copyright 2004,2005,2007,2008,2012 Free Software Foundation, Inc.
#
# This file is part of GNU Radio
#
@@ -23,9 +23,16 @@
# GNU Radio example program to record a dial tone to a WAV file
from gnuradio import gr
+from gnuradio import blocks
from gnuradio.eng_option import eng_option
from optparse import OptionParser
+try:
+ from gnuradio import analog
+except ImportError:
+ sys.stderr.write("Error: Program requires gr-analog.\n")
+ sys.exit(1)
+
class my_top_block(gr.top_block):
def __init__(self):
@@ -45,11 +52,11 @@ class my_top_block(gr.top_block):
sample_rate = int(options.sample_rate)
ampl = 0.1
- src0 = gr.sig_source_f (sample_rate, gr.GR_SIN_WAVE, 350, ampl)
- src1 = gr.sig_source_f (sample_rate, gr.GR_SIN_WAVE, 440, ampl)
+ src0 = analog.sig_source_f(sample_rate, analog.GR_SIN_WAVE, 350, ampl)
+ src1 = analog.sig_source_f(sample_rate, analog.GR_SIN_WAVE, 440, ampl)
head0 = gr.head(gr.sizeof_float, int(options.samples))
head1 = gr.head(gr.sizeof_float, int(options.samples))
- dst = gr.wavfile_sink(args[0], 2, int(options.sample_rate), 16)
+ dst = blocks.wavfile_sink(args[0], 2, int(options.sample_rate), 16)
self.connect(src0, head0, (dst, 0))
self.connect(src1, head1, (dst, 1))
diff --git a/gr-audio/examples/python/mono_tone.py b/gr-audio/examples/python/mono_tone.py
index bce486e4ab..ad73d62327 100755
--- a/gr-audio/examples/python/mono_tone.py
+++ b/gr-audio/examples/python/mono_tone.py
@@ -1,6 +1,6 @@
#!/usr/bin/env python
#
-# Copyright 2004,2005,2007 Free Software Foundation, Inc.
+# Copyright 2004,2005,2007,2012 Free Software Foundation, Inc.
#
# This file is part of GNU Radio
#
@@ -25,6 +25,12 @@ from gnuradio import audio
from gnuradio.eng_option import eng_option
from optparse import OptionParser
+try:
+ from gnuradio import analog
+except ImportError:
+ sys.stderr.write("Error: Program requires gr-analog.\n")
+ sys.exit(1)
+
#import os
#print os.getpid()
#raw_input('Attach gdb and press Enter: ')
@@ -50,11 +56,11 @@ class my_top_block(gr.top_block):
sample_rate = int(options.sample_rate)
ampl = 0.1
- src0 = gr.sig_source_f (sample_rate, gr.GR_SIN_WAVE, 650, ampl)
+ src0 = analog.sig_source_f(sample_rate, analog.GR_SIN_WAVE, 650, ampl)
- dst = audio.sink (sample_rate,
- options.audio_output,
- options.ok_to_block)
+ dst = audio.sink(sample_rate,
+ options.audio_output,
+ options.ok_to_block)
self.connect (src0, (dst, 0))
diff --git a/gr-audio/examples/python/multi_tone.py b/gr-audio/examples/python/multi_tone.py
index 00c213b634..6232cbef52 100755
--- a/gr-audio/examples/python/multi_tone.py
+++ b/gr-audio/examples/python/multi_tone.py
@@ -1,6 +1,6 @@
#!/usr/bin/env python
#
-# Copyright 2004,2006,2007 Free Software Foundation, Inc.
+# Copyright 2004,2006,2007,2012 Free Software Foundation, Inc.
#
# This file is part of GNU Radio
#
@@ -25,6 +25,12 @@ from gnuradio import audio
from gnuradio.eng_option import eng_option
from optparse import OptionParser
+try:
+ from gnuradio import analog
+except ImportError:
+ sys.stderr.write("Error: Program requires gr-analog.\n")
+ sys.exit(1)
+
#import os
#print os.getpid()
#raw_input('Attach gdb and press Enter: ')
@@ -43,7 +49,7 @@ class my_top_block(gr.top_block):
help="set maximum channels to use")
parser.add_option("-D", "--dont-block", action="store_false", default=True,
dest="ok_to_block")
- (options, args) = parser.parse_args ()
+ (options, args) = parser.parse_args()
if len(args) != 0:
parser.print_help()
raise SystemExit, 1
@@ -69,19 +75,19 @@ class my_top_block(gr.top_block):
# progression = (7, 11, 1, 5)
progression = (7, 11, 1, 5, 9)
- dst = audio.sink (sample_rate,
- options.audio_output,
- options.ok_to_block)
+ dst = audio.sink(sample_rate,
+ options.audio_output,
+ options.ok_to_block)
max_chan = dst.input_signature().max_streams()
if (max_chan == -1) or (max_chan > limit_channels):
max_chan = limit_channels
- for i in range (max_chan):
- quo, rem = divmod (i, len (progression))
+ for i in range(max_chan):
+ quo, rem = divmod(i, len (progression))
freq = base * ratios[progression[rem]] * (quo + 1)
- src = gr.sig_source_f (sample_rate, gr.GR_SIN_WAVE, freq, ampl)
- self.connect (src, (dst, i))
+ src = analog.sig_source_f(sample_rate, analog.GR_SIN_WAVE, freq, ampl)
+ self.connect(src, (dst, i))
if __name__ == '__main__':
try:
diff --git a/gr-audio/examples/python/noise.py b/gr-audio/examples/python/noise.py
index 12eee1906d..bba9e83eae 100755
--- a/gr-audio/examples/python/noise.py
+++ b/gr-audio/examples/python/noise.py
@@ -22,6 +22,7 @@
from gnuradio import gr
from gnuradio import audio
+from gnuradio import digital
from gnuradio.eng_option import eng_option
from optparse import OptionParser
@@ -44,7 +45,7 @@ class my_top_block(gr.top_block):
ampl = 0.1
src = gr.glfsr_source_b(32) # Pseudorandom noise source
- b2f = gr.chunks_to_symbols_bf([ampl, -ampl], 1)
+ b2f = digital.chunks_to_symbols_bf([ampl, -ampl], 1)
dst = audio.sink(sample_rate, options.audio_output)
self.connect(src, b2f, dst)
diff --git a/gr-audio/examples/python/spectrum_inversion.py b/gr-audio/examples/python/spectrum_inversion.py
index e152430cdb..63d0c8cc8d 100755
--- a/gr-audio/examples/python/spectrum_inversion.py
+++ b/gr-audio/examples/python/spectrum_inversion.py
@@ -28,6 +28,7 @@
from gnuradio import gr
from gnuradio import audio
+from gnuradio import blocks
from gnuradio.eng_option import eng_option
from optparse import OptionParser
@@ -54,7 +55,7 @@ class my_top_block(gr.top_block):
vec1 = [1, -1]
vsource = gr.vector_source_f(vec1, True)
- multiply = gr.multiply_ff()
+ multiply = blocks.multiply_ff()
self.connect(src, (multiply, 0))
self.connect(vsource, (multiply, 1))
diff --git a/gr-audio/examples/python/test_resampler.py b/gr-audio/examples/python/test_resampler.py
index db7f79fba0..0f5544cfa9 100755
--- a/gr-audio/examples/python/test_resampler.py
+++ b/gr-audio/examples/python/test_resampler.py
@@ -1,6 +1,6 @@
#!/usr/bin/env python
#
-# Copyright 2004,2005,2007 Free Software Foundation, Inc.
+# Copyright 2004,2005,2007,2012 Free Software Foundation, Inc.
#
# This file is part of GNU Radio
#
@@ -20,11 +20,22 @@
# Boston, MA 02110-1301, USA.
#
-from gnuradio import gr, gru, blks2
+from gnuradio import gr, gru
from gnuradio import audio
from gnuradio.eng_option import eng_option
from optparse import OptionParser
+try:
+ from gnuradio import analog
+except ImportError:
+ sys.stderr.write("Error: Program requires gr-analog.\n")
+ sys.exit(1)
+
+try:
+ from gnuradio import blocks
+except ImportError:
+ sys.stderr.write("Error: Program requires gr-blocks.\n")
+ sys.exit(1)
class my_top_block(gr.top_block):
@@ -38,7 +49,7 @@ class my_top_block(gr.top_block):
help="set input sample rate to RATE (%default)")
parser.add_option("-o", "--output-rate", type="eng_float", default=48000,
help="set output sample rate to RATE (%default)")
- (options, args) = parser.parse_args ()
+ (options, args) = parser.parse_args()
if len(args) != 0:
parser.print_help()
raise SystemExit, 1
@@ -53,11 +64,10 @@ class my_top_block(gr.top_block):
print "decim =", decim
ampl = 0.1
- src0 = gr.sig_source_f (input_rate, gr.GR_SIN_WAVE, 650, ampl)
- rr = blks2.rational_resampler_fff(interp, decim)
- dst = audio.sink (output_rate, options.audio_output)
- self.connect (src0, rr, (dst, 0))
-
+ src0 = analog.sig_source_f(input_rate, analog.GR_SIN_WAVE, 650, ampl)
+ rr = blocks.rational_resampler_fff(interp, decim)
+ dst = audio.sink(output_rate, options.audio_output)
+ self.connect(src0, rr, (dst, 0))
if __name__ == '__main__':
try:
diff --git a/gr-audio/include/CMakeLists.txt b/gr-audio/include/audio/CMakeLists.txt
index a415063734..1cfbb9bfa2 100644
--- a/gr-audio/include/CMakeLists.txt
+++ b/gr-audio/include/audio/CMakeLists.txt
@@ -1,4 +1,4 @@
-# Copyright 2011 Free Software Foundation, Inc.
+# Copyright 2011,2013 Free Software Foundation, Inc.
#
# This file is part of GNU Radio
#
@@ -21,9 +21,9 @@
# Install header files
########################################################################
install(FILES
- gr_audio_api.h
- gr_audio_source.h
- gr_audio_sink.h
- DESTINATION ${GR_INCLUDE_DIR}/gnuradio
+ api.h
+ source.h
+ sink.h
+ DESTINATION ${GR_INCLUDE_DIR}/gnuradio/audio
COMPONENT "audio_devel"
)
diff --git a/gr-audio/include/gr_audio_api.h b/gr-audio/include/audio/api.h
index 2ddd0fec6e..2ddd0fec6e 100644
--- a/gr-audio/include/gr_audio_api.h
+++ b/gr-audio/include/audio/api.h
diff --git a/gr-audio/include/audio/sink.h b/gr-audio/include/audio/sink.h
new file mode 100644
index 0000000000..d53ff6f6f1
--- /dev/null
+++ b/gr-audio/include/audio/sink.h
@@ -0,0 +1,63 @@
+/* -*- c++ -*- */
+/*
+ * Copyright 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.
+ */
+
+#ifndef INCLUDED_GR_AUDIO_SINK_H
+#define INCLUDED_GR_AUDIO_SINK_H
+
+#include <audio/api.h>
+#include <gr_sync_block.h>
+
+namespace gr {
+ namespace audio {
+
+ /*!
+ * \brief Creates a sink from an audio device.
+ * \ingroup audio_blk
+ */
+ class GR_AUDIO_API sink : virtual public gr_sync_block
+ {
+ public:
+ typedef boost::shared_ptr<sink> sptr;
+
+ /*!
+ * Creates a sink from an audio device at a specified
+ * sample_rate. The specific audio device to use can be
+ * specified as the device_name parameter. Typical choices are:
+ * \li pulse
+ * \li hw:0,0
+ * \li plughw:0,0
+ * \li surround51
+ * \li /dev/dsp
+ *
+ * \xmlonly
+ * - pulse, hw:0,0, plughw:0,0, surround51, /dev/dsp
+ * \endxmlonly
+ */
+ static sptr make(int sampling_rate,
+ const std::string device_name = "",
+ bool ok_to_block = true);
+ };
+
+ } /* namespace audio */
+} /* namespace gr */
+
+#endif /* INCLUDED_GR_AUDIO_SINK_H */
diff --git a/gr-audio/include/audio/source.h b/gr-audio/include/audio/source.h
new file mode 100644
index 0000000000..f8e21f1567
--- /dev/null
+++ b/gr-audio/include/audio/source.h
@@ -0,0 +1,63 @@
+/* -*- c++ -*- */
+/*
+ * Copyright 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.
+ */
+
+#ifndef INCLUDED_GR_AUDIO_SOURCE_H
+#define INCLUDED_GR_AUDIO_SOURCE_H
+
+#include <audio/api.h>
+#include <gr_sync_block.h>
+
+namespace gr {
+ namespace audio {
+
+ /*!
+ * \brief Creates a source from an audio device.
+ * \ingroup audio_blk
+ */
+ class GR_AUDIO_API source : virtual public gr_sync_block
+ {
+ public:
+ typedef boost::shared_ptr<source> sptr;
+
+ /*!
+ * Creates a source from an audio device at a specified
+ * sample_rate. The specific audio device to use can be
+ * specified as the device_name parameter. Typical choices are:
+ * \li pulse
+ * \li hw:0,0
+ * \li plughw:0,0
+ * \li surround51
+ * \li /dev/dsp
+ *
+ * \xmlonly
+ * - pulse, hw:0,0, plughw:0,0, surround51, /dev/dsp
+ * \endxmlonly
+ */
+ static sptr make(int sampling_rate,
+ const std::string device_name = "",
+ bool ok_to_block = true);
+ };
+
+ } /* namespace audio */
+} /* namespace gr */
+
+#endif /* INCLUDED_GR_AUDIO_SOURCE_H */
diff --git a/gr-audio/include/gr_audio_sink.h b/gr-audio/include/gr_audio_sink.h
deleted file mode 100644
index c2c8bdc308..0000000000
--- a/gr-audio/include/gr_audio_sink.h
+++ /dev/null
@@ -1,57 +0,0 @@
-/*
- * Copyright 2011 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.
- */
-
-#ifndef INCLUDED_GR_AUDIO_SINK_H
-#define INCLUDED_GR_AUDIO_SINK_H
-
-#include <gr_audio_api.h>
-#include <gr_sync_block.h>
-
-/*!
- * \brief Creates a sink from an audio device.
- * \ingroup audio_blk
- */
-class GR_AUDIO_API audio_sink : virtual public gr_sync_block{
-public:
- typedef boost::shared_ptr<audio_sink> sptr;
-};
-
-/*!
- * Creates a sink from an audio device at a specified
- * sample_rate. The specific audio device to use can be specified as
- * the device_name parameter. Typical choices are:
- * \li pulse
- * \li hw:0,0
- * \li plughw:0,0
- * \li surround51
- * \li /dev/dsp
- *
- * \xmlonly
- * - pulse, hw:0,0, plughw:0,0, surround51, /dev/dsp
- * \endxmlonly
- */
-GR_AUDIO_API audio_sink::sptr audio_make_sink(
- int sampling_rate,
- const std::string device_name = "",
- bool ok_to_block = true
-);
-
-#endif /* INCLUDED_GR_AUDIO_SINK_H */
diff --git a/gr-audio/include/gr_audio_source.h b/gr-audio/include/gr_audio_source.h
deleted file mode 100644
index 0e46ab1980..0000000000
--- a/gr-audio/include/gr_audio_source.h
+++ /dev/null
@@ -1,57 +0,0 @@
-/*
- * Copyright 2011 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.
- */
-
-#ifndef INCLUDED_GR_AUDIO_SOURCE_H
-#define INCLUDED_GR_AUDIO_SOURCE_H
-
-#include <gr_audio_api.h>
-#include <gr_sync_block.h>
-
-/*!
- * \brief Creates a source from an audio device.
- * \ingroup audio_blk
- */
-class GR_AUDIO_API audio_source : virtual public gr_sync_block{
-public:
- typedef boost::shared_ptr<audio_source> sptr;
-};
-
-/*!
- * Creates a source from an audio device at a specified
- * sample_rate. The specific audio device to use can be specified as
- * the device_name parameter. Typical choices are:
- * \li pulse
- * \li hw:0,0
- * \li plughw:0,0
- * \li surround51
- * \li /dev/dsp
- *
- * \xmlonly
- * - pulse, hw:0,0, plughw:0,0, surround51, /dev/dsp
- * \endxmlonly
- */
-GR_AUDIO_API audio_source::sptr audio_make_source(
- int sampling_rate,
- const std::string device_name = "",
- bool ok_to_block = true
-);
-
-#endif /* INCLUDED_GR_AUDIO_SOURCE_H */
diff --git a/gr-audio/lib/CMakeLists.txt b/gr-audio/lib/CMakeLists.txt
index c9e2806ebd..73977752f3 100644
--- a/gr-audio/lib/CMakeLists.txt
+++ b/gr-audio/lib/CMakeLists.txt
@@ -25,18 +25,22 @@ include_directories(
${GR_AUDIO_INCLUDE_DIRS}
${GNURADIO_CORE_INCLUDE_DIRS}
${GRUEL_INCLUDE_DIRS}
+ ${LOG4CPP_INCLUDE_DIRS}
${Boost_INCLUDE_DIRS}
)
link_directories(${Boost_LIBRARY_DIRS})
-
-include_directories(${LOG4CPP_INCLUDE_DIRS})
link_directories(${LOG4CPP_LIBRARY_DIRS})
list(APPEND gr_audio_libs gnuradio-core ${Boost_LIBRARIES} ${LOG4CPP_LIBRARIES})
-list(APPEND gr_audio_sources gr_audio_registry.cc)
+list(APPEND gr_audio_sources audio_registry.cc)
list(APPEND gr_audio_confs ${CMAKE_CURRENT_SOURCE_DIR}/gr-audio.conf)
+if(ENABLE_GR_CTRLPORT)
+ ADD_DEFINITIONS(-DGR_CTRLPORT)
+ include_directories(${ICE_INCLUDE_DIR})
+endif(ENABLE_GR_CTRLPORT)
+
########################################################################
## ALSA Support
########################################################################
@@ -47,9 +51,9 @@ if(ALSA_FOUND)
include_directories(${CMAKE_CURRENT_SOURCE_DIR}/alsa ${ALSA_INCLUDE_DIRS})
list(APPEND gr_audio_libs ${ALSA_LIBRARIES})
list(APPEND gr_audio_sources
- ${CMAKE_CURRENT_SOURCE_DIR}/alsa/gri_alsa.cc
- ${CMAKE_CURRENT_SOURCE_DIR}/alsa/audio_alsa_source.cc
- ${CMAKE_CURRENT_SOURCE_DIR}/alsa/audio_alsa_sink.cc
+ ${CMAKE_CURRENT_SOURCE_DIR}/alsa/alsa_impl.cc
+ ${CMAKE_CURRENT_SOURCE_DIR}/alsa/alsa_source.cc
+ ${CMAKE_CURRENT_SOURCE_DIR}/alsa/alsa_sink.cc
)
list(APPEND gr_audio_confs ${CMAKE_CURRENT_SOURCE_DIR}/alsa/gr-audio-alsa.conf)
@@ -64,8 +68,8 @@ if(OSS_FOUND)
include_directories(${CMAKE_CURRENT_SOURCE_DIR}/oss ${OSS_INCLUDE_DIRS})
list(APPEND gr_audio_sources
- ${CMAKE_CURRENT_SOURCE_DIR}/oss/audio_oss_source.cc
- ${CMAKE_CURRENT_SOURCE_DIR}/oss/audio_oss_sink.cc
+ ${CMAKE_CURRENT_SOURCE_DIR}/oss/oss_source.cc
+ ${CMAKE_CURRENT_SOURCE_DIR}/oss/oss_sink.cc
)
list(APPEND gr_audio_confs ${CMAKE_CURRENT_SOURCE_DIR}/oss/gr-audio-oss.conf)
@@ -83,9 +87,9 @@ if(JACK_FOUND)
list(APPEND gr_audio_libs ${JACK_LIBRARIES})
add_definitions(${JACK_DEFINITIONS})
list(APPEND gr_audio_sources
- ${CMAKE_CURRENT_SOURCE_DIR}/jack/gri_jack.cc
- ${CMAKE_CURRENT_SOURCE_DIR}/jack/audio_jack_source.cc
- ${CMAKE_CURRENT_SOURCE_DIR}/jack/audio_jack_sink.cc
+ ${CMAKE_CURRENT_SOURCE_DIR}/jack/jack_impl.cc
+ ${CMAKE_CURRENT_SOURCE_DIR}/jack/jack_source.cc
+ ${CMAKE_CURRENT_SOURCE_DIR}/jack/jack_sink.cc
)
list(APPEND gr_audio_confs ${CMAKE_CURRENT_SOURCE_DIR}/jack/gr-audio-jack.conf)
@@ -108,8 +112,8 @@ if(AUDIO_UNIT_H AND AUDIO_TOOLBOX_H)
"-framework Carbon"
)
list(APPEND gr_audio_sources
- ${CMAKE_CURRENT_SOURCE_DIR}/osx/audio_osx_source.cc
- ${CMAKE_CURRENT_SOURCE_DIR}/osx/audio_osx_sink.cc
+ ${CMAKE_CURRENT_SOURCE_DIR}/osx/osx_source.cc
+ ${CMAKE_CURRENT_SOURCE_DIR}/osx/osx_sink.cc
)
endif(AUDIO_UNIT_H AND AUDIO_TOOLBOX_H)
@@ -125,9 +129,9 @@ if(PORTAUDIO_FOUND)
list(APPEND gr_audio_libs ${PORTAUDIO_LIBRARIES})
add_definitions(${PORTAUDIO_DEFINITIONS})
list(APPEND gr_audio_sources
- ${CMAKE_CURRENT_SOURCE_DIR}/portaudio/gri_portaudio.cc
- ${CMAKE_CURRENT_SOURCE_DIR}/portaudio/audio_portaudio_source.cc
- ${CMAKE_CURRENT_SOURCE_DIR}/portaudio/audio_portaudio_sink.cc
+ ${CMAKE_CURRENT_SOURCE_DIR}/portaudio/portaudio_impl.cc
+ ${CMAKE_CURRENT_SOURCE_DIR}/portaudio/portaudio_source.cc
+ ${CMAKE_CURRENT_SOURCE_DIR}/portaudio/portaudio_sink.cc
)
list(APPEND gr_audio_confs ${CMAKE_CURRENT_SOURCE_DIR}/portaudio/gr-audio-portaudio.conf)
@@ -141,8 +145,8 @@ if(WIN32)
include_directories(${CMAKE_CURRENT_SOURCE_DIR}/windows)
list(APPEND gr_audio_libs winmm.lib)
list(APPEND gr_audio_sources
- ${CMAKE_CURRENT_SOURCE_DIR}/windows/audio_windows_source.cc
- ${CMAKE_CURRENT_SOURCE_DIR}/windows/audio_windows_sink.cc
+ ${CMAKE_CURRENT_SOURCE_DIR}/windows/windows_source.cc
+ ${CMAKE_CURRENT_SOURCE_DIR}/windows/windows_sink.cc
)
#Add Windows DLL resource file if using MSVC
diff --git a/gr-audio/lib/alsa/gri_alsa.cc b/gr-audio/lib/alsa/alsa_impl.cc
index 7bae0937d2..63a0b0de8e 100644
--- a/gr-audio/lib/alsa/gri_alsa.cc
+++ b/gr-audio/lib/alsa/alsa_impl.cc
@@ -24,7 +24,7 @@
#include "config.h"
#endif
-#include <gri_alsa.h>
+#include <alsa/alsa_impl.h>
#include <algorithm>
static snd_pcm_access_t access_types[] = {
diff --git a/gr-audio/lib/alsa/gri_alsa.h b/gr-audio/lib/alsa/alsa_impl.h
index 9c64e2c368..1340ce403a 100644
--- a/gr-audio/lib/alsa/gri_alsa.h
+++ b/gr-audio/lib/alsa/alsa_impl.h
@@ -20,8 +20,8 @@
* Boston, MA 02110-1301, USA.
*/
-#ifndef INCLUDED_GRI_ALSA_H
-#define INCLUDED_GRI_ALSA_H
+#ifndef INCLUDED_ALSA_IMPL_H
+#define INCLUDED_ALSA_IMPL_H
#include <stdio.h>
#include <alsa/asoundlib.h>
diff --git a/gr-audio/lib/alsa/alsa_sink.cc b/gr-audio/lib/alsa/alsa_sink.cc
new file mode 100644
index 0000000000..4af57105a9
--- /dev/null
+++ b/gr-audio/lib/alsa/alsa_sink.cc
@@ -0,0 +1,542 @@
+/* -*- c++ -*- */
+/*
+ * Copyright 2004-2011 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 <alsa_sink.h>
+#include <alsa_impl.h>
+#include <gr_io_signature.h>
+#include <gr_prefs.h>
+#include <stdio.h>
+#include <iostream>
+#include <stdexcept>
+
+namespace gr {
+ namespace audio {
+
+ AUDIO_REGISTER_SINK(REG_PRIO_HIGH, alsa)(int sampling_rate,
+ const std::string &device_name,
+ bool ok_to_block)
+ {
+ return sink::sptr
+ (new alsa_sink(sampling_rate, device_name, ok_to_block));
+ }
+
+ static bool CHATTY_DEBUG = false;
+
+ static snd_pcm_format_t acceptable_formats[] = {
+ // these are in our preferred order...
+ SND_PCM_FORMAT_S32,
+ SND_PCM_FORMAT_S16
+ };
+
+#define NELEMS(x) (sizeof(x)/sizeof(x[0]))
+
+ static std::string
+ default_device_name()
+ {
+ return gr_prefs::singleton()->get_string("audio_alsa", "default_output_device", "hw:0,0");
+ }
+
+ static double
+ default_period_time()
+ {
+ return std::max(0.001,
+ gr_prefs::singleton()->get_double("audio_alsa", "period_time", 0.010));
+ }
+
+ static int
+ default_nperiods()
+ {
+ return std::max(2L,
+ gr_prefs::singleton()->get_long("audio_alsa", "nperiods", 4));
+ }
+
+ // ----------------------------------------------------------------
+
+ alsa_sink::alsa_sink(int sampling_rate,
+ const std::string device_name,
+ bool ok_to_block)
+ : gr_sync_block("audio_alsa_sink",
+ gr_make_io_signature(0, 0, 0),
+ gr_make_io_signature(0, 0, 0)),
+ d_sampling_rate(sampling_rate),
+ d_device_name(device_name.empty() ? default_device_name() : device_name),
+ d_pcm_handle(0),
+ d_hw_params((snd_pcm_hw_params_t*)(new char[snd_pcm_hw_params_sizeof()])),
+ d_sw_params((snd_pcm_sw_params_t*)(new char[snd_pcm_sw_params_sizeof()])),
+ d_nperiods(default_nperiods()),
+ d_period_time_us((unsigned int)(default_period_time() * 1e6)),
+ d_period_size(0),
+ d_buffer_size_bytes(0), d_buffer(0),
+ d_worker(0), d_special_case_mono_to_stereo(false),
+ d_nunderuns(0), d_nsuspends(0), d_ok_to_block(ok_to_block)
+ {
+ CHATTY_DEBUG = gr_prefs::singleton()->get_bool("audio_alsa", "verbose", false);
+
+ int error;
+ int dir;
+
+ // open the device for playback
+ error = snd_pcm_open(&d_pcm_handle, d_device_name.c_str(),
+ SND_PCM_STREAM_PLAYBACK, 0);
+ if(ok_to_block == false)
+ snd_pcm_nonblock(d_pcm_handle, !ok_to_block);
+ if(error < 0){
+ fprintf(stderr, "audio_alsa_sink[%s]: %s\n",
+ d_device_name.c_str(), snd_strerror(error));
+ throw std::runtime_error("audio_alsa_sink");
+ }
+
+ // Fill params with a full configuration space for a PCM.
+ error = snd_pcm_hw_params_any(d_pcm_handle, d_hw_params);
+ if(error < 0)
+ bail("broken configuration for playback", error);
+
+ if(CHATTY_DEBUG)
+ gri_alsa_dump_hw_params(d_pcm_handle, d_hw_params, stdout);
+
+ // now that we know how many channels the h/w can handle, set input signature
+ unsigned int umin_chan, umax_chan;
+ snd_pcm_hw_params_get_channels_min(d_hw_params, &umin_chan);
+ snd_pcm_hw_params_get_channels_max(d_hw_params, &umax_chan);
+ int min_chan = std::min(umin_chan, 1000U);
+ int max_chan = std::min(umax_chan, 1000U);
+
+ // As a special case, if the hw's min_chan is two, we'll accept
+ // a single input and handle the duplication ourselves.
+ if(min_chan == 2) {
+ min_chan = 1;
+ d_special_case_mono_to_stereo = true;
+ }
+ set_input_signature(gr_make_io_signature(min_chan, max_chan,
+ sizeof(float)));
+
+ // fill in portions of the d_hw_params that we know now...
+
+ // Specify the access methods we implement
+ // For now, we only handle RW_INTERLEAVED...
+ snd_pcm_access_mask_t *access_mask;
+ snd_pcm_access_mask_t **access_mask_ptr = &access_mask; // FIXME: workaround for compiler warning
+ snd_pcm_access_mask_alloca(access_mask_ptr);
+ snd_pcm_access_mask_none(access_mask);
+ snd_pcm_access_mask_set(access_mask, SND_PCM_ACCESS_RW_INTERLEAVED);
+ // snd_pcm_access_mask_set(access_mask, SND_PCM_ACCESS_RW_NONINTERLEAVED);
+
+ if((error = snd_pcm_hw_params_set_access_mask(d_pcm_handle,
+ d_hw_params, access_mask)) < 0)
+ bail("failed to set access mask", error);
+
+ // set sample format
+ if(!gri_alsa_pick_acceptable_format(d_pcm_handle, d_hw_params,
+ acceptable_formats,
+ NELEMS(acceptable_formats),
+ &d_format,
+ "audio_alsa_sink",
+ CHATTY_DEBUG))
+ throw std::runtime_error("audio_alsa_sink");
+
+ // sampling rate
+ unsigned int orig_sampling_rate = d_sampling_rate;
+ if((error = snd_pcm_hw_params_set_rate_near(d_pcm_handle, d_hw_params,
+ &d_sampling_rate, 0)) < 0)
+ bail("failed to set rate near", error);
+
+ if(orig_sampling_rate != d_sampling_rate) {
+ fprintf(stderr, "audio_alsa_sink[%s]: unable to support sampling rate %d\n",
+ snd_pcm_name(d_pcm_handle), orig_sampling_rate);
+ fprintf(stderr, " card requested %d instead.\n", d_sampling_rate);
+ }
+
+ /*
+ * ALSA transfers data in units of "periods".
+ * We indirectly determine the underlying buffersize by specifying
+ * the number of periods we want (typically 4) and the length of each
+ * period in units of time (typically 1ms).
+ */
+ unsigned int min_nperiods, max_nperiods;
+ snd_pcm_hw_params_get_periods_min(d_hw_params, &min_nperiods, &dir);
+ snd_pcm_hw_params_get_periods_max(d_hw_params, &max_nperiods, &dir);
+ //fprintf(stderr, "alsa_sink: min_nperiods = %d, max_nperiods = %d\n",
+ // min_nperiods, max_nperiods);
+
+ unsigned int orig_nperiods = d_nperiods;
+ d_nperiods = std::min (std::max (min_nperiods, d_nperiods), max_nperiods);
+
+ // adjust period time so that total buffering remains more-or-less constant
+ d_period_time_us = (d_period_time_us * orig_nperiods) / d_nperiods;
+
+ error = snd_pcm_hw_params_set_periods(d_pcm_handle, d_hw_params,
+ d_nperiods, 0);
+ if(error < 0)
+ bail("set_periods failed", error);
+
+ dir = 0;
+ error = snd_pcm_hw_params_set_period_time_near(d_pcm_handle, d_hw_params,
+ &d_period_time_us, &dir);
+ if(error < 0)
+ bail("set_period_time_near failed", error);
+
+ dir = 0;
+ error = snd_pcm_hw_params_get_period_size(d_hw_params,
+ &d_period_size, &dir);
+ if(error < 0)
+ bail("get_period_size failed", error);
+
+ set_output_multiple(d_period_size);
+ }
+
+ bool
+ alsa_sink::check_topology(int ninputs, int noutputs)
+ {
+ // ninputs is how many channels the user has connected.
+ // Now we can finish up setting up the hw params...
+
+ int nchan = ninputs;
+ int err;
+
+ // Check the state of the stream
+ // Ensure that the pcm is in a state where we can still mess with the hw_params
+ snd_pcm_state_t state;
+ state = snd_pcm_state(d_pcm_handle);
+ if(state == SND_PCM_STATE_RUNNING)
+ return true; // If stream is running, don't change any parameters
+ else if(state == SND_PCM_STATE_XRUN)
+ snd_pcm_prepare(d_pcm_handle); // Prepare stream on underrun, and we can set parameters;
+
+ bool special_case = nchan == 1 && d_special_case_mono_to_stereo;
+ if(special_case)
+ nchan = 2;
+
+ err = snd_pcm_hw_params_set_channels(d_pcm_handle, d_hw_params, nchan);
+
+ if(err < 0) {
+ output_error_msg("set_channels failed", err);
+ return false;
+ }
+
+ // set the parameters into the driver...
+ err = snd_pcm_hw_params(d_pcm_handle, d_hw_params);
+ if(err < 0) {
+ output_error_msg("snd_pcm_hw_params failed", err);
+ return false;
+ }
+
+ // get current s/w params
+ err = snd_pcm_sw_params_current(d_pcm_handle, d_sw_params);
+ if(err < 0)
+ bail("snd_pcm_sw_params_current", err);
+
+ // Tell the PCM device to wait to start until we've filled
+ // it's buffers half way full. This helps avoid audio underruns.
+
+ err = snd_pcm_sw_params_set_start_threshold(d_pcm_handle,
+ d_sw_params,
+ d_nperiods * d_period_size / 2);
+ if(err < 0)
+ bail("snd_pcm_sw_params_set_start_threshold", err);
+
+ // store the s/w params
+ err = snd_pcm_sw_params(d_pcm_handle, d_sw_params);
+ if(err < 0)
+ bail("snd_pcm_sw_params", err);
+
+ d_buffer_size_bytes =
+ d_period_size * nchan * snd_pcm_format_size(d_format, 1);
+
+ d_buffer = new char[d_buffer_size_bytes];
+
+ if(CHATTY_DEBUG)
+ fprintf(stdout, "audio_alsa_sink[%s]: sample resolution = %d bits\n",
+ snd_pcm_name(d_pcm_handle),
+ snd_pcm_hw_params_get_sbits(d_hw_params));
+
+ switch(d_format) {
+ case SND_PCM_FORMAT_S16:
+ if(special_case)
+ d_worker = &alsa_sink::work_s16_1x2;
+ else
+ d_worker = &alsa_sink::work_s16;
+ break;
+
+ case SND_PCM_FORMAT_S32:
+ if(special_case)
+ d_worker = &alsa_sink::work_s32_1x2;
+ else
+ d_worker = &alsa_sink::work_s32;
+ break;
+
+ default:
+ assert(0);
+ }
+ return true;
+ }
+
+ alsa_sink::~alsa_sink()
+ {
+ if(snd_pcm_state(d_pcm_handle) == SND_PCM_STATE_RUNNING)
+ snd_pcm_drop(d_pcm_handle);
+
+ snd_pcm_close(d_pcm_handle);
+ delete [] ((char*)d_hw_params);
+ delete [] ((char*)d_sw_params);
+ delete [] d_buffer;
+ }
+
+ int
+ alsa_sink::work(int noutput_items,
+ gr_vector_const_void_star &input_items,
+ gr_vector_void_star &output_items)
+ {
+ assert((noutput_items % d_period_size) == 0);
+
+ // this is a call through a pointer to a method...
+ return (this->*d_worker)(noutput_items, input_items, output_items);
+ }
+
+ /*
+ * Work function that deals with float to S16 conversion
+ */
+ int
+ alsa_sink::work_s16(int noutput_items,
+ gr_vector_const_void_star &input_items,
+ gr_vector_void_star &output_items)
+ {
+ typedef gr_int16 sample_t; // the type of samples we're creating
+ static const float scale_factor = std::pow(2.0f, 16-1) - 1;
+
+ unsigned int nchan = input_items.size();
+ const float **in = (const float **)&input_items[0];
+ sample_t *buf = (sample_t *)d_buffer;
+ int bi;
+ int n;
+
+ unsigned int sizeof_frame = nchan * sizeof(sample_t);
+ assert(d_buffer_size_bytes == d_period_size * sizeof_frame);
+
+ for(n = 0; n < noutput_items; n += d_period_size) {
+ // process one period of data
+ bi = 0;
+ for(unsigned int i = 0; i < d_period_size; i++) {
+ for (unsigned int chan = 0; chan < nchan; chan++) {
+ buf[bi++] = (sample_t) (in[chan][i] * scale_factor);
+ }
+ }
+
+ // update src pointers
+ for(unsigned int chan = 0; chan < nchan; chan++)
+ in[chan] += d_period_size;
+
+ if(!write_buffer (buf, d_period_size, sizeof_frame))
+ return -1; // No fixing this problem. Say we're done.
+ }
+
+ return n;
+ }
+
+ /*
+ * Work function that deals with float to S32 conversion
+ */
+ int
+ alsa_sink::work_s32(int noutput_items,
+ gr_vector_const_void_star &input_items,
+ gr_vector_void_star &output_items)
+ {
+ typedef gr_int32 sample_t; // the type of samples we're creating
+ static const float scale_factor = std::pow(2.0f, 32-1) - 1;
+
+ unsigned int nchan = input_items.size();
+ const float **in = (const float **)&input_items[0];
+ sample_t *buf = (sample_t *)d_buffer;
+ int bi;
+ int n;
+
+ unsigned int sizeof_frame = nchan * sizeof (sample_t);
+ assert(d_buffer_size_bytes == d_period_size * sizeof_frame);
+
+ for(n = 0; n < noutput_items; n += d_period_size) {
+ // process one period of data
+ bi = 0;
+ for(unsigned int i = 0; i < d_period_size; i++) {
+ for(unsigned int chan = 0; chan < nchan; chan++) {
+ buf[bi++] = (sample_t)(in[chan][i] * scale_factor);
+ }
+ }
+
+ // update src pointers
+ for(unsigned int chan = 0; chan < nchan; chan++)
+ in[chan] += d_period_size;
+
+ if(!write_buffer (buf, d_period_size, sizeof_frame))
+ return -1; // No fixing this problem. Say we're done.
+ }
+
+ return n;
+ }
+
+ /*
+ * Work function that deals with float to S16 conversion and
+ * mono to stereo kludge.
+ */
+ int
+ alsa_sink::work_s16_1x2(int noutput_items,
+ gr_vector_const_void_star &input_items,
+ gr_vector_void_star &output_items)
+ {
+ typedef gr_int16 sample_t; // the type of samples we're creating
+ static const float scale_factor = std::pow(2.0f, 16-1) - 1;
+
+ assert(input_items.size () == 1);
+ static const unsigned int nchan = 2;
+ const float **in = (const float **)&input_items[0];
+ sample_t *buf = (sample_t *)d_buffer;
+ int bi;
+ int n;
+
+ unsigned int sizeof_frame = nchan * sizeof(sample_t);
+ assert(d_buffer_size_bytes == d_period_size * sizeof_frame);
+
+ for(n = 0; n < noutput_items; n += d_period_size) {
+ // process one period of data
+ bi = 0;
+ for(unsigned int i = 0; i < d_period_size; i++) {
+ sample_t t = (sample_t) (in[0][i] * scale_factor);
+ buf[bi++] = t;
+ buf[bi++] = t;
+ }
+
+ // update src pointers
+ in[0] += d_period_size;
+
+ if(!write_buffer (buf, d_period_size, sizeof_frame))
+ return -1; // No fixing this problem. Say we're done.
+ }
+
+ return n;
+ }
+
+ /*
+ * Work function that deals with float to S32 conversion and
+ * mono to stereo kludge.
+ */
+ int
+ alsa_sink::work_s32_1x2(int noutput_items,
+ gr_vector_const_void_star &input_items,
+ gr_vector_void_star &output_items)
+ {
+ typedef gr_int32 sample_t; // the type of samples we're creating
+ static const float scale_factor = std::pow(2.0f, 32-1) - 1;
+
+ assert(input_items.size () == 1);
+ static unsigned int nchan = 2;
+ const float **in = (const float **)&input_items[0];
+ sample_t *buf = (sample_t*)d_buffer;
+ int bi;
+ int n;
+
+ unsigned int sizeof_frame = nchan * sizeof(sample_t);
+ assert(d_buffer_size_bytes == d_period_size * sizeof_frame);
+
+ for(n = 0; n < noutput_items; n += d_period_size) {
+ // process one period of data
+ bi = 0;
+ for(unsigned int i = 0; i < d_period_size; i++) {
+ sample_t t = (sample_t)(in[0][i] * scale_factor);
+ buf[bi++] = t;
+ buf[bi++] = t;
+ }
+
+ // update src pointers
+ in[0] += d_period_size;
+
+ if(!write_buffer (buf, d_period_size, sizeof_frame))
+ return -1; // No fixing this problem. Say we're done.
+ }
+
+ return n;
+ }
+
+ bool
+ alsa_sink::write_buffer(const void *vbuffer,
+ unsigned nframes, unsigned sizeof_frame)
+ {
+ const unsigned char *buffer = (const unsigned char *)vbuffer;
+
+ while(nframes > 0){
+ int r = snd_pcm_writei(d_pcm_handle, buffer, nframes);
+ if(r == -EAGAIN) {
+ if(d_ok_to_block == true)
+ continue; // try again
+ break;
+ }
+
+ else if(r == -EPIPE) { // underrun
+ d_nunderuns++;
+ fputs("aU", stderr);
+ if((r = snd_pcm_prepare (d_pcm_handle)) < 0){
+ output_error_msg("snd_pcm_prepare failed. Can't recover from underrun", r);
+ return false;
+ }
+ continue; // try again
+ }
+
+ else if(r == -ESTRPIPE) { // h/w is suspended (whatever that means)
+ // This is apparently related to power management
+ d_nsuspends++;
+ if((r = snd_pcm_resume (d_pcm_handle)) < 0) {
+ output_error_msg("failed to resume from suspend", r);
+ return false;
+ }
+ continue; // try again
+ }
+
+ else if (r < 0) {
+ output_error_msg("snd_pcm_writei failed", r);
+ return false;
+ }
+
+ nframes -= r;
+ buffer += r * sizeof_frame;
+ }
+
+ return true;
+ }
+
+ void
+ alsa_sink::output_error_msg (const char *msg, int err)
+ {
+ fprintf(stderr, "audio_alsa_sink[%s]: %s: %s\n",
+ snd_pcm_name(d_pcm_handle), msg, snd_strerror(err));
+ }
+
+ void
+ alsa_sink::bail(const char *msg, int err) throw (std::runtime_error)
+ {
+ output_error_msg(msg, err);
+ throw std::runtime_error("audio_alsa_sink");
+ }
+
+ } /* namespace audio */
+} /* namespace gr */
diff --git a/gr-audio/lib/alsa/alsa_sink.h b/gr-audio/lib/alsa/alsa_sink.h
new file mode 100644
index 0000000000..1dea62f56e
--- /dev/null
+++ b/gr-audio/lib/alsa/alsa_sink.h
@@ -0,0 +1,111 @@
+/* -*- 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.
+ */
+
+#ifndef INCLUDED_AUDIO_ALSA_SINK_H
+#define INCLUDED_AUDIO_ALSA_SINK_H
+
+// use new ALSA API
+#define ALSA_PCM_NEW_HW_PARAMS_API
+#define ALSA_PCM_NEW_SW_PARAMS_API
+
+#include <audio/sink.h>
+#include <alsa/asoundlib.h>
+#include <string>
+#include <stdexcept>
+
+namespace gr {
+ namespace audio {
+
+ /*!
+ * \brief audio sink using ALSA
+ * \ingroup audio_blk
+ *
+ * The sink has N input streams of floats, where N depends
+ * on the hardware characteristics of the selected device.
+ *
+ * Input samples must be in the range [-1,1].
+ */
+ class alsa_sink : public sink
+ {
+ // typedef for pointer to class work method
+ typedef int(alsa_sink::*work_t)(int noutput_items,
+ gr_vector_const_void_star &input_items,
+ gr_vector_void_star &output_items);
+
+ unsigned int d_sampling_rate;
+ std::string d_device_name;
+ snd_pcm_t *d_pcm_handle;
+ snd_pcm_hw_params_t *d_hw_params;
+ snd_pcm_sw_params_t *d_sw_params;
+ snd_pcm_format_t d_format;
+ unsigned int d_nperiods;
+ unsigned int d_period_time_us; // microseconds
+ snd_pcm_uframes_t d_period_size; // in frames
+ unsigned int d_buffer_size_bytes; // sizeof of d_buffer
+ char *d_buffer;
+ work_t d_worker; // the work method to use
+ bool d_special_case_mono_to_stereo;
+
+ // random stats
+ int d_nunderuns; // count of underruns
+ int d_nsuspends; // count of suspends
+ bool d_ok_to_block; // defaults to "true", controls blocking/non-block I/O
+
+ void output_error_msg(const char *msg, int err);
+ void bail(const char *msg, int err) throw (std::runtime_error);
+
+ public:
+ alsa_sink(int sampling_rate,
+ const std::string device_name,
+ bool ok_to_block);
+ ~alsa_sink();
+
+ bool check_topology(int ninputs, int noutputs);
+
+ int work(int noutput_items,
+ gr_vector_const_void_star &input_items,
+ gr_vector_void_star &output_items);
+
+ protected:
+ bool write_buffer(const void *buffer, unsigned nframes, unsigned sizeof_frame);
+
+ int work_s16(int noutput_items,
+ gr_vector_const_void_star &input_items,
+ gr_vector_void_star &output_items);
+
+ int work_s16_1x2(int noutput_items,
+ gr_vector_const_void_star &input_items,
+ gr_vector_void_star &output_items);
+
+ int work_s32(int noutput_items,
+ gr_vector_const_void_star &input_items,
+ gr_vector_void_star &output_items);
+
+ int work_s32_1x2(int noutput_items,
+ gr_vector_const_void_star &input_items,
+ gr_vector_void_star &output_items);
+ };
+
+ } /* namespace audio */
+} /* namespace gr */
+
+#endif /* INCLUDED_AUDIO_ALSA_SINK_H */
diff --git a/gr-audio/lib/alsa/alsa_source.cc b/gr-audio/lib/alsa/alsa_source.cc
new file mode 100644
index 0000000000..838da60af3
--- /dev/null
+++ b/gr-audio/lib/alsa/alsa_source.cc
@@ -0,0 +1,514 @@
+/* -*- 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 <alsa_source.h>
+#include <alsa_impl.h>
+#include <gr_io_signature.h>
+#include <gr_prefs.h>
+#include <stdio.h>
+#include <iostream>
+#include <stdexcept>
+
+namespace gr {
+ namespace audio {
+
+ AUDIO_REGISTER_SOURCE(REG_PRIO_HIGH, alsa)(int sampling_rate,
+ const std::string &device_name,
+ bool ok_to_block)
+ {
+ return source::sptr
+ (new alsa_source(sampling_rate, device_name, ok_to_block));
+ }
+
+ static bool CHATTY_DEBUG = false;
+
+ static snd_pcm_format_t acceptable_formats[] = {
+ // these are in our preferred order...
+ SND_PCM_FORMAT_S32,
+ SND_PCM_FORMAT_S16
+ };
+
+#define NELEMS(x) (sizeof(x)/sizeof(x[0]))
+
+ static std::string
+ default_device_name()
+ {
+ return gr_prefs::singleton()->get_string("audio_alsa",
+ "default_input_device",
+ "hw:0,0");
+ }
+
+ static double
+ default_period_time()
+ {
+ return std::max(0.001,
+ gr_prefs::singleton()->get_double("audio_alsa", "period_time", 0.010));
+ }
+
+ static int
+ default_nperiods()
+ {
+ return std::max(2L,
+ gr_prefs::singleton()->get_long("audio_alsa", "nperiods", 4));
+ }
+
+ // ----------------------------------------------------------------
+
+ alsa_source::alsa_source(int sampling_rate,
+ const std::string device_name,
+ bool ok_to_block)
+ : gr_sync_block("audio_alsa_source",
+ gr_make_io_signature(0, 0, 0),
+ gr_make_io_signature(0, 0, 0)),
+ d_sampling_rate(sampling_rate),
+ d_device_name(device_name.empty() ? default_device_name() : device_name),
+ d_pcm_handle(0),
+ d_hw_params((snd_pcm_hw_params_t*)(new char[snd_pcm_hw_params_sizeof()])),
+ d_sw_params((snd_pcm_sw_params_t*)(new char[snd_pcm_sw_params_sizeof()])),
+ d_nperiods(default_nperiods()),
+ d_period_time_us((unsigned int)(default_period_time() * 1e6)),
+ d_period_size(0),
+ d_buffer_size_bytes(0), d_buffer(0),
+ d_worker(0), d_hw_nchan(0),
+ d_special_case_stereo_to_mono(false),
+ d_noverruns(0), d_nsuspends(0)
+ {
+ CHATTY_DEBUG = gr_prefs::singleton()->get_bool("audio_alsa", "verbose", false);
+
+ int error;
+ int dir;
+
+ // open the device for capture
+ error = snd_pcm_open(&d_pcm_handle, d_device_name.c_str(),
+ SND_PCM_STREAM_CAPTURE, 0);
+ if(error < 0){
+ fprintf(stderr, "audio_alsa_source[%s]: %s\n",
+ d_device_name.c_str(), snd_strerror(error));
+ throw std::runtime_error("audio_alsa_source");
+ }
+
+ // Fill params with a full configuration space for a PCM.
+ error = snd_pcm_hw_params_any(d_pcm_handle, d_hw_params);
+ if(error < 0)
+ bail("broken configuration for playback", error);
+
+ if(CHATTY_DEBUG)
+ gri_alsa_dump_hw_params(d_pcm_handle, d_hw_params, stdout);
+
+ // now that we know how many channels the h/w can handle, set output signature
+ unsigned int umax_chan;
+ unsigned int umin_chan;
+ snd_pcm_hw_params_get_channels_min(d_hw_params, &umin_chan);
+ snd_pcm_hw_params_get_channels_max(d_hw_params, &umax_chan);
+ int min_chan = std::min(umin_chan, 1000U);
+ int max_chan = std::min(umax_chan, 1000U);
+
+ // As a special case, if the hw's min_chan is two, we'll accept
+ // a single output and handle the demux ourselves.
+ if(min_chan == 2) {
+ min_chan = 1;
+ d_special_case_stereo_to_mono = true;
+ }
+
+ set_output_signature(gr_make_io_signature(min_chan, max_chan,
+ sizeof(float)));
+
+ // fill in portions of the d_hw_params that we know now...
+
+ // Specify the access methods we implement
+ // For now, we only handle RW_INTERLEAVED...
+ snd_pcm_access_mask_t *access_mask;
+ snd_pcm_access_mask_t **access_mask_ptr = &access_mask; // FIXME: workaround for compiler warning
+ snd_pcm_access_mask_alloca(access_mask_ptr);
+ snd_pcm_access_mask_none(access_mask);
+ snd_pcm_access_mask_set(access_mask, SND_PCM_ACCESS_RW_INTERLEAVED);
+ // snd_pcm_access_mask_set(access_mask, SND_PCM_ACCESS_RW_NONINTERLEAVED);
+
+ if((error = snd_pcm_hw_params_set_access_mask(d_pcm_handle,
+ d_hw_params, access_mask)) < 0)
+ bail("failed to set access mask", error);
+
+ // set sample format
+ if(!gri_alsa_pick_acceptable_format(d_pcm_handle, d_hw_params,
+ acceptable_formats,
+ NELEMS(acceptable_formats),
+ &d_format,
+ "audio_alsa_source",
+ CHATTY_DEBUG))
+ throw std::runtime_error("audio_alsa_source");
+
+ // sampling rate
+ unsigned int orig_sampling_rate = d_sampling_rate;
+ if((error = snd_pcm_hw_params_set_rate_near(d_pcm_handle, d_hw_params,
+ &d_sampling_rate, 0)) < 0)
+ bail("failed to set rate near", error);
+
+ if(orig_sampling_rate != d_sampling_rate){
+ fprintf(stderr, "audio_alsa_source[%s]: unable to support sampling rate %d\n",
+ snd_pcm_name (d_pcm_handle), orig_sampling_rate);
+ fprintf(stderr, " card requested %d instead.\n", d_sampling_rate);
+ }
+
+ /*
+ * ALSA transfers data in units of "periods".
+ * We indirectly determine the underlying buffersize by specifying
+ * the number of periods we want (typically 4) and the length of each
+ * period in units of time (typically 1ms).
+ */
+ unsigned int min_nperiods, max_nperiods;
+ snd_pcm_hw_params_get_periods_min(d_hw_params, &min_nperiods, &dir);
+ snd_pcm_hw_params_get_periods_max(d_hw_params, &max_nperiods, &dir);
+ //fprintf (stderr, "alsa_source: min_nperiods = %d, max_nperiods = %d\n",
+ // min_nperiods, max_nperiods);
+
+ unsigned int orig_nperiods = d_nperiods;
+ d_nperiods = std::min(std::max (min_nperiods, d_nperiods), max_nperiods);
+
+ // adjust period time so that total buffering remains more-or-less constant
+ d_period_time_us = (d_period_time_us * orig_nperiods) / d_nperiods;
+
+ error = snd_pcm_hw_params_set_periods(d_pcm_handle, d_hw_params,
+ d_nperiods, 0);
+ if(error < 0)
+ bail("set_periods failed", error);
+
+ dir = 0;
+ error = snd_pcm_hw_params_set_period_time_near(d_pcm_handle, d_hw_params,
+ &d_period_time_us, &dir);
+ if(error < 0)
+ bail("set_period_time_near failed", error);
+
+ dir = 0;
+ error = snd_pcm_hw_params_get_period_size(d_hw_params,
+ &d_period_size, &dir);
+ if(error < 0)
+ bail("get_period_size failed", error);
+
+ set_output_multiple(d_period_size);
+ }
+
+ bool
+ alsa_source::check_topology(int ninputs, int noutputs)
+ {
+ // noutputs is how many channels the user has connected.
+ // Now we can finish up setting up the hw params...
+
+ unsigned int nchan = noutputs;
+ int err;
+
+ // Check the state of the stream
+ // Ensure that the pcm is in a state where we can still mess with the hw_params
+ snd_pcm_state_t state;
+ state=snd_pcm_state(d_pcm_handle);
+ if(state== SND_PCM_STATE_RUNNING)
+ return true; // If stream is running, don't change any parameters
+ else if(state == SND_PCM_STATE_XRUN)
+ snd_pcm_prepare(d_pcm_handle); // Prepare stream on underrun, and we can set parameters;
+
+ bool special_case = nchan == 1 && d_special_case_stereo_to_mono;
+ if(special_case)
+ nchan = 2;
+
+ d_hw_nchan = nchan;
+ err = snd_pcm_hw_params_set_channels(d_pcm_handle, d_hw_params, d_hw_nchan);
+ if(err < 0) {
+ output_error_msg("set_channels failed", err);
+ return false;
+ }
+
+ // set the parameters into the driver...
+ err = snd_pcm_hw_params(d_pcm_handle, d_hw_params);
+ if(err < 0) {
+ output_error_msg("snd_pcm_hw_params failed", err);
+ return false;
+ }
+
+ d_buffer_size_bytes =
+ d_period_size * d_hw_nchan * snd_pcm_format_size(d_format, 1);
+
+ d_buffer = new char[d_buffer_size_bytes];
+
+ if(CHATTY_DEBUG) {
+ fprintf(stdout, "audio_alsa_source[%s]: sample resolution = %d bits\n",
+ snd_pcm_name(d_pcm_handle),
+ snd_pcm_hw_params_get_sbits(d_hw_params));
+ }
+
+ switch(d_format) {
+ case SND_PCM_FORMAT_S16:
+ if(special_case)
+ d_worker = &alsa_source::work_s16_2x1;
+ else
+ d_worker = &alsa_source::work_s16;
+ break;
+
+ case SND_PCM_FORMAT_S32:
+ if(special_case)
+ d_worker = &alsa_source::work_s32_2x1;
+ else
+ d_worker = &alsa_source::work_s32;
+ break;
+
+ default:
+ assert(0);
+ }
+
+ return true;
+ }
+
+ alsa_source::~alsa_source()
+ {
+ if(snd_pcm_state(d_pcm_handle) == SND_PCM_STATE_RUNNING)
+ snd_pcm_drop(d_pcm_handle);
+
+ snd_pcm_close(d_pcm_handle);
+ delete [] ((char*)d_hw_params);
+ delete [] ((char*)d_sw_params);
+ delete [] d_buffer;
+ }
+
+ int
+ alsa_source::work(int noutput_items,
+ gr_vector_const_void_star &input_items,
+ gr_vector_void_star &output_items)
+ {
+ assert((noutput_items % d_period_size) == 0);
+ assert(noutput_items != 0);
+
+ // this is a call through a pointer to a method...
+ return (this->*d_worker)(noutput_items, input_items, output_items);
+ }
+
+ /*
+ * Work function that deals with float to S16 conversion
+ */
+ int
+ alsa_source::work_s16(int noutput_items,
+ gr_vector_const_void_star &input_items,
+ gr_vector_void_star &output_items)
+ {
+ typedef gr_int16 sample_t; // the type of samples we're creating
+ static const float scale_factor = 1.0 / std::pow(2.0f, 16-1);
+
+ unsigned int nchan = output_items.size ();
+ float **out = (float **)&output_items[0];
+ sample_t *buf = (sample_t *)d_buffer;
+ int bi;
+
+ unsigned int sizeof_frame = d_hw_nchan * sizeof (sample_t);
+ assert(d_buffer_size_bytes == d_period_size * sizeof_frame);
+
+ // To minimize latency, return at most a single period's worth of samples.
+ // [We could also read the first one in a blocking mode and subsequent
+ // ones in non-blocking mode, but we'll leave that for later (or never).]
+
+ if(!read_buffer(buf, d_period_size, sizeof_frame))
+ return -1; // No fixing this problem. Say we're done.
+
+ // process one period of data
+ bi = 0;
+ for(unsigned int i = 0; i < d_period_size; i++) {
+ for(unsigned int chan = 0; chan < nchan; chan++) {
+ out[chan][i] = (float) buf[bi++] * scale_factor;
+ }
+ }
+
+ return d_period_size;
+ }
+
+ /*
+ * Work function that deals with float to S16 conversion
+ * and stereo to mono kludge...
+ */
+ int
+ alsa_source::work_s16_2x1(int noutput_items,
+ gr_vector_const_void_star &input_items,
+ gr_vector_void_star &output_items)
+ {
+ typedef gr_int16 sample_t; // the type of samples we're creating
+ static const float scale_factor = 1.0 / std::pow(2.0f, 16-1);
+
+ float **out = (float**)&output_items[0];
+ sample_t *buf = (sample_t*)d_buffer;
+ int bi;
+
+ assert(output_items.size () == 1);
+
+ unsigned int sizeof_frame = d_hw_nchan * sizeof(sample_t);
+ assert(d_buffer_size_bytes == d_period_size * sizeof_frame);
+
+ // To minimize latency, return at most a single period's worth of samples.
+ // [We could also read the first one in a blocking mode and subsequent
+ // ones in non-blocking mode, but we'll leave that for later (or never).]
+ if(!read_buffer (buf, d_period_size, sizeof_frame))
+ return -1; // No fixing this problem. Say we're done.
+
+ // process one period of data
+ bi = 0;
+ for(unsigned int i = 0; i < d_period_size; i++) {
+ int t = (buf[bi] + buf[bi+1]) / 2;
+ bi += 2;
+ out[0][i] = (float) t * scale_factor;
+ }
+
+ return d_period_size;
+ }
+
+ /*
+ * Work function that deals with float to S32 conversion
+ */
+ int
+ alsa_source::work_s32(int noutput_items,
+ gr_vector_const_void_star &input_items,
+ gr_vector_void_star &output_items)
+ {
+ typedef gr_int32 sample_t; // the type of samples we're creating
+ static const float scale_factor = 1.0 / std::pow(2.0f, 32-1);
+
+ unsigned int nchan = output_items.size ();
+ float **out = (float**)&output_items[0];
+ sample_t *buf = (sample_t*)d_buffer;
+ int bi;
+
+ unsigned int sizeof_frame = d_hw_nchan * sizeof(sample_t);
+ assert(d_buffer_size_bytes == d_period_size * sizeof_frame);
+
+ // To minimize latency, return at most a single period's worth of samples.
+ // [We could also read the first one in a blocking mode and subsequent
+ // ones in non-blocking mode, but we'll leave that for later (or never).]
+
+ if(!read_buffer(buf, d_period_size, sizeof_frame))
+ return -1; // No fixing this problem. Say we're done.
+
+ // process one period of data
+ bi = 0;
+ for(unsigned int i = 0; i < d_period_size; i++) {
+ for(unsigned int chan = 0; chan < nchan; chan++) {
+ out[chan][i] = (float) buf[bi++] * scale_factor;
+ }
+ }
+
+ return d_period_size;
+ }
+
+ /*
+ * Work function that deals with float to S32 conversion
+ * and stereo to mono kludge...
+ */
+ int
+ alsa_source::work_s32_2x1(int noutput_items,
+ gr_vector_const_void_star &input_items,
+ gr_vector_void_star &output_items)
+ {
+ typedef gr_int32 sample_t; // the type of samples we're creating
+ static const float scale_factor = 1.0 / std::pow(2.0f, 32-1);
+
+ float **out = (float**)&output_items[0];
+ sample_t *buf = (sample_t*)d_buffer;
+ int bi;
+
+ assert(output_items.size () == 1);
+
+ unsigned int sizeof_frame = d_hw_nchan * sizeof(sample_t);
+ assert(d_buffer_size_bytes == d_period_size * sizeof_frame);
+
+ // To minimize latency, return at most a single period's worth of samples.
+ // [We could also read the first one in a blocking mode and subsequent
+ // ones in non-blocking mode, but we'll leave that for later (or never).]
+
+ if(!read_buffer(buf, d_period_size, sizeof_frame))
+ return -1; // No fixing this problem. Say we're done.
+
+ // process one period of data
+ bi = 0;
+ for(unsigned int i = 0; i < d_period_size; i++) {
+ int t = (buf[bi] + buf[bi+1]) / 2;
+ bi += 2;
+ out[0][i] = (float)t * scale_factor;
+ }
+
+ return d_period_size;
+ }
+
+ bool
+ alsa_source::read_buffer(void *vbuffer, unsigned nframes, unsigned sizeof_frame)
+ {
+ unsigned char *buffer = (unsigned char*)vbuffer;
+
+ while(nframes > 0) {
+ int r = snd_pcm_readi (d_pcm_handle, buffer, nframes);
+ if(r == -EAGAIN)
+ continue; // try again
+
+ else if(r == -EPIPE) { // overrun
+ d_noverruns++;
+ fputs("aO", stderr);
+ if((r = snd_pcm_prepare (d_pcm_handle)) < 0) {
+ output_error_msg("snd_pcm_prepare failed. Can't recover from overrun", r);
+ return false;
+ }
+ continue; // try again
+ }
+
+ else if(r == -ESTRPIPE) { // h/w is suspended (whatever that means)
+ // This is apparently related to power management
+ d_nsuspends++;
+ if((r = snd_pcm_resume (d_pcm_handle)) < 0) {
+ output_error_msg ("failed to resume from suspend", r);
+ return false;
+ }
+ continue; // try again
+ }
+
+ else if(r < 0) {
+ output_error_msg("snd_pcm_readi failed", r);
+ return false;
+ }
+
+ nframes -= r;
+ buffer += r * sizeof_frame;
+ }
+
+ return true;
+ }
+
+ void
+ alsa_source::output_error_msg(const char *msg, int err)
+ {
+ fprintf(stderr, "audio_alsa_source[%s]: %s: %s\n",
+ snd_pcm_name(d_pcm_handle), msg, snd_strerror (err));
+ }
+
+ void
+ alsa_source::bail(const char *msg, int err) throw (std::runtime_error)
+ {
+ output_error_msg(msg, err);
+ throw std::runtime_error("audio_alsa_source");
+ }
+
+ } /* namespace audio */
+} /* namespace gr */
diff --git a/gr-audio/lib/alsa/alsa_source.h b/gr-audio/lib/alsa/alsa_source.h
new file mode 100644
index 0000000000..6314fc1376
--- /dev/null
+++ b/gr-audio/lib/alsa/alsa_source.h
@@ -0,0 +1,114 @@
+/* -*- 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.
+ */
+
+#ifndef INCLUDED_AUDIO_ALSA_SOURCE_H
+#define INCLUDED_AUDIO_ALSA_SOURCE_H
+
+// use new ALSA API
+#define ALSA_PCM_NEW_HW_PARAMS_API
+#define ALSA_PCM_NEW_SW_PARAMS_API
+
+#include <audio/source.h>
+#include <alsa/asoundlib.h>
+#include <string>
+#include <stdexcept>
+
+namespace gr {
+ namespace audio {
+
+ class alsa_source;
+ typedef boost::shared_ptr<alsa_source> alsa_source_sptr;
+
+ /*!
+ * \brief audio source using ALSA
+ * \ingroup audio_blk
+ *
+ * The source has between 1 and N input streams of floats, where N is
+ * depends on the hardware characteristics of the selected device.
+ *
+ * Output samples will be in the range [-1,1].
+ */
+ class alsa_source : public source
+ {
+ // typedef for pointer to class work method
+ typedef int(alsa_source::*work_t)(int noutput_items,
+ gr_vector_const_void_star &input_items,
+ gr_vector_void_star &output_items);
+
+ unsigned int d_sampling_rate;
+ std::string d_device_name;
+ snd_pcm_t *d_pcm_handle;
+ snd_pcm_hw_params_t *d_hw_params;
+ snd_pcm_sw_params_t *d_sw_params;
+ snd_pcm_format_t d_format;
+ unsigned int d_nperiods;
+ unsigned int d_period_time_us; // microseconds
+ snd_pcm_uframes_t d_period_size; // in frames
+ unsigned int d_buffer_size_bytes; // sizeof of d_buffer
+ char *d_buffer;
+ work_t d_worker; // the work method to use
+ unsigned int d_hw_nchan; // # of configured h/w channels
+ bool d_special_case_stereo_to_mono;
+
+ // random stats
+ int d_noverruns; // count of overruns
+ int d_nsuspends; // count of suspends
+
+ void output_error_msg(const char *msg, int err);
+ void bail(const char *msg, int err) throw (std::runtime_error);
+
+ public:
+ alsa_source(int sampling_rate,
+ const std::string device_name,
+ bool ok_to_block);
+ ~alsa_source();
+
+ bool check_topology(int ninputs, int noutputs);
+
+ int work(int noutput_items,
+ gr_vector_const_void_star &input_items,
+ gr_vector_void_star &output_items);
+
+ protected:
+ bool read_buffer(void *buffer, unsigned nframes, unsigned sizeof_frame);
+
+ int work_s16(int noutput_items,
+ gr_vector_const_void_star &input_items,
+ gr_vector_void_star &output_items);
+
+ int work_s16_2x1(int noutput_items,
+ gr_vector_const_void_star &input_items,
+ gr_vector_void_star &output_items);
+
+ int work_s32(int noutput_items,
+ gr_vector_const_void_star &input_items,
+ gr_vector_void_star &output_items);
+
+ int work_s32_2x1(int noutput_items,
+ gr_vector_const_void_star &input_items,
+ gr_vector_void_star &output_items);
+ };
+
+ } /* namespace audio */
+} /* namespace gr */
+
+#endif /* INCLUDED_AUDIO_ALSA_SOURCE_H */
diff --git a/gr-audio/lib/alsa/audio_alsa_sink.cc b/gr-audio/lib/alsa/audio_alsa_sink.cc
deleted file mode 100644
index 687f24bde2..0000000000
--- a/gr-audio/lib/alsa/audio_alsa_sink.cc
+++ /dev/null
@@ -1,548 +0,0 @@
-/* -*- c++ -*- */
-/*
- * Copyright 2004-2011 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 "gr_audio_registry.h"
-#include <audio_alsa_sink.h>
-#include <gr_io_signature.h>
-#include <gr_prefs.h>
-#include <stdio.h>
-#include <iostream>
-#include <stdexcept>
-#include <gri_alsa.h>
-
-AUDIO_REGISTER_SINK(REG_PRIO_HIGH, alsa)(
- int sampling_rate, const std::string &device_name, bool ok_to_block
-){
- return audio_sink::sptr(new audio_alsa_sink(sampling_rate, device_name, ok_to_block));
-}
-
-static bool CHATTY_DEBUG = false;
-
-
-static snd_pcm_format_t acceptable_formats[] = {
- // these are in our preferred order...
- SND_PCM_FORMAT_S32,
- SND_PCM_FORMAT_S16
-};
-
-#define NELEMS(x) (sizeof(x)/sizeof(x[0]))
-
-
-static std::string
-default_device_name ()
-{
- return gr_prefs::singleton()->get_string("audio_alsa", "default_output_device", "hw:0,0");
-}
-
-static double
-default_period_time ()
-{
- return std::max(0.001, gr_prefs::singleton()->get_double("audio_alsa", "period_time", 0.010));
-}
-
-static int
-default_nperiods ()
-{
- return std::max(2L, gr_prefs::singleton()->get_long("audio_alsa", "nperiods", 4));
-}
-
-// ----------------------------------------------------------------
-
-audio_alsa_sink::audio_alsa_sink (int sampling_rate,
- const std::string device_name,
- bool ok_to_block)
- : gr_sync_block ("audio_alsa_sink",
- gr_make_io_signature (0, 0, 0),
- gr_make_io_signature (0, 0, 0)),
- d_sampling_rate (sampling_rate),
- d_device_name (device_name.empty() ? default_device_name() : device_name),
- d_pcm_handle (0),
- d_hw_params ((snd_pcm_hw_params_t *)(new char[snd_pcm_hw_params_sizeof()])),
- d_sw_params ((snd_pcm_sw_params_t *)(new char[snd_pcm_sw_params_sizeof()])),
- d_nperiods (default_nperiods()),
- d_period_time_us ((unsigned int) (default_period_time() * 1e6)),
- d_period_size (0),
- d_buffer_size_bytes (0), d_buffer (0),
- d_worker (0), d_special_case_mono_to_stereo (false),
- d_nunderuns (0), d_nsuspends (0), d_ok_to_block(ok_to_block)
-{
- CHATTY_DEBUG = gr_prefs::singleton()->get_bool("audio_alsa", "verbose", false);
-
- int error;
- int dir;
-
- // open the device for playback
- error = snd_pcm_open(&d_pcm_handle, d_device_name.c_str (),
- SND_PCM_STREAM_PLAYBACK, 0);
- if (ok_to_block == false)
- snd_pcm_nonblock(d_pcm_handle, !ok_to_block);
- if (error < 0){
- fprintf (stderr, "audio_alsa_sink[%s]: %s\n",
- d_device_name.c_str(), snd_strerror(error));
- throw std::runtime_error ("audio_alsa_sink");
- }
-
- // Fill params with a full configuration space for a PCM.
- error = snd_pcm_hw_params_any(d_pcm_handle, d_hw_params);
- if (error < 0)
- bail ("broken configuration for playback", error);
-
-
- if (CHATTY_DEBUG)
- gri_alsa_dump_hw_params (d_pcm_handle, d_hw_params, stdout);
-
-
- // now that we know how many channels the h/w can handle, set input signature
- unsigned int umin_chan, umax_chan;
- snd_pcm_hw_params_get_channels_min (d_hw_params, &umin_chan);
- snd_pcm_hw_params_get_channels_max (d_hw_params, &umax_chan);
- int min_chan = std::min (umin_chan, 1000U);
- int max_chan = std::min (umax_chan, 1000U);
-
- // As a special case, if the hw's min_chan is two, we'll accept
- // a single input and handle the duplication ourselves.
-
- if (min_chan == 2){
- min_chan = 1;
- d_special_case_mono_to_stereo = true;
- }
- set_input_signature (gr_make_io_signature (min_chan, max_chan,
- sizeof (float)));
-
- // fill in portions of the d_hw_params that we know now...
-
- // Specify the access methods we implement
- // For now, we only handle RW_INTERLEAVED...
- snd_pcm_access_mask_t *access_mask;
- snd_pcm_access_mask_t **access_mask_ptr = &access_mask; // FIXME: workaround for compiler warning
- snd_pcm_access_mask_alloca (access_mask_ptr);
- snd_pcm_access_mask_none (access_mask);
- snd_pcm_access_mask_set (access_mask, SND_PCM_ACCESS_RW_INTERLEAVED);
- // snd_pcm_access_mask_set (access_mask, SND_PCM_ACCESS_RW_NONINTERLEAVED);
-
- if ((error = snd_pcm_hw_params_set_access_mask (d_pcm_handle,
- d_hw_params, access_mask)) < 0)
- bail ("failed to set access mask", error);
-
-
- // set sample format
- if (!gri_alsa_pick_acceptable_format (d_pcm_handle, d_hw_params,
- acceptable_formats,
- NELEMS (acceptable_formats),
- &d_format,
- "audio_alsa_sink",
- CHATTY_DEBUG))
- throw std::runtime_error ("audio_alsa_sink");
-
-
- // sampling rate
- unsigned int orig_sampling_rate = d_sampling_rate;
- if ((error = snd_pcm_hw_params_set_rate_near (d_pcm_handle, d_hw_params,
- &d_sampling_rate, 0)) < 0)
- bail ("failed to set rate near", error);
-
- if (orig_sampling_rate != d_sampling_rate){
- fprintf (stderr, "audio_alsa_sink[%s]: unable to support sampling rate %d\n",
- snd_pcm_name (d_pcm_handle), orig_sampling_rate);
- fprintf (stderr, " card requested %d instead.\n", d_sampling_rate);
- }
-
- /*
- * ALSA transfers data in units of "periods".
- * We indirectly determine the underlying buffersize by specifying
- * the number of periods we want (typically 4) and the length of each
- * period in units of time (typically 1ms).
- */
- unsigned int min_nperiods, max_nperiods;
- snd_pcm_hw_params_get_periods_min (d_hw_params, &min_nperiods, &dir);
- snd_pcm_hw_params_get_periods_max (d_hw_params, &max_nperiods, &dir);
- //fprintf (stderr, "alsa_sink: min_nperiods = %d, max_nperiods = %d\n",
- // min_nperiods, max_nperiods);
-
- unsigned int orig_nperiods = d_nperiods;
- d_nperiods = std::min (std::max (min_nperiods, d_nperiods), max_nperiods);
-
- // adjust period time so that total buffering remains more-or-less constant
- d_period_time_us = (d_period_time_us * orig_nperiods) / d_nperiods;
-
- error = snd_pcm_hw_params_set_periods (d_pcm_handle, d_hw_params,
- d_nperiods, 0);
- if (error < 0)
- bail ("set_periods failed", error);
-
- dir = 0;
- error = snd_pcm_hw_params_set_period_time_near (d_pcm_handle, d_hw_params,
- &d_period_time_us, &dir);
- if (error < 0)
- bail ("set_period_time_near failed", error);
-
- dir = 0;
- error = snd_pcm_hw_params_get_period_size (d_hw_params,
- &d_period_size, &dir);
- if (error < 0)
- bail ("get_period_size failed", error);
-
- set_output_multiple (d_period_size);
-}
-
-
-bool
-audio_alsa_sink::check_topology (int ninputs, int noutputs)
-{
- // ninputs is how many channels the user has connected.
- // Now we can finish up setting up the hw params...
-
- int nchan = ninputs;
- int err;
-
- // Check the state of the stream
- // Ensure that the pcm is in a state where we can still mess with the hw_params
- snd_pcm_state_t state;
- state=snd_pcm_state(d_pcm_handle);
- if ( state== SND_PCM_STATE_RUNNING)
- return true; // If stream is running, don't change any parameters
- else if(state == SND_PCM_STATE_XRUN )
- snd_pcm_prepare ( d_pcm_handle ); // Prepare stream on underrun, and we can set parameters;
-
- bool special_case = nchan == 1 && d_special_case_mono_to_stereo;
- if (special_case)
- nchan = 2;
-
- err = snd_pcm_hw_params_set_channels (d_pcm_handle, d_hw_params, nchan);
-
- if (err < 0){
- output_error_msg ("set_channels failed", err);
- return false;
- }
-
- // set the parameters into the driver...
- err = snd_pcm_hw_params(d_pcm_handle, d_hw_params);
- if (err < 0){
- output_error_msg ("snd_pcm_hw_params failed", err);
- return false;
- }
-
- // get current s/w params
- err = snd_pcm_sw_params_current (d_pcm_handle, d_sw_params);
- if (err < 0)
- bail ("snd_pcm_sw_params_current", err);
-
- // Tell the PCM device to wait to start until we've filled
- // it's buffers half way full. This helps avoid audio underruns.
-
- err = snd_pcm_sw_params_set_start_threshold(d_pcm_handle,
- d_sw_params,
- d_nperiods * d_period_size / 2);
- if (err < 0)
- bail ("snd_pcm_sw_params_set_start_threshold", err);
-
- // store the s/w params
- err = snd_pcm_sw_params (d_pcm_handle, d_sw_params);
- if (err < 0)
- bail ("snd_pcm_sw_params", err);
-
- d_buffer_size_bytes =
- d_period_size * nchan * snd_pcm_format_size (d_format, 1);
-
- d_buffer = new char [d_buffer_size_bytes];
-
- if (CHATTY_DEBUG)
- fprintf (stdout, "audio_alsa_sink[%s]: sample resolution = %d bits\n",
- snd_pcm_name (d_pcm_handle),
- snd_pcm_hw_params_get_sbits (d_hw_params));
-
- switch (d_format){
- case SND_PCM_FORMAT_S16:
- if (special_case)
- d_worker = &audio_alsa_sink::work_s16_1x2;
- else
- d_worker = &audio_alsa_sink::work_s16;
- break;
-
- case SND_PCM_FORMAT_S32:
- if (special_case)
- d_worker = &audio_alsa_sink::work_s32_1x2;
- else
- d_worker = &audio_alsa_sink::work_s32;
- break;
-
- default:
- assert (0);
- }
- return true;
-}
-
-audio_alsa_sink::~audio_alsa_sink ()
-{
- if (snd_pcm_state (d_pcm_handle) == SND_PCM_STATE_RUNNING)
- snd_pcm_drop (d_pcm_handle);
-
- snd_pcm_close(d_pcm_handle);
- delete [] ((char *) d_hw_params);
- delete [] ((char *) d_sw_params);
- delete [] d_buffer;
-}
-
-int
-audio_alsa_sink::work (int noutput_items,
- gr_vector_const_void_star &input_items,
- gr_vector_void_star &output_items)
-{
- assert ((noutput_items % d_period_size) == 0);
-
- // this is a call through a pointer to a method...
- return (this->*d_worker)(noutput_items, input_items, output_items);
-}
-
-/*
- * Work function that deals with float to S16 conversion
- */
-int
-audio_alsa_sink::work_s16 (int noutput_items,
- gr_vector_const_void_star &input_items,
- gr_vector_void_star &output_items)
-{
- typedef gr_int16 sample_t; // the type of samples we're creating
- static const float scale_factor = std::pow(2.0f, 16-1) - 1;
-
- unsigned int nchan = input_items.size ();
- const float **in = (const float **) &input_items[0];
- sample_t *buf = (sample_t *) d_buffer;
- int bi;
- int n;
-
- unsigned int sizeof_frame = nchan * sizeof (sample_t);
- assert (d_buffer_size_bytes == d_period_size * sizeof_frame);
-
- for (n = 0; n < noutput_items; n += d_period_size){
-
- // process one period of data
- bi = 0;
- for (unsigned int i = 0; i < d_period_size; i++){
- for (unsigned int chan = 0; chan < nchan; chan++){
- buf[bi++] = (sample_t) (in[chan][i] * scale_factor);
- }
- }
-
- // update src pointers
- for (unsigned int chan = 0; chan < nchan; chan++)
- in[chan] += d_period_size;
-
- if (!write_buffer (buf, d_period_size, sizeof_frame))
- return -1; // No fixing this problem. Say we're done.
- }
-
- return n;
-}
-
-
-/*
- * Work function that deals with float to S32 conversion
- */
-int
-audio_alsa_sink::work_s32 (int noutput_items,
- gr_vector_const_void_star &input_items,
- gr_vector_void_star &output_items)
-{
- typedef gr_int32 sample_t; // the type of samples we're creating
- static const float scale_factor = std::pow(2.0f, 32-1) - 1;
-
- unsigned int nchan = input_items.size ();
- const float **in = (const float **) &input_items[0];
- sample_t *buf = (sample_t *) d_buffer;
- int bi;
- int n;
-
- unsigned int sizeof_frame = nchan * sizeof (sample_t);
- assert (d_buffer_size_bytes == d_period_size * sizeof_frame);
-
- for (n = 0; n < noutput_items; n += d_period_size){
-
- // process one period of data
- bi = 0;
- for (unsigned int i = 0; i < d_period_size; i++){
- for (unsigned int chan = 0; chan < nchan; chan++){
- buf[bi++] = (sample_t) (in[chan][i] * scale_factor);
- }
- }
-
- // update src pointers
- for (unsigned int chan = 0; chan < nchan; chan++)
- in[chan] += d_period_size;
-
- if (!write_buffer (buf, d_period_size, sizeof_frame))
- return -1; // No fixing this problem. Say we're done.
- }
-
- return n;
-}
-
-/*
- * Work function that deals with float to S16 conversion and
- * mono to stereo kludge.
- */
-int
-audio_alsa_sink::work_s16_1x2 (int noutput_items,
- gr_vector_const_void_star &input_items,
- gr_vector_void_star &output_items)
-{
- typedef gr_int16 sample_t; // the type of samples we're creating
- static const float scale_factor = std::pow(2.0f, 16-1) - 1;
-
- assert (input_items.size () == 1);
- static const unsigned int nchan = 2;
- const float **in = (const float **) &input_items[0];
- sample_t *buf = (sample_t *) d_buffer;
- int bi;
- int n;
-
- unsigned int sizeof_frame = nchan * sizeof (sample_t);
- assert (d_buffer_size_bytes == d_period_size * sizeof_frame);
-
- for (n = 0; n < noutput_items; n += d_period_size){
-
- // process one period of data
- bi = 0;
- for (unsigned int i = 0; i < d_period_size; i++){
- sample_t t = (sample_t) (in[0][i] * scale_factor);
- buf[bi++] = t;
- buf[bi++] = t;
- }
-
- // update src pointers
- in[0] += d_period_size;
-
- if (!write_buffer (buf, d_period_size, sizeof_frame))
- return -1; // No fixing this problem. Say we're done.
- }
-
- return n;
-}
-
-/*
- * Work function that deals with float to S32 conversion and
- * mono to stereo kludge.
- */
-int
-audio_alsa_sink::work_s32_1x2 (int noutput_items,
- gr_vector_const_void_star &input_items,
- gr_vector_void_star &output_items)
-{
- typedef gr_int32 sample_t; // the type of samples we're creating
- static const float scale_factor = std::pow(2.0f, 32-1) - 1;
-
- assert (input_items.size () == 1);
- static unsigned int nchan = 2;
- const float **in = (const float **) &input_items[0];
- sample_t *buf = (sample_t *) d_buffer;
- int bi;
- int n;
-
- unsigned int sizeof_frame = nchan * sizeof (sample_t);
- assert (d_buffer_size_bytes == d_period_size * sizeof_frame);
-
- for (n = 0; n < noutput_items; n += d_period_size){
-
- // process one period of data
- bi = 0;
- for (unsigned int i = 0; i < d_period_size; i++){
- sample_t t = (sample_t) (in[0][i] * scale_factor);
- buf[bi++] = t;
- buf[bi++] = t;
- }
-
- // update src pointers
- in[0] += d_period_size;
-
- if (!write_buffer (buf, d_period_size, sizeof_frame))
- return -1; // No fixing this problem. Say we're done.
- }
-
- return n;
-}
-
-bool
-audio_alsa_sink::write_buffer (const void *vbuffer,
- unsigned nframes, unsigned sizeof_frame)
-{
- const unsigned char *buffer = (const unsigned char *) vbuffer;
-
- while (nframes > 0){
- int r = snd_pcm_writei (d_pcm_handle, buffer, nframes);
- if (r == -EAGAIN)
- {
- if (d_ok_to_block == true)
- continue; // try again
-
- break;
- }
-
- else if (r == -EPIPE){ // underrun
- d_nunderuns++;
- fputs ("aU", stderr);
- if ((r = snd_pcm_prepare (d_pcm_handle)) < 0){
- output_error_msg ("snd_pcm_prepare failed. Can't recover from underrun", r);
- return false;
- }
- continue; // try again
- }
-
- else if (r == -ESTRPIPE){ // h/w is suspended (whatever that means)
- // This is apparently related to power management
- d_nsuspends++;
- if ((r = snd_pcm_resume (d_pcm_handle)) < 0){
- output_error_msg ("failed to resume from suspend", r);
- return false;
- }
- continue; // try again
- }
-
- else if (r < 0){
- output_error_msg ("snd_pcm_writei failed", r);
- return false;
- }
-
- nframes -= r;
- buffer += r * sizeof_frame;
- }
-
- return true;
-}
-
-
-void
-audio_alsa_sink::output_error_msg (const char *msg, int err)
-{
- fprintf (stderr, "audio_alsa_sink[%s]: %s: %s\n",
- snd_pcm_name (d_pcm_handle), msg, snd_strerror (err));
-}
-
-void
-audio_alsa_sink::bail (const char *msg, int err) throw (std::runtime_error)
-{
- output_error_msg (msg, err);
- throw std::runtime_error ("audio_alsa_sink");
-}
diff --git a/gr-audio/lib/alsa/audio_alsa_sink.h b/gr-audio/lib/alsa/audio_alsa_sink.h
deleted file mode 100644
index d456e53de3..0000000000
--- a/gr-audio/lib/alsa/audio_alsa_sink.h
+++ /dev/null
@@ -1,105 +0,0 @@
-/* -*- c++ -*- */
-/*
- * Copyright 2004-2011 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.
- */
-
-#ifndef INCLUDED_AUDIO_ALSA_SINK_H
-#define INCLUDED_AUDIO_ALSA_SINK_H
-
-// use new ALSA API
-#define ALSA_PCM_NEW_HW_PARAMS_API
-#define ALSA_PCM_NEW_SW_PARAMS_API
-
-#include <gr_audio_sink.h>
-#include <string>
-#include <alsa/asoundlib.h>
-#include <stdexcept>
-
-/*!
- * \brief audio sink using ALSA
- * \ingroup audio_blk
- *
- * The sink has N input streams of floats, where N depends
- * on the hardware characteristics of the selected device.
- *
- * Input samples must be in the range [-1,1].
- */
-class audio_alsa_sink : public audio_sink {
- // typedef for pointer to class work method
- typedef int (audio_alsa_sink::*work_t)(int noutput_items,
- gr_vector_const_void_star &input_items,
- gr_vector_void_star &output_items);
-
- unsigned int d_sampling_rate;
- std::string d_device_name;
- snd_pcm_t *d_pcm_handle;
- snd_pcm_hw_params_t *d_hw_params;
- snd_pcm_sw_params_t *d_sw_params;
- snd_pcm_format_t d_format;
- unsigned int d_nperiods;
- unsigned int d_period_time_us; // microseconds
- snd_pcm_uframes_t d_period_size; // in frames
- unsigned int d_buffer_size_bytes; // sizeof of d_buffer
- char *d_buffer;
- work_t d_worker; // the work method to use
- bool d_special_case_mono_to_stereo;
-
- // random stats
- int d_nunderuns; // count of underruns
- int d_nsuspends; // count of suspends
- bool d_ok_to_block; // defaults to "true", controls blocking/non-block I/O
-
- void output_error_msg (const char *msg, int err);
- void bail (const char *msg, int err) throw (std::runtime_error);
-
-public:
- audio_alsa_sink (int sampling_rate, const std::string device_name,
- bool ok_to_block);
-
- ~audio_alsa_sink ();
-
- bool check_topology (int ninputs, int noutputs);
-
- int work (int noutput_items,
- gr_vector_const_void_star &input_items,
- gr_vector_void_star &output_items);
-
-
-protected:
- bool write_buffer (const void *buffer, unsigned nframes, unsigned sizeof_frame);
-
- int work_s16 (int noutput_items,
- gr_vector_const_void_star &input_items,
- gr_vector_void_star &output_items);
-
- int work_s16_1x2 (int noutput_items,
- gr_vector_const_void_star &input_items,
- gr_vector_void_star &output_items);
-
- int work_s32 (int noutput_items,
- gr_vector_const_void_star &input_items,
- gr_vector_void_star &output_items);
-
- int work_s32_1x2 (int noutput_items,
- gr_vector_const_void_star &input_items,
- gr_vector_void_star &output_items);
-};
-
-#endif /* INCLUDED_AUDIO_ALSA_SINK_H */
diff --git a/gr-audio/lib/alsa/audio_alsa_source.cc b/gr-audio/lib/alsa/audio_alsa_source.cc
deleted file mode 100644
index 9fdf80b43f..0000000000
--- a/gr-audio/lib/alsa/audio_alsa_source.cc
+++ /dev/null
@@ -1,509 +0,0 @@
-/* -*- c++ -*- */
-/*
- * Copyright 2004-2011 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 "gr_audio_registry.h"
-#include <audio_alsa_source.h>
-#include <gr_io_signature.h>
-#include <gr_prefs.h>
-#include <stdio.h>
-#include <iostream>
-#include <stdexcept>
-#include <gri_alsa.h>
-
-AUDIO_REGISTER_SOURCE(REG_PRIO_HIGH, alsa)(
- int sampling_rate, const std::string &device_name, bool ok_to_block
-){
- return audio_source::sptr(new audio_alsa_source(sampling_rate, device_name, ok_to_block));
-}
-
-static bool CHATTY_DEBUG = false;
-
-static snd_pcm_format_t acceptable_formats[] = {
- // these are in our preferred order...
- SND_PCM_FORMAT_S32,
- SND_PCM_FORMAT_S16
-};
-
-#define NELEMS(x) (sizeof(x)/sizeof(x[0]))
-
-
-static std::string
-default_device_name ()
-{
- return gr_prefs::singleton()->get_string("audio_alsa", "default_input_device", "hw:0,0");
-}
-
-static double
-default_period_time ()
-{
- return std::max(0.001, gr_prefs::singleton()->get_double("audio_alsa", "period_time", 0.010));
-}
-
-static int
-default_nperiods ()
-{
- return std::max(2L, gr_prefs::singleton()->get_long("audio_alsa", "nperiods", 4));
-}
-
-// ----------------------------------------------------------------
-
-audio_alsa_source::audio_alsa_source (int sampling_rate,
- const std::string device_name,
- bool ok_to_block)
- : gr_sync_block ("audio_alsa_source",
- gr_make_io_signature (0, 0, 0),
- gr_make_io_signature (0, 0, 0)),
- d_sampling_rate (sampling_rate),
- d_device_name (device_name.empty() ? default_device_name() : device_name),
- d_pcm_handle (0),
- d_hw_params ((snd_pcm_hw_params_t *)(new char[snd_pcm_hw_params_sizeof()])),
- d_sw_params ((snd_pcm_sw_params_t *)(new char[snd_pcm_sw_params_sizeof()])),
- d_nperiods (default_nperiods()),
- d_period_time_us ((unsigned int) (default_period_time() * 1e6)),
- d_period_size (0),
- d_buffer_size_bytes (0), d_buffer (0),
- d_worker (0), d_hw_nchan (0),
- d_special_case_stereo_to_mono (false),
- d_noverruns (0), d_nsuspends (0)
-{
-
- CHATTY_DEBUG = gr_prefs::singleton()->get_bool("audio_alsa", "verbose", false);
-
- int error;
- int dir;
-
- // open the device for capture
- error = snd_pcm_open(&d_pcm_handle, d_device_name.c_str (),
- SND_PCM_STREAM_CAPTURE, 0);
- if (error < 0){
- fprintf (stderr, "audio_alsa_source[%s]: %s\n",
- d_device_name.c_str(), snd_strerror(error));
- throw std::runtime_error ("audio_alsa_source");
- }
-
- // Fill params with a full configuration space for a PCM.
- error = snd_pcm_hw_params_any(d_pcm_handle, d_hw_params);
- if (error < 0)
- bail ("broken configuration for playback", error);
-
- if (CHATTY_DEBUG)
- gri_alsa_dump_hw_params (d_pcm_handle, d_hw_params, stdout);
-
- // now that we know how many channels the h/w can handle, set output signature
- unsigned int umax_chan;
- unsigned int umin_chan;
- snd_pcm_hw_params_get_channels_min (d_hw_params, &umin_chan);
- snd_pcm_hw_params_get_channels_max (d_hw_params, &umax_chan);
- int min_chan = std::min (umin_chan, 1000U);
- int max_chan = std::min (umax_chan, 1000U);
-
- // As a special case, if the hw's min_chan is two, we'll accept
- // a single output and handle the demux ourselves.
-
- if (min_chan == 2){
- min_chan = 1;
- d_special_case_stereo_to_mono = true;
- }
-
- set_output_signature (gr_make_io_signature (min_chan, max_chan,
- sizeof (float)));
-
- // fill in portions of the d_hw_params that we know now...
-
- // Specify the access methods we implement
- // For now, we only handle RW_INTERLEAVED...
- snd_pcm_access_mask_t *access_mask;
- snd_pcm_access_mask_t **access_mask_ptr = &access_mask; // FIXME: workaround for compiler warning
- snd_pcm_access_mask_alloca (access_mask_ptr);
- snd_pcm_access_mask_none (access_mask);
- snd_pcm_access_mask_set (access_mask, SND_PCM_ACCESS_RW_INTERLEAVED);
- // snd_pcm_access_mask_set (access_mask, SND_PCM_ACCESS_RW_NONINTERLEAVED);
-
- if ((error = snd_pcm_hw_params_set_access_mask (d_pcm_handle,
- d_hw_params, access_mask)) < 0)
- bail ("failed to set access mask", error);
-
-
- // set sample format
- if (!gri_alsa_pick_acceptable_format (d_pcm_handle, d_hw_params,
- acceptable_formats,
- NELEMS (acceptable_formats),
- &d_format,
- "audio_alsa_source",
- CHATTY_DEBUG))
- throw std::runtime_error ("audio_alsa_source");
-
-
- // sampling rate
- unsigned int orig_sampling_rate = d_sampling_rate;
- if ((error = snd_pcm_hw_params_set_rate_near (d_pcm_handle, d_hw_params,
- &d_sampling_rate, 0)) < 0)
- bail ("failed to set rate near", error);
-
- if (orig_sampling_rate != d_sampling_rate){
- fprintf (stderr, "audio_alsa_source[%s]: unable to support sampling rate %d\n",
- snd_pcm_name (d_pcm_handle), orig_sampling_rate);
- fprintf (stderr, " card requested %d instead.\n", d_sampling_rate);
- }
-
- /*
- * ALSA transfers data in units of "periods".
- * We indirectly determine the underlying buffersize by specifying
- * the number of periods we want (typically 4) and the length of each
- * period in units of time (typically 1ms).
- */
- unsigned int min_nperiods, max_nperiods;
- snd_pcm_hw_params_get_periods_min (d_hw_params, &min_nperiods, &dir);
- snd_pcm_hw_params_get_periods_max (d_hw_params, &max_nperiods, &dir);
- //fprintf (stderr, "alsa_source: min_nperiods = %d, max_nperiods = %d\n",
- // min_nperiods, max_nperiods);
-
-
- unsigned int orig_nperiods = d_nperiods;
- d_nperiods = std::min (std::max (min_nperiods, d_nperiods), max_nperiods);
-
- // adjust period time so that total buffering remains more-or-less constant
- d_period_time_us = (d_period_time_us * orig_nperiods) / d_nperiods;
-
- error = snd_pcm_hw_params_set_periods (d_pcm_handle, d_hw_params,
- d_nperiods, 0);
- if (error < 0)
- bail ("set_periods failed", error);
-
- dir = 0;
- error = snd_pcm_hw_params_set_period_time_near (d_pcm_handle, d_hw_params,
- &d_period_time_us, &dir);
- if (error < 0)
- bail ("set_period_time_near failed", error);
-
- dir = 0;
- error = snd_pcm_hw_params_get_period_size (d_hw_params,
- &d_period_size, &dir);
- if (error < 0)
- bail ("get_period_size failed", error);
-
- set_output_multiple (d_period_size);
-}
-
-bool
-audio_alsa_source::check_topology (int ninputs, int noutputs)
-{
- // noutputs is how many channels the user has connected.
- // Now we can finish up setting up the hw params...
-
- unsigned int nchan = noutputs;
- int err;
-
- // Check the state of the stream
- // Ensure that the pcm is in a state where we can still mess with the hw_params
- snd_pcm_state_t state;
- state=snd_pcm_state(d_pcm_handle);
- if ( state== SND_PCM_STATE_RUNNING)
- return true; // If stream is running, don't change any parameters
- else if(state == SND_PCM_STATE_XRUN )
- snd_pcm_prepare ( d_pcm_handle ); // Prepare stream on underrun, and we can set parameters;
-
- bool special_case = nchan == 1 && d_special_case_stereo_to_mono;
- if (special_case)
- nchan = 2;
-
- d_hw_nchan = nchan;
- err = snd_pcm_hw_params_set_channels (d_pcm_handle, d_hw_params, d_hw_nchan);
- if (err < 0){
- output_error_msg ("set_channels failed", err);
- return false;
- }
-
- // set the parameters into the driver...
- err = snd_pcm_hw_params(d_pcm_handle, d_hw_params);
- if (err < 0){
- output_error_msg ("snd_pcm_hw_params failed", err);
- return false;
- }
-
- d_buffer_size_bytes =
- d_period_size * d_hw_nchan * snd_pcm_format_size (d_format, 1);
-
- d_buffer = new char [d_buffer_size_bytes];
-
- if (CHATTY_DEBUG)
- fprintf (stdout, "audio_alsa_source[%s]: sample resolution = %d bits\n",
- snd_pcm_name (d_pcm_handle),
- snd_pcm_hw_params_get_sbits (d_hw_params));
-
- switch (d_format){
- case SND_PCM_FORMAT_S16:
- if (special_case)
- d_worker = &audio_alsa_source::work_s16_2x1;
- else
- d_worker = &audio_alsa_source::work_s16;
- break;
-
- case SND_PCM_FORMAT_S32:
- if (special_case)
- d_worker = &audio_alsa_source::work_s32_2x1;
- else
- d_worker = &audio_alsa_source::work_s32;
- break;
-
- default:
- assert (0);
- }
-
- return true;
-}
-
-audio_alsa_source::~audio_alsa_source ()
-{
- if (snd_pcm_state (d_pcm_handle) == SND_PCM_STATE_RUNNING)
- snd_pcm_drop (d_pcm_handle);
-
- snd_pcm_close(d_pcm_handle);
- delete [] ((char *) d_hw_params);
- delete [] ((char *) d_sw_params);
- delete [] d_buffer;
-}
-
-int
-audio_alsa_source::work (int noutput_items,
- gr_vector_const_void_star &input_items,
- gr_vector_void_star &output_items)
-{
- assert ((noutput_items % d_period_size) == 0);
- assert (noutput_items != 0);
-
- // this is a call through a pointer to a method...
- return (this->*d_worker)(noutput_items, input_items, output_items);
-}
-
-/*
- * Work function that deals with float to S16 conversion
- */
-int
-audio_alsa_source::work_s16 (int noutput_items,
- gr_vector_const_void_star &input_items,
- gr_vector_void_star &output_items)
-{
- typedef gr_int16 sample_t; // the type of samples we're creating
- static const float scale_factor = 1.0 / std::pow(2.0f, 16-1);
-
- unsigned int nchan = output_items.size ();
- float **out = (float **) &output_items[0];
- sample_t *buf = (sample_t *) d_buffer;
- int bi;
-
- unsigned int sizeof_frame = d_hw_nchan * sizeof (sample_t);
- assert (d_buffer_size_bytes == d_period_size * sizeof_frame);
-
- // To minimize latency, return at most a single period's worth of samples.
- // [We could also read the first one in a blocking mode and subsequent
- // ones in non-blocking mode, but we'll leave that for later (or never).]
-
- if (!read_buffer (buf, d_period_size, sizeof_frame))
- return -1; // No fixing this problem. Say we're done.
-
- // process one period of data
- bi = 0;
- for (unsigned int i = 0; i < d_period_size; i++){
- for (unsigned int chan = 0; chan < nchan; chan++){
- out[chan][i] = (float) buf[bi++] * scale_factor;
- }
- }
-
- return d_period_size;
-}
-
-/*
- * Work function that deals with float to S16 conversion
- * and stereo to mono kludge...
- */
-int
-audio_alsa_source::work_s16_2x1 (int noutput_items,
- gr_vector_const_void_star &input_items,
- gr_vector_void_star &output_items)
-{
- typedef gr_int16 sample_t; // the type of samples we're creating
- static const float scale_factor = 1.0 / std::pow(2.0f, 16-1);
-
- float **out = (float **) &output_items[0];
- sample_t *buf = (sample_t *) d_buffer;
- int bi;
-
- assert (output_items.size () == 1);
-
- unsigned int sizeof_frame = d_hw_nchan * sizeof (sample_t);
- assert (d_buffer_size_bytes == d_period_size * sizeof_frame);
-
- // To minimize latency, return at most a single period's worth of samples.
- // [We could also read the first one in a blocking mode and subsequent
- // ones in non-blocking mode, but we'll leave that for later (or never).]
-
- if (!read_buffer (buf, d_period_size, sizeof_frame))
- return -1; // No fixing this problem. Say we're done.
-
- // process one period of data
- bi = 0;
- for (unsigned int i = 0; i < d_period_size; i++){
- int t = (buf[bi] + buf[bi+1]) / 2;
- bi += 2;
- out[0][i] = (float) t * scale_factor;
- }
-
- return d_period_size;
-}
-
-/*
- * Work function that deals with float to S32 conversion
- */
-int
-audio_alsa_source::work_s32 (int noutput_items,
- gr_vector_const_void_star &input_items,
- gr_vector_void_star &output_items)
-{
- typedef gr_int32 sample_t; // the type of samples we're creating
- static const float scale_factor = 1.0 / std::pow(2.0f, 32-1);
-
- unsigned int nchan = output_items.size ();
- float **out = (float **) &output_items[0];
- sample_t *buf = (sample_t *) d_buffer;
- int bi;
-
- unsigned int sizeof_frame = d_hw_nchan * sizeof (sample_t);
- assert (d_buffer_size_bytes == d_period_size * sizeof_frame);
-
- // To minimize latency, return at most a single period's worth of samples.
- // [We could also read the first one in a blocking mode and subsequent
- // ones in non-blocking mode, but we'll leave that for later (or never).]
-
- if (!read_buffer (buf, d_period_size, sizeof_frame))
- return -1; // No fixing this problem. Say we're done.
-
- // process one period of data
- bi = 0;
- for (unsigned int i = 0; i < d_period_size; i++){
- for (unsigned int chan = 0; chan < nchan; chan++){
- out[chan][i] = (float) buf[bi++] * scale_factor;
- }
- }
-
- return d_period_size;
-}
-
-/*
- * Work function that deals with float to S32 conversion
- * and stereo to mono kludge...
- */
-int
-audio_alsa_source::work_s32_2x1 (int noutput_items,
- gr_vector_const_void_star &input_items,
- gr_vector_void_star &output_items)
-{
- typedef gr_int32 sample_t; // the type of samples we're creating
- static const float scale_factor = 1.0 / std::pow(2.0f, 32-1);
-
- float **out = (float **) &output_items[0];
- sample_t *buf = (sample_t *) d_buffer;
- int bi;
-
- assert (output_items.size () == 1);
-
- unsigned int sizeof_frame = d_hw_nchan * sizeof (sample_t);
- assert (d_buffer_size_bytes == d_period_size * sizeof_frame);
-
- // To minimize latency, return at most a single period's worth of samples.
- // [We could also read the first one in a blocking mode and subsequent
- // ones in non-blocking mode, but we'll leave that for later (or never).]
-
- if (!read_buffer (buf, d_period_size, sizeof_frame))
- return -1; // No fixing this problem. Say we're done.
-
- // process one period of data
- bi = 0;
- for (unsigned int i = 0; i < d_period_size; i++){
- int t = (buf[bi] + buf[bi+1]) / 2;
- bi += 2;
- out[0][i] = (float) t * scale_factor;
- }
-
- return d_period_size;
-}
-
-bool
-audio_alsa_source::read_buffer (void *vbuffer, unsigned nframes, unsigned sizeof_frame)
-{
- unsigned char *buffer = (unsigned char *) vbuffer;
-
- while (nframes > 0){
- int r = snd_pcm_readi (d_pcm_handle, buffer, nframes);
- if (r == -EAGAIN)
- continue; // try again
-
- else if (r == -EPIPE){ // overrun
- d_noverruns++;
- fputs ("aO", stderr);
- if ((r = snd_pcm_prepare (d_pcm_handle)) < 0){
- output_error_msg ("snd_pcm_prepare failed. Can't recover from overrun", r);
- return false;
- }
- continue; // try again
- }
-
- else if (r == -ESTRPIPE){ // h/w is suspended (whatever that means)
- // This is apparently related to power management
- d_nsuspends++;
- if ((r = snd_pcm_resume (d_pcm_handle)) < 0){
- output_error_msg ("failed to resume from suspend", r);
- return false;
- }
- continue; // try again
- }
-
- else if (r < 0){
- output_error_msg ("snd_pcm_readi failed", r);
- return false;
- }
-
- nframes -= r;
- buffer += r * sizeof_frame;
- }
-
- return true;
-}
-
-
-void
-audio_alsa_source::output_error_msg (const char *msg, int err)
-{
- fprintf (stderr, "audio_alsa_source[%s]: %s: %s\n",
- snd_pcm_name (d_pcm_handle), msg, snd_strerror (err));
-}
-
-void
-audio_alsa_source::bail (const char *msg, int err) throw (std::runtime_error)
-{
- output_error_msg (msg, err);
- throw std::runtime_error ("audio_alsa_source");
-}
diff --git a/gr-audio/lib/alsa/audio_alsa_source.h b/gr-audio/lib/alsa/audio_alsa_source.h
deleted file mode 100644
index 320d49bd28..0000000000
--- a/gr-audio/lib/alsa/audio_alsa_source.h
+++ /dev/null
@@ -1,107 +0,0 @@
-/* -*- c++ -*- */
-/*
- * Copyright 2004-2011 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.
- */
-
-#ifndef INCLUDED_AUDIO_ALSA_SOURCE_H
-#define INCLUDED_AUDIO_ALSA_SOURCE_H
-
-// use new ALSA API
-#define ALSA_PCM_NEW_HW_PARAMS_API
-#define ALSA_PCM_NEW_SW_PARAMS_API
-
-#include <gr_audio_source.h>
-#include <string>
-#include <alsa/asoundlib.h>
-#include <stdexcept>
-
-class audio_alsa_source;
-typedef boost::shared_ptr<audio_alsa_source> audio_alsa_source_sptr;
-
-/*!
- * \brief audio source using ALSA
- * \ingroup audio_blk
- *
- * The source has between 1 and N input streams of floats, where N is
- * depends on the hardware characteristics of the selected device.
- *
- * Output samples will be in the range [-1,1].
- */
-class audio_alsa_source : public audio_source {
- // typedef for pointer to class work method
- typedef int (audio_alsa_source::*work_t)(int noutput_items,
- gr_vector_const_void_star &input_items,
- gr_vector_void_star &output_items);
-
- unsigned int d_sampling_rate;
- std::string d_device_name;
- snd_pcm_t *d_pcm_handle;
- snd_pcm_hw_params_t *d_hw_params;
- snd_pcm_sw_params_t *d_sw_params;
- snd_pcm_format_t d_format;
- unsigned int d_nperiods;
- unsigned int d_period_time_us; // microseconds
- snd_pcm_uframes_t d_period_size; // in frames
- unsigned int d_buffer_size_bytes; // sizeof of d_buffer
- char *d_buffer;
- work_t d_worker; // the work method to use
- unsigned int d_hw_nchan; // # of configured h/w channels
- bool d_special_case_stereo_to_mono;
-
- // random stats
- int d_noverruns; // count of overruns
- int d_nsuspends; // count of suspends
-
- void output_error_msg (const char *msg, int err);
- void bail (const char *msg, int err) throw (std::runtime_error);
-
-public:
- audio_alsa_source (int sampling_rate, const std::string device_name,
- bool ok_to_block);
-
- ~audio_alsa_source ();
-
- bool check_topology (int ninputs, int noutputs);
-
- int work (int noutput_items,
- gr_vector_const_void_star &input_items,
- gr_vector_void_star &output_items);
-
-protected:
- bool read_buffer (void *buffer, unsigned nframes, unsigned sizeof_frame);
-
- int work_s16 (int noutput_items,
- gr_vector_const_void_star &input_items,
- gr_vector_void_star &output_items);
-
- int work_s16_2x1 (int noutput_items,
- gr_vector_const_void_star &input_items,
- gr_vector_void_star &output_items);
-
- int work_s32 (int noutput_items,
- gr_vector_const_void_star &input_items,
- gr_vector_void_star &output_items);
-
- int work_s32_2x1 (int noutput_items,
- gr_vector_const_void_star &input_items,
- gr_vector_void_star &output_items);
-};
-
-#endif /* INCLUDED_AUDIO_ALSA_SOURCE_H */
diff --git a/gr-audio/lib/audio_registry.cc b/gr-audio/lib/audio_registry.cc
new file mode 100644
index 0000000000..71f9099a63
--- /dev/null
+++ b/gr-audio/lib/audio_registry.cc
@@ -0,0 +1,154 @@
+/*
+ * Copyright 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.
+ */
+
+#include "audio_registry.h"
+#include <boost/foreach.hpp>
+#include <gr_prefs.h>
+#include <stdexcept>
+#include <vector>
+#include <iostream>
+
+namespace gr {
+ namespace audio {
+
+ /***********************************************************************
+ * Create registries
+ **********************************************************************/
+
+ struct source_entry_t {
+ reg_prio_type prio;
+ std::string arch;
+ source_factory_t source;
+ };
+
+ static std::vector<source_entry_t> &get_source_registry(void)
+ {
+ static std::vector<source_entry_t> d_registry;
+ return d_registry;
+ }
+
+ struct sink_entry_t
+ {
+ reg_prio_type prio;
+ std::string arch;
+ sink_factory_t sink;
+ };
+
+ static std::vector<sink_entry_t> &get_sink_registry(void)
+ {
+ static std::vector<sink_entry_t> d_registry;
+ return d_registry;
+ }
+
+ /***********************************************************************
+ * Register functions
+ **********************************************************************/
+ void
+ register_source(reg_prio_type prio,
+ const std::string &arch,
+ source_factory_t source)
+ {
+ source_entry_t entry;
+ entry.prio = prio;
+ entry.arch = arch;
+ entry.source = source;
+ get_source_registry().push_back(entry);
+ }
+
+ void register_sink(reg_prio_type prio,
+ const std::string &arch,
+ sink_factory_t sink)
+ {
+ sink_entry_t entry;
+ entry.prio = prio;
+ entry.arch = arch;
+ entry.sink = sink;
+ get_sink_registry().push_back(entry);
+ }
+
+ /***********************************************************************
+ * Factory functions
+ **********************************************************************/
+ static std::string default_arch_name(void)
+ {
+ return gr_prefs::singleton()->get_string("audio", "audio_module", "auto");
+ }
+
+ static void do_arch_warning(const std::string &arch)
+ {
+ if(arch == "auto")
+ return; //no warning when arch not specified
+ std::cerr << "Could not find audio architecture \"" << arch
+ << "\" in registry." << std::endl;
+ std::cerr << " Defaulting to the first available architecture..." << std::endl;
+ }
+
+ source::sptr
+ source::make(int sampling_rate,
+ const std::string device_name,
+ bool ok_to_block)
+ {
+ if(get_source_registry().empty()) {
+ throw std::runtime_error("no available audio source factories");
+ }
+
+ std::string arch = default_arch_name();
+ source_entry_t entry = get_source_registry().front();
+
+ BOOST_FOREACH(const source_entry_t &e, get_source_registry()) {
+ if(e.prio > entry.prio)
+ entry = e; //entry is highest prio
+ if(arch != e.arch)
+ continue; //continue when no match
+ return e.source(sampling_rate, device_name, ok_to_block);
+ }
+
+ //std::cout << "Audio source arch: " << entry.name << std::endl;
+ return entry.source(sampling_rate, device_name, ok_to_block);
+ }
+
+ sink::sptr
+ sink::make(int sampling_rate,
+ const std::string device_name,
+ bool ok_to_block)
+ {
+ if(get_sink_registry().empty()) {
+ throw std::runtime_error("no available audio sink factories");
+ }
+
+ std::string arch = default_arch_name();
+ sink_entry_t entry = get_sink_registry().front();
+
+ BOOST_FOREACH(const sink_entry_t &e, get_sink_registry()) {
+ if(e.prio > entry.prio)
+ entry = e; //entry is highest prio
+ if(arch != e.arch)
+ continue; //continue when no match
+ return e.sink(sampling_rate, device_name, ok_to_block);
+ }
+
+ do_arch_warning(arch);
+ //std::cout << "Audio sink arch: " << entry.name << std::endl;
+ return entry.sink(sampling_rate, device_name, ok_to_block);
+ }
+
+ } /* namespace audio */
+} /* namespace gr */
diff --git a/gr-audio/lib/audio_registry.h b/gr-audio/lib/audio_registry.h
new file mode 100644
index 0000000000..70612de574
--- /dev/null
+++ b/gr-audio/lib/audio_registry.h
@@ -0,0 +1,63 @@
+/*
+ * Copyright 2011 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.
+ */
+
+#ifndef INCLUDED_GR_AUDIO_REGISTRY_H
+#define INCLUDED_GR_AUDIO_REGISTRY_H
+
+#include <audio/sink.h>
+#include <audio/source.h>
+#include <string>
+
+namespace gr {
+ namespace audio {
+
+ typedef source::sptr(*source_factory_t)(int, const std::string &, bool);
+ typedef sink::sptr(*sink_factory_t)(int, const std::string &, bool);
+
+ enum reg_prio_type {
+ REG_PRIO_LOW = 100,
+ REG_PRIO_MED = 200,
+ REG_PRIO_HIGH = 300
+ };
+
+ void register_source(reg_prio_type prio, const std::string &arch,
+ source_factory_t source);
+ void register_sink(reg_prio_type prio, const std::string &arch,
+ sink_factory_t sink);
+
+#define AUDIO_REGISTER_FIXTURE(x) static struct x{x();}x;x::x()
+
+#define AUDIO_REGISTER_SOURCE(prio, arch) \
+ static source::sptr arch##_source_fcn(int, const std::string &, bool); \
+ AUDIO_REGISTER_FIXTURE(arch##_source_reg) { \
+ register_source(prio, #arch, &arch##_source_fcn); \
+ } static source::sptr arch##_source_fcn
+
+#define AUDIO_REGISTER_SINK(prio, arch) \
+ static sink::sptr arch##_sink_fcn(int, const std::string &, bool); \
+ AUDIO_REGISTER_FIXTURE(arch##_sink_reg) { \
+ register_sink(prio, #arch, &arch##_sink_fcn); \
+ } static sink::sptr arch##_sink_fcn
+
+ } /* namespace audio */
+} /* namespace gr */
+
+#endif /* INCLUDED_GR_AUDIO_REGISTRY_H */
diff --git a/gr-audio/lib/gr_audio_registry.cc b/gr-audio/lib/gr_audio_registry.cc
deleted file mode 100644
index e07bf844ac..0000000000
--- a/gr-audio/lib/gr_audio_registry.cc
+++ /dev/null
@@ -1,132 +0,0 @@
-/*
- * Copyright 2011 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.
- */
-
-#include "gr_audio_registry.h"
-#include <boost/foreach.hpp>
-#include <gr_prefs.h>
-#include <stdexcept>
-#include <vector>
-#include <iostream>
-
-/***********************************************************************
- * Create registries
- **********************************************************************/
-
-struct source_entry_t{
- reg_prio_type prio;
- std::string arch;
- source_factory_t source;
-};
-
-static std::vector<source_entry_t> &get_source_registry(void){
- static std::vector<source_entry_t> _registry;
- return _registry;
-}
-
-struct sink_entry_t{
- reg_prio_type prio;
- std::string arch;
- sink_factory_t sink;
-};
-
-static std::vector<sink_entry_t> &get_sink_registry(void){
- static std::vector<sink_entry_t> _registry;
- return _registry;
-}
-
-/***********************************************************************
- * Register functions
- **********************************************************************/
-void audio_register_source(
- reg_prio_type prio, const std::string &arch, source_factory_t source
-){
- source_entry_t entry;
- entry.prio = prio;
- entry.arch = arch;
- entry.source = source;
- get_source_registry().push_back(entry);
-}
-
-void audio_register_sink(
- reg_prio_type prio, const std::string &arch, sink_factory_t sink
-){
- sink_entry_t entry;
- entry.prio = prio;
- entry.arch = arch;
- entry.sink = sink;
- get_sink_registry().push_back(entry);
-}
-
-/***********************************************************************
- * Factory functions
- **********************************************************************/
-static std::string default_arch_name(void){
- return gr_prefs::singleton()->get_string("audio", "audio_module", "auto");
-}
-
-static void do_arch_warning(const std::string &arch){
- if (arch == "auto") return; //no warning when arch not specified
- std::cerr << "Could not find audio architecture \"" << arch << "\" in registry." << std::endl;
- std::cerr << " Defaulting to the first available architecture..." << std::endl;
-}
-
-audio_source::sptr audio_make_source(
- int sampling_rate,
- const std::string device_name,
- bool ok_to_block
-){
- if (get_source_registry().empty()){
- throw std::runtime_error("no available audio source factories");
- }
-
- std::string arch = default_arch_name();
- source_entry_t entry = get_source_registry().front();
-
- BOOST_FOREACH(const source_entry_t &e, get_source_registry()){
- if (e.prio > entry.prio) entry = e; //entry is highest prio
- if (arch != e.arch) continue; //continue when no match
- return e.source(sampling_rate, device_name, ok_to_block);
- }
- //std::cout << "Audio source arch: " << entry.name << std::endl;
- return entry.source(sampling_rate, device_name, ok_to_block);
-}
-
-audio_sink::sptr audio_make_sink(
- int sampling_rate,
- const std::string device_name,
- bool ok_to_block
-){
- if (get_sink_registry().empty()){
- throw std::runtime_error("no available audio sink factories");
- }
-
- std::string arch = default_arch_name();
- sink_entry_t entry = get_sink_registry().front();
-
- BOOST_FOREACH(const sink_entry_t &e, get_sink_registry()){
- if (e.prio > entry.prio) entry = e; //entry is highest prio
- if (arch != e.arch) continue; //continue when no match
- return e.sink(sampling_rate, device_name, ok_to_block);
- }
- do_arch_warning(arch);
- //std::cout << "Audio sink arch: " << entry.name << std::endl;
- return entry.sink(sampling_rate, device_name, ok_to_block);
-}
diff --git a/gr-audio/lib/gr_audio_registry.h b/gr-audio/lib/gr_audio_registry.h
deleted file mode 100644
index c40e156579..0000000000
--- a/gr-audio/lib/gr_audio_registry.h
+++ /dev/null
@@ -1,55 +0,0 @@
-/*
- * Copyright 2011 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.
- */
-
-#ifndef INCLUDED_GR_AUDIO_REGISTRY_H
-#define INCLUDED_GR_AUDIO_REGISTRY_H
-
-#include <gr_audio_sink.h>
-#include <gr_audio_source.h>
-#include <string>
-
-typedef audio_source::sptr(*source_factory_t)(int, const std::string &, bool);
-typedef audio_sink::sptr(*sink_factory_t)(int, const std::string &, bool);
-
-enum reg_prio_type{
- REG_PRIO_LOW = 100,
- REG_PRIO_MED = 200,
- REG_PRIO_HIGH = 300
-};
-
-void audio_register_source(reg_prio_type prio, const std::string &arch, source_factory_t source);
-void audio_register_sink(reg_prio_type prio, const std::string &arch, sink_factory_t sink);
-
-#define AUDIO_REGISTER_FIXTURE(x) static struct x{x();}x;x::x()
-
-#define AUDIO_REGISTER_SOURCE(prio, arch) \
- static audio_source::sptr arch##_source_fcn(int, const std::string &, bool); \
- AUDIO_REGISTER_FIXTURE(arch##_source_reg){ \
- audio_register_source(prio, #arch, &arch##_source_fcn); \
- } static audio_source::sptr arch##_source_fcn
-
-#define AUDIO_REGISTER_SINK(prio, arch) \
- static audio_sink::sptr arch##_sink_fcn(int, const std::string &, bool); \
- AUDIO_REGISTER_FIXTURE(arch##_sink_reg){ \
- audio_register_sink(prio, #arch, &arch##_sink_fcn); \
- } static audio_sink::sptr arch##_sink_fcn
-
-#endif /* INCLUDED_GR_AUDIO_REGISTRY_H */
diff --git a/gr-audio/lib/jack/audio_jack_sink.cc b/gr-audio/lib/jack/audio_jack_sink.cc
deleted file mode 100644
index 9caabe8e2f..0000000000
--- a/gr-audio/lib/jack/audio_jack_sink.cc
+++ /dev/null
@@ -1,236 +0,0 @@
-/* -*- c++ -*- */
-/*
- * Copyright 2005-2011 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 "gr_audio_registry.h"
-#include <audio_jack_sink.h>
-#include <gr_io_signature.h>
-#include <gr_prefs.h>
-#include <stdio.h>
-#include <iostream>
-#include <stdexcept>
-#include <gri_jack.h>
-
-#ifndef NO_PTHREAD
-#include <pthread.h>
-#endif
-
-AUDIO_REGISTER_SINK(REG_PRIO_MED, jack)(
- int sampling_rate, const std::string &device_name, bool ok_to_block
-){
- return audio_sink::sptr(new audio_jack_sink(sampling_rate, device_name, ok_to_block));
-}
-
-typedef jack_default_audio_sample_t sample_t;
-
-
-// Number of jack buffers in the ringbuffer
-// TODO: make it to match at least the quantity of items passed by work()
-static const unsigned int N_BUFFERS = 16;
-
-static std::string
-default_device_name ()
-{
- return gr_prefs::singleton()->get_string("audio_jack", "default_output_device", "gr_sink");
-}
-
-int
-jack_sink_process (jack_nframes_t nframes, void *arg)
-{
- audio_jack_sink *self = (audio_jack_sink *)arg;
- unsigned int read_size = nframes*sizeof(sample_t);
-
- if (jack_ringbuffer_read_space (self->d_ringbuffer) < read_size) {
- self->d_nunderuns++;
- // FIXME: move this fputs out, we shouldn't use blocking calls in process()
- fputs ("jU", stderr);
- return 0;
- }
-
- char *buffer = (char *) jack_port_get_buffer (self->d_jack_output_port, nframes);
-
- jack_ringbuffer_read (self->d_ringbuffer, buffer, read_size);
-
-#ifndef NO_PTHREAD
- // Tell the sink thread there is room in the ringbuffer.
- // If it is already running, the lock will not be available.
- // We can't wait here in the process() thread, but we don't
- // need to signal in that case, because the sink thread will
- // check for room availability.
-
- if (pthread_mutex_trylock (&self->d_jack_process_lock) == 0) {
- pthread_cond_signal (&self->d_ringbuffer_ready);
- pthread_mutex_unlock (&self->d_jack_process_lock);
- }
-#endif
-
- return 0;
-}
-
-// ----------------------------------------------------------------
-
-audio_jack_sink::audio_jack_sink (int sampling_rate,
- const std::string device_name,
- bool ok_to_block)
- : gr_sync_block ("audio_jack_sink",
- gr_make_io_signature (0, 0, 0),
- gr_make_io_signature (0, 0, 0)),
- d_sampling_rate (sampling_rate),
- d_device_name (device_name.empty() ? default_device_name() : device_name),
- d_ok_to_block (ok_to_block),
- d_jack_client (0),
- d_ringbuffer (0),
- d_nunderuns (0)
-{
-#ifndef NO_PTHREAD
- pthread_cond_init(&d_ringbuffer_ready, NULL);;
- pthread_mutex_init(&d_jack_process_lock, NULL);
-#endif
-
- // try to become a client of the JACK server
- jack_options_t options = JackNullOption;
- jack_status_t status;
- const char *server_name = NULL;
- if ((d_jack_client = jack_client_open (d_device_name.c_str (),
- options, &status,
- server_name)) == NULL) {
- fprintf (stderr, "audio_jack_sink[%s]: jack server not running?\n",
- d_device_name.c_str());
- throw std::runtime_error ("audio_jack_sink");
- }
-
- // tell the JACK server to call `jack_sink_process()' whenever
- // there is work to be done.
- jack_set_process_callback (d_jack_client, &jack_sink_process, (void*)this);
-
- // tell the JACK server to call `jack_shutdown()' if
- // it ever shuts down, either entirely, or if it
- // just decides to stop calling us.
-
- //jack_on_shutdown (d_jack_client, &jack_shutdown, (void*)this);
-
- d_jack_output_port =
- jack_port_register (d_jack_client, "out",
- JACK_DEFAULT_AUDIO_TYPE, JackPortIsOutput, 0);
-
-
- d_jack_buffer_size = jack_get_buffer_size (d_jack_client);
-
- set_output_multiple (d_jack_buffer_size);
-
- d_ringbuffer =
- jack_ringbuffer_create (N_BUFFERS*d_jack_buffer_size*sizeof(sample_t));
- if (d_ringbuffer == NULL)
- bail ("jack_ringbuffer_create failed", 0);
-
- assert(sizeof(float)==sizeof(sample_t));
- set_input_signature (gr_make_io_signature (1, 1, sizeof (sample_t)));
-
-
- jack_nframes_t sample_rate = jack_get_sample_rate (d_jack_client);
-
- if ((jack_nframes_t)sampling_rate != sample_rate){
- fprintf (stderr, "audio_jack_sink[%s]: unable to support sampling rate %d\n",
- d_device_name.c_str (), sampling_rate);
- fprintf (stderr, " card requested %d instead.\n", sample_rate);
- }
-}
-
-
-bool
-audio_jack_sink::check_topology (int ninputs, int noutputs)
-{
- if (ninputs != 1)
- return false;
-
- // tell the JACK server that we are ready to roll
- if (jack_activate (d_jack_client))
- throw std::runtime_error ("audio_jack_sink");
-
- return true;
-}
-
-audio_jack_sink::~audio_jack_sink ()
-{
- jack_client_close (d_jack_client);
- jack_ringbuffer_free (d_ringbuffer);
-}
-
-int
-audio_jack_sink::work (int noutput_items,
- gr_vector_const_void_star &input_items,
- gr_vector_void_star &output_items)
-{
- // write_size and work_size are in bytes
- int work_size = noutput_items*sizeof(sample_t);
- unsigned int write_size;
-
- while (work_size > 0) {
- unsigned int write_space; // bytes
-
-#ifdef NO_PTHREAD
- while ((write_space=jack_ringbuffer_write_space (d_ringbuffer)) <
- d_jack_buffer_size*sizeof(sample_t)) {
- usleep(1000000*((d_jack_buffer_size-write_space/sizeof(sample_t))/d_sampling_rate));
- }
-#else
- // JACK actually requires POSIX
-
- pthread_mutex_lock (&d_jack_process_lock);
- while ((write_space=jack_ringbuffer_write_space (d_ringbuffer)) <
- d_jack_buffer_size*sizeof(sample_t)) {
-
- // wait until jack_sink_process() signals more room
- pthread_cond_wait (&d_ringbuffer_ready, &d_jack_process_lock);
- }
- pthread_mutex_unlock (&d_jack_process_lock);
-#endif
-
- write_space -= write_space%(d_jack_buffer_size*sizeof(sample_t));
- write_size = std::min(write_space, (unsigned int)work_size);
-
- if (jack_ringbuffer_write (d_ringbuffer, (char *) input_items[0],
- write_size) < write_size) {
- bail ("jack_ringbuffer_write failed", 0);
- }
- work_size -= write_size;
- }
-
- return noutput_items;
-}
-
-void
-audio_jack_sink::output_error_msg (const char *msg, int err)
-{
- fprintf (stderr, "audio_jack_sink[%s]: %s: %d\n",
- d_device_name.c_str (), msg, err);
-}
-
-void
-audio_jack_sink::bail (const char *msg, int err) throw (std::runtime_error)
-{
- output_error_msg (msg, err);
- throw std::runtime_error ("audio_jack_sink");
-}
diff --git a/gr-audio/lib/jack/audio_jack_sink.h b/gr-audio/lib/jack/audio_jack_sink.h
deleted file mode 100644
index 8cc3439370..0000000000
--- a/gr-audio/lib/jack/audio_jack_sink.h
+++ /dev/null
@@ -1,80 +0,0 @@
-/* -*- c++ -*- */
-/*
- * Copyright 2005-2011 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.
- */
-#ifndef INCLUDED_AUDIO_JACK_SINK_H
-#define INCLUDED_AUDIO_JACK_SINK_H
-
-#include <gr_audio_sink.h>
-#include <string>
-#include <jack/jack.h>
-#include <jack/ringbuffer.h>
-#include <stdexcept>
-
-int jack_sink_process (jack_nframes_t nframes, void *arg);
-
-/*!
- * \brief audio sink using JACK
- * \ingroup audio_blk
- *
- * The sink has one input stream of floats.
- *
- * Input samples must be in the range [-1,1].
- */
-class audio_jack_sink : public audio_sink {
-
- friend int jack_sink_process (jack_nframes_t nframes, void *arg);
-
- // typedef for pointer to class work method
- typedef int (audio_jack_sink::*work_t)(int noutput_items,
- gr_vector_const_void_star &input_items,
- gr_vector_void_star &output_items);
-
- unsigned int d_sampling_rate;
- std::string d_device_name;
- bool d_ok_to_block;
-
- jack_client_t *d_jack_client;
- jack_port_t *d_jack_output_port;
- jack_ringbuffer_t *d_ringbuffer;
- jack_nframes_t d_jack_buffer_size;
- pthread_cond_t d_ringbuffer_ready;
- pthread_mutex_t d_jack_process_lock;
-
- // random stats
- int d_nunderuns; // count of underruns
-
- void output_error_msg (const char *msg, int err);
- void bail (const char *msg, int err) throw (std::runtime_error);
-
-
-public:
- audio_jack_sink (int sampling_rate, const std::string device_name, bool ok_to_block);
-
- ~audio_jack_sink ();
-
- bool check_topology (int ninputs, int noutputs);
-
- int work (int noutput_items,
- gr_vector_const_void_star &input_items,
- gr_vector_void_star &output_items);
-};
-
-#endif /* INCLUDED_AUDIO_JACK_SINK_H */
diff --git a/gr-audio/lib/jack/audio_jack_source.cc b/gr-audio/lib/jack/audio_jack_source.cc
deleted file mode 100644
index 137fd538e4..0000000000
--- a/gr-audio/lib/jack/audio_jack_source.cc
+++ /dev/null
@@ -1,237 +0,0 @@
-/* -*- c++ -*- */
-/*
- * Copyright 2005,2006,2010 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 "gr_audio_registry.h"
-#include <audio_jack_source.h>
-#include <gr_io_signature.h>
-#include <gr_prefs.h>
-#include <stdio.h>
-#include <iostream>
-#include <stdexcept>
-#include <gri_jack.h>
-
-#ifndef NO_PTHREAD
-#include <pthread.h>
-#endif
-
-AUDIO_REGISTER_SOURCE(REG_PRIO_MED, jack)(
- int sampling_rate, const std::string &device_name, bool ok_to_block
-){
- return audio_source::sptr(new audio_jack_source(sampling_rate, device_name, ok_to_block));
-}
-
-typedef jack_default_audio_sample_t sample_t;
-
-
-// Number of jack buffers in the ringbuffer
-// TODO: make it to match at least the quantity of items passed to work()
-static const unsigned int N_BUFFERS = 16;
-
-static std::string
-default_device_name ()
-{
- return gr_prefs::singleton()->get_string("audio_jack", "default_input_device", "gr_source");
-}
-
-
-int
-jack_source_process (jack_nframes_t nframes, void *arg)
-{
- audio_jack_source *self = (audio_jack_source *)arg;
- unsigned int write_size = nframes*sizeof(sample_t);
-
- if (jack_ringbuffer_write_space (self->d_ringbuffer) < write_size) {
- self->d_noverruns++;
- // FIXME: move this fputs out, we shouldn't use blocking calls in process()
- fputs ("jO", stderr);
- return 0;
- }
-
- char *buffer = (char *) jack_port_get_buffer (self->d_jack_input_port, nframes);
-
- jack_ringbuffer_write (self->d_ringbuffer, buffer, write_size);
-
-#ifndef NO_PTHREAD
- // Tell the source thread there is data in the ringbuffer.
- // If it is already running, the lock will not be available.
- // We can't wait here in the process() thread, but we don't
- // need to signal in that case, because the source thread will
- // check for data availability.
-
- if (pthread_mutex_trylock (&self->d_jack_process_lock) == 0) {
- pthread_cond_signal (&self->d_ringbuffer_ready);
- pthread_mutex_unlock (&self->d_jack_process_lock);
- }
-#endif
-
- return 0;
-}
-
-// ----------------------------------------------------------------
-
-audio_jack_source::audio_jack_source (int sampling_rate,
- const std::string device_name,
- bool ok_to_block)
- : gr_sync_block ("audio_jack_source",
- gr_make_io_signature (0, 0, 0),
- gr_make_io_signature (0, 0, 0)),
- d_sampling_rate (sampling_rate),
- d_device_name (device_name.empty() ? default_device_name() : device_name),
- d_ok_to_block(ok_to_block),
- d_jack_client (0),
- d_ringbuffer (0),
- d_noverruns (0)
-{
-#ifndef NO_PTHREAD
- pthread_cond_init(&d_ringbuffer_ready, NULL);;
- pthread_mutex_init(&d_jack_process_lock, NULL);
-#endif
-
- // try to become a client of the JACK server
- jack_options_t options = JackNullOption;
- jack_status_t status;
- const char *server_name = NULL;
- if ((d_jack_client = jack_client_open (d_device_name.c_str (),
- options, &status,
- server_name)) == NULL) {
- fprintf (stderr, "audio_jack_source[%s]: jack server not running?\n",
- d_device_name.c_str());
- throw std::runtime_error ("audio_jack_source");
- }
-
- // tell the JACK server to call `jack_source_process()' whenever
- // there is work to be done.
- jack_set_process_callback (d_jack_client, &jack_source_process, (void*)this);
-
- // tell the JACK server to call `jack_shutdown()' if
- // it ever shuts down, either entirely, or if it
- // just decides to stop calling us.
-
- //jack_on_shutdown (d_jack_client, &jack_shutdown, (void*)this);
-
- d_jack_input_port = jack_port_register (d_jack_client, "in",
- JACK_DEFAULT_AUDIO_TYPE,
- JackPortIsInput, 0);
-
-
- d_jack_buffer_size = jack_get_buffer_size (d_jack_client);
-
- set_output_multiple (d_jack_buffer_size);
-
- d_ringbuffer = jack_ringbuffer_create (N_BUFFERS*d_jack_buffer_size*sizeof(sample_t));
- if (d_ringbuffer == NULL)
- bail ("jack_ringbuffer_create failed", 0);
-
- assert(sizeof(float)==sizeof(sample_t));
- set_output_signature (gr_make_io_signature (1, 1, sizeof (sample_t)));
-
-
- jack_nframes_t sample_rate = jack_get_sample_rate (d_jack_client);
-
- if ((jack_nframes_t)sampling_rate != sample_rate){
- fprintf (stderr, "audio_jack_source[%s]: unable to support sampling rate %d\n",
- d_device_name.c_str (), sampling_rate);
- fprintf (stderr, " card requested %d instead.\n", sample_rate);
- }
-}
-
-
-bool
-audio_jack_source::check_topology (int ninputs, int noutputs)
-{
- // tell the JACK server that we are ready to roll
- if (jack_activate (d_jack_client))
- throw std::runtime_error ("audio_jack_source");
-
- return true;
-}
-
-audio_jack_source::~audio_jack_source ()
-{
- jack_client_close (d_jack_client);
- jack_ringbuffer_free (d_ringbuffer);
-}
-
-int
-audio_jack_source::work (int noutput_items,
- gr_vector_const_void_star &input_items,
- gr_vector_void_star &output_items)
-{
- // read_size and work_size are in bytes
- unsigned int read_size;
-
- // Minimize latency
- noutput_items = std::min (noutput_items, (int)d_jack_buffer_size);
-
- int work_size = noutput_items*sizeof(sample_t);
-
- while (work_size > 0) {
- unsigned int read_space; // bytes
-
-#ifdef NO_PTHREAD
- while ((read_space=jack_ringbuffer_read_space (d_ringbuffer)) <
- d_jack_buffer_size*sizeof(sample_t)) {
- usleep(1000000*((d_jack_buffer_size-read_space/sizeof(sample_t))/d_sampling_rate));
- }
-#else
- // JACK actually requires POSIX
-
- pthread_mutex_lock (&d_jack_process_lock);
- while ((read_space=jack_ringbuffer_read_space (d_ringbuffer)) <
- d_jack_buffer_size*sizeof(sample_t)) {
-
- // wait until jack_source_process() signals more data
- pthread_cond_wait (&d_ringbuffer_ready, &d_jack_process_lock);
- }
- pthread_mutex_unlock (&d_jack_process_lock);
-#endif
-
- read_space -= read_space%(d_jack_buffer_size*sizeof(sample_t));
- read_size = std::min(read_space, (unsigned int)work_size);
-
- if (jack_ringbuffer_read (d_ringbuffer, (char *) output_items[0],
- read_size) < read_size) {
- bail ("jack_ringbuffer_read failed", 0);
- }
- work_size -= read_size;
- }
-
- return noutput_items;
-}
-
-void
-audio_jack_source::output_error_msg (const char *msg, int err)
-{
- fprintf (stderr, "audio_jack_source[%s]: %s: %d\n",
- d_device_name.c_str (), msg, err);
-}
-
-void
-audio_jack_source::bail (const char *msg, int err) throw (std::runtime_error)
-{
- output_error_msg (msg, err);
- throw std::runtime_error ("audio_jack_source");
-}
diff --git a/gr-audio/lib/jack/audio_jack_source.h b/gr-audio/lib/jack/audio_jack_source.h
deleted file mode 100644
index 2849c84b0c..0000000000
--- a/gr-audio/lib/jack/audio_jack_source.h
+++ /dev/null
@@ -1,80 +0,0 @@
-/* -*- c++ -*- */
-/*
- * Copyright 2005-2011 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.
- */
-#ifndef INCLUDED_AUDIO_JACK_SOURCE_H
-#define INCLUDED_AUDIO_JACK_SOURCE_H
-
-#include <gr_audio_source.h>
-#include <string>
-#include <jack/jack.h>
-#include <jack/ringbuffer.h>
-#include <stdexcept>
-
-int jack_source_process (jack_nframes_t nframes, void *arg);
-
-/*!
- * \brief audio source using JACK
- * \ingroup audio_blk
- *
- * The source has one input stream of floats.
- *
- * Output samples will be in the range [-1,1].
- */
-class audio_jack_source : public audio_source {
-
- friend int jack_source_process (jack_nframes_t nframes, void *arg);
-
- // typedef for pointer to class work method
- typedef int (audio_jack_source::*work_t)(int noutput_items,
- gr_vector_const_void_star &input_items,
- gr_vector_void_star &output_items);
-
- unsigned int d_sampling_rate;
- std::string d_device_name;
- bool d_ok_to_block;
-
- jack_client_t *d_jack_client;
- jack_port_t *d_jack_input_port;
- jack_ringbuffer_t *d_ringbuffer;
- jack_nframes_t d_jack_buffer_size;
- pthread_cond_t d_ringbuffer_ready;
- pthread_mutex_t d_jack_process_lock;
-
- // random stats
- int d_noverruns; // count of overruns
-
- void output_error_msg (const char *msg, int err);
- void bail (const char *msg, int err) throw (std::runtime_error);
-
-
-public:
- audio_jack_source (int sampling_rate, const std::string device_name, bool ok_to_block);
-
- ~audio_jack_source ();
-
- bool check_topology (int ninputs, int noutputs);
-
- int work (int noutput_items,
- gr_vector_const_void_star &input_items,
- gr_vector_void_star &output_items);
-};
-
-#endif /* INCLUDED_AUDIO_JACK_SOURCE_H */
diff --git a/gr-audio/lib/jack/gri_jack.cc b/gr-audio/lib/jack/jack_impl.cc
index 793ed8336b..c1f3b320fc 100644
--- a/gr-audio/lib/jack/gri_jack.cc
+++ b/gr-audio/lib/jack/jack_impl.cc
@@ -24,7 +24,7 @@
#include "config.h"
#endif
-#include <gri_jack.h>
+#include <jack_impl.h>
#include <algorithm>
diff --git a/gr-audio/lib/jack/gri_jack.h b/gr-audio/lib/jack/jack_impl.h
index 5dcd3b811e..178b6a1388 100644
--- a/gr-audio/lib/jack/gri_jack.h
+++ b/gr-audio/lib/jack/jack_impl.h
@@ -20,9 +20,9 @@
* Boston, MA 02110-1301, USA.
*/
-#ifndef INCLUDED_GRI_JACK_H
-#define INCLUDED_GRI_JACK_H
+#ifndef INCLUDED_AUDIO_JACK_IMPL_H
+#define INCLUDED_AUDIO_JACK_IMPL_H
#include <stdio.h>
-#endif /* INCLUDED_GRI_JACK_H */
+#endif /* INCLUDED_AUDIO_JACK_IMPL_H */
diff --git a/gr-audio/lib/jack/jack_sink.cc b/gr-audio/lib/jack/jack_sink.cc
new file mode 100644
index 0000000000..9e9d1e34db
--- /dev/null
+++ b/gr-audio/lib/jack/jack_sink.cc
@@ -0,0 +1,241 @@
+/* -*- c++ -*- */
+/*
+ * Copyright 2005-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 <jack_sink.h>
+#include <jack_impl.h>
+#include <gr_io_signature.h>
+#include <gr_prefs.h>
+#include <stdio.h>
+#include <iostream>
+#include <stdexcept>
+
+#ifndef NO_PTHREAD
+#include <pthread.h>
+#endif
+
+namespace gr {
+ namespace audio {
+
+ AUDIO_REGISTER_SINK(REG_PRIO_MED, jack)(int sampling_rate,
+ const std::string &device_name,
+ bool ok_to_block)
+ {
+ return sink::sptr
+ (new jack_sink(sampling_rate, device_name, ok_to_block));
+ }
+
+ typedef jack_default_audio_sample_t sample_t;
+
+ // Number of jack buffers in the ringbuffer
+ // TODO: make it to match at least the quantity of items passed by work()
+ static const unsigned int N_BUFFERS = 16;
+
+ static std::string
+ default_device_name()
+ {
+ return gr_prefs::singleton()->get_string
+ ("audio_jack", "default_output_device", "gr_sink");
+ }
+
+ int
+ jack_sink_process(jack_nframes_t nframes, void *arg)
+ {
+ jack_sink *self = (jack_sink *)arg;
+ unsigned int read_size = nframes*sizeof(sample_t);
+
+ if(jack_ringbuffer_read_space(self->d_ringbuffer) < read_size) {
+ self->d_nunderuns++;
+ // FIXME: move this fputs out, we shouldn't use blocking calls in process()
+ fputs("jU", stderr);
+ return 0;
+ }
+
+ char *buffer = (char *)jack_port_get_buffer(self->d_jack_output_port, nframes);
+
+ jack_ringbuffer_read(self->d_ringbuffer, buffer, read_size);
+
+#ifndef NO_PTHREAD
+ // Tell the sink thread there is room in the ringbuffer.
+ // If it is already running, the lock will not be available.
+ // We can't wait here in the process() thread, but we don't
+ // need to signal in that case, because the sink thread will
+ // check for room availability.
+ if(pthread_mutex_trylock (&self->d_jack_process_lock) == 0) {
+ pthread_cond_signal(&self->d_ringbuffer_ready);
+ pthread_mutex_unlock(&self->d_jack_process_lock);
+ }
+#endif
+
+ return 0;
+ }
+
+ // ----------------------------------------------------------------
+
+ jack_sink::jack_sink(int sampling_rate,
+ const std::string device_name,
+ bool ok_to_block)
+ : gr_sync_block("audio_jack_sink",
+ gr_make_io_signature(0, 0, 0),
+ gr_make_io_signature(0, 0, 0)),
+ d_sampling_rate(sampling_rate),
+ d_device_name(device_name.empty() ? default_device_name() : device_name),
+ d_ok_to_block(ok_to_block),
+ d_jack_client(0),
+ d_ringbuffer(0),
+ d_nunderuns(0)
+ {
+#ifndef NO_PTHREAD
+ pthread_cond_init(&d_ringbuffer_ready, NULL);;
+ pthread_mutex_init(&d_jack_process_lock, NULL);
+#endif
+
+ // try to become a client of the JACK server
+ jack_options_t options = JackNullOption;
+ jack_status_t status;
+ const char *server_name = NULL;
+ if((d_jack_client = jack_client_open(d_device_name.c_str(),
+ options, &status,
+ server_name)) == NULL) {
+ fprintf(stderr, "audio_jack_sink[%s]: jack server not running?\n",
+ d_device_name.c_str());
+ throw std::runtime_error("audio_jack_sink");
+ }
+
+ // tell the JACK server to call `jack_sink_process()' whenever
+ // there is work to be done.
+ jack_set_process_callback(d_jack_client, &jack_sink_process, (void*)this);
+
+ // tell the JACK server to call `jack_shutdown()' if
+ // it ever shuts down, either entirely, or if it
+ // just decides to stop calling us.
+
+ //jack_on_shutdown (d_jack_client, &jack_shutdown, (void*)this);
+
+ d_jack_output_port =
+ jack_port_register(d_jack_client, "out",
+ JACK_DEFAULT_AUDIO_TYPE, JackPortIsOutput, 0);
+
+
+ d_jack_buffer_size = jack_get_buffer_size(d_jack_client);
+
+ set_output_multiple(d_jack_buffer_size);
+
+ d_ringbuffer =
+ jack_ringbuffer_create(N_BUFFERS*d_jack_buffer_size*sizeof(sample_t));
+ if(d_ringbuffer == NULL)
+ bail("jack_ringbuffer_create failed", 0);
+
+ assert(sizeof(float)==sizeof(sample_t));
+ set_input_signature(gr_make_io_signature(1, 1, sizeof(sample_t)));
+
+ jack_nframes_t sample_rate = jack_get_sample_rate(d_jack_client);
+
+ if((jack_nframes_t)sampling_rate != sample_rate) {
+ fprintf(stderr, "audio_jack_sink[%s]: unable to support sampling rate %d\n",
+ d_device_name.c_str(), sampling_rate);
+ fprintf(stderr, " card requested %d instead.\n", sample_rate);
+ }
+ }
+
+ bool
+ jack_sink::check_topology (int ninputs, int noutputs)
+ {
+ if(ninputs != 1)
+ return false;
+
+ // tell the JACK server that we are ready to roll
+ if(jack_activate (d_jack_client))
+ throw std::runtime_error("audio_jack_sink");
+
+ return true;
+ }
+
+ jack_sink::~jack_sink()
+ {
+ jack_client_close(d_jack_client);
+ jack_ringbuffer_free(d_ringbuffer);
+ }
+
+ int
+ jack_sink::work(int noutput_items,
+ gr_vector_const_void_star &input_items,
+ gr_vector_void_star &output_items)
+ {
+ // write_size and work_size are in bytes
+ int work_size = noutput_items*sizeof(sample_t);
+ unsigned int write_size;
+
+ while(work_size > 0) {
+ unsigned int write_space; // bytes
+
+#ifdef NO_PTHREAD
+ while((write_space=jack_ringbuffer_write_space(d_ringbuffer)) <
+ d_jack_buffer_size*sizeof(sample_t)) {
+ usleep(1000000*((d_jack_buffer_size-write_space/sizeof(sample_t))/d_sampling_rate));
+ }
+#else
+ // JACK actually requires POSIX
+
+ pthread_mutex_lock(&d_jack_process_lock);
+ while((write_space = jack_ringbuffer_write_space(d_ringbuffer)) <
+ d_jack_buffer_size*sizeof(sample_t)) {
+
+ // wait until jack_sink_process() signals more room
+ pthread_cond_wait(&d_ringbuffer_ready, &d_jack_process_lock);
+ }
+ pthread_mutex_unlock(&d_jack_process_lock);
+#endif
+
+ write_space -= write_space%(d_jack_buffer_size*sizeof(sample_t));
+ write_size = std::min(write_space, (unsigned int)work_size);
+
+ if(jack_ringbuffer_write(d_ringbuffer, (char *) input_items[0],
+ write_size) < write_size) {
+ bail("jack_ringbuffer_write failed", 0);
+ }
+ work_size -= write_size;
+ }
+
+ return noutput_items;
+ }
+
+ void
+ jack_sink::output_error_msg(const char *msg, int err)
+ {
+ fprintf(stderr, "audio_jack_sink[%s]: %s: %d\n",
+ d_device_name.c_str(), msg, err);
+ }
+
+ void
+ jack_sink::bail(const char *msg, int err) throw (std::runtime_error)
+ {
+ output_error_msg(msg, err);
+ throw std::runtime_error("audio_jack_sink");
+ }
+
+ } /* namespace audio */
+} /* namespace gr */
diff --git a/gr-audio/lib/jack/jack_sink.h b/gr-audio/lib/jack/jack_sink.h
new file mode 100644
index 0000000000..2caecbd54c
--- /dev/null
+++ b/gr-audio/lib/jack/jack_sink.h
@@ -0,0 +1,86 @@
+/* -*- c++ -*- */
+/*
+ * Copyright 2005-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.
+ */
+#ifndef INCLUDED_AUDIO_JACK_SINK_H
+#define INCLUDED_AUDIO_JACK_SINK_H
+
+#include <audio/sink.h>
+#include <jack/jack.h>
+#include <jack/ringbuffer.h>
+#include <string>
+#include <stdexcept>
+
+namespace gr {
+ namespace audio {
+
+ int sink_process(jack_nframes_t nframes, void *arg);
+
+ /*!
+ * \brief audio sink using JACK
+ * \ingroup audio_blk
+ *
+ * The sink has one input stream of floats.
+ *
+ * Input samples must be in the range [-1,1].
+ */
+ class jack_sink : public sink
+ {
+ friend int jack_sink_process(jack_nframes_t nframes, void *arg);
+
+ // typedef for pointer to class work method
+ typedef int (jack_sink::*work_t)(int noutput_items,
+ gr_vector_const_void_star &input_items,
+ gr_vector_void_star &output_items);
+
+ unsigned int d_sampling_rate;
+ std::string d_device_name;
+ bool d_ok_to_block;
+
+ jack_client_t *d_jack_client;
+ jack_port_t *d_jack_output_port;
+ jack_ringbuffer_t *d_ringbuffer;
+ jack_nframes_t d_jack_buffer_size;
+ pthread_cond_t d_ringbuffer_ready;
+ pthread_mutex_t d_jack_process_lock;
+
+ // random stats
+ int d_nunderuns; // count of underruns
+
+ void output_error_msg(const char *msg, int err);
+ void bail(const char *msg, int err) throw (std::runtime_error);
+
+ public:
+ jack_sink(int sampling_rate,
+ const std::string device_name,
+ bool ok_to_block);
+ ~jack_sink();
+
+ bool check_topology(int ninputs, int noutputs);
+
+ int work(int noutput_items,
+ gr_vector_const_void_star &input_items,
+ gr_vector_void_star &output_items);
+ };
+
+ } /* namespace audio */
+} /* namespace gr */
+
+#endif /* INCLUDED_AUDIO_JACK_SINK_H */
diff --git a/gr-audio/lib/jack/jack_source.cc b/gr-audio/lib/jack/jack_source.cc
new file mode 100644
index 0000000000..e5a46e3416
--- /dev/null
+++ b/gr-audio/lib/jack/jack_source.cc
@@ -0,0 +1,240 @@
+/* -*- c++ -*- */
+/*
+ * Copyright 2005,2006,2010,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 <jack_source.h>
+#include <jack_impl.h>
+#include <gr_io_signature.h>
+#include <gr_prefs.h>
+#include <stdio.h>
+#include <iostream>
+#include <stdexcept>
+
+#ifndef NO_PTHREAD
+#include <pthread.h>
+#endif
+
+namespace gr {
+ namespace audio {
+
+ AUDIO_REGISTER_SOURCE(REG_PRIO_MED, jack)(int sampling_rate,
+ const std::string &device_name,
+ bool ok_to_block)
+ {
+ return source::sptr
+ (new jack_source(sampling_rate, device_name, ok_to_block));
+ }
+
+ typedef jack_default_audio_sample_t sample_t;
+
+ // Number of jack buffers in the ringbuffer
+ // TODO: make it to match at least the quantity of items passed to work()
+ static const unsigned int N_BUFFERS = 16;
+
+ static std::string
+ default_device_name()
+ {
+ return gr_prefs::singleton()->get_string
+ ("audio_jack", "default_input_device", "gr_source");
+ }
+
+ int
+ jack_source_process(jack_nframes_t nframes, void *arg)
+ {
+ jack_source *self = (jack_source *)arg;
+ unsigned int write_size = nframes*sizeof(sample_t);
+
+ if(jack_ringbuffer_write_space (self->d_ringbuffer) < write_size) {
+ self->d_noverruns++;
+ // FIXME: move this fputs out, we shouldn't use blocking calls in process()
+ fputs ("jO", stderr);
+ return 0;
+ }
+
+ char *buffer = (char *)jack_port_get_buffer(self->d_jack_input_port, nframes);
+
+ jack_ringbuffer_write (self->d_ringbuffer, buffer, write_size);
+
+#ifndef NO_PTHREAD
+ // Tell the source thread there is data in the ringbuffer.
+ // If it is already running, the lock will not be available.
+ // We can't wait here in the process() thread, but we don't
+ // need to signal in that case, because the source thread will
+ // check for data availability.
+
+ if(pthread_mutex_trylock(&self->d_jack_process_lock) == 0) {
+ pthread_cond_signal(&self->d_ringbuffer_ready);
+ pthread_mutex_unlock(&self->d_jack_process_lock);
+ }
+#endif
+
+ return 0;
+ }
+
+ // ----------------------------------------------------------------
+
+ jack_source::jack_source(int sampling_rate,
+ const std::string device_name,
+ bool ok_to_block)
+ : gr_sync_block("audio_jack_source",
+ gr_make_io_signature(0, 0, 0),
+ gr_make_io_signature(0, 0, 0)),
+ d_sampling_rate(sampling_rate),
+ d_device_name(device_name.empty() ? default_device_name() : device_name),
+ d_ok_to_block(ok_to_block),
+ d_jack_client(0),
+ d_ringbuffer(0),
+ d_noverruns(0)
+ {
+#ifndef NO_PTHREAD
+ pthread_cond_init(&d_ringbuffer_ready, NULL);;
+ pthread_mutex_init(&d_jack_process_lock, NULL);
+#endif
+
+ // try to become a client of the JACK server
+ jack_options_t options = JackNullOption;
+ jack_status_t status;
+ const char *server_name = NULL;
+ if((d_jack_client = jack_client_open(d_device_name.c_str(),
+ options, &status,
+ server_name)) == NULL) {
+ fprintf(stderr, "audio_jack_source[%s]: jack server not running?\n",
+ d_device_name.c_str());
+ throw std::runtime_error("audio_jack_source");
+ }
+
+ // tell the JACK server to call `jack_source_process()' whenever
+ // there is work to be done.
+ jack_set_process_callback(d_jack_client, &jack_source_process, (void*)this);
+
+ // tell the JACK server to call `jack_shutdown()' if
+ // it ever shuts down, either entirely, or if it
+ // just decides to stop calling us.
+
+ //jack_on_shutdown (d_jack_client, &jack_shutdown, (void*)this);
+
+ d_jack_input_port = jack_port_register(d_jack_client, "in",
+ JACK_DEFAULT_AUDIO_TYPE,
+ JackPortIsInput, 0);
+
+ d_jack_buffer_size = jack_get_buffer_size(d_jack_client);
+
+ set_output_multiple(d_jack_buffer_size);
+
+ d_ringbuffer = jack_ringbuffer_create(N_BUFFERS*d_jack_buffer_size*sizeof(sample_t));
+ if(d_ringbuffer == NULL)
+ bail("jack_ringbuffer_create failed", 0);
+
+ assert(sizeof(float)==sizeof(sample_t));
+ set_output_signature(gr_make_io_signature(1, 1, sizeof(sample_t)));
+
+ jack_nframes_t sample_rate = jack_get_sample_rate(d_jack_client);
+
+ if((jack_nframes_t)sampling_rate != sample_rate) {
+ fprintf(stderr, "audio_jack_source[%s]: unable to support sampling rate %d\n",
+ d_device_name.c_str(), sampling_rate);
+ fprintf(stderr, " card requested %d instead.\n", sample_rate);
+ }
+ }
+
+ bool
+ jack_source::check_topology(int ninputs, int noutputs)
+ {
+ // tell the JACK server that we are ready to roll
+ if(jack_activate (d_jack_client))
+ throw std::runtime_error("audio_jack_source");
+
+ return true;
+ }
+
+ jack_source::~jack_source()
+ {
+ jack_client_close(d_jack_client);
+ jack_ringbuffer_free(d_ringbuffer);
+ }
+
+ int
+ jack_source::work(int noutput_items,
+ gr_vector_const_void_star &input_items,
+ gr_vector_void_star &output_items)
+ {
+ // read_size and work_size are in bytes
+ unsigned int read_size;
+
+ // Minimize latency
+ noutput_items = std::min (noutput_items, (int)d_jack_buffer_size);
+
+ int work_size = noutput_items*sizeof(sample_t);
+
+ while(work_size > 0) {
+ unsigned int read_space; // bytes
+
+#ifdef NO_PTHREAD
+ while((read_space=jack_ringbuffer_read_space (d_ringbuffer)) <
+ d_jack_buffer_size*sizeof(sample_t)) {
+ usleep(1000000*((d_jack_buffer_size-read_space/sizeof(sample_t))/d_sampling_rate));
+ }
+#else
+
+ // JACK actually requires POSIX
+ pthread_mutex_lock(&d_jack_process_lock);
+ while((read_space = jack_ringbuffer_read_space(d_ringbuffer)) <
+ d_jack_buffer_size*sizeof(sample_t)) {
+ // wait until jack_source_process() signals more data
+ pthread_cond_wait(&d_ringbuffer_ready, &d_jack_process_lock);
+ }
+ pthread_mutex_unlock(&d_jack_process_lock);
+#endif
+
+ read_space -= read_space%(d_jack_buffer_size*sizeof(sample_t));
+ read_size = std::min(read_space, (unsigned int)work_size);
+
+ if(jack_ringbuffer_read(d_ringbuffer, (char *) output_items[0],
+ read_size) < read_size) {
+ bail("jack_ringbuffer_read failed", 0);
+ }
+ work_size -= read_size;
+ }
+
+ return noutput_items;
+ }
+
+ void
+ jack_source::output_error_msg(const char *msg, int err)
+ {
+ fprintf(stderr, "audio_jack_source[%s]: %s: %d\n",
+ d_device_name.c_str(), msg, err);
+ }
+
+ void
+ jack_source::bail(const char *msg, int err) throw (std::runtime_error)
+ {
+ output_error_msg(msg, err);
+ throw std::runtime_error("audio_jack_source");
+ }
+
+ } /* namespace audio */
+} /* namespace gr */
diff --git a/gr-audio/lib/jack/jack_source.h b/gr-audio/lib/jack/jack_source.h
new file mode 100644
index 0000000000..f096220b26
--- /dev/null
+++ b/gr-audio/lib/jack/jack_source.h
@@ -0,0 +1,86 @@
+/* -*- c++ -*- */
+/*
+ * Copyright 2005-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.
+ */
+#ifndef INCLUDED_AUDIO_JACK_SOURCE_H
+#define INCLUDED_AUDIO_JACK_SOURCE_H
+
+#include <audio/source.h>
+#include <jack/jack.h>
+#include <jack/ringbuffer.h>
+#include <string>
+#include <stdexcept>
+
+namespace gr {
+ namespace audio {
+
+ int jack_source_process(jack_nframes_t nframes, void *arg);
+
+ /*!
+ * \brief audio source using JACK
+ * \ingroup audio_blk
+ *
+ * The source has one input stream of floats.
+ *
+ * Output samples will be in the range [-1,1].
+ */
+ class jack_source : public source
+ {
+ friend int jack_source_process(jack_nframes_t nframes, void *arg);
+
+ // typedef for pointer to class work method
+ typedef int(jack_source::*work_t)(int noutput_items,
+ gr_vector_const_void_star &input_items,
+ gr_vector_void_star &output_items);
+
+ unsigned int d_sampling_rate;
+ std::string d_device_name;
+ bool d_ok_to_block;
+
+ jack_client_t *d_jack_client;
+ jack_port_t *d_jack_input_port;
+ jack_ringbuffer_t *d_ringbuffer;
+ jack_nframes_t d_jack_buffer_size;
+ pthread_cond_t d_ringbuffer_ready;
+ pthread_mutex_t d_jack_process_lock;
+
+ // random stats
+ int d_noverruns; // count of overruns
+
+ void output_error_msg(const char *msg, int err);
+ void bail(const char *msg, int err) throw (std::runtime_error);
+
+ public:
+ jack_source(int sampling_rate,
+ const std::string device_name,
+ bool ok_to_block);
+ ~jack_source();
+
+ bool check_topology(int ninputs, int noutputs);
+
+ int work(int noutput_items,
+ gr_vector_const_void_star &input_items,
+ gr_vector_void_star &output_items);
+ };
+
+ } /* namespace audio */
+} /* namespace gr */
+
+#endif /* INCLUDED_AUDIO_JACK_SOURCE_H */
diff --git a/gr-audio/lib/oss/audio_oss_sink.cc b/gr-audio/lib/oss/audio_oss_sink.cc
deleted file mode 100644
index 26b71be241..0000000000
--- a/gr-audio/lib/oss/audio_oss_sink.cc
+++ /dev/null
@@ -1,161 +0,0 @@
-/* -*- c++ -*- */
-/*
- * Copyright 2004-2011 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 "gr_audio_registry.h"
-#include <audio_oss_sink.h>
-#include <gr_io_signature.h>
-#include <gr_prefs.h>
-#include <sys/soundcard.h>
-#include <sys/ioctl.h>
-#include <sys/types.h>
-#include <sys/stat.h>
-#include <fcntl.h>
-#include <unistd.h>
-#include <stdio.h>
-#include <iostream>
-#include <stdexcept>
-
-AUDIO_REGISTER_SINK(REG_PRIO_LOW, oss)(
- int sampling_rate, const std::string &device_name, bool ok_to_block
-){
- return audio_sink::sptr(new audio_oss_sink(sampling_rate, device_name, ok_to_block));
-}
-
-static std::string
-default_device_name ()
-{
- return gr_prefs::singleton()->get_string("audio_oss", "default_output_device", "/dev/dsp");
-}
-
-audio_oss_sink::audio_oss_sink (int sampling_rate,
- const std::string device_name,
- bool ok_to_block)
- : gr_sync_block ("audio_oss_sink",
- gr_make_io_signature (1, 2, sizeof (float)),
- gr_make_io_signature (0, 0, 0)),
- d_sampling_rate (sampling_rate),
- d_device_name (device_name.empty() ? default_device_name() : device_name),
- d_fd (-1), d_buffer (0), d_chunk_size (0)
-{
- if ((d_fd = open (d_device_name.c_str (), O_WRONLY)) < 0){
- fprintf (stderr, "audio_oss_sink: ");
- perror (d_device_name.c_str ());
- throw std::runtime_error ("audio_oss_sink");
- }
-
- double CHUNK_TIME =
- std::max(0.001, gr_prefs::singleton()->get_double("audio_oss", "latency", 0.005));
-
- d_chunk_size = (int) (d_sampling_rate * CHUNK_TIME);
- set_output_multiple (d_chunk_size);
-
- d_buffer = new short [d_chunk_size * 2];
-
- int format = AFMT_S16_NE;
- int orig_format = format;
- if (ioctl (d_fd, SNDCTL_DSP_SETFMT, &format) < 0){
- std::cerr << "audio_oss_sink: " << d_device_name << " ioctl failed\n";
- perror (d_device_name.c_str ());
- throw std::runtime_error ("audio_oss_sink");
- }
-
- if (format != orig_format){
- fprintf (stderr, "audio_oss_sink: unable to support format %d\n", orig_format);
- fprintf (stderr, " card requested %d instead.\n", format);
- }
-
- // set to stereo no matter what. Some hardware only does stereo
- int channels = 2;
- if (ioctl (d_fd, SNDCTL_DSP_CHANNELS, &channels) < 0 || channels != 2){
- perror ("audio_oss_sink: could not set STEREO mode");
- throw std::runtime_error ("audio_oss_sink");
- }
-
- // set sampling freq
- int sf = sampling_rate;
- if (ioctl (d_fd, SNDCTL_DSP_SPEED, &sf) < 0){
- std::cerr << "audio_oss_sink: "
- << d_device_name << ": invalid sampling_rate "
- << sampling_rate << "\n";
- sampling_rate = 8000;
- if (ioctl (d_fd, SNDCTL_DSP_SPEED, &sf) < 0){
- std::cerr << "audio_oss_sink: failed to set sampling_rate to 8000\n";
- throw std::runtime_error ("audio_oss_sink");
- }
- }
-}
-
-audio_oss_sink::~audio_oss_sink ()
-{
- close (d_fd);
- delete [] d_buffer;
-}
-
-
-int
-audio_oss_sink::work (int noutput_items,
- gr_vector_const_void_star &input_items,
- gr_vector_void_star &output_items)
-{
- const float *f0, *f1;
-
- 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 (d_fd, d_buffer, 2 * d_chunk_size * sizeof (short)) < 0)
- perror ("audio_oss_sink: write");
- }
- 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 (d_fd, d_buffer, 2 * d_chunk_size * sizeof (short)) < 0)
- perror ("audio_oss_sink: write");
- }
- break;
- }
-
- return noutput_items;
-}
diff --git a/gr-audio/lib/oss/audio_oss_source.cc b/gr-audio/lib/oss/audio_oss_source.cc
deleted file mode 100644
index e186e30aea..0000000000
--- a/gr-audio/lib/oss/audio_oss_source.cc
+++ /dev/null
@@ -1,177 +0,0 @@
-/* -*- c++ -*- */
-/*
- * Copyright 2004-2011 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 "gr_audio_registry.h"
-#include <audio_oss_source.h>
-#include <gr_io_signature.h>
-#include <gr_prefs.h>
-#include <sys/soundcard.h>
-#include <sys/ioctl.h>
-#include <sys/types.h>
-#include <sys/stat.h>
-#include <fcntl.h>
-#include <unistd.h>
-#include <stdio.h>
-#include <iostream>
-#include <stdexcept>
-
-AUDIO_REGISTER_SOURCE(REG_PRIO_LOW, oss)(
- int sampling_rate, const std::string &device_name, bool ok_to_block
-){
- return audio_source::sptr(new audio_oss_source(sampling_rate, device_name, ok_to_block));
-}
-
-static std::string
-default_device_name ()
-{
- return gr_prefs::singleton()->get_string("audio_oss", "default_input_device", "/dev/dsp");
-}
-
-audio_oss_source::audio_oss_source (int sampling_rate,
- const std::string device_name,
- bool ok_to_block)
- : gr_sync_block ("audio_oss_source",
- gr_make_io_signature (0, 0, 0),
- gr_make_io_signature (1, 2, sizeof (float))),
- d_sampling_rate (sampling_rate),
- d_device_name (device_name.empty() ? default_device_name() : device_name),
- d_fd (-1), d_buffer (0), d_chunk_size (0)
-{
- if ((d_fd = open (d_device_name.c_str (), O_RDONLY)) < 0){
- fprintf (stderr, "audio_oss_source: ");
- perror (d_device_name.c_str ());
- throw std::runtime_error ("audio_oss_source");
- }
-
- double CHUNK_TIME =
- std::max(0.001, gr_prefs::singleton()->get_double("audio_oss", "latency", 0.005));
-
- d_chunk_size = (int) (d_sampling_rate * CHUNK_TIME);
- set_output_multiple (d_chunk_size);
-
- d_buffer = new short [d_chunk_size * 2];
-
- int format = AFMT_S16_NE;
- int orig_format = format;
- if (ioctl (d_fd, SNDCTL_DSP_SETFMT, &format) < 0){
- std::cerr << "audio_oss_source: " << d_device_name << " ioctl failed\n";
- perror (d_device_name.c_str ());
- throw std::runtime_error ("audio_oss_source");
- }
-
- if (format != orig_format){
- fprintf (stderr, "audio_oss_source: unable to support format %d\n", orig_format);
- fprintf (stderr, " card requested %d instead.\n", format);
- }
-
- // set to stereo no matter what. Some hardware only does stereo
- int channels = 2;
- if (ioctl (d_fd, SNDCTL_DSP_CHANNELS, &channels) < 0 || channels != 2){
- perror ("audio_oss_source: could not set STEREO mode");
- throw std::runtime_error ("audio_oss_source");
- }
-
- // set sampling freq
- int sf = sampling_rate;
- if (ioctl (d_fd, SNDCTL_DSP_SPEED, &sf) < 0){
- std::cerr << "audio_oss_source: "
- << d_device_name << ": invalid sampling_rate "
- << sampling_rate << "\n";
- sampling_rate = 8000;
- if (ioctl (d_fd, SNDCTL_DSP_SPEED, &sf) < 0){
- std::cerr << "audio_oss_source: failed to set sampling_rate to 8000\n";
- throw std::runtime_error ("audio_oss_source");
- }
- }
-}
-
-audio_oss_source::~audio_oss_source ()
-{
- close (d_fd);
- delete [] d_buffer;
-}
-
-int
-audio_oss_source::work (int noutput_items,
- gr_vector_const_void_star &input_items,
- gr_vector_void_star &output_items)
-{
- float *f0 = (float *) output_items[0];
- float *f1 = (float *) output_items[1]; // will be invalid if this is mono output
-
- const int shorts_per_item = 2; // L + R
- const int bytes_per_item = shorts_per_item * sizeof (short);
-
- // To minimize latency, never return more than CHUNK_TIME
- // worth of samples per call to work.
-
- noutput_items = std::min (noutput_items, d_chunk_size);
-
- int base = 0;
- int ntogo = noutput_items;
-
- while (ntogo > 0){
- int nbytes = std::min (ntogo, d_chunk_size) * bytes_per_item;
- int result_nbytes = read (d_fd, d_buffer, nbytes);
-
- if (result_nbytes < 0){
- perror ("audio_oss_source");
- return -1; // say we're done
- }
-
- if ((result_nbytes & (bytes_per_item - 1)) != 0){
- fprintf (stderr, "audio_oss_source: internal error.\n");
- throw std::runtime_error ("internal error");
- }
-
- int result_nitems = result_nbytes / bytes_per_item;
-
- // now unpack samples into output streams
-
- switch (output_items.size ()){
- case 1: // mono output
- for (int i = 0; i < result_nitems; i++){
- f0[base+i] = d_buffer[2*i+0] * (1.0 / 32767);
- }
- break;
-
- case 2: // stereo output
- for (int i = 0; i < result_nitems; i++){
- f0[base+i] = d_buffer[2*i+0] * (1.0 / 32767);
- f1[base+i] = d_buffer[2*i+1] * (1.0 / 32767);
- }
- break;
-
- default:
- assert (0);
- }
-
- ntogo -= result_nitems;
- base += result_nitems;
- }
-
- return noutput_items - ntogo;
-}
diff --git a/gr-audio/lib/oss/oss_sink.cc b/gr-audio/lib/oss/oss_sink.cc
new file mode 100644
index 0000000000..129e771fd6
--- /dev/null
+++ b/gr-audio/lib/oss/oss_sink.cc
@@ -0,0 +1,167 @@
+/* -*- 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 <oss_sink.h>
+#include <gr_io_signature.h>
+#include <gr_prefs.h>
+#include <sys/soundcard.h>
+#include <sys/ioctl.h>
+#include <sys/types.h>
+#include <sys/stat.h>
+#include <fcntl.h>
+#include <unistd.h>
+#include <stdio.h>
+#include <iostream>
+#include <stdexcept>
+
+namespace gr {
+ namespace audio {
+
+ AUDIO_REGISTER_SINK(REG_PRIO_LOW, oss)(int sampling_rate,
+ const std::string &device_name,
+ bool ok_to_block)
+ {
+ return sink::sptr
+ (new oss_sink(sampling_rate, device_name, ok_to_block));
+ }
+
+ static std::string
+ default_device_name()
+ {
+ return gr_prefs::singleton()->get_string
+ ("audio_oss", "default_output_device", "/dev/dsp");
+ }
+
+ oss_sink::oss_sink(int sampling_rate,
+ const std::string device_name,
+ bool ok_to_block)
+ : gr_sync_block("audio_oss_sink",
+ gr_make_io_signature(1, 2, sizeof(float)),
+ gr_make_io_signature(0, 0, 0)),
+ d_sampling_rate(sampling_rate),
+ d_device_name(device_name.empty() ? default_device_name() : device_name),
+ d_fd(-1), d_buffer(0), d_chunk_size(0)
+ {
+ if((d_fd = open(d_device_name.c_str(), O_WRONLY)) < 0) {
+ fprintf(stderr, "audio_oss_sink: ");
+ perror(d_device_name.c_str());
+ throw std::runtime_error("audio_oss_sink");
+ }
+
+ double CHUNK_TIME =
+ std::max(0.001,
+ gr_prefs::singleton()->get_double("audio_oss", "latency", 0.005));
+
+ d_chunk_size = (int)(d_sampling_rate * CHUNK_TIME);
+ set_output_multiple(d_chunk_size);
+
+ d_buffer = new short[d_chunk_size * 2];
+
+ int format = AFMT_S16_NE;
+ int orig_format = format;
+ if(ioctl(d_fd, SNDCTL_DSP_SETFMT, &format) < 0) {
+ std::cerr << "audio_oss_sink: " << d_device_name << " ioctl failed\n";
+ perror(d_device_name.c_str ());
+ throw std::runtime_error("audio_oss_sink");
+ }
+
+ if(format != orig_format) {
+ fprintf(stderr, "audio_oss_sink: unable to support format %d\n", orig_format);
+ fprintf(stderr, " card requested %d instead.\n", format);
+ }
+
+ // set to stereo no matter what. Some hardware only does stereo
+ int channels = 2;
+ if(ioctl(d_fd, SNDCTL_DSP_CHANNELS, &channels) < 0 || channels != 2) {
+ perror("audio_oss_sink: could not set STEREO mode");
+ throw std::runtime_error("audio_oss_sink");
+ }
+
+ // set sampling freq
+ int sf = sampling_rate;
+ if(ioctl(d_fd, SNDCTL_DSP_SPEED, &sf) < 0) {
+ std::cerr << "audio_oss_sink: "
+ << d_device_name << ": invalid sampling_rate "
+ << sampling_rate << "\n";
+ sampling_rate = 8000;
+ if(ioctl(d_fd, SNDCTL_DSP_SPEED, &sf) < 0) {
+ std::cerr << "audio_oss_sink: failed to set sampling_rate to 8000\n";
+ throw std::runtime_error("audio_oss_sink");
+ }
+ }
+ }
+
+ oss_sink::~oss_sink()
+ {
+ close(d_fd);
+ delete [] d_buffer;
+ }
+
+ int
+ oss_sink::work(int noutput_items,
+ gr_vector_const_void_star &input_items,
+ gr_vector_void_star &output_items)
+ {
+ const float *f0, *f1;
+
+ 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(d_fd, d_buffer, 2 * d_chunk_size * sizeof (short)) < 0)
+ perror("audio_oss_sink: write");
+ }
+ 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(d_fd, d_buffer, 2 * d_chunk_size * sizeof (short)) < 0)
+ perror("audio_oss_sink: write");
+ }
+ break;
+ }
+
+ return noutput_items;
+ }
+
+ } /* namespace audio */
+} /* namespace gr */
diff --git a/gr-audio/lib/oss/audio_oss_sink.h b/gr-audio/lib/oss/oss_sink.h
index 8148ec34b8..3bb5f4ee1a 100644
--- a/gr-audio/lib/oss/audio_oss_sink.h
+++ b/gr-audio/lib/oss/oss_sink.h
@@ -1,6 +1,6 @@
/* -*- c++ -*- */
/*
- * Copyright 2004-2011 Free Software Foundation, Inc.
+ * Copyright 2004-2011,2013 Free Software Foundation, Inc.
*
* This file is part of GNU Radio
*
@@ -23,33 +23,39 @@
#ifndef INCLUDED_AUDIO_OSS_SINK_H
#define INCLUDED_AUDIO_OSS_SINK_H
-#include <gr_audio_sink.h>
+#include <audio/sink.h>
#include <string>
-/*!
- * \brief audio sink using OSS
- * \ingroup audio_blk
- *
- * input signature is one or two streams of floats.
- * Input samples must be in the range [-1,1].
- */
-
-class audio_oss_sink : public audio_sink {
-
- int d_sampling_rate;
- std::string d_device_name;
- int d_fd;
- short *d_buffer;
- int d_chunk_size;
-
-public:
- audio_oss_sink (int sampling_rate, const std::string device_name = "", bool ok_to_block = true);
-
- ~audio_oss_sink ();
-
- int work (int noutput_items,
- gr_vector_const_void_star &input_items,
- gr_vector_void_star &output_items);
-};
+namespace gr {
+ namespace audio {
+
+ /*!
+ * \brief audio sink using OSS
+ * \ingroup audio_blk
+ *
+ * input signature is one or two streams of floats.
+ * Input samples must be in the range [-1,1].
+ */
+ class oss_sink : public sink
+ {
+ int d_sampling_rate;
+ std::string d_device_name;
+ int d_fd;
+ short *d_buffer;
+ int d_chunk_size;
+
+ public:
+ oss_sink(int sampling_rate,
+ const std::string device_name = "",
+ bool ok_to_block = true);
+ ~oss_sink();
+
+ int work(int noutput_items,
+ gr_vector_const_void_star &input_items,
+ gr_vector_void_star &output_items);
+ };
+
+ } /* namespace audio */
+} /* namespace gr */
#endif /* INCLUDED_AUDIO_OSS_SINK_H */
diff --git a/gr-audio/lib/oss/oss_source.cc b/gr-audio/lib/oss/oss_source.cc
new file mode 100644
index 0000000000..6d6bafceb8
--- /dev/null
+++ b/gr-audio/lib/oss/oss_source.cc
@@ -0,0 +1,186 @@
+/* -*- 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 <oss_source.h>
+#include <gr_io_signature.h>
+#include <gr_prefs.h>
+#include <sys/soundcard.h>
+#include <sys/ioctl.h>
+#include <sys/types.h>
+#include <sys/stat.h>
+#include <fcntl.h>
+#include <unistd.h>
+#include <stdio.h>
+#include <iostream>
+#include <stdexcept>
+
+namespace gr {
+ namespace audio {
+
+ AUDIO_REGISTER_SOURCE(REG_PRIO_LOW, oss)(int sampling_rate,
+ const std::string &device_name,
+ bool ok_to_block)
+ {
+ return source::sptr
+ (new oss_source(sampling_rate, device_name, ok_to_block));
+ }
+
+ static std::string
+ default_device_name()
+ {
+ return gr_prefs::singleton()->get_string
+ ("audio_oss", "default_input_device", "/dev/dsp");
+ }
+
+ oss_source::oss_source(int sampling_rate,
+ const std::string device_name,
+ bool ok_to_block)
+ : gr_sync_block("audio_oss_source",
+ gr_make_io_signature(0, 0, 0),
+ gr_make_io_signature(1, 2, sizeof(float))),
+ d_sampling_rate(sampling_rate),
+ d_device_name(device_name.empty() ? default_device_name() : device_name),
+ d_fd(-1), d_buffer(0), d_chunk_size(0)
+ {
+ if((d_fd = open(d_device_name.c_str(), O_RDONLY)) < 0) {
+ fprintf(stderr, "audio_oss_source: ");
+ perror(d_device_name.c_str());
+ throw std::runtime_error("audio_oss_source");
+ }
+
+ double CHUNK_TIME =
+ std::max(0.001, gr_prefs::singleton()->get_double("audio_oss", "latency", 0.005));
+
+ d_chunk_size = (int)(d_sampling_rate * CHUNK_TIME);
+ set_output_multiple(d_chunk_size);
+
+ d_buffer = new short[d_chunk_size * 2];
+
+ int format = AFMT_S16_NE;
+ int orig_format = format;
+ if(ioctl(d_fd, SNDCTL_DSP_SETFMT, &format) < 0) {
+ std::cerr << "audio_oss_source: " << d_device_name << " ioctl failed\n";
+ perror(d_device_name.c_str ());
+ throw std::runtime_error("audio_oss_source");
+ }
+
+ if(format != orig_format) {
+ fprintf(stderr, "audio_oss_source: unable to support format %d\n", orig_format);
+ fprintf(stderr, " card requested %d instead.\n", format);
+ }
+
+ // set to stereo no matter what. Some hardware only does stereo
+ int channels = 2;
+ if(ioctl(d_fd, SNDCTL_DSP_CHANNELS, &channels) < 0 || channels != 2) {
+ perror("audio_oss_source: could not set STEREO mode");
+ throw std::runtime_error("audio_oss_source");
+ }
+
+ // set sampling freq
+ int sf = sampling_rate;
+ if(ioctl(d_fd, SNDCTL_DSP_SPEED, &sf) < 0) {
+ std::cerr << "audio_oss_source: "
+ << d_device_name << ": invalid sampling_rate "
+ << sampling_rate << "\n";
+ sampling_rate = 8000;
+ if(ioctl(d_fd, SNDCTL_DSP_SPEED, &sf) < 0) {
+ std::cerr << "audio_oss_source: failed to set sampling_rate to 8000\n";
+ throw std::runtime_error ("audio_oss_source");
+ }
+ }
+ }
+
+ oss_source::~oss_source()
+ {
+ close(d_fd);
+ delete [] d_buffer;
+ }
+
+ int
+ oss_source::work(int noutput_items,
+ gr_vector_const_void_star &input_items,
+ gr_vector_void_star &output_items)
+ {
+ float *f0 = (float *)output_items[0];
+ float *f1 = (float *)output_items[1]; // will be invalid if this is mono output
+
+ const int shorts_per_item = 2; // L + R
+ const int bytes_per_item = shorts_per_item * sizeof(short);
+
+ // To minimize latency, never return more than CHUNK_TIME
+ // worth of samples per call to work.
+
+ noutput_items = std::min(noutput_items, d_chunk_size);
+
+ int base = 0;
+ int ntogo = noutput_items;
+
+ while(ntogo > 0) {
+ int nbytes = std::min(ntogo, d_chunk_size) * bytes_per_item;
+ int result_nbytes = read(d_fd, d_buffer, nbytes);
+
+ if(result_nbytes < 0) {
+ perror("audio_oss_source");
+ return -1; // say we're done
+ }
+
+ if((result_nbytes & (bytes_per_item - 1)) != 0) {
+ fprintf(stderr, "audio_oss_source: internal error.\n");
+ throw std::runtime_error("internal error");
+ }
+
+ int result_nitems = result_nbytes / bytes_per_item;
+
+ // now unpack samples into output streams
+
+ switch(output_items.size()) {
+ case 1: // mono output
+ for(int i = 0; i < result_nitems; i++) {
+ f0[base+i] = d_buffer[2*i+0] * (1.0 / 32767);
+ }
+ break;
+
+ case 2: // stereo output
+ for(int i = 0; i < result_nitems; i++) {
+ f0[base+i] = d_buffer[2*i+0] * (1.0 / 32767);
+ f1[base+i] = d_buffer[2*i+1] * (1.0 / 32767);
+ }
+ break;
+
+ default:
+ assert(0);
+ }
+
+ ntogo -= result_nitems;
+ base += result_nitems;
+ }
+
+ return noutput_items - ntogo;
+ }
+
+ } /* namespace audio */
+} /* namespace gr */
diff --git a/gr-audio/lib/oss/audio_oss_source.h b/gr-audio/lib/oss/oss_source.h
index abb2db1f8b..bf5183bd32 100644
--- a/gr-audio/lib/oss/audio_oss_source.h
+++ b/gr-audio/lib/oss/oss_source.h
@@ -1,6 +1,6 @@
/* -*- c++ -*- */
/*
- * Copyright 2004-2011 Free Software Foundation, Inc.
+ * Copyright 2004-2011,2013 Free Software Foundation, Inc.
*
* This file is part of GNU Radio
*
@@ -23,37 +23,40 @@
#ifndef INCLUDED_AUDIO_OSS_SOURCE_H
#define INCLUDED_AUDIO_OSS_SOURCE_H
-#include <gr_audio_source.h>
+#include <audio/source.h>
#include <string>
-/*!
- * \brief audio source using OSS
- * \ingroup audio_blk
- *
- * Output signature is one or two streams of floats.
- * Output samples will be in the range [-1,1].
- */
-
-class audio_oss_source : public audio_source {
-
- int d_sampling_rate;
- std::string d_device_name;
- int d_fd;
- short *d_buffer;
- int d_chunk_size;
-
-public:
- audio_oss_source (int sampling_rate,
- const std::string device_name = "",
- bool ok_to_block = true);
-
- ~audio_oss_source ();
-
- int work (int noutput_items,
- gr_vector_const_void_star &input_items,
- gr_vector_void_star &output_items);
-};
-
-
+namespace gr {
+ namespace audio {
+
+ /*!
+ * \brief audio source using OSS
+ * \ingroup audio_blk
+ *
+ * Output signature is one or two streams of floats.
+ * Output samples will be in the range [-1,1].
+ */
+ class oss_source : public source
+ {
+ int d_sampling_rate;
+ std::string d_device_name;
+ int d_fd;
+ short *d_buffer;
+ int d_chunk_size;
+
+ public:
+ oss_source(int sampling_rate,
+ const std::string device_name = "",
+ bool ok_to_block = true);
+
+ ~oss_source();
+
+ int work(int noutput_items,
+ gr_vector_const_void_star &input_items,
+ gr_vector_void_star &output_items);
+ };
+
+ } /* namespace audio */
+} /* namespace gr */
#endif /* INCLUDED_AUDIO_OSS_SOURCE_H */
diff --git a/gr-audio/lib/osx/audio_osx.h b/gr-audio/lib/osx/audio_osx.h
deleted file mode 100644
index 8c9543d0d6..0000000000
--- a/gr-audio/lib/osx/audio_osx.h
+++ /dev/null
@@ -1,72 +0,0 @@
-/* -*- c++ -*- */
-/*
- * Copyright 2006 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.
- */
-
-#ifndef INCLUDED_AUDIO_OSX_H
-#define INCLUDED_AUDIO_OSX_H
-
-#include <iostream>
-#include <string.h>
-
-#define CheckErrorAndThrow(err,what,throw_str) \
- if (err) { \
- OSStatus error = static_cast<OSStatus>(err); \
- char err_str[4]; \
- strncpy (err_str, (char*)(&err), 4); \
- std::cerr << what << std::endl; \
- std::cerr << " Error# " << error << " ('" << err_str \
- << "')" << std::endl; \
- std::cerr << " " << __FILE__ << ":" << __LINE__ << std::endl; \
- fflush (stderr); \
- throw std::runtime_error (throw_str); \
- }
-
-#define CheckError(err,what) \
- if (err) { \
- OSStatus error = static_cast<OSStatus>(err); \
- char err_str[4]; \
- strncpy (err_str, (char*)(&err), 4); \
- std::cerr << what << std::endl; \
- std::cerr << " Error# " << error << " ('" << err_str \
- << "')" << std::endl; \
- std::cerr << " " << __FILE__ << ":" << __LINE__ << std::endl; \
- fflush (stderr); \
- }
-
-#include <boost/detail/endian.hpp> //BOOST_BIG_ENDIAN
-#ifdef BOOST_BIG_ENDIAN
-#define GR_PCM_ENDIANNESS kLinearPCMFormatFlagIsBigEndian
-#else
-#define GR_PCM_ENDIANNESS 0
-#endif
-
-// Check the version of MacOSX being used
-#ifdef __APPLE_CC__
-#include <AvailabilityMacros.h>
-#ifndef MAC_OS_X_VERSION_10_6
-#define MAC_OS_X_VERSION_10_6 1060
-#endif
-#if MAC_OS_X_VERSION_MAX_ALLOWED < MAC_OS_X_VERSION_10_6
-#define GR_USE_OLD_AUDIO_UNIT
-#endif
-#endif
-
-#endif /* INCLUDED_AUDIO_OSX_H */
diff --git a/gr-audio/lib/osx/audio_osx_sink.cc b/gr-audio/lib/osx/audio_osx_sink.cc
deleted file mode 100644
index 939e5e0a1d..0000000000
--- a/gr-audio/lib/osx/audio_osx_sink.cc
+++ /dev/null
@@ -1,404 +0,0 @@
-/* -*- c++ -*- */
-/*
- * Copyright 2006-2011 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 "gr_audio_registry.h"
-#include <audio_osx_sink.h>
-#include <gr_io_signature.h>
-#include <stdexcept>
-#include <audio_osx.h>
-
-#define _OSX_AU_DEBUG_ 0
-
-AUDIO_REGISTER_SINK(REG_PRIO_HIGH, osx)(
- int sampling_rate, const std::string &device_name, bool ok_to_block
-){
- return audio_sink::sptr(new audio_osx_sink(sampling_rate, device_name, ok_to_block));
-}
-
-audio_osx_sink::audio_osx_sink (int sample_rate,
- const std::string device_name,
- bool do_block,
- int channel_config,
- int max_sample_count)
- : gr_sync_block ("audio_osx_sink",
- gr_make_io_signature (0, 0, 0),
- gr_make_io_signature (0, 0, 0)),
- d_sample_rate (0.0), d_channel_config (0), d_n_channels (0),
- d_queueSampleCount (0), d_max_sample_count (0),
- d_do_block (do_block), d_internal (0), d_cond_data (0),
- d_OutputAU (0)
-{
- if (sample_rate <= 0) {
- std::cerr << "Invalid Sample Rate: " << sample_rate << std::endl;
- throw std::invalid_argument ("audio_osx_sink::audio_osx_sink");
- } else
- d_sample_rate = (Float64) sample_rate;
-
- if (channel_config <= 0 & channel_config != -1) {
- std::cerr << "Invalid Channel Config: " << channel_config << std::endl;
- throw std::invalid_argument ("audio_osx_sink::audio_osx_sink");
- } else if (channel_config == -1) {
-// no user input; try "device name" instead
- int l_n_channels = (int) strtol (device_name.data(), (char **)NULL, 10);
- if (l_n_channels == 0 & errno) {
- std::cerr << "Error Converting Device Name: " << errno << std::endl;
- throw std::invalid_argument ("audio_osx_sink::audio_osx_sink");
- }
- if (l_n_channels <= 0)
- channel_config = 2;
- else
- channel_config = l_n_channels;
- }
-
- d_n_channels = d_channel_config = channel_config;
-
-// set the input signature
-
- set_input_signature (gr_make_io_signature (1, d_n_channels, sizeof (float)));
-
-// check that the max # of samples to store is valid
-
- if (max_sample_count == -1)
- max_sample_count = sample_rate;
- else if (max_sample_count <= 0) {
- std::cerr << "Invalid Max Sample Count: " << max_sample_count << std::endl;
- throw std::invalid_argument ("audio_osx_sink::audio_osx_sink");
- }
-
- d_max_sample_count = max_sample_count;
-
-// allocate the output circular buffer(s), one per channel
-
- d_buffers = (circular_buffer<float>**) new
- circular_buffer<float>* [d_n_channels];
- UInt32 n_alloc = (UInt32) ceil ((double) d_max_sample_count);
- for (UInt32 n = 0; n < d_n_channels; n++) {
- d_buffers[n] = new circular_buffer<float> (n_alloc, false, false);
- }
-
-// create the default AudioUnit for output
- OSStatus err = noErr;
-
-// Open the default output unit
-#ifndef GR_USE_OLD_AUDIO_UNIT
- AudioComponentDescription desc;
-#else
- ComponentDescription desc;
-#endif
-
- desc.componentType = kAudioUnitType_Output;
- desc.componentSubType = kAudioUnitSubType_DefaultOutput;
- desc.componentManufacturer = kAudioUnitManufacturer_Apple;
- desc.componentFlags = 0;
- desc.componentFlagsMask = 0;
-
-#ifndef GR_USE_OLD_AUDIO_UNIT
- AudioComponent comp = AudioComponentFindNext(NULL, &desc);
- if (comp == NULL) {
- std::cerr << "AudioComponentFindNext Error" << std::endl;
- throw std::runtime_error ("audio_osx_sink::audio_osx_sink");
- }
-#else
- Component comp = FindNextComponent (NULL, &desc);
- if (comp == NULL) {
- std::cerr << "FindNextComponent Error" << std::endl;
- throw std::runtime_error ("audio_osx_sink::audio_osx_sink");
- }
-#endif
-
-#ifndef GR_USE_OLD_AUDIO_UNIT
- err = AudioComponentInstanceNew (comp, &d_OutputAU);
- CheckErrorAndThrow (err, "AudioComponentInstanceNew", "audio_osx_sink::audio_osx_sink");
-#else
- err = OpenAComponent (comp, &d_OutputAU);
- CheckErrorAndThrow (err, "OpenAComponent", "audio_osx_sink::audio_osx_sink");
-#endif
-
-// Set up a callback function to generate output to the output unit
-
- AURenderCallbackStruct input;
- input.inputProc = (AURenderCallback)(audio_osx_sink::AUOutputCallback);
- input.inputProcRefCon = this;
-
- err = AudioUnitSetProperty (d_OutputAU,
- kAudioUnitProperty_SetRenderCallback,
- kAudioUnitScope_Input,
- 0,
- &input,
- sizeof (input));
- CheckErrorAndThrow (err, "AudioUnitSetProperty Render Callback", "audio_osx_sink::audio_osx_sink");
-
-// tell the Output Unit what format data will be supplied to it
-// so that it handles any format conversions
-
- AudioStreamBasicDescription streamFormat;
- streamFormat.mSampleRate = (Float64)(sample_rate);
- streamFormat.mFormatID = kAudioFormatLinearPCM;
- streamFormat.mFormatFlags = (kLinearPCMFormatFlagIsFloat |
- GR_PCM_ENDIANNESS |
- kLinearPCMFormatFlagIsPacked |
- kAudioFormatFlagIsNonInterleaved);
- streamFormat.mBytesPerPacket = 4;
- streamFormat.mFramesPerPacket = 1;
- streamFormat.mBytesPerFrame = 4;
- streamFormat.mChannelsPerFrame = d_n_channels;
- streamFormat.mBitsPerChannel = 32;
-
- err = AudioUnitSetProperty (d_OutputAU,
- kAudioUnitProperty_StreamFormat,
- kAudioUnitScope_Input,
- 0,
- &streamFormat,
- sizeof (AudioStreamBasicDescription));
- CheckErrorAndThrow (err, "AudioUnitSetProperty StreamFormat", "audio_osx_sink::audio_osx_sink");
-
-// create the stuff to regulate I/O
-
- d_cond_data = new gruel::condition_variable ();
- if (d_cond_data == NULL)
- CheckErrorAndThrow (errno, "new condition (data)",
- "audio_osx_sink::audio_osx_sink");
-
- d_internal = new gruel::mutex ();
- if (d_internal == NULL)
- CheckErrorAndThrow (errno, "new mutex (internal)",
- "audio_osx_sink::audio_osx_sink");
-
-// initialize the AU for output
-
- err = AudioUnitInitialize (d_OutputAU);
- CheckErrorAndThrow (err, "AudioUnitInitialize",
- "audio_osx_sink::audio_osx_sink");
-
-#if _OSX_AU_DEBUG_
- std::cerr << "audio_osx_sink Parameters:" << std::endl;
- std::cerr << " Sample Rate is " << d_sample_rate << std::endl;
- std::cerr << " Number of Channels is " << d_n_channels << std::endl;
- std::cerr << " Max # samples to store per channel is " << d_max_sample_count << std::endl;
-#endif
-}
-
-bool audio_osx_sink::IsRunning ()
-{
- UInt32 AURunning = 0, AUSize = sizeof (UInt32);
-
- OSStatus err = AudioUnitGetProperty (d_OutputAU,
- kAudioOutputUnitProperty_IsRunning,
- kAudioUnitScope_Global,
- 0,
- &AURunning,
- &AUSize);
- CheckErrorAndThrow (err, "AudioUnitGetProperty IsRunning",
- "audio_osx_sink::IsRunning");
-
- return (AURunning);
-}
-
-bool audio_osx_sink::start ()
-{
- if (! IsRunning ()) {
- OSStatus err = AudioOutputUnitStart (d_OutputAU);
- CheckErrorAndThrow (err, "AudioOutputUnitStart", "audio_osx_sink::start");
- }
-
- return (true);
-}
-
-bool audio_osx_sink::stop ()
-{
- if (IsRunning ()) {
- OSStatus err = AudioOutputUnitStop (d_OutputAU);
- CheckErrorAndThrow (err, "AudioOutputUnitStop", "audio_osx_sink::stop");
-
- for (UInt32 n = 0; n < d_n_channels; n++) {
- d_buffers[n]->abort ();
- }
- }
-
- return (true);
-}
-
-audio_osx_sink::~audio_osx_sink ()
-{
-// stop and close the AudioUnit
- stop ();
- AudioUnitUninitialize (d_OutputAU);
-#ifndef GR_USE_OLD_AUDIO_UNIT
- AudioComponentInstanceDispose (d_OutputAU);
-#else
- CloseComponent (d_OutputAU);
-#endif
-
-// empty and delete the queues
- for (UInt32 n = 0; n < d_n_channels; n++) {
- delete d_buffers[n];
- d_buffers[n] = 0;
- }
- delete [] d_buffers;
- d_buffers = 0;
-
-// close and delete control stuff
- delete d_cond_data;
- d_cond_data = 0;
- delete d_internal;
- d_internal = 0;
-}
-
-int
-audio_osx_sink::work (int noutput_items,
- gr_vector_const_void_star &input_items,
- gr_vector_void_star &output_items)
-{
- gruel::scoped_lock l (*d_internal);
-
- /* take the input data, copy it, and push it to the bottom of the queue
- mono input are pushed onto queue[0];
- stereo input are pushed onto queue[1].
- Start the AudioUnit if necessary. */
-
- UInt32 l_max_count;
- int diff_count = d_max_sample_count - noutput_items;
- if (diff_count < 0)
- l_max_count = 0;
- else
- l_max_count = (UInt32) diff_count;
-
-#if 0
- if (l_max_count < d_queueItemLength->back()) {
-// allow 2 buffers at a time, regardless of length
- l_max_count = d_queueItemLength->back();
- }
-#endif
-
-#if _OSX_AU_DEBUG_
- std::cerr << "work1: qSC = " << d_queueSampleCount << ", lMC = "<< l_max_count
- << ", dmSC = " << d_max_sample_count << ", nOI = " << noutput_items << std::endl;
-#endif
-
- if (d_queueSampleCount > l_max_count) {
-// data coming in too fast; do_block decides what to do
- if (d_do_block == true) {
-// block until there is data to return
- while (d_queueSampleCount > l_max_count) {
-// release control so-as to allow data to be retrieved;
-// block until there is data to return
- d_cond_data->wait (l);
-// the condition's 'notify' was called; acquire control
-// to keep thread safe
- }
- }
- }
-// not blocking case and overflow is handled by the circular buffer
-
-// add the input frames to the buffers' queue, checking for overflow
-
- UInt32 l_counter;
- int res = 0;
- float* inBuffer = (float*) input_items[0];
- const UInt32 l_size = input_items.size();
- for (l_counter = 0; l_counter < l_size; l_counter++) {
- inBuffer = (float*) input_items[l_counter];
- int l_res = d_buffers[l_counter]->enqueue (inBuffer,
- noutput_items);
- if (l_res == -1)
- res = -1;
- }
- while (l_counter < d_n_channels) {
-// for extra channels, copy the last input's data
- int l_res = d_buffers[l_counter++]->enqueue (inBuffer,
- noutput_items);
- if (l_res == -1)
- res = -1;
- }
-
- if (res == -1) {
-// data coming in too fast
-// drop oldest buffer
- fputs ("aO", stderr);
- fflush (stderr);
-// set the local number of samples available to the max
- d_queueSampleCount = d_buffers[0]->buffer_length_items ();
- } else {
-// keep up the local sample count
- d_queueSampleCount += noutput_items;
- }
-
-#if _OSX_AU_DEBUG_
- std::cerr << "work2: #OI = " << noutput_items << ", #Cnt = "
- << d_queueSampleCount << ", mSC = " << d_max_sample_count << std::endl;
-#endif
-
- return (noutput_items);
-}
-
-OSStatus audio_osx_sink::AUOutputCallback
-(void *inRefCon,
- AudioUnitRenderActionFlags *ioActionFlags,
- const AudioTimeStamp *inTimeStamp,
- UInt32 inBusNumber,
- UInt32 inNumberFrames,
- AudioBufferList *ioData)
-{
- audio_osx_sink* This = (audio_osx_sink*) inRefCon;
- OSStatus err = noErr;
-
- gruel::scoped_lock l (*This->d_internal);
-
-#if _OSX_AU_DEBUG_
- std::cerr << "cb_in: SC = " << This->d_queueSampleCount
- << ", in#F = " << inNumberFrames << std::endl;
-#endif
-
- if (This->d_queueSampleCount < inNumberFrames) {
-// not enough data to fill request
- err = -1;
- } else {
-// enough data; remove data from our buffers into the AU's buffers
- int l_counter = This->d_n_channels;
-
- while (--l_counter >= 0) {
- size_t t_n_output_items = inNumberFrames;
- float* outBuffer = (float*) ioData->mBuffers[l_counter].mData;
- This->d_buffers[l_counter]->dequeue (outBuffer, &t_n_output_items);
- if (t_n_output_items != inNumberFrames) {
- throw std::runtime_error ("audio_osx_sink::AUOutputCallback(): "
- "number of available items changing "
- "unexpectedly.\n");
- }
- }
-
- This->d_queueSampleCount -= inNumberFrames;
- }
-
-#if _OSX_AU_DEBUG_
- std::cerr << "cb_out: SC = " << This->d_queueSampleCount << std::endl;
-#endif
-
-// signal that data is available
- This->d_cond_data->notify_one ();
-
- return (err);
-}
diff --git a/gr-audio/lib/osx/audio_osx_sink.h b/gr-audio/lib/osx/audio_osx_sink.h
deleted file mode 100644
index 73b3db40d6..0000000000
--- a/gr-audio/lib/osx/audio_osx_sink.h
+++ /dev/null
@@ -1,80 +0,0 @@
-/* -*- c++ -*- */
-/*
- * Copyright 2006-2011 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.
- */
-
-#ifndef INCLUDED_AUDIO_OSX_SINK_H
-#define INCLUDED_AUDIO_OSX_SINK_H
-
-#include <gr_audio_sink.h>
-#include <string>
-#include <list>
-#include <AudioUnit/AudioUnit.h>
-#include <circular_buffer.h>
-
-/*!
- * \brief audio sink using OSX
- * \ingroup audio_blk
- *
- * input signature is one or two streams of floats.
- * Input samples must be in the range [-1,1].
- */
-
-class audio_osx_sink : public audio_sink {
-
- Float64 d_sample_rate;
- int d_channel_config;
- UInt32 d_n_channels;
- UInt32 d_queueSampleCount, d_max_sample_count;
- bool d_do_block;
- gruel::mutex* d_internal;
- gruel::condition_variable* d_cond_data;
- circular_buffer<float>** d_buffers;
-
-// AudioUnits and Such
- AudioUnit d_OutputAU;
-
-public:
- audio_osx_sink (int sample_rate = 44100,
- const std::string device_name = "2",
- bool do_block = true,
- int channel_config = -1,
- int max_sample_count = -1);
-
- ~audio_osx_sink ();
-
- bool IsRunning ();
- bool start ();
- bool stop ();
-
- int work (int noutput_items,
- gr_vector_const_void_star &input_items,
- gr_vector_void_star &output_items);
-
-private:
- static OSStatus AUOutputCallback (void *inRefCon,
- AudioUnitRenderActionFlags *ioActionFlags,
- const AudioTimeStamp *inTimeStamp,
- UInt32 inBusNumber,
- UInt32 inNumberFrames,
- AudioBufferList *ioData);
-};
-
-#endif /* INCLUDED_AUDIO_OSX_SINK_H */
diff --git a/gr-audio/lib/osx/audio_osx_source.cc b/gr-audio/lib/osx/audio_osx_source.cc
deleted file mode 100644
index 29f0ac3811..0000000000
--- a/gr-audio/lib/osx/audio_osx_source.cc
+++ /dev/null
@@ -1,1065 +0,0 @@
-/* -*- c++ -*- */
-/*
- * Copyright 2006-2011 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 "gr_audio_registry.h"
-#include <audio_osx_source.h>
-#include <gr_io_signature.h>
-#include <stdexcept>
-#include <audio_osx.h>
-
-#define _OSX_AU_DEBUG_ 0
-#define _OSX_DO_LISTENERS_ 0
-
-AUDIO_REGISTER_SOURCE(REG_PRIO_HIGH, osx)(
- int sampling_rate, const std::string &device_name, bool ok_to_block
-){
- return audio_source::sptr(new audio_osx_source(sampling_rate, device_name, ok_to_block));
-}
-
-void PrintStreamDesc (AudioStreamBasicDescription *inDesc)
-{
- if (inDesc == NULL) {
- std::cerr << "PrintStreamDesc: Can't print a NULL desc!" << std::endl;
- return;
- }
-
- std::cerr << " Sample Rate : " << inDesc->mSampleRate << std::endl;
- char format_id[4];
- strncpy (format_id, (char*)(&inDesc->mFormatID), 4);
- std::cerr << " Format ID : " << format_id << std::endl;
- std::cerr << " Format Flags : " << inDesc->mFormatFlags << std::endl;
- std::cerr << " Bytes per Packet : " << inDesc->mBytesPerPacket << std::endl;
- std::cerr << " Frames per Packet : " << inDesc->mFramesPerPacket << std::endl;
- std::cerr << " Bytes per Frame : " << inDesc->mBytesPerFrame << std::endl;
- std::cerr << " Channels per Frame : " << inDesc->mChannelsPerFrame << std::endl;
- std::cerr << " Bits per Channel : " << inDesc->mBitsPerChannel << std::endl;
-}
-
-// FIXME these should query some kind of user preference
-
-audio_osx_source::audio_osx_source (int sample_rate,
- const std::string device_name,
- bool do_block,
- int channel_config,
- int max_sample_count)
- : gr_sync_block ("audio_osx_source",
- gr_make_io_signature (0, 0, 0),
- gr_make_io_signature (0, 0, 0)),
- d_deviceSampleRate (0.0), d_outputSampleRate (0.0),
- d_channel_config (0),
- d_inputBufferSizeFrames (0), d_inputBufferSizeBytes (0),
- d_outputBufferSizeFrames (0), d_outputBufferSizeBytes (0),
- d_deviceBufferSizeFrames (0), d_deviceBufferSizeBytes (0),
- d_leadSizeFrames (0), d_leadSizeBytes (0),
- d_trailSizeFrames (0), d_trailSizeBytes (0),
- d_extraBufferSizeFrames (0), d_extraBufferSizeBytes (0),
- d_queueSampleCount (0), d_max_sample_count (0),
- d_n_AvailableInputFrames (0), d_n_ActualInputFrames (0),
- d_n_user_channels (0), d_n_max_channels (0), d_n_deviceChannels (0),
- d_do_block (do_block), d_passThrough (false),
- d_internal (0), d_cond_data (0),
- d_buffers (0),
- d_InputAU (0), d_InputBuffer (0), d_OutputBuffer (0),
- d_AudioConverter (0)
-{
- if (sample_rate <= 0) {
- std::cerr << "Invalid Sample Rate: " << sample_rate << std::endl;
- throw std::invalid_argument ("audio_osx_source::audio_osx_source");
- } else
- d_outputSampleRate = (Float64) sample_rate;
-
- if (channel_config <= 0 & channel_config != -1) {
- std::cerr << "Invalid Channel Config: " << channel_config << std::endl;
- throw std::invalid_argument ("audio_osx_source::audio_osx_source");
- } else if (channel_config == -1) {
-// no user input; try "device name" instead
- int l_n_channels = (int) strtol (device_name.data(), (char **)NULL, 10);
- if (l_n_channels == 0 & errno) {
- std::cerr << "Error Converting Device Name: " << errno << std::endl;
- throw std::invalid_argument ("audio_osx_source::audio_osx_source");
- }
- if (l_n_channels <= 0)
- channel_config = 2;
- else
- channel_config = l_n_channels;
- }
-
- d_channel_config = channel_config;
-
-// check that the max # of samples to store is valid
-
- if (max_sample_count == -1)
- max_sample_count = sample_rate;
- else if (max_sample_count <= 0) {
- std::cerr << "Invalid Max Sample Count: " << max_sample_count << std::endl;
- throw std::invalid_argument ("audio_osx_source::audio_osx_source");
- }
-
- d_max_sample_count = max_sample_count;
-
-#if _OSX_AU_DEBUG_
- std::cerr << "source(): max # samples = " << d_max_sample_count << std::endl;
-#endif
-
- OSStatus err = noErr;
-
-// create the default AudioUnit for input
-
-// Open the default input unit
-#ifndef GR_USE_OLD_AUDIO_UNIT
- AudioComponentDescription InputDesc;
-#else
- ComponentDescription InputDesc;
-#endif
-
-
- InputDesc.componentType = kAudioUnitType_Output;
- InputDesc.componentSubType = kAudioUnitSubType_HALOutput;
- InputDesc.componentManufacturer = kAudioUnitManufacturer_Apple;
- InputDesc.componentFlags = 0;
- InputDesc.componentFlagsMask = 0;
-
-#ifndef GR_USE_OLD_AUDIO_UNIT
- AudioComponent comp = AudioComponentFindNext (NULL, &InputDesc);
-#else
- Component comp = FindNextComponent (NULL, &InputDesc);
-#endif
-
- if (comp == NULL) {
-#ifndef GR_USE_OLD_AUDIO_UNIT
- std::cerr << "AudioComponentFindNext Error" << std::endl;
-#else
- std::cerr << "FindNextComponent Error" << std::endl;
-#endif
- throw std::runtime_error ("audio_osx_source::audio_osx_source");
- }
-
-#ifndef GR_USE_OLD_AUDIO_UNIT
- err = AudioComponentInstanceNew (comp, &d_InputAU);
- CheckErrorAndThrow (err, "AudioComponentInstanceNew",
- "audio_osx_source::audio_osx_source");
-#else
- err = OpenAComponent (comp, &d_InputAU);
- CheckErrorAndThrow (err, "OpenAComponent",
- "audio_osx_source::audio_osx_source");
-#endif
-
-
- UInt32 enableIO;
-
-// must enable the AUHAL for input and disable output
-// before setting the AUHAL's current device
-
-// Enable input on the AUHAL
- enableIO = 1;
- err = AudioUnitSetProperty (d_InputAU,
- kAudioOutputUnitProperty_EnableIO,
- kAudioUnitScope_Input,
- 1, // input element
- &enableIO,
- sizeof (UInt32));
- CheckErrorAndThrow (err, "AudioUnitSetProperty Input Enable",
- "audio_osx_source::audio_osx_source");
-
-// Disable output on the AUHAL
- enableIO = 0;
- err = AudioUnitSetProperty (d_InputAU,
- kAudioOutputUnitProperty_EnableIO,
- kAudioUnitScope_Output,
- 0, // output element
- &enableIO,
- sizeof (UInt32));
- CheckErrorAndThrow (err, "AudioUnitSetProperty Output Disable",
- "audio_osx_source::audio_osx_source");
-
-// set the default input device for our input AU
-
- SetDefaultInputDeviceAsCurrent ();
-
-#if _OSX_DO_LISTENERS_
-// set up a listener if default hardware input device changes
-
- err = AudioHardwareAddPropertyListener
- (kAudioHardwarePropertyDefaultInputDevice,
- (AudioHardwarePropertyListenerProc) HardwareListener,
- this);
-
- CheckErrorAndThrow (err, "AudioHardwareAddPropertyListener",
- "audio_osx_source::audio_osx_source");
-
-// Add a listener for any changes in the input AU's output stream
-// the function "UnitListener" will be called if the stream format
-// changes for whatever reason
-
- err = AudioUnitAddPropertyListener
- (d_InputAU,
- kAudioUnitProperty_StreamFormat,
- (AudioUnitPropertyListenerProc) UnitListener,
- this);
- CheckErrorAndThrow (err, "Adding Unit Property Listener",
- "audio_osx_source::audio_osx_source");
-#endif
-
-// Now find out if it actually can do input.
-
- UInt32 hasInput = 0;
- UInt32 dataSize = sizeof (hasInput);
- err = AudioUnitGetProperty (d_InputAU,
- kAudioOutputUnitProperty_HasIO,
- kAudioUnitScope_Input,
- 1,
- &hasInput,
- &dataSize);
- CheckErrorAndThrow (err, "AudioUnitGetProperty HasIO",
- "audio_osx_source::audio_osx_source");
- if (hasInput == 0) {
- std::cerr << "Selected Audio Device does not support Input." << std::endl;
- throw std::runtime_error ("audio_osx_source::audio_osx_source");
- }
-
-// Set up a callback function to retrieve input from the Audio Device
-
- AURenderCallbackStruct AUCallBack;
-
- AUCallBack.inputProc = (AURenderCallback)(audio_osx_source::AUInputCallback);
- AUCallBack.inputProcRefCon = this;
-
- err = AudioUnitSetProperty (d_InputAU,
- kAudioOutputUnitProperty_SetInputCallback,
- kAudioUnitScope_Global,
- 0,
- &AUCallBack,
- sizeof (AURenderCallbackStruct));
- CheckErrorAndThrow (err, "AudioUnitSetProperty Input Callback",
- "audio_osx_source::audio_osx_source");
-
- UInt32 propertySize;
- AudioStreamBasicDescription asbd_device, asbd_client, asbd_user;
-
-// asbd_device: ASBD of the device that is creating the input data stream
-// asbd_client: ASBD of the client size (output) of the hardware device
-// asbd_user: ASBD of the user's arguments
-
-// Get the Stream Format (device side)
-
- propertySize = sizeof (asbd_device);
- err = AudioUnitGetProperty (d_InputAU,
- kAudioUnitProperty_StreamFormat,
- kAudioUnitScope_Input,
- 1,
- &asbd_device,
- &propertySize);
- CheckErrorAndThrow (err, "AudioUnitGetProperty Device Input Stream Format",
- "audio_osx_source::audio_osx_source");
-
-#if _OSX_AU_DEBUG_
- std::cerr << std::endl << "---- Device Stream Format ----" << std::endl;
- PrintStreamDesc (&asbd_device);
-#endif
-
-// Get the Stream Format (client side)
- propertySize = sizeof (asbd_client);
- err = AudioUnitGetProperty (d_InputAU,
- kAudioUnitProperty_StreamFormat,
- kAudioUnitScope_Output,
- 1,
- &asbd_client,
- &propertySize);
- CheckErrorAndThrow (err, "AudioUnitGetProperty Device Ouput Stream Format",
- "audio_osx_source::audio_osx_source");
-
-#if _OSX_AU_DEBUG_
- std::cerr << std::endl << "---- Client Stream Format ----" << std::endl;
- PrintStreamDesc (&asbd_client);
-#endif
-
-// Set the format of all the AUs to the input/output devices channel count
-
-// get the max number of input (& thus output) channels supported by
-// this device
- d_n_max_channels = asbd_device.mChannelsPerFrame;
-
-// create the output io signature;
-// no input siganture to set (source is hardware)
- set_output_signature (gr_make_io_signature (1,
- d_n_max_channels,
- sizeof (float)));
-
-// allocate the output circular buffer(s), one per channel
- d_buffers = (circular_buffer<float>**) new
- circular_buffer<float>* [d_n_max_channels];
- UInt32 n_alloc = (UInt32) ceil ((double) d_max_sample_count);
- for (UInt32 n = 0; n < d_n_max_channels; n++) {
- d_buffers[n] = new circular_buffer<float> (n_alloc, false, false);
- }
-
- d_deviceSampleRate = asbd_device.mSampleRate;
- d_n_deviceChannels = asbd_device.mChannelsPerFrame;
-
- asbd_client.mSampleRate = asbd_device.mSampleRate;
- asbd_client.mFormatID = kAudioFormatLinearPCM;
- asbd_client.mFormatFlags = (kAudioFormatFlagIsFloat |
- kAudioFormatFlagIsPacked |
- kAudioFormatFlagIsNonInterleaved);
- if ((asbd_client.mFormatID == kAudioFormatLinearPCM) &&
- (d_n_deviceChannels == 1)) {
- asbd_client.mFormatFlags &= ~kLinearPCMFormatFlagIsNonInterleaved;
- }
- asbd_client.mBytesPerFrame = sizeof (float);
- asbd_client.mFramesPerPacket = 1;
- asbd_client.mBitsPerChannel = asbd_client.mBytesPerFrame * 8;
- asbd_client.mChannelsPerFrame = d_n_deviceChannels;
- asbd_client.mBytesPerPacket = asbd_client.mBytesPerFrame;
-
- propertySize = sizeof(AudioStreamBasicDescription);
- err = AudioUnitSetProperty (d_InputAU,
- kAudioUnitProperty_StreamFormat,
- kAudioUnitScope_Output,
- 1,
- &asbd_client,
- propertySize);
- CheckErrorAndThrow (err, "AudioUnitSetProperty Device Ouput Stream Format",
- "audio_osx_source::audio_osx_source");
-
-// create an ASBD for the user's wants
-
- asbd_user.mSampleRate = d_outputSampleRate;
- asbd_user.mFormatID = kAudioFormatLinearPCM;
- asbd_user.mFormatFlags = (kLinearPCMFormatFlagIsFloat |
- GR_PCM_ENDIANNESS |
- kLinearPCMFormatFlagIsPacked |
- kAudioFormatFlagIsNonInterleaved);
- asbd_user.mBytesPerPacket = sizeof (float);
- asbd_user.mFramesPerPacket = 1;
- asbd_user.mBytesPerFrame = asbd_user.mBytesPerPacket;
- asbd_user.mChannelsPerFrame = d_n_deviceChannels;
- asbd_user.mBitsPerChannel = asbd_user.mBytesPerPacket * 8;
-
- if (d_deviceSampleRate == d_outputSampleRate) {
-// no need to do conversion if asbd_client matches user wants
- d_passThrough = true;
- d_leadSizeFrames = d_trailSizeFrames = 0L;
- } else {
- d_passThrough = false;
-// Create the audio converter
-
- err = AudioConverterNew (&asbd_client, &asbd_user, &d_AudioConverter);
- CheckErrorAndThrow (err, "AudioConverterNew",
- "audio_osx_source::audio_osx_source");
-
-// Set the audio converter sample rate quality to "max" ...
-// requires more samples, but should sound nicer
-
- UInt32 ACQuality = kAudioConverterQuality_Max;
- propertySize = sizeof (ACQuality);
- err = AudioConverterSetProperty (d_AudioConverter,
- kAudioConverterSampleRateConverterQuality,
- propertySize,
- &ACQuality);
- CheckErrorAndThrow (err, "AudioConverterSetProperty "
- "SampleRateConverterQuality",
- "audio_osx_source::audio_osx_source");
-
-// set the audio converter's prime method to "pre",
-// which uses both leading and trailing frames
-// from the "current input". All of this is handled
-// internally by the AudioConverter; we just supply
-// the frames for conversion.
-
-// UInt32 ACPrimeMethod = kConverterPrimeMethod_None;
- UInt32 ACPrimeMethod = kConverterPrimeMethod_Pre;
- propertySize = sizeof (ACPrimeMethod);
- err = AudioConverterSetProperty (d_AudioConverter,
- kAudioConverterPrimeMethod,
- propertySize,
- &ACPrimeMethod);
- CheckErrorAndThrow (err, "AudioConverterSetProperty PrimeMethod",
- "audio_osx_source::audio_osx_source");
-
-// Get the size of the I/O buffer(s) to allow for pre-allocated buffers
-
-// lead frame info (trail frame info is ignored)
-
- AudioConverterPrimeInfo ACPrimeInfo = {0, 0};
- propertySize = sizeof (ACPrimeInfo);
- err = AudioConverterGetProperty (d_AudioConverter,
- kAudioConverterPrimeInfo,
- &propertySize,
- &ACPrimeInfo);
- CheckErrorAndThrow (err, "AudioConverterGetProperty PrimeInfo",
- "audio_osx_source::audio_osx_source");
-
- switch (ACPrimeMethod) {
- case (kConverterPrimeMethod_None):
- d_leadSizeFrames =
- d_trailSizeFrames = 0L;
- break;
- case (kConverterPrimeMethod_Normal):
- d_leadSizeFrames = 0L;
- d_trailSizeFrames = ACPrimeInfo.trailingFrames;
- break;
- default:
- d_leadSizeFrames = ACPrimeInfo.leadingFrames;
- d_trailSizeFrames = ACPrimeInfo.trailingFrames;
- }
- }
- d_leadSizeBytes = d_leadSizeFrames * sizeof (Float32);
- d_trailSizeBytes = d_trailSizeFrames * sizeof (Float32);
-
- propertySize = sizeof (d_deviceBufferSizeFrames);
- err = AudioUnitGetProperty (d_InputAU,
- kAudioDevicePropertyBufferFrameSize,
- kAudioUnitScope_Global,
- 0,
- &d_deviceBufferSizeFrames,
- &propertySize);
- CheckErrorAndThrow (err, "AudioUnitGetProperty Buffer Frame Size",
- "audio_osx_source::audio_osx_source");
-
- d_deviceBufferSizeBytes = d_deviceBufferSizeFrames * sizeof (Float32);
- d_inputBufferSizeBytes = d_deviceBufferSizeBytes + d_leadSizeBytes;
- d_inputBufferSizeFrames = d_deviceBufferSizeFrames + d_leadSizeFrames;
-
-// outBufSizeBytes = floor (inBufSizeBytes * rate_out / rate_in)
-// since this is rarely exact, we need another buffer to hold
-// "extra" samples not processed at any given sampling period
-// this buffer must be at least 4 floats in size, but generally
-// follows the rule that
-// extraBufSize = ceil (rate_in / rate_out)*sizeof(float)
-
- d_extraBufferSizeFrames = ((UInt32) ceil (d_deviceSampleRate
- / d_outputSampleRate)
- * sizeof (float));
- if (d_extraBufferSizeFrames < 4)
- d_extraBufferSizeFrames = 4;
- d_extraBufferSizeBytes = d_extraBufferSizeFrames * sizeof (float);
-
- d_outputBufferSizeFrames = (UInt32) ceil (((Float64) d_inputBufferSizeFrames)
- * d_outputSampleRate
- / d_deviceSampleRate);
- d_outputBufferSizeBytes = d_outputBufferSizeFrames * sizeof (float);
- d_inputBufferSizeFrames += d_extraBufferSizeFrames;
-
-// pre-alloc all buffers
-
- AllocAudioBufferList (&d_InputBuffer, d_n_deviceChannels,
- d_inputBufferSizeBytes);
- if (d_passThrough == false) {
- AllocAudioBufferList (&d_OutputBuffer, d_n_max_channels,
- d_outputBufferSizeBytes);
- } else {
- d_OutputBuffer = d_InputBuffer;
- }
-
-// create the stuff to regulate I/O
-
- d_cond_data = new gruel::condition_variable ();
- if (d_cond_data == NULL)
- CheckErrorAndThrow (errno, "new condition (data)",
- "audio_osx_source::audio_osx_source");
-
- d_internal = new gruel::mutex ();
- if (d_internal == NULL)
- CheckErrorAndThrow (errno, "new mutex (internal)",
- "audio_osx_source::audio_osx_source");
-
-// initialize the AU for input
-
- err = AudioUnitInitialize (d_InputAU);
- CheckErrorAndThrow (err, "AudioUnitInitialize",
- "audio_osx_source::audio_osx_source");
-
-#if _OSX_AU_DEBUG_
- std::cerr << "audio_osx_source Parameters:" << std::endl;
- std::cerr << " Device Sample Rate is " << d_deviceSampleRate << std::endl;
- std::cerr << " User Sample Rate is " << d_outputSampleRate << std::endl;
- std::cerr << " Max Sample Count is " << d_max_sample_count << std::endl;
- std::cerr << " # Device Channels is " << d_n_deviceChannels << std::endl;
- std::cerr << " # Max Channels is " << d_n_max_channels << std::endl;
- std::cerr << " Device Buffer Size is Frames = " << d_deviceBufferSizeFrames << std::endl;
- std::cerr << " Lead Size is Frames = " << d_leadSizeFrames << std::endl;
- std::cerr << " Trail Size is Frames = " << d_trailSizeFrames << std::endl;
- std::cerr << " Input Buffer Size is Frames = " << d_inputBufferSizeFrames << std::endl;
- std::cerr << " Output Buffer Size is Frames = " << d_outputBufferSizeFrames << std::endl;
-#endif
-}
-
-void
-audio_osx_source::AllocAudioBufferList (AudioBufferList** t_ABL,
- UInt32 n_channels,
- UInt32 bufferSizeBytes)
-{
- FreeAudioBufferList (t_ABL);
- UInt32 propertySize = (offsetof (AudioBufferList, mBuffers[0]) +
- (sizeof (AudioBuffer) * n_channels));
- *t_ABL = (AudioBufferList*) calloc (1, propertySize);
- (*t_ABL)->mNumberBuffers = n_channels;
-
- int counter = n_channels;
-
- while (--counter >= 0) {
- (*t_ABL)->mBuffers[counter].mNumberChannels = 1;
- (*t_ABL)->mBuffers[counter].mDataByteSize = bufferSizeBytes;
- (*t_ABL)->mBuffers[counter].mData = calloc (1, bufferSizeBytes);
- }
-}
-
-void
-audio_osx_source::FreeAudioBufferList (AudioBufferList** t_ABL)
-{
-// free pre-allocated audio buffer, if it exists
- if (*t_ABL != NULL) {
- int counter = (*t_ABL)->mNumberBuffers;
- while (--counter >= 0)
- free ((*t_ABL)->mBuffers[counter].mData);
- free (*t_ABL);
- (*t_ABL) = 0;
- }
-}
-
-bool audio_osx_source::IsRunning ()
-{
- UInt32 AURunning = 0, AUSize = sizeof (UInt32);
-
- OSStatus err = AudioUnitGetProperty (d_InputAU,
- kAudioOutputUnitProperty_IsRunning,
- kAudioUnitScope_Global,
- 0,
- &AURunning,
- &AUSize);
- CheckErrorAndThrow (err, "AudioUnitGetProperty IsRunning",
- "audio_osx_source::IsRunning");
-
- return (AURunning);
-}
-
-bool audio_osx_source::start ()
-{
- if (! IsRunning ()) {
- OSStatus err = AudioOutputUnitStart (d_InputAU);
- CheckErrorAndThrow (err, "AudioOutputUnitStart",
- "audio_osx_source::start");
- }
-
- return (true);
-}
-
-bool audio_osx_source::stop ()
-{
- if (IsRunning ()) {
- OSStatus err = AudioOutputUnitStop (d_InputAU);
- CheckErrorAndThrow (err, "AudioOutputUnitStart",
- "audio_osx_source::stop");
- for (UInt32 n = 0; n < d_n_user_channels; n++) {
- d_buffers[n]->abort ();
- }
- }
-
- return (true);
-}
-
-audio_osx_source::~audio_osx_source ()
-{
- OSStatus err = noErr;
-
-// stop the AudioUnit
- stop();
-
-#if _OSX_DO_LISTENERS_
-// remove the listeners
-
- err = AudioUnitRemovePropertyListener
- (d_InputAU,
- kAudioUnitProperty_StreamFormat,
- (AudioUnitPropertyListenerProc) UnitListener);
- CheckError (err, "~audio_osx_source: AudioUnitRemovePropertyListener");
-
- err = AudioHardwareRemovePropertyListener
- (kAudioHardwarePropertyDefaultInputDevice,
- (AudioHardwarePropertyListenerProc) HardwareListener);
- CheckError (err, "~audio_osx_source: AudioHardwareRemovePropertyListener");
-#endif
-
-// free pre-allocated audio buffers
- FreeAudioBufferList (&d_InputBuffer);
-
- if (d_passThrough == false) {
- err = AudioConverterDispose (d_AudioConverter);
- CheckError (err, "~audio_osx_source: AudioConverterDispose");
- FreeAudioBufferList (&d_OutputBuffer);
- }
-
-// remove the audio unit
- err = AudioUnitUninitialize (d_InputAU);
- CheckError (err, "~audio_osx_source: AudioUnitUninitialize");
-
-#ifndef GR_USE_OLD_AUDIO_UNIT
- err = AudioComponentInstanceDispose (d_InputAU);
- CheckError (err, "~audio_osx_source: AudioComponentInstanceDispose");
-#else
- err = CloseComponent (d_InputAU);
- CheckError (err, "~audio_osx_source: CloseComponent");
-#endif
-
-// empty and delete the queues
- for (UInt32 n = 0; n < d_n_max_channels; n++) {
- delete d_buffers[n];
- d_buffers[n] = 0;
- }
- delete [] d_buffers;
- d_buffers = 0;
-
-// close and delete the control stuff
- delete d_cond_data;
- d_cond_data = 0;
- delete d_internal;
- d_internal = 0;
-}
-
-bool
-audio_osx_source::check_topology (int ninputs, int noutputs)
-{
-// check # inputs to make sure it's valid
- if (ninputs != 0) {
- std::cerr << "audio_osx_source::check_topology(): number of input "
- << "streams provided (" << ninputs
- << ") should be 0." << std::endl;
- throw std::runtime_error ("audio_osx_source::check_topology()");
- }
-
-// check # outputs to make sure it's valid
- if ((noutputs < 1) | (noutputs > (int) d_n_max_channels)) {
- std::cerr << "audio_osx_source::check_topology(): number of output "
- << "streams provided (" << noutputs << ") should be in [1,"
- << d_n_max_channels << "] for the selected audio device."
- << std::endl;
- throw std::runtime_error ("audio_osx_source::check_topology()");
- }
-
-// save the actual number of output (user) channels
- d_n_user_channels = noutputs;
-
-#if _OSX_AU_DEBUG_
- std::cerr << "chk_topo: Actual # user output channels = "
- << noutputs << std::endl;
-#endif
-
- return (true);
-}
-
-int
-audio_osx_source::work
-(int noutput_items,
- gr_vector_const_void_star &input_items,
- gr_vector_void_star &output_items)
-{
- // acquire control to do processing here only
- gruel::scoped_lock l (*d_internal);
-
-#if _OSX_AU_DEBUG_
- std::cerr << "work1: SC = " << d_queueSampleCount
- << ", #OI = " << noutput_items
- << ", #Chan = " << output_items.size() << std::endl;
-#endif
-
- // set the actual # of output items to the 'desired' amount then
- // verify that data is available; if not enough data is available,
- // either wait until it is (is "do_block" is true), return (0) is no
- // data is available and "do_block" is false, or process the actual
- // amount of available data.
-
- UInt32 actual_noutput_items = noutput_items;
-
- if (d_queueSampleCount < actual_noutput_items) {
- if (d_queueSampleCount == 0) {
- // no data; do_block decides what to do
- if (d_do_block == true) {
- while (d_queueSampleCount == 0) {
- // release control so-as to allow data to be retrieved;
- // block until there is data to return
- d_cond_data->wait (l);
- // the condition's 'notify' was called; acquire control to
- // keep thread safe
- }
- } else {
- // no data & not blocking; return nothing
- return (0);
- }
- }
- // use the actual amount of available data
- actual_noutput_items = d_queueSampleCount;
- }
-
- // number of channels
- int l_counter = (int) output_items.size();
-
- // copy the items from the circular buffer(s) to 'work's output buffers
- // verify that the number copied out is as expected.
-
- while (--l_counter >= 0) {
- size_t t_n_output_items = actual_noutput_items;
- d_buffers[l_counter]->dequeue ((float*) output_items[l_counter],
- &t_n_output_items);
- if (t_n_output_items != actual_noutput_items) {
- std::cerr << "audio_osx_source::work(): ERROR: number of "
- << "available items changing unexpectedly; expecting "
- << actual_noutput_items << ", got "
- << t_n_output_items << "." << std::endl;
- throw std::runtime_error ("audio_osx_source::work()");
- }
- }
-
- // subtract the actual number of items removed from the buffer(s)
- // from the local accounting of the number of available samples
-
- d_queueSampleCount -= actual_noutput_items;
-
-#if _OSX_AU_DEBUG_
- std::cerr << "work2: SC = " << d_queueSampleCount
- << ", act#OI = " << actual_noutput_items << std::endl
- << "Returning." << std::endl;
-#endif
-
- return (actual_noutput_items);
-}
-
-OSStatus
-audio_osx_source::ConverterCallback
-(AudioConverterRef inAudioConverter,
- UInt32* ioNumberDataPackets,
- AudioBufferList* ioData,
- AudioStreamPacketDescription** ioASPD,
- void* inUserData)
-{
- // take current device buffers and copy them to the tail of the
- // input buffers the lead buffer is already there in the first
- // d_leadSizeFrames slots
-
- audio_osx_source* This = static_cast<audio_osx_source*>(inUserData);
- AudioBufferList* l_inputABL = This->d_InputBuffer;
- UInt32 totalInputBufferSizeBytes = ((*ioNumberDataPackets) * sizeof (float));
- int counter = This->d_n_deviceChannels;
- ioData->mNumberBuffers = This->d_n_deviceChannels;
- This->d_n_ActualInputFrames = (*ioNumberDataPackets);
-
-#if _OSX_AU_DEBUG_
- std::cerr << "cc1: io#DP = " << (*ioNumberDataPackets)
- << ", TIBSB = " << totalInputBufferSizeBytes
- << ", #C = " << counter << std::endl;
-#endif
-
- while (--counter >= 0) {
- AudioBuffer* l_ioD_AB = &(ioData->mBuffers[counter]);
- l_ioD_AB->mNumberChannels = 1;
- l_ioD_AB->mData = (float*)(l_inputABL->mBuffers[counter].mData);
- l_ioD_AB->mDataByteSize = totalInputBufferSizeBytes;
- }
-
-#if _OSX_AU_DEBUG_
- std::cerr << "cc2: Returning." << std::endl;
-#endif
-
- return (noErr);
-}
-
-OSStatus
-audio_osx_source::AUInputCallback (void* inRefCon,
- AudioUnitRenderActionFlags* ioActionFlags,
- const AudioTimeStamp* inTimeStamp,
- UInt32 inBusNumber,
- UInt32 inNumberFrames,
- AudioBufferList* ioData)
-{
- OSStatus err = noErr;
- audio_osx_source* This = static_cast<audio_osx_source*>(inRefCon);
-
- gruel::scoped_lock l (*This->d_internal);
-
-#if _OSX_AU_DEBUG_
- std::cerr << "cb0: in#F = " << inNumberFrames
- << ", inBN = " << inBusNumber
- << ", SC = " << This->d_queueSampleCount << std::endl;
-#endif
-
-// Get the new audio data from the input device
-
- err = AudioUnitRender (This->d_InputAU,
- ioActionFlags,
- inTimeStamp,
- 1, //inBusNumber,
- inNumberFrames,
- This->d_InputBuffer);
- CheckErrorAndThrow (err, "AudioUnitRender",
- "audio_osx_source::AUInputCallback");
-
- UInt32 AvailableInputFrames = inNumberFrames;
- This->d_n_AvailableInputFrames = inNumberFrames;
-
-// get the number of actual output frames,
-// either via converting the buffer or not
-
- UInt32 ActualOutputFrames;
-
- if (This->d_passThrough == true) {
- ActualOutputFrames = AvailableInputFrames;
- } else {
- UInt32 AvailableInputBytes = AvailableInputFrames * sizeof (float);
- UInt32 AvailableOutputBytes = AvailableInputBytes;
- UInt32 AvailableOutputFrames = AvailableOutputBytes / sizeof (float);
- UInt32 propertySize = sizeof (AvailableOutputBytes);
- err = AudioConverterGetProperty (This->d_AudioConverter,
- kAudioConverterPropertyCalculateOutputBufferSize,
- &propertySize,
- &AvailableOutputBytes);
- CheckErrorAndThrow (err, "AudioConverterGetProperty CalculateOutputBufferSize", "audio_osx_source::audio_osx_source");
-
- AvailableOutputFrames = AvailableOutputBytes / sizeof (float);
-
-#if 0
-// when decimating too much, the output sounds warbly due to
-// fluctuating # of output frames
-// This should not be a surprise, but there's probably some
-// clever programming that could lessed the effect ...
-// like finding the "ideal" # of output frames, and keeping
-// that number constant no matter the # of input frames
- UInt32 l_InputBytes = AvailableOutputBytes;
- propertySize = sizeof (AvailableOutputBytes);
- err = AudioConverterGetProperty (This->d_AudioConverter,
- kAudioConverterPropertyCalculateInputBufferSize,
- &propertySize,
- &l_InputBytes);
- CheckErrorAndThrow (err, "AudioConverterGetProperty CalculateInputBufferSize", "audio_osx_source::audio_osx_source");
-
- if (l_InputBytes < AvailableInputBytes) {
-// OK to zero pad the input a little
- AvailableOutputFrames += 1;
- AvailableOutputBytes = AvailableOutputFrames * sizeof (float);
- }
-#endif
-
-#if _OSX_AU_DEBUG_
- std::cerr << "cb1: avail: #IF = " << AvailableInputFrames
- << ", #OF = " << AvailableOutputFrames << std::endl;
-#endif
- ActualOutputFrames = AvailableOutputFrames;
-
-// convert the data to the correct rate
-// on input, ActualOutputFrames is the number of available output frames
-
- err = AudioConverterFillComplexBuffer (This->d_AudioConverter,
- (AudioConverterComplexInputDataProc)(This->ConverterCallback),
- inRefCon,
- &ActualOutputFrames,
- This->d_OutputBuffer,
- NULL);
- CheckErrorAndThrow (err, "AudioConverterFillComplexBuffer",
- "audio_osx_source::AUInputCallback");
-
-// on output, ActualOutputFrames is the actual number of output frames
-
-#if _OSX_AU_DEBUG_
- std::cerr << "cb2: actual: #IF = " << This->d_n_ActualInputFrames
- << ", #OF = " << AvailableOutputFrames << std::endl;
- if (This->d_n_ActualInputFrames != AvailableInputFrames)
- std::cerr << "cb2.1: avail#IF = " << AvailableInputFrames
- << ", actual#IF = " << This->d_n_ActualInputFrames << std::endl;
-#endif
- }
-
-// add the output frames to the buffers' queue, checking for overflow
-
- int l_counter = This->d_n_user_channels;
- int res = 0;
-
- while (--l_counter >= 0) {
- float* inBuffer = (float*) This->d_OutputBuffer->mBuffers[l_counter].mData;
-
-#if _OSX_AU_DEBUG_
- std::cerr << "cb3: enqueuing audio data." << std::endl;
-#endif
-
- int l_res = This->d_buffers[l_counter]->enqueue (inBuffer, ActualOutputFrames);
- if (l_res == -1)
- res = -1;
- }
-
- if (res == -1) {
-// data coming in too fast
-// drop oldest buffer
- fputs ("aO", stderr);
- fflush (stderr);
-// set the local number of samples available to the max
- This->d_queueSampleCount = This->d_buffers[0]->buffer_length_items ();
- } else {
-// keep up the local sample count
- This->d_queueSampleCount += ActualOutputFrames;
- }
-
-#if _OSX_AU_DEBUG_
- std::cerr << "cb4: #OI = " << ActualOutputFrames
- << ", #Cnt = " << This->d_queueSampleCount
- << ", mSC = " << This->d_max_sample_count << std::endl;
-#endif
-
-// signal that data is available, if appropraite
- This->d_cond_data->notify_one ();
-
-#if _OSX_AU_DEBUG_
- std::cerr << "cb5: returning." << std::endl;
-#endif
-
- return (err);
-}
-
-void
-audio_osx_source::SetDefaultInputDeviceAsCurrent
-()
-{
-// set the default input device
- AudioDeviceID deviceID = 0;
- UInt32 dataSize = sizeof (AudioDeviceID);
- OSStatus err = noErr;
-
-#ifndef GR_USE_OLD_AUDIO_UNIT
- AudioObjectPropertyAddress theAddress =
- { kAudioHardwarePropertyDefaultInputDevice,
- kAudioObjectPropertyScopeGlobal,
- kAudioObjectPropertyElementMaster };
-
- err = AudioObjectGetPropertyData
- (kAudioObjectSystemObject,
- &theAddress,
- 0,
- NULL,
- &dataSize,
- &deviceID);
-#else
- err = AudioHardwareGetProperty
- (kAudioHardwarePropertyDefaultInputDevice,
- &dataSize,
- &deviceID);
-#endif
-
- CheckErrorAndThrow (err, "Get Audio Unit Property for Current Device",
- "audio_osx_source::SetDefaultInputDeviceAsCurrent");
-
- err = AudioUnitSetProperty
- (d_InputAU,
- kAudioOutputUnitProperty_CurrentDevice,
- kAudioUnitScope_Global,
- 0,
- &deviceID,
- sizeof (AudioDeviceID));
-
- CheckErrorAndThrow (err, "AudioUnitSetProperty Current Device",
- "audio_osx_source::SetDefaultInputDeviceAsCurrent");
-}
-
-#if _OSX_DO_LISTENERS_
-OSStatus
-audio_osx_source::HardwareListener
-(AudioHardwarePropertyID inPropertyID,
- void *inClientData)
-{
- OSStatus err = noErr;
- audio_osx_source* This = static_cast<audio_osx_source*>(inClientData);
-
- std::cerr << "a_o_s::HardwareListener" << std::endl;
-
-// set the new default hardware input device for use by our AU
-
- This->SetDefaultInputDeviceAsCurrent ();
-
-// reset the converter to tell it that the stream has changed
-
- err = AudioConverterReset (This->d_AudioConverter);
- CheckErrorAndThrow (err, "AudioConverterReset",
- "audio_osx_source::UnitListener");
-
- return (err);
-}
-
-OSStatus
-audio_osx_source::UnitListener
-(void *inRefCon,
- AudioUnit ci,
- AudioUnitPropertyID inID,
- AudioUnitScope inScope,
- AudioUnitElement inElement)
-{
- OSStatus err = noErr;
- audio_osx_source* This = static_cast<audio_osx_source*>(inRefCon);
- AudioStreamBasicDescription asbd;
-
- std::cerr << "a_o_s::UnitListener" << std::endl;
-
-// get the converter's input ASBD (for printing)
-
- UInt32 propertySize = sizeof (asbd);
- err = AudioConverterGetProperty (This->d_AudioConverter,
- kAudioConverterCurrentInputStreamDescription,
- &propertySize,
- &asbd);
- CheckErrorAndThrow (err, "AudioConverterGetProperty "
- "CurrentInputStreamDescription",
- "audio_osx_source::UnitListener");
-
- std::cerr << "UnitListener: Input Source changed." << std::endl
- << "Old Source Output Info:" << std::endl;
- PrintStreamDesc (&asbd);
-
-// get the new input unit's output ASBD
-
- propertySize = sizeof (asbd);
- err = AudioUnitGetProperty (This->d_InputAU,
- kAudioUnitProperty_StreamFormat,
- kAudioUnitScope_Output, 1,
- &asbd, &propertySize);
- CheckErrorAndThrow (err, "AudioUnitGetProperty StreamFormat",
- "audio_osx_source::UnitListener");
-
- std::cerr << "New Source Output Info:" << std::endl;
- PrintStreamDesc (&asbd);
-
-// set the converter's input ASBD to this
-
- err = AudioConverterSetProperty (This->d_AudioConverter,
- kAudioConverterCurrentInputStreamDescription,
- propertySize,
- &asbd);
- CheckErrorAndThrow (err, "AudioConverterSetProperty "
- "CurrentInputStreamDescription",
- "audio_osx_source::UnitListener");
-
-// reset the converter to tell it that the stream has changed
-
- err = AudioConverterReset (This->d_AudioConverter);
- CheckErrorAndThrow (err, "AudioConverterReset",
- "audio_osx_source::UnitListener");
-
- return (err);
-}
-#endif
diff --git a/gr-audio/lib/osx/audio_osx_source.h b/gr-audio/lib/osx/audio_osx_source.h
deleted file mode 100644
index bb34d972cd..0000000000
--- a/gr-audio/lib/osx/audio_osx_source.h
+++ /dev/null
@@ -1,116 +0,0 @@
-/* -*- c++ -*- */
-/*
- * Copyright 2006-2011 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.
- */
-
-#ifndef INCLUDED_AUDIO_OSX_SOURCE_H
-#define INCLUDED_AUDIO_OSX_SOURCE_H
-
-#include <gr_audio_source.h>
-#include <string>
-#include <AudioToolbox/AudioToolbox.h>
-#include <AudioUnit/AudioUnit.h>
-#include <circular_buffer.h>
-
-/*!
- * \brief audio source using OSX
- * \ingroup audio_blk
- *
- * Input signature is one or two streams of floats.
- * Samples must be in the range [-1,1].
- */
-
-class audio_osx_source : public audio_source {
-
- Float64 d_deviceSampleRate, d_outputSampleRate;
- int d_channel_config;
- UInt32 d_inputBufferSizeFrames, d_inputBufferSizeBytes;
- UInt32 d_outputBufferSizeFrames, d_outputBufferSizeBytes;
- UInt32 d_deviceBufferSizeFrames, d_deviceBufferSizeBytes;
- UInt32 d_leadSizeFrames, d_leadSizeBytes;
- UInt32 d_trailSizeFrames, d_trailSizeBytes;
- UInt32 d_extraBufferSizeFrames, d_extraBufferSizeBytes;
- UInt32 d_queueSampleCount, d_max_sample_count;
- UInt32 d_n_AvailableInputFrames, d_n_ActualInputFrames;
- UInt32 d_n_user_channels, d_n_max_channels, d_n_deviceChannels;
- bool d_do_block, d_passThrough, d_waiting_for_data;
- gruel::mutex* d_internal;
- gruel::condition_variable* d_cond_data;
- circular_buffer<float>** d_buffers;
-
-// AudioUnits and Such
- AudioUnit d_InputAU;
- AudioBufferList* d_InputBuffer;
- AudioBufferList* d_OutputBuffer;
- AudioConverterRef d_AudioConverter;
-
-public:
- audio_osx_source (int sample_rate = 44100,
- const std::string device_name = "",
- bool do_block = true,
- int channel_config = -1,
- int max_sample_count = -1);
-
- ~audio_osx_source ();
-
- bool start ();
- bool stop ();
- bool IsRunning ();
-
- bool check_topology (int ninputs, int noutputs);
-
- int work (int noutput_items,
- gr_vector_const_void_star &input_items,
- gr_vector_void_star &output_items);
-
-private:
- void SetDefaultInputDeviceAsCurrent ();
-
- void AllocAudioBufferList (AudioBufferList** t_ABL,
- UInt32 n_channels,
- UInt32 inputBufferSizeBytes);
-
- void FreeAudioBufferList (AudioBufferList** t_ABL);
-
- static OSStatus ConverterCallback (AudioConverterRef inAudioConverter,
- UInt32* ioNumberDataPackets,
- AudioBufferList* ioData,
- AudioStreamPacketDescription** outASPD,
- void* inUserData);
-
- static OSStatus AUInputCallback (void *inRefCon,
- AudioUnitRenderActionFlags *ioActionFlags,
- const AudioTimeStamp *inTimeStamp,
- UInt32 inBusNumber,
- UInt32 inNumberFrames,
- AudioBufferList *ioData);
-#if _OSX_DO_LISTENERS_
- static OSStatus UnitListener (void *inRefCon,
- AudioUnit ci,
- AudioUnitPropertyID inID,
- AudioUnitScope inScope,
- AudioUnitElement inElement);
-
- static OSStatus HardwareListener (AudioHardwarePropertyID inPropertyID,
- void *inClientData);
-#endif
-};
-
-#endif /* INCLUDED_AUDIO_OSX_SOURCE_H */
diff --git a/gr-audio/lib/osx/osx_impl.h b/gr-audio/lib/osx/osx_impl.h
new file mode 100644
index 0000000000..5a12bac71a
--- /dev/null
+++ b/gr-audio/lib/osx/osx_impl.h
@@ -0,0 +1,78 @@
+/* -*- c++ -*- */
+/*
+ * Copyright 2006, 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.
+ */
+
+#ifndef INCLUDED_AUDIO_OSX_IMPL_H
+#define INCLUDED_AUDIO_OSX_IMPL_H
+
+#include <iostream>
+#include <string.h>
+
+namespace gr {
+ namespace audio {
+
+#define CheckErrorAndThrow(err,what,throw_str) \
+ if(err) { \
+ OSStatus error = static_cast<OSStatus>(err); \
+ char err_str[4]; \
+ strncpy(err_str, (char*)(&err), 4); \
+ std::cerr << what << std::endl; \
+ std::cerr << " Error# " << error << " ('" << err_str \
+ << "')" << std::endl; \
+ std::cerr << " " << __FILE__ << ":" << __LINE__ << std::endl; \
+ fflush(stderr); \
+ throw std::runtime_error(throw_str); \
+ }
+
+#define CheckError(err,what) \
+ if(err) { \
+ OSStatus error = static_cast<OSStatus>(err); \
+ char err_str[4]; \
+ strncpy(err_str, (char*)(&err), 4); \
+ std::cerr << what << std::endl; \
+ std::cerr << " Error# " << error << " ('" << err_str \
+ << "')" << std::endl; \
+ std::cerr << " " << __FILE__ << ":" << __LINE__ << std::endl; \
+ fflush(stderr); \
+ }
+
+#include <boost/detail/endian.hpp> //BOOST_BIG_ENDIAN
+#ifdef BOOST_BIG_ENDIAN
+#define GR_PCM_ENDIANNESS kLinearPCMFormatFlagIsBigEndian
+#else
+#define GR_PCM_ENDIANNESS 0
+#endif
+
+// Check the version of MacOSX being used
+#ifdef __APPLE_CC__
+#include <AvailabilityMacros.h>
+#ifndef MAC_OS_X_VERSION_10_6
+#define MAC_OS_X_VERSION_10_6 1060
+#endif
+#if MAC_OS_X_VERSION_MAX_ALLOWED < MAC_OS_X_VERSION_10_6
+#define GR_USE_OLD_AUDIO_UNIT
+#endif
+#endif
+
+ } /* namespace audio */
+} /* namespace gr */
+
+#endif /* INCLUDED_AUDIO_OSX_IMPL_H */
diff --git a/gr-audio/lib/osx/osx_sink.cc b/gr-audio/lib/osx/osx_sink.cc
new file mode 100644
index 0000000000..ace96ec742
--- /dev/null
+++ b/gr-audio/lib/osx/osx_sink.cc
@@ -0,0 +1,429 @@
+/* -*- c++ -*- */
+/*
+ * Copyright 2006-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 <osx_sink.h>
+#include <osx_impl.h>
+#include <gr_io_signature.h>
+#include <stdexcept>
+
+namespace gr {
+ namespace audio {
+
+#define _OSX_AU_DEBUG_ 0
+
+ AUDIO_REGISTER_SINK(REG_PRIO_HIGH, osx)(int sampling_rate,
+ const std::string &device_name,
+ bool ok_to_block)
+ {
+ return sink::sptr
+ (new osx_sink(sampling_rate, device_name, ok_to_block));
+ }
+
+ osx_sink::osx_sink(int sample_rate,
+ const std::string device_name,
+ bool do_block,
+ int channel_config,
+ int max_sample_count)
+ : gr_sync_block("audio_osx_sink",
+ gr_make_io_signature(0, 0, 0),
+ gr_make_io_signature(0, 0, 0)),
+ d_sample_rate(0.0), d_channel_config(0), d_n_channels(0),
+ d_queueSampleCount(0), d_max_sample_count(0),
+ d_do_block(do_block), d_internal(0), d_cond_data(0),
+ d_OutputAU(0)
+ {
+ if(sample_rate <= 0) {
+ std::cerr << "Invalid Sample Rate: " << sample_rate << std::endl;
+ throw std::invalid_argument ("audio_osx_sink::audio_osx_sink");
+ }
+ else
+ d_sample_rate = (Float64)sample_rate;
+
+ if(channel_config <= 0 & channel_config != -1) {
+ std::cerr << "Invalid Channel Config: " << channel_config << std::endl;
+ throw std::invalid_argument ("audio_osx_sink::audio_osx_sink");
+ }
+ else if(channel_config == -1) {
+ // no user input; try "device name" instead
+ int l_n_channels = (int)strtol(device_name.data(), (char**)NULL, 10);
+ if((l_n_channels == 0) & errno) {
+ std::cerr << "Error Converting Device Name: " << errno << std::endl;
+ throw std::invalid_argument("audio_osx_sink::audio_osx_sink");
+ }
+ if(l_n_channels <= 0)
+ channel_config = 2;
+ else
+ channel_config = l_n_channels;
+ }
+
+ d_n_channels = d_channel_config = channel_config;
+
+ // set the input signature
+
+ set_input_signature(gr_make_io_signature(1, d_n_channels, sizeof(float)));
+
+ // check that the max # of samples to store is valid
+
+ if(max_sample_count == -1)
+ max_sample_count = sample_rate;
+ else if(max_sample_count <= 0) {
+ std::cerr << "Invalid Max Sample Count: " << max_sample_count << std::endl;
+ throw std::invalid_argument ("audio_osx_sink::audio_osx_sink");
+ }
+
+ d_max_sample_count = max_sample_count;
+
+ // allocate the output circular buffer(s), one per channel
+
+ d_buffers = (circular_buffer<float>**) new
+ circular_buffer<float>* [d_n_channels];
+ UInt32 n_alloc = (UInt32) ceil((double)d_max_sample_count);
+ for(UInt32 n = 0; n < d_n_channels; n++) {
+ d_buffers[n] = new circular_buffer<float>(n_alloc, false, false);
+ }
+
+ // create the default AudioUnit for output
+ OSStatus err = noErr;
+
+ // Open the default output unit
+#ifndef GR_USE_OLD_AUDIO_UNIT
+ AudioComponentDescription desc;
+#else
+ ComponentDescription desc;
+#endif
+
+ desc.componentType = kAudioUnitType_Output;
+ desc.componentSubType = kAudioUnitSubType_DefaultOutput;
+ desc.componentManufacturer = kAudioUnitManufacturer_Apple;
+ desc.componentFlags = 0;
+ desc.componentFlagsMask = 0;
+
+#ifndef GR_USE_OLD_AUDIO_UNIT
+ AudioComponent comp = AudioComponentFindNext(NULL, &desc);
+ if(comp == NULL) {
+ std::cerr << "AudioComponentFindNext Error" << std::endl;
+ throw std::runtime_error("audio_osx_sink::audio_osx_sink");
+ }
+#else
+ Component comp = FindNextComponent(NULL, &desc);
+ if(comp == NULL) {
+ std::cerr << "FindNextComponent Error" << std::endl;
+ throw std::runtime_error("audio_osx_sink::audio_osx_sink");
+ }
+#endif
+
+#ifndef GR_USE_OLD_AUDIO_UNIT
+ err = AudioComponentInstanceNew(comp, &d_OutputAU);
+ CheckErrorAndThrow(err, "AudioComponentInstanceNew",
+ "audio_osx_sink::audio_osx_sink");
+#else
+ err = OpenAComponent(comp, &d_OutputAU);
+ CheckErrorAndThrow(err, "OpenAComponent",
+ "audio_osx_sink::audio_osx_sink");
+#endif
+
+ // Set up a callback function to generate output to the output unit
+
+ AURenderCallbackStruct input;
+ input.inputProc = (AURenderCallback)(osx_sink::AUOutputCallback);
+ input.inputProcRefCon = this;
+
+ err = AudioUnitSetProperty(d_OutputAU,
+ kAudioUnitProperty_SetRenderCallback,
+ kAudioUnitScope_Input,
+ 0,
+ &input,
+ sizeof (input));
+ CheckErrorAndThrow(err, "AudioUnitSetProperty Render Callback",
+ "audio_osx_sink::audio_osx_sink");
+
+ // tell the Output Unit what format data will be supplied to it
+ // so that it handles any format conversions
+
+ AudioStreamBasicDescription streamFormat;
+ streamFormat.mSampleRate = (Float64)(sample_rate);
+ streamFormat.mFormatID = kAudioFormatLinearPCM;
+ streamFormat.mFormatFlags = (kLinearPCMFormatFlagIsFloat |
+ GR_PCM_ENDIANNESS |
+ kLinearPCMFormatFlagIsPacked |
+ kAudioFormatFlagIsNonInterleaved);
+ streamFormat.mBytesPerPacket = 4;
+ streamFormat.mFramesPerPacket = 1;
+ streamFormat.mBytesPerFrame = 4;
+ streamFormat.mChannelsPerFrame = d_n_channels;
+ streamFormat.mBitsPerChannel = 32;
+
+ err = AudioUnitSetProperty(d_OutputAU,
+ kAudioUnitProperty_StreamFormat,
+ kAudioUnitScope_Input,
+ 0,
+ &streamFormat,
+ sizeof(AudioStreamBasicDescription));
+ CheckErrorAndThrow(err, "AudioUnitSetProperty StreamFormat",
+ "audio_osx_sink::audio_osx_sink");
+
+ // create the stuff to regulate I/O
+
+ d_cond_data = new gruel::condition_variable();
+ if(d_cond_data == NULL)
+ CheckErrorAndThrow(errno, "new condition (data)",
+ "audio_osx_sink::audio_osx_sink");
+
+ d_internal = new gruel::mutex();
+ if(d_internal == NULL)
+ CheckErrorAndThrow(errno, "new mutex (internal)",
+ "audio_osx_sink::audio_osx_sink");
+
+ // initialize the AU for output
+
+ err = AudioUnitInitialize(d_OutputAU);
+ CheckErrorAndThrow(err, "AudioUnitInitialize",
+ "audio_osx_sink::audio_osx_sink");
+
+#if _OSX_AU_DEBUG_
+ std::cerr << "audio_osx_sink Parameters:" << std::endl;
+ std::cerr << " Sample Rate is " << d_sample_rate << std::endl;
+ std::cerr << " Number of Channels is " << d_n_channels << std::endl;
+ std::cerr << " Max # samples to store per channel is " << d_max_sample_count << std::endl;
+#endif
+}
+
+ bool
+ osx_sink::IsRunning()
+ {
+ UInt32 AURunning = 0, AUSize = sizeof(UInt32);
+
+ OSStatus err = AudioUnitGetProperty(d_OutputAU,
+ kAudioOutputUnitProperty_IsRunning,
+ kAudioUnitScope_Global,
+ 0,
+ &AURunning,
+ &AUSize);
+ CheckErrorAndThrow(err, "AudioUnitGetProperty IsRunning",
+ "audio_osx_sink::IsRunning");
+
+ return (AURunning);
+ }
+
+ bool
+ osx_sink::start()
+ {
+ if(!IsRunning()) {
+ OSStatus err = AudioOutputUnitStart(d_OutputAU);
+ CheckErrorAndThrow(err, "AudioOutputUnitStart",
+ "audio_osx_sink::start");
+ }
+
+ return (true);
+ }
+
+ bool
+ osx_sink::stop()
+ {
+ if(IsRunning ()) {
+ OSStatus err = AudioOutputUnitStop(d_OutputAU);
+ CheckErrorAndThrow(err, "AudioOutputUnitStop",
+ "audio_osx_sink::stop");
+
+ for(UInt32 n = 0; n < d_n_channels; n++) {
+ d_buffers[n]->abort();
+ }
+ }
+
+ return (true);
+ }
+
+ osx_sink::~osx_sink()
+ {
+ // stop and close the AudioUnit
+ stop();
+ AudioUnitUninitialize(d_OutputAU);
+#ifndef GR_USE_OLD_AUDIO_UNIT
+ AudioComponentInstanceDispose(d_OutputAU);
+#else
+ CloseComponent(d_OutputAU);
+#endif
+
+ // empty and delete the queues
+ for(UInt32 n = 0; n < d_n_channels; n++) {
+ delete d_buffers[n];
+ d_buffers[n] = 0;
+ }
+ delete [] d_buffers;
+ d_buffers = 0;
+
+ // close and delete control stuff
+ delete d_cond_data;
+ d_cond_data = 0;
+ delete d_internal;
+ d_internal = 0;
+ }
+
+ int
+ osx_sink::work(int noutput_items,
+ gr_vector_const_void_star &input_items,
+ gr_vector_void_star &output_items)
+ {
+ gruel::scoped_lock l(*d_internal);
+
+ /* take the input data, copy it, and push it to the bottom of the queue
+ mono input are pushed onto queue[0];
+ stereo input are pushed onto queue[1].
+ Start the AudioUnit if necessary. */
+
+ UInt32 l_max_count;
+ int diff_count = d_max_sample_count - noutput_items;
+ if(diff_count < 0)
+ l_max_count = 0;
+ else
+ l_max_count = (UInt32)diff_count;
+
+#if 0
+ if(l_max_count < d_queueItemLength->back()) {
+ // allow 2 buffers at a time, regardless of length
+ l_max_count = d_queueItemLength->back();
+ }
+#endif
+
+#if _OSX_AU_DEBUG_
+ std::cerr << "work1: qSC = " << d_queueSampleCount
+ << ", lMC = "<< l_max_count
+ << ", dmSC = " << d_max_sample_count
+ << ", nOI = " << noutput_items << std::endl;
+#endif
+
+ if(d_queueSampleCount > l_max_count) {
+ // data coming in too fast; do_block decides what to do
+ if(d_do_block == true) {
+ // block until there is data to return
+ while(d_queueSampleCount > l_max_count) {
+ // release control so-as to allow data to be retrieved;
+ // block until there is data to return
+ d_cond_data->wait(l);
+ // the condition's 'notify' was called; acquire control
+ // to keep thread safe
+ }
+ }
+ }
+ // not blocking case and overflow is handled by the circular buffer
+
+ // add the input frames to the buffers' queue, checking for overflow
+
+ UInt32 l_counter;
+ int res = 0;
+ float* inBuffer = (float*)input_items[0];
+ const UInt32 l_size = input_items.size();
+ for(l_counter = 0; l_counter < l_size; l_counter++) {
+ inBuffer = (float*)input_items[l_counter];
+ int l_res = d_buffers[l_counter]->enqueue(inBuffer,
+ noutput_items);
+ if(l_res == -1)
+ res = -1;
+ }
+ while(l_counter < d_n_channels) {
+ // for extra channels, copy the last input's data
+ int l_res = d_buffers[l_counter++]->enqueue(inBuffer,
+ noutput_items);
+ if(l_res == -1)
+ res = -1;
+ }
+
+ if(res == -1) {
+ // data coming in too fast
+ // drop oldest buffer
+ fputs("aO", stderr);
+ fflush(stderr);
+ // set the local number of samples available to the max
+ d_queueSampleCount = d_buffers[0]->buffer_length_items();
+ }
+ else {
+ // keep up the local sample count
+ d_queueSampleCount += noutput_items;
+ }
+
+#if _OSX_AU_DEBUG_
+ std::cerr << "work2: #OI = "
+ << noutput_items << ", #Cnt = "
+ << d_queueSampleCount << ", mSC = "
+ << d_max_sample_count << std::endl;
+#endif
+
+ return (noutput_items);
+ }
+
+ OSStatus
+ osx_sink::AUOutputCallback(void *inRefCon,
+ AudioUnitRenderActionFlags *ioActionFlags,
+ const AudioTimeStamp *inTimeStamp,
+ UInt32 inBusNumber,
+ UInt32 inNumberFrames,
+ AudioBufferList *ioData)
+ {
+ osx_sink* This = (osx_sink*)inRefCon;
+ OSStatus err = noErr;
+
+ gruel::scoped_lock l(*This->d_internal);
+
+#if _OSX_AU_DEBUG_
+ std::cerr << "cb_in: SC = " << This->d_queueSampleCount
+ << ", in#F = " << inNumberFrames << std::endl;
+#endif
+
+ if(This->d_queueSampleCount < inNumberFrames) {
+ // not enough data to fill request
+ err = -1;
+ }
+ else {
+ // enough data; remove data from our buffers into the AU's buffers
+ int l_counter = This->d_n_channels;
+
+ while(--l_counter >= 0) {
+ size_t t_n_output_items = inNumberFrames;
+ float* outBuffer = (float*)ioData->mBuffers[l_counter].mData;
+ This->d_buffers[l_counter]->dequeue(outBuffer, &t_n_output_items);
+ if(t_n_output_items != inNumberFrames) {
+ throw std::runtime_error("audio_osx_sink::AUOutputCallback(): "
+ "number of available items changing "
+ "unexpectedly.\n");
+ }
+ }
+
+ This->d_queueSampleCount -= inNumberFrames;
+ }
+
+#if _OSX_AU_DEBUG_
+ std::cerr << "cb_out: SC = " << This->d_queueSampleCount << std::endl;
+#endif
+
+ // signal that data is available
+ This->d_cond_data->notify_one();
+
+ return (err);
+ }
+
+ } /* namespace audio */
+} /* namespace gr */
diff --git a/gr-audio/lib/osx/osx_sink.h b/gr-audio/lib/osx/osx_sink.h
new file mode 100644
index 0000000000..6bbd882239
--- /dev/null
+++ b/gr-audio/lib/osx/osx_sink.h
@@ -0,0 +1,86 @@
+/* -*- c++ -*- */
+/*
+ * Copyright 2006-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.
+ */
+
+#ifndef INCLUDED_AUDIO_OSX_SINK_H
+#define INCLUDED_AUDIO_OSX_SINK_H
+
+#include <audio/sink.h>
+#include <string>
+#include <list>
+#include <AudioUnit/AudioUnit.h>
+#include <circular_buffer.h>
+
+namespace gr {
+ namespace audio {
+
+ /*!
+ * \brief audio sink using OSX
+ * \ingroup audio_blk
+ *
+ * input signature is one or two streams of floats.
+ * Input samples must be in the range [-1,1].
+ */
+
+ class osx_sink : public sink
+ {
+ Float64 d_sample_rate;
+ int d_channel_config;
+ UInt32 d_n_channels;
+ UInt32 d_queueSampleCount, d_max_sample_count;
+ bool d_do_block;
+ gruel::mutex* d_internal;
+ gruel::condition_variable* d_cond_data;
+ circular_buffer<float>** d_buffers;
+
+ // AudioUnits and Such
+ AudioUnit d_OutputAU;
+
+ public:
+ osx_sink(int sample_rate = 44100,
+ const std::string device_name = "2",
+ bool do_block = true,
+ int channel_config = -1,
+ int max_sample_count = -1);
+
+ ~osx_sink();
+
+ bool IsRunning();
+ bool start();
+ bool stop();
+
+ int work(int noutput_items,
+ gr_vector_const_void_star &input_items,
+ gr_vector_void_star &output_items);
+
+ private:
+ static OSStatus AUOutputCallback(void *inRefCon,
+ AudioUnitRenderActionFlags *ioActionFlags,
+ const AudioTimeStamp *inTimeStamp,
+ UInt32 inBusNumber,
+ UInt32 inNumberFrames,
+ AudioBufferList *ioData);
+ };
+
+ } /* namespace audio */
+} /* namespace gr */
+
+#endif /* INCLUDED_AUDIO_OSX_SINK_H */
diff --git a/gr-audio/lib/osx/osx_source.cc b/gr-audio/lib/osx/osx_source.cc
new file mode 100644
index 0000000000..c79a6ec6e9
--- /dev/null
+++ b/gr-audio/lib/osx/osx_source.cc
@@ -0,0 +1,1077 @@
+/* -*- c++ -*- */
+/*
+ * Copyright 2006-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 <osx_source.h>
+#include <osx_impl.h>
+#include <gr_io_signature.h>
+#include <stdexcept>
+
+namespace gr {
+ namespace audio {
+
+#define _OSX_AU_DEBUG_ 0
+#define _OSX_DO_LISTENERS_ 0
+
+ AUDIO_REGISTER_SOURCE(REG_PRIO_HIGH, osx)(int sampling_rate,
+ const std::string &device_name,
+ bool ok_to_block)
+ {
+ return source::sptr
+ (new osx_source(sampling_rate, device_name, ok_to_block));
+ }
+
+ void
+ PrintStreamDesc(AudioStreamBasicDescription *inDesc)
+ {
+ if(inDesc == NULL) {
+ std::cerr << "PrintStreamDesc: Can't print a NULL desc!" << std::endl;
+ return;
+ }
+
+ std::cerr << " Sample Rate : " << inDesc->mSampleRate << std::endl;
+ char format_id[4];
+ strncpy(format_id, (char*)(&inDesc->mFormatID), 4);
+ std::cerr << " Format ID : " << format_id << std::endl;
+ std::cerr << " Format Flags : " << inDesc->mFormatFlags << std::endl;
+ std::cerr << " Bytes per Packet : " << inDesc->mBytesPerPacket << std::endl;
+ std::cerr << " Frames per Packet : " << inDesc->mFramesPerPacket << std::endl;
+ std::cerr << " Bytes per Frame : " << inDesc->mBytesPerFrame << std::endl;
+ std::cerr << " Channels per Frame : " << inDesc->mChannelsPerFrame << std::endl;
+ std::cerr << " Bits per Channel : " << inDesc->mBitsPerChannel << std::endl;
+ }
+
+ // FIXME these should query some kind of user preference
+
+ osx_source::osx_source(int sample_rate,
+ const std::string device_name,
+ bool do_block,
+ int channel_config,
+ int max_sample_count)
+ : gr_sync_block("audio_osx_source",
+ gr_make_io_signature(0, 0, 0),
+ gr_make_io_signature(0, 0, 0)),
+ d_deviceSampleRate(0.0), d_outputSampleRate(0.0),
+ d_channel_config(0),
+ d_inputBufferSizeFrames(0), d_inputBufferSizeBytes(0),
+ d_outputBufferSizeFrames(0), d_outputBufferSizeBytes(0),
+ d_deviceBufferSizeFrames(0), d_deviceBufferSizeBytes(0),
+ d_leadSizeFrames(0), d_leadSizeBytes(0),
+ d_trailSizeFrames(0), d_trailSizeBytes(0),
+ d_extraBufferSizeFrames(0), d_extraBufferSizeBytes(0),
+ d_queueSampleCount(0), d_max_sample_count(0),
+ d_n_AvailableInputFrames(0), d_n_ActualInputFrames(0),
+ d_n_user_channels(0), d_n_max_channels(0), d_n_deviceChannels(0),
+ d_do_block(do_block), d_passThrough(false),
+ d_internal(0), d_cond_data(0),
+ d_buffers(0),
+ d_InputAU(0), d_InputBuffer(0), d_OutputBuffer(0),
+ d_AudioConverter(0)
+ {
+ if(sample_rate <= 0) {
+ std::cerr << "Invalid Sample Rate: " << sample_rate << std::endl;
+ throw std::invalid_argument("audio_osx_source::audio_osx_source");
+ }
+ else
+ d_outputSampleRate = (Float64)sample_rate;
+
+ if(channel_config <= 0 & channel_config != -1) {
+ std::cerr << "Invalid Channel Config: " << channel_config << std::endl;
+ throw std::invalid_argument("audio_osx_source::audio_osx_source");
+ }
+ else if (channel_config == -1) {
+ // no user input; try "device name" instead
+ int l_n_channels = (int)strtol(device_name.data(), (char **)NULL, 10);
+ if((l_n_channels == 0) & errno) {
+ std::cerr << "Error Converting Device Name: " << errno << std::endl;
+ throw std::invalid_argument("audio_osx_source::audio_osx_source");
+ }
+ if(l_n_channels <= 0)
+ channel_config = 2;
+ else
+ channel_config = l_n_channels;
+ }
+
+ d_channel_config = channel_config;
+
+ // check that the max # of samples to store is valid
+
+ if(max_sample_count == -1)
+ max_sample_count = sample_rate;
+ else if(max_sample_count <= 0) {
+ std::cerr << "Invalid Max Sample Count: " << max_sample_count << std::endl;
+ throw std::invalid_argument("audio_osx_source::audio_osx_source");
+ }
+
+ d_max_sample_count = max_sample_count;
+
+#if _OSX_AU_DEBUG_
+ std::cerr << "source(): max # samples = " << d_max_sample_count << std::endl;
+#endif
+
+ OSStatus err = noErr;
+
+ // create the default AudioUnit for input
+
+ // Open the default input unit
+#ifndef GR_USE_OLD_AUDIO_UNIT
+ AudioComponentDescription InputDesc;
+#else
+ ComponentDescription InputDesc;
+#endif
+
+ InputDesc.componentType = kAudioUnitType_Output;
+ InputDesc.componentSubType = kAudioUnitSubType_HALOutput;
+ InputDesc.componentManufacturer = kAudioUnitManufacturer_Apple;
+ InputDesc.componentFlags = 0;
+ InputDesc.componentFlagsMask = 0;
+
+#ifndef GR_USE_OLD_AUDIO_UNIT
+ AudioComponent comp = AudioComponentFindNext(NULL, &InputDesc);
+#else
+ Component comp = FindNextComponent(NULL, &InputDesc);
+#endif
+
+ if(comp == NULL) {
+#ifndef GR_USE_OLD_AUDIO_UNIT
+ std::cerr << "AudioComponentFindNext Error" << std::endl;
+#else
+ std::cerr << "FindNextComponent Error" << std::endl;
+#endif
+ throw std::runtime_error("audio_osx_source::audio_osx_source");
+ }
+
+#ifndef GR_USE_OLD_AUDIO_UNIT
+ err = AudioComponentInstanceNew(comp, &d_InputAU);
+ CheckErrorAndThrow(err, "AudioComponentInstanceNew",
+ "audio_osx_source::audio_osx_source");
+#else
+ err = OpenAComponent(comp, &d_InputAU);
+ CheckErrorAndThrow(err, "OpenAComponent",
+ "audio_osx_source::audio_osx_source");
+#endif
+
+ UInt32 enableIO;
+
+ // must enable the AUHAL for input and disable output
+ // before setting the AUHAL's current device
+
+ // Enable input on the AUHAL
+ enableIO = 1;
+ err = AudioUnitSetProperty(d_InputAU,
+ kAudioOutputUnitProperty_EnableIO,
+ kAudioUnitScope_Input,
+ 1, // input element
+ &enableIO,
+ sizeof(UInt32));
+ CheckErrorAndThrow(err, "AudioUnitSetProperty Input Enable",
+ "audio_osx_source::audio_osx_source");
+
+ // Disable output on the AUHAL
+ enableIO = 0;
+ err = AudioUnitSetProperty(d_InputAU,
+ kAudioOutputUnitProperty_EnableIO,
+ kAudioUnitScope_Output,
+ 0, // output element
+ &enableIO,
+ sizeof (UInt32));
+ CheckErrorAndThrow(err, "AudioUnitSetProperty Output Disable",
+ "audio_osx_source::audio_osx_source");
+
+ // set the default input device for our input AU
+
+ SetDefaultInputDeviceAsCurrent();
+
+#if _OSX_DO_LISTENERS_
+ // set up a listener if default hardware input device changes
+
+ err = AudioHardwareAddPropertyListener
+ (kAudioHardwarePropertyDefaultInputDevice,
+ (AudioHardwarePropertyListenerProc)HardwareListener,
+ this);
+
+ CheckErrorAndThrow(err, "AudioHardwareAddPropertyListener",
+ "audio_osx_source::audio_osx_source");
+
+ // Add a listener for any changes in the input AU's output stream
+ // the function "UnitListener" will be called if the stream format
+ // changes for whatever reason
+
+ err = AudioUnitAddPropertyListener
+ (d_InputAU,
+ kAudioUnitProperty_StreamFormat,
+ (AudioUnitPropertyListenerProc)UnitListener,
+ this);
+ CheckErrorAndThrow(err, "Adding Unit Property Listener",
+ "audio_osx_source::audio_osx_source");
+#endif
+
+ // Now find out if it actually can do input.
+
+ UInt32 hasInput = 0;
+ UInt32 dataSize = sizeof(hasInput);
+ err = AudioUnitGetProperty(d_InputAU,
+ kAudioOutputUnitProperty_HasIO,
+ kAudioUnitScope_Input,
+ 1,
+ &hasInput,
+ &dataSize);
+ CheckErrorAndThrow(err, "AudioUnitGetProperty HasIO",
+ "audio_osx_source::audio_osx_source");
+ if(hasInput == 0) {
+ std::cerr << "Selected Audio Device does not support Input." << std::endl;
+ throw std::runtime_error("audio_osx_source::audio_osx_source");
+ }
+
+ // Set up a callback function to retrieve input from the Audio Device
+
+ AURenderCallbackStruct AUCallBack;
+
+ AUCallBack.inputProc = (AURenderCallback)(osx_source::AUInputCallback);
+ AUCallBack.inputProcRefCon = this;
+
+ err = AudioUnitSetProperty(d_InputAU,
+ kAudioOutputUnitProperty_SetInputCallback,
+ kAudioUnitScope_Global,
+ 0,
+ &AUCallBack,
+ sizeof (AURenderCallbackStruct));
+ CheckErrorAndThrow(err, "AudioUnitSetProperty Input Callback",
+ "audio_osx_source::audio_osx_source");
+
+ UInt32 propertySize;
+ AudioStreamBasicDescription asbd_device, asbd_client, asbd_user;
+
+ // asbd_device: ASBD of the device that is creating the input data stream
+ // asbd_client: ASBD of the client size (output) of the hardware device
+ // asbd_user: ASBD of the user's arguments
+
+ // Get the Stream Format (device side)
+
+ propertySize = sizeof(asbd_device);
+ err = AudioUnitGetProperty(d_InputAU,
+ kAudioUnitProperty_StreamFormat,
+ kAudioUnitScope_Input,
+ 1,
+ &asbd_device,
+ &propertySize);
+ CheckErrorAndThrow(err, "AudioUnitGetProperty Device Input Stream Format",
+ "audio_osx_source::audio_osx_source");
+
+#if _OSX_AU_DEBUG_
+ std::cerr << std::endl << "---- Device Stream Format ----" << std::endl;
+ PrintStreamDesc(&asbd_device);
+#endif
+
+ // Get the Stream Format (client side)
+ propertySize = sizeof(asbd_client);
+ err = AudioUnitGetProperty(d_InputAU,
+ kAudioUnitProperty_StreamFormat,
+ kAudioUnitScope_Output,
+ 1,
+ &asbd_client,
+ &propertySize);
+ CheckErrorAndThrow(err, "AudioUnitGetProperty Device Ouput Stream Format",
+ "audio_osx_source::audio_osx_source");
+
+#if _OSX_AU_DEBUG_
+ std::cerr << std::endl << "---- Client Stream Format ----" << std::endl;
+ PrintStreamDesc(&asbd_client);
+#endif
+
+ // Set the format of all the AUs to the input/output devices channel count
+
+ // get the max number of input (& thus output) channels supported by
+ // this device
+ d_n_max_channels = asbd_device.mChannelsPerFrame;
+
+ // create the output io signature;
+ // no input siganture to set (source is hardware)
+ set_output_signature(gr_make_io_signature(1,
+ d_n_max_channels,
+ sizeof(float)));
+
+ // allocate the output circular buffer(s), one per channel
+ d_buffers = (circular_buffer<float>**)new
+ circular_buffer<float>* [d_n_max_channels];
+ UInt32 n_alloc = (UInt32)ceil((double)d_max_sample_count);
+ for(UInt32 n = 0; n < d_n_max_channels; n++) {
+ d_buffers[n] = new circular_buffer<float>(n_alloc, false, false);
+ }
+
+ d_deviceSampleRate = asbd_device.mSampleRate;
+ d_n_deviceChannels = asbd_device.mChannelsPerFrame;
+
+ asbd_client.mSampleRate = asbd_device.mSampleRate;
+ asbd_client.mFormatID = kAudioFormatLinearPCM;
+ asbd_client.mFormatFlags = (kAudioFormatFlagIsFloat |
+ kAudioFormatFlagIsPacked |
+ kAudioFormatFlagIsNonInterleaved);
+ if((asbd_client.mFormatID == kAudioFormatLinearPCM) &&
+ (d_n_deviceChannels == 1)) {
+ asbd_client.mFormatFlags &= ~kLinearPCMFormatFlagIsNonInterleaved;
+ }
+ asbd_client.mBytesPerFrame = sizeof(float);
+ asbd_client.mFramesPerPacket = 1;
+ asbd_client.mBitsPerChannel = asbd_client.mBytesPerFrame * 8;
+ asbd_client.mChannelsPerFrame = d_n_deviceChannels;
+ asbd_client.mBytesPerPacket = asbd_client.mBytesPerFrame;
+
+ propertySize = sizeof(AudioStreamBasicDescription);
+ err = AudioUnitSetProperty(d_InputAU,
+ kAudioUnitProperty_StreamFormat,
+ kAudioUnitScope_Output,
+ 1,
+ &asbd_client,
+ propertySize);
+ CheckErrorAndThrow(err, "AudioUnitSetProperty Device Ouput Stream Format",
+ "audio_osx_source::audio_osx_source");
+
+ // create an ASBD for the user's wants
+
+ asbd_user.mSampleRate = d_outputSampleRate;
+ asbd_user.mFormatID = kAudioFormatLinearPCM;
+ asbd_user.mFormatFlags = (kLinearPCMFormatFlagIsFloat |
+ GR_PCM_ENDIANNESS |
+ kLinearPCMFormatFlagIsPacked |
+ kAudioFormatFlagIsNonInterleaved);
+ asbd_user.mBytesPerPacket = sizeof(float);
+ asbd_user.mFramesPerPacket = 1;
+ asbd_user.mBytesPerFrame = asbd_user.mBytesPerPacket;
+ asbd_user.mChannelsPerFrame = d_n_deviceChannels;
+ asbd_user.mBitsPerChannel = asbd_user.mBytesPerPacket * 8;
+
+ if(d_deviceSampleRate == d_outputSampleRate) {
+ // no need to do conversion if asbd_client matches user wants
+ d_passThrough = true;
+ d_leadSizeFrames = d_trailSizeFrames = 0L;
+ }
+ else {
+ d_passThrough = false;
+ // Create the audio converter
+
+ err = AudioConverterNew(&asbd_client, &asbd_user, &d_AudioConverter);
+ CheckErrorAndThrow(err, "AudioConverterNew",
+ "audio_osx_source::audio_osx_source");
+
+ // Set the audio converter sample rate quality to "max" ...
+ // requires more samples, but should sound nicer
+
+ UInt32 ACQuality = kAudioConverterQuality_Max;
+ propertySize = sizeof(ACQuality);
+ err = AudioConverterSetProperty(d_AudioConverter,
+ kAudioConverterSampleRateConverterQuality,
+ propertySize,
+ &ACQuality);
+ CheckErrorAndThrow(err, "AudioConverterSetProperty "
+ "SampleRateConverterQuality",
+ "audio_osx_source::audio_osx_source");
+
+ // set the audio converter's prime method to "pre",
+ // which uses both leading and trailing frames
+ // from the "current input". All of this is handled
+ // internally by the AudioConverter; we just supply
+ // the frames for conversion.
+
+ // UInt32 ACPrimeMethod = kConverterPrimeMethod_None;
+ UInt32 ACPrimeMethod = kConverterPrimeMethod_Pre;
+ propertySize = sizeof (ACPrimeMethod);
+ err = AudioConverterSetProperty(d_AudioConverter,
+ kAudioConverterPrimeMethod,
+ propertySize,
+ &ACPrimeMethod);
+ CheckErrorAndThrow(err, "AudioConverterSetProperty PrimeMethod",
+ "audio_osx_source::audio_osx_source");
+
+ // Get the size of the I/O buffer(s) to allow for pre-allocated buffers
+
+ // lead frame info (trail frame info is ignored)
+
+ AudioConverterPrimeInfo ACPrimeInfo = {0, 0};
+ propertySize = sizeof(ACPrimeInfo);
+ err = AudioConverterGetProperty(d_AudioConverter,
+ kAudioConverterPrimeInfo,
+ &propertySize,
+ &ACPrimeInfo);
+ CheckErrorAndThrow(err, "AudioConverterGetProperty PrimeInfo",
+ "audio_osx_source::audio_osx_source");
+
+ switch(ACPrimeMethod) {
+ case(kConverterPrimeMethod_None):
+ d_leadSizeFrames =
+ d_trailSizeFrames = 0L;
+ break;
+ case(kConverterPrimeMethod_Normal):
+ d_leadSizeFrames = 0L;
+ d_trailSizeFrames = ACPrimeInfo.trailingFrames;
+ break;
+ default:
+ d_leadSizeFrames = ACPrimeInfo.leadingFrames;
+ d_trailSizeFrames = ACPrimeInfo.trailingFrames;
+ }
+ }
+ d_leadSizeBytes = d_leadSizeFrames * sizeof(Float32);
+ d_trailSizeBytes = d_trailSizeFrames * sizeof(Float32);
+
+ propertySize = sizeof(d_deviceBufferSizeFrames);
+ err = AudioUnitGetProperty(d_InputAU,
+ kAudioDevicePropertyBufferFrameSize,
+ kAudioUnitScope_Global,
+ 0,
+ &d_deviceBufferSizeFrames,
+ &propertySize);
+ CheckErrorAndThrow(err, "AudioUnitGetProperty Buffer Frame Size",
+ "audio_osx_source::audio_osx_source");
+
+ d_deviceBufferSizeBytes = d_deviceBufferSizeFrames * sizeof(Float32);
+ d_inputBufferSizeBytes = d_deviceBufferSizeBytes + d_leadSizeBytes;
+ d_inputBufferSizeFrames = d_deviceBufferSizeFrames + d_leadSizeFrames;
+
+ // outBufSizeBytes = floor (inBufSizeBytes * rate_out / rate_in)
+ // since this is rarely exact, we need another buffer to hold
+ // "extra" samples not processed at any given sampling period
+ // this buffer must be at least 4 floats in size, but generally
+ // follows the rule that
+ // extraBufSize = ceil (rate_in / rate_out)*sizeof(float)
+
+ d_extraBufferSizeFrames = ((UInt32)ceil(d_deviceSampleRate
+ / d_outputSampleRate)
+ * sizeof(float));
+ if(d_extraBufferSizeFrames < 4)
+ d_extraBufferSizeFrames = 4;
+ d_extraBufferSizeBytes = d_extraBufferSizeFrames * sizeof(float);
+
+ d_outputBufferSizeFrames = (UInt32)ceil(((Float64)d_inputBufferSizeFrames)
+ * d_outputSampleRate
+ / d_deviceSampleRate);
+ d_outputBufferSizeBytes = d_outputBufferSizeFrames * sizeof(float);
+ d_inputBufferSizeFrames += d_extraBufferSizeFrames;
+
+ // pre-alloc all buffers
+
+ AllocAudioBufferList(&d_InputBuffer, d_n_deviceChannels,
+ d_inputBufferSizeBytes);
+ if(d_passThrough == false) {
+ AllocAudioBufferList(&d_OutputBuffer, d_n_max_channels,
+ d_outputBufferSizeBytes);
+ }
+ else {
+ d_OutputBuffer = d_InputBuffer;
+ }
+
+ // create the stuff to regulate I/O
+
+ d_cond_data = new gruel::condition_variable();
+ if(d_cond_data == NULL)
+ CheckErrorAndThrow(errno, "new condition (data)",
+ "audio_osx_source::audio_osx_source");
+
+ d_internal = new gruel::mutex();
+ if(d_internal == NULL)
+ CheckErrorAndThrow(errno, "new mutex (internal)",
+ "audio_osx_source::audio_osx_source");
+
+ // initialize the AU for input
+
+ err = AudioUnitInitialize(d_InputAU);
+ CheckErrorAndThrow(err, "AudioUnitInitialize",
+ "audio_osx_source::audio_osx_source");
+
+#if _OSX_AU_DEBUG_
+ std::cerr << "audio_osx_source Parameters:" << std::endl;
+ std::cerr << " Device Sample Rate is " << d_deviceSampleRate << std::endl;
+ std::cerr << " User Sample Rate is " << d_outputSampleRate << std::endl;
+ std::cerr << " Max Sample Count is " << d_max_sample_count << std::endl;
+ std::cerr << " # Device Channels is " << d_n_deviceChannels << std::endl;
+ std::cerr << " # Max Channels is " << d_n_max_channels << std::endl;
+ std::cerr << " Device Buffer Size is Frames = " << d_deviceBufferSizeFrames << std::endl;
+ std::cerr << " Lead Size is Frames = " << d_leadSizeFrames << std::endl;
+ std::cerr << " Trail Size is Frames = " << d_trailSizeFrames << std::endl;
+ std::cerr << " Input Buffer Size is Frames = " << d_inputBufferSizeFrames << std::endl;
+ std::cerr << " Output Buffer Size is Frames = " << d_outputBufferSizeFrames << std::endl;
+#endif
+ }
+
+ void
+ osx_source::AllocAudioBufferList(AudioBufferList** t_ABL,
+ UInt32 n_channels,
+ UInt32 bufferSizeBytes)
+ {
+ FreeAudioBufferList(t_ABL);
+ UInt32 propertySize = (offsetof(AudioBufferList, mBuffers[0]) +
+ (sizeof(AudioBuffer) * n_channels));
+ *t_ABL = (AudioBufferList*)calloc(1, propertySize);
+ (*t_ABL)->mNumberBuffers = n_channels;
+
+ int counter = n_channels;
+
+ while(--counter >= 0) {
+ (*t_ABL)->mBuffers[counter].mNumberChannels = 1;
+ (*t_ABL)->mBuffers[counter].mDataByteSize = bufferSizeBytes;
+ (*t_ABL)->mBuffers[counter].mData = calloc (1, bufferSizeBytes);
+ }
+ }
+
+ void
+ osx_source::FreeAudioBufferList(AudioBufferList** t_ABL)
+ {
+ // free pre-allocated audio buffer, if it exists
+ if(*t_ABL != NULL) {
+ int counter = (*t_ABL)->mNumberBuffers;
+ while(--counter >= 0)
+ free((*t_ABL)->mBuffers[counter].mData);
+ free(*t_ABL);
+ (*t_ABL) = 0;
+ }
+ }
+
+ bool
+ osx_source::IsRunning()
+ {
+ UInt32 AURunning = 0, AUSize = sizeof(UInt32);
+
+ OSStatus err = AudioUnitGetProperty(d_InputAU,
+ kAudioOutputUnitProperty_IsRunning,
+ kAudioUnitScope_Global,
+ 0,
+ &AURunning,
+ &AUSize);
+ CheckErrorAndThrow(err, "AudioUnitGetProperty IsRunning",
+ "audio_osx_source::IsRunning");
+
+ return (AURunning);
+ }
+
+ bool
+ osx_source::start()
+ {
+ if(! IsRunning ()) {
+ OSStatus err = AudioOutputUnitStart(d_InputAU);
+ CheckErrorAndThrow(err, "AudioOutputUnitStart",
+ "audio_osx_source::start");
+ }
+
+ return (true);
+ }
+
+ bool
+ osx_source::stop()
+ {
+ if(IsRunning ()) {
+ OSStatus err = AudioOutputUnitStop(d_InputAU);
+ CheckErrorAndThrow(err, "AudioOutputUnitStart",
+ "audio_osx_source::stop");
+ for(UInt32 n = 0; n < d_n_user_channels; n++) {
+ d_buffers[n]->abort ();
+ }
+ }
+
+ return (true);
+ }
+
+ osx_source::~osx_source()
+ {
+ OSStatus err = noErr;
+
+ // stop the AudioUnit
+ stop();
+
+#if _OSX_DO_LISTENERS_
+ // remove the listeners
+
+ err = AudioUnitRemovePropertyListener
+ (d_InputAU,
+ kAudioUnitProperty_StreamFormat,
+ (AudioUnitPropertyListenerProc)UnitListener);
+ CheckError(err, "~audio_osx_source: AudioUnitRemovePropertyListener");
+
+ err = AudioHardwareRemovePropertyListener
+ (kAudioHardwarePropertyDefaultInputDevice,
+ (AudioHardwarePropertyListenerProc)HardwareListener);
+ CheckError(err, "~audio_osx_source: AudioHardwareRemovePropertyListener");
+#endif
+
+ // free pre-allocated audio buffers
+ FreeAudioBufferList(&d_InputBuffer);
+
+ if(d_passThrough == false) {
+ err = AudioConverterDispose(d_AudioConverter);
+ CheckError(err, "~audio_osx_source: AudioConverterDispose");
+ FreeAudioBufferList(&d_OutputBuffer);
+ }
+
+ // remove the audio unit
+ err = AudioUnitUninitialize(d_InputAU);
+ CheckError(err, "~audio_osx_source: AudioUnitUninitialize");
+
+#ifndef GR_USE_OLD_AUDIO_UNIT
+ err = AudioComponentInstanceDispose(d_InputAU);
+ CheckError(err, "~audio_osx_source: AudioComponentInstanceDispose");
+#else
+ err = CloseComponent(d_InputAU);
+ CheckError(err, "~audio_osx_source: CloseComponent");
+#endif
+
+ // empty and delete the queues
+ for(UInt32 n = 0; n < d_n_max_channels; n++) {
+ delete d_buffers[n];
+ d_buffers[n] = 0;
+ }
+ delete [] d_buffers;
+ d_buffers = 0;
+
+ // close and delete the control stuff
+ delete d_cond_data;
+ d_cond_data = 0;
+ delete d_internal;
+ d_internal = 0;
+ }
+
+ bool
+ osx_source::check_topology(int ninputs, int noutputs)
+ {
+ // check # inputs to make sure it's valid
+ if(ninputs != 0) {
+ std::cerr << "audio_osx_source::check_topology(): number of input "
+ << "streams provided (" << ninputs
+ << ") should be 0." << std::endl;
+ throw std::runtime_error("audio_osx_source::check_topology()");
+ }
+
+ // check # outputs to make sure it's valid
+ if((noutputs < 1) | (noutputs > (int) d_n_max_channels)) {
+ std::cerr << "audio_osx_source::check_topology(): number of output "
+ << "streams provided (" << noutputs << ") should be in [1,"
+ << d_n_max_channels << "] for the selected audio device."
+ << std::endl;
+ throw std::runtime_error("audio_osx_source::check_topology()");
+ }
+
+ // save the actual number of output (user) channels
+ d_n_user_channels = noutputs;
+
+#if _OSX_AU_DEBUG_
+ std::cerr << "chk_topo: Actual # user output channels = "
+ << noutputs << std::endl;
+#endif
+
+ return (true);
+ }
+
+ int
+ osx_source::work(int noutput_items,
+ gr_vector_const_void_star &input_items,
+ gr_vector_void_star &output_items)
+ {
+ // acquire control to do processing here only
+ gruel::scoped_lock l(*d_internal);
+
+#if _OSX_AU_DEBUG_
+ std::cerr << "work1: SC = " << d_queueSampleCount
+ << ", #OI = " << noutput_items
+ << ", #Chan = " << output_items.size() << std::endl;
+#endif
+
+ // set the actual # of output items to the 'desired' amount then
+ // verify that data is available; if not enough data is available,
+ // either wait until it is (is "do_block" is true), return (0) is no
+ // data is available and "do_block" is false, or process the actual
+ // amount of available data.
+
+ UInt32 actual_noutput_items = noutput_items;
+
+ if(d_queueSampleCount < actual_noutput_items) {
+ if(d_queueSampleCount == 0) {
+ // no data; do_block decides what to do
+ if(d_do_block == true) {
+ while(d_queueSampleCount == 0) {
+ // release control so-as to allow data to be retrieved;
+ // block until there is data to return
+ d_cond_data->wait(l);
+ // the condition's 'notify' was called; acquire control to
+ // keep thread safe
+ }
+ }
+ else {
+ // no data & not blocking; return nothing
+ return (0);
+ }
+ }
+ // use the actual amount of available data
+ actual_noutput_items = d_queueSampleCount;
+ }
+
+ // number of channels
+ int l_counter = (int)output_items.size();
+
+ // copy the items from the circular buffer(s) to 'work's output buffers
+ // verify that the number copied out is as expected.
+
+ while(--l_counter >= 0) {
+ size_t t_n_output_items = actual_noutput_items;
+ d_buffers[l_counter]->dequeue((float*)output_items[l_counter],
+ &t_n_output_items);
+ if(t_n_output_items != actual_noutput_items) {
+ std::cerr << "audio_osx_source::work(): ERROR: number of "
+ << "available items changing unexpectedly; expecting "
+ << actual_noutput_items << ", got "
+ << t_n_output_items << "." << std::endl;
+ throw std::runtime_error("audio_osx_source::work()");
+ }
+ }
+
+ // subtract the actual number of items removed from the buffer(s)
+ // from the local accounting of the number of available samples
+
+ d_queueSampleCount -= actual_noutput_items;
+
+#if _OSX_AU_DEBUG_
+ std::cerr << "work2: SC = " << d_queueSampleCount
+ << ", act#OI = " << actual_noutput_items << std::endl
+ << "Returning." << std::endl;
+#endif
+
+ return (actual_noutput_items);
+ }
+
+ OSStatus
+ osx_source::ConverterCallback(AudioConverterRef inAudioConverter,
+ UInt32* ioNumberDataPackets,
+ AudioBufferList* ioData,
+ AudioStreamPacketDescription** ioASPD,
+ void* inUserData)
+ {
+ // take current device buffers and copy them to the tail of the
+ // input buffers the lead buffer is already there in the first
+ // d_leadSizeFrames slots
+
+ osx_source* This = static_cast<osx_source*>(inUserData);
+ AudioBufferList* l_inputABL = This->d_InputBuffer;
+ UInt32 totalInputBufferSizeBytes = ((*ioNumberDataPackets) * sizeof(float));
+ int counter = This->d_n_deviceChannels;
+ ioData->mNumberBuffers = This->d_n_deviceChannels;
+ This->d_n_ActualInputFrames = (*ioNumberDataPackets);
+
+#if _OSX_AU_DEBUG_
+ std::cerr << "cc1: io#DP = " << (*ioNumberDataPackets)
+ << ", TIBSB = " << totalInputBufferSizeBytes
+ << ", #C = " << counter << std::endl;
+#endif
+
+ while(--counter >= 0) {
+ AudioBuffer* l_ioD_AB = &(ioData->mBuffers[counter]);
+ l_ioD_AB->mNumberChannels = 1;
+ l_ioD_AB->mData = (float*)(l_inputABL->mBuffers[counter].mData);
+ l_ioD_AB->mDataByteSize = totalInputBufferSizeBytes;
+ }
+
+#if _OSX_AU_DEBUG_
+ std::cerr << "cc2: Returning." << std::endl;
+#endif
+
+ return (noErr);
+ }
+
+ OSStatus
+ osx_source::AUInputCallback(void* inRefCon,
+ AudioUnitRenderActionFlags* ioActionFlags,
+ const AudioTimeStamp* inTimeStamp,
+ UInt32 inBusNumber,
+ UInt32 inNumberFrames,
+ AudioBufferList* ioData)
+ {
+ OSStatus err = noErr;
+ osx_source* This = static_cast<osx_source*>(inRefCon);
+
+ gruel::scoped_lock l(*This->d_internal);
+
+#if _OSX_AU_DEBUG_
+ std::cerr << "cb0: in#F = " << inNumberFrames
+ << ", inBN = " << inBusNumber
+ << ", SC = " << This->d_queueSampleCount << std::endl;
+#endif
+
+ // Get the new audio data from the input device
+
+ err = AudioUnitRender(This->d_InputAU,
+ ioActionFlags,
+ inTimeStamp,
+ 1, //inBusNumber,
+ inNumberFrames,
+ This->d_InputBuffer);
+ CheckErrorAndThrow(err, "AudioUnitRender",
+ "audio_osx_source::AUInputCallback");
+
+ UInt32 AvailableInputFrames = inNumberFrames;
+ This->d_n_AvailableInputFrames = inNumberFrames;
+
+ // get the number of actual output frames,
+ // either via converting the buffer or not
+
+ UInt32 ActualOutputFrames;
+
+ if(This->d_passThrough == true) {
+ ActualOutputFrames = AvailableInputFrames;
+ }
+ else {
+ UInt32 AvailableInputBytes = AvailableInputFrames * sizeof(float);
+ UInt32 AvailableOutputBytes = AvailableInputBytes;
+ UInt32 AvailableOutputFrames = AvailableOutputBytes / sizeof(float);
+ UInt32 propertySize = sizeof (AvailableOutputBytes);
+ err = AudioConverterGetProperty(This->d_AudioConverter,
+ kAudioConverterPropertyCalculateOutputBufferSize,
+ &propertySize,
+ &AvailableOutputBytes);
+ CheckErrorAndThrow(err, "AudioConverterGetProperty CalculateOutputBufferSize",
+ "audio_osx_source::audio_osx_source");
+
+ AvailableOutputFrames = AvailableOutputBytes / sizeof(float);
+
+#if 0
+ // when decimating too much, the output sounds warbly due to
+ // fluctuating # of output frames
+ // This should not be a surprise, but there's probably some
+ // clever programming that could lessed the effect ...
+ // like finding the "ideal" # of output frames, and keeping
+ // that number constant no matter the # of input frames
+ UInt32 l_InputBytes = AvailableOutputBytes;
+ propertySize = sizeof(AvailableOutputBytes);
+ err = AudioConverterGetProperty(This->d_AudioConverter,
+ kAudioConverterPropertyCalculateInputBufferSize,
+ &propertySize,
+ &l_InputBytes);
+ CheckErrorAndThrow(err, "AudioConverterGetProperty CalculateInputBufferSize",
+ "audio_osx_source::audio_osx_source");
+
+ if(l_InputBytes < AvailableInputBytes) {
+ // OK to zero pad the input a little
+ AvailableOutputFrames += 1;
+ AvailableOutputBytes = AvailableOutputFrames * sizeof(float);
+ }
+#endif
+
+#if _OSX_AU_DEBUG_
+ std::cerr << "cb1: avail: #IF = " << AvailableInputFrames
+ << ", #OF = " << AvailableOutputFrames << std::endl;
+#endif
+ ActualOutputFrames = AvailableOutputFrames;
+
+ // convert the data to the correct rate
+ // on input, ActualOutputFrames is the number of available output frames
+
+ err = AudioConverterFillComplexBuffer(This->d_AudioConverter,
+ (AudioConverterComplexInputDataProc)
+ (This->ConverterCallback),
+ inRefCon,
+ &ActualOutputFrames,
+ This->d_OutputBuffer,
+ NULL);
+ CheckErrorAndThrow(err, "AudioConverterFillComplexBuffer",
+ "audio_osx_source::AUInputCallback");
+
+ // on output, ActualOutputFrames is the actual number of output frames
+
+#if _OSX_AU_DEBUG_
+ std::cerr << "cb2: actual: #IF = " << This->d_n_ActualInputFrames
+ << ", #OF = " << AvailableOutputFrames << std::endl;
+ if(This->d_n_ActualInputFrames != AvailableInputFrames)
+ std::cerr << "cb2.1: avail#IF = " << AvailableInputFrames
+ << ", actual#IF = " << This->d_n_ActualInputFrames << std::endl;
+#endif
+ }
+
+ // add the output frames to the buffers' queue, checking for overflow
+
+ int l_counter = This->d_n_user_channels;
+ int res = 0;
+
+ while(--l_counter >= 0) {
+ float* inBuffer = (float*) This->d_OutputBuffer->mBuffers[l_counter].mData;
+
+#if _OSX_AU_DEBUG_
+ std::cerr << "cb3: enqueuing audio data." << std::endl;
+#endif
+
+ int l_res = This->d_buffers[l_counter]->enqueue(inBuffer, ActualOutputFrames);
+ if(l_res == -1)
+ res = -1;
+ }
+
+ if(res == -1) {
+ // data coming in too fast
+ // drop oldest buffer
+ fputs("aO", stderr);
+ fflush(stderr);
+ // set the local number of samples available to the max
+ This->d_queueSampleCount = This->d_buffers[0]->buffer_length_items();
+ }
+ else {
+ // keep up the local sample count
+ This->d_queueSampleCount += ActualOutputFrames;
+ }
+
+#if _OSX_AU_DEBUG_
+ std::cerr << "cb4: #OI = " << ActualOutputFrames
+ << ", #Cnt = " << This->d_queueSampleCount
+ << ", mSC = " << This->d_max_sample_count << std::endl;
+#endif
+
+ // signal that data is available, if appropraite
+ This->d_cond_data->notify_one();
+
+#if _OSX_AU_DEBUG_
+ std::cerr << "cb5: returning." << std::endl;
+#endif
+
+ return (err);
+ }
+
+ void
+ osx_source::SetDefaultInputDeviceAsCurrent()
+ {
+ // set the default input device
+ AudioDeviceID deviceID = 0;
+ UInt32 dataSize = sizeof (AudioDeviceID);
+ OSStatus err = noErr;
+
+#ifndef GR_USE_OLD_AUDIO_UNIT
+ AudioObjectPropertyAddress theAddress =
+ { kAudioHardwarePropertyDefaultInputDevice,
+ kAudioObjectPropertyScopeGlobal,
+ kAudioObjectPropertyElementMaster };
+
+ err = AudioObjectGetPropertyData(kAudioObjectSystemObject,
+ &theAddress,
+ 0,
+ NULL,
+ &dataSize,
+ &deviceID);
+#else
+ err = AudioHardwareGetProperty(kAudioHardwarePropertyDefaultInputDevice,
+ &dataSize,
+ &deviceID);
+#endif
+
+ CheckErrorAndThrow(err, "Get Audio Unit Property for Current Device",
+ "audio_osx_source::SetDefaultInputDeviceAsCurrent");
+
+ err = AudioUnitSetProperty(d_InputAU,
+ kAudioOutputUnitProperty_CurrentDevice,
+ kAudioUnitScope_Global,
+ 0,
+ &deviceID,
+ sizeof(AudioDeviceID));
+
+ CheckErrorAndThrow(err, "AudioUnitSetProperty Current Device",
+ "audio_osx_source::SetDefaultInputDeviceAsCurrent");
+}
+
+#if _OSX_DO_LISTENERS_
+ OSStatus
+ osx_source::HardwareListener(AudioHardwarePropertyID inPropertyID,
+ void *inClientData)
+ {
+ OSStatus err = noErr;
+ osx_source* This = static_cast<osx_source*>(inClientData);
+
+ std::cerr << "a_o_s::HardwareListener" << std::endl;
+
+ // set the new default hardware input device for use by our AU
+
+ This->SetDefaultInputDeviceAsCurrent();
+
+ // reset the converter to tell it that the stream has changed
+
+ err = AudioConverterReset(This->d_AudioConverter);
+ CheckErrorAndThrow(err, "AudioConverterReset",
+ "audio_osx_source::UnitListener");
+
+ return (err);
+ }
+
+ OSStatus
+ osx_source::UnitListener(void *inRefCon,
+ AudioUnit ci,
+ AudioUnitPropertyID inID,
+ AudioUnitScope inScope,
+ AudioUnitElement inElement)
+ {
+ OSStatus err = noErr;
+ osx_source* This = static_cast<osx_source*>(inRefCon);
+ AudioStreamBasicDescription asbd;
+
+ std::cerr << "a_o_s::UnitListener" << std::endl;
+
+ // get the converter's input ASBD (for printing)
+
+ UInt32 propertySize = sizeof(asbd);
+ err = AudioConverterGetProperty(This->d_AudioConverter,
+ kAudioConverterCurrentInputStreamDescription,
+ &propertySize,
+ &asbd);
+ CheckErrorAndThrow(err, "AudioConverterGetProperty "
+ "CurrentInputStreamDescription",
+ "audio_osx_source::UnitListener");
+
+ std::cerr << "UnitListener: Input Source changed." << std::endl
+ << "Old Source Output Info:" << std::endl;
+ PrintStreamDesc(&asbd);
+
+ // get the new input unit's output ASBD
+
+ propertySize = sizeof(asbd);
+ err = AudioUnitGetProperty(This->d_InputAU,
+ kAudioUnitProperty_StreamFormat,
+ kAudioUnitScope_Output, 1,
+ &asbd, &propertySize);
+ CheckErrorAndThrow(err, "AudioUnitGetProperty StreamFormat",
+ "audio_osx_source::UnitListener");
+
+ std::cerr << "New Source Output Info:" << std::endl;
+ PrintStreamDesc(&asbd);
+
+ // set the converter's input ASBD to this
+
+ err = AudioConverterSetProperty(This->d_AudioConverter,
+ kAudioConverterCurrentInputStreamDescription,
+ propertySize,
+ &asbd);
+ CheckErrorAndThrow(err, "AudioConverterSetProperty "
+ "CurrentInputStreamDescription",
+ "audio_osx_source::UnitListener");
+
+ // reset the converter to tell it that the stream has changed
+
+ err = AudioConverterReset(This->d_AudioConverter);
+ CheckErrorAndThrow(err, "AudioConverterReset",
+ "audio_osx_source::UnitListener");
+
+ return (err);
+ }
+#endif /* _OSX_DO_LISTENERS_ */
+
+ } /* namespace audio */
+} /* namespace gr */
diff --git a/gr-audio/lib/osx/osx_source.h b/gr-audio/lib/osx/osx_source.h
new file mode 100644
index 0000000000..9315c8e44e
--- /dev/null
+++ b/gr-audio/lib/osx/osx_source.h
@@ -0,0 +1,121 @@
+/* -*- c++ -*- */
+/*
+ * Copyright 2006-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.
+ */
+
+#ifndef INCLUDED_AUDIO_OSX_SOURCE_H
+#define INCLUDED_AUDIO_OSX_SOURCE_H
+
+#include <audio/source.h>
+#include <string>
+#include <AudioToolbox/AudioToolbox.h>
+#include <AudioUnit/AudioUnit.h>
+#include <circular_buffer.h>
+
+namespace gr {
+ namespace audio {
+
+ /*!
+ * \brief audio source using OSX
+ * \ingroup audio_blk
+ *
+ * Input signature is one or two streams of floats.
+ * Samples must be in the range [-1,1].
+ */
+ class osx_source : public source
+ {
+ Float64 d_deviceSampleRate, d_outputSampleRate;
+ int d_channel_config;
+ UInt32 d_inputBufferSizeFrames, d_inputBufferSizeBytes;
+ UInt32 d_outputBufferSizeFrames, d_outputBufferSizeBytes;
+ UInt32 d_deviceBufferSizeFrames, d_deviceBufferSizeBytes;
+ UInt32 d_leadSizeFrames, d_leadSizeBytes;
+ UInt32 d_trailSizeFrames, d_trailSizeBytes;
+ UInt32 d_extraBufferSizeFrames, d_extraBufferSizeBytes;
+ UInt32 d_queueSampleCount, d_max_sample_count;
+ UInt32 d_n_AvailableInputFrames, d_n_ActualInputFrames;
+ UInt32 d_n_user_channels, d_n_max_channels, d_n_deviceChannels;
+ bool d_do_block, d_passThrough, d_waiting_for_data;
+ gruel::mutex* d_internal;
+ gruel::condition_variable* d_cond_data;
+ circular_buffer<float>** d_buffers;
+
+ // AudioUnits and Such
+ AudioUnit d_InputAU;
+ AudioBufferList* d_InputBuffer;
+ AudioBufferList* d_OutputBuffer;
+ AudioConverterRef d_AudioConverter;
+
+ public:
+ osx_source(int sample_rate = 44100,
+ const std::string device_name = "",
+ bool do_block = true,
+ int channel_config = -1,
+ int max_sample_count = -1);
+
+ ~osx_source();
+
+ bool start();
+ bool stop();
+ bool IsRunning();
+
+ bool check_topology(int ninputs, int noutputs);
+
+ int work(int noutput_items,
+ gr_vector_const_void_star &input_items,
+ gr_vector_void_star &output_items);
+
+ private:
+ void SetDefaultInputDeviceAsCurrent();
+
+ void AllocAudioBufferList(AudioBufferList** t_ABL,
+ UInt32 n_channels,
+ UInt32 inputBufferSizeBytes);
+
+ void FreeAudioBufferList(AudioBufferList** t_ABL);
+
+ static OSStatus ConverterCallback(AudioConverterRef inAudioConverter,
+ UInt32* ioNumberDataPackets,
+ AudioBufferList* ioData,
+ AudioStreamPacketDescription** outASPD,
+ void* inUserData);
+
+ static OSStatus AUInputCallback(void *inRefCon,
+ AudioUnitRenderActionFlags *ioActionFlags,
+ const AudioTimeStamp *inTimeStamp,
+ UInt32 inBusNumber,
+ UInt32 inNumberFrames,
+ AudioBufferList *ioData);
+#if _OSX_DO_LISTENERS_
+ static OSStatus UnitListener(void *inRefCon,
+ AudioUnit ci,
+ AudioUnitPropertyID inID,
+ AudioUnitScope inScope,
+ AudioUnitElement inElement);
+
+ static OSStatus HardwareListener(AudioHardwarePropertyID inPropertyID,
+ void *inClientData);
+#endif
+ };
+
+ } /* namespace audio */
+} /* namespace gr */
+
+#endif /* INCLUDED_AUDIO_OSX_SOURCE_H */
diff --git a/gr-audio/lib/portaudio/audio_portaudio_sink.cc b/gr-audio/lib/portaudio/audio_portaudio_sink.cc
deleted file mode 100644
index af7f1e48c5..0000000000
--- a/gr-audio/lib/portaudio/audio_portaudio_sink.cc
+++ /dev/null
@@ -1,362 +0,0 @@
-/* -*- c++ -*- */
-/*
- * Copyright 2006-2011 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 he 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 "gr_audio_registry.h"
-#include <audio_portaudio_sink.h>
-#include <gr_io_signature.h>
-#include <gr_prefs.h>
-#include <stdio.h>
-#include <iostream>
-#include <unistd.h>
-#include <stdexcept>
-#include <gri_portaudio.h>
-#include <string.h>
-
-AUDIO_REGISTER_SINK(REG_PRIO_MED, portaudio)(
- int sampling_rate, const std::string &device_name, bool ok_to_block
-){
- return audio_sink::sptr(new audio_portaudio_sink(sampling_rate, device_name, ok_to_block));
-}
-
-//#define LOGGING 0 // define to 0 or 1
-
-#define SAMPLE_FORMAT paFloat32
-typedef float sample_t;
-
-// Number of portaudio buffers in the ringbuffer
-static const unsigned int N_BUFFERS = 4;
-
-static std::string
-default_device_name ()
-{
- return gr_prefs::singleton()->get_string("audio_portaudio", "default_output_device", "");
-}
-
-void
-audio_portaudio_sink::create_ringbuffer(void)
-{
- int bufsize_samples = d_portaudio_buffer_size_frames * d_output_parameters.channelCount;
-
- if (d_verbose)
- fprintf(stderr,"ring buffer size = %d frames\n",
- N_BUFFERS*bufsize_samples/d_output_parameters.channelCount);
-
- // FYI, the buffer indicies are in units of samples.
- d_writer = gr_make_buffer(N_BUFFERS * bufsize_samples, sizeof(sample_t));
- d_reader = gr_buffer_add_reader(d_writer, 0);
-}
-
-/*
- * This routine will be called by the PortAudio engine when audio is needed.
- * It may called at interrupt level on some machines so don't do anything
- * that could mess up the system like calling malloc() or free().
- *
- * Our job is to write framesPerBuffer frames into outputBuffer.
- */
-int
-portaudio_sink_callback (const void *inputBuffer,
- void *outputBuffer,
- unsigned long framesPerBuffer,
- const PaStreamCallbackTimeInfo* timeInfo,
- PaStreamCallbackFlags statusFlags,
- void *arg)
-{
- audio_portaudio_sink *self = (audio_portaudio_sink *)arg;
- int nreqd_samples =
- framesPerBuffer * self->d_output_parameters.channelCount;
-
- int navail_samples = self->d_reader->items_available();
-
- if (nreqd_samples <= navail_samples) { // We've got enough data...
- {
- gruel::scoped_lock guard(self->d_ringbuffer_mutex);
-
- memcpy(outputBuffer,
- self->d_reader->read_pointer(),
- nreqd_samples * sizeof(sample_t));
- self->d_reader->update_read_pointer(nreqd_samples);
-
- self->d_ringbuffer_ready = true;
- }
-
- // Tell the sink thread there is new room in the ringbuffer.
- self->d_ringbuffer_cond.notify_one();
- return paContinue;
- }
-
- else { // underrun
- self->d_nunderuns++;
- ssize_t r = ::write(2, "aU", 2); // FIXME change to non-blocking call
- if(r == -1) {
- perror("audio_portaudio_source::portaudio_source_callback write error to stderr.");
- }
-
- // FIXME we should transfer what we've got and pad the rest
- memset(outputBuffer, 0, nreqd_samples * sizeof(sample_t));
-
- self->d_ringbuffer_ready = true;
- self->d_ringbuffer_cond.notify_one(); // Tell the sink to get going!
-
- return paContinue;
- }
-}
-
-
-// ----------------------------------------------------------------
-
-audio_portaudio_sink::audio_portaudio_sink(int sampling_rate,
- const std::string device_name,
- bool ok_to_block)
- : gr_sync_block ("audio_portaudio_sink",
- gr_make_io_signature(0, 0, 0),
- gr_make_io_signature(0, 0, 0)),
- d_sampling_rate(sampling_rate),
- d_device_name(device_name.empty() ? default_device_name() : device_name),
- d_ok_to_block(ok_to_block),
- d_verbose(gr_prefs::singleton()->get_bool("audio_portaudio", "verbose", false)),
- d_portaudio_buffer_size_frames(0),
- d_stream(0),
- d_ringbuffer_mutex(),
- d_ringbuffer_cond(),
- d_ringbuffer_ready(false),
- d_nunderuns(0)
-{
- memset(&d_output_parameters, 0, sizeof(d_output_parameters));
- //if (LOGGING)
- // d_log = gri_logger::singleton();
-
- PaError err;
- int i, numDevices;
- PaDeviceIndex device = 0;
- const PaDeviceInfo *deviceInfo = NULL;
-
- err = Pa_Initialize();
- if (err != paNoError) {
- bail ("Initialize failed", err);
- }
-
- if (d_verbose)
- gri_print_devices();
-
- numDevices = Pa_GetDeviceCount();
- if (numDevices < 0)
- bail("Pa Device count failed", 0);
- if (numDevices == 0)
- bail("no devices available", 0);
-
- if (d_device_name.empty())
- {
- // FIXME Get smarter about picking something
- fprintf(stderr,"\nUsing Default Device\n");
- device = Pa_GetDefaultOutputDevice();
- deviceInfo = Pa_GetDeviceInfo(device);
- fprintf(stderr,"%s is the chosen device using %s as the host\n",
- deviceInfo->name, Pa_GetHostApiInfo(deviceInfo->hostApi)->name);
- }
- else
- {
- bool found = false;
- fprintf(stderr,"\nTest Devices\n");
- for (i=0;i<numDevices;i++) {
- deviceInfo = Pa_GetDeviceInfo( i );
- fprintf(stderr,"Testing device name: %s",deviceInfo->name);
- if (deviceInfo->maxOutputChannels <= 0) {
- fprintf(stderr,"\n");
- continue;
- }
- if (strstr(deviceInfo->name, d_device_name.c_str())){
- fprintf(stderr," Chosen!\n");
- device = i;
- fprintf(stderr,"%s using %s as the host\n",d_device_name.c_str(),
- Pa_GetHostApiInfo(deviceInfo->hostApi)->name), fflush(stderr);
- found = true;
- deviceInfo = Pa_GetDeviceInfo(device);
- i = numDevices; // force loop exit
- }
- else
- fprintf(stderr,"\n"),fflush(stderr);
- }
-
- if (!found){
- bail("Failed to find specified device name", 0);
- exit(1);
- }
- }
-
-
- d_output_parameters.device = device;
- d_output_parameters.channelCount = deviceInfo->maxOutputChannels;
- d_output_parameters.sampleFormat = SAMPLE_FORMAT;
- d_output_parameters.suggestedLatency = deviceInfo->defaultLowOutputLatency;
- d_output_parameters.hostApiSpecificStreamInfo = NULL;
-
- // We fill in the real channelCount in check_topology when we know
- // how many inputs are connected to us.
-
- // Now that we know the maximum number of channels (allegedly)
- // supported by the h/w, we can compute a reasonable input
- // signature. The portaudio specs say that they'll accept any
- // number of channels from 1 to max.
- set_input_signature(gr_make_io_signature(1, deviceInfo->maxOutputChannels,
- sizeof (sample_t)));
-}
-
-
-bool
-audio_portaudio_sink::check_topology (int ninputs, int noutputs)
-{
- PaError err;
-
- if (Pa_IsStreamActive(d_stream))
- {
- Pa_CloseStream(d_stream);
- d_stream = 0;
- d_reader.reset(); // boost::shared_ptr for d_reader = 0
- d_writer.reset(); // boost::shared_ptr for d_write = 0
- }
-
- d_output_parameters.channelCount = ninputs; // # of channels we're really using
-
-#if 1
- d_portaudio_buffer_size_frames = (int)(0.0213333333 * d_sampling_rate + 0.5); // Force 1024 frame buffers at 48000
- fprintf(stderr, "Latency = %8.5f, requested sampling_rate = %g\n", // Force latency to 21.3333333.. ms
- 0.0213333333, (double)d_sampling_rate);
-#endif
- err = Pa_OpenStream(&d_stream,
- NULL, // No input
- &d_output_parameters,
- d_sampling_rate,
- d_portaudio_buffer_size_frames,
- paClipOff,
- &portaudio_sink_callback,
- (void*)this);
-
- if (err != paNoError) {
- output_error_msg ("OpenStream failed", err);
- return false;
- }
-
-#if 0
- const PaStreamInfo *psi = Pa_GetStreamInfo(d_stream);
-
- d_portaudio_buffer_size_frames = (int)(d_output_parameters.suggestedLatency * psi->sampleRate);
- fprintf(stderr, "Latency = %7.4f, psi->sampleRate = %g\n",
- d_output_parameters.suggestedLatency, psi->sampleRate);
-#endif
-
- fprintf(stderr, "d_portaudio_buffer_size_frames = %d\n", d_portaudio_buffer_size_frames);
-
- assert(d_portaudio_buffer_size_frames != 0);
-
- create_ringbuffer();
-
- err = Pa_StartStream(d_stream);
- if (err != paNoError) {
- output_error_msg ("StartStream failed", err);
- return false;
- }
-
- return true;
-}
-
-audio_portaudio_sink::~audio_portaudio_sink ()
-{
- Pa_StopStream(d_stream); // wait for output to drain
- Pa_CloseStream(d_stream);
- Pa_Terminate();
-}
-
-/*
- * This version consumes everything sent to it, blocking if required.
- * I think this will allow us better control of the total buffering/latency
- * in the audio path.
- */
-int
-audio_portaudio_sink::work (int noutput_items,
- gr_vector_const_void_star &input_items,
- gr_vector_void_star &output_items)
-{
- const float **in = (const float **) &input_items[0];
- const unsigned nchan = d_output_parameters.channelCount; // # of channels == samples/frame
-
- int k;
-
- for (k = 0; k < noutput_items; ){
- int nframes = d_writer->space_available() / nchan; // How much space in ringbuffer
- if (nframes == 0){ // no room...
- if (d_ok_to_block){
- {
- gruel::scoped_lock guard(d_ringbuffer_mutex);
- while (!d_ringbuffer_ready)
- d_ringbuffer_cond.wait(guard);
- }
-
- continue;
- }
- else {
- // There's no room and we're not allowed to block.
- // (A USRP is most likely controlling the pacing through the pipeline.)
- // We drop the samples on the ground, and say we processed them all ;)
- //
- // FIXME, there's probably room for a bit more finesse here.
- return noutput_items;
- }
- }
-
- // We can write the smaller of the request and the room we've got
- {
- gruel::scoped_lock guard(d_ringbuffer_mutex);
-
- int nf = std::min(noutput_items - k, nframes);
- float *p = (float *) d_writer->write_pointer();
-
- for (int i = 0; i < nf; i++)
- for (unsigned int c = 0; c < nchan; c++)
- *p++ = in[c][k + i];
-
- d_writer->update_write_pointer(nf * nchan);
- k += nf;
-
- d_ringbuffer_ready = false;
- }
- }
-
- return k; // tell how many we actually did
-}
-
-void
-audio_portaudio_sink::output_error_msg (const char *msg, int err)
-{
- fprintf (stderr, "audio_portaudio_sink[%s]: %s: %s\n",
- d_device_name.c_str (), msg, Pa_GetErrorText(err));
-}
-
-void
-audio_portaudio_sink::bail (const char *msg, int err) throw (std::runtime_error)
-{
- output_error_msg (msg, err);
- throw std::runtime_error ("audio_portaudio_sink");
-}
diff --git a/gr-audio/lib/portaudio/audio_portaudio_sink.h b/gr-audio/lib/portaudio/audio_portaudio_sink.h
deleted file mode 100644
index cf64d3da04..0000000000
--- a/gr-audio/lib/portaudio/audio_portaudio_sink.h
+++ /dev/null
@@ -1,86 +0,0 @@
-/* -*- c++ -*- */
-/*
- * Copyright 2006-2011 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.
- */
-#ifndef INCLUDED_AUDIO_PORTAUDIO_SINK_H
-#define INCLUDED_AUDIO_PORTAUDIO_SINK_H
-
-#include <gr_audio_sink.h>
-#include <gr_buffer.h>
-#include <gruel/thread.h>
-#include <string>
-#include <portaudio.h>
-#include <stdexcept>
-//#include <gri_logger.h>
-
-PaStreamCallback portaudio_sink_callback;
-
-
-/*!
- * \brief Audio sink using PORTAUDIO
- * \ingroup audio_blk
- *
- * Input samples must be in the range [-1,1].
- */
-class audio_portaudio_sink : public audio_sink {
-
- friend PaStreamCallback portaudio_sink_callback;
-
-
- unsigned int d_sampling_rate;
- std::string d_device_name;
- bool d_ok_to_block;
- bool d_verbose;
-
- unsigned int d_portaudio_buffer_size_frames; // number of frames in a portaudio buffer
-
- PaStream *d_stream;
- PaStreamParameters d_output_parameters;
-
- gr_buffer_sptr d_writer; // buffer used between work and callback
- gr_buffer_reader_sptr d_reader;
-
- gruel::mutex d_ringbuffer_mutex;
- gruel::condition_variable d_ringbuffer_cond;
- bool d_ringbuffer_ready;
-
- // random stats
- int d_nunderuns; // count of underruns
- //gri_logger_sptr d_log; // handle to non-blocking logging instance
-
- void output_error_msg (const char *msg, int err);
- void bail (const char *msg, int err) throw (std::runtime_error);
- void create_ringbuffer();
-
-
-public:
- audio_portaudio_sink (int sampling_rate, const std::string device_name,
- bool ok_to_block);
-
- ~audio_portaudio_sink ();
-
- bool check_topology (int ninputs, int noutputs);
-
- int work (int noutput_items,
- gr_vector_const_void_star &input_items,
- gr_vector_void_star &output_items);
-};
-
-#endif /* INCLUDED_AUDIO_PORTAUDIO_SINK_H */
diff --git a/gr-audio/lib/portaudio/audio_portaudio_source.cc b/gr-audio/lib/portaudio/audio_portaudio_source.cc
deleted file mode 100644
index ddb1a6fb65..0000000000
--- a/gr-audio/lib/portaudio/audio_portaudio_source.cc
+++ /dev/null
@@ -1,374 +0,0 @@
-/* -*- c++ -*- */
-/*
- * Copyright 2006-2011 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 he 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 "gr_audio_registry.h"
-#include <audio_portaudio_source.h>
-#include <gr_io_signature.h>
-#include <gr_prefs.h>
-#include <stdio.h>
-#include <iostream>
-#include <unistd.h>
-#include <stdexcept>
-#include <gri_portaudio.h>
-#include <string.h>
-
-AUDIO_REGISTER_SOURCE(REG_PRIO_MED, portaudio)(
- int sampling_rate, const std::string &device_name, bool ok_to_block
-){
- return audio_source::sptr(new audio_portaudio_source(sampling_rate, device_name, ok_to_block));
-}
-
-//#define LOGGING 0 // define to 0 or 1
-
-#define SAMPLE_FORMAT paFloat32
-typedef float sample_t;
-
-// Number of portaudio buffers in the ringbuffer
-static const unsigned int N_BUFFERS = 4;
-
-static std::string
-default_device_name ()
-{
- return gr_prefs::singleton()->get_string("audio_portaudio", "default_input_device", "");
-}
-
-void
-audio_portaudio_source::create_ringbuffer(void)
-{
- int bufsize_samples = d_portaudio_buffer_size_frames * d_input_parameters.channelCount;
-
- if (d_verbose)
- fprintf(stderr, "ring buffer size = %d frames\n",
- N_BUFFERS*bufsize_samples/d_input_parameters.channelCount);
-
- // FYI, the buffer indicies are in units of samples.
- d_writer = gr_make_buffer(N_BUFFERS * bufsize_samples, sizeof(sample_t));
- d_reader = gr_buffer_add_reader(d_writer, 0);
-}
-
-/*
- * This routine will be called by the PortAudio engine when audio is needed.
- * It may called at interrupt level on some machines so don't do anything
- * that could mess up the system like calling malloc() or free().
- *
- * Our job is to copy framesPerBuffer frames from inputBuffer.
- */
-int
-portaudio_source_callback (const void *inputBuffer,
- void *outputBuffer,
- unsigned long framesPerBuffer,
- const PaStreamCallbackTimeInfo* timeInfo,
- PaStreamCallbackFlags statusFlags,
- void *arg)
-{
- audio_portaudio_source *self = (audio_portaudio_source *)arg;
- int nchan = self->d_input_parameters.channelCount;
- int nframes_to_copy = framesPerBuffer;
- int nframes_room = self->d_writer->space_available() / nchan;
-
- if (nframes_to_copy <= nframes_room){ // We've got room for the data ..
- //if (LOGGING)
- // self->d_log->printf("PAsrc cb: f/b = %4ld\n", framesPerBuffer);
-
- // copy from input buffer to ringbuffer
- {
- gruel::scoped_lock(d_ringbuffer_mutex);
-
- memcpy(self->d_writer->write_pointer(),
- inputBuffer,
- nframes_to_copy * nchan * sizeof(sample_t));
- self->d_writer->update_write_pointer(nframes_to_copy * nchan);
-
- // Tell the source thread there is new data in the ringbuffer.
- self->d_ringbuffer_ready = true;
- }
-
- self->d_ringbuffer_cond.notify_one();
- return paContinue;
- }
-
- else { // overrun
- self->d_noverruns++;
- ssize_t r = ::write(2, "aO", 2); // FIXME change to non-blocking call
- if(r == -1) {
- perror("audio_portaudio_source::portaudio_source_callback write error to stderr.");
- }
-
- self->d_ringbuffer_ready = false;
- self->d_ringbuffer_cond.notify_one(); // Tell the sink to get going!
- return paContinue;
- }
-}
-
-
-// ----------------------------------------------------------------
-
-audio_portaudio_source::audio_portaudio_source(int sampling_rate,
- const std::string device_name,
- bool ok_to_block)
- : gr_sync_block ("audio_portaudio_source",
- gr_make_io_signature(0, 0, 0),
- gr_make_io_signature(0, 0, 0)),
- d_sampling_rate(sampling_rate),
- d_device_name(device_name.empty() ? default_device_name() : device_name),
- d_ok_to_block(ok_to_block),
- d_verbose(gr_prefs::singleton()->get_bool("audio_portaudio", "verbose", false)),
- d_portaudio_buffer_size_frames(0),
- d_stream(0),
- d_ringbuffer_mutex(),
- d_ringbuffer_cond(),
- d_ringbuffer_ready(false),
- d_noverruns(0)
-{
- memset(&d_input_parameters, 0, sizeof(d_input_parameters));
- //if (LOGGING)
- // d_log = gri_logger::singleton();
-
- PaError err;
- int i, numDevices;
- PaDeviceIndex device = 0;
- const PaDeviceInfo *deviceInfo = NULL;
-
-
- err = Pa_Initialize();
- if (err != paNoError) {
- bail ("Initialize failed", err);
- }
-
- if (d_verbose)
- gri_print_devices();
-
- numDevices = Pa_GetDeviceCount();
- if (numDevices < 0)
- bail("Pa Device count failed", 0);
- if (numDevices == 0)
- bail("no devices available", 0);
-
- if (d_device_name.empty())
- {
- // FIXME Get smarter about picking something
- device = Pa_GetDefaultInputDevice();
- deviceInfo = Pa_GetDeviceInfo(device);
- fprintf(stderr,"%s is the chosen device using %s as the host\n",
- deviceInfo->name, Pa_GetHostApiInfo(deviceInfo->hostApi)->name);
- }
- else
- {
- bool found = false;
-
- for (i=0;i<numDevices;i++) {
- deviceInfo = Pa_GetDeviceInfo( i );
- fprintf(stderr,"Testing device name: %s",deviceInfo->name);
- if (deviceInfo->maxInputChannels <= 0) {
- fprintf(stderr,"\n");
- continue;
- }
- if (strstr(deviceInfo->name, d_device_name.c_str())){
- fprintf(stderr," Chosen!\n");
- device = i;
- fprintf(stderr,"%s using %s as the host\n",d_device_name.c_str(),
- Pa_GetHostApiInfo(deviceInfo->hostApi)->name), fflush(stderr);
- found = true;
- deviceInfo = Pa_GetDeviceInfo(device);
- i = numDevices; // force loop exit
- }
- else
- fprintf(stderr,"\n"),fflush(stderr);
- }
-
- if (!found){
- bail("Failed to find specified device name", 0);
- }
- }
-
-
- d_input_parameters.device = device;
- d_input_parameters.channelCount = deviceInfo->maxInputChannels;
- d_input_parameters.sampleFormat = SAMPLE_FORMAT;
- d_input_parameters.suggestedLatency = deviceInfo->defaultLowInputLatency;
- d_input_parameters.hostApiSpecificStreamInfo = NULL;
-
- // We fill in the real channelCount in check_topology when we know
- // how many inputs are connected to us.
-
- // Now that we know the maximum number of channels (allegedly)
- // supported by the h/w, we can compute a reasonable output
- // signature. The portaudio specs say that they'll accept any
- // number of channels from 1 to max.
- set_output_signature(gr_make_io_signature(1, deviceInfo->maxInputChannels,
- sizeof (sample_t)));
-}
-
-
-bool
-audio_portaudio_source::check_topology (int ninputs, int noutputs)
-{
- PaError err;
-
- if (Pa_IsStreamActive(d_stream))
- {
- Pa_CloseStream(d_stream);
- d_stream = 0;
- d_reader.reset(); // boost::shared_ptr for d_reader = 0
- d_writer.reset(); // boost::shared_ptr for d_write = 0
- }
-
- d_input_parameters.channelCount = noutputs; // # of channels we're really using
-
-#if 1
- d_portaudio_buffer_size_frames = (int)(0.0213333333 * d_sampling_rate + 0.5); // Force 512 frame buffers at 48000
- fprintf(stderr, "Latency = %8.5f, requested sampling_rate = %g\n", // Force latency to 21.3333333.. ms
- 0.0213333333, (double)d_sampling_rate);
-#endif
- err = Pa_OpenStream(&d_stream,
- &d_input_parameters,
- NULL, // No output
- d_sampling_rate,
- d_portaudio_buffer_size_frames,
- paClipOff,
- &portaudio_source_callback,
- (void*)this);
-
- if (err != paNoError) {
- output_error_msg ("OpenStream failed", err);
- return false;
- }
-
-#if 0
- const PaStreamInfo *psi = Pa_GetStreamInfo(d_stream);
-
- d_portaudio_buffer_size_frames = (int)(d_input_parameters.suggestedLatency * psi->sampleRate);
- fprintf(stderr, "Latency = %7.4f, psi->sampleRate = %g\n",
- d_input_parameters.suggestedLatency, psi->sampleRate);
-#endif
-
- fprintf(stderr, "d_portaudio_buffer_size_frames = %d\n", d_portaudio_buffer_size_frames);
-
- assert(d_portaudio_buffer_size_frames != 0);
-
- create_ringbuffer();
-
- err = Pa_StartStream(d_stream);
- if (err != paNoError) {
- output_error_msg ("StartStream failed", err);
- return false;
- }
-
- return true;
-}
-
-audio_portaudio_source::~audio_portaudio_source ()
-{
- Pa_StopStream(d_stream); // wait for output to drain
- Pa_CloseStream(d_stream);
- Pa_Terminate();
-}
-
-int
-audio_portaudio_source::work (int noutput_items,
- gr_vector_const_void_star &input_items,
- gr_vector_void_star &output_items)
-{
- float **out = (float **) &output_items[0];
- const unsigned nchan = d_input_parameters.channelCount; // # of channels == samples/frame
-
- int k;
- for (k = 0; k < noutput_items; ){
-
- int nframes = d_reader->items_available() / nchan; // # of frames in ringbuffer
- if (nframes == 0){ // no data right now...
- if (k > 0) // If we've produced anything so far, return that
- return k;
-
- if (d_ok_to_block) {
- gruel:: scoped_lock guard(d_ringbuffer_mutex);
- while (d_ringbuffer_ready == false)
- d_ringbuffer_cond.wait(guard); // block here, then try again
- continue;
- }
-
- assert(k == 0);
-
- // There's no data and we're not allowed to block.
- // (A USRP is most likely controlling the pacing through the pipeline.)
- // This is an underun. The scheduler wouldn't have called us if it
- // had anything better to do. Thus we really need to produce some amount
- // of "fill".
- //
- // There are lots of options for comfort noise, etc.
- // FIXME We'll fill with zeros for now. Yes, it will "click"...
-
- // Fill with some frames of zeros
- {
- gruel::scoped_lock guard(d_ringbuffer_mutex);
-
- int nf = std::min(noutput_items - k, (int) d_portaudio_buffer_size_frames);
- for (int i = 0; i < nf; i++){
- for (unsigned int c = 0; c < nchan; c++){
- out[c][k + i] = 0;
- }
- }
- k += nf;
-
- d_ringbuffer_ready = false;
- return k;
- }
- }
-
- // We can read the smaller of the request and what's in the buffer.
- {
- gruel::scoped_lock guard(d_ringbuffer_mutex);
-
- int nf = std::min(noutput_items - k, nframes);
-
- const float *p = (const float *) d_reader->read_pointer();
- for (int i = 0; i < nf; i++){
- for (unsigned int c = 0; c < nchan; c++){
- out[c][k + i] = *p++;
- }
- }
- d_reader->update_read_pointer(nf * nchan);
- k += nf;
- d_ringbuffer_ready = false;
- }
- }
-
- return k; // tell how many we actually did
-}
-
-void
-audio_portaudio_source::output_error_msg (const char *msg, int err)
-{
- fprintf (stderr, "audio_portaudio_source[%s]: %s: %s\n",
- d_device_name.c_str (), msg, Pa_GetErrorText(err));
-}
-
-void
-audio_portaudio_source::bail (const char *msg, int err) throw (std::runtime_error)
-{
- output_error_msg (msg, err);
- throw std::runtime_error ("audio_portaudio_source");
-}
diff --git a/gr-audio/lib/portaudio/audio_portaudio_source.h b/gr-audio/lib/portaudio/audio_portaudio_source.h
deleted file mode 100644
index e81389a3b8..0000000000
--- a/gr-audio/lib/portaudio/audio_portaudio_source.h
+++ /dev/null
@@ -1,84 +0,0 @@
-/* -*- c++ -*- */
-/*
- * Copyright 2006-2011 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.
- */
-#ifndef INCLUDED_AUDIO_PORTAUDIO_SOURCE_H
-#define INCLUDED_AUDIO_PORTAUDIO_SOURCE_H
-
-#include <gr_audio_source.h>
-#include <gr_buffer.h>
-#include <gruel/thread.h>
-#include <string>
-#include <portaudio.h>
-#include <stdexcept>
-
-PaStreamCallback portaudio_source_callback;
-
-
-/*!
- * \brief Audio source using PORTAUDIO
- * \ingroup audio_blk
- *
- * Input samples must be in the range [-1,1].
- */
-class audio_portaudio_source : public audio_source {
-
- friend PaStreamCallback portaudio_source_callback;
-
-
- unsigned int d_sampling_rate;
- std::string d_device_name;
- bool d_ok_to_block;
- bool d_verbose;
-
- unsigned int d_portaudio_buffer_size_frames; // number of frames in a portaudio buffer
-
- PaStream *d_stream;
- PaStreamParameters d_input_parameters;
-
- gr_buffer_sptr d_writer; // buffer used between work and callback
- gr_buffer_reader_sptr d_reader;
-
- gruel::mutex d_ringbuffer_mutex;
- gruel::condition_variable d_ringbuffer_cond;
- bool d_ringbuffer_ready;
-
- // random stats
- int d_noverruns; // count of overruns
-
- void output_error_msg (const char *msg, int err);
- void bail (const char *msg, int err) throw (std::runtime_error);
- void create_ringbuffer();
-
-
-public:
- audio_portaudio_source (int sampling_rate, const std::string device_name,
- bool ok_to_block);
-
- ~audio_portaudio_source ();
-
- bool check_topology (int ninputs, int noutputs);
-
- int work (int ninput_items,
- gr_vector_const_void_star &input_items,
- gr_vector_void_star &output_items);
-};
-
-#endif /* INCLUDED_AUDIO_PORTAUDIO_SOURCE_H */
diff --git a/gr-audio/lib/portaudio/gri_portaudio.cc b/gr-audio/lib/portaudio/gri_portaudio.cc
deleted file mode 100644
index 66f3d46472..0000000000
--- a/gr-audio/lib/portaudio/gri_portaudio.cc
+++ /dev/null
@@ -1,111 +0,0 @@
-/* -*- c++ -*- */
-/*
- * Copyright 2006 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 <gri_portaudio.h>
-#include <portaudio.h>
-#include <string.h>
-
-
-PaDeviceIndex
-gri_pa_find_device_by_name(const char *name)
-{
- int i;
- int numDevices;
- const PaDeviceInfo *pdi;
- int len = strlen( name );
- PaDeviceIndex result = paNoDevice;
- numDevices = Pa_GetDeviceCount();
- for( i=0; i<numDevices; i++ )
- {
- pdi = Pa_GetDeviceInfo( i );
- if( strncmp( name, pdi->name, len ) == 0 )
- {
- result = i;
- break;
- }
- }
- return result;
-}
-
-
-void
-gri_print_devices()
-{
- int numDevices, defaultDisplayed, myDevice=0;
- const PaDeviceInfo *deviceInfo;
-
- numDevices = Pa_GetDeviceCount();
- if (numDevices < 0)
- return;
-
- printf("Number of devices found = %d\n", numDevices);
-
- for (int i=0; i < numDevices; i++ ) {
- deviceInfo = Pa_GetDeviceInfo( i );
- printf( "--------------------------------------- device #%d\n", i );
- /* Mark global and API specific default devices */
- defaultDisplayed = 0;
- if( i == Pa_GetDefaultInputDevice() )
- {
- myDevice = i;
- printf( "[ Default Input" );
- defaultDisplayed = 1;
- }
- else if( i == Pa_GetHostApiInfo( deviceInfo->hostApi )->defaultInputDevice )
- {
- const PaHostApiInfo *hostInfo = Pa_GetHostApiInfo( deviceInfo->hostApi );
- printf( "[ Default %s Input", hostInfo->name );
- defaultDisplayed = 1;
- }
-
- if( i == Pa_GetDefaultOutputDevice() )
- {
- printf( (defaultDisplayed ? "," : "[") );
- printf( " Default Output" );
- defaultDisplayed = 1;
- }
- else if( i == Pa_GetHostApiInfo( deviceInfo->hostApi )->defaultOutputDevice )
- {
- const PaHostApiInfo *hostInfo = Pa_GetHostApiInfo( deviceInfo->hostApi );
- printf( (defaultDisplayed ? "," : "[") );
- printf( " Default %s Output", hostInfo->name );
- defaultDisplayed = 1;
- }
- if( defaultDisplayed )
- printf( " ]\n" );
-
- /* print device info fields */
- printf( "Name = %s\n", deviceInfo->name );
- printf( "Host API = %s\n", Pa_GetHostApiInfo( deviceInfo->hostApi )->name );
- printf( "Max inputs = %d", deviceInfo->maxInputChannels );
- printf( ", Max outputs = %d\n", deviceInfo->maxOutputChannels );
-
- printf( "Default low input latency = %8.3f\n", deviceInfo->defaultLowInputLatency );
- printf( "Default low output latency = %8.3f\n", deviceInfo->defaultLowOutputLatency );
- printf( "Default high input latency = %8.3f\n", deviceInfo->defaultHighInputLatency );
- printf( "Default high output latency = %8.3f\n", deviceInfo->defaultHighOutputLatency );
- }
-}
diff --git a/gr-audio/lib/portaudio/portaudio_impl.cc b/gr-audio/lib/portaudio/portaudio_impl.cc
new file mode 100644
index 0000000000..ba37acc5fa
--- /dev/null
+++ b/gr-audio/lib/portaudio/portaudio_impl.cc
@@ -0,0 +1,108 @@
+/* -*- c++ -*- */
+/*
+ * Copyright 2006,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 <portaudio_impl.h>
+#include <portaudio.h>
+#include <string.h>
+
+namespace gr {
+ namespace audio {
+
+ PaDeviceIndex
+ pa_find_device_by_name(const char *name)
+ {
+ int i;
+ int numDevices;
+ const PaDeviceInfo *pdi;
+ int len = strlen(name);
+ PaDeviceIndex result = paNoDevice;
+ numDevices = Pa_GetDeviceCount();
+ for(i = 0; i < numDevices; i++) {
+ pdi = Pa_GetDeviceInfo(i);
+ if(strncmp(name, pdi->name, len) == 0) {
+ result = i;
+ break;
+ }
+ }
+ return result;
+ }
+
+ void
+ print_devices()
+ {
+ int numDevices, defaultDisplayed;
+ const PaDeviceInfo *deviceInfo;
+
+ numDevices = Pa_GetDeviceCount();
+ if(numDevices < 0)
+ return;
+
+ printf("Number of devices found = %d\n", numDevices);
+
+ for(int i = 0; i < numDevices; i++) {
+ deviceInfo = Pa_GetDeviceInfo(i);
+ printf("--------------------------------------- device #%d\n", i);
+ /* Mark global and API specific default devices */
+ defaultDisplayed = 0;
+ if(i == Pa_GetDefaultInputDevice()) {
+ printf("[ Default Input");
+ defaultDisplayed = 1;
+ }
+ else if(i == Pa_GetHostApiInfo(deviceInfo->hostApi)->defaultInputDevice) {
+ const PaHostApiInfo *hostInfo = Pa_GetHostApiInfo(deviceInfo->hostApi);
+ printf("[ Default %s Input", hostInfo->name);
+ defaultDisplayed = 1;
+ }
+
+ if(i == Pa_GetDefaultOutputDevice()) {
+ printf((defaultDisplayed ? "," : "["));
+ printf(" Default Output");
+ defaultDisplayed = 1;
+ }
+ else if(i == Pa_GetHostApiInfo(deviceInfo->hostApi)->defaultOutputDevice) {
+ const PaHostApiInfo *hostInfo = Pa_GetHostApiInfo(deviceInfo->hostApi);
+ printf((defaultDisplayed ? "," : "["));
+ printf(" Default %s Output", hostInfo->name);
+ defaultDisplayed = 1;
+ }
+ if(defaultDisplayed)
+ printf(" ]\n");
+
+ /* print device info fields */
+ printf("Name = %s\n", deviceInfo->name);
+ printf("Host API = %s\n", Pa_GetHostApiInfo(deviceInfo->hostApi)->name );
+ printf("Max inputs = %d", deviceInfo->maxInputChannels);
+ printf(", Max outputs = %d\n", deviceInfo->maxOutputChannels);
+
+ printf("Default low input latency = %8.3f\n", deviceInfo->defaultLowInputLatency);
+ printf("Default low output latency = %8.3f\n", deviceInfo->defaultLowOutputLatency);
+ printf("Default high input latency = %8.3f\n", deviceInfo->defaultHighInputLatency);
+ printf("Default high output latency = %8.3f\n", deviceInfo->defaultHighOutputLatency);
+ }
+ }
+
+ } /* namespace audio */
+} /* namespace gr */
diff --git a/gr-audio/lib/portaudio/gri_portaudio.h b/gr-audio/lib/portaudio/portaudio_impl.h
index c3ea7d064d..0cb099e591 100644
--- a/gr-audio/lib/portaudio/gri_portaudio.h
+++ b/gr-audio/lib/portaudio/portaudio_impl.h
@@ -1,6 +1,6 @@
/* -*- c++ -*- */
/*
- * Copyright 2006 Free Software Foundation, Inc.
+ * Copyright 2006,2013 Free Software Foundation, Inc.
*
* This file is part of GNU Radio
*
@@ -20,13 +20,19 @@
* Boston, MA 02110-1301, USA.
*/
-#ifndef INCLUDED_GRI_PORTAUDIO_H
-#define INCLUDED_GRI_PORTAUDIO_H
+#ifndef INCLUDED_AUDIO_PORTAUDIO_IMPL_H
+#define INCLUDED_AUDIO_PORTAUDIO_IMPL_H
#include <stdio.h>
#include <portaudio.h>
-PaDeviceIndex gri_pa_find_device_by_name(const char *name);
-void gri_print_devices();
+namespace gr {
+ namespace audio {
-#endif /* INCLUDED_GRI_PORTAUDIO_H */
+ PaDeviceIndex pa_find_device_by_name(const char *name);
+ void print_devices();
+
+ } /* namespace audio */
+} /* namespace gr */
+
+#endif /* INCLUDED_AUDIO_PORTAUDIO_IMPL_H */
diff --git a/gr-audio/lib/portaudio/portaudio_sink.cc b/gr-audio/lib/portaudio/portaudio_sink.cc
new file mode 100644
index 0000000000..d1a3f5166c
--- /dev/null
+++ b/gr-audio/lib/portaudio/portaudio_sink.cc
@@ -0,0 +1,370 @@
+/* -*- c++ -*- */
+/*
+ * Copyright 2006-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 he 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 <portaudio_sink.h>
+#include <portaudio_impl.h>
+#include <gr_io_signature.h>
+#include <gr_prefs.h>
+#include <stdio.h>
+#include <iostream>
+#include <unistd.h>
+#include <stdexcept>
+#include <string.h>
+
+namespace gr {
+ namespace audio {
+
+ AUDIO_REGISTER_SINK(REG_PRIO_MED, portaudio)(int sampling_rate,
+ const std::string &device_name,
+ bool ok_to_block)
+ {
+ return sink::sptr
+ (new portaudio_sink(sampling_rate, device_name, ok_to_block));
+ }
+
+//#define LOGGING 0 // define to 0 or 1
+
+#define SAMPLE_FORMAT paFloat32
+
+ typedef float sample_t;
+
+ // Number of portaudio buffers in the ringbuffer
+ static const unsigned int N_BUFFERS = 4;
+
+ static std::string
+ default_device_name()
+ {
+ return gr_prefs::singleton()->get_string
+ ("audio_portaudio", "default_output_device", "");
+ }
+
+ void
+ portaudio_sink::create_ringbuffer(void)
+ {
+ int bufsize_samples = d_portaudio_buffer_size_frames * d_output_parameters.channelCount;
+
+ if(d_verbose) {
+ fprintf(stderr,"ring buffer size = %d frames\n",
+ N_BUFFERS*bufsize_samples/d_output_parameters.channelCount);
+ }
+
+ // FYI, the buffer indicies are in units of samples.
+ d_writer = gr_make_buffer(N_BUFFERS * bufsize_samples, sizeof(sample_t));
+ d_reader = gr_buffer_add_reader(d_writer, 0);
+ }
+
+ /*
+ * This routine will be called by the PortAudio engine when audio is needed.
+ * It may called at interrupt level on some machines so don't do anything
+ * that could mess up the system like calling malloc() or free().
+ *
+ * Our job is to write framesPerBuffer frames into outputBuffer.
+ */
+ int
+ portaudio_sink_callback(const void *inputBuffer,
+ void *outputBuffer,
+ unsigned long framesPerBuffer,
+ const PaStreamCallbackTimeInfo* timeInfo,
+ PaStreamCallbackFlags statusFlags,
+ void *arg)
+ {
+ portaudio_sink *self = (portaudio_sink *)arg;
+ int nreqd_samples =
+ framesPerBuffer * self->d_output_parameters.channelCount;
+
+ int navail_samples = self->d_reader->items_available();
+
+ if(nreqd_samples <= navail_samples) { // We've got enough data...
+ {
+ gruel::scoped_lock guard(self->d_ringbuffer_mutex);
+
+ memcpy(outputBuffer,
+ self->d_reader->read_pointer(),
+ nreqd_samples * sizeof(sample_t));
+ self->d_reader->update_read_pointer(nreqd_samples);
+
+ self->d_ringbuffer_ready = true;
+ }
+
+ // Tell the sink thread there is new room in the ringbuffer.
+ self->d_ringbuffer_cond.notify_one();
+ return paContinue;
+ }
+
+ else { // underrun
+ self->d_nunderuns++;
+ ssize_t r = ::write(2, "aU", 2); // FIXME change to non-blocking call
+ if(r == -1) {
+ perror("audio_portaudio_source::portaudio_source_callback write error to stderr.");
+ }
+
+ // FIXME we should transfer what we've got and pad the rest
+ memset(outputBuffer, 0, nreqd_samples * sizeof(sample_t));
+
+ self->d_ringbuffer_ready = true;
+ self->d_ringbuffer_cond.notify_one(); // Tell the sink to get going!
+
+ return paContinue;
+ }
+ }
+
+ // ----------------------------------------------------------------
+
+ portaudio_sink::portaudio_sink(int sampling_rate,
+ const std::string device_name,
+ bool ok_to_block)
+ : gr_sync_block("audio_portaudio_sink",
+ gr_make_io_signature(0, 0, 0),
+ gr_make_io_signature(0, 0, 0)),
+ d_sampling_rate(sampling_rate),
+ d_device_name(device_name.empty() ? default_device_name() : device_name),
+ d_ok_to_block(ok_to_block),
+ d_verbose(gr_prefs::singleton()->get_bool("audio_portaudio", "verbose", false)),
+ d_portaudio_buffer_size_frames(0),
+ d_stream(0),
+ d_ringbuffer_mutex(),
+ d_ringbuffer_cond(),
+ d_ringbuffer_ready(false),
+ d_nunderuns(0)
+ {
+ memset(&d_output_parameters, 0, sizeof(d_output_parameters));
+ //if(LOGGING)
+ // d_log = gri_logger::singleton();
+
+ PaError err;
+ int i, numDevices;
+ PaDeviceIndex device = 0;
+ const PaDeviceInfo *deviceInfo = NULL;
+
+ err = Pa_Initialize();
+ if(err != paNoError) {
+ bail("Initialize failed", err);
+ }
+
+ if(d_verbose)
+ print_devices();
+
+ numDevices = Pa_GetDeviceCount();
+ if(numDevices < 0)
+ bail("Pa Device count failed", 0);
+ if(numDevices == 0)
+ bail("no devices available", 0);
+
+ if(d_device_name.empty()) {
+ // FIXME Get smarter about picking something
+ fprintf(stderr,"\nUsing Default Device\n");
+ device = Pa_GetDefaultOutputDevice();
+ deviceInfo = Pa_GetDeviceInfo(device);
+ fprintf(stderr,"%s is the chosen device using %s as the host\n",
+ deviceInfo->name, Pa_GetHostApiInfo(deviceInfo->hostApi)->name);
+ }
+ else {
+ bool found = false;
+ fprintf(stderr,"\nTest Devices\n");
+ for(i = 0; i < numDevices; i++) {
+ deviceInfo = Pa_GetDeviceInfo(i);
+ fprintf(stderr,"Testing device name: %s",deviceInfo->name);
+
+ if(deviceInfo->maxOutputChannels <= 0) {
+ fprintf(stderr,"\n");
+ continue;
+ }
+
+ if(strstr(deviceInfo->name, d_device_name.c_str())) {
+ fprintf(stderr," Chosen!\n");
+ device = i;
+ fprintf(stderr,"%s using %s as the host\n",d_device_name.c_str(),
+ Pa_GetHostApiInfo(deviceInfo->hostApi)->name), fflush(stderr);
+ found = true;
+ deviceInfo = Pa_GetDeviceInfo(device);
+ i = numDevices; // force loop exit
+ }
+ else
+ fprintf(stderr,"\n"), fflush(stderr);
+ }
+
+ if(!found) {
+ bail("Failed to find specified device name", 0);
+ exit(1);
+ }
+ }
+
+ d_output_parameters.device = device;
+ d_output_parameters.channelCount = deviceInfo->maxOutputChannels;
+ d_output_parameters.sampleFormat = SAMPLE_FORMAT;
+ d_output_parameters.suggestedLatency = deviceInfo->defaultLowOutputLatency;
+ d_output_parameters.hostApiSpecificStreamInfo = NULL;
+
+ // We fill in the real channelCount in check_topology when we know
+ // how many inputs are connected to us.
+
+ // Now that we know the maximum number of channels (allegedly)
+ // supported by the h/w, we can compute a reasonable input
+ // signature. The portaudio specs say that they'll accept any
+ // number of channels from 1 to max.
+ set_input_signature(gr_make_io_signature(1, deviceInfo->maxOutputChannels,
+ sizeof(sample_t)));
+ }
+
+ bool
+ portaudio_sink::check_topology(int ninputs, int noutputs)
+ {
+ PaError err;
+
+ if(Pa_IsStreamActive(d_stream)) {
+ Pa_CloseStream(d_stream);
+ d_stream = 0;
+ d_reader.reset(); // boost::shared_ptr for d_reader = 0
+ d_writer.reset(); // boost::shared_ptr for d_write = 0
+ }
+
+ d_output_parameters.channelCount = ninputs; // # of channels we're really using
+
+#if 1
+ d_portaudio_buffer_size_frames = (int)(0.0213333333 * d_sampling_rate + 0.5); // Force 1024 frame buffers at 48000
+ fprintf(stderr, "Latency = %8.5f, requested sampling_rate = %g\n", // Force latency to 21.3333333.. ms
+ 0.0213333333, (double)d_sampling_rate);
+#endif
+ err = Pa_OpenStream(&d_stream,
+ NULL, // No input
+ &d_output_parameters,
+ d_sampling_rate,
+ d_portaudio_buffer_size_frames,
+ paClipOff,
+ &portaudio_sink_callback,
+ (void*)this);
+
+ if(err != paNoError) {
+ output_error_msg("OpenStream failed", err);
+ return false;
+ }
+
+#if 0
+ const PaStreamInfo *psi = Pa_GetStreamInfo(d_stream);
+
+ d_portaudio_buffer_size_frames = (int)(d_output_parameters.suggestedLatency * psi->sampleRate);
+ fprintf(stderr, "Latency = %7.4f, psi->sampleRate = %g\n",
+ d_output_parameters.suggestedLatency, psi->sampleRate);
+#endif
+
+ fprintf(stderr, "d_portaudio_buffer_size_frames = %d\n",
+ d_portaudio_buffer_size_frames);
+
+ assert(d_portaudio_buffer_size_frames != 0);
+
+ create_ringbuffer();
+
+ err = Pa_StartStream(d_stream);
+ if(err != paNoError) {
+ output_error_msg("StartStream failed", err);
+ return false;
+ }
+
+ return true;
+ }
+
+ portaudio_sink::~portaudio_sink()
+ {
+ Pa_StopStream(d_stream); // wait for output to drain
+ Pa_CloseStream(d_stream);
+ Pa_Terminate();
+ }
+
+ /*
+ * This version consumes everything sent to it, blocking if required.
+ * I think this will allow us better control of the total buffering/latency
+ * in the audio path.
+ */
+ int
+ portaudio_sink::work(int noutput_items,
+ gr_vector_const_void_star &input_items,
+ gr_vector_void_star &output_items)
+ {
+ const float **in = (const float **)&input_items[0];
+ const unsigned nchan = d_output_parameters.channelCount; // # of channels == samples/frame
+
+ int k;
+ for(k = 0; k < noutput_items;) {
+ int nframes = d_writer->space_available() / nchan; // How much space in ringbuffer
+ if(nframes == 0) { // no room...
+ if(d_ok_to_block) {
+ {
+ gruel::scoped_lock guard(d_ringbuffer_mutex);
+ while(!d_ringbuffer_ready)
+ d_ringbuffer_cond.wait(guard);
+ }
+ continue;
+ }
+ else {
+ // There's no room and we're not allowed to block.
+ // (A USRP is most likely controlling the pacing through the pipeline.)
+ // We drop the samples on the ground, and say we processed them all ;)
+ //
+ // FIXME, there's probably room for a bit more finesse here.
+ return noutput_items;
+ }
+ }
+
+ // We can write the smaller of the request and the room we've got
+ {
+ gruel::scoped_lock guard(d_ringbuffer_mutex);
+
+ int nf = std::min(noutput_items - k, nframes);
+ float *p = (float*)d_writer->write_pointer();
+
+ for(int i = 0; i < nf; i++) {
+ for(unsigned int c = 0; c < nchan; c++) {
+ *p++ = in[c][k + i];
+ }
+ }
+
+ d_writer->update_write_pointer(nf * nchan);
+ k += nf;
+
+ d_ringbuffer_ready = false;
+ }
+ }
+
+ return k; // tell how many we actually did
+ }
+
+ void
+ portaudio_sink::output_error_msg(const char *msg, int err)
+ {
+ fprintf(stderr, "audio_portaudio_sink[%s]: %s: %s\n",
+ d_device_name.c_str(), msg, Pa_GetErrorText(err));
+ }
+
+ void
+ portaudio_sink::bail(const char *msg, int err) throw (std::runtime_error)
+ {
+ output_error_msg(msg, err);
+ throw std::runtime_error("audio_portaudio_sink");
+ }
+
+ } /* namespace audio */
+} /* namespace gr */
diff --git a/gr-audio/lib/portaudio/portaudio_sink.h b/gr-audio/lib/portaudio/portaudio_sink.h
new file mode 100644
index 0000000000..41a725b691
--- /dev/null
+++ b/gr-audio/lib/portaudio/portaudio_sink.h
@@ -0,0 +1,90 @@
+/* -*- c++ -*- */
+/*
+ * Copyright 2006-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.
+ */
+#ifndef INCLUDED_AUDIO_PORTAUDIO_SINK_H
+#define INCLUDED_AUDIO_PORTAUDIO_SINK_H
+
+#include <audio/sink.h>
+#include <gr_buffer.h>
+#include <gruel/thread.h>
+#include <string>
+#include <portaudio.h>
+#include <stdexcept>
+//#include <gri_logger.h>
+
+namespace gr {
+ namespace audio {
+
+ PaStreamCallback portaudio_sink_callback;
+
+ /*!
+ * \brief Audio sink using PORTAUDIO
+ * \ingroup audio_blk
+ *
+ * Input samples must be in the range [-1,1].
+ */
+ class portaudio_sink : public sink
+ {
+ friend PaStreamCallback portaudio_sink_callback;
+
+ unsigned int d_sampling_rate;
+ std::string d_device_name;
+ bool d_ok_to_block;
+ bool d_verbose;
+
+ unsigned int d_portaudio_buffer_size_frames; // number of frames in a portaudio buffer
+
+ PaStream *d_stream;
+ PaStreamParameters d_output_parameters;
+
+ gr_buffer_sptr d_writer; // buffer used between work and callback
+ gr_buffer_reader_sptr d_reader;
+
+ gruel::mutex d_ringbuffer_mutex;
+ gruel::condition_variable d_ringbuffer_cond;
+ bool d_ringbuffer_ready;
+
+ // random stats
+ int d_nunderuns; // count of underruns
+ //gri_logger_sptr d_log; // handle to non-blocking logging instance
+
+ void output_error_msg(const char *msg, int err);
+ void bail(const char *msg, int err) throw (std::runtime_error);
+ void create_ringbuffer();
+
+ public:
+ portaudio_sink(int sampling_rate,
+ const std::string device_name,
+ bool ok_to_block);
+
+ ~portaudio_sink();
+
+ bool check_topology(int ninputs, int noutputs);
+
+ int work(int noutput_items,
+ gr_vector_const_void_star &input_items,
+ gr_vector_void_star &output_items);
+ };
+
+ } /* namespace audio */
+} /* namespace gr */
+
+#endif /* INCLUDED_AUDIO_PORTAUDIO_SINK_H */
diff --git a/gr-audio/lib/portaudio/portaudio_source.cc b/gr-audio/lib/portaudio/portaudio_source.cc
new file mode 100644
index 0000000000..937c1d0dbf
--- /dev/null
+++ b/gr-audio/lib/portaudio/portaudio_source.cc
@@ -0,0 +1,378 @@
+/* -*- c++ -*- */
+/*
+ * Copyright 2006-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 he 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 <portaudio_source.h>
+#include <portaudio_impl.h>
+#include <gr_io_signature.h>
+#include <gr_prefs.h>
+#include <stdio.h>
+#include <iostream>
+#include <unistd.h>
+#include <stdexcept>
+#include <string.h>
+
+namespace gr {
+ namespace audio {
+
+ AUDIO_REGISTER_SOURCE(REG_PRIO_MED, portaudio)(int sampling_rate,
+ const std::string &device_name,
+ bool ok_to_block)
+ {
+ return source::sptr
+ (new portaudio_source(sampling_rate, device_name, ok_to_block));
+ }
+
+//#define LOGGING 0 // define to 0 or 1
+
+#define SAMPLE_FORMAT paFloat32
+
+ typedef float sample_t;
+
+ // Number of portaudio buffers in the ringbuffer
+ static const unsigned int N_BUFFERS = 4;
+
+ static std::string
+ default_device_name()
+ {
+ return gr_prefs::singleton()->get_string
+ ("audio_portaudio", "default_input_device", "");
+ }
+
+ void
+ portaudio_source::create_ringbuffer(void)
+ {
+ int bufsize_samples = d_portaudio_buffer_size_frames * d_input_parameters.channelCount;
+
+ if(d_verbose) {
+ fprintf(stderr, "ring buffer size = %d frames\n",
+ N_BUFFERS*bufsize_samples/d_input_parameters.channelCount);
+ }
+
+ // FYI, the buffer indicies are in units of samples.
+ d_writer = gr_make_buffer(N_BUFFERS * bufsize_samples, sizeof(sample_t));
+ d_reader = gr_buffer_add_reader(d_writer, 0);
+ }
+
+ /*
+ * This routine will be called by the PortAudio engine when audio is needed.
+ * It may called at interrupt level on some machines so don't do anything
+ * that could mess up the system like calling malloc() or free().
+ *
+ * Our job is to copy framesPerBuffer frames from inputBuffer.
+ */
+ int
+ portaudio_source_callback(const void *inputBuffer,
+ void *outputBuffer,
+ unsigned long framesPerBuffer,
+ const PaStreamCallbackTimeInfo* timeInfo,
+ PaStreamCallbackFlags statusFlags,
+ void *arg)
+ {
+ portaudio_source *self = (portaudio_source *)arg;
+ int nchan = self->d_input_parameters.channelCount;
+ int nframes_to_copy = framesPerBuffer;
+ int nframes_room = self->d_writer->space_available() / nchan;
+
+ if(nframes_to_copy <= nframes_room) { // We've got room for the data ..
+ //if (LOGGING)
+ // self->d_log->printf("PAsrc cb: f/b = %4ld\n", framesPerBuffer);
+
+ // copy from input buffer to ringbuffer
+ {
+ gruel::scoped_lock(d_ringbuffer_mutex);
+
+ memcpy(self->d_writer->write_pointer(),
+ inputBuffer,
+ nframes_to_copy * nchan * sizeof(sample_t));
+ self->d_writer->update_write_pointer(nframes_to_copy * nchan);
+
+ // Tell the source thread there is new data in the ringbuffer.
+ self->d_ringbuffer_ready = true;
+ }
+
+ self->d_ringbuffer_cond.notify_one();
+ return paContinue;
+ }
+
+ else { // overrun
+ self->d_noverruns++;
+ ssize_t r = ::write(2, "aO", 2); // FIXME change to non-blocking call
+ if(r == -1) {
+ perror("audio_portaudio_source::portaudio_source_callback write error to stderr.");
+ }
+
+ self->d_ringbuffer_ready = false;
+ self->d_ringbuffer_cond.notify_one(); // Tell the sink to get going!
+ return paContinue;
+ }
+ }
+
+ // ----------------------------------------------------------------
+
+ portaudio_source::portaudio_source(int sampling_rate,
+ const std::string device_name,
+ bool ok_to_block)
+ : gr_sync_block("audio_portaudio_source",
+ gr_make_io_signature(0, 0, 0),
+ gr_make_io_signature(0, 0, 0)),
+ d_sampling_rate(sampling_rate),
+ d_device_name(device_name.empty() ? default_device_name() : device_name),
+ d_ok_to_block(ok_to_block),
+ d_verbose(gr_prefs::singleton()->get_bool("audio_portaudio", "verbose", false)),
+ d_portaudio_buffer_size_frames(0),
+ d_stream(0),
+ d_ringbuffer_mutex(),
+ d_ringbuffer_cond(),
+ d_ringbuffer_ready(false),
+ d_noverruns(0)
+ {
+ memset(&d_input_parameters, 0, sizeof(d_input_parameters));
+ //if(LOGGING)
+ // d_log = gri_logger::singleton();
+
+ PaError err;
+ int i, numDevices;
+ PaDeviceIndex device = 0;
+ const PaDeviceInfo *deviceInfo = NULL;
+
+ err = Pa_Initialize();
+ if(err != paNoError) {
+ bail("Initialize failed", err);
+ }
+
+ if(d_verbose)
+ print_devices();
+
+ numDevices = Pa_GetDeviceCount();
+ if(numDevices < 0)
+ bail("Pa Device count failed", 0);
+ if(numDevices == 0)
+ bail("no devices available", 0);
+
+ if(d_device_name.empty()) {
+ // FIXME Get smarter about picking something
+ device = Pa_GetDefaultInputDevice();
+ deviceInfo = Pa_GetDeviceInfo(device);
+ fprintf(stderr,"%s is the chosen device using %s as the host\n",
+ deviceInfo->name, Pa_GetHostApiInfo(deviceInfo->hostApi)->name);
+ }
+ else {
+ bool found = false;
+
+ for(i = 0; i < numDevices; i++) {
+ deviceInfo = Pa_GetDeviceInfo(i);
+ fprintf(stderr,"Testing device name: %s",deviceInfo->name);
+ if(deviceInfo->maxInputChannels <= 0) {
+ fprintf(stderr,"\n");
+ continue;
+ }
+ if(strstr(deviceInfo->name, d_device_name.c_str())) {
+ fprintf(stderr," Chosen!\n");
+ device = i;
+ fprintf(stderr,"%s using %s as the host\n",d_device_name.c_str(),
+ Pa_GetHostApiInfo(deviceInfo->hostApi)->name), fflush(stderr);
+ found = true;
+ deviceInfo = Pa_GetDeviceInfo(device);
+ i = numDevices; // force loop exit
+ }
+ else
+ fprintf(stderr,"\n"),fflush(stderr);
+ }
+
+ if(!found) {
+ bail("Failed to find specified device name", 0);
+ }
+ }
+
+ d_input_parameters.device = device;
+ d_input_parameters.channelCount = deviceInfo->maxInputChannels;
+ d_input_parameters.sampleFormat = SAMPLE_FORMAT;
+ d_input_parameters.suggestedLatency = deviceInfo->defaultLowInputLatency;
+ d_input_parameters.hostApiSpecificStreamInfo = NULL;
+
+ // We fill in the real channelCount in check_topology when we know
+ // how many inputs are connected to us.
+
+ // Now that we know the maximum number of channels (allegedly)
+ // supported by the h/w, we can compute a reasonable output
+ // signature. The portaudio specs say that they'll accept any
+ // number of channels from 1 to max.
+ set_output_signature(gr_make_io_signature(1, deviceInfo->maxInputChannels,
+ sizeof (sample_t)));
+ }
+
+ bool
+ portaudio_source::check_topology(int ninputs, int noutputs)
+ {
+ PaError err;
+
+ if(Pa_IsStreamActive(d_stream)) {
+ Pa_CloseStream(d_stream);
+ d_stream = 0;
+ d_reader.reset(); // boost::shared_ptr for d_reader = 0
+ d_writer.reset(); // boost::shared_ptr for d_write = 0
+ }
+
+ d_input_parameters.channelCount = noutputs; // # of channels we're really using
+
+#if 1
+ d_portaudio_buffer_size_frames = (int)(0.0213333333 * d_sampling_rate + 0.5); // Force 512 frame buffers at 48000
+ fprintf(stderr, "Latency = %8.5f, requested sampling_rate = %g\n", // Force latency to 21.3333333.. ms
+ 0.0213333333, (double)d_sampling_rate);
+#endif
+ err = Pa_OpenStream(&d_stream,
+ &d_input_parameters,
+ NULL, // No output
+ d_sampling_rate,
+ d_portaudio_buffer_size_frames,
+ paClipOff,
+ &portaudio_source_callback,
+ (void*)this);
+
+ if(err != paNoError) {
+ output_error_msg("OpenStream failed", err);
+ return false;
+ }
+
+#if 0
+ const PaStreamInfo *psi = Pa_GetStreamInfo(d_stream);
+
+ d_portaudio_buffer_size_frames = (int)(d_input_parameters.suggestedLatency * psi->sampleRate);
+ fprintf(stderr, "Latency = %7.4f, psi->sampleRate = %g\n",
+ d_input_parameters.suggestedLatency, psi->sampleRate);
+#endif
+
+ fprintf(stderr, "d_portaudio_buffer_size_frames = %d\n",
+ d_portaudio_buffer_size_frames);
+
+ assert(d_portaudio_buffer_size_frames != 0);
+
+ create_ringbuffer();
+
+ err = Pa_StartStream(d_stream);
+ if(err != paNoError) {
+ output_error_msg("StartStream failed", err);
+ return false;
+ }
+
+ return true;
+ }
+
+ portaudio_source::~portaudio_source()
+ {
+ Pa_StopStream(d_stream); // wait for output to drain
+ Pa_CloseStream(d_stream);
+ Pa_Terminate();
+ }
+
+ int
+ portaudio_source::work(int noutput_items,
+ gr_vector_const_void_star &input_items,
+ gr_vector_void_star &output_items)
+ {
+ float **out = (float **)&output_items[0];
+ const unsigned nchan = d_input_parameters.channelCount; // # of channels == samples/frame
+
+ int k;
+ for(k = 0; k < noutput_items;) {
+ int nframes = d_reader->items_available() / nchan; // # of frames in ringbuffer
+ if(nframes == 0) { // no data right now...
+ if(k > 0) // If we've produced anything so far, return that
+ return k;
+
+ if(d_ok_to_block) {
+ gruel::scoped_lock guard(d_ringbuffer_mutex);
+ while(d_ringbuffer_ready == false)
+ d_ringbuffer_cond.wait(guard); // block here, then try again
+ continue;
+ }
+
+ assert(k == 0);
+
+ // There's no data and we're not allowed to block.
+ // (A USRP is most likely controlling the pacing through the pipeline.)
+ // This is an underun. The scheduler wouldn't have called us if it
+ // had anything better to do. Thus we really need to produce some amount
+ // of "fill".
+ //
+ // There are lots of options for comfort noise, etc.
+ // FIXME We'll fill with zeros for now. Yes, it will "click"...
+
+ // Fill with some frames of zeros
+ {
+ gruel::scoped_lock guard(d_ringbuffer_mutex);
+
+ int nf = std::min(noutput_items - k, (int)d_portaudio_buffer_size_frames);
+ for(int i = 0; i < nf; i++) {
+ for(unsigned int c = 0; c < nchan; c++) {
+ out[c][k + i] = 0;
+ }
+ }
+ k += nf;
+
+ d_ringbuffer_ready = false;
+ return k;
+ }
+ }
+
+ // We can read the smaller of the request and what's in the buffer.
+ {
+ gruel::scoped_lock guard(d_ringbuffer_mutex);
+
+ int nf = std::min(noutput_items - k, nframes);
+
+ const float *p = (const float*)d_reader->read_pointer();
+ for(int i = 0; i < nf; i++) {
+ for(unsigned int c = 0; c < nchan; c++) {
+ out[c][k + i] = *p++;
+ }
+ }
+ d_reader->update_read_pointer(nf * nchan);
+ k += nf;
+ d_ringbuffer_ready = false;
+ }
+ }
+
+ return k; // tell how many we actually did
+ }
+
+ void
+ portaudio_source::output_error_msg(const char *msg, int err)
+ {
+ fprintf(stderr, "audio_portaudio_source[%s]: %s: %s\n",
+ d_device_name.c_str (), msg, Pa_GetErrorText(err));
+ }
+
+ void
+ portaudio_source::bail(const char *msg, int err) throw (std::runtime_error)
+ {
+ output_error_msg(msg, err);
+ throw std::runtime_error("audio_portaudio_source");
+ }
+
+ } /* namespace audio */
+} /* namespace gr */
diff --git a/gr-audio/lib/portaudio/portaudio_source.h b/gr-audio/lib/portaudio/portaudio_source.h
new file mode 100644
index 0000000000..d4f4f01d1f
--- /dev/null
+++ b/gr-audio/lib/portaudio/portaudio_source.h
@@ -0,0 +1,89 @@
+/* -*- c++ -*- */
+/*
+ * Copyright 2006-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.
+ */
+
+#ifndef INCLUDED_AUDIO_PORTAUDIO_SOURCE_H
+#define INCLUDED_AUDIO_PORTAUDIO_SOURCE_H
+
+#include <audio/source.h>
+#include <gr_buffer.h>
+#include <gruel/thread.h>
+#include <string>
+#include <portaudio.h>
+#include <stdexcept>
+
+namespace gr {
+ namespace audio {
+
+ PaStreamCallback portaudio_source_callback;
+
+ /*!
+ * \brief Audio source using PORTAUDIO
+ * \ingroup audio_blk
+ *
+ * Input samples must be in the range [-1,1].
+ */
+ class portaudio_source : public source
+ {
+ friend PaStreamCallback portaudio_source_callback;
+
+ unsigned int d_sampling_rate;
+ std::string d_device_name;
+ bool d_ok_to_block;
+ bool d_verbose;
+
+ unsigned int d_portaudio_buffer_size_frames; // number of frames in a portaudio buffer
+
+ PaStream *d_stream;
+ PaStreamParameters d_input_parameters;
+
+ gr_buffer_sptr d_writer; // buffer used between work and callback
+ gr_buffer_reader_sptr d_reader;
+
+ gruel::mutex d_ringbuffer_mutex;
+ gruel::condition_variable d_ringbuffer_cond;
+ bool d_ringbuffer_ready;
+
+ // random stats
+ int d_noverruns; // count of overruns
+
+ void output_error_msg(const char *msg, int err);
+ void bail(const char *msg, int err) throw (std::runtime_error);
+ void create_ringbuffer();
+
+ public:
+ portaudio_source(int sampling_rate,
+ const std::string device_name,
+ bool ok_to_block);
+
+ ~portaudio_source();
+
+ bool check_topology(int ninputs, int noutputs);
+
+ int work(int ninput_items,
+ gr_vector_const_void_star &input_items,
+ gr_vector_void_star &output_items);
+ };
+
+ } /* namespace audio */
+} /* namespace gr */
+
+#endif /* INCLUDED_AUDIO_PORTAUDIO_SOURCE_H */
diff --git a/gr-audio/lib/windows/audio_windows_sink.cc b/gr-audio/lib/windows/audio_windows_sink.cc
deleted file mode 100644
index 5284ce1734..0000000000
--- a/gr-audio/lib/windows/audio_windows_sink.cc
+++ /dev/null
@@ -1,323 +0,0 @@
-/* -*- c++ -*- */
-/*
-* Copyright 2004-2011 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 "gr_audio_registry.h"
-#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>
-
-AUDIO_REGISTER_SINK(REG_PRIO_HIGH, windows)(
- int sampling_rate, const std::string &device_name, bool
-){
- return audio_sink::sptr(new audio_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";
-}
-
-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;
-}
-
-int
-audio_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
-audio_windows_sink::string_to_int (const std::string & s)
-{
- int i;
- std::istringstream (s) >> i;
- return i;
-} //ToInt()
-
-int
-audio_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
-audio_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;
-}
diff --git a/gr-audio/lib/windows/audio_windows_sink.h b/gr-audio/lib/windows/audio_windows_sink.h
deleted file mode 100644
index d4ca259b3b..0000000000
--- a/gr-audio/lib/windows/audio_windows_sink.h
+++ /dev/null
@@ -1,73 +0,0 @@
-/* -*- c++ -*- */
-/*
- * Copyright 2004-2011 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.
- */
-
-#ifndef INCLUDED_AUDIO_WINDOWS_SINK_H
-#define INCLUDED_AUDIO_WINDOWS_SINK_H
-
-#define WIN32_LEAN_AND_MEAN
-#define NOMINMAX // stops windef.h defining max/min under cygwin
-
-#include <windows.h>
-#include <mmsystem.h>
-
-#include <gr_audio_sink.h>
-#include <string>
-
-/*!
- * \brief audio sink using winmm mmsystem (win32 only)
- * \ingroup audio_blk
- *
- * input signature is one or two streams of floats.
- * Input samples must be in the range [-1,1].
- */
-
-class audio_windows_sink : public audio_sink
-{
- int d_sampling_freq;
- std::string d_device_name;
- int d_fd;
- short *d_buffer;
- int d_chunk_size;
- HWAVEOUT d_h_waveout;
- HGLOBAL d_h_wave_hdr;
- LPWAVEHDR d_lp_wave_hdr;
- HANDLE d_wave_write_event;
-
-protected:
- int
- string_to_int (const std::string & s);
- int
- open_waveout_device (void);
- int
- write_waveout (HPSTR lp_data, DWORD dw_data_size);
-
-public:
- audio_windows_sink (int sampling_freq, const std::string device_name = "");
- ~audio_windows_sink ();
-
- int
- work (int noutput_items,
- gr_vector_const_void_star & input_items,
- gr_vector_void_star & output_items);
-};
-
-#endif /* INCLUDED_AUDIO_WINDOWS_SINK_H */
diff --git a/gr-audio/lib/windows/audio_windows_source.cc b/gr-audio/lib/windows/audio_windows_source.cc
deleted file mode 100644
index 75b0a33bbc..0000000000
--- a/gr-audio/lib/windows/audio_windows_source.cc
+++ /dev/null
@@ -1,205 +0,0 @@
-/* -*- c++ -*- */
-/*
- * Copyright 2004-2011 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 "gr_audio_registry.h"
-#include <audio_windows_source.h>
-#include <gr_io_signature.h>
-//include <sys/soundcard.h>
-//include <sys/ioctl.h>
-#include <sys/types.h>
-#include <sys/stat.h>
-#include <fcntl.h>
-#include <unistd.h>
-#include <stdio.h>
-#include <iostream>
-#include <stdexcept>
-
-AUDIO_REGISTER_SOURCE(REG_PRIO_HIGH, windows)(
- int sampling_rate, const std::string &device_name, bool
-){
- return audio_source::sptr(new audio_windows_source(sampling_rate, device_name));
-}
-
-static const double CHUNK_TIME = 0.005; // 5 ms
-
-// FIXME these should query some kind of user preference
-
-static std::string
-default_device_name ()
-{
- return "/dev/dsp";
-}
-
-audio_windows_source::audio_windows_source (int sampling_freq, const std::string device_name)
- : gr_sync_block ("audio_windows_source",
- gr_make_io_signature (0, 0, 0),
- gr_make_io_signature (1, 2, sizeof (float))),
- 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)
-{
- //FIXME TODO implement me
-#if 0
- if ((d_fd = open (d_device_name.c_str (), O_RDONLY)) < 0)
- {
- fprintf (stderr, "audio_windows_source: ");
- perror (d_device_name.c_str ());
- throw
- std::runtime_error ("audio_windows_source");
- }
-
- d_chunk_size = (int) (d_sampling_freq * CHUNK_TIME);
- set_output_multiple (d_chunk_size);
-
- d_buffer = new short[d_chunk_size * 2];
-
- int format = AFMT_S16_NE;
- int orig_format = format;
- if (ioctl (d_fd, SNDCTL_DSP_SETFMT, &format) < 0)
- {
- std::
- cerr << "audio_windows_source: " << d_device_name <<
- " ioctl failed\n";
- perror (d_device_name.c_str ());
- throw
- std::runtime_error ("audio_windows_source");
- }
-
- if (format != orig_format)
- {
- fprintf (stderr, "audio_windows_source: unable to support format %d\n",
- orig_format);
- fprintf (stderr, " card requested %d instead.\n", format);
- }
-
- // set to stereo no matter what. Some hardware only does stereo
- int channels = 2;
- if (ioctl (d_fd, SNDCTL_DSP_CHANNELS, &channels) < 0 || channels != 2)
- {
- perror ("audio_windows_source: could not set STEREO mode");
- throw
- std::runtime_error ("audio_windows_source");
- }
-
- // set sampling freq
- int sf = sampling_freq;
- if (ioctl (d_fd, SNDCTL_DSP_SPEED, &sf) < 0)
- {
- std::cerr << "audio_windows_source: "
- << d_device_name << ": invalid sampling_freq "
- << sampling_freq << "\n";
- sampling_freq = 8000;
- if (ioctl (d_fd, SNDCTL_DSP_SPEED, &sf) < 0)
- {
- std::
- cerr <<
- "audio_windows_source: failed to set sampling_freq to 8000\n";
- throw
- std::runtime_error ("audio_windows_source");
- }
- }
-#endif
-}
-
-audio_windows_source::~audio_windows_source ()
-{
- /*close (d_fd);
- delete [] d_buffer;
- */
-}
-
-int
-audio_windows_source::work (int noutput_items,
- gr_vector_const_void_star & input_items,
- gr_vector_void_star & output_items)
-{
- //FIXME TODO implement me
-#if 0
- float *f0 = (float *) output_items[0];
- float *f1 = (float *) output_items[1]; // will be invalid if this is mono output
-
- const int shorts_per_item = 2; // L + R
- const int bytes_per_item = shorts_per_item * sizeof (short);
-
- // To minimize latency, never return more than CHUNK_TIME
- // worth of samples per call to work.
- // FIXME, we need an API to set this value
-
- noutput_items = std::min (noutput_items, d_chunk_size);
-
- int base = 0;
- int ntogo = noutput_items;
-
- while (ntogo > 0)
- {
- int nbytes = std::min (ntogo, d_chunk_size) * bytes_per_item;
- int result_nbytes = read (d_fd, d_buffer, nbytes);
-
- if (result_nbytes < 0)
- {
- perror ("audio_windows_source");
- return -1; // say we're done
- }
-
- if ((result_nbytes & (bytes_per_item - 1)) != 0)
- {
- fprintf (stderr, "audio_windows_source: internal error.\n");
- throw std::runtime_error ("internal error");
- }
-
- int result_nitems = result_nbytes / bytes_per_item;
-
- // now unpack samples into output streams
-
- switch (output_items.size ())
- {
- case 1: // mono output
- for (int i = 0; i < result_nitems; i++)
- {
- f0[base + i] = d_buffer[2 * i + 0] * (1.0 / 32767);
- }
- break;
-
- case 2: // stereo output
- for (int i = 0; i < result_nitems; i++)
- {
- f0[base + i] = d_buffer[2 * i + 0] * (1.0 / 32767);
- f1[base + i] = d_buffer[2 * i + 1] * (1.0 / 32767);
- }
- break;
-
- default:
- assert (0);
- }
-
- ntogo -= result_nitems;
- base += result_nitems;
- }
-
- return noutput_items - ntogo;
-#endif
- return -1; // EOF
-}
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 */
diff --git a/gr-audio/lib/windows/windows_sink.h b/gr-audio/lib/windows/windows_sink.h
new file mode 100644
index 0000000000..beaccc9054
--- /dev/null
+++ b/gr-audio/lib/windows/windows_sink.h
@@ -0,0 +1,75 @@
+/* -*- 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.
+ */
+
+#ifndef INCLUDED_AUDIO_WINDOWS_SINK_H
+#define INCLUDED_AUDIO_WINDOWS_SINK_H
+
+#define WIN32_LEAN_AND_MEAN
+#define NOMINMAX // stops windef.h defining max/min under cygwin
+
+#include <windows.h>
+#include <mmsystem.h>
+
+#include <audio/sink.h>
+#include <string>
+
+namespace gr {
+ namespace audio {
+
+ /*!
+ * \brief audio sink using winmm mmsystem (win32 only)
+ * \ingroup audio_blk
+ *
+ * input signature is one or two streams of floats.
+ * Input samples must be in the range [-1,1].
+ */
+ class windows_sink : public sink
+ {
+ int d_sampling_freq;
+ std::string d_device_name;
+ int d_fd;
+ short *d_buffer;
+ int d_chunk_size;
+ HWAVEOUT d_h_waveout;
+ HGLOBAL d_h_wave_hdr;
+ LPWAVEHDR d_lp_wave_hdr;
+ HANDLE d_wave_write_event;
+
+ protected:
+ int string_to_int(const std::string & s);
+ int open_waveout_device(void);
+ int write_waveout(HPSTR lp_data, DWORD dw_data_size);
+
+ public:
+ windows_sink(int sampling_freq,
+ const std::string device_name = "");
+ ~windows_sink();
+
+ int work(int noutput_items,
+ gr_vector_const_void_star & input_items,
+ gr_vector_void_star & output_items);
+ };
+
+ } /* namespace audio */
+} /* namespace gr */
+
+#endif /* INCLUDED_AUDIO_WINDOWS_SINK_H */
diff --git a/gr-audio/lib/windows/windows_source.cc b/gr-audio/lib/windows/windows_source.cc
new file mode 100644
index 0000000000..89371cfa84
--- /dev/null
+++ b/gr-audio/lib/windows/windows_source.cc
@@ -0,0 +1,194 @@
+/* -*- 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_source.h>
+#include <gr_io_signature.h>
+//include <sys/soundcard.h>
+//include <sys/ioctl.h>
+#include <sys/types.h>
+#include <sys/stat.h>
+#include <fcntl.h>
+#include <unistd.h>
+#include <stdio.h>
+#include <iostream>
+#include <stdexcept>
+
+namespace gr {
+ namespace audio {
+
+ AUDIO_REGISTER_SOURCE(REG_PRIO_HIGH, windows)(int sampling_rate,
+ const std::string &device_name,
+ bool)
+ {
+ return source::sptr
+ (new windows_source(sampling_rate, device_name));
+ }
+
+ static const double CHUNK_TIME = 0.005; // 5 ms
+
+ // FIXME these should query some kind of user preference
+
+ static std::string
+ default_device_name()
+ {
+ return "/dev/dsp";
+ }
+
+ windows_source::windows_source(int sampling_freq,
+ const std::string device_name)
+ : gr_sync_block("audio_windows_source",
+ gr_make_io_signature(0, 0, 0),
+ gr_make_io_signature(1, 2, sizeof(float))),
+ 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)
+ {
+ //FIXME TODO implement me
+#if 0
+ if((d_fd = open(d_device_name.c_str(), O_RDONLY)) < 0) {
+ fprintf(stderr, "audio_windows_source: ");
+ perror(d_device_name.c_str());
+ throw std::runtime_error("audio_windows_source");
+ }
+
+ d_chunk_size = (int)(d_sampling_freq * CHUNK_TIME);
+ set_output_multiple(d_chunk_size);
+
+ d_buffer = new short[d_chunk_size * 2];
+
+ int format = AFMT_S16_NE;
+ int orig_format = format;
+ if(ioctl(d_fd, SNDCTL_DSP_SETFMT, &format) < 0) {
+ std::cerr << "audio_windows_source: " << d_device_name
+ << " ioctl failed\n";
+ perror(d_device_name.c_str());
+ throw std::runtime_error("audio_windows_source");
+ }
+
+ if(format != orig_format) {
+ fprintf(stderr, "audio_windows_source: unable to support format %d\n",
+ orig_format);
+ fprintf(stderr, " card requested %d instead.\n", format);
+ }
+
+ // set to stereo no matter what. Some hardware only does stereo
+ int channels = 2;
+ if(ioctl(d_fd, SNDCTL_DSP_CHANNELS, &channels) < 0 || channels != 2) {
+ perror("audio_windows_source: could not set STEREO mode");
+ throw std::runtime_error("audio_windows_source");
+ }
+
+ // set sampling freq
+ int sf = sampling_freq;
+ if(ioctl(d_fd, SNDCTL_DSP_SPEED, &sf) < 0) {
+ std::cerr << "audio_windows_source: "
+ << d_device_name << ": invalid sampling_freq "
+ << sampling_freq << "\n";
+ sampling_freq = 8000;
+ if(ioctl(d_fd, SNDCTL_DSP_SPEED, &sf) < 0) {
+ std::cerr << "audio_windows_source: failed to set sampling_freq to 8000\n";
+ throw std::runtime_error ("audio_windows_source");
+ }
+ }
+#endif
+ }
+
+ windows_source::~windows_source()
+ {
+ /*close(d_fd);
+ delete [] d_buffer;
+ */
+ }
+
+ int
+ windows_source::work(int noutput_items,
+ gr_vector_const_void_star & input_items,
+ gr_vector_void_star & output_items)
+ {
+ //FIXME TODO implement me
+#if 0
+ float *f0 = (float*)output_items[0];
+ float *f1 = (float*)output_items[1]; // will be invalid if this is mono output
+
+ const int shorts_per_item = 2; // L + R
+ const int bytes_per_item = shorts_per_item * sizeof(short);
+
+ // To minimize latency, never return more than CHUNK_TIME
+ // worth of samples per call to work.
+ // FIXME, we need an API to set this value
+
+ noutput_items = std::min(noutput_items, d_chunk_size);
+
+ int base = 0;
+ int ntogo = noutput_items;
+
+ while(ntogo > 0) {
+ int nbytes = std::min(ntogo, d_chunk_size) * bytes_per_item;
+ int result_nbytes = read(d_fd, d_buffer, nbytes);
+
+ if(result_nbytes < 0) {
+ perror("audio_windows_source");
+ return -1; // say we're done
+ }
+
+ if((result_nbytes & (bytes_per_item - 1)) != 0) {
+ fprintf(stderr, "audio_windows_source: internal error.\n");
+ throw std::runtime_error("internal error");
+ }
+
+ int result_nitems = result_nbytes / bytes_per_item;
+
+ // now unpack samples into output streams
+ switch(output_items.size()) {
+ case 1: // mono output
+ for(int i = 0; i < result_nitems; i++) {
+ f0[base + i] = d_buffer[2 * i + 0] * (1.0 / 32767);
+ }
+ break;
+
+ case 2: // stereo output
+ for(int i = 0; i < result_nitems; i++) {
+ f0[base + i] = d_buffer[2 * i + 0] * (1.0 / 32767);
+ f1[base + i] = d_buffer[2 * i + 1] * (1.0 / 32767);
+ }
+ break;
+
+ default:
+ assert(0);
+ }
+
+ ntogo -= result_nitems;
+ base += result_nitems;
+ }
+
+ return noutput_items - ntogo;
+#endif
+ return -1; // EOF
+ }
+
+ } /* namespace audio */
+} /* namespace gr */
diff --git a/gr-audio/lib/windows/audio_windows_source.h b/gr-audio/lib/windows/windows_source.h
index 9cb7895760..18507d14b7 100644
--- a/gr-audio/lib/windows/audio_windows_source.h
+++ b/gr-audio/lib/windows/windows_source.h
@@ -1,6 +1,6 @@
/* -*- c++ -*- */
/*
- * Copyright 2004-2011 Free Software Foundation, Inc.
+ * Copyright 2004-2011,2013 Free Software Foundation, Inc.
*
* This file is part of GNU Radio
*
@@ -23,35 +23,38 @@
#ifndef INCLUDED_AUDIO_WINDOWS_SOURCE_H
#define INCLUDED_AUDIO_WINDOWS_SOURCE_H
-#include <gr_audio_source.h>
+#include <audio/source.h>
#include <string>
-/*!
- * \brief audio source using winmm mmsystem (win32 only)
- * \ingroup audio_blk
- *
- * Output signature is one or two streams of floats.
- * Output samples will be in the range [-1,1].
- */
-
-class audio_windows_source : public audio_source
-{
-
- int d_sampling_freq;
- std::string d_device_name;
- int d_fd;
- short *d_buffer;
- int d_chunk_size;
-
-public:
- audio_windows_source (int sampling_freq, const std::string device_name = "");
-
- ~audio_windows_source ();
-
- int
- work (int noutput_items,
- gr_vector_const_void_star & input_items,
- gr_vector_void_star & output_items);
-};
+namespace gr {
+ namespace audio {
+
+ /*!
+ * \brief audio source using winmm mmsystem (win32 only)
+ * \ingroup audio_blk
+ *
+ * Output signature is one or two streams of floats.
+ * Output samples will be in the range [-1,1].
+ */
+ class windows_source : public source
+ {
+ int d_sampling_freq;
+ std::string d_device_name;
+ int d_fd;
+ short *d_buffer;
+ int d_chunk_size;
+
+ public:
+ windows_source(int sampling_freq,
+ const std::string device_name = "");
+ ~windows_source();
+
+ int work(int noutput_items,
+ gr_vector_const_void_star & input_items,
+ gr_vector_void_star & output_items);
+ };
+
+ } /* namespace audio */
+} /* namespace gr */
#endif /* INCLUDED_AUDIO_WINDOWS_SOURCE_H */
diff --git a/gr-audio/swig/__init__.py b/gr-audio/swig/__init__.py
index ff10a8976a..7a73e3ca83 100644
--- a/gr-audio/swig/__init__.py
+++ b/gr-audio/swig/__init__.py
@@ -20,10 +20,11 @@
#
'''
-This is the gr-audio package. This package is used to connect to audio
-sources (mic-in) and sinks (speaker-out) ports on a computer. The
-underlying hardware driver is system and OS dependent and this module
-should automatically discover the correct one to use.
+Blocks to connect to audio sources (mic-in) and sinks (speaker-out)
+ports on a computer.
+
+The underlying hardware driver is system and OS dependent and this
+module should automatically discover the correct one to use.
'''
from audio_swig import *
diff --git a/gr-audio/swig/audio_swig.i b/gr-audio/swig/audio_swig.i
index 089055dac4..449fb5da51 100644
--- a/gr-audio/swig/audio_swig.i
+++ b/gr-audio/swig/audio_swig.i
@@ -1,5 +1,5 @@
/*
- * Copyright 2011 Free Software Foundation, Inc.
+ * Copyright 2011,2013 Free Software Foundation, Inc.
*
* This file is part of GNU Radio
*
@@ -22,27 +22,18 @@
#define GR_AUDIO_API
-////////////////////////////////////////////////////////////////////////
-// standard includes
-////////////////////////////////////////////////////////////////////////
%include "gnuradio.i"
//load generated python docstrings
%include "audio_swig_doc.i"
-////////////////////////////////////////////////////////////////////////
-// block headers
-////////////////////////////////////////////////////////////////////////
%{
-#include <gr_audio_source.h>
-#include <gr_audio_sink.h>
+#include <audio/source.h>
+#include <audio/sink.h>
%}
-////////////////////////////////////////////////////////////////////////
-// block magic
-////////////////////////////////////////////////////////////////////////
-GR_SWIG_BLOCK_MAGIC(audio,source)
-%include <gr_audio_source.h>
+%include <audio/source.h>
+%include <audio/sink.h>
-GR_SWIG_BLOCK_MAGIC(audio,sink)
-%include <gr_audio_sink.h>
+GR_SWIG_BLOCK_MAGIC2(audio, source)
+GR_SWIG_BLOCK_MAGIC2(audio, sink)