Blame tlv.c

Packit 9c3e7e
/**
Packit 9c3e7e
 * @file tlv.c
Packit 9c3e7e
 * @note Copyright (C) 2012 Richard Cochran <richardcochran@gmail.com>
Packit 9c3e7e
 *
Packit 9c3e7e
 * This program is free software; you can redistribute it and/or modify
Packit 9c3e7e
 * it under the terms of the GNU General Public License as published by
Packit 9c3e7e
 * the Free Software Foundation; either version 2 of the License, or
Packit 9c3e7e
 * (at your option) any later version.
Packit 9c3e7e
 *
Packit 9c3e7e
 * This program is distributed in the hope that it will be useful,
Packit 9c3e7e
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
Packit 9c3e7e
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
Packit 9c3e7e
 * GNU General Public License for more details.
Packit 9c3e7e
 *
Packit 9c3e7e
 * You should have received a copy of the GNU General Public License along
Packit 9c3e7e
 * with this program; if not, write to the Free Software Foundation, Inc.,
Packit 9c3e7e
 * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
Packit 9c3e7e
 */
Packit 9c3e7e
#include <arpa/inet.h>
Packit 9c3e7e
#include <errno.h>
Packit 9c3e7e
#include <stdlib.h>
Packit 9c3e7e
#include <string.h>
Packit 9c3e7e
Packit 9c3e7e
#include "port.h"
Packit 9c3e7e
#include "tlv.h"
Packit 9c3e7e
#include "msg.h"
Packit 9c3e7e
Packit 9c3e7e
#define HTONS(x) (x) = htons(x)
Packit 9c3e7e
#define HTONL(x) (x) = htonl(x)
Packit 9c3e7e
#define NTOHS(x) (x) = ntohs(x)
Packit 9c3e7e
#define NTOHL(x) (x) = ntohl(x)
Packit 9c3e7e
Packit 9c3e7e
#define TLV_LENGTH_INVALID(tlv, type) \
Packit 9c3e7e
	(tlv->length < sizeof(struct type) - sizeof(struct TLV))
Packit 9c3e7e
Packit 9c3e7e
uint8_t ieee8021_id[3] = { IEEE_802_1_COMMITTEE };
Packit 9c3e7e
Packit 9c3e7e
static TAILQ_HEAD(tlv_pool, tlv_extra) tlv_pool =
Packit 9c3e7e
	TAILQ_HEAD_INITIALIZER(tlv_pool);
Packit 9c3e7e
Packit 9c3e7e
static void scaled_ns_n2h(ScaledNs *sns)
Packit 9c3e7e
{
Packit 9c3e7e
	sns->nanoseconds_msb = ntohs(sns->nanoseconds_msb);
Packit 9c3e7e
	sns->nanoseconds_lsb = net2host64(sns->nanoseconds_msb);
Packit 9c3e7e
	sns->fractional_nanoseconds = ntohs(sns->fractional_nanoseconds);
Packit 9c3e7e
}
Packit 9c3e7e
Packit 9c3e7e
static void scaled_ns_h2n(ScaledNs *sns)
Packit 9c3e7e
{
Packit 9c3e7e
	sns->nanoseconds_msb = htons(sns->nanoseconds_msb);
Packit 9c3e7e
	sns->nanoseconds_lsb = host2net64(sns->nanoseconds_msb);
Packit 9c3e7e
	sns->fractional_nanoseconds = htons(sns->fractional_nanoseconds);
Packit 9c3e7e
}
Packit 9c3e7e
Packit 9c3e7e
static uint16_t flip16(uint16_t *p)
Packit 9c3e7e
{
Packit 9c3e7e
	uint16_t v;
Packit 9c3e7e
	memcpy(&v, p, sizeof(v));
Packit 9c3e7e
	v = htons(v);
Packit 9c3e7e
	memcpy(p, &v, sizeof(v));
Packit 9c3e7e
	return v;
Packit 9c3e7e
}
Packit 9c3e7e
Packit 9c3e7e
static int64_t host2net64_unaligned(int64_t *p)
Packit 9c3e7e
{
Packit 9c3e7e
	int64_t v;
Packit 9c3e7e
	memcpy(&v, p, sizeof(v));
Packit 9c3e7e
	v = host2net64(v);
Packit 9c3e7e
	memcpy(p, &v, sizeof(v));
Packit 9c3e7e
	return v;
Packit 9c3e7e
}
Packit 9c3e7e
Packit 9c3e7e
static int64_t net2host64_unaligned(int64_t *p)
Packit 9c3e7e
{
Packit 9c3e7e
	int64_t v;
Packit 9c3e7e
	memcpy(&v, p, sizeof(v));
Packit 9c3e7e
	v = net2host64(v);
Packit 9c3e7e
	memcpy(p, &v, sizeof(v));
Packit 9c3e7e
	return v;
Packit 9c3e7e
}
Packit 9c3e7e
Packit 9c3e7e
static int mgt_post_recv(struct management_tlv *m, uint16_t data_len,
Packit 9c3e7e
			 struct tlv_extra *extra)
