/*********************************************************************** Moses - factored phrase-based language decoder Copyright (C) 2011 University of Edinburgh This library is free software; you can redistribute it and/or modify it under the terms of the GNU Lesser General Public License as published by the Free Software Foundation; either version 2.1 of the License, or (at your option) any later version. This library 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 Lesser General Public License for more details. You should have received a copy of the GNU Lesser General Public License along with this library; if not, write to the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA ***********************************************************************/ #pragma once #ifndef moses_OutputCollector_h #define moses_OutputCollector_h #ifdef WITH_THREADS #include #endif #ifdef BOOST_HAS_PTHREADS #include #endif #include #include #include #include namespace Moses { /** * Makes sure output goes in the correct order when multi-threading **/ class OutputCollector { public: OutputCollector(std::ostream* outStream= &std::cout, std::ostream* debugStream=&std::cerr) : m_nextOutput(0),m_outStream(outStream),m_debugStream(debugStream), m_isHoldingOutputStream(false), m_isHoldingDebugStream(false) {} ~OutputCollector() { if (m_isHoldingOutputStream) delete m_outStream; if (m_isHoldingDebugStream) delete m_debugStream; } void HoldOutputStream() { m_isHoldingOutputStream = true; } void HoldDebugStream() { m_isHoldingDebugStream = true; } bool OutputIsCout() const { return (m_outStream == &std::cout); } /** * Write or cache the output, as appropriate. **/ void Write(int sourceId,const std::string& output,const std::string& debug="") { #ifdef WITH_THREADS boost::mutex::scoped_lock lock(m_mutex); #endif if (sourceId == m_nextOutput) { //This is the one we were expecting *m_outStream << output << std::flush; *m_debugStream << debug << std::flush; ++m_nextOutput; //see if there's any more std::map::iterator iter; while ((iter = m_outputs.find(m_nextOutput)) != m_outputs.end()) { *m_outStream << iter->second << std::flush; ++m_nextOutput; std::map::iterator debugIter = m_debugs.find(iter->first); m_outputs.erase(iter); if (debugIter != m_debugs.end()) { *m_debugStream << debugIter->second << std::flush; m_debugs.erase(debugIter); } } } else { //save for later m_outputs[sourceId] = output; m_debugs[sourceId] = debug; } } private: std::map m_outputs; std::map m_debugs; int m_nextOutput; std::ostream* m_outStream; std::ostream* m_debugStream; bool m_isHoldingOutputStream; bool m_isHoldingDebugStream; #ifdef WITH_THREADS boost::mutex m_mutex; #endif public: void SetOutputStream(std::ostream* outStream) { m_outStream = outStream; } }; } // namespace Moses #endif