ostinato/common/sign.cpp

296 lines
7.6 KiB
C++

/*
Copyright (C) 2016 Srivats P.
This file is part of "Ostinato"
This is free software: you can redistribute it and/or modify
it under the terms of the GNU General Public License as published by
the Free Software Foundation, either version 3 of the License, or
(at your option) any later version.
This program 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.
You should have received a copy of the GNU General Public License
along with this program. If not, see <http://www.gnu.org/licenses/>
*/
#include "sign.h"
#include "../common/streambase.h"
SignProtocol::SignProtocol(StreamBase *stream, AbstractProtocol *parent)
: AbstractProtocol(stream, parent)
{
}
SignProtocol::~SignProtocol()
{
}
AbstractProtocol* SignProtocol::createInstance(StreamBase *stream,
AbstractProtocol *parent)
{
return new SignProtocol(stream, parent);
}
quint32 SignProtocol::protocolNumber() const
{
return OstProto::Protocol::kSignFieldNumber;
}
void SignProtocol::protoDataCopyInto(OstProto::Protocol &protocol) const
{
protocol.MutableExtension(OstProto::sign)->CopyFrom(data);
protocol.mutable_protocol_id()->set_id(protocolNumber());
}
void SignProtocol::protoDataCopyFrom(const OstProto::Protocol &protocol)
{
if (protocol.protocol_id().id() == protocolNumber() &&
protocol.HasExtension(OstProto::sign))
data.MergeFrom(protocol.GetExtension(OstProto::sign));
}
QString SignProtocol::name() const
{
return QString("Signature");
}
QString SignProtocol::shortName() const
{
return QString("SIGN");
}
int SignProtocol::fieldCount() const
{
return sign_fieldCount;
}
AbstractProtocol::FieldFlags SignProtocol::fieldFlags(int index) const
{
AbstractProtocol::FieldFlags flags;
flags = AbstractProtocol::fieldFlags(index);
switch (index)
{
case sign_magic:
case sign_tlv_tx_port:
case sign_tlv_guid:
case sign_tlv_ttag:
case sign_tlv_end:
break;
default:
qFatal("%s: unimplemented case %d in switch", __PRETTY_FUNCTION__,
index);
break;
}
return flags;
}
QVariant SignProtocol::fieldData(int index, FieldAttrib attrib,
int streamIndex) const
{
switch (index)
{
case sign_magic:
{
switch(attrib)
{
case FieldName:
return QString("Magic");
case FieldValue:
return kSignMagic;
case FieldTextValue:
return QString("%1").arg(kSignMagic);
case FieldFrameValue:
{
QByteArray fv;
fv.resize(4);
qToBigEndian(kSignMagic, (uchar*) fv.data());
return fv;
}
default:
break;
}
break;
}
case sign_tlv_ttag:
{
switch(attrib)
{
case FieldName:
return QString("T-Tag");
case FieldValue:
return 0;
case FieldTextValue:
return QString("%1").arg(0);
case FieldFrameValue:
{
QByteArray fv;
fv.resize(2);
fv[0] = 0;
fv[1] = kTypeLenTtagPlaceholder;
return fv;
}
default:
break;
}
break;
}
case sign_tlv_tx_port:
{
switch(attrib)
{
case FieldName:
return QString("TxPort");
case FieldValue:
return mpStream->portId();
case FieldTextValue:
return QString("%1").arg(mpStream->portId());
case FieldFrameValue:
{
QByteArray fv;
fv.resize(2);
fv[0] = mpStream->portId() & 0xFF;
fv[1] = kTypeLenTxPort;
return fv;
}
default:
break;
}
break;
}
case sign_tlv_guid:
{
quint32 guid = data.stream_guid() & 0xFFFFFF;
switch(attrib)
{
case FieldName:
return QString("Stream GUID");
case FieldValue:
return guid;
case FieldTextValue:
return QString("%1").arg(guid);
case FieldFrameValue:
{
QByteArray fv;
fv.resize(4);
fv[0] = (guid >> 16) & 0xff;
fv[1] = (guid >> 8) & 0xff;
fv[2] = (guid >> 0) & 0xff;
fv[3] = kTypeLenGuid;
return fv;
}
default:
break;
}
break;
}
case sign_tlv_end:
{
switch(attrib)
{
case FieldName:
return QString("End TLV");
case FieldValue:
return 0;
case FieldTextValue:
return QString("NA");
case FieldFrameValue:
return QByteArray(1, kTypeLenEnd);
default:
break;
}
break;
}
default:
qFatal("%s: unimplemented case %d in switch", __PRETTY_FUNCTION__,
index);
break;
}
return AbstractProtocol::fieldData(index, attrib, streamIndex);
}
bool SignProtocol::setFieldData(int index, const QVariant &value,
FieldAttrib attrib)
{
bool isOk = false;
if (attrib != FieldValue)
goto _exit;
switch (index)
{
case sign_tlv_guid:
{
uint guid = value.toUInt(&isOk);
if (isOk)
data.set_stream_guid(guid & 0xFFFFFF);
break;
}
default:
qFatal("%s: unimplemented case %d in switch", __PRETTY_FUNCTION__,
index);
break;
}
_exit:
return isOk;
}
quint32 SignProtocol::magic()
{
return kSignMagic;
}
bool SignProtocol::packetGuid(const uchar *pkt, int pktLen, uint *guid)
{
const uchar *p = pkt + pktLen - sizeof(kSignMagic);
quint32 magic = qFromBigEndian<quint32>(p);
if (magic != kSignMagic)
return false;
p--;
while (*p != kTypeLenEnd) {
if (*p == kTypeLenGuid) {
*guid = qFromBigEndian<quint32>(p - 3) >> 8;
return true;
}
p -= 1 + (*p >> 5); // move to next TLV
}
return false;
}
bool SignProtocol::packetTtagId(const uchar *pkt, int pktLen, uint *ttagId, uint *guid)
{
bool ret = false;
const uchar *p = pkt + pktLen - sizeof(kSignMagic);
quint32 magic = qFromBigEndian<quint32>(p);
if (magic != kSignMagic)
return ret;
*guid = kInvalidGuid;
p--;
while (*p != kTypeLenEnd) {
if (*p == kTypeLenTtag) {
*ttagId = *(p - 1);
ret = true;
} else if (*p == kTypeLenGuid) {
*guid = qFromBigEndian<quint32>(p - 3) >> 8;
} else if (*p == kTypeLenTxPort) {
#ifdef Q_OS_WIN32
*ttagId |= uint(*(p - 1)) << 8;
#endif
}
p -= 1 + (*p >> 5); // move to next TLV
}
return ret;
}