/*** This file is part of snapcast Copyright (C) 2014-2019 Johannes Pohl This program 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 of the License, or (at your option) any later version. This program 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 this program. If not, see . ***/ #include #include #include #include #include #include "common/aixlog.hpp" #include "common/snap_exception.hpp" #include "common/str_compat.hpp" #include "pipe_stream.hpp" using namespace std; PipeStream::PipeStream(PcmListener* pcmListener, boost::asio::io_context& ioc, const StreamUri& uri) : AsioStream(pcmListener, ioc, uri) { umask(0); string mode = uri_.getQuery("mode", "create"); LOG(INFO) << "PipeStream mode: " << mode << "\n"; if ((mode != "read") && (mode != "create")) throw SnapException("create mode for fifo must be \"read\" or \"create\""); if (mode == "create") { if ((mkfifo(uri_.path.c_str(), 0666) != 0) && (errno != EEXIST)) throw SnapException("failed to make fifo \"" + uri_.path + "\": " + cpt::to_string(errno)); } chunk_ = make_unique(sampleFormat_, pcmReadMs_); } void PipeStream::connect() { LOG(DEBUG) << "connect\n"; auto self = shared_from_this(); auto fd = open(uri_.path.c_str(), O_RDONLY | O_NONBLOCK); stream_ = std::make_unique(ioc_, fd); on_connect(); } void PipeStream::disconnect() { stream_->close(); } // void PipeStream::worker() // { // timeval tvChunk; // std::unique_ptr chunk(new msg::PcmChunk(sampleFormat_, pcmReadMs_)); // string lastException = ""; // while (active_) // { // if (fd_ != -1) // close(fd_); // fd_ = open(uri_.path.c_str(), O_RDONLY | O_NONBLOCK); // chronos::systemtimeofday(&tvChunk); // tvEncodedChunk_ = tvChunk; // long nextTick = chronos::getTickCount(); // int idleBytes = 0; // int maxIdleBytes = sampleFormat_.rate * sampleFormat_.frameSize * dryoutMs_ / 1000; // try // { // if (fd_ == -1) // throw SnapException("failed to open fifo: \"" + uri_.path + "\""); // while (active_) // { // chunk->timestamp.sec = tvChunk.tv_sec; // chunk->timestamp.usec = tvChunk.tv_usec; // int toRead = chunk->payloadSize; // int len = 0; // do // { // int count = read(fd_, chunk->payload + len, toRead - len); // if (count < 0 && idleBytes < maxIdleBytes) // { // memset(chunk->payload + len, 0, toRead - len); // idleBytes += toRead - len; // len += toRead - len; // continue; // } // if (count < 0) // { // setState(kIdle); // if (!sleep(100)) // break; // } // else if (count == 0) // throw SnapException("end of file"); // else // { // len += count; // idleBytes = 0; // } // } while ((len < toRead) && active_); // if (!active_) // break; // /// TODO: use less raw pointers, make this encoding more transparent // encoder_->encode(chunk.get()); // if (!active_) // break; // nextTick += pcmReadMs_; // chronos::addUs(tvChunk, pcmReadMs_ * 1000); // long currentTick = chronos::getTickCount(); // if (nextTick >= currentTick) // { // setState(kPlaying); // if (!sleep(nextTick - currentTick)) // break; // } // else // { // chronos::systemtimeofday(&tvChunk); // tvEncodedChunk_ = tvChunk; // pcmListener_->onResync(this, currentTick - nextTick); // nextTick = currentTick; // } // lastException = ""; // } // } // catch (const std::exception& e) // { // if (lastException != e.what()) // { // LOG(ERROR) << "(PipeStream) Exception: " << e.what() << std::endl; // lastException = e.what(); // } // if (!sleep(100)) // break; // } // } // }