2016-05-03 01:16:26 +00:00
|
|
|
#include "amuse/BooBackend.hpp"
|
|
|
|
#include "amuse/Voice.hpp"
|
2016-05-07 22:10:57 +00:00
|
|
|
#include "amuse/Submix.hpp"
|
2016-05-20 06:17:02 +00:00
|
|
|
#include "amuse/Engine.hpp"
|
2016-05-03 01:16:26 +00:00
|
|
|
|
2018-12-08 05:20:09 +00:00
|
|
|
namespace amuse {
|
2016-05-03 01:16:26 +00:00
|
|
|
|
2018-12-08 05:20:09 +00:00
|
|
|
void BooBackendVoice::VoiceCallback::preSupplyAudio(boo::IAudioVoice&, double dt) {
|
|
|
|
m_parent.m_clientVox.preSupplyAudio(dt);
|
2016-07-05 01:08:00 +00:00
|
|
|
}
|
|
|
|
|
2018-12-08 05:20:09 +00:00
|
|
|
size_t BooBackendVoice::VoiceCallback::supplyAudio(boo::IAudioVoice&, size_t frames, int16_t* data) {
|
|
|
|
return m_parent.m_clientVox.supplyAudio(frames, data);
|
2016-05-03 01:16:26 +00:00
|
|
|
}
|
|
|
|
|
2018-12-08 05:20:09 +00:00
|
|
|
void BooBackendVoice::VoiceCallback::routeAudio(size_t frames, size_t channels, double dt, int busId, int16_t* in,
|
|
|
|
int16_t* out) {
|
|
|
|
m_parent.m_clientVox.routeAudio(frames, dt, busId, in, out);
|
2016-05-11 04:48:08 +00:00
|
|
|
}
|
|
|
|
|
2018-12-08 05:20:09 +00:00
|
|
|
void BooBackendVoice::VoiceCallback::routeAudio(size_t frames, size_t channels, double dt, int busId, int32_t* in,
|
|
|
|
int32_t* out) {
|
|
|
|
m_parent.m_clientVox.routeAudio(frames, dt, busId, in, out);
|
2016-05-03 01:16:26 +00:00
|
|
|
}
|
|
|
|
|
2018-12-08 05:20:09 +00:00
|
|
|
void BooBackendVoice::VoiceCallback::routeAudio(size_t frames, size_t channels, double dt, int busId, float* in,
|
|
|
|
float* out) {
|
|
|
|
m_parent.m_clientVox.routeAudio(frames, dt, busId, in, out);
|
2016-05-31 05:15:20 +00:00
|
|
|
}
|
|
|
|
|
2016-07-14 04:54:46 +00:00
|
|
|
BooBackendVoice::BooBackendVoice(boo::IAudioVoiceEngine& engine, Voice& clientVox, double sampleRate, bool dynamicPitch)
|
2018-12-08 05:20:09 +00:00
|
|
|
: m_clientVox(clientVox), m_cb(*this), m_booVoice(engine.allocateNewMonoVoice(sampleRate, &m_cb, dynamicPitch)) {}
|
2016-05-03 01:16:26 +00:00
|
|
|
|
2016-07-14 04:54:46 +00:00
|
|
|
void BooBackendVoice::resetSampleRate(double sampleRate) { m_booVoice->resetSampleRate(sampleRate); }
|
2016-05-03 01:16:26 +00:00
|
|
|
|
2016-07-14 04:54:46 +00:00
|
|
|
void BooBackendVoice::resetChannelLevels() { m_booVoice->resetChannelLevels(); }
|
2016-05-03 01:16:26 +00:00
|
|
|
|
2018-12-08 05:20:09 +00:00
|
|
|
void BooBackendVoice::setChannelLevels(IBackendSubmix* submix, const float coefs[8], bool slew) {
|
|
|
|
BooBackendSubmix& smx = *reinterpret_cast<BooBackendSubmix*>(submix);
|
|
|
|
m_booVoice->setMonoChannelLevels(smx.m_booSubmix.get(), coefs, slew);
|
2016-05-07 22:10:57 +00:00
|
|
|
}
|
|
|
|
|
2016-07-14 04:54:46 +00:00
|
|
|
void BooBackendVoice::setPitchRatio(double ratio, bool slew) { m_booVoice->setPitchRatio(ratio, slew); }
|
|
|
|
|
|
|
|
void BooBackendVoice::start() { m_booVoice->start(); }
|
|
|
|
|
|
|
|
void BooBackendVoice::stop() { m_booVoice->stop(); }
|
|
|
|
|
|
|
|
bool BooBackendSubmix::SubmixCallback::canApplyEffect() const { return m_parent.m_clientSmx.canApplyEffect(); }
|
|
|
|
|
|
|
|
void BooBackendSubmix::SubmixCallback::applyEffect(int16_t* audio, size_t frameCount, const boo::ChannelMap& chanMap,
|
2018-12-08 05:20:09 +00:00
|
|
|
double) const {
|
|
|
|
return m_parent.m_clientSmx.applyEffect(audio, frameCount, reinterpret_cast<const ChannelMap&>(chanMap));
|
2016-05-07 22:10:57 +00:00
|
|
|
}
|
|
|
|
|
2016-07-14 04:54:46 +00:00
|
|
|
void BooBackendSubmix::SubmixCallback::applyEffect(int32_t* audio, size_t frameCount, const boo::ChannelMap& chanMap,
|
2018-12-08 05:20:09 +00:00
|
|
|
double) const {
|
|
|
|
return m_parent.m_clientSmx.applyEffect(audio, frameCount, reinterpret_cast<const ChannelMap&>(chanMap));
|
2016-05-07 22:10:57 +00:00
|
|
|
}
|
|
|
|
|
2016-07-14 04:54:46 +00:00
|
|
|
void BooBackendSubmix::SubmixCallback::applyEffect(float* audio, size_t frameCount, const boo::ChannelMap& chanMap,
|
2018-12-08 05:20:09 +00:00
|
|
|
double) const {
|
|
|
|
return m_parent.m_clientSmx.applyEffect(audio, frameCount, reinterpret_cast<const ChannelMap&>(chanMap));
|
2016-05-07 22:10:57 +00:00
|
|
|
}
|
|
|
|
|
2018-12-08 05:20:09 +00:00
|
|
|
void BooBackendSubmix::SubmixCallback::resetOutputSampleRate(double sampleRate) {
|
|
|
|
m_parent.m_clientSmx.resetOutputSampleRate(sampleRate);
|
2016-06-01 04:49:35 +00:00
|
|
|
}
|
|
|
|
|
2016-07-14 04:54:46 +00:00
|
|
|
BooBackendSubmix::BooBackendSubmix(boo::IAudioVoiceEngine& engine, Submix& clientSmx, bool mainOut, int busId)
|
2018-12-08 05:20:09 +00:00
|
|
|
: m_clientSmx(clientSmx), m_cb(*this), m_booSubmix(engine.allocateNewSubmix(mainOut, &m_cb, busId)) {}
|
2016-05-07 22:10:57 +00:00
|
|
|
|
2018-12-08 05:20:09 +00:00
|
|
|
void BooBackendSubmix::setSendLevel(IBackendSubmix* submix, float level, bool slew) {
|
|
|
|
BooBackendSubmix& smx = *reinterpret_cast<BooBackendSubmix*>(submix);
|
|
|
|
m_booSubmix->setSendLevel(smx.m_booSubmix.get(), level, slew);
|
2016-05-07 22:10:57 +00:00
|
|
|
}
|
|
|
|
|
2016-07-14 04:54:46 +00:00
|
|
|
double BooBackendSubmix::getSampleRate() const { return m_booSubmix->getSampleRate(); }
|
2016-05-14 04:46:39 +00:00
|
|
|
|
2016-07-14 04:54:46 +00:00
|
|
|
SubmixFormat BooBackendSubmix::getSampleFormat() const { return SubmixFormat(m_booSubmix->getSampleFormat()); }
|
2016-05-20 06:17:02 +00:00
|
|
|
|
2016-05-20 22:56:25 +00:00
|
|
|
BooBackendMIDIReader::~BooBackendMIDIReader() {}
|
|
|
|
|
2018-08-19 00:28:52 +00:00
|
|
|
BooBackendMIDIReader::BooBackendMIDIReader(Engine& engine, bool useLock)
|
2018-12-08 05:20:09 +00:00
|
|
|
: m_engine(engine), m_decoder(*this), m_useLock(useLock) {
|
|
|
|
BooBackendVoiceAllocator& voxAlloc = static_cast<BooBackendVoiceAllocator&>(engine.getBackend());
|
|
|
|
auto devices = voxAlloc.m_booEngine.enumerateMIDIInputs();
|
|
|
|
for (const auto& dev : devices) {
|
|
|
|
auto midiIn =
|
|
|
|
voxAlloc.m_booEngine.newRealMIDIIn(dev.first.c_str(), std::bind(&BooBackendMIDIReader::_MIDIReceive, this,
|
|
|
|
std::placeholders::_1, std::placeholders::_2));
|
2018-08-19 00:28:52 +00:00
|
|
|
if (midiIn)
|
2018-12-08 05:20:09 +00:00
|
|
|
m_midiIns[dev.first] = std::move(midiIn);
|
|
|
|
}
|
|
|
|
if (voxAlloc.m_booEngine.supportsVirtualMIDIIn())
|
|
|
|
m_virtualIn = voxAlloc.m_booEngine.newVirtualMIDIIn(
|
|
|
|
std::bind(&BooBackendMIDIReader::_MIDIReceive, this, std::placeholders::_1, std::placeholders::_2));
|
2018-08-19 00:28:52 +00:00
|
|
|
}
|
|
|
|
|
2018-12-08 05:20:09 +00:00
|
|
|
void BooBackendMIDIReader::addMIDIIn(const char* name) {
|
|
|
|
BooBackendVoiceAllocator& voxAlloc = static_cast<BooBackendVoiceAllocator&>(m_engine.getBackend());
|
|
|
|
auto midiIn = voxAlloc.m_booEngine.newRealMIDIIn(
|
|
|
|
name, std::bind(&BooBackendMIDIReader::_MIDIReceive, this, std::placeholders::_1, std::placeholders::_2));
|
|
|
|
if (midiIn)
|
|
|
|
m_midiIns[name] = std::move(midiIn);
|
2018-08-19 00:28:52 +00:00
|
|
|
}
|
|
|
|
|
2018-12-08 05:20:09 +00:00
|
|
|
void BooBackendMIDIReader::removeMIDIIn(const char* name) { m_midiIns.erase(name); }
|
2018-08-19 00:28:52 +00:00
|
|
|
|
2018-12-08 05:20:09 +00:00
|
|
|
bool BooBackendMIDIReader::hasMIDIIn(const char* name) const { return m_midiIns.find(name) != m_midiIns.cend(); }
|
2018-08-19 00:28:52 +00:00
|
|
|
|
2018-12-08 05:20:09 +00:00
|
|
|
void BooBackendMIDIReader::setVirtualIn(bool v) {
|
|
|
|
if (v) {
|
|
|
|
BooBackendVoiceAllocator& voxAlloc = static_cast<BooBackendVoiceAllocator&>(m_engine.getBackend());
|
|
|
|
if (voxAlloc.m_booEngine.supportsVirtualMIDIIn())
|
|
|
|
m_virtualIn = voxAlloc.m_booEngine.newVirtualMIDIIn(
|
|
|
|
std::bind(&BooBackendMIDIReader::_MIDIReceive, this, std::placeholders::_1, std::placeholders::_2));
|
|
|
|
} else {
|
|
|
|
m_virtualIn.reset();
|
|
|
|
}
|
2016-05-20 22:56:25 +00:00
|
|
|
}
|
|
|
|
|
2018-12-08 05:20:09 +00:00
|
|
|
bool BooBackendMIDIReader::hasVirtualIn() const { return m_virtualIn.operator bool(); }
|
|
|
|
|
|
|
|
void BooBackendMIDIReader::_MIDIReceive(std::vector<uint8_t>&& bytes, double time) {
|
|
|
|
std::unique_lock<std::mutex> lk(m_midiMutex, std::defer_lock_t{});
|
|
|
|
if (m_useLock)
|
|
|
|
lk.lock();
|
|
|
|
m_queue.emplace_back(time, std::move(bytes));
|
2016-06-08 04:33:15 +00:00
|
|
|
#if 0
|
|
|
|
openlog("LogIt", (LOG_CONS|LOG_PERROR|LOG_PID), LOG_DAEMON);
|
|
|
|
syslog(LOG_EMERG, "MIDI receive %f\n", time);
|
|
|
|
closelog();
|
|
|
|
#endif
|
2016-05-20 22:56:25 +00:00
|
|
|
}
|
|
|
|
|
2018-12-08 05:20:09 +00:00
|
|
|
void BooBackendMIDIReader::pumpReader(double dt) {
|
|
|
|
dt += 0.001; /* Add 1ms to ensure consumer keeps up with producer */
|
|
|
|
|
|
|
|
std::unique_lock<std::mutex> lk(m_midiMutex, std::defer_lock_t{});
|
|
|
|
if (m_useLock)
|
|
|
|
lk.lock();
|
|
|
|
if (m_queue.empty())
|
|
|
|
return;
|
|
|
|
|
|
|
|
/* Determine range of buffer updates within this period */
|
|
|
|
auto periodEnd = m_queue.cbegin();
|
|
|
|
double startPt = m_queue.front().first;
|
|
|
|
for (; periodEnd != m_queue.cend(); ++periodEnd) {
|
|
|
|
double delta = periodEnd->first - startPt;
|
|
|
|
if (delta > dt)
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (m_queue.cbegin() == periodEnd)
|
|
|
|
return;
|
|
|
|
|
|
|
|
/* Dispatch buffers */
|
|
|
|
for (auto it = m_queue.begin(); it != periodEnd;) {
|
2016-06-08 04:33:15 +00:00
|
|
|
#if 0
|
|
|
|
char str[64];
|
|
|
|
sprintf(str, "MIDI %zu %f ", it->second.size(), it->first);
|
|
|
|
for (uint8_t byte : it->second)
|
|
|
|
sprintf(str + strlen(str), "%02X ", byte);
|
|
|
|
openlog("LogIt", (LOG_CONS|LOG_PERROR|LOG_PID), LOG_DAEMON);
|
|
|
|
syslog(LOG_EMERG, "%s\n", str);
|
|
|
|
closelog();
|
|
|
|
#endif
|
2018-12-08 05:20:09 +00:00
|
|
|
m_decoder.receiveBytes(it->second.cbegin(), it->second.cend());
|
|
|
|
it = m_queue.erase(it);
|
|
|
|
}
|
2016-05-20 06:17:02 +00:00
|
|
|
}
|
|
|
|
|
2018-12-08 05:20:09 +00:00
|
|
|
void BooBackendMIDIReader::noteOff(uint8_t chan, uint8_t key, uint8_t velocity) {
|
|
|
|
for (ObjToken<Sequencer>& seq : m_engine.getActiveSequencers())
|
|
|
|
seq->keyOff(chan, key, velocity);
|
2016-06-08 04:33:15 +00:00
|
|
|
#if 0
|
|
|
|
openlog("LogIt", (LOG_CONS|LOG_PERROR|LOG_PID), LOG_DAEMON);
|
|
|
|
syslog(LOG_EMERG, "NoteOff %d", key);
|
|
|
|
closelog();
|
|
|
|
#endif
|
2016-05-20 06:17:02 +00:00
|
|
|
}
|
|
|
|
|
2018-12-08 05:20:09 +00:00
|
|
|
void BooBackendMIDIReader::noteOn(uint8_t chan, uint8_t key, uint8_t velocity) {
|
|
|
|
for (ObjToken<Sequencer>& seq : m_engine.getActiveSequencers())
|
|
|
|
seq->keyOn(chan, key, velocity);
|
2016-06-08 04:33:15 +00:00
|
|
|
#if 0
|
|
|
|
openlog("LogIt", (LOG_CONS|LOG_PERROR|LOG_PID), LOG_DAEMON);
|
|
|
|
syslog(LOG_EMERG, "NoteOn %d", key);
|
|
|
|
closelog();
|
|
|
|
#endif
|
2016-05-20 06:17:02 +00:00
|
|
|
}
|
|
|
|
|
2016-07-14 04:54:46 +00:00
|
|
|
void BooBackendMIDIReader::notePressure(uint8_t /*chan*/, uint8_t /*key*/, uint8_t /*pressure*/) {}
|
2016-05-20 06:17:02 +00:00
|
|
|
|
2018-12-08 05:20:09 +00:00
|
|
|
void BooBackendMIDIReader::controlChange(uint8_t chan, uint8_t control, uint8_t value) {
|
|
|
|
for (ObjToken<Sequencer>& seq : m_engine.getActiveSequencers())
|
|
|
|
seq->setCtrlValue(chan, control, value);
|
2016-05-20 06:17:02 +00:00
|
|
|
}
|
|
|
|
|
2018-12-08 05:20:09 +00:00
|
|
|
void BooBackendMIDIReader::programChange(uint8_t chan, uint8_t program) {
|
|
|
|
for (ObjToken<Sequencer>& seq : m_engine.getActiveSequencers())
|
|
|
|
seq->setChanProgram(chan, program);
|
2016-05-20 06:17:02 +00:00
|
|
|
}
|
|
|
|
|
2016-07-14 04:54:46 +00:00
|
|
|
void BooBackendMIDIReader::channelPressure(uint8_t /*chan*/, uint8_t /*pressure*/) {}
|
2016-05-20 06:17:02 +00:00
|
|
|
|
2018-12-08 05:20:09 +00:00
|
|
|
void BooBackendMIDIReader::pitchBend(uint8_t chan, int16_t pitch) {
|
|
|
|
for (ObjToken<Sequencer>& seq : m_engine.getActiveSequencers())
|
|
|
|
seq->setPitchWheel(chan, (pitch - 0x2000) / float(0x2000));
|
2016-05-20 06:17:02 +00:00
|
|
|
}
|
|
|
|
|
2018-12-08 05:20:09 +00:00
|
|
|
void BooBackendMIDIReader::allSoundOff(uint8_t chan) {
|
|
|
|
for (ObjToken<Sequencer>& seq : m_engine.getActiveSequencers())
|
|
|
|
seq->allOff(chan, true);
|
2016-05-20 06:17:02 +00:00
|
|
|
}
|
|
|
|
|
2016-07-14 04:54:46 +00:00
|
|
|
void BooBackendMIDIReader::resetAllControllers(uint8_t /*chan*/) {}
|
2016-05-20 06:17:02 +00:00
|
|
|
|
2016-07-14 04:54:46 +00:00
|
|
|
void BooBackendMIDIReader::localControl(uint8_t /*chan*/, bool /*on*/) {}
|
2016-05-20 06:17:02 +00:00
|
|
|
|
2018-12-08 05:20:09 +00:00
|
|
|
void BooBackendMIDIReader::allNotesOff(uint8_t chan) {
|
|
|
|
for (ObjToken<Sequencer>& seq : m_engine.getActiveSequencers())
|
|
|
|
seq->allOff(chan, false);
|
2016-05-20 06:17:02 +00:00
|
|
|
}
|
|
|
|
|
2016-07-14 04:54:46 +00:00
|
|
|
void BooBackendMIDIReader::omniMode(uint8_t /*chan*/, bool /*on*/) {}
|
2016-05-20 06:17:02 +00:00
|
|
|
|
2016-07-14 04:54:46 +00:00
|
|
|
void BooBackendMIDIReader::polyMode(uint8_t /*chan*/, bool /*on*/) {}
|
2016-05-20 06:17:02 +00:00
|
|
|
|
2016-07-14 04:54:46 +00:00
|
|
|
void BooBackendMIDIReader::sysex(const void* /*data*/, size_t /*len*/) {}
|
2016-05-20 06:17:02 +00:00
|
|
|
|
2016-07-14 04:54:46 +00:00
|
|
|
void BooBackendMIDIReader::timeCodeQuarterFrame(uint8_t /*message*/, uint8_t /*value*/) {}
|
2016-05-20 06:17:02 +00:00
|
|
|
|
2016-07-14 04:54:46 +00:00
|
|
|
void BooBackendMIDIReader::songPositionPointer(uint16_t /*pointer*/) {}
|
2016-05-20 06:17:02 +00:00
|
|
|
|
2016-07-14 04:54:46 +00:00
|
|
|
void BooBackendMIDIReader::songSelect(uint8_t /*song*/) {}
|
2016-05-20 06:17:02 +00:00
|
|
|
|
2016-07-14 04:54:46 +00:00
|
|
|
void BooBackendMIDIReader::tuneRequest() {}
|
2016-05-20 06:17:02 +00:00
|
|
|
|
2016-07-14 04:54:46 +00:00
|
|
|
void BooBackendMIDIReader::startSeq() {}
|
2016-05-20 06:17:02 +00:00
|
|
|
|
2016-07-14 04:54:46 +00:00
|
|
|
void BooBackendMIDIReader::continueSeq() {}
|
2016-05-20 06:17:02 +00:00
|
|
|
|
2016-07-14 04:54:46 +00:00
|
|
|
void BooBackendMIDIReader::stopSeq() {}
|
2016-05-20 06:17:02 +00:00
|
|
|
|
2016-07-14 04:54:46 +00:00
|
|
|
void BooBackendMIDIReader::reset() {}
|
2016-05-20 06:17:02 +00:00
|
|
|
|
2018-12-08 05:20:09 +00:00
|
|
|
BooBackendVoiceAllocator::BooBackendVoiceAllocator(boo::IAudioVoiceEngine& booEngine) : m_booEngine(booEngine) {
|
|
|
|
booEngine.setCallbackInterface(this);
|
2017-02-15 06:01:39 +00:00
|
|
|
}
|
2016-05-03 01:16:26 +00:00
|
|
|
|
2016-07-14 04:54:46 +00:00
|
|
|
std::unique_ptr<IBackendVoice> BooBackendVoiceAllocator::allocateVoice(Voice& clientVox, double sampleRate,
|
2018-12-08 05:20:09 +00:00
|
|
|
bool dynamicPitch) {
|
|
|
|
return std::make_unique<BooBackendVoice>(m_booEngine, clientVox, sampleRate, dynamicPitch);
|
2016-05-03 01:16:26 +00:00
|
|
|
}
|
|
|
|
|
2018-12-08 05:20:09 +00:00
|
|
|
std::unique_ptr<IBackendSubmix> BooBackendVoiceAllocator::allocateSubmix(Submix& clientSmx, bool mainOut, int busId) {
|
|
|
|
return std::make_unique<BooBackendSubmix>(m_booEngine, clientSmx, mainOut, busId);
|
2016-05-07 22:10:57 +00:00
|
|
|
}
|
|
|
|
|
2018-12-08 05:20:09 +00:00
|
|
|
std::vector<std::pair<std::string, std::string>> BooBackendVoiceAllocator::enumerateMIDIDevices() {
|
|
|
|
return m_booEngine.enumerateMIDIInputs();
|
2016-05-20 06:17:02 +00:00
|
|
|
}
|
|
|
|
|
2018-12-08 05:20:09 +00:00
|
|
|
std::unique_ptr<IMIDIReader> BooBackendVoiceAllocator::allocateMIDIReader(Engine& engine) {
|
|
|
|
return std::make_unique<BooBackendMIDIReader>(engine, m_booEngine.useMIDILock());
|
2016-05-20 06:17:02 +00:00
|
|
|
}
|
|
|
|
|
2018-12-08 05:20:09 +00:00
|
|
|
void BooBackendVoiceAllocator::setCallbackInterface(Engine* engine) { m_cbInterface = engine; }
|
2017-01-23 07:21:50 +00:00
|
|
|
|
2016-07-14 04:54:46 +00:00
|
|
|
AudioChannelSet BooBackendVoiceAllocator::getAvailableSet() { return AudioChannelSet(m_booEngine.getAvailableSet()); }
|
2016-05-14 04:46:39 +00:00
|
|
|
|
2016-07-14 06:16:00 +00:00
|
|
|
void BooBackendVoiceAllocator::setVolume(float vol) { m_booEngine.setVolume(vol); }
|
2017-02-15 06:01:39 +00:00
|
|
|
|
2018-12-08 05:20:09 +00:00
|
|
|
void BooBackendVoiceAllocator::on5MsInterval(boo::IAudioVoiceEngine& engine, double dt) {
|
|
|
|
if (m_cbInterface)
|
|
|
|
m_cbInterface->_on5MsInterval(*this, dt);
|
2017-02-15 06:01:39 +00:00
|
|
|
}
|
|
|
|
|
2018-12-08 05:20:09 +00:00
|
|
|
void BooBackendVoiceAllocator::onPumpCycleComplete(boo::IAudioVoiceEngine& engine) {
|
|
|
|
if (m_cbInterface)
|
|
|
|
m_cbInterface->_onPumpCycleComplete(*this);
|
2016-05-03 01:16:26 +00:00
|
|
|
}
|
2018-12-08 05:20:09 +00:00
|
|
|
} // namespace amuse
|