Merge branch 'master' of git://git.ucc.asn.au/tpg/acess2
[tpg/acess2.git] / Usermode / Libraries / libaxwin4.so_src / ipc.cpp
diff --git a/Usermode/Libraries/libaxwin4.so_src/ipc.cpp b/Usermode/Libraries/libaxwin4.so_src/ipc.cpp
new file mode 100644 (file)
index 0000000..d0455f5
--- /dev/null
@@ -0,0 +1,178 @@
+/*
+ * AxWin4 Interface Library
+ * - By John Hodge (thePowersGang)
+ *
+ * ipc.c
+ * - IPC Abstraction
+ */
+#include <axwin4/axwin.h>
+#include "include/common.hpp"
+#include "include/IIPCChannel.hpp"
+#include "include/CIPCChannel_AcessIPCPipe.hpp"
+#include <ipc_proto.hpp>
+#include <algorithm>
+#include <mutex>
+#include <stdexcept>
+
+#include <cstring>
+#include <cstdio>
+
+namespace AxWin {
+
+IIPCChannel*   gIPCChannel;
+::std::mutex   glSyncReply;
+bool   gSyncReplyActive;
+bool   gSyncReplyValid;
+CDeserialiser  gSyncReplyBuf;
+
+extern "C" bool AxWin4_Connect(const char *URI)
+{
+       _SysDebug("AxWin4_Connect('%s')", URI);
+       if( gIPCChannel ) {
+               return false;
+       }
+       try {
+               if( strncmp(URI, "ipcpipe://", 3+4+3) == 0 )
+               {
+                       gIPCChannel = new CIPCChannel_AcessIPCPipe(URI+3+4+3);
+               }
+               else
+               {
+                       _SysDebug("Unknown protocol");
+                       return false;
+               }
+       }
+       catch( const ::std::exception& e )
+       {
+               fprintf(stderr, "AxWin4_Connect: %s\n", e.what());
+               return false;
+       }
+       return true;
+}
+
+extern "C" bool AxWin4_PeekEventQueue(void)
+{
+       return false;
+}
+
+extern "C" bool AxWin4_WaitEventQueue(uint64_t Timeout)
+{
+       return AxWin4_WaitEventQueueSelect(0, NULL, NULL, NULL, Timeout);
+}
+
+extern "C" bool AxWin4_WaitEventQueueSelect(int nFDs, fd_set *rfds, fd_set *wfds, fd_set *efds, uint64_t Timeout)
+{
+       fd_set  local_rfds;
+       if( !rfds ) {
+               FD_ZERO(&local_rfds);
+               rfds = &local_rfds;
+       }
+       
+       int64_t select_timeout = Timeout;
+       int64_t *select_timeout_p = (Timeout ? &select_timeout : 0);
+       
+       nFDs = ::std::max(nFDs, gIPCChannel->FillSelect(*rfds));
+       _SysSelect(nFDs, rfds, wfds, efds, select_timeout_p, 0);
+       return gIPCChannel->HandleSelect(*rfds);
+}
+
+void SendMessage(CSerialiser& message)
+{
+       gIPCChannel->Send(message);
+}
+void RecvMessage(CDeserialiser& message)
+{
+       uint8_t id = message.ReadU8();
+       _SysDebug("RecvMessage: id=%i", id);
+       switch(id)
+       {
+       case IPCMSG_PING:
+               // If we hear ping, we must pong
+               {
+               CSerialiser     pong;
+               pong.WriteU8(IPCMSG_REPLY);
+               pong.WriteU8(IPCMSG_PING);
+               SendMessage(pong);
+               }
+               break;
+       case IPCMSG_REPLY:
+               // Flag reply and take a copy of this message
+               if( !gSyncReplyActive )
+               {
+                       _SysDebug("Unexpected reply message %i", message.ReadU8());
+               }
+               else if( gSyncReplyValid )
+               {
+                       // Oh... that was unexpected
+                       _SysDebug("Unexpected second reply message %i", message.ReadU8());
+               }
+               else
+               {
+                       gSyncReplyValid = true;
+                       gSyncReplyBuf = message;
+               }
+               break;
+       // TODO: Handle messages from server (input events, IPC)
+       // TODO: If an event is currently being processed, save the message in a queue to be handled when processing is complete
+       // - This will prevent deep recursion (and make server errors aparent)
+       case IPCMSG_INPUTEVENT:
+               _SysDebug("TODO: Input events");
+               break;
+       default:
+               _SysDebug("TODO: RecvMessage(%i)", id);
+               break;
+       }
+}
+
+CDeserialiser GetSyncReply(CSerialiser& request, unsigned int Message)
+{
+       ::std::lock_guard<std::mutex>   lock(glSyncReply);
+       gSyncReplyActive = true;
+       gSyncReplyValid = false;
+       // Send once lock is acquired
+       SendMessage(request);
+       
+       while( !gSyncReplyValid )
+       {
+               // Tick along
+               if( !AxWin4_WaitEventQueue(0) )
+                       throw ::std::runtime_error("Connection dropped while waiting for reply");
+       }
+       gSyncReplyActive = false;
+       
+       uint8_t id = gSyncReplyBuf.ReadU8();
+       if( id != Message )
+       {
+               _SysDebug("Unexpected reply message id=%i, expected %i",
+                       id, Message);
+               throw ::std::runtime_error("Sequencing error, unexpected reply");
+       }
+       
+       // Use move to avoid copying
+       return ::std::move(gSyncReplyBuf);
+}
+
+extern "C" void AxWin4_GetScreenDimensions(unsigned int ScreenIndex, unsigned int *Width, unsigned int *Height)
+{
+       CSerialiser     req;
+       req.WriteU8(IPCMSG_GETGLOBAL);
+       req.WriteU16(IPC_GLOBATTR_SCREENDIMS);
+       req.WriteU8(ScreenIndex);
+       
+       CDeserialiser   response = GetSyncReply(req, IPCMSG_GETGLOBAL);
+       if( response.ReadU16() != IPC_GLOBATTR_SCREENDIMS ) {
+               
+       }
+       
+       *Width = response.ReadU16();
+       *Height = response.ReadU16();
+       
+       _SysDebug("AxWin4_GetScreenDimensions: %i = %ix%i", ScreenIndex, *Width, *Height);
+}
+
+IIPCChannel::~IIPCChannel()
+{
+}
+
+};     // namespace AxWin
+

UCC git Repository :: git.ucc.asn.au