#ifndef BLENDERCONNECTION_HPP #define BLENDERCONNECTION_HPP #if _WIN32 #ifndef WIN32_LEAN_AND_MEAN #define WIN32_LEAN_AND_MEAN 1 #endif #ifndef NOMINMAX #define NOMINMAX #endif #include #else #include #endif #include #include #include #include #include #include "HECL/HECL.hpp" #include namespace HECL { extern LogVisor::LogModule BlenderLog; extern class BlenderConnection* SharedBlenderConnection; class BlenderConnection { public: enum BlendType { TypeNone, TypeMesh, TypeActor, TypeArea }; private: bool m_lock = false; #if _WIN32 HANDLE m_blenderProc; #else pid_t m_blenderProc; #endif int m_readpipe[2]; int m_writepipe[2]; BlendType m_loadedType = TypeNone; ProjectPath m_loadedBlend; std::string m_startupBlend; size_t _readLine(char* buf, size_t bufSz); size_t _writeLine(const char* buf); size_t _readBuf(void* buf, size_t len); size_t _writeBuf(const void* buf, size_t len); void _closePipe(); public: BlenderConnection(int verbosityLevel=1); ~BlenderConnection(); bool createBlend(const ProjectPath& path, BlendType type); BlendType getBlendType() const {return m_loadedType;} bool openBlend(const ProjectPath& path, bool force=false); bool saveBlend(); void deleteBlend(); class PyOutStream : public std::ostream { friend class BlenderConnection; 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_parent || !m_parent.m_parent->m_lock) BlenderLog.report(LogVisor::FatalError, "lock not held for PyOutStream writing"); if (ch != traits_type::eof() && ch != '\n' && ch != '\0') { m_lineBuf += char_type(ch); return ch; } //printf("FLUSHING %s\n", m_lineBuf.c_str()); 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) : std::ostream(&m_sbuf), m_parent(parent), m_deleteOnError(deleteOnError), m_sbuf(*this, deleteOnError) { m_parent->m_lock = true; 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) : std::ostream(&m_sbuf), m_parent(other.m_parent), m_sbuf(std::move(other.m_sbuf)) {other.m_parent = nullptr;} ~PyOutStream() {close();} void close() { if (m_parent && m_parent->m_lock) { 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_parent->m_lock = false; } } #if __GNUC__ __attribute__((__format__ (__printf__, 2, 3))) #endif void format(const char* fmt, ...) { if (!m_parent || !m_parent->m_lock) BlenderLog.report(LogVisor::FatalError, "lock not held for PyOutStream::format()"); va_list ap; va_start(ap, fmt); char* result = nullptr; #ifdef _WIN32 int length = _vscprintf(fmt, ap); result = (char*)malloc(length); vsnprintf(result, length, fmt, ap); #else int length = vasprintf(&result, fmt, ap); #endif va_end(ap); if (length > 0) this->write(result, length); free(result); } void linkBlend(const std::string& target, const std::string& objName, bool link=true); class ANIMOutStream { BlenderConnection* m_parent; unsigned m_curCount = 0; unsigned m_totalCount = 0; bool m_inCurve = false; public: enum CurveType { CurveRotate, CurveTranslate, CurveScale }; ANIMOutStream(BlenderConnection* parent) : m_parent(parent) { m_parent->_writeLine("PYANIM"); char readBuf[16]; m_parent->_readLine(readBuf, 16); if (strcmp(readBuf, "ANIMREADY")) BlenderLog.report(LogVisor::FatalError, "unable to open ANIMOutStream"); } ~ANIMOutStream() { char tp = -1; m_parent->_writeBuf(&tp, 1); char readBuf[16]; m_parent->_readLine(readBuf, 16); if (strcmp(readBuf, "ANIMDONE")) BlenderLog.report(LogVisor::FatalError, "unable to close ANIMOutStream"); } void changeCurve(CurveType type, unsigned crvIdx, unsigned keyCount) { if (m_curCount != m_totalCount) BlenderLog.report(LogVisor::FatalError, "incomplete ANIMOutStream for change"); m_curCount = 0; m_totalCount = keyCount; char tp = char(type); m_parent->_writeBuf(&tp, 1); struct { uint32_t ci; uint32_t kc; } info = {uint32_t(crvIdx), uint32_t(keyCount)}; m_parent->_writeBuf(reinterpret_cast(&info), 8); m_inCurve = true; } void write(unsigned frame, float val) { if (!m_inCurve) BlenderLog.report(LogVisor::FatalError, "changeCurve not called before write"); if (m_curCount < m_totalCount) { struct { uint32_t frm; float val; } key = {uint32_t(frame), val}; m_parent->_writeBuf(reinterpret_cast(&key), 8); ++m_curCount; } else BlenderLog.report(LogVisor::FatalError, "ANIMOutStream keyCount overflow"); } }; ANIMOutStream beginANIMCurve() { return ANIMOutStream(m_parent); } }; PyOutStream beginPythonOut(bool deleteOnError=false) { if (m_lock) BlenderLog.report(LogVisor::FatalError, "lock already held for BlenderConnection::beginPythonOut()"); return PyOutStream(this, deleteOnError); } class DataStream { friend class BlenderConnection; BlenderConnection* m_parent; DataStream(BlenderConnection* parent) : m_parent(parent) { m_parent->m_lock = true; m_parent->_writeLine("DATABEGIN"); char readBuf[16]; m_parent->_readLine(readBuf, 16); if (strcmp(readBuf, "READY")) BlenderLog.report(LogVisor::FatalError, "unable to open DataStream with blender"); } public: DataStream(const DataStream& other) = delete; DataStream(DataStream&& other) : m_parent(other.m_parent) {other.m_parent = nullptr;} ~DataStream() {close();} void close() { if (m_parent && m_parent->m_lock) { m_parent->_writeLine("DATAEND"); char readBuf[16]; m_parent->_readLine(readBuf, 16); if (strcmp(readBuf, "DONE")) BlenderLog.report(LogVisor::FatalError, "unable to close DataStream with blender"); m_parent->m_lock = false; } } std::vector getMeshList() { m_parent->_writeLine("MESHLIST"); uint32_t count; m_parent->_readBuf(&count, 4); std::vector retval; retval.reserve(count); for (uint32_t i=0 ; i_readLine(name, 128); retval.push_back(name); } return retval; } /* Intermediate mesh representation prepared by blender from a single mesh object */ struct Mesh { struct Vector2f { atVec2f val; Vector2f(BlenderConnection& conn) {conn._readBuf(&val, 8);} }; struct Vector3f { atVec3f val; Vector3f(BlenderConnection& conn) {conn._readBuf(&val, 12);} }; struct Index { uint32_t val; Index(BlenderConnection& conn) {conn._readBuf(&val, 4);} }; /* Cumulative AABB */ Vector3f aabbMin; Vector3f aabbMax; /* HECL source of each material */ struct Material { std::string name; std::string source; std::vector texs; Material(BlenderConnection& conn); }; std::vector> materialSets; /* Vertex buffer data */ std::vector pos; std::vector norm; uint32_t colorLayerCount = 0; std::vector color; uint32_t uvLayerCount = 0; std::vector uv; /* Skinning data */ std::vector boneNames; struct SkinBind { uint32_t boneIdx; float weight; SkinBind(BlenderConnection& conn) {conn._readBuf(&boneIdx, 8);} }; std::vector> skins; /* Islands of the same material/skinBank are represented here */ struct Surface { Vector3f centroid; Index materialIdx; Vector3f aabbMin; Vector3f aabbMax; Vector3f reflectionNormal; uint32_t skinBankIdx; /* Vertex indexing data (all primitives joined as degenerate tri-strip) */ struct Vert { uint32_t iPos; uint32_t iNorm; uint32_t iColor[4] = {uint32_t(-1)}; uint32_t iUv[8] = {uint32_t(-1)}; uint32_t iSkin; uint32_t iBankSkin = -1; Vert(BlenderConnection& conn, const Mesh& parent); }; std::vector verts; Surface(BlenderConnection& conn, Mesh& parent, int skinSlotCount); }; std::vector surfaces; struct SkinBanks { std::vector> banks; std::vector>::iterator addSkinBank(int skinSlotCount) { banks.emplace_back(); banks.back().reserve(skinSlotCount); return banks.end() - 1; } uint32_t addSurface(const Surface& surf, int skinSlotCount); } skinBanks; Mesh(BlenderConnection& conn, int skinSlotCount); }; /* Compile mesh by context */ Mesh compileMesh(int skinSlotCount=10) { char req[128]; snprintf(req, 128, "MESHCOMPILE %d", skinSlotCount); m_parent->_writeLine(req); char readBuf[256]; m_parent->_readLine(readBuf, 256); if (strcmp(readBuf, "OK")) BlenderLog.report(LogVisor::FatalError, "unable to cook mesh: %s", readBuf); return Mesh(*m_parent, skinSlotCount); } /* Compile mesh by name */ Mesh compileMesh(const std::string& name, int skinSlotCount=10) { char req[128]; snprintf(req, 128, "MESHCOMPILENAME %s %d", name.c_str(), skinSlotCount); m_parent->_writeLine(req); char readBuf[256]; m_parent->_readLine(readBuf, 256); if (strcmp(readBuf, "OK")) BlenderLog.report(LogVisor::FatalError, "unable to cook mesh '%s': %s", name.c_str(), readBuf); return Mesh(*m_parent, skinSlotCount); } /* Compile all meshes into one */ Mesh compileAllMeshes(int skinSlotCount=10, float maxOctantLength=5.0) { char req[128]; snprintf(req, 128, "MESHCOMPILEALL %d %f", skinSlotCount, maxOctantLength); m_parent->_writeLine(req); char readBuf[256]; m_parent->_readLine(readBuf, 256); if (strcmp(readBuf, "OK")) BlenderLog.report(LogVisor::FatalError, "unable to cook all meshes: %s", readBuf); return Mesh(*m_parent, skinSlotCount); } }; DataStream beginData() { if (m_lock) BlenderLog.report(LogVisor::FatalError, "lock already held for BlenderConnection::beginDataIn()"); return DataStream(this); } void quitBlender(); static BlenderConnection& SharedConnection() { if (!SharedBlenderConnection) SharedBlenderConnection = new BlenderConnection(HECL::VerbosityLevel); return *SharedBlenderConnection; } void closeStream() { if (m_lock) deleteBlend(); } static void Shutdown() { if (SharedBlenderConnection) { SharedBlenderConnection->closeStream(); SharedBlenderConnection->quitBlender(); delete SharedBlenderConnection; SharedBlenderConnection = nullptr; BlenderLog.report(LogVisor::Info, "BlenderConnection Shutdown Successful"); } } }; } #endif // BLENDERCONNECTION_HPP