2015-03-22 20:44:37 +01:00
|
|
|
/** @file
|
|
|
|
* @author Edouard DUPIN
|
|
|
|
* @copyright 2015, Edouard DUPIN, all right reserved
|
2017-01-05 21:28:23 +01:00
|
|
|
* @license MPL v2.0 (see license file)
|
2015-03-22 20:44:37 +01:00
|
|
|
*/
|
|
|
|
|
2016-09-14 22:19:22 +02:00
|
|
|
//! [audio_river_sample_write_all]
|
|
|
|
|
2016-10-02 22:19:26 +02:00
|
|
|
#include <audio/river/river.hpp>
|
|
|
|
#include <audio/river/Manager.hpp>
|
|
|
|
#include <audio/river/Interface.hpp>
|
|
|
|
#include <etk/etk.hpp>
|
2017-09-07 23:38:26 +02:00
|
|
|
#include <ethread/Thread.hpp>
|
2017-11-07 13:38:04 +01:00
|
|
|
#include <ethread/tools.hpp>
|
|
|
|
#include <test-debug/debug.hpp>
|
2016-10-13 21:29:18 +02:00
|
|
|
|
2015-03-22 20:44:37 +01:00
|
|
|
|
2016-09-14 22:19:22 +02:00
|
|
|
//! [audio_river_sample_write_config_file]
|
2017-08-28 00:08:50 +02:00
|
|
|
static const etk::String configurationRiver =
|
2015-03-22 20:44:37 +01:00
|
|
|
"{\n"
|
|
|
|
" speaker:{\n"
|
|
|
|
" io:'output',\n"
|
|
|
|
" map-on:{\n"
|
2016-04-20 21:19:11 +02:00
|
|
|
" interface:'alsa',\n"
|
2017-04-26 22:09:23 +00:00
|
|
|
" name:'hw:0,3',\n"
|
2015-03-22 20:44:37 +01:00
|
|
|
" },\n"
|
2016-08-22 21:52:31 +02:00
|
|
|
" frequency:48000,\n"
|
2016-04-20 21:19:11 +02:00
|
|
|
//" channel-map:['front-left', 'front-right', 'rear-left', 'rear-right'],\n"
|
2015-03-22 20:44:37 +01:00
|
|
|
" channel-map:['front-left', 'front-right'],\n"
|
2016-08-22 21:52:31 +02:00
|
|
|
" type:'int16',\n"
|
2015-03-22 20:44:37 +01:00
|
|
|
" nb-chunk:1024,\n"
|
|
|
|
" volume-name:'MASTER'\n"
|
|
|
|
" }\n"
|
|
|
|
"}\n";
|
2016-09-14 22:19:22 +02:00
|
|
|
//! [audio_river_sample_write_config_file]
|
2015-03-22 20:44:37 +01:00
|
|
|
|
2016-04-20 21:19:11 +02:00
|
|
|
static const int32_t nbChannelMax=8;
|
|
|
|
|
2016-09-14 22:19:22 +02:00
|
|
|
//! [audio_river_sample_callback_implement]
|
2015-03-22 20:44:37 +01:00
|
|
|
void onDataNeeded(void* _data,
|
2015-05-08 22:36:02 +02:00
|
|
|
const audio::Time& _time,
|
2015-03-22 20:44:37 +01:00
|
|
|
size_t _nbChunk,
|
|
|
|
enum audio::format _format,
|
2017-01-05 21:28:23 +01:00
|
|
|
uint32_t _sampleRate,
|
2017-08-28 00:08:50 +02:00
|
|
|
const etk::Vector<audio::channel>& _map) {
|
2016-04-20 21:19:11 +02:00
|
|
|
static double phase[8] = {0,0,0,0,0,0,0,0};
|
|
|
|
|
2015-03-22 20:44:37 +01:00
|
|
|
if (_format != audio::format_int16) {
|
2017-09-26 15:57:44 +02:00
|
|
|
TEST_ERROR("Call wrong type ... (need int16_t)");
|
2015-03-22 20:44:37 +01:00
|
|
|
}
|
2017-09-26 15:57:44 +02:00
|
|
|
//TEST_VERBOSE("Map " << _map);
|
2015-03-22 20:44:37 +01:00
|
|
|
int16_t* data = static_cast<int16_t*>(_data);
|
2016-04-20 21:19:11 +02:00
|
|
|
double baseCycle = 2.0*M_PI/double(48000) * double(440);
|
2015-03-22 20:44:37 +01:00
|
|
|
for (int32_t iii=0; iii<_nbChunk; iii++) {
|
|
|
|
for (int32_t jjj=0; jjj<_map.size(); jjj++) {
|
2016-04-20 21:19:11 +02:00
|
|
|
data[_map.size()*iii+jjj] = cos(phase[jjj]) * 30000;
|
|
|
|
phase[jjj] += baseCycle*jjj;
|
|
|
|
if (phase[jjj] >= 2*M_PI) {
|
|
|
|
phase[jjj] -= 2*M_PI;
|
|
|
|
}
|
2015-03-22 20:44:37 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2016-09-14 22:19:22 +02:00
|
|
|
//! [audio_river_sample_callback_implement]
|
2015-03-22 20:44:37 +01:00
|
|
|
|
2015-05-08 22:36:02 +02:00
|
|
|
int main(int _argc, const char **_argv) {
|
2015-03-22 20:44:37 +01:00
|
|
|
// the only one init for etk:
|
|
|
|
etk::init(_argc, _argv);
|
|
|
|
for (int32_t iii=0; iii<_argc ; ++iii) {
|
2017-08-28 00:08:50 +02:00
|
|
|
etk::String data = _argv[iii];
|
2015-03-22 20:44:37 +01:00
|
|
|
if ( data == "-h"
|
|
|
|
|| data == "--help") {
|
2017-09-26 15:57:44 +02:00
|
|
|
TEST_PRINT("Help:");
|
|
|
|
TEST_PRINT(" ./xxx ---");
|
2015-03-22 20:44:37 +01:00
|
|
|
exit(0);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
// initialize river interface
|
2015-05-08 22:36:02 +02:00
|
|
|
audio::river::initString(configurationRiver);
|
2015-03-22 20:44:37 +01:00
|
|
|
// Create the River manager for tha application or part of the application.
|
2016-07-19 21:43:58 +02:00
|
|
|
ememory::SharedPtr<audio::river::Manager> manager = audio::river::Manager::create("river_sample_read");
|
2016-09-14 22:19:22 +02:00
|
|
|
//! [audio_river_sample_create_write_interface]
|
2015-03-22 20:44:37 +01:00
|
|
|
// create interface:
|
2016-07-19 21:43:58 +02:00
|
|
|
ememory::SharedPtr<audio::river::Interface> interface;
|
2015-03-22 20:44:37 +01:00
|
|
|
//Get the generic input:
|
|
|
|
interface = manager->createOutput(48000,
|
2017-08-28 00:08:50 +02:00
|
|
|
etk::Vector<audio::channel>(),
|
2015-03-22 20:44:37 +01:00
|
|
|
audio::format_int16,
|
|
|
|
"speaker");
|
|
|
|
if(interface == nullptr) {
|
2017-09-26 15:57:44 +02:00
|
|
|
TEST_ERROR("nullptr interface");
|
2015-03-22 20:44:37 +01:00
|
|
|
return -1;
|
|
|
|
}
|
2016-09-14 22:19:22 +02:00
|
|
|
//! [audio_river_sample_create_write_interface]
|
|
|
|
//! [audio_river_sample_set_callback]
|
2015-03-22 20:44:37 +01:00
|
|
|
// set callback mode ...
|
2017-11-07 13:38:04 +01:00
|
|
|
interface->setOutputCallback([=](void* _data,
|
2017-09-26 15:57:44 +02:00
|
|
|
const audio::Time& _time,
|
|
|
|
size_t _nbChunk,
|
|
|
|
enum audio::format _format,
|
|
|
|
uint32_t _frequency,
|
|
|
|
const etk::Vector<audio::channel>& _map) {
|
|
|
|
onDataNeeded(_data, _time, _nbChunk, _format, _frequency, _map);
|
|
|
|
});
|
2016-09-14 22:19:22 +02:00
|
|
|
//! [audio_river_sample_set_callback]
|
2015-03-22 20:44:37 +01:00
|
|
|
// start the stream
|
|
|
|
interface->start();
|
|
|
|
// wait 10 second ...
|
2017-09-26 15:57:44 +02:00
|
|
|
ethread::sleepMilliSeconds(1000*(10));
|
2015-03-22 20:44:37 +01:00
|
|
|
// stop the stream
|
|
|
|
interface->stop();
|
|
|
|
// remove interface and manager.
|
|
|
|
interface.reset();
|
|
|
|
manager.reset();
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2016-09-14 22:19:22 +02:00
|
|
|
//! [audio_river_sample_write_all]
|
|
|
|
|