#ifndef BLENDERCONNECTION_HPP #define BLENDERCONNECTION_HPP #if _WIN32 #define _WIN32_LEAN_AND_MEAN 1 #include #else #include #endif #include #include #include #include #include #include "HECL/HECL.hpp" namespace HECL { extern LogVisor::LogModule BlenderLog; extern class BlenderConnection* SharedBlenderConnection; class BlenderConnection { std::mutex m_lock; #if _WIN32 HANDLE m_blenderProc; HANDLE m_readpipe[2]; HANDLE m_writepipe[2]; #else pid_t m_blenderProc; int m_readpipe[2]; int m_writepipe[2]; #endif std::string m_loadedBlend; size_t _readLine(char* buf, size_t bufSz); size_t _writeLine(const char* buf); size_t _readBuf(char* buf, size_t len); size_t _writeBuf(const char* buf, size_t len); void _closePipe(); public: BlenderConnection(bool silenceBlender=false); ~BlenderConnection(); bool createBlend(const SystemString& path); bool openBlend(const SystemString& path); bool saveBlend(); void deleteBlend(); enum CookPlatform { CP_MODERN = 0, CP_GX = 1, }; bool cookBlend(std::function bufGetter, const std::string& expectedType, const std::string& platform, bool bigEndian=false); class PyOutStream : public std::ostream { friend class BlenderConnection; std::unique_lock m_lk; BlenderConnection* m_parent; bool m_deleteOnError; struct StreamBuf : std::streambuf { PyOutStream& m_parent; std::string m_lineBuf; bool m_deleteOnError; StreamBuf(PyOutStream& parent, bool deleteOnError) : m_parent(parent), m_deleteOnError(deleteOnError) {} StreamBuf(const StreamBuf& other) = delete; StreamBuf(StreamBuf&& other) = default; int_type overflow(int_type ch) { if (!m_parent.m_lk) BlenderLog.report(LogVisor::FatalError, "lock not held for PyOutStream writing"); if (ch != traits_type::eof() && ch != '\n') { m_lineBuf += char_type(ch); return ch; } m_parent.m_parent->_writeLine(m_lineBuf.c_str()); char readBuf[16]; m_parent.m_parent->_readLine(readBuf, 16); if (strcmp(readBuf, "OK")) { if (m_deleteOnError) m_parent.m_parent->deleteBlend(); BlenderLog.report(LogVisor::FatalError, "error sending '%s' to blender", m_lineBuf.c_str()); } m_lineBuf.clear(); return ch; } } m_sbuf; PyOutStream(BlenderConnection* parent, bool deleteOnError) : m_lk(parent->m_lock), m_parent(parent), m_sbuf(*this, deleteOnError), m_deleteOnError(deleteOnError), std::ostream(&m_sbuf) { m_parent->_writeLine("PYBEGIN"); char readBuf[16]; m_parent->_readLine(readBuf, 16); if (strcmp(readBuf, "READY")) BlenderLog.report(LogVisor::FatalError, "unable to open PyOutStream with blender"); } public: PyOutStream(const PyOutStream& other) = delete; PyOutStream(PyOutStream&& other) : m_lk(std::move(other.m_lk)), m_parent(other.m_parent), m_sbuf(std::move(other.m_sbuf)) {other.m_parent = nullptr;} ~PyOutStream() {close();} void close() { if (m_lk) { if (m_parent) { m_parent->_writeLine("PYEND"); char readBuf[16]; m_parent->_readLine(readBuf, 16); if (strcmp(readBuf, "DONE")) BlenderLog.report(LogVisor::FatalError, "unable to close PyOutStream with blender"); } m_lk.unlock(); } } void format(const char* fmt, ...) { if (!m_lk) BlenderLog.report(LogVisor::FatalError, "lock not held for PyOutStream::format()"); va_list ap; va_start(ap, fmt); char* result = nullptr; int length = vasprintf(&result, fmt, ap); if (length > 0) this->write(result, length); free(result); va_end(ap); } void linkBlend(const SystemString& target, const std::string& objName, bool link=true); }; inline PyOutStream beginPythonOut(bool deleteOnError=false) { return PyOutStream(this, deleteOnError); } void quitBlender(); static inline BlenderConnection& SharedConnection() { if (!SharedBlenderConnection) SharedBlenderConnection = new BlenderConnection(HECL::VerbosityLevel?false:true); return *SharedBlenderConnection; } static inline void Shutdown() { delete SharedBlenderConnection; } }; } #endif // BLENDERCONNECTION_HPP