Merge pull request #1564 from Diapolo/IPC-server

IPC-server hardening and update
This commit is contained in:
Wladimir J. van der Laan 2012-07-17 00:54:12 -07:00
commit d47afc7f4c
4 changed files with 75 additions and 39 deletions

View File

@ -126,16 +126,24 @@ int main(int argc, char *argv[])
const char *strURI = argv[i]; const char *strURI = argv[i];
try { try {
boost::interprocess::message_queue mq(boost::interprocess::open_only, BITCOINURI_QUEUE_NAME); boost::interprocess::message_queue mq(boost::interprocess::open_only, BITCOINURI_QUEUE_NAME);
if(mq.try_send(strURI, strlen(strURI), 0)) if (mq.try_send(strURI, strlen(strURI), 0))
// if URI could be sent to the message queue exit here
exit(0); exit(0);
else else
// if URI could not be sent to the message queue do a normal Bitcoin-Qt startup
break; break;
} }
catch (boost::interprocess::interprocess_exception &ex) { catch (boost::interprocess::interprocess_exception &ex) {
// don't log the "file not found" exception, because that's normal for
// the first start of the first instance
if (ex.get_error_code() != boost::interprocess::not_found_error)
{
printf("main() - boost interprocess exception #%d: %s\n", ex.get_error_code(), ex.what());
break; break;
} }
} }
} }
}
#endif #endif
// Internal string conversion is all UTF-8 // Internal string conversion is all UTF-8
@ -278,6 +286,8 @@ int main(int argc, char *argv[])
mq.try_send(strURI, strlen(strURI), 0); mq.try_send(strURI, strlen(strURI), 0);
} }
catch (boost::interprocess::interprocess_exception &ex) { catch (boost::interprocess::interprocess_exception &ex) {
printf("main() - boost interprocess exception #%d: %s\n", ex.get_error_code(), ex.what());
break;
} }
} }
} }

View File

@ -4,10 +4,10 @@
/* Milliseconds between model updates */ /* Milliseconds between model updates */
static const int MODEL_UPDATE_DELAY = 500; static const int MODEL_UPDATE_DELAY = 500;
/* Maximum passphrase length */ /* AskPassphraseDialog -- Maximum passphrase length */
static const int MAX_PASSPHRASE_SIZE = 1024; static const int MAX_PASSPHRASE_SIZE = 1024;
/* Size of icons in status bar */ /* BitcoinGUI -- Size of icons in status bar */
static const int STATUSBAR_ICONSIZE = 16; static const int STATUSBAR_ICONSIZE = 16;
/* Invalid field background style */ /* Invalid field background style */

View File

