123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265 |
- /*
- ==============================================================================
- This file is part of the JUCE library.
- Copyright (c) 2015 - ROLI Ltd.
- Permission is granted to use this software under the terms of either:
- a) the GPL v2 (or any later version)
- b) the Affero GPL v3
- Details of these licenses can be found at: www.gnu.org/licenses
- JUCE 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 General Public License for more details.
- ------------------------------------------------------------------------------
- To release a closed-source product which uses JUCE, commercial licenses are
- available: visit www.juce.com for more information.
- ==============================================================================
- */
- enum { magicMastSlaveConnectionHeader = 0x712baf04 };
- static const char* startMessage = "__ipc_st";
- static const char* killMessage = "__ipc_k_";
- static const char* pingMessage = "__ipc_p_";
- enum { specialMessageSize = 8, defaultTimeoutMs = 8000 };
- static String getCommandLinePrefix (const String& commandLineUniqueID)
- {
- return "--" + commandLineUniqueID + ":";
- }
- //==============================================================================
- // This thread sends and receives ping messages every second, so that it
- // can find out if the other process has stopped running.
- struct ChildProcessPingThread : public Thread,
- private AsyncUpdater
- {
- ChildProcessPingThread (int timeout) : Thread ("IPC ping"), timeoutMs (timeout)
- {
- pingReceived();
- }
- static bool isPingMessage (const MemoryBlock& m) noexcept
- {
- return memcmp (m.getData(), pingMessage, specialMessageSize) == 0;
- }
- void pingReceived() noexcept { countdown = timeoutMs / 1000 + 1; }
- void triggerConnectionLostMessage() { triggerAsyncUpdate(); }
- virtual bool sendPingMessage (const MemoryBlock&) = 0;
- virtual void pingFailed() = 0;
- int timeoutMs;
- private:
- Atomic<int> countdown;
- void handleAsyncUpdate() override { pingFailed(); }
- void run() override
- {
- while (! threadShouldExit())
- {
- if (--countdown <= 0 || ! sendPingMessage (MemoryBlock (pingMessage, specialMessageSize)))
- {
- triggerConnectionLostMessage();
- break;
- }
- wait (1000);
- }
- }
- JUCE_DECLARE_NON_COPYABLE_WITH_LEAK_DETECTOR (ChildProcessPingThread)
- };
- //==============================================================================
- struct ChildProcessMaster::Connection : public InterprocessConnection,
- private ChildProcessPingThread
- {
- Connection (ChildProcessMaster& m, const String& pipeName, int timeout)
- : InterprocessConnection (false, magicMastSlaveConnectionHeader),
- ChildProcessPingThread (timeout),
- owner (m)
- {
- if (createPipe (pipeName, timeoutMs))
- startThread (4);
- }
- ~Connection()
- {
- stopThread (10000);
- }
- private:
- void connectionMade() override {}
- void connectionLost() override { owner.handleConnectionLost(); }
- bool sendPingMessage (const MemoryBlock& m) override { return owner.sendMessageToSlave (m); }
- void pingFailed() override { connectionLost(); }
- void messageReceived (const MemoryBlock& m) override
- {
- pingReceived();
- if (m.getSize() != specialMessageSize || ! isPingMessage (m))
- owner.handleMessageFromSlave (m);
- }
- ChildProcessMaster& owner;
- JUCE_DECLARE_NON_COPYABLE_WITH_LEAK_DETECTOR (Connection)
- };
- //==============================================================================
- ChildProcessMaster::ChildProcessMaster() {}
- ChildProcessMaster::~ChildProcessMaster()
- {
- if (connection != nullptr)
- {
- sendMessageToSlave (MemoryBlock (killMessage, specialMessageSize));
- connection->disconnect();
- connection = nullptr;
- }
- }
- void ChildProcessMaster::handleConnectionLost() {}
- bool ChildProcessMaster::sendMessageToSlave (const MemoryBlock& mb)
- {
- if (connection != nullptr)
- return connection->sendMessage (mb);
- jassertfalse; // this can only be used when the connection is active!
- return false;
- }
- bool ChildProcessMaster::launchSlaveProcess (const File& executable, const String& commandLineUniqueID, int timeoutMs, int streamFlags)
- {
- connection = nullptr;
- jassert (childProcess.kill());
- const String pipeName ("p" + String::toHexString (Random().nextInt64()));
- StringArray args;
- args.add (executable.getFullPathName());
- args.add (getCommandLinePrefix (commandLineUniqueID) + pipeName);
- if (childProcess.start (args, streamFlags))
- {
- connection = new Connection (*this, pipeName, timeoutMs <= 0 ? defaultTimeoutMs : timeoutMs);
- if (connection->isConnected())
- {
- sendMessageToSlave (MemoryBlock (startMessage, specialMessageSize));
- return true;
- }
- connection = nullptr;
- }
- return false;
- }
- //==============================================================================
- struct ChildProcessSlave::Connection : public InterprocessConnection,
- private ChildProcessPingThread
- {
- Connection (ChildProcessSlave& p, const String& pipeName, int timeout)
- : InterprocessConnection (false, magicMastSlaveConnectionHeader),
- ChildProcessPingThread (timeout),
- owner (p)
- {
- connectToPipe (pipeName, timeoutMs);
- startThread (4);
- }
- ~Connection()
- {
- stopThread (10000);
- }
- private:
- ChildProcessSlave& owner;
- void connectionMade() override {}
- void connectionLost() override { owner.handleConnectionLost(); }
- bool sendPingMessage (const MemoryBlock& m) override { return owner.sendMessageToMaster (m); }
- void pingFailed() override { connectionLost(); }
- void messageReceived (const MemoryBlock& m) override
- {
- pingReceived();
- if (m.getSize() == specialMessageSize)
- {
- if (isPingMessage (m))
- return;
- if (memcmp (m.getData(), killMessage, specialMessageSize) == 0)
- {
- triggerConnectionLostMessage();
- return;
- }
- if (memcmp (m.getData(), startMessage, specialMessageSize) == 0)
- {
- owner.handleConnectionMade();
- return;
- }
- }
- owner.handleMessageFromMaster (m);
- }
- JUCE_DECLARE_NON_COPYABLE_WITH_LEAK_DETECTOR (Connection)
- };
- //==============================================================================
- ChildProcessSlave::ChildProcessSlave() {}
- ChildProcessSlave::~ChildProcessSlave() {}
- void ChildProcessSlave::handleConnectionMade() {}
- void ChildProcessSlave::handleConnectionLost() {}
- bool ChildProcessSlave::sendMessageToMaster (const MemoryBlock& mb)
- {
- if (connection != nullptr)
- return connection->sendMessage (mb);
- jassertfalse; // this can only be used when the connection is active!
- return false;
- }
- bool ChildProcessSlave::initialiseFromCommandLine (const String& commandLine,
- const String& commandLineUniqueID,
- int timeoutMs)
- {
- String prefix (getCommandLinePrefix (commandLineUniqueID));
- if (commandLine.trim().startsWith (prefix))
- {
- String pipeName (commandLine.fromFirstOccurrenceOf (prefix, false, false)
- .upToFirstOccurrenceOf (" ", false, false).trim());
- if (pipeName.isNotEmpty())
- {
- connection = new Connection (*this, pipeName, timeoutMs <= 0 ? defaultTimeoutMs : timeoutMs);
- if (! connection->isConnected())
- connection = nullptr;
- }
- }
- return connection != nullptr;
- }
|