2012-11-29 17:48:28 -05:00
|
|
|
/*
|
|
|
|
* libretroshare/src/serialiser: rswikiitems.cc
|
|
|
|
*
|
|
|
|
* RetroShare C++ Interface.
|
|
|
|
*
|
|
|
|
* Copyright 2012-2012 by Robert Fernie
|
|
|
|
*
|
|
|
|
* This library is free software; you can redistribute it and/or
|
|
|
|
* modify it under the terms of the GNU Library General Public
|
|
|
|
* License Version 2.1 as published by the Free Software Foundation.
|
|
|
|
*
|
|
|
|
* This library 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
|
|
|
|
* Library General Public License for more details.
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU Library General Public
|
|
|
|
* License along with this library; if not, write to the Free Software
|
|
|
|
* Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307
|
|
|
|
* USA.
|
|
|
|
*
|
|
|
|
* Please report all bugs and problems to "retroshare@lunamutt.com".
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <iostream>
|
|
|
|
|
2012-11-30 19:16:24 -05:00
|
|
|
#include "rsgxscircleitems.h"
|
2012-11-29 17:48:28 -05:00
|
|
|
#include "serialiser/rstlvbase.h"
|
|
|
|
#include "serialiser/rsbaseserial.h"
|
|
|
|
|
2016-03-25 18:54:13 -04:00
|
|
|
//#define CIRCLE_DEBUG 1
|
2012-11-29 17:48:28 -05:00
|
|
|
|
|
|
|
uint32_t RsGxsCircleSerialiser::size(RsItem *item)
|
|
|
|
{
|
|
|
|
RsGxsCircleGroupItem* grp_item = NULL;
|
|
|
|
RsGxsCircleMsgItem* snap_item = NULL;
|
|
|
|
|
|
|
|
if((grp_item = dynamic_cast<RsGxsCircleGroupItem*>(item)) != NULL)
|
|
|
|
{
|
|
|
|
return sizeGxsCircleGroupItem(grp_item);
|
|
|
|
}
|
|
|
|
else if((snap_item = dynamic_cast<RsGxsCircleMsgItem*>(item)) != NULL)
|
|
|
|
{
|
|
|
|
return sizeGxsCircleMsgItem(snap_item);
|
2015-03-14 10:33:23 -04:00
|
|
|
}
|
|
|
|
else
|
|
|
|
return 0 ;
|
2012-11-29 17:48:28 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
bool RsGxsCircleSerialiser::serialise(RsItem *item, void *data, uint32_t *size)
|
|
|
|
{
|
|
|
|
RsGxsCircleGroupItem* grp_item = NULL;
|
|
|
|
RsGxsCircleMsgItem* snap_item = NULL;
|
|
|
|
|
|
|
|
if((grp_item = dynamic_cast<RsGxsCircleGroupItem*>(item)) != NULL)
|
|
|
|
{
|
|
|
|
return serialiseGxsCircleGroupItem(grp_item, data, size);
|
|
|
|
}
|
|
|
|
else if((snap_item = dynamic_cast<RsGxsCircleMsgItem*>(item)) != NULL)
|
|
|
|
{
|
|
|
|
return serialiseGxsCircleMsgItem(snap_item, data, size);
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
RsItem* RsGxsCircleSerialiser::deserialise(void* data, uint32_t* size)
|
|
|
|
{
|
|
|
|
|
|
|
|
#ifdef CIRCLE_DEBUG
|
|
|
|
std::cerr << "RsGxsCircleSerialiser::deserialise()" << std::endl;
|
|
|
|
#endif
|
|
|
|
/* get the type and size */
|
|
|
|
uint32_t rstype = getRsItemId(data);
|
|
|
|
|
|
|
|
if ((RS_PKT_VERSION_SERVICE != getRsItemVersion(rstype)) ||
|
2014-03-29 11:34:37 -04:00
|
|
|
(RS_SERVICE_GXS_TYPE_GXSCIRCLE != getRsItemService(rstype)))
|
2012-11-29 17:48:28 -05:00
|
|
|
{
|
|
|
|
return NULL; /* wrong type */
|
|
|
|
}
|
|
|
|
|
|
|
|
switch(getRsItemSubType(rstype))
|
|
|
|
{
|
|
|
|
|
|
|
|
case RS_PKT_SUBTYPE_GXSCIRCLE_GROUP_ITEM:
|
|
|
|
return deserialiseGxsCircleGroupItem(data, size);
|
|
|
|
break;
|
|
|
|
case RS_PKT_SUBTYPE_GXSCIRCLE_MSG_ITEM:
|
|
|
|
return deserialiseGxsCircleMsgItem(data, size);
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
#ifdef CIRCLE_DEBUG
|
|
|
|
std::cerr << "RsGxsCircleSerialiser::deserialise(): unknown subtype";
|
|
|
|
std::cerr << std::endl;
|
|
|
|
#endif
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/*****************************************************************************************/
|
|
|
|
/*****************************************************************************************/
|
|
|
|
/*****************************************************************************************/
|
|
|
|
|
|
|
|
|
|
|
|
void RsGxsCircleGroupItem::clear()
|
|
|
|
{
|
2013-06-04 17:00:43 -04:00
|
|
|
pgpIdSet.TlvClear();
|
2012-11-30 19:16:24 -05:00
|
|
|
gxsIdSet.TlvClear();
|
|
|
|
subCircleSet.TlvClear();
|
2012-11-29 17:48:28 -05:00
|
|
|
}
|
|
|
|
|
2012-11-30 19:16:24 -05:00
|
|
|
bool RsGxsCircleGroupItem::convertFrom(const RsGxsCircleGroup &group)
|
|
|
|
{
|
|
|
|
clear();
|
|
|
|
|
|
|
|
meta = group.mMeta;
|
2013-06-04 17:00:43 -04:00
|
|
|
|
|
|
|
// Enforce the local rules.
|
|
|
|
if (meta.mCircleType == GXS_CIRCLE_TYPE_LOCAL)
|
|
|
|
{
|
2014-04-01 04:00:20 -04:00
|
|
|
pgpIdSet.ids = group.mLocalFriends;
|
2013-06-04 17:00:43 -04:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2014-04-01 04:00:20 -04:00
|
|
|
gxsIdSet.ids = group.mInvitedMembers;
|
2013-06-04 17:00:43 -04:00
|
|
|
}
|
2014-03-17 16:56:06 -04:00
|
|
|
|
2014-04-01 04:00:20 -04:00
|
|
|
subCircleSet.ids = group.mSubCircles;
|
2012-11-30 19:16:24 -05:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool RsGxsCircleGroupItem::convertTo(RsGxsCircleGroup &group) const
|
|
|
|
{
|
|
|
|
group.mMeta = meta;
|
2013-06-04 17:00:43 -04:00
|
|
|
|
|
|
|
// Enforce the local rules.
|
|
|
|
if (meta.mCircleType == GXS_CIRCLE_TYPE_LOCAL)
|
|
|
|
{
|
2014-04-01 04:00:20 -04:00
|
|
|
group.mLocalFriends = pgpIdSet.ids;
|
2013-06-04 17:00:43 -04:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2014-04-01 04:00:20 -04:00
|
|
|
group.mInvitedMembers = gxsIdSet.ids;
|
2013-06-04 17:00:43 -04:00
|
|
|
}
|
2014-03-17 16:56:06 -04:00
|
|
|
|
2014-04-01 04:00:20 -04:00
|
|
|
group.mSubCircles = subCircleSet.ids;
|
2012-11-30 19:16:24 -05:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2012-11-29 17:48:28 -05:00
|
|
|
std::ostream& RsGxsCircleGroupItem::print(std::ostream& out, uint16_t indent)
|
|
|
|
{
|
|
|
|
printRsItemBase(out, "RsGxsCircleGroupItem", indent);
|
|
|
|
uint16_t int_Indent = indent + 2;
|
|
|
|
|
2013-06-04 17:00:43 -04:00
|
|
|
if (meta.mCircleType == GXS_CIRCLE_TYPE_LOCAL)
|
|
|
|
{
|
|
|
|
printRsItemBase(out, "Local Circle: PGP Ids:", indent);
|
|
|
|
pgpIdSet.print(out, int_Indent);
|
|
|
|
printRsItemBase(out, "GXS Ids (should be empty):", indent);
|
|
|
|
gxsIdSet.print(out, int_Indent);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
printRsItemBase(out, "External Circle: GXS Ids", indent);
|
|
|
|
gxsIdSet.print(out, int_Indent);
|
|
|
|
printRsItemBase(out, "PGP Ids (should be empty):", indent);
|
|
|
|
pgpIdSet.print(out, int_Indent);
|
|
|
|
}
|
|
|
|
|
2012-11-30 19:16:24 -05:00
|
|
|
subCircleSet.print(out, int_Indent);
|
2012-11-29 17:48:28 -05:00
|
|
|
printRsItemEnd(out ,"RsGxsCircleGroupItem", indent);
|
|
|
|
return out;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
uint32_t RsGxsCircleSerialiser::sizeGxsCircleGroupItem(RsGxsCircleGroupItem *item)
|
|
|
|
{
|
|
|
|
uint32_t s = 8; // header
|
|
|
|
|
2013-06-04 17:00:43 -04:00
|
|
|
s += item->pgpIdSet.TlvSize();
|
2012-11-30 19:16:24 -05:00
|
|
|
s += item->gxsIdSet.TlvSize();
|
|
|
|
s += item->subCircleSet.TlvSize();
|
2012-11-29 17:48:28 -05:00
|
|
|
|
|
|
|
return s;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool RsGxsCircleSerialiser::serialiseGxsCircleGroupItem(RsGxsCircleGroupItem *item, void *data, uint32_t *size)
|
|
|
|
{
|
|
|
|
|
|
|
|
#ifdef CIRCLE_DEBUG
|
|
|
|
std::cerr << "RsGxsCircleSerialiser::serialiseGxsCircleGroupItem()" << std::endl;
|
|
|
|
#endif
|
|
|
|
|
|
|
|
uint32_t tlvsize = sizeGxsCircleGroupItem(item);
|
|
|
|
uint32_t offset = 0;
|
|
|
|
|
|
|
|
if(*size < tlvsize)
|
|
|
|
{
|
|
|
|
#ifdef CIRCLE_DEBUG
|
|
|
|
std::cerr << "RsGxsCircleSerialiser::serialiseGxsCircleGroupItem()" << std::endl;
|
|
|
|
#endif
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
*size = tlvsize;
|
|
|
|
|
|
|
|
bool ok = true;
|
|
|
|
|
|
|
|
ok &= setRsItemHeader(data, tlvsize, item->PacketId(), tlvsize);
|
|
|
|
|
|
|
|
/* skip the header */
|
|
|
|
offset += 8;
|
|
|
|
|
|
|
|
/* GxsCircleGroupItem */
|
2013-06-04 17:00:43 -04:00
|
|
|
ok &= item->pgpIdSet.SetTlv(data, tlvsize, &offset);
|
2012-11-30 19:16:24 -05:00
|
|
|
ok &= item->gxsIdSet.SetTlv(data, tlvsize, &offset);
|
|
|
|
ok &= item->subCircleSet.SetTlv(data, tlvsize, &offset);
|
2012-11-29 17:48:28 -05:00
|
|
|
|
|
|
|
if(offset != tlvsize)
|
|
|
|
{
|
|
|
|
#ifdef CIRCLE_DEBUG
|
|
|
|
std::cerr << "RsGxsCircleSerialiser::serialiseGxsCircleGroupItem() FAIL Size Error! " << std::endl;
|
|
|
|
#endif
|
|
|
|
ok = false;
|
|
|
|
}
|
|
|
|
|
|
|
|
#ifdef CIRCLE_DEBUG
|
|
|
|
if (!ok)
|
|
|
|
{
|
|
|
|
std::cerr << "RsGxsCircleSerialiser::serialiseGxsCircleGroupItem() NOK" << std::endl;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
return ok;
|
|
|
|
}
|
|
|
|
|
|
|
|
RsGxsCircleGroupItem* RsGxsCircleSerialiser::deserialiseGxsCircleGroupItem(void *data, uint32_t *size)
|
|
|
|
{
|
|
|
|
|
|
|
|
#ifdef CIRCLE_DEBUG
|
|
|
|
std::cerr << "RsGxsCircleSerialiser::deserialiseGxsCircleGroupItem()" << std::endl;
|
|
|
|
#endif
|
|
|
|
/* get the type and size */
|
|
|
|
uint32_t rstype = getRsItemId(data);
|
|
|
|
uint32_t rssize = getRsItemSize(data);
|
|
|
|
|
|
|
|
uint32_t offset = 0;
|
|
|
|
|
|
|
|
|
|
|
|
if ((RS_PKT_VERSION_SERVICE != getRsItemVersion(rstype)) ||
|
2014-03-29 11:34:37 -04:00
|
|
|
(RS_SERVICE_GXS_TYPE_GXSCIRCLE != getRsItemService(rstype)) ||
|
2012-11-29 17:48:28 -05:00
|
|
|
(RS_PKT_SUBTYPE_GXSCIRCLE_GROUP_ITEM != getRsItemSubType(rstype)))
|
|
|
|
{
|
|
|
|
#ifdef CIRCLE_DEBUG
|
|
|
|
std::cerr << "RsGxsCircleSerialiser::deserialiseGxsCircleGroupItem() FAIL wrong type" << std::endl;
|
|
|
|
#endif
|
|
|
|
return NULL; /* wrong type */
|
|
|
|
}
|
|
|
|
|
|
|
|
if (*size < rssize) /* check size */
|
|
|
|
{
|
|
|
|
#ifdef CIRCLE_DEBUG
|
|
|
|
std::cerr << "RsGxsCircleSerialiser::deserialiseGxsCircleGroupItem() FAIL wrong size" << std::endl;
|
|
|
|
#endif
|
|
|
|
return NULL; /* not enough data */
|
|
|
|
}
|
|
|
|
|
|
|
|
/* set the packet length */
|
|
|
|
*size = rssize;
|
|
|
|
|
|
|
|
bool ok = true;
|
|
|
|
|
|
|
|
RsGxsCircleGroupItem* item = new RsGxsCircleGroupItem();
|
|
|
|
/* skip the header */
|
|
|
|
offset += 8;
|
|
|
|
|
2013-06-04 17:00:43 -04:00
|
|
|
ok &= item->pgpIdSet.GetTlv(data, rssize, &offset);
|
2012-11-30 19:16:24 -05:00
|
|
|
ok &= item->gxsIdSet.GetTlv(data, rssize, &offset);
|
|
|
|
ok &= item->subCircleSet.GetTlv(data, rssize, &offset);
|
2012-11-29 17:48:28 -05:00
|
|
|
|
|
|
|
if (offset != rssize)
|
|
|
|
{
|
|
|
|
#ifdef CIRCLE_DEBUG
|
|
|
|
std::cerr << "RsGxsCircleSerialiser::deserialiseGxsCircleGroupItem() FAIL size mismatch" << std::endl;
|
|
|
|
#endif
|
|
|
|
/* error */
|
|
|
|
delete item;
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!ok)
|
|
|
|
{
|
|
|
|
#ifdef CIRCLE_DEBUG
|
|
|
|
std::cerr << "RsGxsCircleSerialiser::deserialiseGxsCircleGroupItem() NOK" << std::endl;
|
|
|
|
#endif
|
|
|
|
delete item;
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
return item;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/*****************************************************************************************/
|
|
|
|
/*****************************************************************************************/
|
|
|
|
/*****************************************************************************************/
|
|
|
|
|
|
|
|
|
|
|
|
void RsGxsCircleMsgItem::clear()
|
|
|
|
{
|
2012-11-30 19:16:24 -05:00
|
|
|
msg.stuff.clear();
|
2012-11-29 17:48:28 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
std::ostream& RsGxsCircleMsgItem::print(std::ostream& out, uint16_t indent)
|
|
|
|
{
|
|
|
|
printRsItemBase(out, "RsGxsCircleMsgItem", indent);
|
|
|
|
uint16_t int_Indent = indent + 2;
|
|
|
|
|
|
|
|
printIndent(out, int_Indent);
|
2012-11-30 19:16:24 -05:00
|
|
|
out << "Stuff: " << msg.stuff << std::endl;
|
2012-11-29 17:48:28 -05:00
|
|
|
|
|
|
|
printRsItemEnd(out ,"RsGxsCircleMsgItem", indent);
|
|
|
|
return out;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
uint32_t RsGxsCircleSerialiser::sizeGxsCircleMsgItem(RsGxsCircleMsgItem *item)
|
|
|
|
{
|
|
|
|
|
2012-11-30 19:16:24 -05:00
|
|
|
const RsGxsCircleMsg &msg = item->msg;
|
2012-11-29 17:48:28 -05:00
|
|
|
uint32_t s = 8; // header
|
|
|
|
|
2012-11-30 19:16:24 -05:00
|
|
|
s += GetTlvStringSize(msg.stuff);
|
2012-11-29 17:48:28 -05:00
|
|
|
|
|
|
|
return s;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool RsGxsCircleSerialiser::serialiseGxsCircleMsgItem(RsGxsCircleMsgItem *item, void *data, uint32_t *size)
|
|
|
|
{
|
|
|
|
|
|
|
|
#ifdef CIRCLE_DEBUG
|
|
|
|
std::cerr << "RsGxsCircleSerialiser::serialiseGxsCircleMsgItem()" << std::endl;
|
|
|
|
#endif
|
|
|
|
|
|
|
|
uint32_t tlvsize = sizeGxsCircleMsgItem(item);
|
|
|
|
uint32_t offset = 0;
|
|
|
|
|
|
|
|
if(*size < tlvsize)
|
|
|
|
{
|
|
|
|
#ifdef CIRCLE_DEBUG
|
|
|
|
std::cerr << "RsGxsCircleSerialiser::serialiseGxsCircleMsgItem()" << std::endl;
|
|
|
|
#endif
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
*size = tlvsize;
|
|
|
|
|
|
|
|
bool ok = true;
|
|
|
|
|
|
|
|
ok &= setRsItemHeader(data, tlvsize, item->PacketId(), tlvsize);
|
|
|
|
|
|
|
|
/* skip the header */
|
|
|
|
offset += 8;
|
|
|
|
|
|
|
|
/* GxsCircleMsgItem */
|
2014-02-08 06:13:10 -05:00
|
|
|
ok &= SetTlvString(data, tlvsize, &offset, TLV_TYPE_STR_MSG, item->msg.stuff);
|
2012-11-29 17:48:28 -05:00
|
|
|
|
|
|
|
if(offset != tlvsize)
|
|
|
|
{
|
|
|
|
#ifdef CIRCLE_DEBUG
|
|
|
|
std::cerr << "RsGxsCircleSerialiser::serialiseGxsCircleMsgItem() FAIL Size Error! " << std::endl;
|
|
|
|
#endif
|
|
|
|
ok = false;
|
|
|
|
}
|
|
|
|
|
|
|
|
#ifdef CIRCLE_DEBUG
|
|
|
|
if (!ok)
|
|
|
|
{
|
|
|
|
std::cerr << "RsGxsCircleSerialiser::serialiseGxsCircleMsgItem() NOK" << std::endl;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
return ok;
|
|
|
|
}
|
|
|
|
|
|
|
|
RsGxsCircleMsgItem* RsGxsCircleSerialiser::deserialiseGxsCircleMsgItem(void *data, uint32_t *size)
|
|
|
|
{
|
|
|
|
|
|
|
|
#ifdef CIRCLE_DEBUG
|
|
|
|
std::cerr << "RsGxsCircleSerialiser::deserialiseGxsCircleMsgItem()" << std::endl;
|
|
|
|
#endif
|
|
|
|
/* get the type and size */
|
|
|
|
uint32_t rstype = getRsItemId(data);
|
|
|
|
uint32_t rssize = getRsItemSize(data);
|
|
|
|
|
|
|
|
uint32_t offset = 0;
|
|
|
|
|
|
|
|
|
|
|
|
if ((RS_PKT_VERSION_SERVICE != getRsItemVersion(rstype)) ||
|
2014-03-29 11:34:37 -04:00
|
|
|
(RS_SERVICE_GXS_TYPE_GXSCIRCLE != getRsItemService(rstype)) ||
|
2012-11-29 17:48:28 -05:00
|
|
|
(RS_PKT_SUBTYPE_GXSCIRCLE_MSG_ITEM != getRsItemSubType(rstype)))
|
|
|
|
{
|
|
|
|
#ifdef CIRCLE_DEBUG
|
|
|
|
std::cerr << "RsGxsCircleSerialiser::deserialiseGxsCircleMsgItem() FAIL wrong type" << std::endl;
|
|
|
|
#endif
|
|
|
|
return NULL; /* wrong type */
|
|
|
|
}
|
|
|
|
|
|
|
|
if (*size < rssize) /* check size */
|
|
|
|
{
|
|
|
|
#ifdef CIRCLE_DEBUG
|
|
|
|
std::cerr << "RsGxsCircleSerialiser::deserialiseGxsCircleMsgItem() FAIL wrong size" << std::endl;
|
|
|
|
#endif
|
|
|
|
return NULL; /* not enough data */
|
|
|
|
}
|
|
|
|
|
|
|
|
/* set the packet length */
|
|
|
|
*size = rssize;
|
|
|
|
|
|
|
|
bool ok = true;
|
|
|
|
|
|
|
|
RsGxsCircleMsgItem* item = new RsGxsCircleMsgItem();
|
|
|
|
/* skip the header */
|
|
|
|
offset += 8;
|
|
|
|
|
2014-02-08 06:13:10 -05:00
|
|
|
ok &= GetTlvString(data, rssize, &offset, TLV_TYPE_STR_MSG, item->msg.stuff);
|
2012-11-29 17:48:28 -05:00
|
|
|
|
|
|
|
if (offset != rssize)
|
|
|
|
{
|
|
|
|
#ifdef CIRCLE_DEBUG
|
|
|
|
std::cerr << "RsGxsCircleSerialiser::deserialiseGxsCircleMsgItem() FAIL size mismatch" << std::endl;
|
|
|
|
#endif
|
|
|
|
/* error */
|
|
|
|
delete item;
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!ok)
|
|
|
|
{
|
|
|
|
#ifdef CIRCLE_DEBUG
|
|
|
|
std::cerr << "RsGxsCircleSerialiser::deserialiseGxsCircleMsgItem() NOK" << std::endl;
|
|
|
|
#endif
|
|
|
|
delete item;
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
return item;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*****************************************************************************************/
|
|
|
|
/*****************************************************************************************/
|
|
|
|
/*****************************************************************************************/
|
|
|
|
|