/*
|
* =====================================================================================
|
*
|
* Filename: shm_queue.cpp
|
*
|
* Description:
|
*
|
* Version: 1.0
|
* Created: 2021年03月25日 10时34分42秒
|
* Revision: none
|
* Compiler: gcc
|
*
|
* Author: Li Chao (),
|
* Organization:
|
*
|
* =====================================================================================
|
*/
|
|
#include "shm_queue.h"
|
#include "bh_util.h"
|
#include <boost/uuid/uuid_generators.hpp>
|
#include <boost/uuid/uuid_io.hpp>
|
|
namespace bhome_shm
|
{
|
using namespace bhome_msg;
|
using namespace boost::interprocess;
|
using namespace boost::uuids;
|
|
namespace
|
{
|
std::string MsgQIdToName(const MQId &id) { return "shmq" + to_string(id); }
|
// MQId EmptyId() { return nil_uuid(); }
|
MQId NewId() { return random_generator()(); }
|
const int AdjustMQLength(const int len)
|
{
|
const int kMaxLength = 10000;
|
const int kDefaultLen = 12;
|
if (len <= 0) {
|
return kDefaultLen;
|
} else if (len < kMaxLength) {
|
return len;
|
} else {
|
return kMaxLength;
|
}
|
}
|
|
} // namespace
|
|
// ShmMsgQueue memory usage: (320 + 16*length) bytes, length >= 2
|
ShmMsgQueue::ShmMsgQueue(const MQId &id, ShmType &segment, const int len) :
|
Super(segment, MsgQIdToName(id), AdjustMQLength(len), segment.get_segment_manager()),
|
id_(id)
|
{
|
}
|
|
ShmMsgQueue::ShmMsgQueue(ShmType &segment, const int len) :
|
ShmMsgQueue(NewId(), segment, len) {}
|
|
ShmMsgQueue::~ShmMsgQueue() {}
|
|
bool ShmMsgQueue::Remove(SharedMemory &shm, const MQId &id)
|
{
|
return Super::Remove(shm, MsgQIdToName(id));
|
}
|
|
ShmMsgQueue::Queue *ShmMsgQueue::FindRemote(SharedMemory &shm, const MQId &remote_id)
|
{
|
return Find(shm, MsgQIdToName(remote_id));
|
}
|
bool ShmMsgQueue::Send(SharedMemory &shm, const MQId &remote_id, const MsgI &msg, const int timeout_ms, OnSend const &onsend)
|
{
|
Queue *remote = FindRemote(shm, remote_id);
|
if (remote) {
|
if (onsend) {
|
return remote->Write(msg, timeout_ms, [&onsend](const MsgI &msg) { onsend(); msg.AddRef(); });
|
} else {
|
return remote->Write(msg, timeout_ms, [](const MsgI &msg) { msg.AddRef(); });
|
}
|
} else {
|
// SetLestError(eNotFound);
|
return false;
|
}
|
}
|
|
bool ShmMsgQueue::TrySend(SharedMemory &shm, const MQId &remote_id, const MsgI &msg, OnSend const &onsend)
|
{
|
Queue *remote = FindRemote(shm, remote_id);
|
if (remote) {
|
if (onsend) {
|
return remote->TryWrite(msg, [&onsend](const MsgI &msg) { onsend(); msg.AddRef(); });
|
} else {
|
return remote->TryWrite(msg, [](const MsgI &msg) { msg.AddRef(); });
|
}
|
} else {
|
// SetLestError(eNotFound);
|
return false;
|
}
|
}
|
|
// Test shows that in the 2 cases:
|
// 1) build msg first, then find remote queue;
|
// 2) find remote queue first, then build msg;
|
// 1 is about 50% faster than 2, maybe cache related.
|
|
} // namespace bhome_shm
|