Packit 9c3e7e
{
Packit 9c3e7e
	struct defaultDS *dds;
Packit 9c3e7e
	struct currentDS *cds;
Packit 9c3e7e
	struct parentDS *pds;
Packit 9c3e7e
	struct timePropertiesDS *tp;
Packit 9c3e7e
	struct portDS *p;
Packit 9c3e7e
	struct port_ds_np *pdsnp;
Packit 9c3e7e
	struct time_status_np *tsn;
Packit 9c3e7e
	struct grandmaster_settings_np *gsn;
Packit 9c3e7e
	struct subscribe_events_np *sen;
Packit 9c3e7e
	struct port_properties_np *ppn;
Packit 9c3e7e
	struct mgmt_clock_description *cd;
Packit 9c3e7e
	int extra_len = 0, len;
Packit 9c3e7e
	uint8_t *buf;
Packit 9c3e7e
	uint16_t u16;
Packit 9c3e7e
	switch (m->id) {
Packit 9c3e7e
	case TLV_CLOCK_DESCRIPTION:
Packit 9c3e7e
		cd = &extra->cd;
Packit 9c3e7e
		buf = m->data;
Packit 9c3e7e
		len = data_len;
Packit 9c3e7e
Packit 9c3e7e
		cd->clockType = (UInteger16 *) buf;
Packit 9c3e7e
		buf += sizeof(*cd->clockType);
Packit 9c3e7e
		len -= sizeof(*cd->clockType);
Packit 9c3e7e
		if (len < 0)
Packit 9c3e7e
			goto bad_length;
Packit 9c3e7e
		flip16(cd->clockType);
Packit 9c3e7e
Packit 9c3e7e
		cd->physicalLayerProtocol = (struct PTPText *) buf;
Packit 9c3e7e
		buf += sizeof(struct PTPText);
Packit 9c3e7e
		len -= sizeof(struct PTPText);
Packit 9c3e7e
		if (len < 0)
Packit 9c3e7e
			goto bad_length;
Packit 9c3e7e
Packit 9c3e7e
		buf += cd->physicalLayerProtocol->length;
Packit 9c3e7e
		len -= cd->physicalLayerProtocol->length;
Packit 9c3e7e
		if (len < 0)
Packit 9c3e7e
			goto bad_length;
Packit 9c3e7e
Packit 9c3e7e
		cd->physicalAddress = (struct PhysicalAddress *) buf;
Packit 9c3e7e
		buf += sizeof(struct PhysicalAddress);
Packit 9c3e7e
		len -= sizeof(struct PhysicalAddress);
Packit 9c3e7e
		if (len < 0)
Packit 9c3e7e
			goto bad_length;
Packit 9c3e7e
Packit 9c3e7e
		u16 = flip16(&cd->physicalAddress->length);
Packit 9c3e7e
		if (u16 > TRANSPORT_ADDR_LEN)
Packit 9c3e7e
			goto bad_length;
Packit 9c3e7e
		buf += u16;
Packit 9c3e7e
		len -= u16;
Packit 9c3e7e
		if (len < 0)
Packit 9c3e7e
			goto bad_length;
Packit 9c3e7e
Packit 9c3e7e
		cd->protocolAddress = (struct PortAddress *) buf;
Packit 9c3e7e
		buf += sizeof(struct PortAddress);
Packit 9c3e7e
		len -= sizeof(struct PortAddress);
Packit 9c3e7e
		if (len < 0)
Packit 9c3e7e
			goto bad_length;
Packit 9c3e7e
Packit 9c3e7e
		flip16(&cd->protocolAddress->networkProtocol);
Packit 9c3e7e
		u16 = flip16(&cd->protocolAddress->addressLength);
Packit 9c3e7e
		if (u16 > TRANSPORT_ADDR_LEN)
Packit 9c3e7e
			goto bad_length;
Packit 9c3e7e
		buf += u16;
Packit 9c3e7e
		len -= u16;
Packit 9c3e7e
		if (len < 0)
Packit 9c3e7e
			goto bad_length;
Packit 9c3e7e
Packit 9c3e7e
		cd->manufacturerIdentity = buf;
Packit 9c3e7e
		buf += OUI_LEN + 1;
Packit 9c3e7e
		len -= OUI_LEN + 1;
Packit 9c3e7e
		if (len < 0)
Packit 9c3e7e
			goto bad_length;
Packit 9c3e7e
Packit 9c3e7e
		cd->productDescription = (struct PTPText *) buf;
Packit 9c3e7e
		buf += sizeof(struct PTPText);
Packit 9c3e7e
		len -= sizeof(struct PTPText);
Packit 9c3e7e
		if (len < 0)
Packit 9c3e7e
			goto bad_length;
Packit 9c3e7e
Packit 9c3e7e
		buf += cd->productDescription->length;
Packit 9c3e7e
		len -= cd->productDescription->length;
Packit 9c3e7e
		if (len < 0)
Packit 9c3e7e
			goto bad_length;
Packit 9c3e7e
Packit 9c3e7e
		cd->revisionData = (struct PTPText *) buf;
Packit 9c3e7e
		buf += sizeof(struct PTPText);
Packit 9c3e7e
		len -= sizeof(struct PTPText);
Packit 9c3e7e
		if (len < 0)
Packit 9c3e7e
			goto bad_length;
Packit 9c3e7e
Packit 9c3e7e
		buf += cd->revisionData->length;
Packit 9c3e7e
		len -= cd->revisionData->length;
Packit 9c3e7e
		if (len < 0)
Packit 9c3e7e
			goto bad_length;
Packit 9c3e7e
Packit 9c3e7e
		cd->userDescription = (struct PTPText *) buf;
Packit 9c3e7e
		buf += sizeof(struct PTPText);
Packit 9c3e7e
		len -= sizeof(struct PTPText);
Packit 9c3e7e
		if (len < 0)
Packit 9c3e7e
			goto bad_length;
Packit 9c3e7e
Packit 9c3e7e
		buf += cd->userDescription->length;
Packit 9c3e7e
		len -= cd->userDescription->length;
Packit 9c3e7e
		if (len < 0)
Packit 9c3e7e
			goto bad_length;
Packit 9c3e7e
Packit 9c3e7e
		cd->profileIdentity = buf;
Packit 9c3e7e
		buf += PROFILE_ID_LEN;
Packit 9c3e7e
		len -= PROFILE_ID_LEN;
Packit 9c3e7e
		if (len < 0)
Packit 9c3e7e
			goto bad_length;
Packit 9c3e7e
Packit 9c3e7e
		extra_len = buf - m->data;
Packit 9c3e7e
		break;
Packit 9c3e7e
	case TLV_USER_DESCRIPTION:
Packit 9c3e7e
		if (data_len < sizeof(struct PTPText))
Packit 9c3e7e
			goto bad_length;
Packit 9c3e7e
		extra->cd.userDescription = (struct PTPText *) m->data;
Packit 9c3e7e
		extra_len = sizeof(struct PTPText);
Packit 9c3e7e
		extra_len += extra->cd.userDescription->length;
Packit 9c3e7e
		break;
Packit 9c3e7e
	case TLV_DEFAULT_DATA_SET:
Packit 9c3e7e
		if (data_len != sizeof(struct defaultDS))
Packit 9c3e7e
			goto bad_length;
Packit 9c3e7e
		dds = (struct defaultDS *) m->data;
Packit 9c3e7e
		dds->numberPorts = ntohs(dds->numberPorts);
Packit 9c3e7e
		dds->clockQuality.offsetScaledLogVariance =
Packit 9c3e7e
			ntohs(dds->clockQuality.offsetScaledLogVariance);
Packit 9c3e7e
		break;
Packit 9c3e7e
	case TLV_CURRENT_DATA_SET:
Packit 9c3e7e
		if (data_len != sizeof(struct currentDS))
Packit 9c3e7e
			goto bad_length;
Packit 9c3e7e
		cds = (struct currentDS *) m->data;
Packit 9c3e7e
		cds->stepsRemoved = ntohs(cds->stepsRemoved);
Packit 9c3e7e
		cds->offsetFromMaster = net2host64(cds->offsetFromMaster);
Packit 9c3e7e
		cds->meanPathDelay = net2host64(cds->meanPathDelay);
Packit 9c3e7e
		break;
Packit 9c3e7e
	case TLV_PARENT_DATA_SET:
Packit 9c3e7e
		if (data_len != sizeof(struct parentDS))
Packit 9c3e7e
			goto bad_length;
Packit 9c3e7e
		pds = (struct parentDS *) m->data;
Packit 9c3e7e
		pds->parentPortIdentity.portNumber =
Packit 9c3e7e
			ntohs(pds->parentPortIdentity.portNumber);
Packit 9c3e7e
		pds->observedParentOffsetScaledLogVariance =
Packit 9c3e7e
			ntohs(pds->observedParentOffsetScaledLogVariance);
Packit 9c3e7e
		pds->observedParentClockPhaseChangeRate =
Packit 9c3e7e
			ntohl(pds->observedParentClockPhaseChangeRate);
Packit 9c3e7e
		pds->grandmasterClockQuality.offsetScaledLogVariance =
Packit 9c3e7e
			ntohs(pds->grandmasterClockQuality.offsetScaledLogVariance);
Packit 9c3e7e
		break;
Packit 9c3e7e
	case TLV_TIME_PROPERTIES_DATA_SET:
Packit 9c3e7e
		if (data_len != sizeof(struct timePropertiesDS))
Packit 9c3e7e
			goto bad_length;
Packit 9c3e7e
		tp = (struct timePropertiesDS *) m->data;
Packit 9c3e7e
		tp->currentUtcOffset = ntohs(tp->currentUtcOffset);
Packit 9c3e7e
		break;
Packit 9c3e7e
	case TLV_PORT_DATA_SET:
Packit 9c3e7e
		if (data_len != sizeof(struct portDS))
Packit 9c3e7e
			goto bad_length;
Packit 9c3e7e
		p = (struct portDS *) m->data;
Packit 9c3e7e
		p->portIdentity.portNumber = ntohs(p->portIdentity.portNumber);
Packit 9c3e7e
		p->peerMeanPathDelay = net2host64(p->peerMeanPathDelay);
Packit 9c3e7e
		break;
Packit 9c3e7e
	case TLV_TIME_STATUS_NP:
Packit 9c3e7e
		if (data_len != sizeof(struct time_status_np))
Packit 9c3e7e
			goto bad_length;
Packit 9c3e7e
		tsn = (struct time_status_np *) m->data;
Packit 9c3e7e
		tsn->master_offset = net2host64(tsn->master_offset);
Packit 9c3e7e
		tsn->ingress_time = net2host64(tsn->ingress_time);
Packit 9c3e7e
		tsn->cumulativeScaledRateOffset = ntohl(tsn->cumulativeScaledRateOffset);
Packit 9c3e7e
		tsn->scaledLastGmPhaseChange = ntohl(tsn->scaledLastGmPhaseChange);
Packit 9c3e7e
		tsn->gmTimeBaseIndicator = ntohs(tsn->gmTimeBaseIndicator);
Packit 9c3e7e
		scaled_ns_n2h(&tsn->lastGmPhaseChange);
Packit 9c3e7e
		tsn->gmPresent = ntohl(tsn->gmPresent);
Packit 9c3e7e
		break;
Packit 9c3e7e
	case TLV_GRANDMASTER_SETTINGS_NP:
Packit 9c3e7e
		if (data_len != sizeof(struct grandmaster_settings_np))
Packit 9c3e7e
			goto bad_length;
Packit 9c3e7e
		gsn = (struct grandmaster_settings_np *) m->data;
Packit 9c3e7e
		gsn->clockQuality.offsetScaledLogVariance =
Packit 9c3e7e
			ntohs(gsn->clockQuality.offsetScaledLogVariance);
Packit 9c3e7e
		gsn->utc_offset = ntohs(gsn->utc_offset);
Packit 9c3e7e
		break;
Packit 9c3e7e
	case TLV_PORT_DATA_SET_NP:
Packit 9c3e7e
		if (data_len != sizeof(struct port_ds_np))
Packit 9c3e7e
			goto bad_length;
Packit 9c3e7e
		pdsnp = (struct port_ds_np *) m->data;
Packit 9c3e7e
		pdsnp->neighborPropDelayThresh = ntohl(pdsnp->neighborPropDelayThresh);
Packit 9c3e7e
		pdsnp->asCapable = ntohl(pdsnp->asCapable);
Packit 9c3e7e
		break;
Packit 9c3e7e
	case TLV_SUBSCRIBE_EVENTS_NP:
Packit 9c3e7e
		if (data_len != sizeof(struct subscribe_events_np))
Packit 9c3e7e
			goto bad_length;
Packit 9c3e7e
		sen = (struct subscribe_events_np *)m->data;
Packit 9c3e7e
		sen->duration = ntohs(sen->duration);
Packit 9c3e7e
		break;
Packit 9c3e7e
	case TLV_PORT_PROPERTIES_NP:
Packit 9c3e7e
		if (data_len < sizeof(struct port_properties_np))
Packit 9c3e7e
			goto bad_length;
Packit 9c3e7e
		ppn = (struct port_properties_np *)m->data;
Packit 9c3e7e
		ppn->portIdentity.portNumber = ntohs(ppn->portIdentity.portNumber);
Packit 9c3e7e
		extra_len = sizeof(struct port_properties_np);
Packit 9c3e7e
		extra_len += ppn->interface.length;
Packit 9c3e7e
		break;
Packit 9c3e7e
	case TLV_SAVE_IN_NON_VOLATILE_STORAGE:
Packit 9c3e7e
	case TLV_RESET_NON_VOLATILE_STORAGE:
Packit 9c3e7e
	case TLV_INITIALIZE:
Packit 9c3e7e
	case TLV_FAULT_LOG_RESET:
Packit 9c3e7e
	case TLV_ENABLE_PORT:
Packit 9c3e7e
	case TLV_DISABLE_PORT:
Packit 9c3e7e
		if (data_len != 0)
Packit 9c3e7e
			goto bad_length;
Packit 9c3e7e
		break;
Packit 9c3e7e
	}
Packit 9c3e7e
	if (extra_len) {
Packit 9c3e7e
		if (extra_len % 2)
Packit 9c3e7e
			extra_len++;
Packit 9c3e7e
		if (extra_len + sizeof(m->id) != m->length)
Packit 9c3e7e
			goto bad_length;
Packit 9c3e7e
	}
Packit 9c3e7e
	return 0;
Packit 9c3e7e
bad_length:
Packit 9c3e7e
	return -EBADMSG;
Packit 9c3e7e
}
Packit 9c3e7e
Packit 9c3e7e
static void mgt_pre_send(struct management_tlv *m, struct tlv_extra *extra)
Packit 9c3e7e
{
Packit 9c3e7e
	struct defaultDS *dds;
Packit 9c3e7e
	struct currentDS *cds;
Packit 9c3e7e
	struct parentDS *pds;
Packit 9c3e7e
	struct timePropertiesDS *tp;
Packit 9c3e7e
	struct portDS *p;
Packit 9c3e7e
	struct port_ds_np *pdsnp;
Packit 9c3e7e
	struct time_status_np *tsn;
Packit 9c3e7e
	struct grandmaster_settings_np *gsn;
Packit 9c3e7e
	struct subscribe_events_np *sen;
Packit 9c3e7e
	struct port_properties_np *ppn;
Packit 9c3e7e
	struct mgmt_clock_description *cd;
Packit 9c3e7e
	switch (m->id) {
Packit 9c3e7e
	case TLV_CLOCK_DESCRIPTION:
Packit 9c3e7e
		if (extra) {
Packit 9c3e7e
			cd = &extra->cd;
Packit 9c3e7e
			flip16(cd->clockType);
Packit 9c3e7e
			flip16(&cd->physicalAddress->length);
Packit 9c3e7e
			flip16(&cd->protocolAddress->networkProtocol);
Packit 9c3e7e
			flip16(&cd->protocolAddress->addressLength);
Packit 9c3e7e
		}
Packit 9c3e7e
		break;
Packit 9c3e7e
	case TLV_DEFAULT_DATA_SET:
Packit 9c3e7e
		dds = (struct defaultDS *) m->data;
Packit 9c3e7e
		dds->numberPorts = htons(dds->numberPorts);
Packit 9c3e7e
		dds->clockQuality.offsetScaledLogVariance =
Packit 9c3e7e
			htons(dds->clockQuality.offsetScaledLogVariance);
Packit 9c3e7e
		break;
Packit 9c3e7e
	case TLV_CURRENT_DATA_SET:
Packit 9c3e7e
		cds = (struct currentDS *) m->data;
Packit 9c3e7e
		cds->stepsRemoved = htons(cds->stepsRemoved);
Packit 9c3e7e
		cds->offsetFromMaster = host2net64(cds->offsetFromMaster);
Packit 9c3e7e
		cds->meanPathDelay = host2net64(cds->meanPathDelay);
Packit 9c3e7e
		break;
Packit 9c3e7e
	case TLV_PARENT_DATA_SET:
Packit 9c3e7e
		pds = (struct parentDS *) m->data;
Packit 9c3e7e
		pds->parentPortIdentity.portNumber =
Packit 9c3e7e
			htons(pds->parentPortIdentity.portNumber);
Packit 9c3e7e
		pds->observedParentOffsetScaledLogVariance =
Packit 9c3e7e
			htons(pds->observedParentOffsetScaledLogVariance);
Packit 9c3e7e
		pds->observedParentClockPhaseChangeRate =
Packit 9c3e7e
			htonl(pds->observedParentClockPhaseChangeRate);
Packit 9c3e7e
		pds->grandmasterClockQuality.offsetScaledLogVariance =
Packit 9c3e7e
			htons(pds->grandmasterClockQuality.offsetScaledLogVariance);
Packit 9c3e7e
		break;
Packit 9c3e7e
	case TLV_TIME_PROPERTIES_DATA_SET:
Packit 9c3e7e
		tp = (struct timePropertiesDS *) m->data;
Packit 9c3e7e
		tp->currentUtcOffset = htons(tp->currentUtcOffset);
Packit 9c3e7e
		break;
Packit 9c3e7e
	case TLV_PORT_DATA_SET:
Packit 9c3e7e
		p = (struct portDS *) m->data;
Packit 9c3e7e
		p->portIdentity.portNumber = htons(p->portIdentity.portNumber);
Packit 9c3e7e
		p->peerMeanPathDelay = host2net64(p->peerMeanPathDelay);
Packit 9c3e7e
		break;
Packit 9c3e7e
	case TLV_TIME_STATUS_NP:
Packit 9c3e7e
		tsn = (struct time_status_np *) m->data;
Packit 9c3e7e
		tsn->master_offset = host2net64(tsn->master_offset);
Packit 9c3e7e
		tsn->ingress_time = host2net64(tsn->ingress_time);
Packit 9c3e7e
		tsn->cumulativeScaledRateOffset = htonl(tsn->cumulativeScaledRateOffset);
Packit 9c3e7e
		tsn->scaledLastGmPhaseChange = htonl(tsn->scaledLastGmPhaseChange);
Packit 9c3e7e
		tsn->gmTimeBaseIndicator = htons(tsn->gmTimeBaseIndicator);
Packit 9c3e7e
		scaled_ns_h2n(&tsn->lastGmPhaseChange);
Packit 9c3e7e
		tsn->gmPresent = htonl(tsn->gmPresent);
Packit 9c3e7e
		break;
Packit 9c3e7e
	case TLV_GRANDMASTER_SETTINGS_NP:
Packit 9c3e7e
		gsn = (struct grandmaster_settings_np *) m->data;
Packit 9c3e7e
		gsn->clockQuality.offsetScaledLogVariance =
Packit 9c3e7e
			htons(gsn->clockQuality.offsetScaledLogVariance);
Packit 9c3e7e
		gsn->utc_offset = htons(gsn->utc_offset);
Packit 9c3e7e
		break;
Packit 9c3e7e
	case TLV_PORT_DATA_SET_NP:
Packit 9c3e7e
		pdsnp = (struct port_ds_np *) m->data;
Packit 9c3e7e
		pdsnp->neighborPropDelayThresh = htonl(pdsnp->neighborPropDelayThresh);
Packit 9c3e7e
		pdsnp->asCapable = htonl(pdsnp->asCapable);
Packit 9c3e7e
		break;
Packit 9c3e7e
	case TLV_SUBSCRIBE_EVENTS_NP:
Packit 9c3e7e
		sen = (struct subscribe_events_np *)m->data;
Packit 9c3e7e
		sen->duration = htons(sen->duration);
Packit 9c3e7e
		break;
Packit 9c3e7e
	case TLV_PORT_PROPERTIES_NP:
Packit 9c3e7e
		ppn = (struct port_properties_np *)m->data;
Packit 9c3e7e
		ppn->portIdentity.portNumber = htons(ppn->portIdentity.portNumber);
Packit 9c3e7e
		break;
Packit 9c3e7e
	}
Packit 9c3e7e
}
Packit 9c3e7e
Packit 9c3e7e
static int nsm_resp_post_recv(struct tlv_extra *extra)
Packit 9c3e7e
{
Packit 9c3e7e
	struct nsm_resp_tlv_head *head;
Packit 9c3e7e
	struct TLV *tlv = extra->tlv;
Packit 9c3e7e
	struct timePropertiesDS *tp;
Packit 9c3e7e
	struct PortAddress *paddr;
Packit 9c3e7e
	struct currentDS *cds;
Packit 9c3e7e
	struct parentDS *pds;
Packit 9c3e7e
	unsigned char *ptr;
Packit 9c3e7e
	uint16_t expected;
Packit 9c3e7e
Packit 9c3e7e
	if (tlv->length < sizeof(*head) + sizeof(*extra->foot)
Packit 9c3e7e
	    - sizeof(head->type) - sizeof(head->length)) {
Packit 9c3e7e
		return -EBADMSG;
Packit 9c3e7e
	}
Packit 9c3e7e
	head = (struct nsm_resp_tlv_head *) tlv;
Packit 9c3e7e
	paddr = &head->parent_addr;
Packit 9c3e7e
	NTOHS(paddr->networkProtocol);
Packit 9c3e7e
	NTOHS(paddr->addressLength);
Packit 9c3e7e
Packit 9c3e7e
	switch (paddr->networkProtocol) {
Packit 9c3e7e
	case TRANS_UDP_IPV4:
Packit 9c3e7e
		expected = 4;
Packit 9c3e7e
		break;
Packit 9c3e7e
	case TRANS_UDP_IPV6:
Packit 9c3e7e
		expected = 16;
Packit 9c3e7e
		break;
Packit 9c3e7e
	case TRANS_IEEE_802_3:
Packit 9c3e7e
		expected = 6;
Packit 9c3e7e
		break;
Packit 9c3e7e
	default:
Packit 9c3e7e
		return -EBADMSG;
Packit 9c3e7e
	}
Packit 9c3e7e
	if (paddr->addressLength != expected) {
Packit 9c3e7e
		return -EBADMSG;
Packit 9c3e7e
	}
Packit 9c3e7e
	if (tlv->length != sizeof(*head) + sizeof(*extra->foot) +
Packit 9c3e7e
	    paddr->addressLength - sizeof(head->type) - sizeof(head->length)) {
Packit 9c3e7e
		return -EBADMSG;
Packit 9c3e7e
	}
Packit 9c3e7e
Packit 9c3e7e
	ptr = (unsigned char *) tlv;
Packit 9c3e7e
	ptr += sizeof(*head) + paddr->addressLength;
Packit 9c3e7e
	extra->foot = (struct nsm_resp_tlv_foot *) ptr;
Packit 9c3e7e
Packit 9c3e7e
	pds = &extra->foot->parent;
Packit 9c3e7e
	cds = &extra->foot->current;
Packit 9c3e7e
	tp = &extra->foot->timeprop;
Packit 9c3e7e
Packit 9c3e7e
	/*
Packit 9c3e7e
	 * At this point the alignment only 2 bytes worst case.
Packit 9c3e7e
	 * So we need to be careful with the 64 bit words.
Packit 9c3e7e
	 */
Packit 9c3e7e
	NTOHS(pds->parentPortIdentity.portNumber);
Packit 9c3e7e
	NTOHS(pds->observedParentOffsetScaledLogVariance);
Packit 9c3e7e
	NTOHL(pds->observedParentClockPhaseChangeRate);
Packit 9c3e7e
	NTOHS(pds->grandmasterClockQuality.offsetScaledLogVariance);
Packit 9c3e7e
Packit 9c3e7e
	NTOHS(cds->stepsRemoved);
Packit 9c3e7e
	net2host64_unaligned(&cds->offsetFromMaster);
Packit 9c3e7e
	net2host64_unaligned(&cds->meanPathDelay);
Packit 9c3e7e
Packit 9c3e7e
	NTOHS(tp->currentUtcOffset);
Packit 9c3e7e
Packit 9c3e7e
	NTOHL(extra->foot->lastsync.seconds_lsb);
Packit 9c3e7e
	NTOHS(extra->foot->lastsync.seconds_msb);
Packit 9c3e7e
	NTOHL(extra->foot->lastsync.nanoseconds);
Packit 9c3e7e
Packit 9c3e7e
	return 0;
Packit 9c3e7e
}
Packit 9c3e7e
Packit 9c3e7e
static void nsm_resp_pre_send(struct tlv_extra *extra)
Packit 9c3e7e
{
Packit 9c3e7e
	struct nsm_resp_tlv_head *head;
Packit 9c3e7e
	struct timePropertiesDS *tp;
Packit 9c3e7e
	struct PortAddress *paddr;
Packit 9c3e7e
	struct currentDS *cds;
Packit 9c3e7e
	struct parentDS *pds;
Packit 9c3e7e
Packit 9c3e7e
	head = (struct nsm_resp_tlv_head *) extra->tlv;
Packit 9c3e7e
	paddr = &head->parent_addr;
Packit 9c3e7e
Packit 9c3e7e
	pds = &extra->foot->parent;
Packit 9c3e7e
	cds = &extra->foot->current;
Packit 9c3e7e
	tp = &extra->foot->timeprop;
Packit 9c3e7e
Packit 9c3e7e
	NTOHS(paddr->networkProtocol);
Packit 9c3e7e
	NTOHS(paddr->addressLength);
Packit 9c3e7e
Packit 9c3e7e
	HTONS(pds->parentPortIdentity.portNumber);
Packit 9c3e7e
	HTONS(pds->observedParentOffsetScaledLogVariance);
Packit 9c3e7e
	HTONL(pds->observedParentClockPhaseChangeRate);
Packit 9c3e7e
	HTONS(pds->grandmasterClockQuality.offsetScaledLogVariance);
Packit 9c3e7e
Packit 9c3e7e
	HTONS(cds->stepsRemoved);
Packit 9c3e7e
	host2net64_unaligned(&cds->offsetFromMaster);
Packit 9c3e7e
	host2net64_unaligned(&cds->meanPathDelay);
Packit 9c3e7e
Packit 9c3e7e
	HTONS(tp->currentUtcOffset);
Packit 9c3e7e
Packit 9c3e7e
	HTONL(extra->foot->lastsync.seconds_lsb);
Packit 9c3e7e
	HTONS(extra->foot->lastsync.seconds_msb);
Packit 9c3e7e
	HTONL(extra->foot->lastsync.nanoseconds);
Packit 9c3e7e
}
Packit 9c3e7e
Packit 9c3e7e
static int org_post_recv(struct organization_tlv *org)
Packit 9c3e7e
{
Packit 9c3e7e
	struct follow_up_info_tlv *f;
Packit 9c3e7e
Packit 9c3e7e
	if (0 == memcmp(org->id, ieee8021_id, sizeof(ieee8021_id))) {
Packit 9c3e7e
		if (org->subtype[0] || org->subtype[1]) {
Packit 9c3e7e
			return 0;
Packit 9c3e7e
		}
Packit 9c3e7e
		switch (org->subtype[2]) {
Packit 9c3e7e
		case 1:
Packit 9c3e7e
			if (org->length + sizeof(struct TLV) != sizeof(struct follow_up_info_tlv))
Packit 9c3e7e
				goto bad_length;
Packit 9c3e7e
			f = (struct follow_up_info_tlv *) org;
Packit 9c3e7e
			f->cumulativeScaledRateOffset = ntohl(f->cumulativeScaledRateOffset);
Packit 9c3e7e
			f->gmTimeBaseIndicator = ntohs(f->gmTimeBaseIndicator);
Packit 9c3e7e
			scaled_ns_n2h(&f->lastGmPhaseChange);
Packit 9c3e7e
			f->scaledLastGmPhaseChange = ntohl(f->scaledLastGmPhaseChange);
Packit 9c3e7e
			break;
Packit 9c3e7e
		}
Packit 9c3e7e
	}
Packit 9c3e7e
	return 0;
Packit 9c3e7e
bad_length:
Packit 9c3e7e
	return -EBADMSG;
Packit 9c3e7e
}
Packit 9c3e7e
Packit 9c3e7e
static void org_pre_send(struct organization_tlv *org)
Packit 9c3e7e
{
Packit 9c3e7e
	struct follow_up_info_tlv *f;
Packit 9c3e7e
Packit 9c3e7e
	if (0 == memcmp(org->id, ieee8021_id, sizeof(ieee8021_id))) {
Packit 9c3e7e
		if (org->subtype[0] || org->subtype[1]) {
Packit 9c3e7e
			return;
Packit 9c3e7e
		}
Packit 9c3e7e
		switch (org->subtype[2]) {
Packit 9c3e7e
		case 1:
Packit 9c3e7e
			f = (struct follow_up_info_tlv *) org;
Packit 9c3e7e
			f->cumulativeScaledRateOffset = htonl(f->cumulativeScaledRateOffset);
Packit 9c3e7e
			f->gmTimeBaseIndicator = htons(f->gmTimeBaseIndicator);
Packit 9c3e7e
			scaled_ns_h2n(&f->lastGmPhaseChange);
Packit 9c3e7e
			f->scaledLastGmPhaseChange = htonl(f->scaledLastGmPhaseChange);
Packit 9c3e7e
			break;
Packit 9c3e7e
		}
Packit 9c3e7e
	}
Packit 9c3e7e
}
Packit 9c3e7e
Packit 9c3e7e
static int unicast_message_type_valid(uint8_t message_type)
Packit 9c3e7e
{
Packit 9c3e7e
	message_type >>= 4;
Packit 9c3e7e
	switch (message_type) {
Packit 9c3e7e
	case ANNOUNCE:
Packit 9c3e7e
	case SYNC:
Packit 9c3e7e
	case DELAY_RESP:
Packit 9c3e7e
	case PDELAY_RESP:
Packit 9c3e7e
		return 1;
Packit 9c3e7e
	default:
Packit 9c3e7e
		return 0;
Packit 9c3e7e
	}
Packit 9c3e7e
}
Packit 9c3e7e
Packit 9c3e7e
static int unicast_negotiation_post_recv(struct tlv_extra *extra)
Packit 9c3e7e
{
Packit 9c3e7e
	struct request_unicast_xmit_tlv *request;
Packit 9c3e7e
	struct ack_cancel_unicast_xmit_tlv *ack;
Packit 9c3e7e
	struct cancel_unicast_xmit_tlv *cancel;
Packit 9c3e7e
	struct grant_unicast_xmit_tlv *grant;
Packit 9c3e7e
	struct TLV *tlv = extra->tlv;
Packit 9c3e7e
Packit 9c3e7e
	switch (tlv->type) {
Packit 9c3e7e
	case TLV_REQUEST_UNICAST_TRANSMISSION:
Packit 9c3e7e
		if (TLV_LENGTH_INVALID(tlv, request_unicast_xmit_tlv)) {
Packit 9c3e7e
			return -EBADMSG;
Packit 9c3e7e
		}
Packit 9c3e7e
		request = (struct request_unicast_xmit_tlv *) tlv;
Packit 9c3e7e
		if (!unicast_message_type_valid(request->message_type)) {
Packit 9c3e7e
			return -EBADMSG;
Packit 9c3e7e
		}
Packit 9c3e7e
		NTOHL(request->durationField);
Packit 9c3e7e
		break;
Packit 9c3e7e
	case TLV_GRANT_UNICAST_TRANSMISSION:
Packit 9c3e7e
		if (TLV_LENGTH_INVALID(tlv, grant_unicast_xmit_tlv)) {
Packit 9c3e7e
			return -EBADMSG;
Packit 9c3e7e
		}
Packit 9c3e7e
		grant = (struct grant_unicast_xmit_tlv *) tlv;
Packit 9c3e7e
		if (!unicast_message_type_valid(grant->message_type)) {
Packit 9c3e7e
			return -EBADMSG;
Packit 9c3e7e
		}
Packit 9c3e7e
		NTOHL(grant->durationField);
Packit 9c3e7e
		break;
Packit 9c3e7e
	case TLV_CANCEL_UNICAST_TRANSMISSION:
Packit 9c3e7e
		if (TLV_LENGTH_INVALID(tlv, cancel_unicast_xmit_tlv)) {
Packit 9c3e7e
			return -EBADMSG;
Packit 9c3e7e
		}
Packit 9c3e7e
		cancel = (struct cancel_unicast_xmit_tlv *) tlv;
Packit 9c3e7e
		if (!unicast_message_type_valid(cancel->message_type_flags)) {
Packit 9c3e7e
			return -EBADMSG;
Packit 9c3e7e
		}
Packit 9c3e7e
		break;
Packit 9c3e7e
	case TLV_ACKNOWLEDGE_CANCEL_UNICAST_TRANSMISSION:
Packit 9c3e7e
		if (TLV_LENGTH_INVALID(tlv, ack_cancel_unicast_xmit_tlv)) {
Packit 9c3e7e
			return -EBADMSG;
Packit 9c3e7e
		}
Packit 9c3e7e
		ack = (struct ack_cancel_unicast_xmit_tlv *) tlv;
Packit 9c3e7e
		if (!unicast_message_type_valid(ack->message_type_flags)) {
Packit 9c3e7e
			return -EBADMSG;
Packit 9c3e7e
		}
Packit 9c3e7e
		break;
Packit 9c3e7e
	}
Packit 9c3e7e
	return 0;
Packit 9c3e7e
}
Packit 9c3e7e
Packit 9c3e7e
static void unicast_negotiation_pre_send(struct TLV *tlv)
Packit 9c3e7e
{
Packit 9c3e7e
	struct request_unicast_xmit_tlv *request;
Packit 9c3e7e
	struct grant_unicast_xmit_tlv *grant;
Packit 9c3e7e
Packit 9c3e7e
	switch (tlv->type) {
Packit 9c3e7e
	case TLV_REQUEST_UNICAST_TRANSMISSION:
Packit 9c3e7e
		request = (struct request_unicast_xmit_tlv *) tlv;
Packit 9c3e7e
		HTONL(request->durationField);
Packit 9c3e7e
		break;
Packit 9c3e7e
	case TLV_GRANT_UNICAST_TRANSMISSION:
Packit 9c3e7e
		grant = (struct grant_unicast_xmit_tlv *) tlv;
Packit 9c3e7e
		HTONL(grant->durationField);
Packit 9c3e7e
		break;
Packit 9c3e7e
	case TLV_CANCEL_UNICAST_TRANSMISSION:
Packit 9c3e7e
	case TLV_ACKNOWLEDGE_CANCEL_UNICAST_TRANSMISSION:
Packit 9c3e7e
		break;
Packit 9c3e7e
	}
Packit 9c3e7e
}
Packit 9c3e7e
Packit 9c3e7e
struct tlv_extra *tlv_extra_alloc(void)
Packit 9c3e7e
{
Packit 9c3e7e
	struct tlv_extra *extra = TAILQ_FIRST(&tlv_pool);
Packit 9c3e7e
Packit 9c3e7e
	if (extra) {
Packit 9c3e7e
		TAILQ_REMOVE(&tlv_pool, extra, list);
Packit 9c3e7e
	} else {
Packit 9c3e7e
		extra = calloc(1, sizeof(*extra));
Packit 9c3e7e
	}
Packit 9c3e7e
	return extra;
Packit 9c3e7e
}
Packit 9c3e7e
Packit 9c3e7e
void tlv_extra_cleanup(void)
Packit 9c3e7e
{
Packit 9c3e7e
	struct tlv_extra *extra;
Packit 9c3e7e
Packit 9c3e7e
	while ((extra = TAILQ_FIRST(&tlv_pool)) != NULL) {
Packit 9c3e7e
		TAILQ_REMOVE(&tlv_pool, extra, list);
Packit 9c3e7e
		free(extra);
Packit 9c3e7e
	}
Packit 9c3e7e
}
Packit 9c3e7e
Packit 9c3e7e
void tlv_extra_recycle(struct tlv_extra *extra)
Packit 9c3e7e
{
Packit 9c3e7e
	memset(extra, 0, sizeof(*extra));
Packit 9c3e7e
	TAILQ_INSERT_HEAD(&tlv_pool, extra, list);
Packit 9c3e7e
}
Packit 9c3e7e
Packit 9c3e7e
int tlv_post_recv(struct tlv_extra *extra)
Packit 9c3e7e
{
Packit 9c3e7e
	int result = 0;
Packit 9c3e7e
	struct management_tlv *mgt;
Packit 9c3e7e
	struct management_error_status *mes;
Packit 9c3e7e
	struct TLV *tlv = extra->tlv;
Packit 9c3e7e
	struct path_trace_tlv *ptt;
Packit 9c3e7e
Packit 9c3e7e
	switch (tlv->type) {
Packit 9c3e7e
	case TLV_MANAGEMENT:
Packit 9c3e7e
		if (TLV_LENGTH_INVALID(tlv, management_tlv))
Packit 9c3e7e
			goto bad_length;
Packit 9c3e7e
		mgt = (struct management_tlv *) tlv;
Packit 9c3e7e
		mgt->id = ntohs(mgt->id);
Packit 9c3e7e
		if (tlv->length > sizeof(mgt->id))
Packit 9c3e7e
			result = mgt_post_recv(mgt, tlv->length - sizeof(mgt->id), extra);
Packit 9c3e7e
		break;
Packit 9c3e7e
	case TLV_MANAGEMENT_ERROR_STATUS:
Packit 9c3e7e
		if (TLV_LENGTH_INVALID(tlv, management_error_status))
Packit 9c3e7e
			goto bad_length;
Packit 9c3e7e
		mes = (struct management_error_status *) tlv;
Packit 9c3e7e
		mes->error = ntohs(mes->error);
Packit 9c3e7e
		mes->id = ntohs(mes->id);
Packit 9c3e7e
		break;
Packit 9c3e7e
	case TLV_ORGANIZATION_EXTENSION:
Packit 9c3e7e
		if (TLV_LENGTH_INVALID(tlv, organization_tlv))
Packit 9c3e7e
			goto bad_length;
Packit 9c3e7e
		result = org_post_recv((struct organization_tlv *) tlv);
Packit 9c3e7e
		break;
Packit 9c3e7e
	case TLV_REQUEST_UNICAST_TRANSMISSION:
Packit 9c3e7e
	case TLV_GRANT_UNICAST_TRANSMISSION:
Packit 9c3e7e
	case TLV_CANCEL_UNICAST_TRANSMISSION:
Packit 9c3e7e
	case TLV_ACKNOWLEDGE_CANCEL_UNICAST_TRANSMISSION:
Packit 9c3e7e
		result = unicast_negotiation_post_recv(extra);
Packit 9c3e7e
		break;
Packit 9c3e7e
	case TLV_PATH_TRACE:
Packit 9c3e7e
		ptt = (struct path_trace_tlv *) tlv;
Packit 9c3e7e
		if (path_length(ptt) > PATH_TRACE_MAX) {
Packit 9c3e7e
			ptt->length = PATH_TRACE_MAX * sizeof(struct ClockIdentity);
Packit 9c3e7e
		}
Packit 9c3e7e
		break;
Packit 9c3e7e
	case TLV_ALTERNATE_TIME_OFFSET_INDICATOR:
Packit 9c3e7e
	case TLV_AUTHENTICATION:
Packit 9c3e7e
	case TLV_AUTHENTICATION_CHALLENGE:
Packit 9c3e7e
	case TLV_SECURITY_ASSOCIATION_UPDATE:
Packit 9c3e7e
	case TLV_CUM_FREQ_SCALE_FACTOR_OFFSET:
Packit 9c3e7e
	case TLV_PTPMON_REQ:
Packit 9c3e7e
		break;
Packit 9c3e7e
	case TLV_PTPMON_RESP:
Packit 9c3e7e
		result = nsm_resp_post_recv(extra);
Packit 9c3e7e
		break;
Packit 9c3e7e
	default:
Packit 9c3e7e
		break;
Packit 9c3e7e
	}
Packit 9c3e7e
	return result;
Packit 9c3e7e
bad_length:
Packit 9c3e7e
	return -EBADMSG;
Packit 9c3e7e
}
Packit 9c3e7e
Packit 9c3e7e
void tlv_pre_send(struct TLV *tlv, struct tlv_extra *extra)
Packit 9c3e7e
{
Packit 9c3e7e
	struct management_tlv *mgt;
Packit 9c3e7e
	struct management_error_status *mes;
Packit 9c3e7e
Packit 9c3e7e
	switch (tlv->type) {
Packit 9c3e7e
	case TLV_MANAGEMENT:
Packit 9c3e7e
		mgt = (struct management_tlv *) tlv;
Packit 9c3e7e
		if (tlv->length > sizeof(mgt->id))
Packit 9c3e7e
			mgt_pre_send(mgt, extra);
Packit 9c3e7e
		mgt->id = htons(mgt->id);
Packit 9c3e7e
		break;
Packit 9c3e7e
	case TLV_MANAGEMENT_ERROR_STATUS:
Packit 9c3e7e
		mes = (struct management_error_status *) tlv;
Packit 9c3e7e
		mes->error = htons(mes->error);
Packit 9c3e7e
		mes->id = htons(mes->id);
Packit 9c3e7e
		break;
Packit 9c3e7e
	case TLV_ORGANIZATION_EXTENSION:
Packit 9c3e7e
		org_pre_send((struct organization_tlv *) tlv);
Packit 9c3e7e
		break;
Packit 9c3e7e
	case TLV_REQUEST_UNICAST_TRANSMISSION:
Packit 9c3e7e
	case TLV_GRANT_UNICAST_TRANSMISSION:
Packit 9c3e7e
	case TLV_CANCEL_UNICAST_TRANSMISSION:
Packit 9c3e7e
	case TLV_ACKNOWLEDGE_CANCEL_UNICAST_TRANSMISSION:
Packit 9c3e7e
		unicast_negotiation_pre_send(tlv);
Packit 9c3e7e
		break;
Packit 9c3e7e
	case TLV_PATH_TRACE:
Packit 9c3e7e
	case TLV_ALTERNATE_TIME_OFFSET_INDICATOR:
Packit 9c3e7e
	case TLV_AUTHENTICATION:
Packit 9c3e7e
	case TLV_AUTHENTICATION_CHALLENGE:
Packit 9c3e7e
	case TLV_SECURITY_ASSOCIATION_UPDATE:
Packit 9c3e7e
	case TLV_CUM_FREQ_SCALE_FACTOR_OFFSET:
Packit 9c3e7e
	case TLV_PTPMON_REQ:
Packit 9c3e7e
		break;
Packit 9c3e7e
	case TLV_PTPMON_RESP:
Packit 9c3e7e
		nsm_resp_pre_send(extra);
Packit 9c3e7e
		break;
Packit 9c3e7e
	default:
Packit 9c3e7e
		break;
Packit 9c3e7e
	}
Packit 9c3e7e
}