static int org_post_recv(struct organization_tlv *org) { struct follow_up_info_tlv *f; if (0 == memcmp(org->id, ieee8021_id, sizeof(ieee8021_id))) { if (org->subtype[0] || org->subtype[1]) { return 0; } switch (org->subtype[2]) { case 1: if (org->length + sizeof(struct TLV) != sizeof(struct follow_up_info_tlv)) goto bad_length; f = (struct follow_up_info_tlv *) org; f->cumulativeScaledRateOffset = ntohl(f->cumulativeScaledRateOffset); f->gmTimeBaseIndicator = ntohs(f->gmTimeBaseIndicator); scaled_ns_n2h(&f->lastGmPhaseChange); f->scaledLastGmPhaseChange = ntohl(f->scaledLastGmPhaseChange); break; case 2: if (org->length + sizeof(struct TLV) != sizeof(struct msg_interval_req_tlv)) goto bad_length; } } return 0; bad_length: return -EBADMSG; }
static int mgt_post_recv(struct management_tlv *m, uint16_t data_len, struct tlv_extra *extra) { struct defaultDS *dds; struct currentDS *cds; struct parentDS *pds; struct timePropertiesDS *tp; struct portDS *p; struct port_ds_np *pdsnp; struct time_status_np *tsn; struct grandmaster_settings_np *gsn; struct subscribe_events_np *sen; struct port_properties_np *ppn; struct mgmt_clock_description *cd; int extra_len = 0, len; uint8_t *buf; uint16_t u16; switch (m->id) { case TLV_CLOCK_DESCRIPTION: cd = &extra->cd; buf = m->data; len = data_len; cd->clockType = (UInteger16 *) buf; buf += sizeof(*cd->clockType); len -= sizeof(*cd->clockType); if (len < 0) goto bad_length; flip16(cd->clockType); cd->physicalLayerProtocol = (struct PTPText *) buf; buf += sizeof(struct PTPText); len -= sizeof(struct PTPText); if (len < 0) goto bad_length; buf += cd->physicalLayerProtocol->length; len -= cd->physicalLayerProtocol->length; if (len < 0) goto bad_length; cd->physicalAddress = (struct PhysicalAddress *) buf; buf += sizeof(struct PhysicalAddress); len -= sizeof(struct PhysicalAddress); if (len < 0) goto bad_length; u16 = flip16(&cd->physicalAddress->length); if (u16 > TRANSPORT_ADDR_LEN) goto bad_length; buf += u16; len -= u16; if (len < 0) goto bad_length; cd->protocolAddress = (struct PortAddress *) buf; buf += sizeof(struct PortAddress); len -= sizeof(struct PortAddress); if (len < 0) goto bad_length; flip16(&cd->protocolAddress->networkProtocol); u16 = flip16(&cd->protocolAddress->addressLength); if (u16 > TRANSPORT_ADDR_LEN) goto bad_length; buf += u16; len -= u16; if (len < 0) goto bad_length; cd->manufacturerIdentity = buf; buf += OUI_LEN + 1; len -= OUI_LEN + 1; if (len < 0) goto bad_length; cd->productDescription = (struct PTPText *) buf; buf += sizeof(struct PTPText); len -= sizeof(struct PTPText); if (len < 0) goto bad_length; buf += cd->productDescription->length; len -= cd->productDescription->length; if (len < 0) goto bad_length; cd->revisionData = (struct PTPText *) buf; buf += sizeof(struct PTPText); len -= sizeof(struct PTPText); if (len < 0) goto bad_length; buf += cd->revisionData->length; len -= cd->revisionData->length; if (len < 0) goto bad_length; cd->userDescription = (struct PTPText *) buf; buf += sizeof(struct PTPText); len -= sizeof(struct PTPText); if (len < 0) goto bad_length; buf += cd->userDescription->length; len -= cd->userDescription->length; if (len < 0) goto bad_length; cd->profileIdentity = buf; buf += PROFILE_ID_LEN; len -= PROFILE_ID_LEN; if (len < 0) goto bad_length; extra_len = buf - m->data; break; case TLV_USER_DESCRIPTION: if (data_len < sizeof(struct PTPText)) goto bad_length; extra->cd.userDescription = (struct PTPText *) m->data; extra_len = sizeof(struct PTPText); extra_len += extra->cd.userDescription->length; break; case TLV_DEFAULT_DATA_SET: if (data_len != sizeof(struct defaultDS)) goto bad_length; dds = (struct defaultDS *) m->data; dds->numberPorts = ntohs(dds->numberPorts); dds->clockQuality.offsetScaledLogVariance = ntohs(dds->clockQuality.offsetScaledLogVariance); break; case TLV_CURRENT_DATA_SET: if (data_len != sizeof(struct currentDS)) goto bad_length; cds = (struct currentDS *) m->data; cds->stepsRemoved = ntohs(cds->stepsRemoved); cds->offsetFromMaster = net2host64(cds->offsetFromMaster); cds->meanPathDelay = net2host64(cds->meanPathDelay); break; case TLV_PARENT_DATA_SET: if (data_len != sizeof(struct parentDS)) goto bad_length; pds = (struct parentDS *) m->data; pds->parentPortIdentity.portNumber = ntohs(pds->parentPortIdentity.portNumber); pds->observedParentOffsetScaledLogVariance = ntohs(pds->observedParentOffsetScaledLogVariance); pds->observedParentClockPhaseChangeRate = ntohl(pds->observedParentClockPhaseChangeRate); pds->grandmasterClockQuality.offsetScaledLogVariance = ntohs(pds->grandmasterClockQuality.offsetScaledLogVariance); break; case TLV_TIME_PROPERTIES_DATA_SET: if (data_len != sizeof(struct timePropertiesDS)) goto bad_length; tp = (struct timePropertiesDS *) m->data; tp->currentUtcOffset = ntohs(tp->currentUtcOffset); break; case TLV_PORT_DATA_SET: if (data_len != sizeof(struct portDS)) goto bad_length; p = (struct portDS *) m->data; p->portIdentity.portNumber = ntohs(p->portIdentity.portNumber); p->peerMeanPathDelay = net2host64(p->peerMeanPathDelay); break; case TLV_TIME_STATUS_NP: if (data_len != sizeof(struct time_status_np)) goto bad_length; tsn = (struct time_status_np *) m->data; tsn->master_offset = net2host64(tsn->master_offset); tsn->ingress_time = net2host64(tsn->ingress_time); tsn->cumulativeScaledRateOffset = ntohl(tsn->cumulativeScaledRateOffset); tsn->scaledLastGmPhaseChange = ntohl(tsn->scaledLastGmPhaseChange); tsn->gmTimeBaseIndicator = ntohs(tsn->gmTimeBaseIndicator); scaled_ns_n2h(&tsn->lastGmPhaseChange); tsn->gmPresent = ntohl(tsn->gmPresent); break; case TLV_GRANDMASTER_SETTINGS_NP: if (data_len != sizeof(struct grandmaster_settings_np)) goto bad_length; gsn = (struct grandmaster_settings_np *) m->data; gsn->clockQuality.offsetScaledLogVariance = ntohs(gsn->clockQuality.offsetScaledLogVariance); gsn->utc_offset = ntohs(gsn->utc_offset); break; case TLV_PORT_DATA_SET_NP: if (data_len != sizeof(struct port_ds_np)) goto bad_length; pdsnp = (struct port_ds_np *) m->data; pdsnp->neighborPropDelayThresh = ntohl(pdsnp->neighborPropDelayThresh); pdsnp->asCapable = ntohl(pdsnp->asCapable); break; case TLV_SUBSCRIBE_EVENTS_NP: if (data_len != sizeof(struct subscribe_events_np)) goto bad_length; sen = (struct subscribe_events_np *)m->data; sen->duration = ntohs(sen->duration); break; case TLV_PORT_PROPERTIES_NP: if (data_len < sizeof(struct port_properties_np)) goto bad_length; ppn = (struct port_properties_np *)m->data; ppn->portIdentity.portNumber = ntohs(ppn->portIdentity.portNumber); extra_len = sizeof(struct port_properties_np); extra_len += ppn->interface.length; break; case TLV_SAVE_IN_NON_VOLATILE_STORAGE: case TLV_RESET_NON_VOLATILE_STORAGE: case TLV_INITIALIZE: case TLV_FAULT_LOG_RESET: case TLV_ENABLE_PORT: case TLV_DISABLE_PORT: if (data_len != 0) goto bad_length; break; } if (extra_len) { if (extra_len % 2) extra_len++; if (extra_len + sizeof(m->id) != m->length) goto bad_length; } return 0; bad_length: return -EBADMSG; }