@ -2,80 +2,96 @@
// Distributed under the MIT/X11 software license, see the accompanying // Distributed under the MIT/X11 software license, see the accompanying
// file license.txt or http://www.opensource.org/licenses/mit-license.php. // file license.txt or http://www.opensource.org/licenses/mit-license.php.
#include <boost/algorithm/string.hpp> #include "qtipcserver.h"
#include <boost/interprocess/ipc/message_queue.hpp> #include "guiconstants.h"
#include <boost/tokenizer.hpp> #include "ui_interface.h"
#include "util.h"
#include <boost/date_time/posix_time/posix_time.hpp> #include <boost/date_time/posix_time/posix_time.hpp>
#include <boost/interprocess/ipc/message_queue.hpp>
#include <boost/version.hpp> #include <boost/version.hpp>
#if defined(WIN32) && (!defined(BOOST_INTERPROCESS_HAS_WINDOWS_KERNEL_BOOTTIME) || !defined(BOOST_INTERPROCESS_HAS_KERNEL_BOOTTIME) || BOOST_VERSION < 104900) #if defined(WIN32) && (!defined(BOOST_INTERPROCESS_HAS_WINDOWS_KERNEL_BOOTTIME) || !defined(BOOST_INTERPROCESS_HAS_KERNEL_BOOTTIME) || BOOST_VERSION < 104900)
#warning Compiling without BOOST_INTERPROCESS_HAS_WINDOWS_KERNEL_BOOTTIME and BOOST_INTERPROCESS_HAS_KERNEL_BOOTTIME uncommented in boost/interprocess/detail/tmp_dir_helpers.hpp or using a boost version before 1.49 may have unintended results see svn.boost.org/trac/boost/ticket/5392 #warning Compiling without BOOST_INTERPROCESS_HAS_WINDOWS_KERNEL_BOOTTIME and BOOST_INTERPROCESS_HAS_KERNEL_BOOTTIME uncommented in boost/interprocess/detail/tmp_dir_helpers.hpp or using a boost version before 1.49 may have unintended results see svn.boost.org/trac/boost/ticket/5392
#endif #endif
#include "ui_interface.h" using namespace boost;
#include "qtipcserver.h"
#include "util.h"
using namespace boost::interprocess; using namespace boost::interprocess;
using namespace boost::posix_time; using namespace boost::posix_time;
using namespace boost;
using namespace std; static void ipcThread2(void* pArg);
#ifdef MAC_OSX #ifdef MAC_OSX
// URI handling not implemented on OSX yet // URI handling not implemented on OSX yet
void ipcInit() { } void ipcInit() { }
void ipcShutdown() { }
#else #else
void ipcShutdown() static void ipcThread(void* pArg)
{ {
message_queue::remove(BITCOINURI_QUEUE_NAME); IMPLEMENT_RANDOMIZE_STACK(ipcThread(pArg));
}
void ipcThread(void* parg)
{
// Make this thread recognisable as the GUI-IPC thread // Make this thread recognisable as the GUI-IPC thread
RenameThread("bitcoin-gui-ipc"); RenameThread("bitcoin-gui-ipc");
message_queue* mq = (message_queue*)parg; try
char strBuf[257]; {
size_t nSize; ipcThread2(pArg);
unsigned int nPriority; }
catch (std::exception& e) {
PrintExceptionContinue(&e, "ipcThread()");
} catch (...) {
PrintExceptionContinue(NULL, "ipcThread()");
}
printf("ipcThread exited\n");
}
static void ipcThread2(void* pArg)
{
printf("ipcThread started\n");
message_queue* mq = (message_queue*)pArg;
char buffer[MAX_URI_LENGTH + 1] = "";
size_t nSize = 0;
unsigned int nPriority = 0;
loop loop
{ {
ptime d = boost::posix_time::microsec_clock::universal_time() + millisec(100); ptime d = boost::posix_time::microsec_clock::universal_time() + millisec(100);
if(mq->timed_receive(&strBuf, sizeof(strBuf), nSize, nPriority, d)) if (mq->timed_receive(&buffer, sizeof(buffer), nSize, nPriority, d))
{ {
uiInterface.ThreadSafeHandleURI(std::string(strBuf, nSize)); uiInterface.ThreadSafeHandleURI(std::string(buffer, nSize));
Sleep(1000); Sleep(1000);
} }
if (fShutdown) if (fShutdown)
{
ipcShutdown();
break; break;
} }
}
ipcShutdown(); // Remove message queue
message_queue::remove(BITCOINURI_QUEUE_NAME);
// Cleanup allocated memory
delete mq;
} }
void ipcInit() void ipcInit()
{ {
message_queue* mq; message_queue* mq = NULL;
char strBuf[257]; char buffer[MAX_URI_LENGTH + 1] = "";
size_t nSize; size_t nSize = 0;
unsigned int nPriority; unsigned int nPriority = 0;
try { try {
mq = new message_queue(open_or_create, BITCOINURI_QUEUE_NAME, 2, 256); mq = new message_queue(open_or_create, BITCOINURI_QUEUE_NAME, 2, MAX_URI_LENGTH);
// Make sure we don't lose any bitcoin: URIs // Make sure we don't lose any bitcoin: URIs
for (int i = 0; i < 2; i++) for (int i = 0; i < 2; i++)
{ {
ptime d = boost::posix_time::microsec_clock::universal_time() + millisec(1); ptime d = boost::posix_time::microsec_clock::universal_time() + millisec(1);
if(mq->timed_receive(&strBuf, sizeof(strBuf), nSize, nPriority, d)) if (mq->timed_receive(&buffer, sizeof(buffer), nSize, nPriority, d))
{ {
uiInterface.ThreadSafeHandleURI(std::string(strBuf, nSize)); uiInterface.ThreadSafeHandleURI(std::string(buffer, nSize));
} }
else else
break; break;
@ -83,14 +99,19 @@ void ipcInit()
// Make sure only one bitcoin instance is listening // Make sure only one bitcoin instance is listening
message_queue::remove(BITCOINURI_QUEUE_NAME); message_queue::remove(BITCOINURI_QUEUE_NAME);
mq = new message_queue(open_or_create, BITCOINURI_QUEUE_NAME, 2, 256); delete mq;
mq = new message_queue(open_or_create, BITCOINURI_QUEUE_NAME, 2, MAX_URI_LENGTH);
} }
catch (interprocess_exception &ex) { catch (interprocess_exception &ex) {
printf("ipcInit() - boost interprocess exception #%d: %s\n", ex.get_error_code(), ex.what());
return; return;
} }
if (!CreateThread(ipcThread, mq)) if (!CreateThread(ipcThread, mq))
{ {
delete mq; delete mq;
return;
} }
} }

View File

@ -1,4 +1,9 @@
#ifndef QTIPCSERVER_H
#define QTIPCSERVER_H
// Define Bitcoin-Qt message queue name
#define BITCOINURI_QUEUE_NAME "BitcoinURI" #define BITCOINURI_QUEUE_NAME "BitcoinURI"
void ipcInit(); void ipcInit();
void ipcShutdown();
#endif // QTIPCSERVER_H