mirror of https://github.com/AxioDL/metaforce.git
169 lines
5.1 KiB
C++
169 lines
5.1 KiB
C++
#ifndef BLENDERCONNECTION_HPP
|
|
#define BLENDERCONNECTION_HPP
|
|
|
|
#if _WIN32
|
|
#define _WIN32_LEAN_AND_MEAN 1
|
|
#include <windows.h>
|
|
#else
|
|
#include <unistd.h>
|
|
#endif
|
|
|
|
#include <stdint.h>
|
|
#include <stdio.h>
|
|
#include <string>
|
|
#include <functional>
|
|
#include <mutex>
|
|
|
|
#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<char*(uint32_t)> bufGetter,
|
|
const std::string& expectedType,
|
|
const std::string& platform,
|
|
bool bigEndian=false);
|
|
|
|
class PyOutStream : public std::ostream
|
|
{
|
|
friend class BlenderConnection;
|
|
std::unique_lock<std::mutex> 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
|