![](/img/trans.png)
[英]getting library_error exception when trying to use boost::interprocess::message_queue
[英]Boost IPC Message_Queue try_receive throws interprocess_exception::library_error
我正在使用boost :: interprocess :: message_queue在两个进程之间进行进程间通信。
这是我第一次使用它,因此我看不到该异常,因为我找不到关于它的任何文档。
我的班级设置如下:
struct Pos{float X,Y,Z;};
struct Quat{float W,X,Y,Z;};
typedef unsigned char Byte;
struct NPCDataFoot
{
//some Pos and Quat variables here too
unsigned short AnimationIndex;
void Apply(NPCDataFoot &data){AnimationIndex=data.AnimationIndex;}
NPCDataFoot(){AnimationIndex=0;}
};
struct NPCDataVehicle
{
//many many more
unsigned short lrAnalog;
void Apply(NPCDataVehicle &data){lrAnalog=data.lrAnalog;}
NPCDataVehicle(){lrAnalog = 0;}
};
enum TransmissionDataType{
TDT_NewNPC,//many more...
};
const unsigned short QueueMaxSize = 256;
struct ExchangeData
{
unsigned short CommandType;
unsigned short NPCPlayerID;
Byte State;
NPCDataFoot OnFootData;
NPCDataVehicle InCarData;
float MoveSpeed;
Pos MoveToPos;
//206
ExchangeData(unsigned short CommandType = 0, unsigned short NPCPlayerid = 0xFFFF)
: CommandType(CommandType), NPCPlayerID(NPCPlayerid)
{}
ExchangeData(unsigned short CommandType, unsigned short NPCPlayerid, NPCDataFoot& foot_data, NPCDataVehicle& car_data)
: CommandType(CommandType), NPCPlayerID(NPCPlayerid), OnFootData(foot_data), InCarData(car_data)
{}
};
我的两个程序都设置为使用/ zp1标志进行编译(将结构/类对齐为1字节对齐)。
现在,只要我到达此代码:
ServerMsgQueue * message_queue = NULL;
PLUGIN_EXPORT void PLUGIN_CALL
ProcessTick()
{
static bool init = false;
static ExchangeData DataTransmision;
if(!init)
{
try
{
ServerMsgQueue = new message_queue(open_or_create
,string_format("REMOTESHAREDMEMORYBTWNPRCS%04x",GetServerVarAsInt("port")).c_str()
,1024 * QueueMaxSize,sizeof(ExchangeData));
}
catch(interprocess_exception &ex)
{
std::cout << ex.what() << ":" << __FILE__ << ":" << __FUNCTION__ << ":" << __LINE__ << std::endl;
}
init = true;
}
static unsigned int unused;
if(ServerMsgQueue)
{
try
{
if(ServerMsgQueue->try_receive(&DataTransmision,sizeof(ExchangeData),unused,unused))
{
switch(DataTransmision.CommandType)
{//some cases
default:{std::cout << "UNKNOWN RECEIVED DATA!!!" << std::endl;break;}
}
}
}
catch(interprocess_exception &ex)
{
//this happens always
std::cout << ex.what() << ":" << __FILE__ << ":" << __FUNCTION__ << ":" << __LINE__ << std::endl;
}
}
}
应用程序在使用try_receive
的块中不断发出“ boost :: interprocess_exception :: library_error”。
我在这种情况下做错了什么? 我确定数据发送的大小是相同的,因为1)我使用相同的标头,并且2)我使用相同的选项进行编译。
我已经确认两个程序中的变量大小与以下代码相同:
MessageBox(NULL,string_format(
"Pos(%d):Quat(%d):NPCDataFoot(%d):NPCDataVehicle(%d):ExchangeData(%d)",
sizeof(Pos),sizeof(Quat),sizeof(NPCDataFoot),sizeof(NPCDataVehicle),sizeof(ExchangeData)).c_str()
,"Reported sizes Client",0);
编辑:我似乎已经“解决”了它。
似乎此代码是错误:
static unsigned int unused;
if(ServerMsgQueue->try_receive(&DataTransmision,sizeof(ExchangeData),unused,unused))
改成
unsigned int Priority;
size_t sizexxx;
if(ServerMsgQueue->try_receive(&DataTransmision,sizeof(ExchangeData),sizexxx,Priority))
有人可以解释为什么这行得通,而其他代码行不通吗?
这很可能是由于您使用/zp1
引起的。 您的程序及其链接到的库将认为结构具有不同的大小和/或成员位置。 不要那样做
参见message_queue::do_receive
的源代码(从message_queue::try_receive
); do_receive
首先将消息大小写入“ out”参数recvd_size
(对代码中unused
的引用),然后将优先级写入“ out”参数priority
( 也对代码中unused
的引用),从而有效地覆盖了recvd_size
priority
值)。 在紧下方,对memcpy
的调用使用了现在不正确的recvd_size
,从而导致您观察到的行为(不完整的副本或缓冲区溢出,具体取决于priority
)。
可以说,该库的作者应该对memcpy
字节计数使用原始的top_msg->len
(而不是“ out”参数recvd_size
)。 但是,为了避免此类意外,请始终对“输出”参数使用不同的变量(如果需要,可以将它们unused1
“ unused1
和“ unused2
)。
声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.